@orbcharts/plugins-basic 3.0.1 → 3.0.3

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.
Files changed (133) hide show
  1. package/LICENSE +200 -200
  2. package/dist/lib/gridObservables.d.ts +1 -0
  3. package/dist/orbcharts-plugins-basic.es.js +14800 -13543
  4. package/dist/orbcharts-plugins-basic.umd.js +129 -99
  5. package/dist/src/base/BaseOrdinalBubbles.d.ts +26 -0
  6. package/dist/src/base/BaseRacingBars.d.ts +2 -1
  7. package/dist/src/base/BaseRacingLabels.d.ts +1 -0
  8. package/dist/src/base/BaseXZoom.d.ts +18 -0
  9. package/dist/src/grid/gridObservables.d.ts +4 -7
  10. package/dist/src/multiValue/defaults.d.ts +5 -1
  11. package/dist/src/multiValue/index.d.ts +4 -0
  12. package/dist/src/multiValue/multiValueObservables.d.ts +14 -1
  13. package/dist/src/multiValue/plugins/OrdinalAux.d.ts +3 -0
  14. package/dist/src/multiValue/plugins/OrdinalAxis.d.ts +3 -0
  15. package/dist/src/multiValue/plugins/OrdinalBubbles.d.ts +3 -0
  16. package/dist/src/multiValue/plugins/OrdinalZoom.d.ts +1 -0
  17. package/lib/core-types.ts +7 -7
  18. package/lib/core.ts +6 -6
  19. package/lib/gridObservables.ts +6 -0
  20. package/lib/plugins-basic-types.ts +6 -6
  21. package/package.json +48 -44
  22. package/src/base/BaseBars.ts +765 -765
  23. package/src/base/BaseBarsTriangle.ts +676 -676
  24. package/src/base/BaseDots.ts +464 -464
  25. package/src/base/BaseGroupAxis.ts +691 -691
  26. package/src/base/BaseLegend.ts +684 -684
  27. package/src/base/BaseLineAreas.ts +629 -629
  28. package/src/base/BaseLines.ts +706 -706
  29. package/src/base/BaseOrdinalBubbles.ts +728 -0
  30. package/src/base/BaseRacingBars.ts +582 -551
  31. package/src/base/BaseRacingLabels.ts +404 -396
  32. package/src/base/BaseRacingValueLabels.ts +403 -403
  33. package/src/base/BaseStackedBars.ts +782 -782
  34. package/src/base/BaseTooltip.ts +386 -386
  35. package/src/base/BaseValueAxis.ts +600 -600
  36. package/src/base/BaseXAxis.ts +427 -427
  37. package/src/base/BaseXZoom.ts +242 -0
  38. package/src/base/BaseYAxis.ts +389 -389
  39. package/src/base/types.ts +2 -2
  40. package/src/const.ts +30 -30
  41. package/src/grid/defaults.ts +213 -213
  42. package/src/grid/gridObservables.ts +635 -612
  43. package/src/grid/index.ts +16 -16
  44. package/src/grid/plugins/Bars.ts +69 -69
  45. package/src/grid/plugins/BarsPN.ts +66 -66
  46. package/src/grid/plugins/BarsTriangle.ts +73 -73
  47. package/src/grid/plugins/Dots.ts +68 -68
  48. package/src/grid/plugins/GridLegend.ts +107 -107
  49. package/src/grid/plugins/GridTooltip.ts +66 -66
  50. package/src/grid/plugins/GroupAux.ts +1095 -1120
  51. package/src/grid/plugins/GroupAxis.ts +73 -73
  52. package/src/grid/plugins/GroupZoom.ts +218 -218
  53. package/src/grid/plugins/LineAreas.ts +65 -65
  54. package/src/grid/plugins/Lines.ts +59 -59
  55. package/src/grid/plugins/StackedBars.ts +64 -64
  56. package/src/grid/plugins/StackedValueAxis.ts +96 -96
  57. package/src/grid/plugins/ValueAxis.ts +94 -94
  58. package/src/index.ts +6 -6
  59. package/src/multiGrid/defaults.ts +244 -244
  60. package/src/multiGrid/index.ts +14 -14
  61. package/src/multiGrid/multiGridObservables.ts +50 -50
  62. package/src/multiGrid/plugins/MultiBars.ts +108 -108
  63. package/src/multiGrid/plugins/MultiBarsTriangle.ts +114 -114
  64. package/src/multiGrid/plugins/MultiDots.ts +102 -102
  65. package/src/multiGrid/plugins/MultiGridLegend.ts +169 -169
  66. package/src/multiGrid/plugins/MultiGridTooltip.ts +66 -66
  67. package/src/multiGrid/plugins/MultiGroupAxis.ts +137 -137
  68. package/src/multiGrid/plugins/MultiLineAreas.ts +107 -107
  69. package/src/multiGrid/plugins/MultiLines.ts +101 -101
  70. package/src/multiGrid/plugins/MultiStackedBars.ts +106 -106
  71. package/src/multiGrid/plugins/MultiStackedValueAxis.ts +134 -134
  72. package/src/multiGrid/plugins/MultiValueAxis.ts +134 -134
  73. package/src/multiGrid/plugins/OverlappingStackedValueAxes.ts +300 -300
  74. package/src/multiGrid/plugins/OverlappingValueAxes.ts +300 -300
  75. package/src/multiValue/defaults.ts +523 -431
  76. package/src/multiValue/index.ts +16 -12
  77. package/src/multiValue/multiValueObservables.ts +781 -666
  78. package/src/multiValue/plugins/MultiValueLegend.ts +107 -107
  79. package/src/multiValue/plugins/MultiValueTooltip.ts +66 -66
  80. package/src/multiValue/plugins/OrdinalAux.ts +661 -0
  81. package/src/multiValue/plugins/OrdinalAxis.ts +525 -0
  82. package/src/multiValue/plugins/OrdinalBubbles.ts +226 -0
  83. package/src/multiValue/plugins/OrdinalZoom.ts +57 -0
  84. package/src/multiValue/plugins/RacingBars.ts +375 -373
  85. package/src/multiValue/plugins/RacingCounterTexts.ts +300 -300
  86. package/src/multiValue/plugins/RacingValueAxis.ts +114 -114
  87. package/src/multiValue/plugins/Scatter.ts +486 -426
  88. package/src/multiValue/plugins/ScatterBubbles.ts +635 -554
  89. package/src/multiValue/plugins/XAxis.ts +107 -107
  90. package/src/multiValue/plugins/XYAux.ts +683 -682
  91. package/src/multiValue/plugins/XYAxes.ts +194 -194
  92. package/src/multiValue/plugins/XYAxes_legacy.ts +683 -683
  93. package/src/multiValue/plugins/XZoom.ts +40 -299
  94. package/src/noneData/defaults.ts +102 -102
  95. package/src/noneData/index.ts +3 -3
  96. package/src/noneData/plugins/Container.ts +27 -27
  97. package/src/noneData/plugins/Tooltip.ts +373 -373
  98. package/src/relationship/defaults.ts +221 -221
  99. package/src/relationship/index.ts +5 -5
  100. package/src/relationship/plugins/ForceDirected.ts +1173 -1173
  101. package/src/relationship/plugins/ForceDirectedBubbles.ts +1411 -1411
  102. package/src/relationship/plugins/RelationshipLegend.ts +100 -100
  103. package/src/relationship/plugins/RelationshipTooltip.ts +66 -66
  104. package/src/relationship/relationshipObservables.ts +49 -49
  105. package/src/series/defaults.ts +221 -221
  106. package/src/series/index.ts +9 -9
  107. package/src/series/plugins/Bubbles.ts +636 -636
  108. package/src/series/plugins/Pie.ts +623 -623
  109. package/src/series/plugins/PieEventTexts.ts +284 -284
  110. package/src/series/plugins/PieLabels.ts +640 -640
  111. package/src/series/plugins/Rose.ts +516 -516
  112. package/src/series/plugins/RoseLabels.ts +600 -600
  113. package/src/series/plugins/SeriesLegend.ts +107 -107
  114. package/src/series/plugins/SeriesTooltip.ts +66 -66
  115. package/src/series/seriesObservables.ts +145 -145
  116. package/src/series/seriesUtils.ts +51 -51
  117. package/src/tree/defaults.ts +102 -102
  118. package/src/tree/index.ts +4 -4
  119. package/src/tree/plugins/TreeLegend.ts +100 -100
  120. package/src/tree/plugins/TreeMap.ts +341 -341
  121. package/src/tree/plugins/TreeTooltip.ts +66 -66
  122. package/src/utils/commonUtils.ts +31 -31
  123. package/src/utils/d3Graphics.ts +176 -176
  124. package/src/utils/d3Utils.ts +92 -92
  125. package/src/utils/observables.ts +14 -14
  126. package/src/utils/orbchartsUtils.ts +129 -129
  127. package/tsconfig.base.json +13 -13
  128. package/tsconfig.json +2 -2
  129. package/vite.config.js +22 -22
  130. package/dist/src/multiValue/plugins/OrdinalXAxis.d.ts +0 -0
  131. package/dist/src/multiValue/plugins/RankingAxis_legacy.d.ts +0 -0
  132. package/src/multiValue/plugins/OrdinalXAxis.ts +0 -0
  133. package/src/multiValue/plugins/RankingAxis_legacy.ts +0 -109
@@ -1,552 +1,583 @@
1
- import * as d3 from 'd3'
2
- import {
3
- combineLatest,
4
- map,
5
- switchMap,
6
- takeUntil,
7
- distinctUntilChanged,
8
- shareReplay,
9
- Observable,
10
- Subject } from 'rxjs'
11
- import type { BasePluginFn } from './types'
12
- import type {
13
- ComputedDatumMultiValue,
14
- ComputedDataMultiValue,
15
- // ComputedLayoutDataGrid,
16
- DataFormatterTypeMap,
17
- ContainerPositionScaled,
18
- ContainerSize,
19
- EventMultiValue,
20
- ChartParams,
21
- Layout,
22
- TransformData } from '../../lib/core-types'
23
- import type { BaseRacingBarsParams } from '../../lib/plugins-basic-types'
24
- import { getD3TransitionEase } from '../utils/d3Utils'
25
- import { getClassName, getUniID } from '../utils/orbchartsUtils'
26
- import { multiValueContainerSelectionsObservable } from '../multiValue/multiValueObservables'
27
-
28
- // export interface BaseBarsParams {
29
- // // barType: BarType
30
- // barWidth: number
31
- // barPadding: number
32
- // barGroupPadding: number // 群組和群組間的間隔
33
- // barRadius: number | boolean
34
- // }
35
-
36
- interface BaseRacingBarsContext {
37
- selection: d3.Selection<any, unknown, any, unknown>
38
- computedData$: Observable<ComputedDataMultiValue>
39
- visibleComputedRankingData$: Observable<ComputedDatumMultiValue[][]>
40
- CategoryDataMap$: Observable<Map<string, ComputedDatumMultiValue[]>>
41
- fullParams$: Observable<BaseRacingBarsParams>
42
- fullChartParams$: Observable<ChartParams>
43
- // xyValueIndex$: Observable<[number, number]>
44
- highlight$: Observable<ComputedDatumMultiValue[]>
45
- rankingItemHeight$: Observable<number>
46
- rankingScaleList$: Observable<d3.ScalePoint<string>[]>
47
- containerPosition$: Observable<ContainerPositionScaled[]>
48
- containerSize$: Observable<ContainerSize>
49
- xScale$: Observable<(n: number) => number>
50
- isCategorySeprate$: Observable<boolean>
51
- event$: Subject<EventMultiValue>
52
- }
53
-
54
- interface RenderGraphicGParams {
55
- containerSelection: d3.Selection<SVGGElement, ComputedDatumMultiValue[], any, any>
56
- visibleComputedRankingData: ComputedDatumMultiValue[][]
57
- rankingScaleList: d3.ScalePoint<string>[]
58
- transitionDuration: number
59
- }
60
-
61
- interface RenderBarParams {
62
- graphicGSelection: d3.Selection<SVGGElement, ComputedDatumMultiValue, any, any>
63
- rectClassName: string
64
- // xyValueIndex: [number, number]
65
- xScale: (n: number) => number
66
- fullParams: BaseRacingBarsParams
67
- barWidth: number
68
- transitionDuration: number
69
- }
70
-
71
- type ClipPathDatum = {
72
- id: string;
73
- // x: number;
74
- // y: number;
75
- width: number;
76
- height: number;
77
- }
78
-
79
-
80
- function renderGraphicG ({ containerSelection, visibleComputedRankingData, rankingScaleList, transitionDuration }: RenderGraphicGParams) {
81
- containerSelection
82
- .each((_, categoryIndex, g) => {
83
- const container = d3.select(g[categoryIndex])
84
- const graphicG = container.selectAll<SVGGElement, ComputedDatumMultiValue>(`g`)
85
- .data(visibleComputedRankingData[categoryIndex] ?? [], d => d.id)
86
- .join(
87
- enter => {
88
- return enter
89
- .append('g')
90
- .attr('cursor', 'pointer')
91
- .attr('transform', d => {
92
- return `translate(0, ${rankingScaleList[categoryIndex] && rankingScaleList[categoryIndex](d.label)})`
93
- })
94
- },
95
- update => {
96
- return update
97
- .transition()
98
- .duration(transitionDuration)
99
- .ease(d3.easeLinear)
100
- .attr('transform', d => {
101
- return `translate(0, ${rankingScaleList[categoryIndex] && rankingScaleList[categoryIndex](d.label)})`
102
- })
103
- },
104
- exit => exit.remove()
105
- )
106
- })
107
-
108
- const graphicBarSelection: d3.Selection<SVGRectElement, ComputedDatumMultiValue, SVGGElement, unknown> = containerSelection.selectAll(`g`)
109
-
110
- return graphicBarSelection
111
- }
112
-
113
- function renderRectBars ({ graphicGSelection, rectClassName, xScale, fullParams, barWidth, transitionDuration }: RenderBarParams) {
114
-
115
- graphicGSelection
116
- .each((datum, i, g) => {
117
- // const containerIndex = isCategorySeprate ? datum.categoryIndex : 0
118
- // const barWidth = barWidthList[containerIndex]
119
- const barHalfWidth = barWidth / 2
120
- const radius = fullParams.bar.barRadius === true ? barHalfWidth
121
- : fullParams.bar.barRadius === false ? 0
122
- : typeof fullParams.bar.barRadius == 'number' ? fullParams.bar.barRadius
123
- : 0
124
-
125
- const gSelection = d3.select(g[i])
126
- gSelection.selectAll<SVGRectElement, ComputedDatumMultiValue>(`rect.${rectClassName}`)
127
- .data([datum], d => d.id)
128
- .join(
129
- enter => {
130
- return enter
131
- .append('rect')
132
- .classed(rectClassName, true)
133
- .attr('cursor', 'pointer')
134
- // .attr('width', d => 1)
135
- .attr('width', d => xScale(d.value[d.xValueIndex]) ?? 1)
136
- .attr('height', barWidth)
137
- },
138
- update => {
139
- return update
140
- .transition()
141
- .duration(transitionDuration)
142
- .ease(d3.easeLinear)
143
- .attr('width', d => xScale(d.value[d.xValueIndex]) ?? 1)
144
- .attr('height', barWidth)
145
- },
146
- exit => exit.remove()
147
- )
148
- .attr('transform', `translate(0, ${-barHalfWidth})`)
149
- .attr('fill', d => d.color)
150
- .attr('rx', radius)
151
- .attr('ry', radius)
152
- })
153
-
154
- const graphicBarSelection: d3.Selection<SVGRectElement, ComputedDatumMultiValue, SVGGElement, unknown> = graphicGSelection.selectAll(`rect.${rectClassName}`)
155
-
156
- return graphicBarSelection
157
- }
158
-
159
- function renderClipPath ({ defsSelection, clipPathData }: {
160
- defsSelection: d3.Selection<SVGDefsElement, any, any, any>
161
- clipPathData: ClipPathDatum[]
162
- }) {
163
- const clipPath = defsSelection
164
- .selectAll<SVGClipPathElement, Layout>('clipPath')
165
- .data(clipPathData)
166
- .join(
167
- enter => {
168
- return enter
169
- .append('clipPath')
170
- },
171
- update => update,
172
- exit => exit.remove()
173
- )
174
- .attr('id', d => d.id)
175
- .each((d, i, g) => {
176
- const rect = d3.select(g[i])
177
- .selectAll<SVGRectElement, typeof d>('rect')
178
- .data([d])
179
- .join(
180
- enter => {
181
- return enter
182
- .append('rect')
183
- },
184
- update => update,
185
- exit => exit.remove()
186
- )
187
- .attr('x', 0)
188
- .attr('y', 0)
189
- .attr('width', _d => _d.width)
190
- .attr('height', _d => _d.height)
191
- })
192
- }
193
-
194
- function highlight ({ selection, ids, fullChartParams }: {
195
- selection: d3.Selection<any, ComputedDatumMultiValue, any, any>
196
- ids: string[]
197
- fullChartParams: ChartParams
198
- }) {
199
- selection.interrupt('highlight')
200
-
201
- if (!ids.length) {
202
- // remove highlight
203
- selection
204
- .transition('highlight')
205
- .duration(200)
206
- .style('opacity', 1)
207
- return
208
- }
209
-
210
- selection
211
- .each((d, i, n) => {
212
- if (ids.includes(d.id)) {
213
- d3.select(n[i])
214
- .style('opacity', 1)
215
- } else {
216
- d3.select(n[i])
217
- .style('opacity', fullChartParams.styles.unhighlightedOpacity)
218
- }
219
- })
220
- }
221
-
222
-
223
- export const createBaseRacingBars: BasePluginFn<BaseRacingBarsContext> = (pluginName: string, {
224
- selection,
225
- computedData$,
226
- visibleComputedRankingData$,
227
- // xyValueIndex$,
228
- // categoryLabels$,
229
- CategoryDataMap$,
230
- fullParams$,
231
- fullChartParams$,
232
- // layout$,
233
- // graphicTransform$,
234
- // graphicReverseScale$,
235
- highlight$,
236
- // computedRankingAmountList$,
237
- rankingItemHeight$,
238
- rankingScaleList$,
239
- containerPosition$,
240
- containerSize$,
241
- // layout$,
242
- xScale$,
243
- isCategorySeprate$,
244
- event$
245
- }) => {
246
-
247
- const destroy$ = new Subject()
248
-
249
- const clipPathID = getUniID(pluginName, 'clipPath-box')
250
- const rectClassName = getClassName(pluginName, 'rect')
251
- // const containerClassName = getClassName(pluginName, 'container')
252
-
253
- // const {
254
- // categorySelection$,
255
- // axesSelection$,
256
- // defsSelection$,
257
- // graphicGSelection$
258
- // } = multiValueSelectionsObservable({
259
- // selection,
260
- // pluginName,
261
- // clipPathID,
262
- // categoryLabels$: categoryLabels$,
263
- // containerPosition$: containerPosition$,
264
- // graphicTransform$: graphicTransform$
265
- // })
266
-
267
- // const clipPathSubscription = combineLatest({
268
- // defsSelection: defsSelection$,
269
- // layout: layout$,
270
- // }).pipe(
271
- // takeUntil(destroy$),
272
- // switchMap(async (d) => d),
273
- // ).subscribe(data => {
274
- // // 外層的遮罩
275
- // const clipPathData = [{
276
- // id: clipPathID,
277
- // width: data.layout.width,
278
- // height: data.layout.height
279
- // }]
280
- // renderClipPath({
281
- // defsSelection: data.defsSelection,
282
- // clipPathData,
283
- // })
284
- // })
285
-
286
- const containerSelection$ = multiValueContainerSelectionsObservable({
287
- selection,
288
- pluginName,
289
- clipPathID,
290
- computedData$,
291
- containerPosition$,
292
- isCategorySeprate$,
293
- }).pipe(
294
- takeUntil(destroy$),
295
- )
296
-
297
- // const containerSelection$ = combineLatest({
298
- // computedData: computedData$.pipe(
299
- // distinctUntilChanged((a, b) => {
300
- // // 只有當series的數量改變時,才重新計算
301
- // return a.length === b.length
302
- // }),
303
- // ),
304
- // isCategorySeprate: isCategorySeprate$
305
- // }).pipe(
306
- // takeUntil(destroy$),
307
- // switchMap(async (d) => d),
308
- // map(data => {
309
- // return data.isCategorySeprate
310
- // // category分開的時候顯示各別axis
311
- // ? data.computedData
312
- // // category合併的時候只顯示第一個axis
313
- // : [data.computedData[0]]
314
- // }),
315
- // map((computedData, i) => {
316
- // return selection
317
- // .selectAll<SVGGElement, ComputedDatumMultiValue[]>(`g.${containerClassName}`)
318
- // .data(computedData, d => d[0] ? d[0].categoryIndex : i)
319
- // .join('g')
320
- // .classed(containerClassName, true)
321
- // .attr('clip-path', `url(#${clipPathID})`)
322
- // })
323
- // )
324
-
325
- containerSize$.subscribe(data => {
326
- const defsSelection = selection.selectAll<SVGDefsElement, any>('defs')
327
- .data([clipPathID])
328
- .join('defs')
329
- const clipPathData = [{
330
- id: clipPathID,
331
- width: data.width,
332
- height: data.height
333
- }]
334
- renderClipPath({
335
- defsSelection: defsSelection,
336
- clipPathData,
337
- // textReverseTransform: data.textReverseTransform
338
- })
339
- })
340
-
341
- // combineLatest({
342
- // containerSelection: containerSelection$,
343
- // containerPosition: containerPosition$
344
- // }).pipe(
345
- // takeUntil(destroy$),
346
- // switchMap(async d => d)
347
- // ).subscribe(data => {
348
- // data.containerSelection
349
- // .attr('transform', (d, i) => {
350
- // const containerPosition = data.containerPosition[i] ?? data.containerPosition[0]
351
- // const translate = containerPosition.translate
352
- // const scale = containerPosition.scale
353
- // // return `translate(${translate[0]}, ${translate[1]}) scale(${scale[0]}, ${scale[1]})`
354
- // return `translate(${translate[0]}, ${translate[1]})`
355
- // })
356
- // // .attr('opacity', 0)
357
- // // .transition()
358
- // // .attr('opacity', 1)
359
- // })
360
-
361
- const barWidth$ = combineLatest({
362
- fullParams: fullParams$,
363
- rankingItemHeight: rankingItemHeight$,
364
- }).pipe(
365
- takeUntil(destroy$),
366
- switchMap(async d => d),
367
- map(data => {
368
- if (data.fullParams.bar.barWidth) {
369
- return data.fullParams.bar.barWidth
370
- } else {
371
- return data.rankingItemHeight - data.fullParams.bar.barPadding
372
- }
373
- }),
374
- distinctUntilChanged()
375
- )
376
-
377
- const transitionDuration$ = fullChartParams$.pipe(
378
- takeUntil(destroy$),
379
- map(d => d.transitionDuration),
380
- distinctUntilChanged()
381
- )
382
-
383
- const graphicGSelection$ = combineLatest({
384
- containerSelection: containerSelection$,
385
- visibleComputedRankingData: visibleComputedRankingData$,
386
- rankingScaleList: rankingScaleList$,
387
- transitionDuration: transitionDuration$,
388
- }).pipe(
389
- takeUntil(destroy$),
390
- switchMap(async (d) => d),
391
- map(data => {
392
-
393
- return renderGraphicG({
394
- containerSelection: data.containerSelection,
395
- visibleComputedRankingData: data.visibleComputedRankingData,
396
- rankingScaleList: data.rankingScaleList,
397
- transitionDuration: data.transitionDuration
398
- })
399
- })
400
- )
401
-
402
- const graphicSelection$ = combineLatest({
403
- graphicGSelection: graphicGSelection$,
404
- // xyValueIndex: xyValueIndex$,
405
- xScale: xScale$,
406
- barWidth: barWidth$,
407
- transitionDuration: transitionDuration$,
408
- fullParams: fullParams$,
409
- }).pipe(
410
- takeUntil(destroy$),
411
- switchMap(async (d) => d),
412
- map(data => {
413
-
414
- return renderRectBars({
415
- graphicGSelection: data.graphicGSelection,
416
- rectClassName,
417
- // xyValueIndex: data.xyValueIndex,
418
- xScale: data.xScale,
419
- fullParams: data.fullParams,
420
- barWidth: data.barWidth,
421
- transitionDuration: data.transitionDuration,
422
- })
423
- }),
424
- shareReplay(1)
425
- )
426
-
427
- const highlightTarget$ = fullChartParams$.pipe(
428
- takeUntil(destroy$),
429
- map(d => d.highlightTarget),
430
- distinctUntilChanged()
431
- )
432
-
433
- combineLatest({
434
- graphicSelection: graphicSelection$,
435
- computedData: computedData$,
436
- CategoryDataMap: CategoryDataMap$,
437
- highlightTarget: highlightTarget$
438
- }).pipe(
439
- takeUntil(destroy$),
440
- switchMap(async (d) => d),
441
- ).subscribe(data => {
442
-
443
- data.graphicSelection
444
- .on('mouseover', (event, datum) => {
445
- // event.stopPropagation()
446
- // console.log({
447
- // type: 'multiValue',
448
- // eventName: 'mouseover',
449
- // pluginName,
450
- // highlightTarget: data.highlightTarget,
451
- // datum,
452
- // category: data.CategoryDataMap.get(datum.categoryLabel)!,
453
- // categoryIndex: datum.categoryIndex,
454
- // categoryLabel: datum.categoryLabel,
455
- // data: data.computedData,
456
- // event,
457
- // })
458
-
459
- // 只顯示目前的值
460
- datum._visibleValue = [datum.value[datum.xValueIndex]]
461
-
462
- event$.next({
463
- type: 'multiValue',
464
- eventName: 'mouseover',
465
- pluginName,
466
- highlightTarget: data.highlightTarget,
467
- datum,
468
- category: data.CategoryDataMap.get(datum.categoryLabel)!,
469
- categoryIndex: datum.categoryIndex,
470
- categoryLabel: datum.categoryLabel,
471
- data: data.computedData,
472
- event,
473
- })
474
- })
475
- .on('mousemove', (event, datum) => {
476
- // event.stopPropagation()
477
-
478
- // 只顯示目前的值
479
- datum._visibleValue = [datum.value[datum.xValueIndex]]
480
-
481
- event$.next({
482
- type: 'multiValue',
483
- eventName: 'mousemove',
484
- pluginName,
485
- highlightTarget: data.highlightTarget,
486
- datum,
487
- category: data.CategoryDataMap.get(datum.categoryLabel)!,
488
- categoryIndex: datum.categoryIndex,
489
- categoryLabel: datum.categoryLabel,
490
- data: data.computedData,
491
- event,
492
- })
493
- })
494
- .on('mouseout', (event, datum) => {
495
- // event.stopPropagation()
496
-
497
- event$.next({
498
- type: 'multiValue',
499
- eventName: 'mouseout',
500
- pluginName,
501
- highlightTarget: data.highlightTarget,
502
- datum,
503
- category: data.CategoryDataMap.get(datum.categoryLabel)!,
504
- categoryIndex: datum.categoryIndex,
505
- categoryLabel: datum.categoryLabel,
506
- data: data.computedData,
507
- event,
508
- })
509
- })
510
- .on('click', (event, datum) => {
511
- // event.stopPropagation()
512
-
513
- // 只顯示目前的值
514
- datum._visibleValue = [datum.value[datum.xValueIndex]]
515
-
516
- event$.next({
517
- type: 'multiValue',
518
- eventName: 'click',
519
- pluginName,
520
- highlightTarget: data.highlightTarget,
521
- datum,
522
- category: data.CategoryDataMap.get(datum.categoryLabel)!,
523
- categoryIndex: datum.categoryIndex,
524
- categoryLabel: datum.categoryLabel,
525
- data: data.computedData,
526
- event,
527
- })
528
- })
529
-
530
- })
531
-
532
- combineLatest({
533
- graphicSelection: graphicSelection$,
534
- highlight: highlight$.pipe(
535
- map(data => data.map(d => d.id))
536
- ),
537
- fullChartParams: fullChartParams$
538
- }).pipe(
539
- takeUntil(destroy$),
540
- switchMap(async d => d)
541
- ).subscribe(data => {
542
- highlight({
543
- selection: data.graphicSelection,
544
- ids: data.highlight,
545
- fullChartParams: data.fullChartParams
546
- })
547
- })
548
-
549
- return () => {
550
- destroy$.next(undefined)
551
- }
1
+ import * as d3 from 'd3'
2
+ import {
3
+ combineLatest,
4
+ map,
5
+ switchMap,
6
+ takeUntil,
7
+ distinctUntilChanged,
8
+ shareReplay,
9
+ Observable,
10
+ Subject } from 'rxjs'
11
+ import type { BasePluginFn } from './types'
12
+ import type {
13
+ ComputedDatumMultiValue,
14
+ ComputedDataMultiValue,
15
+ // ComputedLayoutDataGrid,
16
+ DataFormatterMultiValue,
17
+ ContainerPositionScaled,
18
+ ContainerSize,
19
+ EventMultiValue,
20
+ ChartParams,
21
+ Layout,
22
+ TransformData } from '../../lib/core-types'
23
+ import type { BaseRacingBarsParams } from '../../lib/plugins-basic-types'
24
+ import { getD3TransitionEase } from '../utils/d3Utils'
25
+ import { getClassName, getUniID } from '../utils/orbchartsUtils'
26
+ import { multiValueContainerSelectionsObservable } from '../multiValue/multiValueObservables'
27
+
28
+ // export interface BaseBarsParams {
29
+ // // barType: BarType
30
+ // barWidth: number
31
+ // barPadding: number
32
+ // barGroupPadding: number // 群組和群組間的間隔
33
+ // barRadius: number | boolean
34
+ // }
35
+
36
+ interface BaseRacingBarsContext {
37
+ selection: d3.Selection<any, unknown, any, unknown>
38
+ computedData$: Observable<ComputedDataMultiValue>
39
+ visibleComputedRankingData$: Observable<ComputedDatumMultiValue[][]>
40
+ CategoryDataMap$: Observable<Map<string, ComputedDatumMultiValue[]>>
41
+ fullParams$: Observable<BaseRacingBarsParams>
42
+ fullChartParams$: Observable<ChartParams>
43
+ fullDataFormatter$: Observable<DataFormatterMultiValue>
44
+ // xyValueIndex$: Observable<[number, number]>
45
+ highlight$: Observable<ComputedDatumMultiValue[]>
46
+ rankingItemHeight$: Observable<number>
47
+ rankingScaleList$: Observable<d3.ScalePoint<string>[]>
48
+ containerPosition$: Observable<ContainerPositionScaled[]>
49
+ containerSize$: Observable<ContainerSize>
50
+ xScale$: Observable<(n: number) => number>
51
+ isCategorySeprate$: Observable<boolean>
52
+ event$: Subject<EventMultiValue>
53
+ }
54
+
55
+ interface RenderGraphicGParams {
56
+ containerSelection: d3.Selection<SVGGElement, ComputedDatumMultiValue[], any, any>
57
+ visibleComputedRankingData: ComputedDatumMultiValue[][]
58
+ rankingScaleList: d3.ScalePoint<string>[]
59
+ transitionDuration: number
60
+ }
61
+
62
+ interface RenderBarParams {
63
+ graphicGSelection: d3.Selection<SVGGElement, ComputedDatumMultiValue, any, any>
64
+ rectClassName: string
65
+ // xyValueIndex: [number, number]
66
+ xScale: (n: number) => number
67
+ fullParams: BaseRacingBarsParams
68
+ barWidth: number
69
+ transitionDuration: number
70
+ }
71
+
72
+ type ClipPathDatum = {
73
+ id: string;
74
+ // x: number;
75
+ // y: number;
76
+ width: number;
77
+ height: number;
78
+ }
79
+
80
+
81
+ function renderGraphicG ({ containerSelection, visibleComputedRankingData, rankingScaleList, transitionDuration }: RenderGraphicGParams) {
82
+ containerSelection
83
+ .each((_, categoryIndex, g) => {
84
+ const container = d3.select(g[categoryIndex])
85
+ const graphicG = container.selectAll<SVGGElement, ComputedDatumMultiValue>(`g`)
86
+ .data(visibleComputedRankingData[categoryIndex] ?? [], d => d.id)
87
+ .join(
88
+ enter => {
89
+ return enter
90
+ .append('g')
91
+ .attr('cursor', 'pointer')
92
+ .attr('transform', d => {
93
+ return `translate(0, ${rankingScaleList[categoryIndex] && rankingScaleList[categoryIndex](d.label)})`
94
+ })
95
+ },
96
+ update => {
97
+ return update
98
+ .transition()
99
+ .duration(transitionDuration)
100
+ .ease(d3.easeLinear)
101
+ .attr('transform', d => {
102
+ return `translate(0, ${rankingScaleList[categoryIndex] && rankingScaleList[categoryIndex](d.label)})`
103
+ })
104
+ },
105
+ exit => exit.remove()
106
+ )
107
+ })
108
+
109
+ const graphicBarSelection: d3.Selection<SVGRectElement, ComputedDatumMultiValue, SVGGElement, unknown> = containerSelection.selectAll(`g`)
110
+
111
+ return graphicBarSelection
112
+ }
113
+
114
+ function renderRectBars ({ graphicGSelection, rectClassName, xScale, fullParams, barWidth, transitionDuration }: RenderBarParams) {
115
+
116
+ graphicGSelection
117
+ .each((datum, i, g) => {
118
+ // const containerIndex = isCategorySeprate ? datum.categoryIndex : 0
119
+ // const barWidth = barWidthList[containerIndex]
120
+ const barHalfWidth = barWidth / 2
121
+ const radius = fullParams.bar.barRadius === true ? barHalfWidth
122
+ : fullParams.bar.barRadius === false ? 0
123
+ : typeof fullParams.bar.barRadius == 'number' ? fullParams.bar.barRadius
124
+ : 0
125
+
126
+ const gSelection = d3.select(g[i])
127
+ gSelection.selectAll<SVGRectElement, ComputedDatumMultiValue>(`rect.${rectClassName}`)
128
+ .data([datum], d => d.id)
129
+ .join(
130
+ enter => {
131
+ return enter
132
+ .append('rect')
133
+ .classed(rectClassName, true)
134
+ .attr('cursor', 'pointer')
135
+ // .attr('width', d => 1)
136
+ .attr('width', d => xScale(d.value[d.xValueIndex]) ?? 1)
137
+ .attr('height', barWidth)
138
+ },
139
+ update => {
140
+ return update
141
+ .transition()
142
+ .duration(transitionDuration)
143
+ .ease(d3.easeLinear)
144
+ .attr('width', d => xScale(d.value[d.xValueIndex]) ?? 1)
145
+ .attr('height', barWidth)
146
+ },
147
+ exit => exit.remove()
148
+ )
149
+ .attr('transform', `translate(0, ${-barHalfWidth})`)
150
+ .attr('fill', d => d.color)
151
+ .attr('rx', radius)
152
+ .attr('ry', radius)
153
+ })
154
+
155
+ const graphicBarSelection: d3.Selection<SVGRectElement, ComputedDatumMultiValue, SVGGElement, unknown> = graphicGSelection.selectAll(`rect.${rectClassName}`)
156
+
157
+ return graphicBarSelection
158
+ }
159
+
160
+ function renderClipPath ({ defsSelection, clipPathData }: {
161
+ defsSelection: d3.Selection<SVGDefsElement, any, any, any>
162
+ clipPathData: ClipPathDatum[]
163
+ }) {
164
+ const clipPath = defsSelection
165
+ .selectAll<SVGClipPathElement, Layout>('clipPath')
166
+ .data(clipPathData)
167
+ .join(
168
+ enter => {
169
+ return enter
170
+ .append('clipPath')
171
+ },
172
+ update => update,
173
+ exit => exit.remove()
174
+ )
175
+ .attr('id', d => d.id)
176
+ .each((d, i, g) => {
177
+ const rect = d3.select(g[i])
178
+ .selectAll<SVGRectElement, typeof d>('rect')
179
+ .data([d])
180
+ .join(
181
+ enter => {
182
+ return enter
183
+ .append('rect')
184
+ },
185
+ update => update,
186
+ exit => exit.remove()
187
+ )
188
+ .attr('x', 0)
189
+ .attr('y', 0)
190
+ .attr('width', _d => _d.width)
191
+ .attr('height', _d => _d.height)
192
+ })
193
+ }
194
+
195
+ function highlight ({ selection, ids, fullChartParams }: {
196
+ selection: d3.Selection<any, ComputedDatumMultiValue, any, any>
197
+ ids: string[]
198
+ fullChartParams: ChartParams
199
+ }) {
200
+ selection.interrupt('highlight')
201
+
202
+ if (!ids.length) {
203
+ // remove highlight
204
+ selection
205
+ .transition('highlight')
206
+ .duration(200)
207
+ .style('opacity', 1)
208
+ return
209
+ }
210
+
211
+ selection
212
+ .each((d, i, n) => {
213
+ if (ids.includes(d.id)) {
214
+ d3.select(n[i])
215
+ .style('opacity', 1)
216
+ } else {
217
+ d3.select(n[i])
218
+ .style('opacity', fullChartParams.styles.unhighlightedOpacity)
219
+ }
220
+ })
221
+ }
222
+
223
+
224
+ export const createBaseRacingBars: BasePluginFn<BaseRacingBarsContext> = (pluginName: string, {
225
+ selection,
226
+ computedData$,
227
+ visibleComputedRankingData$,
228
+ // xyValueIndex$,
229
+ // categoryLabels$,
230
+ CategoryDataMap$,
231
+ fullParams$,
232
+ fullDataFormatter$,
233
+ fullChartParams$,
234
+ // layout$,
235
+ // graphicTransform$,
236
+ // graphicReverseScale$,
237
+ highlight$,
238
+ // computedRankingAmountList$,
239
+ rankingItemHeight$,
240
+ rankingScaleList$,
241
+ containerPosition$,
242
+ containerSize$,
243
+ // layout$,
244
+ xScale$,
245
+ isCategorySeprate$,
246
+ event$
247
+ }) => {
248
+
249
+ const destroy$ = new Subject()
250
+
251
+ const clipPathID = getUniID(pluginName, 'clipPath-box')
252
+ const rectClassName = getClassName(pluginName, 'rect')
253
+ // const containerClassName = getClassName(pluginName, 'container')
254
+
255
+ // const {
256
+ // categorySelection$,
257
+ // axesSelection$,
258
+ // defsSelection$,
259
+ // graphicGSelection$
260
+ // } = multiValueSelectionsObservable({
261
+ // selection,
262
+ // pluginName,
263
+ // clipPathID,
264
+ // categoryLabels$: categoryLabels$,
265
+ // containerPosition$: containerPosition$,
266
+ // graphicTransform$: graphicTransform$
267
+ // })
268
+
269
+ // const clipPathSubscription = combineLatest({
270
+ // defsSelection: defsSelection$,
271
+ // layout: layout$,
272
+ // }).pipe(
273
+ // takeUntil(destroy$),
274
+ // switchMap(async (d) => d),
275
+ // ).subscribe(data => {
276
+ // // 外層的遮罩
277
+ // const clipPathData = [{
278
+ // id: clipPathID,
279
+ // width: data.layout.width,
280
+ // height: data.layout.height
281
+ // }]
282
+ // renderClipPath({
283
+ // defsSelection: data.defsSelection,
284
+ // clipPathData,
285
+ // })
286
+ // })
287
+
288
+ const containerSelection$ = multiValueContainerSelectionsObservable({
289
+ selection,
290
+ pluginName,
291
+ clipPathID,
292
+ computedData$,
293
+ containerPosition$,
294
+ isCategorySeprate$,
295
+ }).pipe(
296
+ takeUntil(destroy$),
297
+ )
298
+
299
+ // const containerSelection$ = combineLatest({
300
+ // computedData: computedData$.pipe(
301
+ // distinctUntilChanged((a, b) => {
302
+ // // 只有當series的數量改變時,才重新計算
303
+ // return a.length === b.length
304
+ // }),
305
+ // ),
306
+ // isCategorySeprate: isCategorySeprate$
307
+ // }).pipe(
308
+ // takeUntil(destroy$),
309
+ // switchMap(async (d) => d),
310
+ // map(data => {
311
+ // return data.isCategorySeprate
312
+ // // category分開的時候顯示各別axis
313
+ // ? data.computedData
314
+ // // category合併的時候只顯示第一個axis
315
+ // : [data.computedData[0]]
316
+ // }),
317
+ // map((computedData, i) => {
318
+ // return selection
319
+ // .selectAll<SVGGElement, ComputedDatumMultiValue[]>(`g.${containerClassName}`)
320
+ // .data(computedData, d => d[0] ? d[0].categoryIndex : i)
321
+ // .join('g')
322
+ // .classed(containerClassName, true)
323
+ // .attr('clip-path', `url(#${clipPathID})`)
324
+ // })
325
+ // )
326
+
327
+ containerSize$.subscribe(data => {
328
+ const defsSelection = selection.selectAll<SVGDefsElement, any>('defs')
329
+ .data([clipPathID])
330
+ .join('defs')
331
+ const clipPathData = [{
332
+ id: clipPathID,
333
+ width: data.width,
334
+ height: data.height
335
+ }]
336
+ renderClipPath({
337
+ defsSelection: defsSelection,
338
+ clipPathData,
339
+ // textReverseTransform: data.textReverseTransform
340
+ })
341
+ })
342
+
343
+ // combineLatest({
344
+ // containerSelection: containerSelection$,
345
+ // containerPosition: containerPosition$
346
+ // }).pipe(
347
+ // takeUntil(destroy$),
348
+ // switchMap(async d => d)
349
+ // ).subscribe(data => {
350
+ // data.containerSelection
351
+ // .attr('transform', (d, i) => {
352
+ // const containerPosition = data.containerPosition[i] ?? data.containerPosition[0]
353
+ // const translate = containerPosition.translate
354
+ // const scale = containerPosition.scale
355
+ // // return `translate(${translate[0]}, ${translate[1]}) scale(${scale[0]}, ${scale[1]})`
356
+ // return `translate(${translate[0]}, ${translate[1]})`
357
+ // })
358
+ // // .attr('opacity', 0)
359
+ // // .transition()
360
+ // // .attr('opacity', 1)
361
+ // })
362
+
363
+ const barWidth$ = combineLatest({
364
+ fullParams: fullParams$,
365
+ rankingItemHeight: rankingItemHeight$,
366
+ }).pipe(
367
+ takeUntil(destroy$),
368
+ switchMap(async d => d),
369
+ map(data => {
370
+ if (data.fullParams.bar.barWidth) {
371
+ return data.fullParams.bar.barWidth
372
+ } else {
373
+ return data.rankingItemHeight - data.fullParams.bar.barPadding
374
+ }
375
+ }),
376
+ distinctUntilChanged()
377
+ )
378
+
379
+ const transitionDuration$ = fullChartParams$.pipe(
380
+ takeUntil(destroy$),
381
+ map(d => d.transitionDuration),
382
+ distinctUntilChanged()
383
+ )
384
+
385
+ const graphicGSelection$ = combineLatest({
386
+ containerSelection: containerSelection$,
387
+ visibleComputedRankingData: visibleComputedRankingData$,
388
+ rankingScaleList: rankingScaleList$,
389
+ transitionDuration: transitionDuration$,
390
+ }).pipe(
391
+ takeUntil(destroy$),
392
+ switchMap(async (d) => d),
393
+ map(data => {
394
+
395
+ return renderGraphicG({
396
+ containerSelection: data.containerSelection,
397
+ visibleComputedRankingData: data.visibleComputedRankingData,
398
+ rankingScaleList: data.rankingScaleList,
399
+ transitionDuration: data.transitionDuration
400
+ })
401
+ })
402
+ )
403
+
404
+ const graphicSelection$ = combineLatest({
405
+ graphicGSelection: graphicGSelection$,
406
+ // xyValueIndex: xyValueIndex$,
407
+ xScale: xScale$,
408
+ barWidth: barWidth$,
409
+ transitionDuration: transitionDuration$,
410
+ fullParams: fullParams$,
411
+ }).pipe(
412
+ takeUntil(destroy$),
413
+ switchMap(async (d) => d),
414
+ map(data => {
415
+
416
+ return renderRectBars({
417
+ graphicGSelection: data.graphicGSelection,
418
+ rectClassName,
419
+ // xyValueIndex: data.xyValueIndex,
420
+ xScale: data.xScale,
421
+ fullParams: data.fullParams,
422
+ barWidth: data.barWidth,
423
+ transitionDuration: data.transitionDuration,
424
+ })
425
+ }),
426
+ shareReplay(1)
427
+ )
428
+
429
+ const highlightTarget$ = fullChartParams$.pipe(
430
+ takeUntil(destroy$),
431
+ map(d => d.highlightTarget),
432
+ distinctUntilChanged()
433
+ )
434
+
435
+ combineLatest({
436
+ graphicSelection: graphicSelection$,
437
+ computedData: computedData$,
438
+ CategoryDataMap: CategoryDataMap$,
439
+ highlightTarget: highlightTarget$,
440
+ fullDataFormatter: fullDataFormatter$
441
+ }).pipe(
442
+ takeUntil(destroy$),
443
+ switchMap(async (d) => d),
444
+ ).subscribe(data => {
445
+
446
+ data.graphicSelection
447
+ .on('mouseover', (event, datum) => {
448
+ // event.stopPropagation()
449
+ // console.log({
450
+ // type: 'multiValue',
451
+ // eventName: 'mouseover',
452
+ // pluginName,
453
+ // highlightTarget: data.highlightTarget,
454
+ // datum,
455
+ // category: data.CategoryDataMap.get(datum.categoryLabel)!,
456
+ // categoryIndex: datum.categoryIndex,
457
+ // categoryLabel: datum.categoryLabel,
458
+ // data: data.computedData,
459
+ // event,
460
+ // })
461
+
462
+ // 只顯示目前的值
463
+ // datum._visibleValue = [datum.value[datum.xValueIndex]]
464
+
465
+ event$.next({
466
+ type: 'multiValue',
467
+ eventName: 'mouseover',
468
+ pluginName,
469
+ highlightTarget: data.highlightTarget,
470
+ valueDetail: [
471
+ {
472
+ value: datum.value[datum.xValueIndex],
473
+ valueIndex: datum.xValueIndex,
474
+ valueLabel: data.fullDataFormatter.valueLabels[datum.xValueIndex] ?? String(datum.xValueIndex)
475
+ },
476
+ ],
477
+ datum,
478
+ category: data.CategoryDataMap.get(datum.categoryLabel)!,
479
+ categoryIndex: datum.categoryIndex,
480
+ categoryLabel: datum.categoryLabel,
481
+ data: data.computedData,
482
+ event,
483
+ })
484
+ })
485
+ .on('mousemove', (event, datum) => {
486
+ // event.stopPropagation()
487
+
488
+ // 只顯示目前的值
489
+ // datum._visibleValue = [datum.value[datum.xValueIndex]]
490
+
491
+ event$.next({
492
+ type: 'multiValue',
493
+ eventName: 'mousemove',
494
+ pluginName,
495
+ highlightTarget: data.highlightTarget,
496
+ valueDetail: [
497
+ {
498
+ value: datum.value[datum.xValueIndex],
499
+ valueIndex: datum.xValueIndex,
500
+ valueLabel: data.fullDataFormatter.valueLabels[datum.xValueIndex] ?? String(datum.xValueIndex)
501
+ },
502
+ ],
503
+ datum,
504
+ category: data.CategoryDataMap.get(datum.categoryLabel)!,
505
+ categoryIndex: datum.categoryIndex,
506
+ categoryLabel: datum.categoryLabel,
507
+ data: data.computedData,
508
+ event,
509
+ })
510
+ })
511
+ .on('mouseout', (event, datum) => {
512
+ // event.stopPropagation()
513
+
514
+ event$.next({
515
+ type: 'multiValue',
516
+ eventName: 'mouseout',
517
+ pluginName,
518
+ highlightTarget: data.highlightTarget,
519
+ valueDetail: [
520
+ {
521
+ value: datum.value[datum.xValueIndex],
522
+ valueIndex: datum.xValueIndex,
523
+ valueLabel: data.fullDataFormatter.valueLabels[datum.xValueIndex] ?? String(datum.xValueIndex)
524
+ },
525
+ ],
526
+ datum,
527
+ category: data.CategoryDataMap.get(datum.categoryLabel)!,
528
+ categoryIndex: datum.categoryIndex,
529
+ categoryLabel: datum.categoryLabel,
530
+ data: data.computedData,
531
+ event,
532
+ })
533
+ })
534
+ .on('click', (event, datum) => {
535
+ // event.stopPropagation()
536
+
537
+ // 只顯示目前的值
538
+ // datum._visibleValue = [datum.value[datum.xValueIndex]]
539
+
540
+ event$.next({
541
+ type: 'multiValue',
542
+ eventName: 'click',
543
+ pluginName,
544
+ highlightTarget: data.highlightTarget,
545
+ valueDetail: [
546
+ {
547
+ value: datum.value[datum.xValueIndex],
548
+ valueIndex: datum.xValueIndex,
549
+ valueLabel: data.fullDataFormatter.valueLabels[datum.xValueIndex] ?? String(datum.xValueIndex)
550
+ },
551
+ ],
552
+ datum,
553
+ category: data.CategoryDataMap.get(datum.categoryLabel)!,
554
+ categoryIndex: datum.categoryIndex,
555
+ categoryLabel: datum.categoryLabel,
556
+ data: data.computedData,
557
+ event,
558
+ })
559
+ })
560
+
561
+ })
562
+
563
+ combineLatest({
564
+ graphicSelection: graphicSelection$,
565
+ highlight: highlight$.pipe(
566
+ map(data => data.map(d => d.id))
567
+ ),
568
+ fullChartParams: fullChartParams$
569
+ }).pipe(
570
+ takeUntil(destroy$),
571
+ switchMap(async d => d)
572
+ ).subscribe(data => {
573
+ highlight({
574
+ selection: data.graphicSelection,
575
+ ids: data.highlight,
576
+ fullChartParams: data.fullChartParams
577
+ })
578
+ })
579
+
580
+ return () => {
581
+ destroy$.next(undefined)
582
+ }
552
583
  }