{"version":3,"file":"group_chart_controller-CN2_S8jN.js","sources":["../../../app/javascript/entrypoints/controllers/group_chart_controller.jsx"],"sourcesContent":["/** @jsx renderer.create */\nimport { Controller } from \"@hotwired/stimulus\";\nimport dayjs from \"dayjs\";\n\nimport {\n Chart,\n CategoryScale,\n LinearScale,\n PointElement,\n LineElement,\n Title,\n Tooltip,\n Legend,\n} from \"chart.js\";\nimport ChartDataLabels from \"chartjs-plugin-datalabels\";\n\nimport _ from \"lodash\";\n\nimport { useWindowResize } from \"stimulus-use\";\nimport { useResize } from \"stimulus-use\";\nimport currency from \"currency.js\";\n\nimport numbro from \"numbro\";\n\nimport { CommonDOMRenderer } from \"render-jsx/dom\";\n\nimport { numberFormatter, ticksCallback } from \"../utils\";\n\nconst tooltipWidth = 350;\n\nconst red = \"rgba(255, 0, 0, 1)\";\nconst black = \"rgb(0, 0, 0, 1)\";\n\nChart.register(\n CategoryScale,\n LinearScale,\n PointElement,\n LineElement,\n Title,\n Tooltip,\n Legend,\n ChartDataLabels\n);\n\nexport default class extends Controller {\n chart = null;\n initial = false;\n values = [];\n labels = [];\n\n static targets = [\"chart\"];\n\n static values = {\n jsonData: Object,\n };\n\n initialize() {\n this.resize = _.throttle(this.resize, 750).bind(this);\n this.windowResize = _.throttle(this.windowResize, 750).bind(this);\n }\n\n chartTargetDisconnect(element) {\n const chart = Chart.getChart(`chart-${this.statId}`);\n if (chart) {\n chart.destroy();\n }\n }\n\n jsonDataValueChanged() {\n if (!this.initial) {\n this.initial = true;\n } else {\n const data = this.jsonDataValue;\n const rows = this.jsonDataValue.rows;\n\n this.statId = data.id;\n this.dataType = data.data_type;\n this.reverse = data.upside_down;\n this.decimalPlaces = data.decimal_places;\n this.tracking = data.tracking;\n this.endOfWeek = data.end_of_week;\n\n this.min = data.min;\n this.max = data.max;\n\n this.values = Object.values(rows) || [];\n\n if (this.tracking == \"week\") {\n this.labels = Object.keys(rows).map((key) => this.endOfWeek[key]);\n } else {\n this.labels = Object.keys(rows) || [];\n }\n\n let chart = Chart.getChart(`chart-${this.statId}`);\n\n if (chart) {\n chart.data.labels = this.labels;\n chart.data.datasets[0].data = this.values;\n\n chart.options = this.options(rows);\n\n chart.update(\"reset\");\n chart.update(\"normal\");\n } else {\n const ctx = document\n .getElementById(`chart-${this.statId}`)\n .getContext(\"2d\");\n this.createChart(ctx, rows, this.element);\n }\n }\n }\n\n chartTargetConnected(element) {\n const parsedData = this.jsonDataValue;\n\n if (_.isEmpty(parsedData) || _.isEmpty(parsedData.rows)) {\n return;\n }\n\n const data = parsedData.rows;\n\n this.values = Object.values(data) || [];\n\n this.tracking = parsedData.tracking;\n this.min = parsedData.min;\n this.max = parsedData.max;\n\n this.statId = parsedData.id;\n this.dataType = parsedData.data_type;\n this.reverse = parsedData.upside_down;\n this.decimalPlaces = parsedData.decimal_places;\n this.endOfWeek = parsedData.end_of_week;\n\n if (this.tracking == \"week\") {\n this.labels = Object.keys(data).map((key) => this.endOfWeek[key]);\n } else {\n this.labels = Object.keys(data) || [];\n }\n\n if (_.isEmpty(this.labels) || _.isEmpty(this.values)) {\n return;\n }\n\n const ctx = document\n .getElementById(`chart-${this.statId}`)\n .getContext(\"2d\");\n\n this.createChart(ctx, data, element);\n }\n\n getOrCreateTooltip = (chart) => {\n let tooltipEl = chart.canvas.parentNode.querySelector(\"div.chart-tooltip\");\n\n if (!tooltipEl) {\n tooltipEl = document.createElement(\"div\");\n\n tooltipEl.classList.add(\n \"bg-gray-100\",\n \"text-black\",\n \"rounded\",\n \"shadow-lg\",\n \"border\",\n \"chart-tooltip\"\n );\n\n tooltipEl.style.opacity = 1;\n tooltipEl.style.pointerEvents = \"none\";\n tooltipEl.style.position = \"absolute\";\n tooltipEl.style.transform = \"translate(-50%, 0)\";\n tooltipEl.style.transition = \"all .1s ease\";\n tooltipEl.style.zIndex = \"9999999999999999999999\";\n\n const table = document.createElement(\"table\");\n table.style.margin = \"0px\";\n\n tooltipEl.appendChild(table);\n chart.canvas.parentNode.appendChild(tooltipEl);\n }\n\n return tooltipEl;\n };\n\n createChart = (ctx, data, element) => {\n useWindowResize(this);\n useResize(this);\n\n let delayed;\n\n new Chart(ctx, {\n type: \"line\",\n data: {\n labels: this.labels,\n datasets: [\n {\n data: this.values,\n ...this.basicLineChartSettings(),\n datalabels: {\n display: true,\n color: \"black\",\n textStrokeWidth: 0.5,\n align: \"top\",\n offset: 12,\n font: {\n size: 12,\n },\n },\n },\n ],\n },\n options: {\n animation: {\n onComplete: () => {\n delayed = true;\n },\n delay: (context) => {\n let delay = 0;\n if (\n context.type === \"data\" &&\n context.mode === \"default\" &&\n !delayed\n ) {\n delay = context.dataIndex * 50 + context.datasetIndex * 25;\n }\n\n return delay;\n },\n },\n font: {\n family: \"Inter var\",\n size: 12,\n },\n ...this.options(data),\n },\n });\n\n document.addEventListener(\"chart:updated\", (event) => {\n const chart = Chart.getChart(element.id);\n const index = chart.data.labels.indexOf(event.detail.entryAt);\n\n if (index != undefined || index != -1) {\n let value = event.detail.entry;\n\n if (_.isEmpty(value) || value == \"No Report\" || value == \"NR\") {\n chart.data.datasets[0].data[index] = null;\n } else {\n try {\n const parsedValue = parseFloat(value.replace(/[^0-9.]/g, \"\"));\n chart.data.datasets[0].data[index] = parsedValue;\n } catch (error) {\n console.log(error);\n return;\n }\n }\n\n chart.update(\"active\");\n }\n });\n };\n\n externalTooltipHandler = (context) => {\n const { chart, tooltip } = context;\n const tooltipEl = this.getOrCreateTooltip(chart);\n\n tooltipEl.innerHTML = \"\";\n\n if (tooltip.opacity === 0) {\n tooltipEl.style.opacity = 0;\n return;\n }\n\n if (tooltip.body) {\n const bodyLines = tooltip.body.map((b) => b.lines);\n const titleLines = tooltip.title || [];\n const renderer = new CommonDOMRenderer();\n\n renderer\n .render(\n