@orbcharts/plugins-basic 3.0.4 → 3.0.6

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 (117) hide show
  1. package/LICENSE +200 -200
  2. package/dist/orbcharts-plugins-basic.es.js +4738 -4700
  3. package/dist/orbcharts-plugins-basic.umd.js +48 -48
  4. package/dist/src/series/seriesObservables.d.ts +5 -7
  5. package/lib/core-types.ts +7 -7
  6. package/lib/core.ts +6 -6
  7. package/lib/gridObservables.ts +6 -6
  8. package/lib/plugins-basic-types.ts +6 -6
  9. package/package.json +48 -48
  10. package/src/base/BaseBars.ts +765 -765
  11. package/src/base/BaseBarsTriangle.ts +676 -676
  12. package/src/base/BaseDots.ts +464 -464
  13. package/src/base/BaseGroupAxis.ts +691 -691
  14. package/src/base/BaseLegend.ts +684 -684
  15. package/src/base/BaseLineAreas.ts +629 -629
  16. package/src/base/BaseLines.ts +706 -706
  17. package/src/base/BaseOrdinalBubbles.ts +729 -729
  18. package/src/base/BaseRacingBars.ts +582 -582
  19. package/src/base/BaseRacingLabels.ts +404 -404
  20. package/src/base/BaseRacingValueLabels.ts +403 -403
  21. package/src/base/BaseStackedBars.ts +782 -782
  22. package/src/base/BaseTooltip.ts +386 -386
  23. package/src/base/BaseValueAxis.ts +600 -600
  24. package/src/base/BaseXAxis.ts +427 -427
  25. package/src/base/BaseXZoom.ts +241 -241
  26. package/src/base/BaseYAxis.ts +389 -389
  27. package/src/base/types.ts +2 -2
  28. package/src/const.ts +30 -30
  29. package/src/grid/defaults.ts +213 -213
  30. package/src/grid/gridObservables.ts +635 -635
  31. package/src/grid/index.ts +16 -16
  32. package/src/grid/plugins/Bars.ts +69 -69
  33. package/src/grid/plugins/BarsPN.ts +66 -66
  34. package/src/grid/plugins/BarsTriangle.ts +73 -73
  35. package/src/grid/plugins/Dots.ts +68 -68
  36. package/src/grid/plugins/GridLegend.ts +107 -107
  37. package/src/grid/plugins/GridTooltip.ts +66 -66
  38. package/src/grid/plugins/GroupAux.ts +1095 -1095
  39. package/src/grid/plugins/GroupAxis.ts +73 -73
  40. package/src/grid/plugins/GroupZoom.ts +218 -218
  41. package/src/grid/plugins/LineAreas.ts +65 -65
  42. package/src/grid/plugins/Lines.ts +59 -59
  43. package/src/grid/plugins/StackedBars.ts +64 -64
  44. package/src/grid/plugins/StackedValueAxis.ts +96 -96
  45. package/src/grid/plugins/ValueAxis.ts +94 -94
  46. package/src/index.ts +6 -6
  47. package/src/multiGrid/defaults.ts +244 -244
  48. package/src/multiGrid/index.ts +14 -14
  49. package/src/multiGrid/multiGridObservables.ts +50 -50
  50. package/src/multiGrid/plugins/MultiBars.ts +108 -108
  51. package/src/multiGrid/plugins/MultiBarsTriangle.ts +114 -114
  52. package/src/multiGrid/plugins/MultiDots.ts +102 -102
  53. package/src/multiGrid/plugins/MultiGridLegend.ts +169 -169
  54. package/src/multiGrid/plugins/MultiGridTooltip.ts +66 -66
  55. package/src/multiGrid/plugins/MultiGroupAxis.ts +137 -137
  56. package/src/multiGrid/plugins/MultiLineAreas.ts +107 -107
  57. package/src/multiGrid/plugins/MultiLines.ts +101 -101
  58. package/src/multiGrid/plugins/MultiStackedBars.ts +106 -106
  59. package/src/multiGrid/plugins/MultiStackedValueAxis.ts +134 -134
  60. package/src/multiGrid/plugins/MultiValueAxis.ts +134 -134
  61. package/src/multiGrid/plugins/OverlappingStackedValueAxes.ts +300 -300
  62. package/src/multiGrid/plugins/OverlappingValueAxes.ts +300 -300
  63. package/src/multiValue/defaults.ts +523 -523
  64. package/src/multiValue/index.ts +16 -16
  65. package/src/multiValue/multiValueObservables.ts +781 -781
  66. package/src/multiValue/plugins/MultiValueLegend.ts +107 -107
  67. package/src/multiValue/plugins/MultiValueTooltip.ts +66 -66
  68. package/src/multiValue/plugins/OrdinalAux.ts +660 -660
  69. package/src/multiValue/plugins/OrdinalAxis.ts +524 -524
  70. package/src/multiValue/plugins/OrdinalBubbles.ts +226 -226
  71. package/src/multiValue/plugins/OrdinalZoom.ts +57 -57
  72. package/src/multiValue/plugins/RacingBars.ts +375 -375
  73. package/src/multiValue/plugins/RacingCounterTexts.ts +300 -300
  74. package/src/multiValue/plugins/RacingValueAxis.ts +114 -114
  75. package/src/multiValue/plugins/Scatter.ts +486 -486
  76. package/src/multiValue/plugins/ScatterBubbles.ts +635 -635
  77. package/src/multiValue/plugins/XAxis.ts +107 -107
  78. package/src/multiValue/plugins/XYAux.ts +683 -683
  79. package/src/multiValue/plugins/XYAxes.ts +194 -194
  80. package/src/multiValue/plugins/XYAxes_legacy.ts +683 -683
  81. package/src/multiValue/plugins/XZoom.ts +40 -40
  82. package/src/noneData/defaults.ts +102 -102
  83. package/src/noneData/index.ts +3 -3
  84. package/src/noneData/plugins/Container.ts +27 -27
  85. package/src/noneData/plugins/Tooltip.ts +373 -373
  86. package/src/relationship/defaults.ts +221 -221
  87. package/src/relationship/index.ts +5 -5
  88. package/src/relationship/plugins/ForceDirected.ts +1056 -1173
  89. package/src/relationship/plugins/ForceDirectedBubbles.ts +1294 -1411
  90. package/src/relationship/plugins/RelationshipLegend.ts +100 -100
  91. package/src/relationship/plugins/RelationshipTooltip.ts +66 -66
  92. package/src/relationship/relationshipObservables.ts +49 -49
  93. package/src/series/defaults.ts +223 -221
  94. package/src/series/index.ts +9 -9
  95. package/src/series/plugins/Bubbles.ts +781 -636
  96. package/src/series/plugins/Pie.ts +622 -623
  97. package/src/series/plugins/PieEventTexts.ts +283 -284
  98. package/src/series/plugins/PieLabels.ts +639 -640
  99. package/src/series/plugins/Rose.ts +515 -516
  100. package/src/series/plugins/RoseLabels.ts +599 -600
  101. package/src/series/plugins/SeriesLegend.ts +107 -107
  102. package/src/series/plugins/SeriesTooltip.ts +66 -66
  103. package/src/series/seriesObservables.ts +168 -145
  104. package/src/series/seriesUtils.ts +51 -51
  105. package/src/tree/defaults.ts +102 -102
  106. package/src/tree/index.ts +4 -4
  107. package/src/tree/plugins/TreeLegend.ts +100 -100
  108. package/src/tree/plugins/TreeMap.ts +341 -341
  109. package/src/tree/plugins/TreeTooltip.ts +66 -66
  110. package/src/utils/commonUtils.ts +31 -31
  111. package/src/utils/d3Graphics.ts +176 -176
  112. package/src/utils/d3Utils.ts +92 -92
  113. package/src/utils/observables.ts +14 -14
  114. package/src/utils/orbchartsUtils.ts +129 -129
  115. package/tsconfig.base.json +13 -13
  116. package/tsconfig.json +2 -2
  117. package/vite.config.js +22 -22
@@ -1,637 +1,782 @@
1
- import * as d3 from 'd3'
2
- import {
3
- combineLatest,
4
- map,
5
- switchMap,
6
- first,
7
- takeUntil,
8
- debounceTime,
9
- Subject,
10
- Observable,
11
- distinctUntilChanged,
12
- shareReplay} from 'rxjs'
13
- import type { DefinePluginConfig } from '../../../lib/core-types'
14
- import type {
15
- ChartParams,
16
- DatumValue,
17
- DataSeries,
18
- EventName,
19
- ComputedDataSeries,
20
- ComputedDatumSeries,
21
- ContainerPosition } from '../../../lib/core-types'
22
- import {
23
- defineSeriesPlugin } from '../../../lib/core'
24
- import type { BubblesParams, ArcScaleType } from '../../../lib/plugins-basic-types'
25
- import { DEFAULT_BUBBLES_PARAMS } from '../defaults'
26
- import { renderCircleText } from '../../utils/d3Graphics'
27
- import { LAYER_INDEX_OF_GRAPHIC } from '../../const'
28
- import { getDatumColor } from '../../utils/orbchartsUtils'
29
-
30
- interface BubblesDatum extends ComputedDatumSeries {
31
- x: number
32
- y: number
33
- r: number
34
- _originR: number // 紀錄變化前的r
35
- }
36
-
37
- type BubblesSimulationDatum = BubblesDatum & d3.SimulationNodeDatum
38
-
39
- const pluginName = 'Bubbles'
40
-
41
- const baseLineHeight = 12 // 未變形前的字體大小(代入計算用而已,數字多少都不會有影響)
42
-
43
- const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_BUBBLES_PARAMS> = {
44
- name: pluginName,
45
- defaultParams: DEFAULT_BUBBLES_PARAMS,
46
- layerIndex: LAYER_INDEX_OF_GRAPHIC,
47
- validator: (params, { validateColumns }) => {
48
- const result = validateColumns(params, {
49
- force: {
50
- toBeTypes: ['object']
51
- },
52
- bubbleLabel: {
53
- toBeTypes: ['object']
54
- },
55
- arcScaleType: {
56
- toBe: '"area" | "radius"',
57
- test: (value) => value === 'area' || value === 'radius'
58
- }
59
- })
60
- if (params.force) {
61
- const forceResult = validateColumns(params.force, {
62
- velocityDecay: {
63
- toBeTypes: ['number']
64
- },
65
- collisionSpacing: {
66
- toBeTypes: ['number']
67
- },
68
- strength: {
69
- toBeTypes: ['number']
70
- },
71
- })
72
- if (forceResult.status === 'error') {
73
- return forceResult
74
- }
75
- }
76
- if (params.bubbleLabel) {
77
- const bubbleLabelResult = validateColumns(params.bubbleLabel, {
78
- colorType: {
79
- toBeOption: 'ColorType'
80
- },
81
- fillRate: {
82
- toBeTypes: ['number']
83
- },
84
- lineHeight: {
85
- toBeTypes: ['number']
86
- },
87
- maxLineLength: {
88
- toBeTypes: ['number']
89
- },
90
- })
91
- if (bubbleLabelResult.status === 'error') {
92
- return bubbleLabelResult
93
- }
94
- }
95
- return result
96
- }
97
- }
98
-
99
-
100
- // let isRunning = false
101
-
102
- function createSimulation (bubblesSelection: d3.Selection<SVGGElement, BubblesDatum, any, any>, fullParams: BubblesParams) {
103
- return d3.forceSimulation()
104
- .velocityDecay(fullParams.force!.velocityDecay!)
105
- // .alphaDecay(0.2)
106
- .force(
107
- "collision",
108
- d3.forceCollide()
109
- .radius((d: d3.SimulationNodeDatum & BubblesDatum) => {
110
- return d.r + fullParams.force!.collisionSpacing
111
- })
112
- // .strength(0.01)
113
- )
114
- .force("charge", d3.forceManyBody().strength((d: d3.SimulationNodeDatum & BubblesDatum) => {
115
- return - Math.pow(d.r, 2.0) * fullParams.force!.strength
116
- }))
117
- // .force("charge", d3.forceManyBody().strength(-2000))
118
- // .force("collision", d3.forceCollide(60).strength(1)) // @Q@ 60為泡泡的R,暫時是先寫死的
119
- // .force("x", d3.forceX().strength(forceStrength).x(this.graphicWidth / 2))
120
- // .force("y", d3.forceY().strength(forceStrength).y(this.graphicHeight / 2))
121
- .on("tick", () => {
122
- // if (!bubblesSelection) {
123
- // return
124
- // }
125
- bubblesSelection
126
- .attr("transform", (d) => {
127
- return `translate(${d.x},${d.y})`
128
- })
129
- // .attr("cx", (d) => d.x)
130
- // .attr("cy", (d) => d.y)
131
-
132
-
133
- })
134
- // .on("end", () => {
135
-
136
- // })
137
-
138
- }
139
-
140
-
141
- // // 計算最大泡泡的半徑
142
- // function getMaxR ({ data, totalR, maxValue, avgValue }: {
143
- // data: DatumValue[]
144
- // totalR: number
145
- // maxValue: number
146
- // avgValue: number
147
- // }) {
148
- // // 平均r(假想是正方型來計算的,比如說大正方型裡有4個正方型,則 r = width/Math.sqrt(4)/2)
149
- // const avgR = totalR / Math.sqrt(data.length)
150
- // const avgSize = avgR * avgR * Math.PI
151
- // const sizeRate = avgSize / avgValue
152
- // const maxSize = maxValue * sizeRate
153
- // const maxR = Math.pow(maxSize / Math.PI, 0.5)
154
-
155
- // const modifier = 0.785 // @Q@ 因為以下公式是假設泡泡是正方型來計算,所以畫出來的圖會偏大一些,這個數值是用來修正用的
156
- // return maxR * modifier
157
- // }
158
-
159
- function createBubblesData ({ visibleComputedSortedData, LastBubbleDataMap, graphicWidth, graphicHeight, SeriesContainerPositionMap, scaleType }: {
160
- visibleComputedSortedData: ComputedDataSeries
161
- LastBubbleDataMap: Map<string, BubblesDatum>
162
- graphicWidth: number
163
- graphicHeight: number
164
- SeriesContainerPositionMap: Map<string, ContainerPosition>
165
- scaleType: ArcScaleType
166
- // highlightIds: string[]
167
- }): BubblesDatum[] {
168
- // 虛擬大圓(所有小圓聚合起來的大圓)的半徑
169
- const totalR = Math.min(...[graphicWidth, graphicHeight]) / 2
170
-
171
- const data = visibleComputedSortedData.flat()
172
-
173
- const totalValue = data.reduce((acc, current) => acc + current.value, 0)
174
-
175
- // 半徑比例尺
176
- const radiusScale = d3.scalePow()
177
- .domain([0, totalValue])
178
- .range([0, totalR])
179
- .exponent(scaleType === 'area'
180
- ? 0.5 // 數值映射面積(0.5為取平方根)
181
- : 1 // 數值映射半徑
182
- )
183
-
184
- // 縮放比例 - 確保多個小圓的總面積等於大圓的面積
185
- const scaleFactor = scaleType === 'area'
186
- ? 1
187
- // 當數值映射半徑時,多個小圓的總面積會小於大圓的面積,所以要計算縮放比例
188
- : (() => {
189
- const totalArea = totalR * totalR * Math.PI
190
- return Math.sqrt(totalArea / d3.sum(data, d => Math.PI * Math.pow(radiusScale(d.value), 2)))
191
- })()
192
-
193
- // 調整係數 - 因為圓和圓之間的空隙造成聚合起來的大圓會略大,所以稍作微調
194
- const adjustmentFactor = 0.9
195
-
196
- return data.map((_d) => {
197
- const d: BubblesDatum = _d as BubblesDatum
198
-
199
- const existDatum = LastBubbleDataMap.get(d.id)
200
-
201
- if (existDatum) {
202
- // 使用現有的座標
203
- d.x = existDatum.x
204
- d.y = existDatum.y
205
- } else {
206
- const seriesContainerPosition = SeriesContainerPositionMap.get(d.seriesLabel)!
207
- d.x = Math.random() * seriesContainerPosition.width
208
- d.y = Math.random() * seriesContainerPosition.height
209
- }
210
- const r = radiusScale!(d.value ?? 0)! * scaleFactor * adjustmentFactor
211
- d.r = r
212
- d._originR = r
213
-
214
- return d
215
- })
216
- }
217
-
218
- function renderBubbles ({ selection, bubblesData, fullParams, fullChartParams, sumSeries }: {
219
- selection: d3.Selection<SVGGElement, any, any, any>
220
- bubblesData: BubblesDatum[]
221
- fullParams: BubblesParams
222
- fullChartParams: ChartParams
223
- sumSeries: boolean
224
- }) {
225
- const bubblesSelection = selection.selectAll<SVGGElement, BubblesDatum>("g")
226
- .data(bubblesData, (d) => d.id)
227
- .join(
228
- enter => {
229
- const enterSelection = enter
230
- .append('g')
231
- .attr('cursor', 'pointer')
232
- .attr('font-size', baseLineHeight)
233
- .style('fill', '#ffffff')
234
- .attr("text-anchor", "middle")
235
-
236
- enterSelection
237
- .append("circle")
238
- .attr("class", "node")
239
- .attr("cx", 0)
240
- .attr("cy", 0)
241
- // .attr("r", 1e-6)
242
- .attr('fill', (d) => d.color)
243
- // .transition()
244
- // .duration(500)
245
-
246
- enterSelection
247
- .append('text')
248
- .style('opacity', 0.8)
249
- .attr('pointer-events', 'none')
250
-
251
- return enterSelection
252
- },
253
- update => {
254
- return update
255
- },
256
- exit => {
257
- return exit
258
- .remove()
259
- }
260
- )
261
- .attr("transform", (d) => {
262
- return `translate(${d.x},${d.y})`
263
- })
264
-
265
- // 泡泡文字要使用的的資料欄位
266
- const textDataColumn = sumSeries ? 'seriesLabel' : 'label'// 如果有合併series則使用seriesLabel
267
-
268
- bubblesSelection.select('circle')
269
- .transition()
270
- .duration(200)
271
- .attr("r", (d) => d.r)
272
- .attr('fill', (d) => d.color)
273
- bubblesSelection
274
- .each((d,i,g) => {
275
- const gSelection = d3.select(g[i])
276
- const text = d[textDataColumn] ?? ''
277
-
278
- gSelection.call(renderCircleText, {
279
- text,
280
- radius: d.r * fullParams.bubbleLabel.fillRate,
281
- lineHeight: baseLineHeight * fullParams.bubbleLabel.lineHeight,
282
- isBreakAll: text.length <= fullParams.bubbleLabel.maxLineLength
283
- ? false
284
- : fullParams.bubbleLabel.wordBreakAll
285
- })
286
-
287
- // -- text color --
288
- gSelection.select('text').attr('fill', _ => getDatumColor({
289
- datum: d,
290
- colorType: fullParams.bubbleLabel.colorType,
291
- fullChartParams: fullChartParams
292
- }))
293
-
294
- })
295
-
296
- return bubblesSelection
297
- }
298
-
299
- function setHighlightData ({ data, highlightRIncrease, highlightIds }: {
300
- data: BubblesDatum[]
301
- // fullParams: BubblesParams
302
- highlightRIncrease: number
303
- highlightIds: string[]
304
- }) {
305
- if (highlightRIncrease == 0) {
306
- return
307
- }
308
- if (!highlightIds.length) {
309
- data.forEach(d => d.r = d._originR)
310
- return
311
- }
312
- data.forEach(d => {
313
- if (highlightIds.includes(d.id)) {
314
- d.r = d._originR + highlightRIncrease
315
- } else {
316
- d.r = d._originR
317
- }
318
- })
319
- }
320
-
321
- function drag (_simulation: d3.Simulation<d3.SimulationNodeDatum, undefined>): d3.DragBehavior<Element, unknown, unknown> {
322
- return d3.drag()
323
- .on("start", (event, d: any) => {
324
- if (!event.active) {
325
- _simulation!.alpha(1).restart()
326
- }
327
- d.fx = d.x
328
- d.fy = d.y
329
- })
330
- .on("drag", (event, d: any) => {
331
- if (!event.active) {
332
- _simulation!.alphaTarget(0)
333
- }
334
- d.fx = event.x
335
- d.fy = event.y
336
- })
337
- .on("end", (event, d: any) => {
338
- d.fx = null
339
- d.fy = null
340
- _simulation!.alpha(1).restart()
341
- })
342
- }
343
-
344
-
345
- // private nodeTypePos (d: any) {
346
- // console.log(d)
347
- // console.log(this.TypeCenters.get(d.type)!)
348
- // const typeCenter = this.TypeCenters.get(d.type)!
349
- // return typeCenter ? typeCenter.x : 0
350
- // }
351
-
352
- function groupBubbles ({ _simulation, fullParams, SeriesContainerPositionMap }: {
353
- _simulation: d3.Simulation<d3.SimulationNodeDatum, undefined>
354
- fullParams: BubblesParams
355
- // graphicWidth: number
356
- // graphicHeight: number
357
- SeriesContainerPositionMap: Map<string, ContainerPosition>
358
- }) {
359
- // console.log('groupBubbles')
360
- _simulation!
361
- // .force('x', d3.forceX().strength(fullParams.force.strength).x(graphicWidth / 2))
362
- // .force('y', d3.forceY().strength(fullParams.force.strength).y(graphicHeight / 2))
363
- .force('x', d3.forceX().strength(fullParams.force.strength).x((data: BubblesSimulationDatum) => {
364
- return SeriesContainerPositionMap.get(data.seriesLabel)!.centerX
365
- }))
366
- .force('y', d3.forceY().strength(fullParams.force.strength).y((data: BubblesSimulationDatum) => {
367
- return SeriesContainerPositionMap.get(data.seriesLabel)!.centerY
368
- }))
369
-
370
- // force!.alpha(1).restart()
371
- }
372
-
373
- function highlight ({ bubblesSelection, highlightIds, fullChartParams }: {
374
- bubblesSelection: d3.Selection<SVGGElement, BubblesDatum, any, any>
375
- fullChartParams: ChartParams
376
- highlightIds: string[]
377
- }) {
378
- bubblesSelection.interrupt('highlight')
379
-
380
- if (!highlightIds.length) {
381
- bubblesSelection
382
- .transition('highlight')
383
- .style('opacity', 1)
384
- return
385
- }
386
-
387
- bubblesSelection.each((d, i, n) => {
388
- const segment = d3.select(n[i])
389
-
390
- if (highlightIds.includes(d.id)) {
391
- segment
392
- .style('opacity', 1)
393
- .transition('highlight')
394
- .ease(d3.easeElastic)
395
- .duration(500)
396
- } else {
397
- // 取消放大
398
- segment
399
- .style('opacity', fullChartParams.styles.unhighlightedOpacity)
400
- }
401
- })
402
- }
403
-
404
-
405
- export const Bubbles = defineSeriesPlugin(pluginConfig)(({ selection, name, observer, subject }) => {
406
-
407
- const destroy$ = new Subject()
408
-
409
- let simulation: d3.Simulation<d3.SimulationNodeDatum, undefined> | undefined
410
-
411
- // 紀錄前一次bubble data
412
- let LastBubbleDataMap: Map<string, BubblesDatum> = new Map()
413
-
414
-
415
- const sumSeries$ = observer.fullDataFormatter$.pipe(
416
- map(d => d.sumSeries),
417
- distinctUntilChanged()
418
- )
419
-
420
- const scaleType$ = observer.fullParams$.pipe(
421
- takeUntil(destroy$),
422
- map(d => d.arcScaleType),
423
- distinctUntilChanged()
424
- )
425
-
426
- const bubblesData$ = combineLatest({
427
- layout: observer.layout$,
428
- SeriesContainerPositionMap: observer.SeriesContainerPositionMap$,
429
- visibleComputedSortedData: observer.visibleComputedSortedData$,
430
- scaleType: scaleType$
431
- }).pipe(
432
- takeUntil(destroy$),
433
- switchMap(async (d) => d),
434
- map(data => {
435
- // console.log(data.visibleComputedSortedData)
436
- return createBubblesData({
437
- visibleComputedSortedData: data.visibleComputedSortedData,
438
- LastBubbleDataMap,
439
- graphicWidth: data.layout.width,
440
- graphicHeight: data.layout.height,
441
- SeriesContainerPositionMap: data.SeriesContainerPositionMap,
442
- scaleType: data.scaleType
443
- })
444
- }),
445
- shareReplay(1)
446
- )
447
-
448
- // 紀錄前一次bubble data
449
- bubblesData$.subscribe(d => {
450
- LastBubbleDataMap = new Map(d.map(_d => [_d.id, _d])) // key: id, value: datum
451
- })
452
-
453
- const highlightTarget$ = observer.fullChartParams$.pipe(
454
- takeUntil(destroy$),
455
- map(d => d.highlightTarget),
456
- distinctUntilChanged()
457
- )
458
-
459
- const bubblesSelection$ = combineLatest({
460
- bubblesData: bubblesData$,
461
- fullParams: observer.fullParams$,
462
- fullChartParams: observer.fullChartParams$,
463
- SeriesContainerPositionMap: observer.SeriesContainerPositionMap$,
464
- sumSeries: sumSeries$
465
- }).pipe(
466
- takeUntil(destroy$),
467
- switchMap(async (d) => d),
468
- map(data => {
469
- if (simulation) {
470
- // 先停止,重新計算之後再restart
471
- simulation.stop()
472
- }
473
-
474
- const bubblesSelection = renderBubbles({
475
- selection,
476
- bubblesData: data.bubblesData,
477
- fullParams: data.fullParams,
478
- fullChartParams: data.fullChartParams,
479
- sumSeries: data.sumSeries
480
- })
481
-
482
- simulation = createSimulation(bubblesSelection, data.fullParams)
483
-
484
- simulation.nodes(data.bubblesData)
485
-
486
- groupBubbles({
487
- _simulation: simulation,
488
- fullParams: data.fullParams,
489
- SeriesContainerPositionMap: data.SeriesContainerPositionMap
490
- // graphicWidth: data.layout.width,
491
- // graphicHeight: data.layout.height
492
- })
493
-
494
- simulation!.alpha(1).restart()
495
-
496
- return bubblesSelection
497
- }),
498
- shareReplay(1)
499
- )
500
-
501
- combineLatest({
502
- bubblesSelection: bubblesSelection$,
503
- computedData: observer.computedData$,
504
- SeriesDataMap: observer.SeriesDataMap$,
505
- highlightTarget: highlightTarget$,
506
- }).pipe(
507
- takeUntil(destroy$),
508
- switchMap(async (d) => d)
509
- ).subscribe(data => {
510
-
511
- data.bubblesSelection
512
- .on('mouseover', (event, datum) => {
513
- // this.tooltip!.setDatum({
514
- // data: d,
515
- // x: d3.event.clientX,
516
- // y: d3.event.clientY
517
- // })
518
-
519
- subject.event$.next({
520
- type: 'series',
521
- eventName: 'mouseover',
522
- pluginName: name,
523
- highlightTarget: data.highlightTarget,
524
- datum,
525
- series: data.SeriesDataMap.get(datum.seriesLabel)!,
526
- seriesIndex: datum.seriesIndex,
527
- seriesLabel: datum.seriesLabel,
528
- event,
529
- data: data.computedData
530
- })
531
- })
532
- .on('mousemove', (event, datum) => {
533
- // this.tooltip!.setDatum({
534
- // x: d3.event.clientX,
535
- // y: d3.event.clientY
536
- // })
537
-
538
- subject.event$.next({
539
- type: 'series',
540
- eventName: 'mousemove',
541
- pluginName: name,
542
- highlightTarget: data.highlightTarget,
543
- datum,
544
- series: data.SeriesDataMap.get(datum.seriesLabel)!,
545
- seriesIndex: datum.seriesIndex,
546
- seriesLabel: datum.seriesLabel,
547
- event,
548
- data: data.computedData
549
- })
550
- })
551
- .on('mouseout', (event, datum) => {
552
- // this.tooltip!.remove()
553
-
554
- subject.event$.next({
555
- type: 'series',
556
- eventName: 'mouseout',
557
- pluginName: name,
558
- highlightTarget: data.highlightTarget,
559
- datum,
560
- series: data.SeriesDataMap.get(datum.seriesLabel)!,
561
- seriesIndex: datum.seriesIndex,
562
- seriesLabel: datum.seriesLabel,
563
- event,
564
- data: data.computedData
565
- })
566
- })
567
- .on('click', (event, datum) => {
568
-
569
- subject.event$.next({
570
- type: 'series',
571
- eventName: 'click',
572
- pluginName: name,
573
- highlightTarget: data.highlightTarget,
574
- datum,
575
- series: data.SeriesDataMap.get(datum.seriesLabel)!,
576
- seriesIndex: datum.seriesIndex,
577
- seriesLabel: datum.seriesLabel,
578
- event,
579
- data: data.computedData
580
- })
581
- })
582
- .call(drag(simulation) as any)
583
-
584
-
585
- })
586
-
587
- combineLatest({
588
- bubblesSelection: bubblesSelection$,
589
- // bubblesData: bubblesData$,
590
- highlight: observer.seriesHighlight$.pipe(
591
- map(data => data.map(d => d.id))
592
- ),
593
- fullChartParams: observer.fullChartParams$,
594
- // fullParams: observer.fullParams$,
595
- // sumSeries: sumSeries$,
596
- // // layout: observer.layout$,
597
- // SeriesContainerPositionMap: observer.SeriesContainerPositionMap$,
598
- }).pipe(
599
- takeUntil(destroy$),
600
- switchMap(async d => d)
601
- ).subscribe(data => {
602
- highlight({
603
- bubblesSelection: data.bubblesSelection,
604
- highlightIds: data.highlight,
605
- fullChartParams: data.fullChartParams
606
- })
607
-
608
- // if (data.fullParams.highlightRIncrease) {
609
- // setHighlightData ({
610
- // data: data.bubblesData,
611
- // highlightRIncrease: data.fullParams.highlightRIncrease,
612
- // highlightIds: data.highlight
613
- // })
614
- // data.bubblesSelection.select('circle')
615
- // // .transition()
616
- // // .duration(200)
617
- // .attr("r", (d) => d.r)
618
-
619
- // force!.nodes(data.bubblesData)
620
-
621
- // groupBubbles({
622
- // fullParams: data.fullParams,
623
- // SeriesContainerPositionMap: data.SeriesContainerPositionMap
624
- // })
625
- // }
626
-
627
- })
628
-
629
-
630
- return () => {
631
- destroy$.next(undefined)
632
- if (simulation) {
633
- simulation.stop()
634
- simulation = undefined
635
- }
636
- }
1
+ import * as d3 from 'd3'
2
+ import {
3
+ combineLatest,
4
+ map,
5
+ switchMap,
6
+ first,
7
+ takeUntil,
8
+ debounceTime,
9
+ of,
10
+ iif,
11
+ Subject,
12
+ Observable,
13
+ distinctUntilChanged,
14
+ shareReplay} from 'rxjs'
15
+ import type { DefinePluginConfig } from '../../../lib/core-types'
16
+ import type {
17
+ ChartParams,
18
+ DatumValue,
19
+ DataSeries,
20
+ EventName,
21
+ ComputedDataSeries,
22
+ ComputedDatumSeries,
23
+ ContainerPosition } from '../../../lib/core-types'
24
+ import {
25
+ defineSeriesPlugin } from '../../../lib/core'
26
+ import type { BubblesParams, ArcScaleType } from '../../../lib/plugins-basic-types'
27
+ import { DEFAULT_BUBBLES_PARAMS } from '../defaults'
28
+ import { renderCircleText } from '../../utils/d3Graphics'
29
+ import { LAYER_INDEX_OF_GRAPHIC } from '../../const'
30
+ import { getDatumColor } from '../../utils/orbchartsUtils'
31
+
32
+ interface BubblesDatum extends ComputedDatumSeries {
33
+ x: number
34
+ y: number
35
+ r: number
36
+ renderLabel: string
37
+ _originR: number // 紀錄變化前的r
38
+ }
39
+
40
+ type BubblesSimulationDatum = BubblesDatum & d3.SimulationNodeDatum
41
+
42
+ const pluginName = 'Bubbles'
43
+
44
+ const baseLineHeight = 12 // 未變形前的字體大小(代入計算用而已,數字多少都不會有影響)
45
+
46
+ const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_BUBBLES_PARAMS> = {
47
+ name: pluginName,
48
+ defaultParams: DEFAULT_BUBBLES_PARAMS,
49
+ layerIndex: LAYER_INDEX_OF_GRAPHIC,
50
+ validator: (params, { validateColumns }) => {
51
+ const result = validateColumns(params, {
52
+ force: {
53
+ toBeTypes: ['object']
54
+ },
55
+ bubbleLabel: {
56
+ toBeTypes: ['object']
57
+ },
58
+ arcScaleType: {
59
+ toBe: '"area" | "radius"',
60
+ test: (value) => value === 'area' || value === 'radius'
61
+ }
62
+ })
63
+ if (params.force) {
64
+ const forceResult = validateColumns(params.force, {
65
+ velocityDecay: {
66
+ toBeTypes: ['number']
67
+ },
68
+ collisionSpacing: {
69
+ toBeTypes: ['number']
70
+ },
71
+ strength: {
72
+ toBeTypes: ['number']
73
+ },
74
+ })
75
+ if (forceResult.status === 'error') {
76
+ return forceResult
77
+ }
78
+ }
79
+ if (params.bubbleLabel) {
80
+ const bubbleLabelResult = validateColumns(params.bubbleLabel, {
81
+ labelFn: {
82
+ toBeTypes: ['Function'],
83
+ },
84
+ colorType: {
85
+ toBeOption: 'ColorType'
86
+ },
87
+ fillRate: {
88
+ toBeTypes: ['number']
89
+ },
90
+ lineHeight: {
91
+ toBeTypes: ['number']
92
+ },
93
+ maxLineLength: {
94
+ toBeTypes: ['number']
95
+ },
96
+ })
97
+ if (bubbleLabelResult.status === 'error') {
98
+ return bubbleLabelResult
99
+ }
100
+ }
101
+ return result
102
+ }
103
+ }
104
+
105
+
106
+ // let isRunning = false
107
+
108
+ function createSimulation (bubblesSelection: d3.Selection<SVGGElement, BubblesDatum, any, any>, fullParams: BubblesParams) {
109
+ return d3.forceSimulation()
110
+ .velocityDecay(fullParams.force!.velocityDecay!)
111
+ // .alphaDecay(0.2)
112
+ .force(
113
+ "collision",
114
+ d3.forceCollide()
115
+ .radius((d: d3.SimulationNodeDatum & BubblesDatum) => {
116
+ return d.r + fullParams.force!.collisionSpacing
117
+ })
118
+ // .strength(0.01)
119
+ )
120
+ .force("charge", d3.forceManyBody().strength((d: d3.SimulationNodeDatum & BubblesDatum) => {
121
+ return - Math.pow(d.r, 2.0) * fullParams.force!.strength
122
+ }))
123
+ // .force("charge", d3.forceManyBody().strength(-2000))
124
+ // .force("collision", d3.forceCollide(60).strength(1)) // @Q@ 60為泡泡的R,暫時是先寫死的
125
+ // .force("x", d3.forceX().strength(forceStrength).x(this.graphicWidth / 2))
126
+ // .force("y", d3.forceY().strength(forceStrength).y(this.graphicHeight / 2))
127
+ .on("tick", () => {
128
+ // if (!bubblesSelection) {
129
+ // return
130
+ // }
131
+ bubblesSelection
132
+ .attr("transform", (d) => {
133
+ return `translate(${d.x},${d.y})`
134
+ })
135
+ // .attr("cx", (d) => d.x)
136
+ // .attr("cy", (d) => d.y)
137
+
138
+
139
+ })
140
+ // .on("end", () => {
141
+
142
+ // })
143
+
144
+ }
145
+
146
+
147
+ // // 計算最大泡泡的半徑
148
+ // function getMaxR ({ data, totalR, maxValue, avgValue }: {
149
+ // data: DatumValue[]
150
+ // totalR: number
151
+ // maxValue: number
152
+ // avgValue: number
153
+ // }) {
154
+ // // 平均r(假想是正方型來計算的,比如說大正方型裡有4個正方型,則 r = width/Math.sqrt(4)/2)
155
+ // const avgR = totalR / Math.sqrt(data.length)
156
+ // const avgSize = avgR * avgR * Math.PI
157
+ // const sizeRate = avgSize / avgValue
158
+ // const maxSize = maxValue * sizeRate
159
+ // const maxR = Math.pow(maxSize / Math.PI, 0.5)
160
+
161
+ // const modifier = 0.785 // @Q@ 因為以下公式是假設泡泡是正方型來計算,所以畫出來的圖會偏大一些,這個數值是用來修正用的
162
+ // return maxR * modifier
163
+ // }
164
+
165
+ // function createBubblesData ({ visibleComputedSortedData, LastBubbleDataMap, fullParams, graphicWidth, graphicHeight, DatumContainerPositionMap, scaleType }: {
166
+ // visibleComputedSortedData: ComputedDataSeries
167
+ // LastBubbleDataMap: Map<string, BubblesDatum>
168
+ // fullParams: BubblesParams
169
+ // graphicWidth: number
170
+ // graphicHeight: number
171
+ // DatumContainerPositionMap: Map<string, ContainerPosition>
172
+ // scaleType: ArcScaleType
173
+ // // highlightIds: string[]
174
+ // }): BubblesDatum[] {
175
+ // // 虛擬大圓(所有小圓聚合起來的大圓)的半徑
176
+ // const totalR = Math.min(...[graphicWidth, graphicHeight]) / 2
177
+
178
+ // const data = visibleComputedSortedData.flat()
179
+
180
+ // const totalValue = data.reduce((acc, current) => acc + current.value, 0)
181
+
182
+ // // 半徑比例尺
183
+ // const radiusScale = d3.scalePow()
184
+ // .domain([0, totalValue])
185
+ // .range([0, totalR])
186
+ // .exponent(scaleType === 'area'
187
+ // ? 0.5 // 數值映射面積(0.5為取平方根)
188
+ // : 1 // 數值映射半徑
189
+ // )
190
+
191
+ // // 縮放比例 - 確保多個小圓的總面積等於大圓的面積
192
+ // const scaleFactor = scaleType === 'area'
193
+ // ? 1
194
+ // // 當數值映射半徑時,多個小圓的總面積會小於大圓的面積,所以要計算縮放比例
195
+ // : (() => {
196
+ // const totalArea = totalR * totalR * Math.PI
197
+ // return Math.sqrt(totalArea / d3.sum(data, d => Math.PI * Math.pow(radiusScale(d.value), 2)))
198
+ // })()
199
+
200
+ // // 調整係數 - 因為圓和圓之間的空隙造成聚合起來的大圓會略大,所以稍作微調
201
+ // const adjustmentFactor = 0.9
202
+
203
+ // return data.map((_d) => {
204
+ // const d: BubblesDatum = _d as BubblesDatum
205
+
206
+ // d.renderLabel = fullParams.bubbleLabel.labelFn(d)
207
+
208
+ // const existDatum = LastBubbleDataMap.get(d.id)
209
+
210
+ // if (existDatum) {
211
+ // // 使用現有的座標
212
+ // d.x = existDatum.x
213
+ // d.y = existDatum.y
214
+ // } else {
215
+ // const seriesContainerPosition = DatumContainerPositionMap.get(d.id)!
216
+ // d.x = Math.random() * seriesContainerPosition.width
217
+ // d.y = Math.random() * seriesContainerPosition.height
218
+ // }
219
+ // const r = radiusScale!(d.value ?? 0)! * scaleFactor * adjustmentFactor
220
+ // d.r = r
221
+ // d._originR = r
222
+
223
+ // return d
224
+ // })
225
+ // }
226
+
227
+ function renderBubbles ({ selection, bubblesData, fullParams, fullChartParams }: {
228
+ selection: d3.Selection<SVGGElement, any, any, any>
229
+ bubblesData: BubblesDatum[]
230
+ fullParams: BubblesParams
231
+ fullChartParams: ChartParams
232
+ }) {
233
+ const bubblesSelection = selection.selectAll<SVGGElement, BubblesDatum>("g")
234
+ .data(bubblesData, (d) => d.id)
235
+ .join(
236
+ enter => {
237
+ const enterSelection = enter
238
+ .append('g')
239
+ .attr('cursor', 'pointer')
240
+ .attr('font-size', baseLineHeight)
241
+ .style('fill', '#ffffff')
242
+ .attr("text-anchor", "middle")
243
+
244
+ enterSelection
245
+ .append("circle")
246
+ .attr("class", "node")
247
+ .attr("cx", 0)
248
+ .attr("cy", 0)
249
+ // .attr("r", 1e-6)
250
+ .attr('fill', (d) => d.color)
251
+ // .transition()
252
+ // .duration(500)
253
+
254
+ enterSelection
255
+ .append('text')
256
+ .style('opacity', 0.8)
257
+ .attr('pointer-events', 'none')
258
+
259
+ return enterSelection
260
+ },
261
+ update => {
262
+ return update
263
+ },
264
+ exit => {
265
+ return exit
266
+ .remove()
267
+ }
268
+ )
269
+ .attr("transform", (d) => {
270
+ return `translate(${d.x},${d.y})`
271
+ })
272
+
273
+ bubblesSelection.select('circle')
274
+ .transition()
275
+ .duration(200)
276
+ // .ease(d3.easeLinear)
277
+ .attr("r", (d) => d.r)
278
+ .attr('fill', (d) => d.color)
279
+ bubblesSelection
280
+ .each((d,i,g) => {
281
+ const gSelection = d3.select(g[i])
282
+ const text = d.renderLabel
283
+
284
+ gSelection.call(renderCircleText, {
285
+ text,
286
+ radius: d.r * fullParams.bubbleLabel.fillRate,
287
+ lineHeight: baseLineHeight * fullParams.bubbleLabel.lineHeight,
288
+ isBreakAll: text.length <= fullParams.bubbleLabel.maxLineLength
289
+ ? false
290
+ : fullParams.bubbleLabel.wordBreakAll
291
+ })
292
+
293
+ // -- text color --
294
+ gSelection.select('text').attr('fill', _ => getDatumColor({
295
+ datum: d,
296
+ colorType: fullParams.bubbleLabel.colorType,
297
+ fullChartParams: fullChartParams
298
+ }))
299
+
300
+ })
301
+
302
+ return bubblesSelection
303
+ }
304
+
305
+ function setHighlightData ({ data, highlightRIncrease, highlightIds }: {
306
+ data: BubblesDatum[]
307
+ // fullParams: BubblesParams
308
+ highlightRIncrease: number
309
+ highlightIds: string[]
310
+ }) {
311
+ if (highlightRIncrease == 0) {
312
+ return
313
+ }
314
+ if (!highlightIds.length) {
315
+ data.forEach(d => d.r = d._originR)
316
+ return
317
+ }
318
+ data.forEach(d => {
319
+ if (highlightIds.includes(d.id)) {
320
+ d.r = d._originR + highlightRIncrease
321
+ } else {
322
+ d.r = d._originR
323
+ }
324
+ })
325
+ }
326
+
327
+ function drag (_simulation: d3.Simulation<d3.SimulationNodeDatum, undefined>): d3.DragBehavior<Element, unknown, unknown> {
328
+ return d3.drag()
329
+ .on("start", (event, d: any) => {
330
+ if (!event.active) {
331
+ _simulation!.alpha(1).restart()
332
+ }
333
+ d.fx = d.x
334
+ d.fy = d.y
335
+ })
336
+ .on("drag", (event, d: any) => {
337
+ if (!event.active) {
338
+ _simulation!.alphaTarget(0)
339
+ }
340
+ d.fx = event.x
341
+ d.fy = event.y
342
+ })
343
+ .on("end", (event, d: any) => {
344
+ d.fx = null
345
+ d.fy = null
346
+ _simulation!.alpha(1).restart()
347
+ })
348
+ }
349
+
350
+
351
+ // private nodeTypePos (d: any) {
352
+ // console.log(d)
353
+ // console.log(this.TypeCenters.get(d.type)!)
354
+ // const typeCenter = this.TypeCenters.get(d.type)!
355
+ // return typeCenter ? typeCenter.x : 0
356
+ // }
357
+
358
+ function groupBubbles ({ _simulation, fullParams, DatumContainerPositionMap }: {
359
+ _simulation: d3.Simulation<d3.SimulationNodeDatum, undefined>
360
+ fullParams: BubblesParams
361
+ // graphicWidth: number
362
+ // graphicHeight: number
363
+ DatumContainerPositionMap: Map<string, ContainerPosition>
364
+ }) {
365
+ // console.log('groupBubbles')
366
+
367
+ _simulation!
368
+ // .force('x', d3.forceX().strength(fullParams.force.strength).x(graphicWidth / 2))
369
+ // .force('y', d3.forceY().strength(fullParams.force.strength).y(graphicHeight / 2))
370
+ .force('x', d3.forceX().strength(fullParams.force.strength).x((data: BubblesSimulationDatum) => {
371
+ let position = DatumContainerPositionMap.get(data.id)!
372
+ if (!position) {
373
+ // 有時候可能會因為時間差而找不到,這時候取第一筆
374
+ position = DatumContainerPositionMap.get(Array.from(DatumContainerPositionMap.keys())[0])
375
+ }
376
+
377
+ return position?.centerX ?? null
378
+ }))
379
+ .force('y', d3.forceY().strength(fullParams.force.strength).y((data: BubblesSimulationDatum) => {
380
+ let position = DatumContainerPositionMap.get(data.id)!
381
+ if (!position) {
382
+ position = DatumContainerPositionMap.get(Array.from(DatumContainerPositionMap.keys())[0])
383
+ }
384
+
385
+ return position?.centerY ?? null
386
+ }))
387
+
388
+ // force!.alpha(1).restart()
389
+ }
390
+
391
+ function highlight ({ bubblesSelection, highlightIds, fullChartParams }: {
392
+ bubblesSelection: d3.Selection<SVGGElement, BubblesDatum, any, any>
393
+ fullChartParams: ChartParams
394
+ highlightIds: string[]
395
+ }) {
396
+ bubblesSelection.interrupt('highlight')
397
+
398
+ if (!highlightIds.length) {
399
+ bubblesSelection
400
+ .transition('highlight')
401
+ .style('opacity', 1)
402
+ return
403
+ }
404
+
405
+ bubblesSelection.each((d, i, n) => {
406
+ const segment = d3.select(n[i])
407
+
408
+ if (highlightIds.includes(d.id)) {
409
+ segment
410
+ .style('opacity', 1)
411
+ .transition('highlight')
412
+ .ease(d3.easeElastic)
413
+ .duration(500)
414
+ } else {
415
+ // 取消放大
416
+ segment
417
+ .style('opacity', fullChartParams.styles.unhighlightedOpacity)
418
+ }
419
+ })
420
+ }
421
+
422
+
423
+ export const Bubbles = defineSeriesPlugin(pluginConfig)(({ selection, name, observer, subject }) => {
424
+
425
+ const destroy$ = new Subject()
426
+
427
+ let simulation: d3.Simulation<d3.SimulationNodeDatum, undefined> | undefined
428
+
429
+ // 紀錄前一次bubble data
430
+ let LastBubbleDataMap: Map<string, BubblesDatum> = new Map()
431
+
432
+
433
+ const scaleType$ = observer.fullParams$.pipe(
434
+ takeUntil(destroy$),
435
+ map(d => d.arcScaleType),
436
+ distinctUntilChanged(),
437
+ shareReplay(1)
438
+ )
439
+
440
+ // 虛擬大圓(所有小圓聚合起來的大圓)的半徑
441
+ const totalR$ = observer.layout$.pipe(
442
+ takeUntil(destroy$),
443
+ map(d => Math.min(d.width, d.height) / 2),
444
+ distinctUntilChanged(),
445
+ shareReplay(1)
446
+ )
447
+
448
+ const totalValue$ = observer.visibleComputedSortedData$.pipe(
449
+ takeUntil(destroy$),
450
+ map(d => d.flat().reduce((acc, current) => acc + current.value, 0)),
451
+ distinctUntilChanged(),
452
+ shareReplay(1)
453
+ )
454
+
455
+ // 半徑比例尺
456
+ const radiusScale$ = combineLatest({
457
+ totalR: totalR$,
458
+ totalValue: totalValue$,
459
+ scaleType: scaleType$
460
+ }).pipe(
461
+ takeUntil(destroy$),
462
+ switchMap(async (d) => d),
463
+ map(data => {
464
+ return d3.scalePow()
465
+ .domain([0, data.totalValue])
466
+ .range([0, data.totalR])
467
+ .exponent(data.scaleType === 'area'
468
+ ? 0.5 // 數值映射面積(0.5為取平方根)
469
+ : 1 // 數值映射半徑
470
+ )
471
+ }),
472
+ shareReplay(1)
473
+ )
474
+
475
+ // 縮放比例 - 確保多個小圓的總面積等於大圓的面積
476
+ const scaleFactor$ = scaleType$.pipe(
477
+ takeUntil(destroy$),
478
+ switchMap(scaleType => {
479
+ return iif(
480
+ () => scaleType === 'area',
481
+ of(1),
482
+ combineLatest({
483
+ totalR: totalR$,
484
+ radiusScale: radiusScale$,
485
+ visibleComputedSortedData: observer.visibleComputedSortedData$
486
+ }).pipe(
487
+ switchMap(async (d) => d),
488
+ map(data => {
489
+ // 當數值映射半徑時,多個小圓的總面積會小於大圓的面積,所以要計算縮放比例
490
+ const totalArea = data.totalR * data.totalR * Math.PI
491
+ return Math.sqrt(totalArea / d3.sum(data.visibleComputedSortedData.flat(), d => Math.PI * Math.pow(data.radiusScale(d.value), 2)))
492
+ })
493
+ )
494
+ )
495
+ })
496
+ )
497
+
498
+ const DatumRMap$ = combineLatest({
499
+ visibleComputedSortedData: observer.visibleComputedSortedData$,
500
+ radiusScale: radiusScale$,
501
+ scaleFactor: scaleFactor$
502
+ }).pipe(
503
+ takeUntil(destroy$),
504
+ switchMap(async (d) => d),
505
+ map(data => {
506
+ // 調整係數 - 因為圓和圓之間的空隙造成聚合起來的大圓會略大,所以稍作微調
507
+ const adjustmentFactor = 0.9
508
+
509
+ return new Map<string, number>(
510
+ data.visibleComputedSortedData
511
+ .flat()
512
+ .map(d => [d.id, data.radiusScale(d.value ?? 0) * data.scaleFactor * adjustmentFactor])
513
+ )
514
+ }),
515
+ shareReplay(1)
516
+ )
517
+
518
+ // 初始座標
519
+ const DatumInitXYMap$ = observer.DatumContainerPositionMap$.pipe(
520
+ takeUntil(destroy$),
521
+ map(data => {
522
+ return new Map<string, { x: number, y: number }>(
523
+ Array.from(data).map(([id, position]) => {
524
+ return [
525
+ id,
526
+ {
527
+ x: position.startX + (position.width * Math.random()),
528
+ y: position.startY + (position.height * Math.random())
529
+ }
530
+ ]
531
+ })
532
+ )
533
+ }),
534
+ first(), // 只算一次
535
+ shareReplay(1)
536
+ )
537
+
538
+ const bubblesData$ = combineLatest({
539
+ visibleComputedSortedData: observer.visibleComputedSortedData$,
540
+ DatumRMap: DatumRMap$,
541
+ DatumInitXYMap: DatumInitXYMap$,
542
+ fullParams: observer.fullParams$,
543
+ }).pipe(
544
+ takeUntil(destroy$),
545
+ switchMap(async (d) => d),
546
+ map(data => {
547
+ return data.visibleComputedSortedData
548
+ .flat()
549
+ .map(_d => {
550
+ // 傳址,附加計算的欄位資料會 reference 到始資料上
551
+ const d: BubblesDatum = _d as BubblesDatum
552
+
553
+ // 第一次計算時沒有 x, y 座標,取得預設座標。第二次之後計算使用原有的座標
554
+ if (d.x === undefined || d.y === undefined) {
555
+ let xy = data.DatumInitXYMap.get(d.id)!
556
+ if (!xy) {
557
+ xy = data.DatumInitXYMap.get(Array.from(data.DatumInitXYMap.keys())[0])
558
+ }
559
+ d.x = xy.x
560
+ d.y = xy.y
561
+ }
562
+ d.r = data.DatumRMap.get(d.id)!
563
+ d._originR = d.r
564
+ d.renderLabel = data.fullParams.bubbleLabel.labelFn(d)
565
+ return d
566
+ })
567
+ }),
568
+ shareReplay(1)
569
+ )
570
+
571
+ // const bubblesData$ = combineLatest({
572
+ // layout: observer.layout$,
573
+ // fullParams: observer.fullParams$,
574
+ // DatumContainerPositionMap: observer.DatumContainerPositionMap$,
575
+ // visibleComputedSortedData: observer.visibleComputedSortedData$,
576
+ // scaleType: scaleType$,
577
+ // }).pipe(
578
+ // takeUntil(destroy$),
579
+ // switchMap(async (d) => d),
580
+ // map(data => {
581
+ // // console.log(data.visibleComputedSortedData)
582
+ // return createBubblesData({
583
+ // visibleComputedSortedData: data.visibleComputedSortedData,
584
+ // LastBubbleDataMap,
585
+ // fullParams: data.fullParams,
586
+ // graphicWidth: data.layout.width,
587
+ // graphicHeight: data.layout.height,
588
+ // DatumContainerPositionMap: data.DatumContainerPositionMap,
589
+ // scaleType: data.scaleType
590
+ // })
591
+ // }),
592
+ // shareReplay(1)
593
+ // )
594
+
595
+ // // 紀錄前一次bubble data
596
+ // bubblesData$.subscribe(d => {
597
+ // LastBubbleDataMap = new Map(d.map(_d => [_d.id, _d])) // key: id, value: datum
598
+ // })
599
+
600
+ const highlightTarget$ = observer.fullChartParams$.pipe(
601
+ takeUntil(destroy$),
602
+ map(d => d.highlightTarget),
603
+ distinctUntilChanged()
604
+ )
605
+
606
+ const bubblesSelection$ = combineLatest({
607
+ bubblesData: bubblesData$,
608
+ fullParams: observer.fullParams$,
609
+ fullChartParams: observer.fullChartParams$,
610
+ DatumContainerPositionMap: observer.DatumContainerPositionMap$,
611
+ }).pipe(
612
+ takeUntil(destroy$),
613
+ switchMap(async (d) => d),
614
+ map(data => {
615
+ if (simulation) {
616
+ // 先停止,重新計算之後再restart
617
+ simulation.stop()
618
+ }
619
+
620
+ const bubblesSelection = renderBubbles({
621
+ selection,
622
+ bubblesData: data.bubblesData,
623
+ fullParams: data.fullParams,
624
+ fullChartParams: data.fullChartParams,
625
+ })
626
+
627
+ simulation = createSimulation(bubblesSelection, data.fullParams)
628
+
629
+ simulation.nodes(data.bubblesData)
630
+
631
+ groupBubbles({
632
+ _simulation: simulation,
633
+ fullParams: data.fullParams,
634
+ DatumContainerPositionMap: data.DatumContainerPositionMap
635
+ // graphicWidth: data.layout.width,
636
+ // graphicHeight: data.layout.height
637
+ })
638
+
639
+ simulation!.alpha(1).restart()
640
+
641
+ return bubblesSelection
642
+ }),
643
+ shareReplay(1)
644
+ )
645
+
646
+ combineLatest({
647
+ bubblesSelection: bubblesSelection$,
648
+ computedData: observer.computedData$,
649
+ SeriesDataMap: observer.SeriesDataMap$,
650
+ highlightTarget: highlightTarget$,
651
+ }).pipe(
652
+ takeUntil(destroy$),
653
+ switchMap(async (d) => d)
654
+ ).subscribe(data => {
655
+
656
+ data.bubblesSelection
657
+ .on('mouseover', (event, datum) => {
658
+ // this.tooltip!.setDatum({
659
+ // data: d,
660
+ // x: d3.event.clientX,
661
+ // y: d3.event.clientY
662
+ // })
663
+
664
+ subject.event$.next({
665
+ type: 'series',
666
+ eventName: 'mouseover',
667
+ pluginName: name,
668
+ highlightTarget: data.highlightTarget,
669
+ datum,
670
+ series: data.SeriesDataMap.get(datum.seriesLabel)!,
671
+ seriesIndex: datum.seriesIndex,
672
+ seriesLabel: datum.seriesLabel,
673
+ event,
674
+ data: data.computedData
675
+ })
676
+ })
677
+ .on('mousemove', (event, datum) => {
678
+ // this.tooltip!.setDatum({
679
+ // x: d3.event.clientX,
680
+ // y: d3.event.clientY
681
+ // })
682
+
683
+ subject.event$.next({
684
+ type: 'series',
685
+ eventName: 'mousemove',
686
+ pluginName: name,
687
+ highlightTarget: data.highlightTarget,
688
+ datum,
689
+ series: data.SeriesDataMap.get(datum.seriesLabel)!,
690
+ seriesIndex: datum.seriesIndex,
691
+ seriesLabel: datum.seriesLabel,
692
+ event,
693
+ data: data.computedData
694
+ })
695
+ })
696
+ .on('mouseout', (event, datum) => {
697
+ // this.tooltip!.remove()
698
+
699
+ subject.event$.next({
700
+ type: 'series',
701
+ eventName: 'mouseout',
702
+ pluginName: name,
703
+ highlightTarget: data.highlightTarget,
704
+ datum,
705
+ series: data.SeriesDataMap.get(datum.seriesLabel)!,
706
+ seriesIndex: datum.seriesIndex,
707
+ seriesLabel: datum.seriesLabel,
708
+ event,
709
+ data: data.computedData
710
+ })
711
+ })
712
+ .on('click', (event, datum) => {
713
+
714
+ subject.event$.next({
715
+ type: 'series',
716
+ eventName: 'click',
717
+ pluginName: name,
718
+ highlightTarget: data.highlightTarget,
719
+ datum,
720
+ series: data.SeriesDataMap.get(datum.seriesLabel)!,
721
+ seriesIndex: datum.seriesIndex,
722
+ seriesLabel: datum.seriesLabel,
723
+ event,
724
+ data: data.computedData
725
+ })
726
+ })
727
+ .call(drag(simulation) as any)
728
+
729
+
730
+ })
731
+
732
+ combineLatest({
733
+ bubblesSelection: bubblesSelection$,
734
+ // bubblesData: bubblesData$,
735
+ highlight: observer.seriesHighlight$.pipe(
736
+ map(data => data.map(d => d.id))
737
+ ),
738
+ fullChartParams: observer.fullChartParams$,
739
+ // fullParams: observer.fullParams$,
740
+ // sumSeries: sumSeries$,
741
+ // // layout: observer.layout$,
742
+ // DatumContainerPositionMap: observer.DatumContainerPositionMap$,
743
+ }).pipe(
744
+ takeUntil(destroy$),
745
+ switchMap(async d => d)
746
+ ).subscribe(data => {
747
+ highlight({
748
+ bubblesSelection: data.bubblesSelection,
749
+ highlightIds: data.highlight,
750
+ fullChartParams: data.fullChartParams
751
+ })
752
+
753
+ // if (data.fullParams.highlightRIncrease) {
754
+ // setHighlightData ({
755
+ // data: data.bubblesData,
756
+ // highlightRIncrease: data.fullParams.highlightRIncrease,
757
+ // highlightIds: data.highlight
758
+ // })
759
+ // data.bubblesSelection.select('circle')
760
+ // // .transition()
761
+ // // .duration(200)
762
+ // .attr("r", (d) => d.r)
763
+
764
+ // force!.nodes(data.bubblesData)
765
+
766
+ // groupBubbles({
767
+ // fullParams: data.fullParams,
768
+ // DatumContainerPositionMap: data.DatumContainerPositionMap
769
+ // })
770
+ // }
771
+
772
+ })
773
+
774
+
775
+ return () => {
776
+ destroy$.next(undefined)
777
+ if (simulation) {
778
+ simulation.stop()
779
+ simulation = undefined
780
+ }
781
+ }
637
782
  })