UNPKG

ng-chartjs

Version:
317 lines 41 kB
import { EventEmitter, Input, Output, Directive } from '@angular/core'; // import { Chart, ChartConfiguration, ChartEvent, DefaultDataPoint, registerables } from 'chart.js'; import Chart from 'chart.js/auto'; import { deepCopyJson, mergeJson } from './ng-chartjs.service'; import { getColors } from './colors'; import * as i0 from "@angular/core"; import * as i1 from "./ng-chartjs.service"; import * as i2 from "./store.service"; /* tslint:disable-next-line */ export class NgChartjsDirective { constructor(element, ngChartjsService, storeService, zone) { this.ngChartjsService = ngChartjsService; this.storeService = storeService; this.zone = zone; // x轴标签。这对图表来说是必要的:线,条和雷达。并且只是图表的标签(悬停):polarArea,pie和doughnut this.labels = []; this.noZone = true; // disable angular NgZone // @ts-ignore this.id = null; // chart instance id // 鼠标点击图表所有的区域 this.chartClick = new EventEmitter(); // 鼠标悬浮在标签或者活跃的点上面时 this.chartHover = new EventEmitter(); this.initFlag = false; this.hasChanges = false; this.element = element; // 获取指令所在canvas元素 } ngOnInit() { this.ctx = this.element.nativeElement.getContext('2d'); // 获取元素的ctx this.initFlag = true; // 是否初始化了的标志 if (this.data || this.datasets) { // 判断data和datasets有一个有数据就刷新 if (this.noZone) { this.zone.runOutsideAngular(() => { this.refresh(); }); } else { this.refresh(); } } } ngOnChanges(changes) { // TODO: 插件变化刷新,开放刷新按钮 if (this.initFlag) { // Check if the changes are in the data or datasets if (changes.hasOwnProperty('data') || changes.hasOwnProperty('datasets')) { if (changes.data) { this.updateChartData(changes.data.currentValue); } else { this.updateChartData(changes.datasets.currentValue); } this.hasChanges = true; } if (changes.hasOwnProperty('labels')) { this.chart.data.labels = changes.labels.currentValue; this.hasChanges = true; } if (changes.hasOwnProperty('legend')) { if (changes.legend.currentValue !== changes.legend.previousValue) { // @ts-ignore this.chart.options.plugins.legend.display = changes.legend.currentValue; this.hasChanges = true; } } if (changes.hasOwnProperty('adding')) { this.addData_(changes.adding.currentValue.labels, changes.adding.currentValue.data); this.hasChanges = true; } if (changes.hasOwnProperty('removing')) { if (changes.removing.currentValue.orientation === 'oldest' || changes.removing.currentValue.orientation === 'latest') { this.removeData_(changes.removing.currentValue.orientation); this.hasChanges = true; } } if (changes.hasOwnProperty('chartType')) { this.refresh(); } if (changes.hasOwnProperty('resetOption')) { const resetOption = deepCopyJson(changes.resetOption.currentValue); this.chart.options = mergeJson(resetOption, this.chart.options); this.hasChanges = true; } if (this.hasChanges) { this.chart.update(); this.hasChanges = false; } // change chart id if (changes.hasOwnProperty('id')) { this.removeChart(changes.id.previousValue); this.addChart(changes.id.currentValue); } } } ngOnDestroy() { if (this.chart) { this.chart.destroy(); // @ts-ignore this.chart = void 0; this.removeChart(this.id); } } // update chartjs update() { this.chart.update(); } // Dynamic add data addData(labels, data) { this.addData_(labels, data); this.update(); } // Dynamic remove data, orientation is 'ildest' or 'latest' removeData(orientation) { this.removeData_(orientation); this.update(); } refresh() { this.ngOnDestroy(); this.chart = this.getChartBuilder(this.ctx /*, data, this.options*/); this.addChart(this.id); } removeChart(id) { if (this.element.nativeElement.hasAttribute('id')) { this.storeService.removeChart(this.element.nativeElement.id); return; } if (id !== null && id !== undefined) { this.storeService.removeChart(id); // delete chart instance. } } addChart(id) { if (this.element.nativeElement.hasAttribute('id')) { this.storeService.addChart(this.element.nativeElement.id, this.chart); return; } if (id !== null && id !== undefined) { this.storeService.addChart(id, this.chart); } } updateChartData(newDataValues) { if (Array.isArray(newDataValues[0].data)) { // @ts-ignore // this.chart.data.datasets.forEach((dataset: ChartDataset, i: number) => { // dataset.data = newDataValues[i].data; // if (newDataValues[i].label) { // dataset.label = newDataValues[i].label; // } // }); // @ts-ignore this.chart.data.datasets = newDataValues; } else { // @ts-ignore this.chart.data.datasets[0].data = newDataValues; } // update colors // @ts-ignore this.chart.data.datasets = this.updateColors(this.chart.data.datasets); } getChartBuilder(ctx /*, data:Array<any>, options:any*/) { const datasets = this.getDatasets(); let options = this.options || {}; options = Object.assign({}, this.options); // 深复制options mergeJson(options, { plugins: { legend: { display: this.legend } } }); // hock for onHover and onClick events options.hover = options.hover || {}; if (!options.onHover) { options.onHover = (event, active) => { if (active && !active.length) { return; } this.chartHover.emit({ event, active }); }; } if (!options.onClick) { options.onClick = (event, active) => { this.chartClick.emit({ event, active }); }; } const inlinePlugins = this.inlinePlugins || []; const opts = { type: this.chartType, data: { labels: this.labels, datasets: datasets // TODO: 后续更改这个属性名字,否则警告 }, options: options, plugins: inlinePlugins }; return new Chart(ctx, opts); } // 获取 chart.js的datasets数据 getDatasets() { // @ts-ignore let datasets = void 0; // in case if datasets is not provided, but data is present if (!this.datasets || !this.datasets.length && (this.data && this.data.length)) { if (Array.isArray(this.data[0])) { datasets = this.data.map((data, index) => { return { data, label: this.labels[index] || `Label ${index}` }; }); } else { datasets = [{ data: this.data, label: `Label 0` }]; } } datasets = this.updateColors(datasets); // update colors if (!datasets) { throw new Error(`ng-chartjs configuration error, data or datasets field are required to render char ${this.chartType}`); } return datasets; } // update dataset colors updateColors(datasets) { if (this.datasets && this.datasets.length || (datasets && datasets.length)) { // fix elm type, pre type is number datasets = (this.datasets || datasets).map((elm, index) => { const newElm = Object.assign({}, elm); if (this.colors && this.colors.length) { Object.assign(newElm, this.colors[index]); } else { // @ts-ignore Object.assign(newElm, getColors(this.chartType, index, newElm.data.length)); } return newElm; }); } return datasets; } addData_(labels, data) { if (labels.length === 0 || data.length === 0) { return; } // update labels // @ts-ignore labels.forEach((label) => { this.chart.data.labels.push(label); }); // @ts-ignore this.chart.data.datasets.forEach((dataset, index) => { if (data[index]) { for (let i = 0; i < data[index].length; i++) { // @ts-ignore dataset.data.push(data[index][i]); } } else { console.log('The added data does not match the original data'); return; } }); } removeData_(orientation) { // fix: support to oldest feature if (orientation === 'latest') { // @ts-ignore this.chart.data.labels.pop(); // @ts-ignore this.chart.data.datasets.forEach((dataset) => { // @ts-ignore dataset.data.pop(); }); } else if (orientation === 'oldest') { // @ts-ignore this.chart.data.labels.shift(); // @ts-ignore this.chart.data.datasets.forEach((dataset) => { // @ts-ignore dataset.data.shift(); }); } } } NgChartjsDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: NgChartjsDirective, deps: [{ token: i0.ElementRef }, { token: i1.NgChartjsService }, { token: i2.StoreService }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Directive }); NgChartjsDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.9", type: NgChartjsDirective, selector: "canvas[ngChartjs]", inputs: { data: "data", datasets: "datasets", labels: "labels", options: "options", inlinePlugins: "inlinePlugins", chartType: "chartType", colors: "colors", legend: "legend", adding: "adding", removing: "removing", resetOption: "resetOption", noZone: "noZone", id: "id" }, outputs: { chartClick: "chartClick", chartHover: "chartHover" }, exportAs: ["ngChartjs"], usesOnChanges: true, ngImport: i0 }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: NgChartjsDirective, decorators: [{ type: Directive, args: [{ selector: 'canvas[ngChartjs]', exportAs: 'ngChartjs' }] }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i1.NgChartjsService }, { type: i2.StoreService }, { type: i0.NgZone }]; }, propDecorators: { data: [{ type: Input }], datasets: [{ type: Input }], labels: [{ type: Input }], options: [{ type: Input }], inlinePlugins: [{ type: Input }], chartType: [{ type: Input }], colors: [{ type: Input }], legend: [{ type: Input }], adding: [{ type: Input }], removing: [{ type: Input }], resetOption: [{ type: Input }], noZone: [{ type: Input }], id: [{ type: Input }], chartClick: [{ type: Output }], chartHover: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-chartjs.directive.js","sourceRoot":"","sources":["../../../../projects/ng-chartjs/src/lib/ng-chartjs.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  OnDestroy,\n  OnInit,\n  OnChanges,\n  EventEmitter,\n  ElementRef,\n  Input,\n  Output,\n  SimpleChanges,\n  Directive,\n  NgZone\n} from '@angular/core';\n// import { Chart, ChartConfiguration, ChartEvent, DefaultDataPoint, registerables } from 'chart.js';\nimport Chart, \n{ \n  ChartEvent, \n  ChartConfiguration, \n  ChartOptions, \n  ChartDataset, \n  ChartData \n} from 'chart.js/auto';\nimport { StoreService } from './store.service';\nimport { deepCopyJson, mergeJson, NgChartjsService } from './ng-chartjs.service';\nimport { getColors, Colors } from './colors';\n\nexport type Labels = Array<string | string[] | number | number[] | Date | Date[] | any | any[]>;\nexport type Orientation = 'oldest' | 'latest';\nexport interface NgChartjsEvent { event: ChartEvent; active: Array<{}>; }\n\n/* tslint:disable-next-line */\n@Directive({ selector: 'canvas[ngChartjs]', exportAs: 'ngChartjs' })\nexport class NgChartjsDirective implements OnDestroy, OnChanges, OnInit {\n\n  // 图表的点集，它应该是数组<number []>仅用于线，条和雷达，否则数字[];\n  // @ts-ignore\n  @Input() data: number[] | any[];\n  // 相当于chart.js内 data: {datasets: [{...}]}\n  // @ts-ignore\n  @Input() datasets: ChartData['datasets'];\n  // x轴标签。这对图表来说是必要的：线，条和雷达。并且只是图表的标签（悬停）：polarArea，pie和doughnut\n  @Input() labels: Labels = [];\n  // 相当于chart.js的option\n  @Input() options?: ChartConfiguration['options'];\n  // 内联插件属性\n  @Input() inlinePlugins?: any[];\n  // chartType line, bar, radar, pie, polarArea, doughnut\n  // @ts-ignore\n  @Input() chartType: ChartConfiguration['type'];\n  // 数据颜色，如果没有指定，将使用默认和|或随机颜色\n  // @ts-ignore\n  @Input() colors: Colors[];\n  // 是否显示图例\n  // @ts-ignore\n  @Input() legend: boolean;\n\n  // @ts-ignore\n  @Input() adding: { labels: Labels[], data: any[][] };\n  // @ts-ignore\n  @Input() removing: { orientation: Orientation };  // orientation is 'oldest' or 'latest\n  // @ts-ignore\n  @Input() resetOption?: ChartConfiguration['options'];\n\n  @Input() noZone = true; // disable angular NgZone\n  // @ts-ignore\n  @Input() id: string = null; // chart instance id\n\n  // 鼠标点击图表所有的区域\n  @Output() chartClick: EventEmitter<NgChartjsEvent> = new EventEmitter();\n  // 鼠标悬浮在标签或者活跃的点上面时\n  @Output() chartHover: EventEmitter<NgChartjsEvent> = new EventEmitter();\n\n  // get Chartjs object\n  chart!: Chart;\n  // @ts-ignore\n  private ctx: CanvasRenderingContext2D;\n  private initFlag = false;\n  private hasChanges = false;\n\n  private element: ElementRef;\n\n  public constructor(\n    element: ElementRef,\n    private ngChartjsService: NgChartjsService,\n    private storeService: StoreService,\n    private zone: NgZone) {\n    this.element = element;   // 获取指令所在canvas元素\n  }\n\n  ngOnInit(): void {\n    this.ctx = this.element.nativeElement.getContext('2d'); // 获取元素的ctx\n    this.initFlag = true; // 是否初始化了的标志\n\n    if (this.data || this.datasets) { // 判断data和datasets有一个有数据就刷新\n      if (this.noZone) {\n        this.zone.runOutsideAngular(() => {\n         this.refresh();\n        });\n      } else {\n        this.refresh();\n      }\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    // TODO: 插件变化刷新，开放刷新按钮\n    if (this.initFlag) {\n      // Check if the changes are in the data or datasets\n      if (changes.hasOwnProperty('data') || changes.hasOwnProperty('datasets')) {\n        if (changes.data) {\n          this.updateChartData(changes.data.currentValue);\n        } else {\n          this.updateChartData(changes.datasets.currentValue);\n        }\n        this.hasChanges = true;\n      }\n\n      if (changes.hasOwnProperty('labels')) {\n        this.chart.data.labels = changes.labels.currentValue;\n        this.hasChanges = true;\n      }\n\n      if (changes.hasOwnProperty('legend')) {\n        if (changes.legend.currentValue !== changes.legend.previousValue) {\n          // @ts-ignore\n          this.chart.options.plugins.legend.display = changes.legend.currentValue;\n          this.hasChanges = true;\n        }\n      }\n\n      if (changes.hasOwnProperty('adding')) {\n        this.addData_(changes.adding.currentValue.labels, changes.adding.currentValue.data);\n        this.hasChanges = true;\n      }\n\n      if (changes.hasOwnProperty('removing')) {\n        if (changes.removing.currentValue.orientation === 'oldest' || changes.removing.currentValue.orientation === 'latest') {\n          this.removeData_(changes.removing.currentValue.orientation);\n          this.hasChanges = true;\n        }\n      }\n\n      if (changes.hasOwnProperty('chartType')) {\n        this.refresh();\n      }\n\n      if (changes.hasOwnProperty('resetOption')) {\n        const resetOption = deepCopyJson(changes.resetOption.currentValue);\n        this.chart.options = mergeJson(resetOption, this.chart.options);\n        this.hasChanges = true;\n      }\n\n      if (this.hasChanges) {\n        this.chart.update();\n        this.hasChanges = false;\n      }\n\n      // change chart id\n      if (changes.hasOwnProperty('id')) {\n        this.removeChart(changes.id.previousValue);\n        this.addChart(changes.id.currentValue);\n      }\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (this.chart) {\n      this.chart.destroy();\n      // @ts-ignore\n      this.chart = void 0;\n\n      this.removeChart(this.id);\n    }\n  }\n\n  // update chartjs\n  update(): void {\n    this.chart.update();\n  }\n\n  // Dynamic add data\n  addData(labels: Labels[], data: any[][]): void {\n    this.addData_(labels, data);\n    this.update();\n  }\n  // Dynamic remove data, orientation is 'ildest' or 'latest'\n  removeData(orientation: Orientation): void {\n    this.removeData_(orientation);\n    this.update();\n  }\n\n  private refresh(): void {\n    this.ngOnDestroy();\n    this.chart = this.getChartBuilder(this.ctx/*, data, this.options*/);\n    this.addChart(this.id);\n  }\n\n  private removeChart(id: string): void {\n    if (this.element.nativeElement.hasAttribute('id')) {\n      this.storeService.removeChart(this.element.nativeElement.id);\n      return;\n    }\n    if (id !== null && id !== undefined) {\n      this.storeService.removeChart(id);  // delete chart instance.\n    }\n  }\n\n  private addChart(id: string): void {\n    if (this.element.nativeElement.hasAttribute('id')) {\n      this.storeService.addChart(this.element.nativeElement.id, this.chart);\n      return;\n    }\n    if (id !== null && id !== undefined) {\n      this.storeService.addChart(id, this.chart);\n    }\n  }\n\n  private updateChartData(newDataValues: number[] | any[]): void {\n    if (Array.isArray(newDataValues[0].data)) {\n      // @ts-ignore\n      // this.chart.data.datasets.forEach((dataset: ChartDataset, i: number) => {\n      //   dataset.data = newDataValues[i].data;\n\n      //   if (newDataValues[i].label) {\n      //     dataset.label = newDataValues[i].label;\n      //   }\n      // });\n\n      // @ts-ignore\n      this.chart.data.datasets = newDataValues;\n    } else {\n      // @ts-ignore\n      this.chart.data.datasets[0].data = newDataValues;\n    }\n    // update colors\n    // @ts-ignore\n    this.chart.data.datasets = this.updateColors(this.chart.data.datasets);\n  }\n\n  private getChartBuilder(ctx: CanvasRenderingContext2D/*, data:Array<any>, options:any*/): Chart {\n    const datasets = this.getDatasets();\n    let options: ChartOptions = this.options || {};\n    options = Object.assign({}, this.options); // 深复制options\n    mergeJson(options, {\n      plugins: {\n        legend: {\n          display: this.legend\n        }\n      }\n    })\n\n    // hock for onHover and onClick events\n    options.hover = options.hover || {};\n    if (!options.onHover) {\n      options.onHover = (event: ChartEvent, active: Array<{}>) => {\n        if (active && !active.length) {\n          return;\n        }\n        this.chartHover.emit({ event, active });\n      };\n    }\n\n    if (!options.onClick) {\n      options.onClick = (event: ChartEvent, active: Array<{}>) => {\n        this.chartClick.emit({ event, active });\n      };\n    }\n\n    const inlinePlugins = this.inlinePlugins || [];\n\n    const opts: ChartConfiguration = {\n      type: this.chartType,\n      data: {\n        labels: this.labels,\n        datasets: datasets   // TODO: 后续更改这个属性名字，否则警告\n      },\n      options: options,   // TODO: 后续更改这个属性名字，否则警告\n      plugins: inlinePlugins\n    };\n\n    return new Chart(ctx, opts);\n  }\n\n  // 获取 chart.js的datasets数据\n  private getDatasets(): ChartData['datasets'] {\n    // @ts-ignore\n    let datasets: ChartData['datasets'] = void 0;\n    // in case if datasets is not provided, but data is present\n    if (!this.datasets || !this.datasets.length && (this.data && this.data.length)) {\n      if (Array.isArray(this.data[0])) {\n        datasets = (this.data as number[][]).map((data: number[], index: number) => {\n          return { data, label: this.labels[index] || `Label ${index}` };\n        });\n      } else {\n        datasets = [{ data: this.data, label: `Label 0` }];\n      }\n    }\n\n    datasets = this.updateColors(datasets); // update colors\n\n    if (!datasets) {\n      throw new Error(`ng-chartjs configuration error,\n      data or datasets field are required to render char ${this.chartType}`);\n    }\n\n    return datasets;\n  }\n\n  // update dataset colors\n  private updateColors(datasets: ChartData['datasets']): ChartData['datasets'] {\n    if (this.datasets && this.datasets.length || (datasets && datasets.length)) {\n      // fix elm type, pre type is number\n      datasets = (this.datasets || datasets).map((elm: any, index: number) => {\n        const newElm = Object.assign({}, elm);\n        if (this.colors && this.colors.length) {\n          Object.assign(newElm, this.colors[index]);\n        } else {\n          // @ts-ignore\n          Object.assign(newElm, getColors(this.chartType, index, newElm.data.length));\n        }\n        return newElm;\n      });\n    }\n    return datasets;\n  }\n\n  private addData_(labels: Labels[], data: any[][]): void {\n    if (labels.length === 0 || data.length === 0) {\n      return;\n    }\n    // update labels\n    // @ts-ignore\n    labels.forEach((label) => { this.chart.data.labels.push(label); });\n    // @ts-ignore\n    this.chart.data.datasets.forEach((dataset, index) => {\n      if (data[index]) {\n        for (let i = 0; i < data[index].length; i++) {\n          // @ts-ignore\n          dataset.data.push(data[index][i]);\n        }\n      } else {\n        console.log('The added data does not match the original data');\n        return;\n      }\n    });\n  }\n\n  private removeData_(orientation: Orientation): void {\n    // fix: support to oldest feature\n    if (orientation === 'latest') {\n      // @ts-ignore\n      this.chart.data.labels.pop();\n      // @ts-ignore\n      this.chart.data.datasets.forEach((dataset: ChartData['datasets']) => {\n        // @ts-ignore\n        dataset.data.pop();\n      });\n    } else if (orientation === 'oldest') {\n      // @ts-ignore\n      this.chart.data.labels.shift();\n      // @ts-ignore\n      this.chart.data.datasets.forEach((dataset: ChartData['datasets']) => {\n        // @ts-ignore\n        dataset.data.shift();\n      });\n    }\n  }\n}\n"]}