@orbcharts/plugins-basic 3.0.2 → 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 -48
  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,624 +1,624 @@
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 { DefinePluginConfig } from '../../../lib/core-types'
12
- import type {
13
- ComputedDataSeries,
14
- ComputedDatumSeries,
15
- ContainerPosition,
16
- ChartParams,
17
- EventSeries,
18
- Layout } from '../../../lib/core-types'
19
- import type { PieDatum } from '../seriesUtils'
20
- import type { PieParams } from '../../../lib/plugins-basic-types'
21
- import {
22
- defineSeriesPlugin } from '../../../lib/core'
23
- import { DEFAULT_PIE_PARAMS } from '../defaults'
24
- import { makePieData } from '../seriesUtils'
25
- import { getD3TransitionEase, makeD3Arc } from '../../utils/d3Utils'
26
- import { getDatumColor, getClassName } from '../../utils/orbchartsUtils'
27
- import { seriesCenterSelectionObservable } from '../seriesObservables'
28
- import { LAYER_INDEX_OF_GRAPHIC } from '../../const'
29
-
30
- const pluginName = 'Pie'
31
-
32
- const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_PIE_PARAMS> = {
33
- name: pluginName,
34
- defaultParams: DEFAULT_PIE_PARAMS,
35
- layerIndex: LAYER_INDEX_OF_GRAPHIC,
36
- validator: (params, { validateColumns }) => {
37
- const result = validateColumns(params, {
38
- outerRadius: {
39
- toBeTypes: ['number'],
40
- },
41
- innerRadius: {
42
- toBeTypes: ['number'],
43
- },
44
- outerRadiusWhileHighlight: {
45
- toBeTypes: ['number'],
46
- },
47
- startAngle: {
48
- toBeTypes: ['number'],
49
- },
50
- endAngle: {
51
- toBeTypes: ['number'],
52
- },
53
- padAngle: {
54
- toBeTypes: ['number'],
55
- },
56
- strokeColorType: {
57
- toBeTypes: ['string'],
58
- },
59
- strokeWidth: {
60
- toBeTypes: ['number'],
61
- },
62
- cornerRadius: {
63
- toBeTypes: ['number'],
64
- }
65
- })
66
-
67
- return result
68
- }
69
- }
70
-
71
- function makeTweenPieRenderDataFn ({ enter, exit, data, lastTweenData, fullParams }: {
72
- enter: d3.Selection<d3.EnterElement, PieDatum, any, any>
73
- exit: d3.Selection<SVGPathElement, unknown, any, any>
74
- data: PieDatum[]
75
- lastTweenData: PieDatum[]
76
- fullParams: PieParams
77
- }): (t: number) => PieDatum[] {
78
- // 無更新資料項目則只計算資料變化 (新資料 * t + 舊資料 * (1 - t))
79
- if (!enter.size() && !exit.size()) {
80
- // console.log('case1')
81
- return (t: number) => {
82
- const tweenData: PieDatum[] = data.map((_d, _i) => {
83
- const lastDatum = lastTweenData[_i] ?? {
84
- startAngle: 0,
85
- endAngle: 0,
86
- value: 0
87
- }
88
- return {
89
- ..._d,
90
- startAngle: (_d.startAngle * t) + (lastDatum.startAngle * (1 - t)),
91
- endAngle: (_d.endAngle * t) + (lastDatum.endAngle * (1 - t)),
92
- value: (_d.value * t) + (lastDatum.value * (1 - t))
93
- }
94
- })
95
-
96
- return makePieRenderData(
97
- tweenData,
98
- fullParams.startAngle!,
99
- fullParams.endAngle!,
100
- 1
101
- )
102
- }
103
- // 有更新資料則重新繪圖
104
- } else {
105
- // console.log('case2')
106
- return (t: number) => {
107
- return makePieRenderData(
108
- data,
109
- fullParams.startAngle!,
110
- fullParams.endAngle!,
111
- t
112
- )
113
- }
114
- }
115
- }
116
-
117
- function makePieRenderData (data: PieDatum[], startAngle: number, endAngle: number, t: number): PieDatum[] {
118
- return data.map((d, i) => {
119
- const _startAngle = startAngle + (d.startAngle - startAngle) * t
120
- const _endAngle = _startAngle + (d.endAngle - d.startAngle) * t
121
- return {
122
- ...d,
123
- startAngle: _startAngle,
124
- endAngle: _endAngle
125
- }
126
- })
127
- }
128
-
129
- function renderPie ({ selection, data, arc, pathClassName, fullParams, fullChartParams }: {
130
- selection: d3.Selection<SVGGElement, unknown, any, unknown>
131
- data: PieDatum[]
132
- arc: d3.Arc<any, d3.DefaultArcObject>
133
- pathClassName: string
134
- fullParams: PieParams
135
- fullChartParams: ChartParams
136
- }): d3.Selection<SVGPathElement, PieDatum, any, any> {
137
- // console.log('data', data)
138
- const pathSelection: d3.Selection<SVGPathElement, PieDatum, any, any> = selection
139
- .selectAll<SVGPathElement, PieDatum>('path')
140
- .data(data, d => d.id)
141
- .join('path')
142
- .classed(pathClassName, true)
143
- .style('cursor', 'pointer')
144
- .attr('fill', (d, i) => d.data.color)
145
- .attr('stroke', (d, i) => getDatumColor({ datum: d.data, colorType: fullParams.strokeColorType, fullChartParams }))
146
- .attr('stroke-width', fullParams.strokeWidth)
147
- .attr('d', (d, i) => {
148
- return arc!(d as any)
149
- })
150
-
151
- return pathSelection
152
- }
153
-
154
- function highlight ({ pathSelection, ids, fullChartParams, arc, arcHighlight }: {
155
- pathSelection: d3.Selection<SVGPathElement, PieDatum, any, any>
156
- ids: string[]
157
- fullChartParams: ChartParams
158
- arc: d3.Arc<any, d3.DefaultArcObject>
159
- arcHighlight: d3.Arc<any, d3.DefaultArcObject>
160
- }) {
161
- pathSelection.interrupt('highlight')
162
-
163
- if (!ids.length) {
164
- // 取消放大
165
- pathSelection
166
- .transition('highlight')
167
- .style('opacity', 1)
168
- .attr('d', (d) => {
169
- return arc!(d as any)
170
- })
171
- return
172
- }
173
-
174
- pathSelection.each((d, i, n) => {
175
- const segment = d3.select(n[i])
176
-
177
- if (ids.includes(d.data.id)) {
178
- segment
179
- .style('opacity', 1)
180
- .transition('highlight')
181
- .ease(d3.easeElastic)
182
- .duration(500)
183
- .attr('d', (d: any) => {
184
- return arcHighlight!(d)
185
- })
186
- // .on('interrupt', () => {
187
- // // this.pathSelection!.select('path').attr('d', (d) => {
188
- // // return this.arc!(d as any)
189
- // // })
190
- // this.initHighlight()
191
- // })
192
- } else {
193
- // 取消放大
194
- segment
195
- .style('opacity', fullChartParams.styles.unhighlightedOpacity)
196
- .transition('highlight')
197
- .attr('d', (d) => {
198
- return arc!(d as any)
199
- })
200
- }
201
- })
202
- }
203
-
204
- // 各別的pie
205
- function createEachPie (pluginName: string, context: {
206
- containerSelection: d3.Selection<SVGGElement, any, any, unknown>
207
- computedData$: Observable<ComputedDatumSeries[][]>
208
- containerVisibleComputedSortedData$: Observable<ComputedDatumSeries[]>
209
- SeriesDataMap$: Observable<Map<string, ComputedDatumSeries[]>>
210
- fullParams$: Observable<PieParams>
211
- fullChartParams$: Observable<ChartParams>
212
- seriesHighlight$: Observable<ComputedDatumSeries[]>
213
- seriesContainerPosition$: Observable<ContainerPosition>
214
- event$: Subject<EventSeries>
215
- }) {
216
- const destroy$ = new Subject()
217
-
218
- const pathClassName = getClassName(pluginName, 'path')
219
-
220
- let lastTweenData: PieDatum[] = [] // 紀錄補間動畫前次的資料
221
- let tweenData: PieDatum[] = [] // 紀錄補間動畫用的資料
222
- // let originHighlight: Highlight | null = null
223
-
224
- // context.layout$
225
- // .pipe(
226
- // first()
227
- // )
228
- // .subscribe(size => {
229
- // selection
230
- // .attr('transform', `translate(${size.width / 2}, ${size.height / 2})`)
231
- // context.layout$
232
- // .pipe(
233
- // takeUntil(destroy$)
234
- // )
235
- // .subscribe(size => {
236
- // selection
237
- // .transition()
238
- // .attr('transform', `translate(${size.width / 2}, ${size.height / 2})`)
239
- // })
240
- // })
241
-
242
-
243
- const shorterSideWith$ = context.seriesContainerPosition$.pipe(
244
- takeUntil(destroy$),
245
- map(d => d.width < d.height ? d.width : d.height),
246
- distinctUntilChanged()
247
- )
248
-
249
- const pieData$: Observable<PieDatum[]> = new Observable(subscriber => {
250
- combineLatest({
251
- containerVisibleComputedSortedData: context.containerVisibleComputedSortedData$,
252
- fullParams: context.fullParams$,
253
- }).pipe(
254
- takeUntil(destroy$),
255
- switchMap(async (d) => d),
256
- ).subscribe(data => {
257
- // console.log('pieData', data)
258
- const pieData: PieDatum[] = makePieData({
259
- data: data.containerVisibleComputedSortedData,
260
- startAngle: data.fullParams.startAngle,
261
- endAngle: data.fullParams.endAngle
262
- })
263
- // console.log('pieData', pieData)
264
- subscriber.next(pieData)
265
- })
266
- })
267
-
268
- // const SeriesDataMap$ = context.computedData$.pipe(
269
- // takeUntil(destroy$),
270
- // map(d => makeSeriesDataMap(d))
271
- // )
272
-
273
- const arc$: Observable<d3.Arc<any, d3.DefaultArcObject>> = new Observable(subscriber => {
274
- combineLatest({
275
- shorterSideWith: shorterSideWith$,
276
- fullParams: context.fullParams$,
277
- }).pipe(
278
- takeUntil(destroy$),
279
- switchMap(async (d) => d),
280
- ).subscribe(data => {
281
- const arc = makeD3Arc({
282
- axisWidth: data.shorterSideWith,
283
- innerRadius: data.fullParams.innerRadius,
284
- outerRadius: data.fullParams.outerRadius,
285
- padAngle: data.fullParams.padAngle,
286
- cornerRadius: data.fullParams.cornerRadius
287
- })
288
- subscriber.next(arc)
289
- })
290
- })
291
-
292
- const arcHighlight$: Observable<d3.Arc<any, d3.DefaultArcObject>> = new Observable(subscriber => {
293
- combineLatest({
294
- shorterSideWith: shorterSideWith$,
295
- fullParams: context.fullParams$,
296
- }).pipe(
297
- takeUntil(destroy$),
298
- switchMap(async (d) => d),
299
- ).subscribe(data => {
300
- const arcHighlight = makeD3Arc({
301
- axisWidth: data.shorterSideWith,
302
- innerRadius: data.fullParams.innerRadius,
303
- outerRadius: data.fullParams.outerRadiusWhileHighlight, // 外半徑變化
304
- padAngle: data.fullParams.padAngle,
305
- cornerRadius: data.fullParams.cornerRadius
306
- })
307
- subscriber.next(arcHighlight)
308
- })
309
- })
310
-
311
- const highlightTarget$ = context.fullChartParams$.pipe(
312
- takeUntil(destroy$),
313
- map(d => d.highlightTarget),
314
- distinctUntilChanged()
315
- )
316
-
317
- const pathSelection$ = new Observable<d3.Selection<SVGPathElement, PieDatum, any, any>>(subscriber => {
318
- combineLatest({
319
- pieData: pieData$,
320
- arc: arc$,
321
- computedData: context.computedData$,
322
- fullParams: context.fullParams$,
323
- fullChartParams: context.fullChartParams$,
324
- highlightTarget: highlightTarget$
325
- }).pipe(
326
- takeUntil(destroy$),
327
- switchMap(async d => d)
328
- ).subscribe(data => {
329
- context.containerSelection.interrupt('graphicMove')
330
- // console.log('graphic', data)
331
- const update: d3.Selection<SVGPathElement, PieDatum, any, any> = context.containerSelection
332
- .selectAll<SVGPathElement, PieDatum>('path')
333
- .data(data.pieData, d => d.id)
334
- const enter = update.enter()
335
- const exit = update.exit()
336
-
337
- const makeTweenPieRenderData = makeTweenPieRenderDataFn({
338
- enter,
339
- exit,
340
- data: data.pieData,
341
- lastTweenData,
342
- fullParams: data.fullParams
343
- })
344
-
345
- // -- 使用補間動畫 --
346
- context.containerSelection
347
- .transition('graphicMove')
348
- .duration(data.fullChartParams.transitionDuration)
349
- // .ease(getD3TransitionEase(data.fullChartParams.transitionEase))
350
- .tween('move', (self, t) => {
351
- return (t) => {
352
- tweenData = makeTweenPieRenderData(t)
353
-
354
- const pathSelection = renderPie({
355
- selection: context.containerSelection,
356
- data: tweenData,
357
- arc: data.arc,
358
- pathClassName,
359
- fullParams: data.fullParams,
360
- fullChartParams: data.fullChartParams,
361
- })
362
-
363
- // @Q@ 想盡量減清效能負擔所以取消掉
364
- // context.event$.next({
365
- // type: 'series',
366
- // pluginName,
367
- // eventName: 'transitionMove',
368
- // event: undefined,
369
- // highlightTarget: data.highlightTarget,
370
- // datum: null,
371
- // series: [],
372
- // seriesIndex: -1,
373
- // seriesLabel: '',
374
- // data: data.computedData
375
- // })
376
-
377
- // const callbackData = makeEnterDurationCallbackData(data.computedData, )
378
- // enterDurationCallback(callbackData, t)
379
- }
380
- })
381
- .on('end', (self, t) => {
382
- tweenData = makePieRenderData(
383
- data.pieData,
384
- data.fullParams.startAngle,
385
- data.fullParams.endAngle,
386
- 1
387
- )
388
- // console.log('tweenData', tweenData)
389
- const pathSelection = renderPie({
390
- selection: context.containerSelection,
391
- data: tweenData,
392
- arc: data.arc,
393
- pathClassName,
394
- fullParams: data.fullParams,
395
- fullChartParams: data.fullChartParams,
396
- })
397
-
398
- // if (data.fullParams.highlightTarget && data.fullParams.highlightTarget != 'none') {
399
- // if (data.fullChartParams.highlightTarget && data.fullChartParams.highlightTarget != 'none') {
400
- // pathSelection!.style('cursor', 'pointer')
401
- // }
402
-
403
- subscriber.next(pathSelection)
404
-
405
- // pathSelection && setPathEvent({
406
- // pathSelection,
407
- // pluginName: name,
408
- // data: data.computedData,
409
- // fullChartParams: data.fullChartParams,
410
- // arc: data.arc,
411
- // arcHighlight: data.arcHighlight,
412
- // SeriesDataMap: data.SeriesDataMap,
413
- // event$: store.event$
414
- // })
415
-
416
- // 渲染完後紀錄為前次的資料
417
- lastTweenData = Object.assign([], data.pieData)
418
-
419
- context.event$.next({
420
- type: 'series',
421
- pluginName,
422
- eventName: 'transitionEnd',
423
- event: undefined,
424
- highlightTarget: data.highlightTarget,
425
- datum: null,
426
- series: [],
427
- seriesIndex: -1,
428
- seriesLabel: '',
429
- data: data.computedData
430
- })
431
-
432
-
433
- })
434
-
435
- // -- 更新資料 --
436
- // if (!enter.size() && update.size() > 0) {
437
- // // console.log('test')
438
- // const pathSelection = renderPie({
439
- // selection: context.containerSelection,
440
- // data: data.pieData,
441
- // arc: data.arc,
442
- // pathClassName
443
- // })
444
- // subscriber.next(pathSelection)
445
- // }
446
- })
447
- }).pipe(
448
- shareReplay(1)
449
- )
450
-
451
- // pathSelection$.subscribe(data => {
452
- // console.log('pathSelection', data)
453
- // })
454
- // context.SeriesDataMap$.subscribe(data => {
455
- // console.log('SeriesDataMap', data)
456
- // })
457
- // context.computedData$.subscribe(data => {
458
- // console.log('computedData', data)
459
- // })
460
- // highlightTarget$.subscribe(data => {
461
- // console.log('highlightTarget', data)
462
- // })
463
-
464
- combineLatest({
465
- pathSelection: pathSelection$,
466
- SeriesDataMap: context.SeriesDataMap$,
467
- computedData: context.computedData$,
468
- highlightTarget: highlightTarget$
469
- }).pipe(
470
- takeUntil(destroy$),
471
- switchMap(async d => d)
472
- ).subscribe(data => {
473
- data.pathSelection
474
- .on('mouseover', (event, pieDatum) => {
475
- event.stopPropagation()
476
-
477
- context.event$.next({
478
- type: 'series',
479
- eventName: 'mouseover',
480
- pluginName,
481
- highlightTarget: data.highlightTarget,
482
- datum: pieDatum.data,
483
- series: data.SeriesDataMap.get(pieDatum.data.seriesLabel)!,
484
- seriesIndex: pieDatum.data.seriesIndex,
485
- seriesLabel: pieDatum.data.seriesLabel,
486
- event,
487
- data: data.computedData
488
- })
489
- })
490
- .on('mousemove', (event, pieDatum) => {
491
- event.stopPropagation()
492
-
493
- context.event$.next({
494
- type: 'series',
495
- eventName: 'mousemove',
496
- pluginName,
497
- highlightTarget: data.highlightTarget,
498
- datum: pieDatum.data,
499
- series: data.SeriesDataMap.get(pieDatum.data.seriesLabel)!,
500
- seriesIndex: pieDatum.data.seriesIndex,
501
- seriesLabel: pieDatum.data.seriesLabel,
502
- event,
503
- data: data.computedData,
504
- })
505
- })
506
- .on('mouseout', (event, pieDatum) => {
507
- event.stopPropagation()
508
-
509
- context.event$.next({
510
- type: 'series',
511
- eventName: 'mouseout',
512
- pluginName,
513
- highlightTarget: data.highlightTarget,
514
- datum: pieDatum.data,
515
- series: data.SeriesDataMap.get(pieDatum.data.seriesLabel)!,
516
- seriesIndex: pieDatum.data.seriesIndex,
517
- seriesLabel: pieDatum.data.seriesLabel,
518
- event,
519
- data: data.computedData,
520
- })
521
- })
522
- .on('click', (event, pieDatum) => {
523
- event.stopPropagation()
524
-
525
- context.event$.next({
526
- type: 'series',
527
- eventName: 'click',
528
- pluginName,
529
- highlightTarget: data.highlightTarget,
530
- datum: pieDatum.data,
531
- series: data.SeriesDataMap.get(pieDatum.data.seriesLabel)!,
532
- seriesIndex: pieDatum.data.seriesIndex,
533
- seriesLabel: pieDatum.data.seriesLabel,
534
- event,
535
- data: data.computedData,
536
- })
537
- })
538
- })
539
-
540
- combineLatest({
541
- pathSelection: pathSelection$,
542
- highlight: context.seriesHighlight$.pipe(
543
- map(data => data.map(d => d.id))
544
- ),
545
- fullChartParams: context.fullChartParams$,
546
- arc: arc$,
547
- arcHighlight: arcHighlight$
548
- }).pipe(
549
- takeUntil(destroy$),
550
- switchMap(async d => d)
551
- ).subscribe(data => {
552
- highlight({
553
- pathSelection: data.pathSelection,
554
- ids: data.highlight,
555
- fullChartParams: data.fullChartParams,
556
- arc: data.arc,
557
- arcHighlight: data.arcHighlight
558
- })
559
- })
560
-
561
- return () => {
562
- destroy$.next(undefined)
563
- }
564
- }
565
-
566
- export const Pie = defineSeriesPlugin(pluginConfig)(({ selection, name, subject, observer }) => {
567
- const destroy$ = new Subject()
568
-
569
- const { seriesCenterSelection$ } = seriesCenterSelectionObservable({
570
- selection: selection,
571
- pluginName,
572
- separateSeries$: observer.separateSeries$,
573
- seriesLabels$: observer.seriesLabels$,
574
- seriesContainerPosition$: observer.seriesContainerPosition$
575
- })
576
-
577
- const unsubscribeFnArr: (() => void)[] = []
578
-
579
- seriesCenterSelection$
580
- .pipe(
581
- takeUntil(destroy$)
582
- )
583
- .subscribe(seriesCenterSelection => {
584
- // 每次重新計算時,清除之前的訂閱
585
- unsubscribeFnArr.forEach(fn => fn())
586
-
587
- // observer.fullParams$.subscribe(data => {
588
- // console.log('observer.fullParams$', data)
589
- // })
590
-
591
- seriesCenterSelection.each((d, containerIndex, g) => {
592
- // console.log('containerIndex', containerIndex)
593
- const containerSelection = d3.select(g[containerIndex])
594
-
595
- const containerVisibleComputedSortedData$ = observer.visibleComputedSortedData$.pipe(
596
- takeUntil(destroy$),
597
- map(data => data[containerIndex] ?? data[0])
598
- )
599
-
600
- const containerPosition$ = observer.seriesContainerPosition$.pipe(
601
- takeUntil(destroy$),
602
- map(data => data[containerIndex] ?? data[0])
603
- )
604
-
605
- unsubscribeFnArr[containerIndex] = createEachPie(pluginName, {
606
- containerSelection: containerSelection,
607
- computedData$: observer.computedData$,
608
- containerVisibleComputedSortedData$: containerVisibleComputedSortedData$,
609
- SeriesDataMap$: observer.SeriesDataMap$,
610
- fullParams$: observer.fullParams$,
611
- fullChartParams$: observer.fullChartParams$,
612
- seriesHighlight$: observer.seriesHighlight$,
613
- seriesContainerPosition$: containerPosition$,
614
- event$: subject.event$,
615
- })
616
-
617
- })
618
- })
619
-
620
- return () => {
621
- destroy$.next(undefined)
622
- unsubscribeFnArr.forEach(fn => fn())
623
- }
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 { DefinePluginConfig } from '../../../lib/core-types'
12
+ import type {
13
+ ComputedDataSeries,
14
+ ComputedDatumSeries,
15
+ ContainerPosition,
16
+ ChartParams,
17
+ EventSeries,
18
+ Layout } from '../../../lib/core-types'
19
+ import type { PieDatum } from '../seriesUtils'
20
+ import type { PieParams } from '../../../lib/plugins-basic-types'
21
+ import {
22
+ defineSeriesPlugin } from '../../../lib/core'
23
+ import { DEFAULT_PIE_PARAMS } from '../defaults'
24
+ import { makePieData } from '../seriesUtils'
25
+ import { getD3TransitionEase, makeD3Arc } from '../../utils/d3Utils'
26
+ import { getDatumColor, getClassName } from '../../utils/orbchartsUtils'
27
+ import { seriesCenterSelectionObservable } from '../seriesObservables'
28
+ import { LAYER_INDEX_OF_GRAPHIC } from '../../const'
29
+
30
+ const pluginName = 'Pie'
31
+
32
+ const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_PIE_PARAMS> = {
33
+ name: pluginName,
34
+ defaultParams: DEFAULT_PIE_PARAMS,
35
+ layerIndex: LAYER_INDEX_OF_GRAPHIC,
36
+ validator: (params, { validateColumns }) => {
37
+ const result = validateColumns(params, {
38
+ outerRadius: {
39
+ toBeTypes: ['number'],
40
+ },
41
+ innerRadius: {
42
+ toBeTypes: ['number'],
43
+ },
44
+ outerRadiusWhileHighlight: {
45
+ toBeTypes: ['number'],
46
+ },
47
+ startAngle: {
48
+ toBeTypes: ['number'],
49
+ },
50
+ endAngle: {
51
+ toBeTypes: ['number'],
52
+ },
53
+ padAngle: {
54
+ toBeTypes: ['number'],
55
+ },
56
+ strokeColorType: {
57
+ toBeTypes: ['string'],
58
+ },
59
+ strokeWidth: {
60
+ toBeTypes: ['number'],
61
+ },
62
+ cornerRadius: {
63
+ toBeTypes: ['number'],
64
+ }
65
+ })
66
+
67
+ return result
68
+ }
69
+ }
70
+
71
+ function makeTweenPieRenderDataFn ({ enter, exit, data, lastTweenData, fullParams }: {
72
+ enter: d3.Selection<d3.EnterElement, PieDatum, any, any>
73
+ exit: d3.Selection<SVGPathElement, unknown, any, any>
74
+ data: PieDatum[]
75
+ lastTweenData: PieDatum[]
76
+ fullParams: PieParams
77
+ }): (t: number) => PieDatum[] {
78
+ // 無更新資料項目則只計算資料變化 (新資料 * t + 舊資料 * (1 - t))
79
+ if (!enter.size() && !exit.size()) {
80
+ // console.log('case1')
81
+ return (t: number) => {
82
+ const tweenData: PieDatum[] = data.map((_d, _i) => {
83
+ const lastDatum = lastTweenData[_i] ?? {
84
+ startAngle: 0,
85
+ endAngle: 0,
86
+ value: 0
87
+ }
88
+ return {
89
+ ..._d,
90
+ startAngle: (_d.startAngle * t) + (lastDatum.startAngle * (1 - t)),
91
+ endAngle: (_d.endAngle * t) + (lastDatum.endAngle * (1 - t)),
92
+ value: (_d.value * t) + (lastDatum.value * (1 - t))
93
+ }
94
+ })
95
+
96
+ return makePieRenderData(
97
+ tweenData,
98
+ fullParams.startAngle!,
99
+ fullParams.endAngle!,
100
+ 1
101
+ )
102
+ }
103
+ // 有更新資料則重新繪圖
104
+ } else {
105
+ // console.log('case2')
106
+ return (t: number) => {
107
+ return makePieRenderData(
108
+ data,
109
+ fullParams.startAngle!,
110
+ fullParams.endAngle!,
111
+ t
112
+ )
113
+ }
114
+ }
115
+ }
116
+
117
+ function makePieRenderData (data: PieDatum[], startAngle: number, endAngle: number, t: number): PieDatum[] {
118
+ return data.map((d, i) => {
119
+ const _startAngle = startAngle + (d.startAngle - startAngle) * t
120
+ const _endAngle = _startAngle + (d.endAngle - d.startAngle) * t
121
+ return {
122
+ ...d,
123
+ startAngle: _startAngle,
124
+ endAngle: _endAngle
125
+ }
126
+ })
127
+ }
128
+
129
+ function renderPie ({ selection, data, arc, pathClassName, fullParams, fullChartParams }: {
130
+ selection: d3.Selection<SVGGElement, unknown, any, unknown>
131
+ data: PieDatum[]
132
+ arc: d3.Arc<any, d3.DefaultArcObject>
133
+ pathClassName: string
134
+ fullParams: PieParams
135
+ fullChartParams: ChartParams
136
+ }): d3.Selection<SVGPathElement, PieDatum, any, any> {
137
+ // console.log('data', data)
138
+ const pathSelection: d3.Selection<SVGPathElement, PieDatum, any, any> = selection
139
+ .selectAll<SVGPathElement, PieDatum>('path')
140
+ .data(data, d => d.id)
141
+ .join('path')
142
+ .classed(pathClassName, true)
143
+ .style('cursor', 'pointer')
144
+ .attr('fill', (d, i) => d.data.color)
145
+ .attr('stroke', (d, i) => getDatumColor({ datum: d.data, colorType: fullParams.strokeColorType, fullChartParams }))
146
+ .attr('stroke-width', fullParams.strokeWidth)
147
+ .attr('d', (d, i) => {
148
+ return arc!(d as any)
149
+ })
150
+
151
+ return pathSelection
152
+ }
153
+
154
+ function highlight ({ pathSelection, ids, fullChartParams, arc, arcHighlight }: {
155
+ pathSelection: d3.Selection<SVGPathElement, PieDatum, any, any>
156
+ ids: string[]
157
+ fullChartParams: ChartParams
158
+ arc: d3.Arc<any, d3.DefaultArcObject>
159
+ arcHighlight: d3.Arc<any, d3.DefaultArcObject>
160
+ }) {
161
+ pathSelection.interrupt('highlight')
162
+
163
+ if (!ids.length) {
164
+ // 取消放大
165
+ pathSelection
166
+ .transition('highlight')
167
+ .style('opacity', 1)
168
+ .attr('d', (d) => {
169
+ return arc!(d as any)
170
+ })
171
+ return
172
+ }
173
+
174
+ pathSelection.each((d, i, n) => {
175
+ const segment = d3.select(n[i])
176
+
177
+ if (ids.includes(d.data.id)) {
178
+ segment
179
+ .style('opacity', 1)
180
+ .transition('highlight')
181
+ .ease(d3.easeElastic)
182
+ .duration(500)
183
+ .attr('d', (d: any) => {
184
+ return arcHighlight!(d)
185
+ })
186
+ // .on('interrupt', () => {
187
+ // // this.pathSelection!.select('path').attr('d', (d) => {
188
+ // // return this.arc!(d as any)
189
+ // // })
190
+ // this.initHighlight()
191
+ // })
192
+ } else {
193
+ // 取消放大
194
+ segment
195
+ .style('opacity', fullChartParams.styles.unhighlightedOpacity)
196
+ .transition('highlight')
197
+ .attr('d', (d) => {
198
+ return arc!(d as any)
199
+ })
200
+ }
201
+ })
202
+ }
203
+
204
+ // 各別的pie
205
+ function createEachPie (pluginName: string, context: {
206
+ containerSelection: d3.Selection<SVGGElement, any, any, unknown>
207
+ computedData$: Observable<ComputedDatumSeries[][]>
208
+ containerVisibleComputedSortedData$: Observable<ComputedDatumSeries[]>
209
+ SeriesDataMap$: Observable<Map<string, ComputedDatumSeries[]>>
210
+ fullParams$: Observable<PieParams>
211
+ fullChartParams$: Observable<ChartParams>
212
+ seriesHighlight$: Observable<ComputedDatumSeries[]>
213
+ seriesContainerPosition$: Observable<ContainerPosition>
214
+ event$: Subject<EventSeries>
215
+ }) {
216
+ const destroy$ = new Subject()
217
+
218
+ const pathClassName = getClassName(pluginName, 'path')
219
+
220
+ let lastTweenData: PieDatum[] = [] // 紀錄補間動畫前次的資料
221
+ let tweenData: PieDatum[] = [] // 紀錄補間動畫用的資料
222
+ // let originHighlight: Highlight | null = null
223
+
224
+ // context.layout$
225
+ // .pipe(
226
+ // first()
227
+ // )
228
+ // .subscribe(size => {
229
+ // selection
230
+ // .attr('transform', `translate(${size.width / 2}, ${size.height / 2})`)
231
+ // context.layout$
232
+ // .pipe(
233
+ // takeUntil(destroy$)
234
+ // )
235
+ // .subscribe(size => {
236
+ // selection
237
+ // .transition()
238
+ // .attr('transform', `translate(${size.width / 2}, ${size.height / 2})`)
239
+ // })
240
+ // })
241
+
242
+
243
+ const shorterSideWith$ = context.seriesContainerPosition$.pipe(
244
+ takeUntil(destroy$),
245
+ map(d => d.width < d.height ? d.width : d.height),
246
+ distinctUntilChanged()
247
+ )
248
+
249
+ const pieData$: Observable<PieDatum[]> = new Observable(subscriber => {
250
+ combineLatest({
251
+ containerVisibleComputedSortedData: context.containerVisibleComputedSortedData$,
252
+ fullParams: context.fullParams$,
253
+ }).pipe(
254
+ takeUntil(destroy$),
255
+ switchMap(async (d) => d),
256
+ ).subscribe(data => {
257
+ // console.log('pieData', data)
258
+ const pieData: PieDatum[] = makePieData({
259
+ data: data.containerVisibleComputedSortedData,
260
+ startAngle: data.fullParams.startAngle,
261
+ endAngle: data.fullParams.endAngle
262
+ })
263
+ // console.log('pieData', pieData)
264
+ subscriber.next(pieData)
265
+ })
266
+ })
267
+
268
+ // const SeriesDataMap$ = context.computedData$.pipe(
269
+ // takeUntil(destroy$),
270
+ // map(d => makeSeriesDataMap(d))
271
+ // )
272
+
273
+ const arc$: Observable<d3.Arc<any, d3.DefaultArcObject>> = new Observable(subscriber => {
274
+ combineLatest({
275
+ shorterSideWith: shorterSideWith$,
276
+ fullParams: context.fullParams$,
277
+ }).pipe(
278
+ takeUntil(destroy$),
279
+ switchMap(async (d) => d),
280
+ ).subscribe(data => {
281
+ const arc = makeD3Arc({
282
+ axisWidth: data.shorterSideWith,
283
+ innerRadius: data.fullParams.innerRadius,
284
+ outerRadius: data.fullParams.outerRadius,
285
+ padAngle: data.fullParams.padAngle,
286
+ cornerRadius: data.fullParams.cornerRadius
287
+ })
288
+ subscriber.next(arc)
289
+ })
290
+ })
291
+
292
+ const arcHighlight$: Observable<d3.Arc<any, d3.DefaultArcObject>> = new Observable(subscriber => {
293
+ combineLatest({
294
+ shorterSideWith: shorterSideWith$,
295
+ fullParams: context.fullParams$,
296
+ }).pipe(
297
+ takeUntil(destroy$),
298
+ switchMap(async (d) => d),
299
+ ).subscribe(data => {
300
+ const arcHighlight = makeD3Arc({
301
+ axisWidth: data.shorterSideWith,
302
+ innerRadius: data.fullParams.innerRadius,
303
+ outerRadius: data.fullParams.outerRadiusWhileHighlight, // 外半徑變化
304
+ padAngle: data.fullParams.padAngle,
305
+ cornerRadius: data.fullParams.cornerRadius
306
+ })
307
+ subscriber.next(arcHighlight)
308
+ })
309
+ })
310
+
311
+ const highlightTarget$ = context.fullChartParams$.pipe(
312
+ takeUntil(destroy$),
313
+ map(d => d.highlightTarget),
314
+ distinctUntilChanged()
315
+ )
316
+
317
+ const pathSelection$ = new Observable<d3.Selection<SVGPathElement, PieDatum, any, any>>(subscriber => {
318
+ combineLatest({
319
+ pieData: pieData$,
320
+ arc: arc$,
321
+ computedData: context.computedData$,
322
+ fullParams: context.fullParams$,
323
+ fullChartParams: context.fullChartParams$,
324
+ highlightTarget: highlightTarget$
325
+ }).pipe(
326
+ takeUntil(destroy$),
327
+ switchMap(async d => d)
328
+ ).subscribe(data => {
329
+ context.containerSelection.interrupt('graphicMove')
330
+ // console.log('graphic', data)
331
+ const update: d3.Selection<SVGPathElement, PieDatum, any, any> = context.containerSelection
332
+ .selectAll<SVGPathElement, PieDatum>('path')
333
+ .data(data.pieData, d => d.id)
334
+ const enter = update.enter()
335
+ const exit = update.exit()
336
+
337
+ const makeTweenPieRenderData = makeTweenPieRenderDataFn({
338
+ enter,
339
+ exit,
340
+ data: data.pieData,
341
+ lastTweenData,
342
+ fullParams: data.fullParams
343
+ })
344
+
345
+ // -- 使用補間動畫 --
346
+ context.containerSelection
347
+ .transition('graphicMove')
348
+ .duration(data.fullChartParams.transitionDuration)
349
+ // .ease(getD3TransitionEase(data.fullChartParams.transitionEase))
350
+ .tween('move', (self, t) => {
351
+ return (t) => {
352
+ tweenData = makeTweenPieRenderData(t)
353
+
354
+ const pathSelection = renderPie({
355
+ selection: context.containerSelection,
356
+ data: tweenData,
357
+ arc: data.arc,
358
+ pathClassName,
359
+ fullParams: data.fullParams,
360
+ fullChartParams: data.fullChartParams,
361
+ })
362
+
363
+ // @Q@ 想盡量減清效能負擔所以取消掉
364
+ // context.event$.next({
365
+ // type: 'series',
366
+ // pluginName,
367
+ // eventName: 'transitionMove',
368
+ // event: undefined,
369
+ // highlightTarget: data.highlightTarget,
370
+ // datum: null,
371
+ // series: [],
372
+ // seriesIndex: -1,
373
+ // seriesLabel: '',
374
+ // data: data.computedData
375
+ // })
376
+
377
+ // const callbackData = makeEnterDurationCallbackData(data.computedData, )
378
+ // enterDurationCallback(callbackData, t)
379
+ }
380
+ })
381
+ .on('end', (self, t) => {
382
+ tweenData = makePieRenderData(
383
+ data.pieData,
384
+ data.fullParams.startAngle,
385
+ data.fullParams.endAngle,
386
+ 1
387
+ )
388
+ // console.log('tweenData', tweenData)
389
+ const pathSelection = renderPie({
390
+ selection: context.containerSelection,
391
+ data: tweenData,
392
+ arc: data.arc,
393
+ pathClassName,
394
+ fullParams: data.fullParams,
395
+ fullChartParams: data.fullChartParams,
396
+ })
397
+
398
+ // if (data.fullParams.highlightTarget && data.fullParams.highlightTarget != 'none') {
399
+ // if (data.fullChartParams.highlightTarget && data.fullChartParams.highlightTarget != 'none') {
400
+ // pathSelection!.style('cursor', 'pointer')
401
+ // }
402
+
403
+ subscriber.next(pathSelection)
404
+
405
+ // pathSelection && setPathEvent({
406
+ // pathSelection,
407
+ // pluginName: name,
408
+ // data: data.computedData,
409
+ // fullChartParams: data.fullChartParams,
410
+ // arc: data.arc,
411
+ // arcHighlight: data.arcHighlight,
412
+ // SeriesDataMap: data.SeriesDataMap,
413
+ // event$: store.event$
414
+ // })
415
+
416
+ // 渲染完後紀錄為前次的資料
417
+ lastTweenData = Object.assign([], data.pieData)
418
+
419
+ context.event$.next({
420
+ type: 'series',
421
+ pluginName,
422
+ eventName: 'transitionEnd',
423
+ event: undefined,
424
+ highlightTarget: data.highlightTarget,
425
+ datum: null,
426
+ series: [],
427
+ seriesIndex: -1,
428
+ seriesLabel: '',
429
+ data: data.computedData
430
+ })
431
+
432
+
433
+ })
434
+
435
+ // -- 更新資料 --
436
+ // if (!enter.size() && update.size() > 0) {
437
+ // // console.log('test')
438
+ // const pathSelection = renderPie({
439
+ // selection: context.containerSelection,
440
+ // data: data.pieData,
441
+ // arc: data.arc,
442
+ // pathClassName
443
+ // })
444
+ // subscriber.next(pathSelection)
445
+ // }
446
+ })
447
+ }).pipe(
448
+ shareReplay(1)
449
+ )
450
+
451
+ // pathSelection$.subscribe(data => {
452
+ // console.log('pathSelection', data)
453
+ // })
454
+ // context.SeriesDataMap$.subscribe(data => {
455
+ // console.log('SeriesDataMap', data)
456
+ // })
457
+ // context.computedData$.subscribe(data => {
458
+ // console.log('computedData', data)
459
+ // })
460
+ // highlightTarget$.subscribe(data => {
461
+ // console.log('highlightTarget', data)
462
+ // })
463
+
464
+ combineLatest({
465
+ pathSelection: pathSelection$,
466
+ SeriesDataMap: context.SeriesDataMap$,
467
+ computedData: context.computedData$,
468
+ highlightTarget: highlightTarget$
469
+ }).pipe(
470
+ takeUntil(destroy$),
471
+ switchMap(async d => d)
472
+ ).subscribe(data => {
473
+ data.pathSelection
474
+ .on('mouseover', (event, pieDatum) => {
475
+ event.stopPropagation()
476
+
477
+ context.event$.next({
478
+ type: 'series',
479
+ eventName: 'mouseover',
480
+ pluginName,
481
+ highlightTarget: data.highlightTarget,
482
+ datum: pieDatum.data,
483
+ series: data.SeriesDataMap.get(pieDatum.data.seriesLabel)!,
484
+ seriesIndex: pieDatum.data.seriesIndex,
485
+ seriesLabel: pieDatum.data.seriesLabel,
486
+ event,
487
+ data: data.computedData
488
+ })
489
+ })
490
+ .on('mousemove', (event, pieDatum) => {
491
+ event.stopPropagation()
492
+
493
+ context.event$.next({
494
+ type: 'series',
495
+ eventName: 'mousemove',
496
+ pluginName,
497
+ highlightTarget: data.highlightTarget,
498
+ datum: pieDatum.data,
499
+ series: data.SeriesDataMap.get(pieDatum.data.seriesLabel)!,
500
+ seriesIndex: pieDatum.data.seriesIndex,
501
+ seriesLabel: pieDatum.data.seriesLabel,
502
+ event,
503
+ data: data.computedData,
504
+ })
505
+ })
506
+ .on('mouseout', (event, pieDatum) => {
507
+ event.stopPropagation()
508
+
509
+ context.event$.next({
510
+ type: 'series',
511
+ eventName: 'mouseout',
512
+ pluginName,
513
+ highlightTarget: data.highlightTarget,
514
+ datum: pieDatum.data,
515
+ series: data.SeriesDataMap.get(pieDatum.data.seriesLabel)!,
516
+ seriesIndex: pieDatum.data.seriesIndex,
517
+ seriesLabel: pieDatum.data.seriesLabel,
518
+ event,
519
+ data: data.computedData,
520
+ })
521
+ })
522
+ .on('click', (event, pieDatum) => {
523
+ event.stopPropagation()
524
+
525
+ context.event$.next({
526
+ type: 'series',
527
+ eventName: 'click',
528
+ pluginName,
529
+ highlightTarget: data.highlightTarget,
530
+ datum: pieDatum.data,
531
+ series: data.SeriesDataMap.get(pieDatum.data.seriesLabel)!,
532
+ seriesIndex: pieDatum.data.seriesIndex,
533
+ seriesLabel: pieDatum.data.seriesLabel,
534
+ event,
535
+ data: data.computedData,
536
+ })
537
+ })
538
+ })
539
+
540
+ combineLatest({
541
+ pathSelection: pathSelection$,
542
+ highlight: context.seriesHighlight$.pipe(
543
+ map(data => data.map(d => d.id))
544
+ ),
545
+ fullChartParams: context.fullChartParams$,
546
+ arc: arc$,
547
+ arcHighlight: arcHighlight$
548
+ }).pipe(
549
+ takeUntil(destroy$),
550
+ switchMap(async d => d)
551
+ ).subscribe(data => {
552
+ highlight({
553
+ pathSelection: data.pathSelection,
554
+ ids: data.highlight,
555
+ fullChartParams: data.fullChartParams,
556
+ arc: data.arc,
557
+ arcHighlight: data.arcHighlight
558
+ })
559
+ })
560
+
561
+ return () => {
562
+ destroy$.next(undefined)
563
+ }
564
+ }
565
+
566
+ export const Pie = defineSeriesPlugin(pluginConfig)(({ selection, name, subject, observer }) => {
567
+ const destroy$ = new Subject()
568
+
569
+ const { seriesCenterSelection$ } = seriesCenterSelectionObservable({
570
+ selection: selection,
571
+ pluginName,
572
+ separateSeries$: observer.separateSeries$,
573
+ seriesLabels$: observer.seriesLabels$,
574
+ seriesContainerPosition$: observer.seriesContainerPosition$
575
+ })
576
+
577
+ const unsubscribeFnArr: (() => void)[] = []
578
+
579
+ seriesCenterSelection$
580
+ .pipe(
581
+ takeUntil(destroy$)
582
+ )
583
+ .subscribe(seriesCenterSelection => {
584
+ // 每次重新計算時,清除之前的訂閱
585
+ unsubscribeFnArr.forEach(fn => fn())
586
+
587
+ // observer.fullParams$.subscribe(data => {
588
+ // console.log('observer.fullParams$', data)
589
+ // })
590
+
591
+ seriesCenterSelection.each((d, containerIndex, g) => {
592
+ // console.log('containerIndex', containerIndex)
593
+ const containerSelection = d3.select(g[containerIndex])
594
+
595
+ const containerVisibleComputedSortedData$ = observer.visibleComputedSortedData$.pipe(
596
+ takeUntil(destroy$),
597
+ map(data => data[containerIndex] ?? data[0])
598
+ )
599
+
600
+ const containerPosition$ = observer.seriesContainerPosition$.pipe(
601
+ takeUntil(destroy$),
602
+ map(data => data[containerIndex] ?? data[0])
603
+ )
604
+
605
+ unsubscribeFnArr[containerIndex] = createEachPie(pluginName, {
606
+ containerSelection: containerSelection,
607
+ computedData$: observer.computedData$,
608
+ containerVisibleComputedSortedData$: containerVisibleComputedSortedData$,
609
+ SeriesDataMap$: observer.SeriesDataMap$,
610
+ fullParams$: observer.fullParams$,
611
+ fullChartParams$: observer.fullChartParams$,
612
+ seriesHighlight$: observer.seriesHighlight$,
613
+ seriesContainerPosition$: containerPosition$,
614
+ event$: subject.event$,
615
+ })
616
+
617
+ })
618
+ })
619
+
620
+ return () => {
621
+ destroy$.next(undefined)
622
+ unsubscribeFnArr.forEach(fn => fn())
623
+ }
624
624
  })