ng-chartjs
Version:
This is a Angular chart.js library.
317 lines • 41 kB
JavaScript
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"]}