reports.blade.php 62 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314
  1. {{-- resources/views/livewire/reports.blade.php --}}
  2. <div>
  3. <script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
  4. <link rel="stylesheet" href="/css/chart-reports.css">
  5. <div class="dashboard-container">
  6. <div wire:ignore class="chart-row" style="display: grid; grid-template-columns: 1fr 1fr; gap: 1rem; align-items: start;">
  7. <div class="chart-card">
  8. <div class="chart-header">
  9. <h3 class="chart-title">Entrate/Uscite totali</h3>
  10. </div>
  11. <div class="chart-body">
  12. <div class="yearly-table-container" id="yearly-table"></div>
  13. </div>
  14. </div>
  15. {{-- <div class="chart-card">
  16. <div class="chart-header">
  17. <h3 class="chart-title">Tesserati per Stagione</h3>
  18. </div>
  19. <div class="chart-body">
  20. <div class="members-table-container" id="members-table"></div>
  21. </div>
  22. </div> --}}
  23. </div>
  24. <div class="controls-section">
  25. <div class="control-group">
  26. <label for="season-filter">Stagione di Riferimento:</label>
  27. <select class="form-select" wire:model="seasonFilter" wire:change="updateCharts">
  28. @foreach($this->getAvailableSeasons() as $season)
  29. <option value="{{ $season }}">{{ $season }}</option>
  30. @endforeach
  31. </select>
  32. </div>
  33. </div>
  34. @php
  35. $summary = $this->getYearlySummary();
  36. @endphp
  37. <div class="summary-cards">
  38. <div class="summary-card income">
  39. <h3>Entrate Totali</h3>
  40. <div class="value">€{{ number_format($summary['totalIncome'], 2, ',', '.') }}</div>
  41. </div>
  42. <div class="summary-card expense">
  43. <h3>Uscite Totali</h3>
  44. <div class="value">€{{ number_format($summary['totalExpenses'], 2, ',', '.') }}</div>
  45. </div>
  46. <div class="summary-card delta {{ $summary['delta'] < 0 ? 'negative' : '' }}">
  47. <h3>Bilancio Netto</h3>
  48. <div class="value">€{{ number_format($summary['delta'], 2, ',', '.') }}</div>
  49. </div>
  50. </div>
  51. <div wire:ignore>
  52. <div class="chart-row">
  53. <div class="chart-card">
  54. <div class="chart-header">
  55. <h3 class="chart-title">Entrate e Uscite Mensili - <span id="monthly-season-title">{{ $seasonFilter }}</span></h3>
  56. </div>
  57. <div class="chart-body">
  58. <div style="display: grid; grid-template-columns: 1fr 300px; align-items: start;">
  59. <div class="chart-container">
  60. <canvas id="monthly-chart-{{ str_replace('-', '', $seasonFilter) }}"></canvas>
  61. </div>
  62. <div class="monthly-table-container" id="monthly-table">
  63. </div>
  64. </div>
  65. </div>
  66. </div>
  67. </div>
  68. <div class="chart-row">
  69. <div class="chart-card">
  70. <div class="chart-header">
  71. <h3 class="chart-title">Redditività per Causale - <span id="causals-season-title">{{ $seasonFilter }}</span></h3>
  72. </div>
  73. <div class="chart-body">
  74. <div style="display: grid; grid-template-columns: 1fr 700px; gap: 1rem; align-items: start;">
  75. <div class="causals-table-container" id="causals-table">
  76. </div>
  77. <div class="chart-container">
  78. <canvas id="causals-chart-{{ str_replace('-', '', $seasonFilter) }}"></canvas>
  79. </div>
  80. </div>
  81. </div>
  82. </div>
  83. </div>
  84. <div class="chart-row">
  85. <div class="chart-card">
  86. <div class="chart-header">
  87. <h3 class="chart-title">Tesserati per Stagione</h3>
  88. </div>
  89. <div class="chart-body">
  90. <div style="display: grid; grid-template-columns: 1fr 500px; gap: 1rem; align-items: start;">
  91. <div class="chart-container">
  92. <canvas id="members-chart"></canvas>
  93. </div>
  94. <div class="chart-body">
  95. <div class="members-table-container" id="members-table"></div>
  96. </div>
  97. </div>
  98. </div>
  99. </div>
  100. </div>
  101. </div>
  102. <div class="chart-row">
  103. <div class="chart-card modern-course-card">
  104. <div class="chart-header">
  105. <h3 class="chart-title">Analisi incassi corsi</h3>
  106. </div>
  107. <div class="chart-body">
  108. <div class="course-controls">
  109. <div class="control-group">
  110. <label>Seleziona Corso ({{ $seasonFilter }}):</label>
  111. <select class="form-select modern-select" wire:model.live="selectedCourse">
  112. <option value="">Seleziona un Corso</option>
  113. @foreach($this->getCoursesForSelect() as $course)
  114. <option value="{{ $course['id'] }}">{{ $course['full_name'] }}</option>
  115. @endforeach
  116. </select>
  117. </div>
  118. </div>
  119. @if($selectedCourse)
  120. <div wire:ignore wire:key="course-chart-{{ $seasonFilter }}-{{ $selectedCourse }}">
  121. <div class="modern-chart-layout">
  122. <div class="course-delta-table" id="course-delta-table-{{ str_replace('-', '', $seasonFilter) }}-{{ $selectedCourse }}">
  123. </div>
  124. <div class="modern-chart-container">
  125. <canvas id="courses-chart-{{ str_replace('-', '', $seasonFilter) }}-{{ $selectedCourse }}"></canvas>
  126. </div>
  127. </div>
  128. </div>
  129. @else
  130. <div class="chart-placeholder">
  131. <div style="text-align: center;">
  132. <div style="font-size: 3rem; margin-bottom: 1rem; opacity: 0.3;">📊</div>
  133. <p style="font-size: 1.25rem; font-weight: 600; margin: 0;">Seleziona un corso per
  134. visualizzare il grafico</p>
  135. <p style="font-size: 1rem; opacity: 0.7; margin-top: 0.5rem;">Usa il menu a tendina sopra
  136. per scegliere un corso</p>
  137. </div>
  138. </div>
  139. @endif
  140. </div>
  141. </div>
  142. </div>
  143. </div>
  144. <script>
  145. window.ReportsChartManager = window.ReportsChartManager || {
  146. charts: {},
  147. currentSeason: null,
  148. destroyChart: function (chartId) {
  149. if (this.charts[chartId]) {
  150. this.charts[chartId].destroy();
  151. delete this.charts[chartId];
  152. }
  153. },
  154. destroyAllCharts: function () {
  155. Object.keys(this.charts).forEach(chartId => {
  156. this.destroyChart(chartId);
  157. });
  158. },
  159. destroySeasonCharts: function (oldSeasonKey) {
  160. const chartsToDestroy = Object.keys(this.charts).filter(chartId =>
  161. chartId.includes(oldSeasonKey)
  162. );
  163. chartsToDestroy.forEach(chartId => this.destroyChart(chartId));
  164. },
  165. updateMainCharts: function () {
  166. console.log('=== updateMainCharts called ===');
  167. const seasonFilter = @this.get('seasonFilter');
  168. const monthlyTitle = document.getElementById('monthly-season-title');
  169. const causalsTitle = document.getElementById('causals-season-title');
  170. if (monthlyTitle) {
  171. monthlyTitle.textContent = seasonFilter;
  172. }
  173. if (causalsTitle) {
  174. causalsTitle.textContent = seasonFilter;
  175. }
  176. const originalSeasonKey = '{{ str_replace('-', '', $seasonFilter) }}';
  177. console.log('Using original season key for canvas IDs:', originalSeasonKey);
  178. @this.call('getMonthlyTotals').then(monthlyData => {
  179. console.log('Got monthly data:', monthlyData);
  180. this.createMonthlyChart(originalSeasonKey, monthlyData);
  181. this.updateMonthlyTable(monthlyData);
  182. });
  183. @this.call('getYearlyTotals').then(yearlyData => {
  184. console.log('Got yearly data:', yearlyData);
  185. this.updateYearlyTable(yearlyData);
  186. });
  187. @this.call('getTopCausalsByAmount').then(causalsData => {
  188. console.log('Got causals data:', causalsData);
  189. this.createCausalsChart(originalSeasonKey, causalsData);
  190. });
  191. @this.call('getTesseratiData').then(membersData => {
  192. console.log('Got members data:', membersData);
  193. this.createMembersChart(originalSeasonKey, membersData);
  194. this.updateMembersTable(membersData);
  195. });
  196. },
  197. forceUpdateCharts: function () {
  198. console.log('Force updating charts...');
  199. this.currentSeason = null;
  200. this.updateMainCharts();
  201. },
  202. createMonthlyChart: function (seasonKey, monthlyData) {
  203. const chartId = `monthly-chart-${seasonKey}`;
  204. const canvas = document.getElementById(chartId);
  205. if (!canvas) {
  206. console.error('Canvas not found for ID:', chartId);
  207. return;
  208. }
  209. this.destroyChart(chartId);
  210. const ctx = canvas.getContext('2d');
  211. const incomeGradient = ctx.createLinearGradient(0, 0, 0, 400);
  212. incomeGradient.addColorStop(0, 'rgba(0, 184, 148, 1)');
  213. incomeGradient.addColorStop(1, 'rgba(0, 184, 148, 1)');
  214. const expenseGradient = ctx.createLinearGradient(0, 0, 0, 400);
  215. expenseGradient.addColorStop(0, 'rgba(255, 107, 107, 1)');
  216. expenseGradient.addColorStop(1, 'rgba(255, 107, 107, 1)');
  217. this.charts[chartId] = new Chart(ctx, {
  218. type: 'bar',
  219. data: {
  220. labels: monthlyData.labels,
  221. datasets: [
  222. {
  223. label: 'Entrate',
  224. data: monthlyData.datasets[0].data,
  225. backgroundColor: incomeGradient,
  226. borderColor: '#00b894',
  227. borderWidth: 0,
  228. borderRadius: 0,
  229. borderSkipped: false,
  230. barThickness: "flex",
  231. barPercentage: 0.65,
  232. categoryPercentage: 0.4,
  233. },
  234. {
  235. label: 'Uscite',
  236. data: monthlyData.datasets[1].data,
  237. backgroundColor: expenseGradient,
  238. borderColor: '#ff6b6b',
  239. borderWidth: 0,
  240. borderRadius: 0,
  241. borderSkipped: false,
  242. barThickness: "flex",
  243. barPercentage: 0.65,
  244. categoryPercentage: 0.4,
  245. }
  246. ]
  247. },
  248. options: {
  249. responsive: true,
  250. maintainAspectRatio: false,
  251. interaction: {
  252. mode: 'index',
  253. intersect: false,
  254. },
  255. plugins: {
  256. legend: {
  257. position: 'bottom',
  258. labels: {
  259. usePointStyle: true,
  260. padding: 20,
  261. pointStyle: "rect",
  262. font: { weight: '500' }
  263. }
  264. },
  265. tooltip: {
  266. backgroundColor: 'rgba(255, 255, 255, 1)',
  267. titleColor: '#212529',
  268. bodyColor: '#495057',
  269. borderColor: '#e9ecef',
  270. borderWidth: 2,
  271. cornerRadius: 0,
  272. callbacks: {
  273. label: function (context) {
  274. return context.dataset.label + ': €' +
  275. new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(context.parsed.y);
  276. }
  277. }
  278. },
  279. },
  280. scales: {
  281. x: {
  282. grid: { display: false },
  283. ticks: { font: { weight: '500' } }
  284. },
  285. y: {
  286. beginAtZero: true,
  287. grid: { color: 'rgba(0, 0, 0, 0.05)' },
  288. ticks: {
  289. callback: function (value) {
  290. return '€' + new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(value);
  291. }
  292. }
  293. }
  294. },
  295. elements: {
  296. bar: {
  297. borderRadius: {
  298. topLeft: 12,
  299. topRight: 12,
  300. bottomLeft: 0,
  301. bottomRight: 0
  302. }
  303. }
  304. },
  305. animation: {
  306. duration: 1000,
  307. easing: 'easeOutQuart'
  308. }
  309. }
  310. });
  311. },
  312. createCausalsChart: function (seasonKey, causalsData) {
  313. const chartId = `causals-chart-${seasonKey}`;
  314. const canvas = document.getElementById(chartId);
  315. if (!canvas) return;
  316. this.destroyChart(chartId);
  317. const ctx = canvas.getContext('2d');
  318. const colors = [
  319. 'rgba(59, 91, 219, 0.8)',
  320. 'rgba(0, 184, 148, 0.8)',
  321. 'rgba(34, 184, 207, 0.8)',
  322. 'rgba(255, 212, 59, 0.8)',
  323. 'rgba(255, 107, 107, 0.8)',
  324. 'rgba(142, 68, 173, 0.8)',
  325. 'rgba(230, 126, 34, 0.8)',
  326. 'rgba(149, 165, 166, 0.8)',
  327. 'rgba(241, 196, 15, 0.8)',
  328. 'rgba(231, 76, 60, 0.8)'
  329. ];
  330. const dataValues = causalsData.inData.map(item => parseFloat(item.value));
  331. const total = dataValues.reduce((sum, value) => sum + value, 0);
  332. // alwaysTooltip
  333. const alwaysShowTooltip = {
  334. id: "alwaysShowTooltip",
  335. afterDraw(chart) {
  336. const { ctx } = chart;
  337. ctx.save();
  338. const linePad = 12; // distanza dal bordo della ciambella
  339. const textOffset = 20; // distanza fissa del testo dal bordo
  340. const lineGap = 8; // margine tra linea e testo
  341. const lineWidth = 1;
  342. chart.data.datasets.forEach((dataset, di) => {
  343. const meta = chart.getDatasetMeta(di);
  344. const total = dataset.data.reduce((s, v) => s + v, 0);
  345. meta.data.forEach((datapoint, index) => {
  346. const value = dataset.data[index];
  347. const perc = total > 0 ? (value / total) * 100 : 0;
  348. const text = `${perc.toFixed(1)}%`;
  349. // calcolo angolo e anchor
  350. const { x, y } = datapoint.tooltipPosition();
  351. const coords = outsideLabelPoint({
  352. cx: datapoint.x,
  353. cy: datapoint.y,
  354. px: x,
  355. py: y,
  356. radius: datapoint.outerRadius,
  357. pad: linePad,
  358. });
  359. const ux = Math.cos(coords.theta);
  360. const uy = Math.sin(coords.theta);
  361. const ax = coords.anchor.x;
  362. const ay = coords.anchor.y;
  363. // centro testo
  364. const tx = ax + ux * textOffset;
  365. const ty = ay + uy * textOffset;
  366. // fine della linea = poco prima del testo
  367. const lx = tx - ux * lineGap;
  368. const ly = ty - uy * lineGap;
  369. // linea: anchor -> vicino al testo
  370. ctx.lineWidth = lineWidth;
  371. ctx.strokeStyle = datapoint.options.borderColor;
  372. ctx.beginPath();
  373. ctx.moveTo(ax, ay);
  374. ctx.lineTo(lx, ly);
  375. ctx.stroke();
  376. // testo
  377. ctx.font = "12px greycliff-cf, sans-serif";
  378. ctx.fillStyle = datapoint.options.borderColor;
  379. ctx.textAlign = "center";
  380. ctx.textBaseline = "middle";
  381. ctx.fillText(text, tx, ty);
  382. });
  383. });
  384. ctx.restore();
  385. },
  386. };
  387. function outsideLabelPoint({ cx, cy, px, py, radius, pad = 20 }) {
  388. const theta = Math.atan2(py - cy, px - cx); // angolo rad
  389. const anchor = { // punto sul bordo del pie
  390. x: cx + radius * Math.cos(theta),
  391. y: cy + radius * Math.sin(theta)
  392. };
  393. const label = { // punto esterno per il tooltip/etichetta
  394. x: cx + (radius + pad) * Math.cos(theta),
  395. y: cy + (radius + pad) * Math.sin(theta)
  396. };
  397. const align = {
  398. textAlign: Math.cos(theta) >= 0 ? 'left' : 'right',
  399. textBaseline: Math.sin(theta) > 0 ? 'top' : 'bottom'
  400. };
  401. return { theta, anchor, label, align };
  402. }
  403. this.charts[chartId] = new Chart(ctx, {
  404. type: 'doughnut',
  405. data: {
  406. labels: causalsData.inLabels,
  407. datasets: [{
  408. label: 'Importo',
  409. data: dataValues,
  410. backgroundColor: colors,
  411. borderColor: colors.map(color => color.replace('0.8', '1')),
  412. borderWidth: 2,
  413. hoverOffset: 8
  414. }]
  415. },
  416. options: {
  417. responsive: true,
  418. maintainAspectRatio: false,
  419. cutout: '30%',
  420. layout: {
  421. padding: {
  422. top: 30,
  423. right: 30,
  424. bottom: 30,
  425. left: 30
  426. }
  427. },
  428. plugins: {
  429. legend: {
  430. display: false
  431. },
  432. tooltip: {
  433. backgroundColor: 'rgba(255, 255, 255, 1)',
  434. titleColor: '#212529',
  435. bodyColor: '#495057',
  436. borderColor: '#e9ecef',
  437. borderWidth: 2,
  438. cornerRadius: 0,
  439. titleFont: {
  440. size: 13,
  441. weight: 'bold'
  442. },
  443. bodyFont: {
  444. size: 12,
  445. weight: '500'
  446. },
  447. padding: 12,
  448. callbacks: {
  449. title: function (context) {
  450. return context[0].label;
  451. },
  452. label: function (context) {
  453. const value = context.raw;
  454. const percentage = total > 0 ? ((value / total) * 100).toFixed(1) : 0;
  455. return [
  456. `Importo: €${new Intl.NumberFormat('it-IT', {
  457. minimumFractionDigits: 2,
  458. maximumFractionDigits: 2
  459. }).format(value)}`,
  460. `Percentuale: ${percentage}%`
  461. ];
  462. }
  463. }
  464. },
  465. },
  466. animation: {
  467. animateRotate: true,
  468. duration: 1000
  469. }
  470. },
  471. plugins: [alwaysShowTooltip]
  472. });
  473. this.updateCausalsTable(causalsData, dataValues, total);
  474. },
  475. updateCausalsTable: function (causalsData, dataValues, total) {
  476. const container = document.getElementById('causals-table');
  477. if (!container) return;
  478. const colors = [
  479. '#3b5bdb', '#00b894', '#22b8cf', '#ffd43b', '#ff6b6b',
  480. '#8e44ad', '#e67e22', '#95a5a6', '#f1c40f', '#e74c3c'
  481. ];
  482. let tableHtml = `
  483. <div class="causals-table compact">
  484. <div class="table-header">
  485. <div class="table-cell causale">Causale</div>
  486. <div class="table-cell euro">Importo</div>
  487. <!-- <div class="table-cell percent">%</div> -->
  488. </div>
  489. `;
  490. causalsData.inLabels.forEach((label, index) => {
  491. const value = dataValues[index] || 0;
  492. const percentage = total > 0 ? ((value / total) * 100).toFixed(1) : 0;
  493. const color = colors[index % colors.length];
  494. tableHtml += `
  495. <div class="table-row">
  496. <div class="table-cell causale">
  497. <span class="causale-indicator" style="background-color: ${color}"></span>
  498. ${label}
  499. </div>
  500. <div class="table-cell euro">€${new Intl.NumberFormat('it-IT', {
  501. minimumFractionDigits: 2,
  502. maximumFractionDigits: 2
  503. }).format(value)}</div>
  504. <!-- <div class="table-cell percent">${percentage}%</div> -->
  505. </div>
  506. `;
  507. });
  508. tableHtml += '</div>';
  509. container.innerHTML = tableHtml;
  510. },
  511. createMembersChart: function (seasonKey, membersData) {
  512. const chartId = `members-chart`;
  513. const canvas = document.getElementById(chartId);
  514. if (!canvas) return;
  515. this.destroyChart(chartId);
  516. const ctx = canvas.getContext('2d');
  517. const gradient = ctx.createLinearGradient(0, 0, 0, 400);
  518. gradient.addColorStop(0, 'rgba(59, 91, 219, 0.3)');
  519. gradient.addColorStop(1, 'rgba(59, 91, 219, 0.05)');
  520. const processedDatasets = membersData.datasets.map((dataset, index) => {
  521. if (dataset.label === 'Totale Membri Tesserati') {
  522. return {
  523. ...dataset,
  524. // backgroundColor: gradient,
  525. backgroundColor: dataset.borderColor,
  526. borderColor: dataset.borderColor,
  527. borderWidth: 0,
  528. type: 'bar',
  529. order: 1,
  530. fill: true,
  531. barThickness: "flex",
  532. barPercentage: 0.65,
  533. categoryPercentage: 0.4,
  534. };
  535. } else {
  536. return {
  537. ...dataset,
  538. backgroundColor: dataset.borderColor,
  539. borderColor: dataset.borderColor,
  540. borderWidth: 0,
  541. type: 'bar',
  542. order: 2,
  543. fill: true,
  544. barThickness: "flex",
  545. barPercentage: 0.65,
  546. categoryPercentage: 0.4,
  547. };
  548. }
  549. });
  550. let options = {
  551. type: 'bar',
  552. data: {
  553. labels: membersData.labels,
  554. datasets: processedDatasets
  555. },
  556. options: {
  557. responsive: true,
  558. maintainAspectRatio: false,
  559. interaction: {
  560. mode: 'index',
  561. intersect: false,
  562. },
  563. plugins: {
  564. legend: {
  565. position: 'bottom',
  566. labels: {
  567. usePointStyle: true,
  568. padding: 20,
  569. pointStyle: "rect",
  570. font: { weight: '500' }
  571. }
  572. },
  573. tooltip: {
  574. backgroundColor: 'rgba(255, 255, 255, 1)',
  575. titleColor: '#212529',
  576. bodyColor: '#495057',
  577. borderColor: '#e9ecef',
  578. borderWidth: 2,
  579. cornerRadius: 0,
  580. callbacks: {
  581. title: function (context) {
  582. return 'Stagione: ' + context[0].label;
  583. },
  584. label: function (context) {
  585. return context.dataset.label + ': ' + context.parsed.y;
  586. }
  587. }
  588. }
  589. },
  590. scales: {
  591. x: {
  592. grid: { display: false },
  593. ticks: {
  594. font: { weight: '500' }
  595. }
  596. },
  597. y: {
  598. beginAtZero: true,
  599. grid: { color: 'rgba(0, 0, 0, 0.05)' },
  600. ticks: {
  601. precision: 0,
  602. callback: function (value) {
  603. return Math.floor(value); // Ensure integer values
  604. }
  605. }
  606. }
  607. },
  608. animation: {
  609. duration: 1000,
  610. easing: 'easeOutQuart'
  611. }
  612. }
  613. };
  614. this.charts[chartId] = new Chart(ctx, options);
  615. },
  616. updateMonthlyTable: function (monthlyData) {
  617. const container = document.getElementById('monthly-table');
  618. if (!container) return;
  619. const incomeData = monthlyData.datasets[0].data;
  620. const expenseData = monthlyData.datasets[1].data;
  621. const monthNames = monthlyData.labels;
  622. let tableHtml = `
  623. <div class="monthly-table">
  624. <div class="table-header">
  625. <div class="table-cell month">Mese</div>
  626. <div class="table-cell">Entrate</div>
  627. <div class="table-cell">Uscite</div>
  628. <div class="table-cell">Delta</div>
  629. </div>
  630. `;
  631. monthNames.forEach((month, index) => {
  632. const income = parseFloat(incomeData[index] || 0);
  633. const expense = parseFloat(expenseData[index] || 0);
  634. const net = income - expense;
  635. const rowClass = net < 0 ? 'negative' : (net > 0 ? 'positive' : 'neutral');
  636. const plusSymbol = net > 0 ? "+" : "";
  637. tableHtml += `
  638. <div class="table-row ${rowClass}">
  639. <div class="table-cell month-name">${month}</div>
  640. <div class="table-cell income">€${new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(income)}</div>
  641. <div class="table-cell expense">€${new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(expense)}</div>
  642. <div class="table-cell net">€${plusSymbol}${new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(net)}</div>
  643. </div>
  644. `;
  645. });
  646. tableHtml += '</div>';
  647. container.innerHTML = tableHtml;
  648. },
  649. updateYearlyTable: function (yearlyData) {
  650. const container = document.getElementById('yearly-table');
  651. if (!container) return;
  652. const incomeData = yearlyData.datasets[0].data;
  653. const expenseData = yearlyData.datasets[1].data;
  654. const years = yearlyData.labels;
  655. let tableHtml = `
  656. <div class="monthly-table">
  657. <div class="table-header">
  658. <div class="table-cell month">Anno</div>
  659. <div class="table-cell">Entrate</div>
  660. <div class="table-cell">Uscite</div>
  661. <div class="table-cell">Delta</div>
  662. </div>
  663. `;
  664. years.forEach((year, index) => {
  665. const income = parseFloat(incomeData[index] || 0);
  666. const expense = parseFloat(expenseData[index] || 0);
  667. const net = income - expense;
  668. const rowClass = net < 0 ? 'negative' : (net > 0 ? 'positive' : 'neutral');
  669. tableHtml += `
  670. <div class="table-row ${rowClass}">
  671. <div class="table-cell month-name">${year}</div>
  672. <div class="table-cell income">€${new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(income)}</div>
  673. <div class="table-cell expense">€${new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(expense)}</div>
  674. <div class="table-cell net">€${new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(net)}</div>
  675. </div>
  676. `;
  677. });
  678. tableHtml += '</div>';
  679. container.innerHTML = tableHtml;
  680. },
  681. updateMembersTable: function (membersData) {
  682. const container = document.getElementById('members-table');
  683. if (!container) return;
  684. const seasonLabels = membersData.labels;
  685. const totalDataset = membersData.datasets.find(d => d.label === 'Totale Membri Tesserati');
  686. const cardTypeDatasets = membersData.datasets.filter(d => d.label !== 'Totale Membri Tesserati');
  687. const memberCounts = totalDataset ? totalDataset.data : [];
  688. let tableHtml = `
  689. <div class="members-table">
  690. <div class="table-header">
  691. <div class="table-cell">Stagione</div>
  692. <div class="table-cell">Totale</div>
  693. <div class="table-cell">Variazione</div>
  694. <div class="table-cell">Ente</div>
  695. </div>
  696. `;
  697. seasonLabels.forEach((season, index) => {
  698. const current = parseInt(memberCounts[index] || 0);
  699. const previous = index > 0 ? parseInt(memberCounts[index - 1] || 0) : 0;
  700. const variation = index > 0 ? current - previous : 0;
  701. const variationPercent = previous > 0 ? Math.round((variation / previous) * 100 * 10) / 10 : 0;
  702. const rowClass = variation > 0 ? 'positive' : (variation < 0 ? 'negative' : 'neutral');
  703. let variationText = '—';
  704. if (index > 0) {
  705. if (variation > 0) {
  706. variationText = `<span class="variation-positive">+${variation} (+${variationPercent}%)</span>`;
  707. } else if (variation < 0) {
  708. variationText = `<span class="variation-negative">${variation} (${variationPercent}%)</span>`;
  709. } else {
  710. variationText = `<span class="variation-neutral">${variation}</span>`;
  711. }
  712. }
  713. // Build card type breakdown
  714. let cardTypeBreakdown = '';
  715. cardTypeDatasets.forEach((dataset, datasetIndex) => {
  716. const count = dataset.data[index] || 0;
  717. if (count > 0) {
  718. const color = dataset.borderColor || '#6b7280';
  719. cardTypeBreakdown += `
  720. <div class="card-type-item">
  721. <span class="card-type-indicator" style="background-color: ${color}"></span>
  722. <span class="card-type-name">${dataset.label}</span>
  723. <span class="card-type-count">${count}</span>
  724. </div>
  725. `;
  726. }
  727. });
  728. if (!cardTypeBreakdown) {
  729. cardTypeBreakdown = '<div class="no-card-types">Nessun dettaglio</div>';
  730. }
  731. tableHtml += `
  732. <div class="table-row ${rowClass}">
  733. <div class="table-cell season-name">${season}</div>
  734. <div class="table-cell members-count">${new Intl.NumberFormat('it-IT').format(current)}</div>
  735. <div class="table-cell variation">${variationText}</div>
  736. <div class="table-cell card-types">
  737. <div class="card-types-container">
  738. ${cardTypeBreakdown}
  739. </div>
  740. </div>
  741. </div>
  742. `;
  743. });
  744. tableHtml += '</div>';
  745. container.innerHTML = tableHtml;
  746. },
  747. createCourseChart: function () {
  748. console.log('Creating course chart...');
  749. const seasonFilter = @this.seasonFilter;
  750. const selectedCourse = @this.selectedCourse ?? '';
  751. const seasonKey = seasonFilter.replace('-', '');
  752. console.log('Selected course:', selectedCourse, 'for season:', seasonFilter);
  753. if (!selectedCourse || selectedCourse.trim() === '') {
  754. console.log('No course selected, skipping chart creation');
  755. return;
  756. }
  757. const chartId = `courses-chart-${seasonKey}-${selectedCourse}`;
  758. const canvas = document.getElementById(chartId);
  759. if (!canvas) return;
  760. this.destroyChart(chartId);
  761. const courseData = @json($this->getCourseMonthlyEarnings());
  762. const ctx = canvas.getContext('2d');
  763. this.charts[chartId] = new Chart(ctx, {
  764. type: 'bar',
  765. data: {
  766. labels: courseData.labels,
  767. datasets: courseData.datasets.map(dataset => {
  768. if (dataset.type === 'line') {
  769. return {
  770. ...dataset,
  771. type: 'line',
  772. fill: false,
  773. backgroundColor: 'transparent'
  774. };
  775. }
  776. return dataset;
  777. })
  778. },
  779. options: {
  780. responsive: true,
  781. maintainAspectRatio: false,
  782. interaction: {
  783. mode: 'index',
  784. intersect: false,
  785. },
  786. scales: {
  787. x: {
  788. grid: { display: false },
  789. ticks: { font: { weight: '500' } }
  790. },
  791. y: {
  792. beginAtZero: true,
  793. grid: {
  794. color: 'rgba(0, 0, 0, 1)',
  795. borderDash: [5, 5]
  796. },
  797. ticks: {
  798. callback: function (value) {
  799. return '€' + new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(value);
  800. }
  801. }
  802. }
  803. },
  804. plugins: {
  805. legend: {
  806. position: 'bottom',
  807. labels: {
  808. usePointStyle: true,
  809. padding: 20,
  810. pointStyle: "rect",
  811. font: { weight: '500' }
  812. }
  813. },
  814. tooltip: {
  815. backgroundColor: 'rgba(255, 255, 255, 1)',
  816. titleColor: '#212529',
  817. bodyColor: '#495057',
  818. borderColor: '#e9ecef',
  819. borderWidth: 2,
  820. cornerRadius: 0,
  821. callbacks: {
  822. label: function (context) {
  823. return context.dataset.label + ': €' +
  824. new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(context.parsed.y);
  825. }
  826. }
  827. }
  828. },
  829. animation: {
  830. duration: 1000,
  831. easing: 'easeOutQuart'
  832. }
  833. }
  834. });
  835. },
  836. createCourseChartWithValue: function (selectedCourseValue) {
  837. console.log('Creating modern course chart with value:', selectedCourseValue);
  838. const seasonFilter = @this.seasonFilter;
  839. const seasonKey = seasonFilter.replace('-', '');
  840. const chartId = `courses-chart-${seasonKey}-${selectedCourseValue}`;
  841. const tableId = `course-delta-table-${seasonKey}-${selectedCourseValue}`;
  842. let canvas = document.getElementById(chartId);
  843. const tableContainer = document.getElementById(tableId);
  844. if (!canvas) {
  845. console.log('Canvas not found for chart ID:', chartId);
  846. const chartContainer = document.querySelector('.modern-chart-container');
  847. if (chartContainer) {
  848. chartContainer.innerHTML = `<div class="chart-empty-state">
  849. <div style="text-align: center; padding: 4rem 2rem;">
  850. <div style="font-size: 4rem; margin-bottom: 1.5rem; opacity: 0.3;">📊</div>
  851. <h3 style="font-size: 1.5rem; font-weight: 600; margin-bottom: 1rem; color: #374151;">
  852. Grafico non disponibile
  853. </h3>
  854. <p style="font-size: 1rem; opacity: 0.7; margin: 0; max-width: 400px; margin-left: auto; margin-right: auto; line-height: 1.5;">
  855. Il grafico per questo corso non può essere visualizzato nella stagione selezionata.
  856. </p>
  857. </div>
  858. </div>`;
  859. }
  860. if (tableContainer) {
  861. tableContainer.innerHTML = '';
  862. }
  863. return;
  864. }
  865. this.destroyChart(chartId);
  866. @this.call('getCourseData', selectedCourseValue).then(courseData => {
  867. console.log('Received course data:', courseData);
  868. if (courseData.isEmpty) {
  869. console.log('No data available for course, showing message');
  870. if (tableContainer) {
  871. tableContainer.innerHTML = '';
  872. }
  873. const chartContainer = canvas.parentElement;
  874. chartContainer.innerHTML = `<div class="chart-empty-state">
  875. <div style="text-align: center; padding: 4rem 2rem;">
  876. <div style="font-size: 4rem; margin-bottom: 1.5rem; opacity: 0.3;">📊</div>
  877. <h3 style="font-size: 1.5rem; font-weight: 600; margin-bottom: 1rem; color: #374151;">
  878. ${courseData.message}
  879. </h3>
  880. <p style="font-size: 1rem; opacity: 0.7; margin: 0; max-width: 400px; margin-left: auto; margin-right: auto; line-height: 1.5;">
  881. Questo corso non ha pagamenti registrati per la stagione selezionata.
  882. </p>
  883. </div>
  884. </div>`;
  885. return;
  886. }
  887. if (!courseData || !courseData.labels || courseData.labels.length === 0) {
  888. console.log('No data available for chart');
  889. return;
  890. }
  891. let canvasElement = document.getElementById(chartId);
  892. if (!canvasElement) {
  893. const chartContainer = canvas.parentElement;
  894. chartContainer.innerHTML = `<canvas id="${chartId}"></canvas>`;
  895. canvasElement = document.getElementById(chartId);
  896. }
  897. this.updateCourseTable(tableContainer, courseData.tableData);
  898. const participantData = courseData.datasets.find(d => d.participantData)?.participantData || [];
  899. const suspendedData = courseData.datasets.find(d => d.suspendedData)?.suspendedData || [];
  900. const monthNamesExtended = courseData.datasets.find(d => d.monthNamesExtended)?.monthNamesExtended || [];
  901. const ctx = canvasElement.getContext('2d');
  902. const earnedGradient = ctx.createLinearGradient(0, 0, 0, 400);
  903. earnedGradient.addColorStop(0, 'rgba(16, 185, 129, 1)');
  904. earnedGradient.addColorStop(1, 'rgba(16, 185, 129, 1)');
  905. const totalData = courseData.datasets.find(d => d.label === 'TOT. DA INCASSARE')?.data || [];
  906. const earnedData = courseData.datasets.find(d => d.label === 'TOT. INCASSATO')?.data || [];
  907. this.charts[chartId] = new Chart(ctx, {
  908. type: 'bar',
  909. data: {
  910. labels: courseData.labels,
  911. datasets: [
  912. {
  913. label: 'TOT. INCASSATO',
  914. backgroundColor: earnedGradient,
  915. borderColor: 'rgba(16, 185, 129, 1)',
  916. borderWidth: 0,
  917. borderRadius: {
  918. topLeft: 8,
  919. topRight: 8,
  920. bottomLeft: 0,
  921. bottomRight: 0,
  922. },
  923. borderSkipped: true,
  924. data: earnedData,
  925. type: 'bar',
  926. barThickness: "flex",
  927. barPercentage: 0.65,
  928. categoryPercentage: 0.25,
  929. order: 1,
  930. participantData: participantData,
  931. suspendedData: suspendedData,
  932. monthNamesExtended: monthNamesExtended,
  933. },
  934. {
  935. label: 'TOT. DA INCASSARE',
  936. backgroundColor: '#F28322',
  937. borderColor: '#F28322',
  938. borderWidth: 0,
  939. borderRadius: {
  940. topLeft: 8,
  941. topRight: 8,
  942. bottomLeft: 0,
  943. bottomRight: 0,
  944. },
  945. borderSkipped: true,
  946. data: totalData,
  947. type: 'bar',
  948. barThickness: "flex",
  949. barPercentage: 0.65,
  950. categoryPercentage: 0.25,
  951. order: 2,
  952. participantData: participantData,
  953. suspendedData: suspendedData,
  954. monthNamesExtended: monthNamesExtended,
  955. }
  956. ]
  957. },
  958. options: {
  959. responsive: true,
  960. maintainAspectRatio: false,
  961. interaction: {
  962. mode: 'index',
  963. intersect: false,
  964. },
  965. layout: {
  966. padding: {
  967. top: 20,
  968. right: 20,
  969. bottom: 20,
  970. left: 10
  971. }
  972. },
  973. scales: {
  974. x: {
  975. stacked: true,
  976. grid: {
  977. display: false
  978. },
  979. ticks: {
  980. font: {
  981. weight: '600',
  982. size: 13
  983. },
  984. color: '#6b7280'
  985. },
  986. border: {
  987. display: false
  988. }
  989. },
  990. y: {
  991. stacked: true,
  992. beginAtZero: true,
  993. grid: {
  994. color: 'rgba(156, 163, 175, 0.15)',
  995. borderDash: [3, 3]
  996. },
  997. border: {
  998. display: false
  999. },
  1000. ticks: {
  1001. font: {
  1002. size: 12,
  1003. weight: '500'
  1004. },
  1005. color: '#6b7280',
  1006. callback: function (value) {
  1007. return '€' + new Intl.NumberFormat('it-IT', {
  1008. minimumFractionDigits: 0,
  1009. maximumFractionDigits: 0
  1010. }).format(value);
  1011. }
  1012. }
  1013. }
  1014. },
  1015. plugins: {
  1016. legend: {
  1017. position: 'bottom',
  1018. labels: {
  1019. usePointStyle: true,
  1020. padding: 20,
  1021. pointStyle: "rect",
  1022. font: { weight: '500' }
  1023. }
  1024. },
  1025. tooltip: {
  1026. backgroundColor: 'rgba(255, 255, 255, 1)',
  1027. borderColor: 'rgba(229, 231, 235, 0.8)',
  1028. borderWidth: 2,
  1029. cornerRadius: 0,
  1030. titleFont: {
  1031. size: 15,
  1032. weight: 'bold',
  1033. },
  1034. titleColor: '#111827',
  1035. bodyFont: {
  1036. size: 14,
  1037. weight: '400',
  1038. },
  1039. bodyColor: '#111827',
  1040. footerFont: {
  1041. size: 14,
  1042. weight: '400',
  1043. },
  1044. // footerColor: '#0C6197',
  1045. footerSpacing: 0,
  1046. footerMarginTop: 0,
  1047. padding: 16,
  1048. boxPadding: 8,
  1049. usePointStyle: true,
  1050. displayColors: true,
  1051. callbacks: {
  1052. title: function (tooltipItems) {
  1053. let sum = 0;
  1054. tooltipItems.forEach(function(tooltipItem) {
  1055. sum += tooltipItem.parsed.y;
  1056. });
  1057. let item = tooltipItems[0];
  1058. let index = item.dataIndex;
  1059. let monthNameExtended = item.dataset["monthNamesExtended"] ? item.dataset["monthNamesExtended"][index] : 0;
  1060. // return item.label + '\n' + 'TOTALE ATTESO: €' + new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(sum);
  1061. return monthNameExtended + '\n' + 'TOTALE ATTESO: €' + new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(sum);
  1062. },
  1063. // labelTextColor: function(tooltipItems) {
  1064. // return tooltipItems.dataset.backgroundColor;
  1065. // },
  1066. label: function (tooltipItems) {
  1067. let label = tooltipItems.dataset.label + ': €' + new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(tooltipItems.parsed.y);
  1068. return label;
  1069. },
  1070. // footer: function (tooltipItems) {
  1071. // let item = tooltipItems[0];
  1072. // let index = item.dataIndex;
  1073. // let suspendedData = item.dataset["suspendedData"] ? item.dataset["suspendedData"][index] : 0;
  1074. // return "TOTALE SOSPESI: " + suspendedData;
  1075. // }
  1076. }
  1077. }
  1078. },
  1079. animation: {
  1080. duration: 1500,
  1081. easing: 'easeOutCubic'
  1082. },
  1083. elements: {
  1084. bar: {
  1085. borderRadius: {
  1086. topLeft: 8,
  1087. topRight: 8,
  1088. bottomLeft: 0,
  1089. bottomRight: 0
  1090. }
  1091. },
  1092. line: {
  1093. borderCapStyle: 'round',
  1094. borderJoinStyle: 'round'
  1095. },
  1096. point: {
  1097. hoverBorderWidth: 2,
  1098. borderWidth: 1
  1099. }
  1100. }
  1101. }
  1102. });
  1103. }).catch(error => {
  1104. console.error('Error calling getCourseData:', error);
  1105. });
  1106. },
  1107. updateCourseTable: function (container, tableData) {
  1108. if (!container || !tableData) return;
  1109. let tableHtml = `<div class="course-table">
  1110. <div class="table-header">
  1111. <div class="table-cell month">MESE</div>
  1112. <div class="table-cell percentage">TOT. INCASSATO</div>
  1113. <div class="table-cell delta">TOT. DA INCASSARE</div>
  1114. <div class="table-cell suspended">SOSPESI</div>
  1115. </div>`;
  1116. tableData.forEach(row => {
  1117. const earned = parseFloat(row.earned) || 0;
  1118. const total = parseFloat(row.total) || 0;
  1119. const delta = Math.max(0, total - earned);
  1120. let percentage = 0;
  1121. let percentageDisplay = '—';
  1122. let percentageClass = 'neutral';
  1123. if (total > 0) {
  1124. percentage = Math.round((earned / total) * 100);
  1125. percentageDisplay = percentage + '%';
  1126. // Color based on completion
  1127. if (percentage >= 100) {
  1128. percentageClass = 'good';
  1129. } else if (percentage >= 80) {
  1130. percentageClass = 'warning';
  1131. } else {
  1132. percentageClass = 'bad';
  1133. }
  1134. }
  1135. // Delta styling: positive when delta is 0 (fully paid), negative when there's missing amount
  1136. const deltaClass = (total > 0 && delta === 0) ? 'positive' : (delta > 0) ? 'negative' : 'neutral';
  1137. const earnedClass = (earned > 0) ? 'positive' : 'neutral';
  1138. tableHtml += `
  1139. <div class="table-row">
  1140. <div class="table-cell month">${row.month}</div>
  1141. <!-- <div class="table-cell percentage ${percentageClass}">${percentageDisplay}</div> -->
  1142. <div class="table-cell earned ${earnedClass}">€${new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(earned)}</div>
  1143. <div class="table-cell delta ${deltaClass}">€${new Intl.NumberFormat('it-IT', {minimumFractionDigits: 2, maximumFractionDigits: 2}).format(delta)}</div>
  1144. <div class="table-cell suspended">${row.suspended}</div>
  1145. </div>
  1146. `;
  1147. });
  1148. tableHtml += '</div>';
  1149. container.innerHTML = tableHtml;
  1150. },
  1151. updateCourseChart: function () {
  1152. if (this.selectedCourse) {
  1153. const seasonFilter = @json($seasonFilter);
  1154. const seasonKey = seasonFilter.replace('-', '');
  1155. this.createCourseChartWithValue(this.selectedCourse);
  1156. }
  1157. }
  1158. };
  1159. document.addEventListener('DOMContentLoaded', function () {
  1160. setTimeout(() => {
  1161. window.ReportsChartManager.updateMainCharts();
  1162. }, 100);
  1163. });
  1164. document.addEventListener('livewire:navigated', function () {
  1165. setTimeout(() => {
  1166. window.ReportsChartManager.updateMainCharts();
  1167. }, 100);
  1168. });
  1169. document.addEventListener('livewire:updated', function (event) {
  1170. console.log('Livewire updated, waiting for component to fully update...');
  1171. setTimeout(() => {
  1172. console.log('Now updating charts after delay');
  1173. window.ReportsChartManager.forceUpdateCharts();
  1174. }, 800);
  1175. });
  1176. document.addEventListener('livewire:load', function () {
  1177. Livewire.on('courseSelected', (courseId) => {
  1178. console.log('Course selected event received:', courseId);
  1179. setTimeout(() => {
  1180. window.ReportsChartManager.createCourseChartWithValue(courseId);
  1181. }, 200);
  1182. });
  1183. Livewire.on('chartsUpdated', () => {
  1184. console.log('Charts updated event received from Livewire');
  1185. setTimeout(() => {
  1186. window.ReportsChartManager.forceUpdateCharts();
  1187. }, 200);
  1188. });
  1189. });
  1190. </script>
  1191. </div>