ng-prime-tools 1.0.24 → 1.0.25

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -636,6 +636,10 @@ export class FormBuilderTesterComponent implements OnInit {
636
636
 
637
637
  ## Changelog
638
638
 
639
+ ### Version 1.0.25 - Release Date: 06/03/2025
640
+
641
+ - adding pt-chart-comparison
642
+
639
643
  ### Version 1.0.24 - Release Date: 05/03/2025
640
644
 
641
645
  - adding pt-group
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhcnQtY29uZmlnLm1vZGVsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctcHJpbWUtdG9vbHMvc3JjL2xpYi9tb2RlbHMvY2hhcnQtY29uZmlnLm1vZGVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDaGFydERhdGEsIENoYXJ0T3B0aW9ucywgQ2hhcnRUeXBlIH0gZnJvbSAnY2hhcnQuanMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIERhdGFzZXQge1xuICBsYWJlbD86IHN0cmluZztcbiAgZGF0YTogbnVtYmVyW10gfCBhbnlbXTtcbiAgYmFja2dyb3VuZENvbG9yPzogc3RyaW5nIHwgc3RyaW5nW107XG4gIGJvcmRlckNvbG9yPzogc3RyaW5nIHwgc3RyaW5nW107XG4gIGJvcmRlcldpZHRoPzogbnVtYmVyO1xuICBmaWxsPzogYm9vbGVhbjtcbiAgdGVuc2lvbj86IG51bWJlcjtcbiAgc3RhY2s/OiBzdHJpbmc7XG4gIHR5cGU/OiBDaGFydFR5cGU7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ2hhcnRDb25maWcge1xuICB0eXBlOiBDaGFydFR5cGU7XG4gIGRhdGE6IENoYXJ0RGF0YTtcbiAgb3B0aW9ucz86IENoYXJ0T3B0aW9ucztcbn1cbiJdfQ==
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhcnQtY29uZmlnLm1vZGVsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctcHJpbWUtdG9vbHMvc3JjL2xpYi9tb2RlbHMvY2hhcnQtY29uZmlnLm1vZGVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDaGFydERhdGEsIENoYXJ0T3B0aW9ucywgQ2hhcnRUeXBlIH0gZnJvbSAnY2hhcnQuanMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIERhdGFzZXQge1xuICBsYWJlbD86IHN0cmluZztcbiAgZGF0YTogbnVtYmVyW10gfCBhbnlbXTtcbiAgYmFja2dyb3VuZENvbG9yPzogc3RyaW5nIHwgc3RyaW5nW107XG4gIGJvcmRlckNvbG9yPzogc3RyaW5nIHwgc3RyaW5nW107XG4gIGJvcmRlcldpZHRoPzogbnVtYmVyO1xuICBmaWxsPzogYm9vbGVhbjtcbiAgdGVuc2lvbj86IG51bWJlcjtcbiAgc3RhY2s/OiBzdHJpbmc7XG4gIHR5cGU/OiBDaGFydFR5cGU7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ2hhcnRDb25maWcge1xuICB0eXBlOiBDaGFydFR5cGU7XG4gIGRhdGE6IENoYXJ0RGF0YTtcbiAgY2hhcnRIZWlnaHQ/OiBzdHJpbmc7IC8vIERpcmVjdGx5IGluc2lkZSBDaGFydENvbmZpZ1xuICBjaGFydFdpZHRoPzogc3RyaW5nOyAvLyBEaXJlY3RseSBpbnNpZGUgQ2hhcnRDb25maWdcbiAgbWVkaWFuVGl0bGU/OiBzdHJpbmc7IC8vIERpcmVjdGx5IGluc2lkZSBDaGFydENvbmZpZ1xuICB4QXhpc1RpdGxlPzogc3RyaW5nOyAvLyBEaXJlY3RseSBpbnNpZGUgQ2hhcnRDb25maWdcbiAgeUF4aXNUaXRsZT86IHN0cmluZzsgLy8gRGlyZWN0bHkgaW5zaWRlIENoYXJ0Q29uZmlnXG4gIHNjYWxlcz86IHtcbiAgICB5Pzoge1xuICAgICAgbWluPzogbnVtYmVyO1xuICAgICAgbWF4PzogbnVtYmVyO1xuICAgICAgdGlja3M/OiB7XG4gICAgICAgIHN0ZXBTaXplPzogbnVtYmVyO1xuICAgICAgICBiZWdpbkF0WmVybz86IGJvb2xlYW47XG4gICAgICB9O1xuICAgIH07XG4gIH07XG4gIG9wdGlvbnM/OiBDaGFydE9wdGlvbnM7IC8vIEtlZXBpbmcgdGhpcyBmb3Igb3RoZXIgQ2hhcnQuanMtc3BlY2lmaWMgY29uZmlndXJhdGlvbnNcbn1cbiJdfQ==
@@ -0,0 +1,2 @@
1
+ export * from './public-api';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1wcmltZS10b29scy9zcmMvbGliL3B0LWNoYXJ0LWNvbXBhcmlzb24vaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxjQUFjLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3B1YmxpYy1hcGknO1xuIl19
@@ -0,0 +1,153 @@
1
+ import { Component, Input, ViewChild, } from '@angular/core';
2
+ import { Chart, registerables, } from 'chart.js';
3
+ import ChartDataLabels from 'chartjs-plugin-datalabels';
4
+ import * as i0 from "@angular/core";
5
+ export class PTChartComparisonComponent {
6
+ constructor() {
7
+ this.medianTitle = 'Médiane';
8
+ this.xAxisTitle = 'Time';
9
+ this.yAxisTitle = 'Value';
10
+ // Chart dimension inputs
11
+ this.chartHeight = '400px';
12
+ this.chartWidth = '1200px';
13
+ Chart.register(...registerables, ChartDataLabels);
14
+ }
15
+ ngOnInit() {
16
+ this.initializeChart();
17
+ }
18
+ ngOnDestroy() {
19
+ this.destroyChart();
20
+ }
21
+ /**
22
+ * Initializes the chart with median and datasets
23
+ */
24
+ initializeChart() {
25
+ const canvas = this.canvasRef.nativeElement;
26
+ this.destroyChart();
27
+ const config = {
28
+ type: this.chartConfig.type || 'line',
29
+ data: this.getFormattedChartData(), // Get data with median
30
+ options: this.getChartOptions(),
31
+ };
32
+ this.chart = new Chart(canvas, config);
33
+ }
34
+ /**
35
+ * Computes median values and adds them to the dataset
36
+ */
37
+ getFormattedChartData() {
38
+ const medianValues = this.calculateMedian();
39
+ return {
40
+ labels: this.chartConfig.data.labels,
41
+ datasets: [
42
+ // Add median line dataset
43
+ {
44
+ label: this.medianTitle || 'Médiane',
45
+ data: medianValues,
46
+ borderColor: '#0000FF', // Blue color
47
+ borderWidth: 2,
48
+ backgroundColor: 'transparent',
49
+ pointRadius: 0,
50
+ fill: false,
51
+ tension: 0.1,
52
+ borderDash: [5, 5], // Dashed line
53
+ },
54
+ // Keep existing datasets
55
+ ...this.chartConfig.data.datasets,
56
+ ],
57
+ };
58
+ }
59
+ /**
60
+ * Computes median values from all datasets at each time point
61
+ */
62
+ calculateMedian() {
63
+ const datasets = this.chartConfig.data.datasets;
64
+ return this.chartConfig.data.labels.map((_, index) => {
65
+ const valuesAtTime = datasets
66
+ .map((dataset) => dataset.data[index])
67
+ .filter((val) => val !== undefined);
68
+ if (valuesAtTime.length === 0)
69
+ return 0;
70
+ valuesAtTime.sort((a, b) => a - b);
71
+ const middle = Math.floor(valuesAtTime.length / 2);
72
+ return valuesAtTime.length % 2 === 0
73
+ ? (valuesAtTime[middle - 1] + valuesAtTime[middle]) / 2
74
+ : valuesAtTime[middle];
75
+ });
76
+ }
77
+ /**
78
+ * Defines chart options including Y-axis scaling
79
+ */
80
+ getChartOptions() {
81
+ return {
82
+ responsive: true,
83
+ maintainAspectRatio: false,
84
+ plugins: {
85
+ legend: { display: true, position: 'top' },
86
+ tooltip: { mode: 'index', intersect: false },
87
+ },
88
+ scales: {
89
+ x: {
90
+ title: { display: true, text: this.chartConfig.xAxisTitle || 'Time' },
91
+ ticks: { font: { size: 12 } },
92
+ },
93
+ y: {
94
+ title: {
95
+ display: true,
96
+ text: this.chartConfig.yAxisTitle || 'Value',
97
+ },
98
+ min: this.chartConfig.scales?.y?.min,
99
+ max: this.chartConfig.scales?.y?.max,
100
+ ticks: {
101
+ stepSize: this.chartConfig.scales?.y?.ticks?.stepSize,
102
+ font: { size: 16, weight: 'bold' },
103
+ color: '#333',
104
+ },
105
+ grid: {
106
+ color: 'rgba(0,0,0,0.1)',
107
+ },
108
+ border: {
109
+ display: true,
110
+ color: '#000',
111
+ },
112
+ },
113
+ },
114
+ };
115
+ }
116
+ /**
117
+ * Destroys the existing chart instance if present
118
+ */
119
+ destroyChart() {
120
+ if (this.chart) {
121
+ this.chart.destroy();
122
+ this.chart = undefined;
123
+ }
124
+ }
125
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTChartComparisonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
126
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.11", type: PTChartComparisonComponent, selector: "pt-chart-comparison", inputs: { chartConfig: "chartConfig", medianTitle: "medianTitle", xAxisTitle: "xAxisTitle", yAxisTitle: "yAxisTitle", yMin: "yMin", yMax: "yMax", yStepSize: "yStepSize", chartHeight: "chartHeight", chartWidth: "chartWidth" }, viewQueries: [{ propertyName: "canvasRef", first: true, predicate: ["chartCanvas"], descendants: true, static: true }], ngImport: i0, template: "<div class=\"chart-scroll-container\">\n <div\n class=\"chart-inner\"\n [style.width]=\"chartConfig.chartWidth || '1200px'\"\n [style.height]=\"chartConfig.chartHeight || '400px'\"\n >\n <canvas #chartCanvas></canvas>\n </div>\n</div>\n", styles: [".chart-scroll-container{width:100%;overflow-x:auto;white-space:nowrap;padding-bottom:10px}.chart-inner{display:inline-block}\n"] }); }
127
+ }
128
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTChartComparisonComponent, decorators: [{
129
+ type: Component,
130
+ args: [{ selector: 'pt-chart-comparison', template: "<div class=\"chart-scroll-container\">\n <div\n class=\"chart-inner\"\n [style.width]=\"chartConfig.chartWidth || '1200px'\"\n [style.height]=\"chartConfig.chartHeight || '400px'\"\n >\n <canvas #chartCanvas></canvas>\n </div>\n</div>\n", styles: [".chart-scroll-container{width:100%;overflow-x:auto;white-space:nowrap;padding-bottom:10px}.chart-inner{display:inline-block}\n"] }]
131
+ }], ctorParameters: () => [], propDecorators: { chartConfig: [{
132
+ type: Input
133
+ }], medianTitle: [{
134
+ type: Input
135
+ }], xAxisTitle: [{
136
+ type: Input
137
+ }], yAxisTitle: [{
138
+ type: Input
139
+ }], yMin: [{
140
+ type: Input
141
+ }], yMax: [{
142
+ type: Input
143
+ }], yStepSize: [{
144
+ type: Input
145
+ }], chartHeight: [{
146
+ type: Input
147
+ }], chartWidth: [{
148
+ type: Input
149
+ }], canvasRef: [{
150
+ type: ViewChild,
151
+ args: ['chartCanvas', { static: true }]
152
+ }] } });
153
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pt-chart-comparison.component.js","sourceRoot":"","sources":["../../../../../projects/ng-prime-tools/src/lib/pt-chart-comparison/pt-chart-comparison.component.ts","../../../../../projects/ng-prime-tools/src/lib/pt-chart-comparison/pt-chart-comparison.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  OnInit,\n  OnDestroy,\n  ViewChild,\n  ElementRef,\n} from '@angular/core';\nimport {\n  Chart,\n  ChartConfiguration,\n  registerables,\n  ChartData,\n  ChartOptions,\n} from 'chart.js';\nimport ChartDataLabels from 'chartjs-plugin-datalabels';\nimport { ChartConfig, Dataset } from '../models/chart-config.model';\n\n@Component({\n  selector: 'pt-chart-comparison',\n  templateUrl: './pt-chart-comparison.component.html',\n  styleUrls: ['./pt-chart-comparison.component.css'],\n})\nexport class PTChartComparisonComponent implements OnInit, OnDestroy {\n  @Input() chartConfig!: ChartConfig;\n  @Input() medianTitle: string = 'Médiane';\n  @Input() xAxisTitle: string = 'Time';\n  @Input() yAxisTitle: string = 'Value';\n\n  // Y-Axis manual inputs\n  @Input() yMin!: number;\n  @Input() yMax!: number;\n  @Input() yStepSize!: number;\n\n  // Chart dimension inputs\n  @Input() chartHeight: string = '400px';\n  @Input() chartWidth: string = '1200px';\n\n  @ViewChild('chartCanvas', { static: true })\n  canvasRef!: ElementRef<HTMLCanvasElement>;\n\n  private chart!: Chart;\n\n  constructor() {\n    Chart.register(...registerables, ChartDataLabels);\n  }\n\n  ngOnInit() {\n    this.initializeChart();\n  }\n\n  ngOnDestroy() {\n    this.destroyChart();\n  }\n\n  /**\n   * Initializes the chart with median and datasets\n   */\n  private initializeChart() {\n    const canvas = this.canvasRef.nativeElement;\n    this.destroyChart();\n\n    const config: ChartConfiguration = {\n      type: this.chartConfig.type || 'line',\n      data: this.getFormattedChartData(), // Get data with median\n      options: this.getChartOptions(),\n    };\n\n    this.chart = new Chart(canvas, config);\n  }\n\n  /**\n   * Computes median values and adds them to the dataset\n   */\n  private getFormattedChartData(): ChartData {\n    const medianValues = this.calculateMedian();\n\n    return {\n      labels: this.chartConfig.data.labels,\n      datasets: [\n        // Add median line dataset\n        {\n          label: this.medianTitle || 'Médiane',\n          data: medianValues,\n          borderColor: '#0000FF', // Blue color\n          borderWidth: 2,\n          backgroundColor: 'transparent',\n          pointRadius: 0,\n          fill: false,\n          tension: 0.1,\n          borderDash: [5, 5], // Dashed line\n        },\n        // Keep existing datasets\n        ...(this.chartConfig.data.datasets as Dataset[]),\n      ],\n    };\n  }\n\n  /**\n   * Computes median values from all datasets at each time point\n   */\n  private calculateMedian(): number[] {\n    const datasets = this.chartConfig.data.datasets as Dataset[];\n\n    return this.chartConfig.data.labels!.map((_, index) => {\n      const valuesAtTime = datasets\n        .map((dataset) => dataset.data[index])\n        .filter((val) => val !== undefined);\n\n      if (valuesAtTime.length === 0) return 0;\n\n      valuesAtTime.sort((a, b) => a - b);\n      const middle = Math.floor(valuesAtTime.length / 2);\n\n      return valuesAtTime.length % 2 === 0\n        ? (valuesAtTime[middle - 1] + valuesAtTime[middle]) / 2\n        : valuesAtTime[middle];\n    });\n  }\n\n  /**\n   * Defines chart options including Y-axis scaling\n   */\n  private getChartOptions(): ChartOptions {\n    return {\n      responsive: true,\n      maintainAspectRatio: false,\n      plugins: {\n        legend: { display: true, position: 'top' },\n        tooltip: { mode: 'index', intersect: false },\n      },\n      scales: {\n        x: {\n          title: { display: true, text: this.chartConfig.xAxisTitle || 'Time' },\n          ticks: { font: { size: 12 } },\n        },\n        y: {\n          title: {\n            display: true,\n            text: this.chartConfig.yAxisTitle || 'Value',\n          },\n          min: this.chartConfig.scales?.y?.min,\n          max: this.chartConfig.scales?.y?.max,\n          ticks: {\n            stepSize: this.chartConfig.scales?.y?.ticks?.stepSize,\n            font: { size: 16, weight: 'bold' },\n            color: '#333',\n          },\n          grid: {\n            color: 'rgba(0,0,0,0.1)',\n          },\n          border: {\n            display: true,\n            color: '#000',\n          },\n        },\n      },\n    };\n  }\n\n  /**\n   * Destroys the existing chart instance if present\n   */\n  private destroyChart() {\n    if (this.chart) {\n      this.chart.destroy();\n      this.chart = undefined as any;\n    }\n  }\n}\n","<div class=\"chart-scroll-container\">\n  <div\n    class=\"chart-inner\"\n    [style.width]=\"chartConfig.chartWidth || '1200px'\"\n    [style.height]=\"chartConfig.chartHeight || '400px'\"\n  >\n    <canvas #chartCanvas></canvas>\n  </div>\n</div>\n"]}
@@ -0,0 +1,18 @@
1
+ import { NgModule } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { PTChartComparisonComponent } from './pt-chart-comparison.component';
4
+ import * as i0 from "@angular/core";
5
+ export class PTChartComparisonModule {
6
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTChartComparisonModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule }); }
7
+ static { this.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "17.3.11", ngImport: i0, type: PTChartComparisonModule, declarations: [PTChartComparisonComponent], imports: [CommonModule], exports: [PTChartComparisonComponent] }); }
8
+ static { this.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTChartComparisonModule, imports: [CommonModule] }); }
9
+ }
10
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTChartComparisonModule, decorators: [{
11
+ type: NgModule,
12
+ args: [{
13
+ declarations: [PTChartComparisonComponent],
14
+ imports: [CommonModule],
15
+ exports: [PTChartComparisonComponent],
16
+ }]
17
+ }] });
18
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHQtY2hhcnQtY29tcGFyaXNvbi5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1wcmltZS10b29scy9zcmMvbGliL3B0LWNoYXJ0LWNvbXBhcmlzb24vcHQtY2hhcnQtY29tcGFyaXNvbi5tb2R1bGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN6QyxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLDBCQUEwQixFQUFFLE1BQU0saUNBQWlDLENBQUM7O0FBTzdFLE1BQU0sT0FBTyx1QkFBdUI7K0dBQXZCLHVCQUF1QjtnSEFBdkIsdUJBQXVCLGlCQUpuQiwwQkFBMEIsYUFDL0IsWUFBWSxhQUNaLDBCQUEwQjtnSEFFekIsdUJBQXVCLFlBSHhCLFlBQVk7OzRGQUdYLHVCQUF1QjtrQkFMbkMsUUFBUTttQkFBQztvQkFDUixZQUFZLEVBQUUsQ0FBQywwQkFBMEIsQ0FBQztvQkFDMUMsT0FBTyxFQUFFLENBQUMsWUFBWSxDQUFDO29CQUN2QixPQUFPLEVBQUUsQ0FBQywwQkFBMEIsQ0FBQztpQkFDdEMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IFBUQ2hhcnRDb21wYXJpc29uQ29tcG9uZW50IH0gZnJvbSAnLi9wdC1jaGFydC1jb21wYXJpc29uLmNvbXBvbmVudCc7XG5cbkBOZ01vZHVsZSh7XG4gIGRlY2xhcmF0aW9uczogW1BUQ2hhcnRDb21wYXJpc29uQ29tcG9uZW50XSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZV0sXG4gIGV4cG9ydHM6IFtQVENoYXJ0Q29tcGFyaXNvbkNvbXBvbmVudF0sXG59KVxuZXhwb3J0IGNsYXNzIFBUQ2hhcnRDb21wYXJpc29uTW9kdWxlIHt9XG4iXX0=
@@ -0,0 +1,3 @@
1
+ export * from './pt-chart-comparison.component';
2
+ export * from './pt-chart-comparison.module';
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25nLXByaW1lLXRvb2xzL3NyYy9saWIvcHQtY2hhcnQtY29tcGFyaXNvbi9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsaUNBQWlDLENBQUM7QUFDaEQsY0FBYyw4QkFBOEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vcHQtY2hhcnQtY29tcGFyaXNvbi5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9wdC1jaGFydC1jb21wYXJpc29uLm1vZHVsZSc7XG4iXX0=
@@ -17,6 +17,7 @@ export * from './lib/pt-metric-card';
17
17
  export * from './lib/pt-metric-card-group';
18
18
  // Chart
19
19
  export * from './lib/pt-chart';
20
+ export * from './lib/pt-chart-comparison';
20
21
  export * from './lib/pt-line-chart';
21
22
  // Cards
22
23
  export * from './lib/pt-card';
@@ -44,4 +45,4 @@ export * from './lib/models';
44
45
  export * from './lib/enums';
45
46
  export * from './lib/types';
46
47
  export * from './lib/pt-group';
47
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL25nLXByaW1lLXRvb2xzL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsNkJBQTZCLENBQUM7QUFFNUMsaUJBQWlCO0FBQ2pCLGNBQWMsK0JBQStCLENBQUM7QUFDOUMsY0FBYyw2QkFBNkIsQ0FBQztBQUU1QyxTQUFTO0FBQ1QsY0FBYywwQkFBMEIsQ0FBQztBQUN6QyxjQUFjLHFCQUFxQixDQUFDO0FBQ3BDLGNBQWMsbUJBQW1CLENBQUM7QUFDbEMsY0FBYyx1QkFBdUIsQ0FBQztBQUN0QyxjQUFjLHVCQUF1QixDQUFDO0FBQ3RDLGNBQWMsdUJBQXVCLENBQUM7QUFDdEMsY0FBYywwQkFBMEIsQ0FBQztBQUN6QyxjQUFjLHFCQUFxQixDQUFDO0FBRXBDLFlBQVk7QUFDWixjQUFjLHVCQUF1QixDQUFDO0FBQ3RDLGNBQWMsc0JBQXNCLENBQUM7QUFDckMsY0FBYyw0QkFBNEIsQ0FBQztBQUUzQyxRQUFRO0FBQ1IsY0FBYyxnQkFBZ0IsQ0FBQztBQUMvQixjQUFjLHFCQUFxQixDQUFDO0FBRXBDLFFBQVE7QUFDUixjQUFjLGVBQWUsQ0FBQztBQUU5QixPQUFPO0FBQ1AsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxxQkFBcUIsQ0FBQztBQUVwQyxTQUFTO0FBQ1QsY0FBYyx1QkFBdUIsQ0FBQztBQUV0QyxTQUFTO0FBQ1QsY0FBYyx3QkFBd0IsQ0FBQztBQUV2QyxnQkFBZ0I7QUFDaEIsY0FBYyx3QkFBd0IsQ0FBQztBQUV2QyxTQUFTO0FBQ1QsY0FBYyxpQkFBaUIsQ0FBQztBQUVoQyxjQUFjO0FBQ2QsY0FBYyxzQkFBc0IsQ0FBQztBQUVyQyxhQUFhO0FBQ2IsY0FBYyxxQkFBcUIsQ0FBQztBQUVwQyxTQUFTO0FBQ1QsY0FBYyxpQkFBaUIsQ0FBQztBQUVoQyxTQUFTO0FBQ1QsY0FBYyxpQkFBaUIsQ0FBQztBQUVoQyxVQUFVO0FBQ1YsY0FBYyxjQUFjLENBQUM7QUFDN0IsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxnQkFBZ0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vbGliL25nLXByaW1lLXRvb2xzLm1vZHVsZSc7XG5cbi8vIEFkdmFuY2VkIHRhYmxlXG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1hZHZhbmNlZC1wcmltZS10YWJsZSc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9tdWx0aS1zZWFyY2gtY3JpdGVyaWEnO1xuXG4vLyBJbnB1dHNcbmV4cG9ydCAqIGZyb20gJy4vbGliL3B0LWNoZWNrLWJveC1pbnB1dCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1kYXRlLWlucHV0JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3B0LWRyb3Bkb3duJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3B0LWZvcm0tYnVpbGRlcic7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1udW1iZXItaW5wdXQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtc3dpdGNoLWlucHV0JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3B0LXRleHQtYXJlYS1pbnB1dCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC10ZXh0LWlucHV0JztcblxuLy8gRGFzaGJvYXJkXG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1tZXRyaWMtcGFuZWwnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtbWV0cmljLWNhcmQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtbWV0cmljLWNhcmQtZ3JvdXAnO1xuXG4vLyBDaGFydFxuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtY2hhcnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtbGluZS1jaGFydCc7XG5cbi8vIENhcmRzXG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1jYXJkJztcblxuLy8gTWVudVxuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtbWVudSc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1tZW51LWZhbmN5JztcblxuLy8gTmF2YmFyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1uYXYtYmFyLW1lbnUnO1xuXG4vLyBOYXZiYXJcbmV4cG9ydCAqIGZyb20gJy4vbGliL3B0LXNpZGUtYmFyLW1lbnUnO1xuXG4vLyBQYWdlIHNrZWxldG9uXG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1wYWdlLXNrZWxldG9uJztcblxuLy8gZm9vdGVyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1mb290ZXInO1xuXG4vLyBicmVhZCBjcnVtYlxuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtYnJlYWQtY3J1bWInO1xuXG4vLyBsb2dpbiBwYWdlXG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1sb2dpbi1wYWdlJztcblxuLy8gYnV0dG9uXG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1idXR0b24nO1xuXG4vLyBkaWFsb2dcbmV4cG9ydCAqIGZyb20gJy4vbGliL3B0LWRpYWxvZyc7XG5cbi8vIEdlbmVyaWNcbmV4cG9ydCAqIGZyb20gJy4vbGliL21vZGVscyc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9lbnVtcyc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi90eXBlcyc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1ncm91cCc7XG4iXX0=
48
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL25nLXByaW1lLXRvb2xzL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsNkJBQTZCLENBQUM7QUFFNUMsaUJBQWlCO0FBQ2pCLGNBQWMsK0JBQStCLENBQUM7QUFDOUMsY0FBYyw2QkFBNkIsQ0FBQztBQUU1QyxTQUFTO0FBQ1QsY0FBYywwQkFBMEIsQ0FBQztBQUN6QyxjQUFjLHFCQUFxQixDQUFDO0FBQ3BDLGNBQWMsbUJBQW1CLENBQUM7QUFDbEMsY0FBYyx1QkFBdUIsQ0FBQztBQUN0QyxjQUFjLHVCQUF1QixDQUFDO0FBQ3RDLGNBQWMsdUJBQXVCLENBQUM7QUFDdEMsY0FBYywwQkFBMEIsQ0FBQztBQUN6QyxjQUFjLHFCQUFxQixDQUFDO0FBRXBDLFlBQVk7QUFDWixjQUFjLHVCQUF1QixDQUFDO0FBQ3RDLGNBQWMsc0JBQXNCLENBQUM7QUFDckMsY0FBYyw0QkFBNEIsQ0FBQztBQUUzQyxRQUFRO0FBQ1IsY0FBYyxnQkFBZ0IsQ0FBQztBQUMvQixjQUFjLDJCQUEyQixDQUFDO0FBQzFDLGNBQWMscUJBQXFCLENBQUM7QUFFcEMsUUFBUTtBQUNSLGNBQWMsZUFBZSxDQUFDO0FBRTlCLE9BQU87QUFDUCxjQUFjLGVBQWUsQ0FBQztBQUM5QixjQUFjLHFCQUFxQixDQUFDO0FBRXBDLFNBQVM7QUFDVCxjQUFjLHVCQUF1QixDQUFDO0FBRXRDLFNBQVM7QUFDVCxjQUFjLHdCQUF3QixDQUFDO0FBRXZDLGdCQUFnQjtBQUNoQixjQUFjLHdCQUF3QixDQUFDO0FBRXZDLFNBQVM7QUFDVCxjQUFjLGlCQUFpQixDQUFDO0FBRWhDLGNBQWM7QUFDZCxjQUFjLHNCQUFzQixDQUFDO0FBRXJDLGFBQWE7QUFDYixjQUFjLHFCQUFxQixDQUFDO0FBRXBDLFNBQVM7QUFDVCxjQUFjLGlCQUFpQixDQUFDO0FBRWhDLFNBQVM7QUFDVCxjQUFjLGlCQUFpQixDQUFDO0FBRWhDLFVBQVU7QUFDVixjQUFjLGNBQWMsQ0FBQztBQUM3QixjQUFjLGFBQWEsQ0FBQztBQUM1QixjQUFjLGFBQWEsQ0FBQztBQUM1QixjQUFjLGdCQUFnQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9saWIvbmctcHJpbWUtdG9vbHMubW9kdWxlJztcblxuLy8gQWR2YW5jZWQgdGFibGVcbmV4cG9ydCAqIGZyb20gJy4vbGliL3B0LWFkdmFuY2VkLXByaW1lLXRhYmxlJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL211bHRpLXNlYXJjaC1jcml0ZXJpYSc7XG5cbi8vIElucHV0c1xuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtY2hlY2stYm94LWlucHV0JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3B0LWRhdGUtaW5wdXQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtZHJvcGRvd24nO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtZm9ybS1idWlsZGVyJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3B0LW51bWJlci1pbnB1dCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1zd2l0Y2gtaW5wdXQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtdGV4dC1hcmVhLWlucHV0JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3B0LXRleHQtaW5wdXQnO1xuXG4vLyBEYXNoYm9hcmRcbmV4cG9ydCAqIGZyb20gJy4vbGliL3B0LW1ldHJpYy1wYW5lbCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1tZXRyaWMtY2FyZCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1tZXRyaWMtY2FyZC1ncm91cCc7XG5cbi8vIENoYXJ0XG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1jaGFydCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1jaGFydC1jb21wYXJpc29uJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3B0LWxpbmUtY2hhcnQnO1xuXG4vLyBDYXJkc1xuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtY2FyZCc7XG5cbi8vIE1lbnVcbmV4cG9ydCAqIGZyb20gJy4vbGliL3B0LW1lbnUnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtbWVudS1mYW5jeSc7XG5cbi8vIE5hdmJhclxuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtbmF2LWJhci1tZW51JztcblxuLy8gTmF2YmFyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1zaWRlLWJhci1tZW51JztcblxuLy8gUGFnZSBza2VsZXRvblxuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtcGFnZS1za2VsZXRvbic7XG5cbi8vIGZvb3RlclxuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtZm9vdGVyJztcblxuLy8gYnJlYWQgY3J1bWJcbmV4cG9ydCAqIGZyb20gJy4vbGliL3B0LWJyZWFkLWNydW1iJztcblxuLy8gbG9naW4gcGFnZVxuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtbG9naW4tcGFnZSc7XG5cbi8vIGJ1dHRvblxuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtYnV0dG9uJztcblxuLy8gZGlhbG9nXG5leHBvcnQgKiBmcm9tICcuL2xpYi9wdC1kaWFsb2cnO1xuXG4vLyBHZW5lcmljXG5leHBvcnQgKiBmcm9tICcuL2xpYi9tb2RlbHMnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvZW51bXMnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvdHlwZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvcHQtZ3JvdXAnO1xuIl19
@@ -3948,6 +3948,169 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImpo
3948
3948
  }]
3949
3949
  }] });
3950
3950
 
3951
+ class PTChartComparisonComponent {
3952
+ constructor() {
3953
+ this.medianTitle = 'Médiane';
3954
+ this.xAxisTitle = 'Time';
3955
+ this.yAxisTitle = 'Value';
3956
+ // Chart dimension inputs
3957
+ this.chartHeight = '400px';
3958
+ this.chartWidth = '1200px';
3959
+ Chart.register(...registerables, ChartDataLabels);
3960
+ }
3961
+ ngOnInit() {
3962
+ this.initializeChart();
3963
+ }
3964
+ ngOnDestroy() {
3965
+ this.destroyChart();
3966
+ }
3967
+ /**
3968
+ * Initializes the chart with median and datasets
3969
+ */
3970
+ initializeChart() {
3971
+ const canvas = this.canvasRef.nativeElement;
3972
+ this.destroyChart();
3973
+ const config = {
3974
+ type: this.chartConfig.type || 'line',
3975
+ data: this.getFormattedChartData(), // Get data with median
3976
+ options: this.getChartOptions(),
3977
+ };
3978
+ this.chart = new Chart(canvas, config);
3979
+ }
3980
+ /**
3981
+ * Computes median values and adds them to the dataset
3982
+ */
3983
+ getFormattedChartData() {
3984
+ const medianValues = this.calculateMedian();
3985
+ return {
3986
+ labels: this.chartConfig.data.labels,
3987
+ datasets: [
3988
+ // Add median line dataset
3989
+ {
3990
+ label: this.medianTitle || 'Médiane',
3991
+ data: medianValues,
3992
+ borderColor: '#0000FF', // Blue color
3993
+ borderWidth: 2,
3994
+ backgroundColor: 'transparent',
3995
+ pointRadius: 0,
3996
+ fill: false,
3997
+ tension: 0.1,
3998
+ borderDash: [5, 5], // Dashed line
3999
+ },
4000
+ // Keep existing datasets
4001
+ ...this.chartConfig.data.datasets,
4002
+ ],
4003
+ };
4004
+ }
4005
+ /**
4006
+ * Computes median values from all datasets at each time point
4007
+ */
4008
+ calculateMedian() {
4009
+ const datasets = this.chartConfig.data.datasets;
4010
+ return this.chartConfig.data.labels.map((_, index) => {
4011
+ const valuesAtTime = datasets
4012
+ .map((dataset) => dataset.data[index])
4013
+ .filter((val) => val !== undefined);
4014
+ if (valuesAtTime.length === 0)
4015
+ return 0;
4016
+ valuesAtTime.sort((a, b) => a - b);
4017
+ const middle = Math.floor(valuesAtTime.length / 2);
4018
+ return valuesAtTime.length % 2 === 0
4019
+ ? (valuesAtTime[middle - 1] + valuesAtTime[middle]) / 2
4020
+ : valuesAtTime[middle];
4021
+ });
4022
+ }
4023
+ /**
4024
+ * Defines chart options including Y-axis scaling
4025
+ */
4026
+ getChartOptions() {
4027
+ return {
4028
+ responsive: true,
4029
+ maintainAspectRatio: false,
4030
+ plugins: {
4031
+ legend: { display: true, position: 'top' },
4032
+ tooltip: { mode: 'index', intersect: false },
4033
+ },
4034
+ scales: {
4035
+ x: {
4036
+ title: { display: true, text: this.chartConfig.xAxisTitle || 'Time' },
4037
+ ticks: { font: { size: 12 } },
4038
+ },
4039
+ y: {
4040
+ title: {
4041
+ display: true,
4042
+ text: this.chartConfig.yAxisTitle || 'Value',
4043
+ },
4044
+ min: this.chartConfig.scales?.y?.min,
4045
+ max: this.chartConfig.scales?.y?.max,
4046
+ ticks: {
4047
+ stepSize: this.chartConfig.scales?.y?.ticks?.stepSize,
4048
+ font: { size: 16, weight: 'bold' },
4049
+ color: '#333',
4050
+ },
4051
+ grid: {
4052
+ color: 'rgba(0,0,0,0.1)',
4053
+ },
4054
+ border: {
4055
+ display: true,
4056
+ color: '#000',
4057
+ },
4058
+ },
4059
+ },
4060
+ };
4061
+ }
4062
+ /**
4063
+ * Destroys the existing chart instance if present
4064
+ */
4065
+ destroyChart() {
4066
+ if (this.chart) {
4067
+ this.chart.destroy();
4068
+ this.chart = undefined;
4069
+ }
4070
+ }
4071
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTChartComparisonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
4072
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.11", type: PTChartComparisonComponent, selector: "pt-chart-comparison", inputs: { chartConfig: "chartConfig", medianTitle: "medianTitle", xAxisTitle: "xAxisTitle", yAxisTitle: "yAxisTitle", yMin: "yMin", yMax: "yMax", yStepSize: "yStepSize", chartHeight: "chartHeight", chartWidth: "chartWidth" }, viewQueries: [{ propertyName: "canvasRef", first: true, predicate: ["chartCanvas"], descendants: true, static: true }], ngImport: i0, template: "<div class=\"chart-scroll-container\">\n <div\n class=\"chart-inner\"\n [style.width]=\"chartConfig.chartWidth || '1200px'\"\n [style.height]=\"chartConfig.chartHeight || '400px'\"\n >\n <canvas #chartCanvas></canvas>\n </div>\n</div>\n", styles: [".chart-scroll-container{width:100%;overflow-x:auto;white-space:nowrap;padding-bottom:10px}.chart-inner{display:inline-block}\n"] }); }
4073
+ }
4074
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTChartComparisonComponent, decorators: [{
4075
+ type: Component,
4076
+ args: [{ selector: 'pt-chart-comparison', template: "<div class=\"chart-scroll-container\">\n <div\n class=\"chart-inner\"\n [style.width]=\"chartConfig.chartWidth || '1200px'\"\n [style.height]=\"chartConfig.chartHeight || '400px'\"\n >\n <canvas #chartCanvas></canvas>\n </div>\n</div>\n", styles: [".chart-scroll-container{width:100%;overflow-x:auto;white-space:nowrap;padding-bottom:10px}.chart-inner{display:inline-block}\n"] }]
4077
+ }], ctorParameters: () => [], propDecorators: { chartConfig: [{
4078
+ type: Input
4079
+ }], medianTitle: [{
4080
+ type: Input
4081
+ }], xAxisTitle: [{
4082
+ type: Input
4083
+ }], yAxisTitle: [{
4084
+ type: Input
4085
+ }], yMin: [{
4086
+ type: Input
4087
+ }], yMax: [{
4088
+ type: Input
4089
+ }], yStepSize: [{
4090
+ type: Input
4091
+ }], chartHeight: [{
4092
+ type: Input
4093
+ }], chartWidth: [{
4094
+ type: Input
4095
+ }], canvasRef: [{
4096
+ type: ViewChild,
4097
+ args: ['chartCanvas', { static: true }]
4098
+ }] } });
4099
+
4100
+ class PTChartComparisonModule {
4101
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTChartComparisonModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule }); }
4102
+ static { this.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "17.3.11", ngImport: i0, type: PTChartComparisonModule, declarations: [PTChartComparisonComponent], imports: [CommonModule], exports: [PTChartComparisonComponent] }); }
4103
+ static { this.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTChartComparisonModule, imports: [CommonModule] }); }
4104
+ }
4105
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTChartComparisonModule, decorators: [{
4106
+ type: NgModule,
4107
+ args: [{
4108
+ declarations: [PTChartComparisonComponent],
4109
+ imports: [CommonModule],
4110
+ exports: [PTChartComparisonComponent],
4111
+ }]
4112
+ }] });
4113
+
3951
4114
  class PTLineChartComponent {
3952
4115
  constructor() {
3953
4116
  this.config = {
@@ -4119,5 +4282,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImpo
4119
4282
  * Generated bundle index. Do not edit.
4120
4283
  */
4121
4284
 
4122
- export { BadgeType, BadgeTypeStyles, ButtonColorEnum, FormInputTypeEnum, InputValidationEnum, MultiSearchCriteriaComponent, MultiSearchCriteriaModule, NgPrimeToolsModule, PTAdvancedPrimeTableComponent, PTAdvancedPrimeTableModule, PTBreadCrumbComponent, PTBreadCrumbModule, PTButtonComponent, PTButtonModule, PTCardComponent, PTCardModule, PTChartComponent, PTChartModule, PTCheckBoxInputComponent, PTCheckBoxInputModule, PTDateInputComponent, PTDateInputModule, PTDialogComponent, PTDialogModule, PTDropdownComponent, PTDropdownModule, PTFooterComponent, PTFooterModule, PTFormBuilderComponent, PTFormBuilderModule, PTGroupComponent, PTGroupModule, PTLineChartComponent, PTLineChartModule, PTLoginPageComponent, PTLoginPageModule, PTMenuComponent, PTMenuFancyComponent, PTMenuFancyModule, PTMenuModule, PTMetricCardComponent, PTMetricCardGroupComponent, PTMetricCardGroupModule, PTMetricCardModule, PTMetricPanelComponent, PTMetricPanelModule, PTNavbarMenuComponent, PTNavbarMenuModule, PTNumberInputComponent, PTNumberInputModule, PTPageSkeletonComponent, PTPageSkeletonModule, PTSideBarMenuComponent, PTSideBarMenuModule, PTSwitchInputComponent, PTSwitchInputModule, PTTextAreaInputComponent, PTTextAreaInputModule, PTTextInputComponent, PTTextInputModule, SearchCriteriaTypeEnum, TableTypeEnum };
4285
+ export { BadgeType, BadgeTypeStyles, ButtonColorEnum, FormInputTypeEnum, InputValidationEnum, MultiSearchCriteriaComponent, MultiSearchCriteriaModule, NgPrimeToolsModule, PTAdvancedPrimeTableComponent, PTAdvancedPrimeTableModule, PTBreadCrumbComponent, PTBreadCrumbModule, PTButtonComponent, PTButtonModule, PTCardComponent, PTCardModule, PTChartComparisonComponent, PTChartComparisonModule, PTChartComponent, PTChartModule, PTCheckBoxInputComponent, PTCheckBoxInputModule, PTDateInputComponent, PTDateInputModule, PTDialogComponent, PTDialogModule, PTDropdownComponent, PTDropdownModule, PTFooterComponent, PTFooterModule, PTFormBuilderComponent, PTFormBuilderModule, PTGroupComponent, PTGroupModule, PTLineChartComponent, PTLineChartModule, PTLoginPageComponent, PTLoginPageModule, PTMenuComponent, PTMenuFancyComponent, PTMenuFancyModule, PTMenuModule, PTMetricCardComponent, PTMetricCardGroupComponent, PTMetricCardGroupModule, PTMetricCardModule, PTMetricPanelComponent, PTMetricPanelModule, PTNavbarMenuComponent, PTNavbarMenuModule, PTNumberInputComponent, PTNumberInputModule, PTPageSkeletonComponent, PTPageSkeletonModule, PTSideBarMenuComponent, PTSideBarMenuModule, PTSwitchInputComponent, PTSwitchInputModule, PTTextAreaInputComponent, PTTextAreaInputModule, PTTextInputComponent, PTTextInputModule, SearchCriteriaTypeEnum, TableTypeEnum };
4123
4286
  //# sourceMappingURL=ng-prime-tools.mjs.map