ng-prime-tools 1.0.64 → 1.0.65
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/esm2022/lib/pt-chart-comparison/pt-chart-comparison.component.mjs +15 -25
- package/fesm2022/ng-prime-tools.mjs +14 -24
- package/fesm2022/ng-prime-tools.mjs.map +1 -1
- package/lib/pt-chart-comparison/pt-chart-comparison.component.d.ts +0 -15
- package/lib/pt-chart-comparison/pt-chart-comparison.component.d.ts.map +1 -1
- package/lib/pt-metric-panel/pt-metric-panel.component.d.ts +1 -1
- package/package.json +1 -1
|
@@ -7,7 +7,6 @@ export class PTChartComparisonComponent {
|
|
|
7
7
|
this.medianTitle = 'Médiane';
|
|
8
8
|
this.xAxisTitle = 'Time';
|
|
9
9
|
this.yAxisTitle = 'Value';
|
|
10
|
-
// Chart dimension inputs
|
|
11
10
|
this.chartHeight = '400px';
|
|
12
11
|
this.chartWidth = '1200px';
|
|
13
12
|
Chart.register(...registerables, ChartDataLabels);
|
|
@@ -18,53 +17,42 @@ export class PTChartComparisonComponent {
|
|
|
18
17
|
ngOnDestroy() {
|
|
19
18
|
this.destroyChart();
|
|
20
19
|
}
|
|
21
|
-
/**
|
|
22
|
-
* Initializes the chart with median and datasets
|
|
23
|
-
*/
|
|
24
20
|
initializeChart() {
|
|
25
21
|
const canvas = this.canvasRef.nativeElement;
|
|
26
22
|
this.destroyChart();
|
|
27
23
|
const config = {
|
|
28
24
|
type: this.chartConfig.type || 'line',
|
|
29
|
-
data: this.getFormattedChartData(),
|
|
25
|
+
data: this.getFormattedChartData(),
|
|
30
26
|
options: this.getChartOptions(),
|
|
31
27
|
};
|
|
32
28
|
this.chart = new Chart(canvas, config);
|
|
33
29
|
}
|
|
34
|
-
/**
|
|
35
|
-
* Computes median values and adds them to the dataset
|
|
36
|
-
*/
|
|
37
30
|
getFormattedChartData() {
|
|
38
31
|
const medianValues = this.calculateMedian();
|
|
39
32
|
return {
|
|
40
33
|
labels: this.chartConfig.data.labels,
|
|
41
34
|
datasets: [
|
|
42
|
-
// Add median line dataset
|
|
43
35
|
{
|
|
44
|
-
label: this.medianTitle || 'Médiane',
|
|
36
|
+
label: this.chartConfig.medianTitle || this.medianTitle || 'Médiane',
|
|
45
37
|
data: medianValues,
|
|
46
|
-
borderColor: '#0000FF',
|
|
47
|
-
borderWidth:
|
|
38
|
+
borderColor: '#0000FF',
|
|
39
|
+
borderWidth: 3,
|
|
48
40
|
backgroundColor: 'transparent',
|
|
49
41
|
pointRadius: 0,
|
|
50
42
|
fill: false,
|
|
51
43
|
tension: 0.1,
|
|
52
|
-
borderDash: [
|
|
44
|
+
borderDash: [], // ✅ médiane continue
|
|
53
45
|
},
|
|
54
|
-
// Keep existing datasets
|
|
55
46
|
...this.chartConfig.data.datasets,
|
|
56
47
|
],
|
|
57
48
|
};
|
|
58
49
|
}
|
|
59
|
-
/**
|
|
60
|
-
* Computes median values from all datasets at each time point
|
|
61
|
-
*/
|
|
62
50
|
calculateMedian() {
|
|
63
51
|
const datasets = this.chartConfig.data.datasets;
|
|
64
52
|
return this.chartConfig.data.labels.map((_, index) => {
|
|
65
53
|
const valuesAtTime = datasets
|
|
66
54
|
.map((dataset) => dataset.data[index])
|
|
67
|
-
.filter((val) => val
|
|
55
|
+
.filter((val) => typeof val === 'number' && !Number.isNaN(val));
|
|
68
56
|
if (valuesAtTime.length === 0)
|
|
69
57
|
return 0;
|
|
70
58
|
valuesAtTime.sort((a, b) => a - b);
|
|
@@ -74,21 +62,24 @@ export class PTChartComparisonComponent {
|
|
|
74
62
|
: valuesAtTime[middle];
|
|
75
63
|
});
|
|
76
64
|
}
|
|
77
|
-
/**
|
|
78
|
-
* Defines chart options including Y-axis scaling
|
|
79
|
-
*/
|
|
80
65
|
getChartOptions() {
|
|
66
|
+
const externalOptions = this.chartConfig.options ?? {};
|
|
81
67
|
return {
|
|
82
68
|
responsive: true,
|
|
83
69
|
maintainAspectRatio: false,
|
|
84
70
|
plugins: {
|
|
85
71
|
legend: { display: true, position: 'top' },
|
|
86
72
|
tooltip: { mode: 'index', intersect: false },
|
|
73
|
+
datalabels: {
|
|
74
|
+
display: false, // ✅ supprime les nombres au-dessus des lignes
|
|
75
|
+
},
|
|
76
|
+
...(externalOptions.plugins ?? {}),
|
|
87
77
|
},
|
|
88
78
|
scales: {
|
|
89
79
|
x: {
|
|
90
80
|
title: { display: true, text: this.chartConfig.xAxisTitle || 'Time' },
|
|
91
81
|
ticks: { font: { size: 12 } },
|
|
82
|
+
...externalOptions.scales?.x,
|
|
92
83
|
},
|
|
93
84
|
y: {
|
|
94
85
|
title: {
|
|
@@ -109,13 +100,12 @@ export class PTChartComparisonComponent {
|
|
|
109
100
|
display: true,
|
|
110
101
|
color: '#000',
|
|
111
102
|
},
|
|
103
|
+
...externalOptions.scales?.y,
|
|
112
104
|
},
|
|
113
105
|
},
|
|
106
|
+
...externalOptions,
|
|
114
107
|
};
|
|
115
108
|
}
|
|
116
|
-
/**
|
|
117
|
-
* Destroys the existing chart instance if present
|
|
118
|
-
*/
|
|
119
109
|
destroyChart() {
|
|
120
110
|
if (this.chart) {
|
|
121
111
|
this.chart.destroy();
|
|
@@ -150,4 +140,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImpo
|
|
|
150
140
|
type: ViewChild,
|
|
151
141
|
args: ['chartCanvas', { static: true }]
|
|
152
142
|
}] } });
|
|
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"]}
|
|
143
|
+
//# 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  @Input() yMin!: number;\n  @Input() yMax!: number;\n  @Input() yStepSize!: number;\n\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  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(),\n      options: this.getChartOptions(),\n    };\n\n    this.chart = new Chart(canvas, config);\n  }\n\n  private getFormattedChartData(): ChartData {\n    const medianValues = this.calculateMedian();\n\n    return {\n      labels: this.chartConfig.data.labels,\n      datasets: [\n        {\n          label: this.chartConfig.medianTitle || this.medianTitle || 'Médiane',\n          data: medianValues,\n          borderColor: '#0000FF',\n          borderWidth: 3,\n          backgroundColor: 'transparent',\n          pointRadius: 0,\n          fill: false,\n          tension: 0.1,\n          borderDash: [], // ✅ médiane continue\n        },\n        ...(this.chartConfig.data.datasets as Dataset[]),\n      ],\n    };\n  }\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(\n          (val): val is number => typeof val === 'number' && !Number.isNaN(val),\n        );\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  private getChartOptions(): ChartOptions {\n    const externalOptions = this.chartConfig.options ?? {};\n\n    return {\n      responsive: true,\n      maintainAspectRatio: false,\n      plugins: {\n        legend: { display: true, position: 'top' },\n        tooltip: { mode: 'index', intersect: false },\n        datalabels: {\n          display: false, // ✅ supprime les nombres au-dessus des lignes\n        },\n        ...(externalOptions.plugins ?? {}),\n      },\n      scales: {\n        x: {\n          title: { display: true, text: this.chartConfig.xAxisTitle || 'Time' },\n          ticks: { font: { size: 12 } },\n          ...(externalOptions.scales as any)?.x,\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          ...(externalOptions.scales as any)?.y,\n        },\n      },\n      ...externalOptions,\n    };\n  }\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"]}
|
|
@@ -5482,7 +5482,6 @@ class PTChartComparisonComponent {
|
|
|
5482
5482
|
this.medianTitle = 'Médiane';
|
|
5483
5483
|
this.xAxisTitle = 'Time';
|
|
5484
5484
|
this.yAxisTitle = 'Value';
|
|
5485
|
-
// Chart dimension inputs
|
|
5486
5485
|
this.chartHeight = '400px';
|
|
5487
5486
|
this.chartWidth = '1200px';
|
|
5488
5487
|
Chart.register(...registerables, ChartDataLabels);
|
|
@@ -5493,53 +5492,42 @@ class PTChartComparisonComponent {
|
|
|
5493
5492
|
ngOnDestroy() {
|
|
5494
5493
|
this.destroyChart();
|
|
5495
5494
|
}
|
|
5496
|
-
/**
|
|
5497
|
-
* Initializes the chart with median and datasets
|
|
5498
|
-
*/
|
|
5499
5495
|
initializeChart() {
|
|
5500
5496
|
const canvas = this.canvasRef.nativeElement;
|
|
5501
5497
|
this.destroyChart();
|
|
5502
5498
|
const config = {
|
|
5503
5499
|
type: this.chartConfig.type || 'line',
|
|
5504
|
-
data: this.getFormattedChartData(),
|
|
5500
|
+
data: this.getFormattedChartData(),
|
|
5505
5501
|
options: this.getChartOptions(),
|
|
5506
5502
|
};
|
|
5507
5503
|
this.chart = new Chart(canvas, config);
|
|
5508
5504
|
}
|
|
5509
|
-
/**
|
|
5510
|
-
* Computes median values and adds them to the dataset
|
|
5511
|
-
*/
|
|
5512
5505
|
getFormattedChartData() {
|
|
5513
5506
|
const medianValues = this.calculateMedian();
|
|
5514
5507
|
return {
|
|
5515
5508
|
labels: this.chartConfig.data.labels,
|
|
5516
5509
|
datasets: [
|
|
5517
|
-
// Add median line dataset
|
|
5518
5510
|
{
|
|
5519
|
-
label: this.medianTitle || 'Médiane',
|
|
5511
|
+
label: this.chartConfig.medianTitle || this.medianTitle || 'Médiane',
|
|
5520
5512
|
data: medianValues,
|
|
5521
|
-
borderColor: '#0000FF',
|
|
5522
|
-
borderWidth:
|
|
5513
|
+
borderColor: '#0000FF',
|
|
5514
|
+
borderWidth: 3,
|
|
5523
5515
|
backgroundColor: 'transparent',
|
|
5524
5516
|
pointRadius: 0,
|
|
5525
5517
|
fill: false,
|
|
5526
5518
|
tension: 0.1,
|
|
5527
|
-
borderDash: [
|
|
5519
|
+
borderDash: [], // ✅ médiane continue
|
|
5528
5520
|
},
|
|
5529
|
-
// Keep existing datasets
|
|
5530
5521
|
...this.chartConfig.data.datasets,
|
|
5531
5522
|
],
|
|
5532
5523
|
};
|
|
5533
5524
|
}
|
|
5534
|
-
/**
|
|
5535
|
-
* Computes median values from all datasets at each time point
|
|
5536
|
-
*/
|
|
5537
5525
|
calculateMedian() {
|
|
5538
5526
|
const datasets = this.chartConfig.data.datasets;
|
|
5539
5527
|
return this.chartConfig.data.labels.map((_, index) => {
|
|
5540
5528
|
const valuesAtTime = datasets
|
|
5541
5529
|
.map((dataset) => dataset.data[index])
|
|
5542
|
-
.filter((val) => val
|
|
5530
|
+
.filter((val) => typeof val === 'number' && !Number.isNaN(val));
|
|
5543
5531
|
if (valuesAtTime.length === 0)
|
|
5544
5532
|
return 0;
|
|
5545
5533
|
valuesAtTime.sort((a, b) => a - b);
|
|
@@ -5549,21 +5537,24 @@ class PTChartComparisonComponent {
|
|
|
5549
5537
|
: valuesAtTime[middle];
|
|
5550
5538
|
});
|
|
5551
5539
|
}
|
|
5552
|
-
/**
|
|
5553
|
-
* Defines chart options including Y-axis scaling
|
|
5554
|
-
*/
|
|
5555
5540
|
getChartOptions() {
|
|
5541
|
+
const externalOptions = this.chartConfig.options ?? {};
|
|
5556
5542
|
return {
|
|
5557
5543
|
responsive: true,
|
|
5558
5544
|
maintainAspectRatio: false,
|
|
5559
5545
|
plugins: {
|
|
5560
5546
|
legend: { display: true, position: 'top' },
|
|
5561
5547
|
tooltip: { mode: 'index', intersect: false },
|
|
5548
|
+
datalabels: {
|
|
5549
|
+
display: false, // ✅ supprime les nombres au-dessus des lignes
|
|
5550
|
+
},
|
|
5551
|
+
...(externalOptions.plugins ?? {}),
|
|
5562
5552
|
},
|
|
5563
5553
|
scales: {
|
|
5564
5554
|
x: {
|
|
5565
5555
|
title: { display: true, text: this.chartConfig.xAxisTitle || 'Time' },
|
|
5566
5556
|
ticks: { font: { size: 12 } },
|
|
5557
|
+
...externalOptions.scales?.x,
|
|
5567
5558
|
},
|
|
5568
5559
|
y: {
|
|
5569
5560
|
title: {
|
|
@@ -5584,13 +5575,12 @@ class PTChartComparisonComponent {
|
|
|
5584
5575
|
display: true,
|
|
5585
5576
|
color: '#000',
|
|
5586
5577
|
},
|
|
5578
|
+
...externalOptions.scales?.y,
|
|
5587
5579
|
},
|
|
5588
5580
|
},
|
|
5581
|
+
...externalOptions,
|
|
5589
5582
|
};
|
|
5590
5583
|
}
|
|
5591
|
-
/**
|
|
5592
|
-
* Destroys the existing chart instance if present
|
|
5593
|
-
*/
|
|
5594
5584
|
destroyChart() {
|
|
5595
5585
|
if (this.chart) {
|
|
5596
5586
|
this.chart.destroy();
|