@alauda-fe/common
Version:
Alauda frontend team common codes.
237 lines • 32.6 kB
JavaScript
import { Inject, Injectable, InjectionToken, Optional } from '@angular/core';
import { scaleBand, scaleLinear, scalePoint, scaleTime, } from 'd3';
import { BehaviorSubject, Subject, map, tap } from 'rxjs';
import { publishRef } from '../core/utils/public-api';
import { DEFAULT_COLORS, basics } from './constants';
import { Orientation, } from './types';
import { ScaleType, calculateViewDimensions, getScaleType, getXDomain, getYDomain, } from './utils';
import * as i0 from "@angular/core";
export const CHART_COLORS_TOKEN = new InjectionToken('chart_colors');
export class ContextChartService {
get scaleType() {
return this._scaleType || getScaleType(this.xSeriesValue);
}
set scaleType(type) {
this._scaleType = type;
}
get x() {
if (this.adoptCache('x')) {
return this.cacheXScale;
}
return (this.cacheXScale = this.getXScale(getXDomain(this.xSeriesValue, this.scaleType)));
}
get y() {
if (this.adoptCache('y')) {
return this.cacheYScale;
}
return (this.cacheYScale = this.calculateY());
}
get innerDomain() {
const domainSet = new Set();
for (const [, item] of this.chartData.entries()) {
for (const [, value] of item.values.entries()) {
domainSet.add(value.x);
}
}
return [...domainSet];
}
get colors() {
return this.chartColors?.length ? this.chartColors : DEFAULT_COLORS;
}
get horizontal() {
return (this.barOptions.isBand ||
this.barOptions.orientation === Orientation.HORIZONTAL);
}
constructor(chartColors) {
this.chartColors = chartColors;
this.chartView = {
width: 0,
height: 0,
xOffset: 0,
};
this.update$$ = new Subject();
this.legendChange$$ = new Subject();
this.legendActivate$$ = new Subject();
this.disabledLegend = new Set();
this.showXLabel = false;
this.showYLabel = false;
this.xSeriesValue = [];
this.ySeriesValue = [];
this.barOptions = {
isBand: false,
orientation: Orientation.VERTICAL,
type: 'standard',
};
this.chartData = [];
this.chartData$ = new BehaviorSubject([]);
this.seriesData$ = new BehaviorSubject([]);
this.triggerInitData$ = new BehaviorSubject([]);
this.ySeriesDomain = [0, 0];
this.xSeriesDomain = [0, 0];
this.cacheDomainParams = {
yDomain: [0, 0],
xDomain: [0, 0],
height: 0,
width: 0,
};
this.noData = true;
this.noData$ = this.chartData$.pipe(map(res => res.every(item => item.values.every(d => d.y === null))), tap(noData => (this.noData = noData)), publishRef());
this.barInnerScale = (barPadding = 8) => {
const width = this.x.bandwidth();
const spacing = this.innerDomain.length / (width / barPadding + 1);
return scaleBand()
.range([0, width])
.paddingInner(spacing)
.domain(this.innerDomain);
};
this.calculateY = () => {
const horizontal = this.barOptions.orientation === Orientation.HORIZONTAL;
const height = horizontal ? this.chartView.width : this.chartView.height;
this.cacheDomainParams = {
...this.cacheDomainParams,
yDomain: this.ySeriesDomain,
height,
};
return scaleLinear().domain(this.ySeriesDomain).range([height, 0]).nice();
};
this.getChartDefaultColor = (index) => {
const colorIndex = index % this.colors.length;
return this.colors[colorIndex];
};
}
initContext() {
this.xSeriesValue = [];
this.ySeriesValue = [];
}
setChartData(data) {
this.chartData$.next(data);
this.calculationYDomain();
this.xSeriesDomain = getXDomain(this.xSeriesValue, this.scaleType);
}
updateChartView(width, height, margin) {
this.chartView = calculateViewDimensions({
width,
height,
margin: margin || basics.margin,
showXLabel: this.showXLabel,
showYLabel: this.showYLabel,
});
this.cacheDomainParams.width = this.chartView.width;
this.update$$.next();
}
calculationYDomain() {
const horizontal = this.barOptions.orientation === Orientation.HORIZONTAL;
const defaultDomain = getYDomain(this.ySeriesValue);
const min = Math.min(0, this.yScaleMin || 0, defaultDomain[0]);
const max = Math.max(0, this.yScaleMax || 0, defaultDomain[1]);
this.ySeriesDomain = horizontal ? [max, min] : [min, max];
}
calculationYAxisMax() {
this.ySeriesValue = this.chartData.reduce((prev, curr) => {
if (!this.disabledLegend.has(curr.name)) {
return [
...prev,
...(this.barOptions.isBand
? this.getYSeriesItemDomain(curr.values)
: curr.values.map(d => d.y)),
];
}
return prev;
}, []);
this.calculationYDomain();
}
getYSeriesItemDomain(values) {
return this.barOptions.type === 'stacked'
? [
0,
values.reduce((prev, curr) => {
if (!this.disabledLegend.has(curr.x)) {
return curr.y + prev;
}
return prev;
}, 0),
]
: values.reduce((prev, curr) => {
if (!this.disabledLegend.has(curr.x)) {
return [...prev, curr.y];
}
return prev;
}, []);
}
getXScale(domain) {
const width = this.barOptions.orientation === Orientation.HORIZONTAL
? this.chartView.height
: this.chartView.width;
if (this.barOptions.isBand) {
return scaleBand().range([0, width]).domain(domain).padding(0.2);
}
switch (this.scaleType) {
case ScaleType.TIME: {
return scaleTime().domain(domain).range([0, width]);
}
case ScaleType.LINEAR: {
return scaleLinear().domain(domain).range([0, width]);
}
case ScaleType.ORDINAL: {
return scalePoint().domain(domain).range([0, width]);
}
default: {
break;
}
}
}
adoptCache(type) {
const { xDomain, width, yDomain, height } = this.cacheDomainParams;
if (type === 'x') {
return (xDomain[0] === this.ySeriesDomain[0] &&
xDomain[1] === this.ySeriesDomain[1] &&
width === this.chartView.width &&
this.chartView.width &&
this.ySeriesDomain[1]);
}
return (yDomain[0] === this.ySeriesDomain[0] &&
yDomain[1] === this.ySeriesDomain[1] &&
height === this.chartView.height &&
this.chartView.height);
}
selectedLegend(payload) {
const legend = payload.legendItem;
this.disabledLegend.delete(legend.name);
this.legendActivate$$.next(legend.name);
this.calculationYAxisMax();
this.legendChange$$.next(payload);
this.legendActivate$$.next('');
this.update$$.next();
}
selectedAllLegend() {
this.disabledLegend.clear();
this.calculationYAxisMax();
this.legendChange$$.next(null);
this.update$$.next();
}
unselectAllLegend(legends) {
this.disabledLegend = new Set(legends);
this.calculationYAxisMax();
this.legendChange$$.next(null);
this.update$$.next();
}
deactivateLegend(payload) {
const legend = payload.legendItem;
this.disabledLegend.add(legend.name);
this.calculationYAxisMax();
this.legendActivate$$.next('');
this.legendChange$$.next(payload);
this.update$$.next();
}
static { this.ɵfac = function ContextChartService_Factory(t) { return new (t || ContextChartService)(i0.ɵɵinject(CHART_COLORS_TOKEN, 8)); }; }
static { this.ɵprov = /*@__PURE__*/ i0.ɵɵdefineInjectable({ token: ContextChartService, factory: ContextChartService.ɵfac }); }
}
(() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(ContextChartService, [{
type: Injectable
}], () => [{ type: undefined, decorators: [{
type: Optional
}, {
type: Inject,
args: [CHART_COLORS_TOKEN]
}] }], null); })();
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chart.service.js","sourceRoot":"","sources":["../../../../../libs/common/src/chart/chart.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, InjectionToken, Optional } from '@angular/core';\nimport {\n  ScaleBand,\n  ScaleLinear,\n  ScalePoint,\n  ScaleTime,\n  scaleBand,\n  scaleLinear,\n  scalePoint,\n  scaleTime,\n} from 'd3';\nimport { BehaviorSubject, Subject, map, tap } from 'rxjs';\n\nimport { publishRef } from '../core/utils/public-api';\nimport { ChartData, ChartDataItem } from '../view-chart/types';\n\nimport { LegendItem } from './common/legend/legend.component';\nimport { DEFAULT_COLORS, basics } from './constants';\nimport {\n  ChartMargin,\n  ChartView,\n  DomainParams,\n  Orientation,\n  SeriesData,\n} from './types';\nimport {\n  ScaleType,\n  calculateViewDimensions,\n  getScaleType,\n  getXDomain,\n  getYDomain,\n} from './utils';\n\nexport const CHART_COLORS_TOKEN = new InjectionToken<string[]>('chart_colors');\n\ninterface LegendChangePayload {\n  legendItem: LegendItem;\n  legends: LegendItem[];\n}\n\n@Injectable()\nexport class ContextChartService {\n  margin: ChartMargin;\n\n  chartView: ChartView = {\n    width: 0,\n    height: 0,\n    xOffset: 0,\n  };\n\n  update$$ = new Subject<void>();\n\n  legendChange$$ = new Subject<LegendChangePayload>();\n\n  legendActivate$$ = new Subject<string>();\n\n  disabledLegend: Set<string> = new Set();\n\n  showXLabel = false;\n\n  showYLabel = false;\n\n  yScaleMin: number;\n\n  yScaleMax: number;\n\n  xSeriesValue: Array<string | number | Date> = [];\n\n  ySeriesValue: number[] = [];\n\n  _scaleType: ScaleType;\n\n  barOptions: {\n    isBand: boolean;\n    orientation: Orientation | string;\n    type: string;\n  } = {\n    isBand: false,\n    orientation: Orientation.VERTICAL,\n    type: 'standard',\n  };\n\n  chartData: ChartData[] = [];\n\n  chartData$ = new BehaviorSubject<ChartData[]>([]);\n\n  seriesData$ = new BehaviorSubject<SeriesData[]>([]);\n\n  triggerInitData$ = new BehaviorSubject<ChartData[]>([]);\n\n  ySeriesDomain = [0, 0];\n\n  xSeriesDomain: unknown[] = [0, 0];\n\n  cacheDomainParams: DomainParams = {\n    yDomain: [0, 0],\n    xDomain: [0, 0],\n    height: 0,\n    width: 0,\n  };\n\n  cacheXScale:\n    | ScaleLinear<number, number>\n    | ScaleTime<number, number>\n    | ScalePoint<string>;\n\n  cacheYScale: ScaleLinear<number, number>;\n\n  noData = true;\n\n  cacheScale: ScaleLinear<number, number>;\n\n  noData$ = this.chartData$.pipe(\n    map(res => res.every(item => item.values.every(d => d.y === null))),\n    tap(noData => (this.noData = noData)),\n    publishRef(),\n  );\n\n  get scaleType() {\n    return this._scaleType || getScaleType(this.xSeriesValue);\n  }\n\n  set scaleType(type) {\n    this._scaleType = type;\n  }\n\n  get x() {\n    if (this.adoptCache('x')) {\n      return this.cacheXScale;\n    }\n    return (this.cacheXScale = this.getXScale(\n      getXDomain(this.xSeriesValue, this.scaleType),\n    ));\n  }\n\n  get y() {\n    if (this.adoptCache('y')) {\n      return this.cacheYScale;\n    }\n    return (this.cacheYScale = this.calculateY());\n  }\n\n  get innerDomain() {\n    const domainSet: Set<string> = new Set();\n    for (const [, item] of this.chartData.entries()) {\n      for (const [, value] of item.values.entries()) {\n        domainSet.add(value.x as string);\n      }\n    }\n    return [...domainSet];\n  }\n\n  get colors() {\n    return this.chartColors?.length ? this.chartColors : DEFAULT_COLORS;\n  }\n\n  get horizontal() {\n    return (\n      this.barOptions.isBand ||\n      this.barOptions.orientation === Orientation.HORIZONTAL\n    );\n  }\n\n  constructor(\n    @Optional()\n    @Inject(CHART_COLORS_TOKEN)\n    private readonly chartColors: string[],\n  ) {}\n\n  initContext() {\n    this.xSeriesValue = [];\n    this.ySeriesValue = [];\n  }\n\n  setChartData(data: ChartData[]) {\n    this.chartData$.next(data);\n    this.calculationYDomain();\n    this.xSeriesDomain = getXDomain(this.xSeriesValue, this.scaleType);\n  }\n\n  barInnerScale = (barPadding = 8) => {\n    const width = (<ScaleBand<string>>this.x).bandwidth();\n    const spacing = this.innerDomain.length / (width / barPadding + 1);\n    return scaleBand()\n      .range([0, width])\n      .paddingInner(spacing)\n      .domain(this.innerDomain);\n  };\n\n  calculateY = () => {\n    const horizontal = this.barOptions.orientation === Orientation.HORIZONTAL;\n    const height = horizontal ? this.chartView.width : this.chartView.height;\n    this.cacheDomainParams = {\n      ...this.cacheDomainParams,\n      yDomain: this.ySeriesDomain,\n      height,\n    };\n    return scaleLinear().domain(this.ySeriesDomain).range([height, 0]).nice();\n  };\n\n  updateChartView(width?: number, height?: number, margin?: ChartMargin) {\n    this.chartView = calculateViewDimensions({\n      width,\n      height,\n      margin: margin || basics.margin,\n      showXLabel: this.showXLabel,\n      showYLabel: this.showYLabel,\n    });\n    this.cacheDomainParams.width = this.chartView.width;\n    this.update$$.next();\n  }\n\n  getChartDefaultColor = (index: number) => {\n    const colorIndex = index % this.colors.length;\n    return this.colors[colorIndex];\n  };\n\n  private calculationYDomain() {\n    const horizontal = this.barOptions.orientation === Orientation.HORIZONTAL;\n    const defaultDomain = getYDomain(this.ySeriesValue);\n    const min = Math.min(0, this.yScaleMin || 0, defaultDomain[0]);\n    const max = Math.max(0, this.yScaleMax || 0, defaultDomain[1]);\n    this.ySeriesDomain = horizontal ? [max, min] : [min, max];\n  }\n\n  calculationYAxisMax() {\n    this.ySeriesValue = this.chartData.reduce((prev, curr) => {\n      if (!this.disabledLegend.has(curr.name)) {\n        return [\n          ...prev,\n          ...(this.barOptions.isBand\n            ? this.getYSeriesItemDomain(curr.values)\n            : curr.values.map(d => d.y)),\n        ];\n      }\n      return prev;\n    }, []);\n    this.calculationYDomain();\n  }\n\n  private getYSeriesItemDomain(values: ChartDataItem[]) {\n    return this.barOptions.type === 'stacked'\n      ? [\n          0,\n          values.reduce((prev, curr) => {\n            if (!this.disabledLegend.has(curr.x as string)) {\n              return curr.y + prev;\n            }\n            return prev;\n          }, 0),\n        ]\n      : values.reduce((prev, curr) => {\n          if (!this.disabledLegend.has(curr.x as string)) {\n            return [...prev, curr.y];\n          }\n          return prev;\n        }, []);\n  }\n\n  private getXScale(domain: Array<Date | number | any>) {\n    const width =\n      this.barOptions.orientation === Orientation.HORIZONTAL\n        ? this.chartView.height\n        : this.chartView.width;\n    if (this.barOptions.isBand) {\n      return scaleBand().range([0, width]).domain(domain).padding(0.2);\n    }\n    switch (this.scaleType) {\n      case ScaleType.TIME: {\n        return scaleTime().domain(domain).range([0, width]);\n      }\n      case ScaleType.LINEAR: {\n        return scaleLinear().domain(domain).range([0, width]);\n      }\n      case ScaleType.ORDINAL: {\n        return scalePoint().domain(domain).range([0, width]);\n      }\n      default: {\n        break;\n      }\n    }\n  }\n\n  private adoptCache(type: 'y' | 'x') {\n    const { xDomain, width, yDomain, height } = this.cacheDomainParams;\n    if (type === 'x') {\n      return (\n        xDomain[0] === this.ySeriesDomain[0] &&\n        xDomain[1] === this.ySeriesDomain[1] &&\n        width === this.chartView.width &&\n        this.chartView.width &&\n        this.ySeriesDomain[1]\n      );\n    }\n    return (\n      yDomain[0] === this.ySeriesDomain[0] &&\n      yDomain[1] === this.ySeriesDomain[1] &&\n      height === this.chartView.height &&\n      this.chartView.height\n    );\n  }\n\n  selectedLegend(payload: LegendChangePayload) {\n    const legend = payload.legendItem;\n    this.disabledLegend.delete(legend.name);\n    this.legendActivate$$.next(legend.name);\n    this.calculationYAxisMax();\n    this.legendChange$$.next(payload);\n    this.legendActivate$$.next('');\n    this.update$$.next();\n  }\n\n  selectedAllLegend() {\n    this.disabledLegend.clear();\n    this.calculationYAxisMax();\n    this.legendChange$$.next(null);\n    this.update$$.next();\n  }\n\n  unselectAllLegend(legends: string[]) {\n    this.disabledLegend = new Set(legends);\n    this.calculationYAxisMax();\n    this.legendChange$$.next(null);\n    this.update$$.next();\n  }\n\n  deactivateLegend(payload: LegendChangePayload) {\n    const legend = payload.legendItem;\n    this.disabledLegend.add(legend.name);\n    this.calculationYAxisMax();\n    this.legendActivate$$.next('');\n    this.legendChange$$.next(payload);\n    this.update$$.next();\n  }\n}\n"]}