analytics.js 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  1. import ApolloClient, { gql } from "apollo-boost"
  2. import React from "react"
  3. import ReactDOM from "react-dom"
  4. import Chart from "react-apexcharts"
  5. import { ApolloProvider, Query } from "react-apollo"
  6. import moment from "moment"
  7. const initAnalytics = ({ elementId, errorMessage, labels, title, uri }) => {
  8. const element = document.getElementById(elementId)
  9. if (!element) console.error("Element with id " + element + "doesn't exist!")
  10. const client = new ApolloClient({
  11. credentials: "same-origin",
  12. uri: uri
  13. })
  14. ReactDOM.render(
  15. <ApolloProvider client={client}>
  16. <Analytics errorMessage={errorMessage} labels={labels} title={title} />
  17. </ApolloProvider>,
  18. element
  19. )
  20. }
  21. const getAnalytics = gql`
  22. query getAnalytics($span: Int!) {
  23. analytics(span: $span) {
  24. users {
  25. current
  26. previous
  27. }
  28. threads {
  29. current
  30. previous
  31. }
  32. posts {
  33. current
  34. previous
  35. }
  36. attachments {
  37. current
  38. previous
  39. }
  40. dataDownloads {
  41. current
  42. previous
  43. }
  44. }
  45. }
  46. `
  47. class Analytics extends React.Component {
  48. state = { span: 30 }
  49. setSpan = span => {
  50. this.setState({ span })
  51. }
  52. render() {
  53. const { errorMessage, labels, title } = this.props
  54. const { span } = this.state
  55. return (
  56. <div className="card card-admin-info">
  57. <div className="card-body">
  58. <div className="row align-items-center">
  59. <div className="col">
  60. <h4 className="card-title">{title}</h4>
  61. </div>
  62. <div className="col-auto">
  63. <SpanPicker span={span} setSpan={this.setSpan} />
  64. </div>
  65. </div>
  66. </div>
  67. <Query query={getAnalytics} variables={{ span }}>
  68. {({ loading, error, data }) => {
  69. if (loading) return <Spinner />
  70. if (error) return <Error message={errorMessage} />
  71. const { analytics } = data
  72. return (
  73. <>
  74. <AnalyticsItem
  75. data={analytics.users}
  76. name={labels.users}
  77. span={span}
  78. />
  79. <AnalyticsItem
  80. data={analytics.threads}
  81. name={labels.threads}
  82. span={span}
  83. />
  84. <AnalyticsItem
  85. data={analytics.posts}
  86. name={labels.posts}
  87. span={span}
  88. />
  89. <AnalyticsItem
  90. data={analytics.attachments}
  91. name={labels.attachments}
  92. span={span}
  93. />
  94. <AnalyticsItem
  95. data={analytics.dataDownloads}
  96. name={labels.dataDownloads}
  97. span={span}
  98. />
  99. </>
  100. )
  101. }}
  102. </Query>
  103. </div>
  104. )
  105. }
  106. }
  107. const SpanPicker = ({ span, setSpan }) => (
  108. <div>
  109. {[30, 90, 180, 360].map(choice => (
  110. <button
  111. key={choice}
  112. className={
  113. choice === span
  114. ? "btn btn-primary btn-sm ml-3"
  115. : "btn btn-light btn-sm ml-3"
  116. }
  117. type="button"
  118. onClick={() => setSpan(choice)}
  119. >
  120. {choice}
  121. </button>
  122. ))}
  123. </div>
  124. )
  125. const Spinner = () => (
  126. <div className="card-body border-top">
  127. <div className="text-center py-5">
  128. <div className="spinner-border text-light" role="status">
  129. <span className="sr-only">Loading...</span>
  130. </div>
  131. </div>
  132. </div>
  133. )
  134. const Error = ({ message }) => (
  135. <div className="card-body border-top">
  136. <div className="text-center py-5">{message}</div>
  137. </div>
  138. )
  139. const CURRENT = "C"
  140. const PREVIOUS = "P"
  141. const AnalyticsItem = ({ data, legend, name, span }) => {
  142. const options = {
  143. legend: {
  144. show: false
  145. },
  146. chart: {
  147. animations: {
  148. enabled: false
  149. },
  150. parentHeightOffset: 0,
  151. toolbar: {
  152. show: false
  153. }
  154. },
  155. colors: ["#6554c0", "#b3d4ff"],
  156. grid: {
  157. padding: {
  158. top: 0
  159. }
  160. },
  161. stroke: {
  162. width: 2
  163. },
  164. tooltip: {
  165. x: {
  166. show: false
  167. },
  168. y: {
  169. title: {
  170. formatter: function(series, { dataPointIndex }) {
  171. const now = moment()
  172. if (series === PREVIOUS) now.subtract(span, "days")
  173. now.subtract(span - dataPointIndex - 1, "days")
  174. return now.format("ll")
  175. }
  176. }
  177. }
  178. },
  179. xaxis: {
  180. axisBorder: {
  181. show: false
  182. },
  183. axisTicks: {
  184. show: false
  185. },
  186. labels: {
  187. show: false
  188. },
  189. categories: [],
  190. tooltip: {
  191. enabled: false
  192. }
  193. },
  194. yaxis: {
  195. tickAmount: 2,
  196. max: m => m || 1,
  197. show: false
  198. }
  199. }
  200. const series = [
  201. { name: CURRENT, data: data.current },
  202. { name: PREVIOUS, data: data.previous }
  203. ]
  204. return (
  205. <div className="card-body border-top pb-1">
  206. <h5 className="m-0">{name}</h5>
  207. <div className="row align-items-center">
  208. <div className="col-auto">
  209. <Summary data={data} />
  210. </div>
  211. <div className="col">
  212. <ChartContainer>
  213. {({ width }) =>
  214. width > 1 && (
  215. <Chart
  216. options={options}
  217. series={series}
  218. type="line"
  219. width={width}
  220. height={140}
  221. />
  222. )
  223. }
  224. </ChartContainer>
  225. </div>
  226. </div>
  227. </div>
  228. )
  229. }
  230. const Summary = ({ data }) => {
  231. const current = data.current.reduce((a, b) => a + b)
  232. const previous = data.previous.reduce((a, b) => a + b)
  233. const diff = current - previous
  234. let color = "text-light"
  235. let icon = "fas fa-equals"
  236. if (diff > 0) {
  237. color = "text-success"
  238. icon = "fas fa-chevron-up"
  239. }
  240. if (diff < 0) {
  241. color = "text-danger"
  242. icon = "fas fa-chevron-down"
  243. }
  244. return (
  245. <div className="card-admin-analytics-summary">
  246. <div>{current}</div>
  247. <small className={color}>
  248. <span className={icon} /> {Math.abs(diff)}
  249. </small>
  250. </div>
  251. )
  252. }
  253. class ChartContainer extends React.Component {
  254. state = { width: 1, height: 1 }
  255. element = React.createRef()
  256. componentDidMount() {
  257. this.timer = window.setInterval(this.updateSize, 3000)
  258. this.updateSize()
  259. }
  260. componentWillUnmount() {
  261. window.clearInterval(this.timer)
  262. }
  263. updateSize = () => {
  264. this.setState({
  265. width: this.element.current.clientWidth,
  266. height: this.element.current.clientHeight
  267. })
  268. }
  269. render() {
  270. return (
  271. <div className="card-admin-analytics-chart" ref={this.element}>
  272. {this.props.children(this.state)}
  273. </div>
  274. )
  275. }
  276. }
  277. export default initAnalytics