@orbcharts/plugins-basic 3.0.0-beta.7 → 3.0.0-beta.8

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 (126) hide show
  1. package/LICENSE +200 -200
  2. package/dist/orbcharts-plugins-basic/src/base/{BaseBarStack.d.ts → BaseStackedBar.d.ts} +4 -4
  3. package/dist/orbcharts-plugins-basic/src/grid/defaults.d.ts +3 -3
  4. package/dist/orbcharts-plugins-basic/src/grid/index.d.ts +2 -2
  5. package/dist/orbcharts-plugins-basic/src/grid/plugins/StackedBar.d.ts +1 -0
  6. package/dist/orbcharts-plugins-basic/src/grid/plugins/StackedValueAxis.d.ts +1 -0
  7. package/dist/orbcharts-plugins-basic/src/index.d.ts +1 -0
  8. package/dist/orbcharts-plugins-basic/src/multiGrid/defaults.d.ts +4 -4
  9. package/dist/orbcharts-plugins-basic/src/multiGrid/index.d.ts +3 -3
  10. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiStackedBar.d.ts +1 -0
  11. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiStackedValueAxis.d.ts +1 -0
  12. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/OverlappingStackedValueAxes.d.ts +1 -0
  13. package/dist/orbcharts-plugins-basic/src/noneData/plugins/Container.d.ts +0 -1
  14. package/dist/orbcharts-plugins-basic/src/noneData/plugins/Tooltip.d.ts +0 -3
  15. package/dist/orbcharts-plugins-basic/src/relationship/defaults.d.ts +5 -0
  16. package/dist/orbcharts-plugins-basic/src/relationship/index.d.ts +4 -0
  17. package/dist/orbcharts-plugins-basic/src/relationship/plugins/ForceDirected.d.ts +3 -0
  18. package/dist/orbcharts-plugins-basic/src/relationship/plugins/RelationshipLegend.d.ts +1 -0
  19. package/dist/orbcharts-plugins-basic/src/relationship/plugins/RelationshipTooltip.d.ts +1 -0
  20. package/dist/orbcharts-plugins-basic/src/relationship/relationshipObservables.d.ts +1 -0
  21. package/dist/orbcharts-plugins-basic/src/utils/commonUtils.d.ts +1 -1
  22. package/dist/orbcharts-plugins-basic/src/utils/orbchartsUtils.d.ts +1 -1
  23. package/dist/orbcharts-plugins-basic.es.js +8122 -7467
  24. package/dist/orbcharts-plugins-basic.umd.js +45 -32
  25. package/lib/core-types.ts +7 -7
  26. package/lib/core.ts +6 -6
  27. package/lib/plugins-basic-types.ts +6 -6
  28. package/package.json +44 -44
  29. package/src/base/BaseBars.ts +765 -765
  30. package/src/base/BaseBarsTriangle.ts +676 -676
  31. package/src/base/BaseDots.ts +464 -464
  32. package/src/base/BaseGroupAxis.ts +679 -679
  33. package/src/base/BaseLegend.ts +684 -684
  34. package/src/base/BaseLineAreas.ts +629 -629
  35. package/src/base/BaseLines.ts +706 -706
  36. package/src/base/{BaseBarStack.ts → BaseStackedBar.ts} +782 -782
  37. package/src/base/BaseTooltip.ts +385 -385
  38. package/src/base/BaseValueAxis.ts +583 -583
  39. package/src/base/types.ts +2 -2
  40. package/src/const.ts +30 -30
  41. package/src/grid/defaults.ts +246 -246
  42. package/src/grid/gridObservables.ts +554 -554
  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/GridZoom.ts +218 -218
  51. package/src/grid/plugins/GroupAux.ts +1103 -1103
  52. package/src/grid/plugins/GroupAxis.ts +97 -97
  53. package/src/grid/plugins/LineAreas.ts +65 -65
  54. package/src/grid/plugins/Lines.ts +59 -59
  55. package/src/grid/plugins/{BarStack.ts → StackedBar.ts} +64 -64
  56. package/src/grid/plugins/{ValueStackAxis.ts → StackedValueAxis.ts} +96 -96
  57. package/src/grid/plugins/ValueAxis.ts +94 -94
  58. package/src/index.ts +6 -10
  59. package/src/multiGrid/defaults.ts +224 -224
  60. package/src/multiGrid/index.ts +15 -15
  61. package/src/multiGrid/multiGridObservables.ts +49 -49
  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 +159 -159
  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/{MultiBarStack.ts → MultiStackedBar.ts} +106 -106
  71. package/src/multiGrid/plugins/{MultiValueStackAxis.ts → MultiStackedValueAxis.ts} +134 -134
  72. package/src/multiGrid/plugins/MultiValueAxis.ts +134 -134
  73. package/src/multiGrid/plugins/{OverlappingValueStackAxes.ts → OverlappingStackedValueAxes.ts} +299 -299
  74. package/src/multiGrid/plugins/OverlappingValueAxes.ts +300 -300
  75. package/src/multiValue/defaults.ts +166 -166
  76. package/src/multiValue/index.ts +8 -8
  77. package/src/multiValue/multiValueObservables.ts +297 -297
  78. package/src/multiValue/plugins/MultiValueLegend.ts +107 -107
  79. package/src/multiValue/plugins/MultiValueTooltip.ts +66 -66
  80. package/src/multiValue/plugins/Scatter.ts +426 -426
  81. package/src/multiValue/plugins/ScatterBubbles.ts +554 -554
  82. package/src/multiValue/plugins/XYAux.ts +681 -681
  83. package/src/multiValue/plugins/XYAxes.ts +684 -684
  84. package/src/multiValue/plugins/XYZoom.ts +299 -299
  85. package/src/noneData/defaults.ts +102 -102
  86. package/src/noneData/index.ts +3 -3
  87. package/src/noneData/plugins/Container.ts +28 -28
  88. package/src/noneData/plugins/Tooltip.ts +374 -374
  89. package/src/relationship/defaults.ts +113 -0
  90. package/src/relationship/index.ts +4 -0
  91. package/src/relationship/plugins/ForceDirected.ts +1148 -0
  92. package/src/relationship/plugins/RelationshipLegend.ts +100 -0
  93. package/src/relationship/plugins/RelationshipTooltip.ts +66 -0
  94. package/src/relationship/relationshipObservables.ts +50 -0
  95. package/src/series/defaults.ts +206 -206
  96. package/src/series/index.ts +9 -9
  97. package/src/series/plugins/Bubbles.ts +604 -603
  98. package/src/series/plugins/Pie.ts +623 -623
  99. package/src/series/plugins/PieEventTexts.ts +284 -283
  100. package/src/series/plugins/PieLabels.ts +640 -640
  101. package/src/series/plugins/Rose.ts +516 -516
  102. package/src/series/plugins/RoseLabels.ts +600 -600
  103. package/src/series/plugins/SeriesLegend.ts +107 -107
  104. package/src/series/plugins/SeriesTooltip.ts +66 -66
  105. package/src/series/seriesObservables.ts +145 -145
  106. package/src/series/seriesUtils.ts +51 -51
  107. package/src/tree/defaults.ts +78 -78
  108. package/src/tree/index.ts +4 -4
  109. package/src/tree/plugins/TreeLegend.ts +100 -100
  110. package/src/tree/plugins/TreeMap.ts +333 -333
  111. package/src/tree/plugins/TreeTooltip.ts +66 -66
  112. package/src/utils/commonUtils.ts +21 -21
  113. package/src/utils/d3Graphics.ts +174 -174
  114. package/src/utils/d3Utils.ts +74 -74
  115. package/src/utils/observables.ts +14 -14
  116. package/src/utils/orbchartsUtils.ts +115 -115
  117. package/tsconfig.base.json +13 -13
  118. package/tsconfig.json +2 -2
  119. package/vite.config.js +22 -22
  120. package/dist/orbcharts-plugins-basic/src/grid/plugins/BarStack.d.ts +0 -1
  121. package/dist/orbcharts-plugins-basic/src/grid/plugins/ValueStackAxis.d.ts +0 -1
  122. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiBarStack.d.ts +0 -1
  123. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiValueStackAxis.d.ts +0 -1
  124. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/OverlappingValueStackAxes.d.ts +0 -1
  125. /package/dist/orbcharts-plugins-basic/src/relationship/plugins/{Relationship.d.ts → ForceDirectedBubbles.d.ts} +0 -0
  126. /package/src/relationship/plugins/{Relationship.ts → ForceDirectedBubbles.ts} +0 -0
@@ -1,604 +1,605 @@
1
- import * as d3 from 'd3'
2
- import {
3
- combineLatest,
4
- map,
5
- switchMap,
6
- first,
7
- takeUntil,
8
- Subject,
9
- Observable,
10
- distinctUntilChanged,
11
- shareReplay} from 'rxjs'
12
- import type { DefinePluginConfig } from '../../../lib/core-types'
13
- import type {
14
- ChartParams,
15
- DatumValue,
16
- DataSeries,
17
- EventName,
18
- ComputedDataSeries,
19
- ComputedDatumSeries,
20
- ContainerPosition } from '../../../lib/core-types'
21
- import {
22
- defineSeriesPlugin } from '../../../lib/core'
23
- import type { BubblesParams, ArcScaleType } from '../../../lib/plugins-basic-types'
24
- import { DEFAULT_BUBBLES_PARAMS } from '../defaults'
25
- import { renderCircleText } from '../../utils/d3Graphics'
26
- import { LAYER_INDEX_OF_GRAPHIC } from '../../const'
27
-
28
- interface BubblesDatum extends ComputedDatumSeries {
29
- x: number
30
- y: number
31
- r: number
32
- _originR: number // 紀錄變化前的r
33
- }
34
-
35
- type BubblesSimulationDatum = BubblesDatum & d3.SimulationNodeDatum
36
-
37
- const pluginName = 'Bubbles'
38
-
39
- const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_BUBBLES_PARAMS> = {
40
- name: pluginName,
41
- defaultParams: DEFAULT_BUBBLES_PARAMS,
42
- layerIndex: LAYER_INDEX_OF_GRAPHIC,
43
- validator: (params, { validateColumns }) => {
44
- const result = validateColumns(params, {
45
- force: {
46
- toBeTypes: ['object']
47
- },
48
- bubbleText: {
49
- toBeTypes: ['object']
50
- },
51
- arcScaleType: {
52
- toBe: '"area" | "radius"',
53
- test: (value) => value === 'area' || value === 'radius'
54
- }
55
- })
56
- if (params.force) {
57
- const forceResult = validateColumns(params.force, {
58
- velocityDecay: {
59
- toBeTypes: ['number']
60
- },
61
- collisionSpacing: {
62
- toBeTypes: ['number']
63
- },
64
- strength: {
65
- toBeTypes: ['number']
66
- },
67
- })
68
- if (forceResult.status === 'error') {
69
- return forceResult
70
- }
71
- }
72
- if (params.bubbleText) {
73
- const bubbleTextResult = validateColumns(params.bubbleText, {
74
- fillRate: {
75
- toBeTypes: ['number']
76
- },
77
- lineHeight: {
78
- toBeTypes: ['number']
79
- },
80
- lineLengthMin: {
81
- toBeTypes: ['number']
82
- },
83
- })
84
- if (bubbleTextResult.status === 'error') {
85
- return bubbleTextResult
86
- }
87
- }
88
- return result
89
- }
90
- }
91
-
92
- let force: d3.Simulation<d3.SimulationNodeDatum, undefined> | undefined
93
-
94
- function makeForce (bubblesSelection: d3.Selection<SVGGElement, any, any, any>, fullParams: BubblesParams) {
95
- return d3.forceSimulation()
96
- .velocityDecay(fullParams.force!.velocityDecay!)
97
- // .alphaDecay(0.2)
98
- .force(
99
- "collision",
100
- d3.forceCollide()
101
- .radius(d => {
102
- // @ts-ignore
103
- return d.r + fullParams.force!.collisionSpacing
104
- })
105
- // .strength(0.01)
106
- )
107
- .force("charge", d3.forceManyBody().strength((d) => {
108
- // @ts-ignore
109
- return - Math.pow(d.r, 2.0) * fullParams.force!.strength
110
- }))
111
- // .force("x", d3.forceX().strength(forceStrength).x(this.graphicWidth / 2))
112
- // .force("y", d3.forceY().strength(forceStrength).y(this.graphicHeight / 2))
113
- .on("tick", () => {
114
- // if (!bubblesSelection) {
115
- // return
116
- // }
117
- bubblesSelection
118
- .attr("transform", (d) => {
119
- return `translate(${d.x},${d.y})`
120
- })
121
- // .attr("cx", (d) => d.x)
122
- // .attr("cy", (d) => d.y)
123
- })
124
- }
125
-
126
-
127
- // // 計算最大泡泡的半徑
128
- // function getMaxR ({ data, totalR, maxValue, avgValue }: {
129
- // data: DatumValue[]
130
- // totalR: number
131
- // maxValue: number
132
- // avgValue: number
133
- // }) {
134
- // // 平均r(假想是正方型來計算的,比如說大正方型裡有4個正方型,則 r = width/Math.sqrt(4)/2)
135
- // const avgR = totalR / Math.sqrt(data.length)
136
- // const avgSize = avgR * avgR * Math.PI
137
- // const sizeRate = avgSize / avgValue
138
- // const maxSize = maxValue * sizeRate
139
- // const maxR = Math.pow(maxSize / Math.PI, 0.5)
140
-
141
- // const modifier = 0.785 // @Q@ 因為以下公式是假設泡泡是正方型來計算,所以畫出來的圖會偏大一些,這個數值是用來修正用的
142
- // return maxR * modifier
143
- // }
144
-
145
- function createBubblesData ({ visibleComputedLayoutData, LastBubbleDataMap, graphicWidth, graphicHeight, SeriesContainerPositionMap, scaleType }: {
146
- visibleComputedLayoutData: ComputedDataSeries
147
- LastBubbleDataMap: Map<string, BubblesDatum>
148
- graphicWidth: number
149
- graphicHeight: number
150
- SeriesContainerPositionMap: Map<string, ContainerPosition>
151
- scaleType: ArcScaleType
152
- // highlightIds: string[]
153
- }): BubblesDatum[] {
154
- // 虛擬大圓(所有小圓聚合起來的大圓)的半徑
155
- const totalR = Math.min(...[graphicWidth, graphicHeight]) / 2
156
-
157
- const data = visibleComputedLayoutData.flat()
158
-
159
- const totalValue = data.reduce((acc, current) => acc + current.value, 0)
160
-
161
- // 半徑比例尺
162
- const radiusScale = d3.scalePow()
163
- .domain([0, totalValue])
164
- .range([0, totalR])
165
- .exponent(scaleType === 'area'
166
- ? 0.5 // 數值映射面積(0.5為取平方根)
167
- : 1 // 數值映射半徑
168
- )
169
-
170
- // 縮放比例 - 確保多個小圓的總面積等於大圓的面積
171
- const scaleFactor = scaleType === 'area'
172
- ? 1
173
- // 當數值映射半徑時,多個小圓的總面積會小於大圓的面積,所以要計算縮放比例
174
- : (() => {
175
- const totalArea = totalR * totalR * Math.PI
176
- return Math.sqrt(totalArea / d3.sum(data, d => Math.PI * Math.pow(radiusScale(d.value), 2)))
177
- })()
178
-
179
- // 調整係數 - 因為圓和圓之間的空隙造成聚合起來的大圓會略大,所以稍作微調
180
- const adjustmentFactor = 0.9
181
-
182
- return data.map((_d) => {
183
- const d: BubblesDatum = _d as BubblesDatum
184
-
185
- const existDatum = LastBubbleDataMap.get(d.id)
186
-
187
- if (existDatum) {
188
- // 使用現有的座標
189
- d.x = existDatum.x
190
- d.y = existDatum.y
191
- } else {
192
- const seriesContainerPosition = SeriesContainerPositionMap.get(d.seriesLabel)!
193
- d.x = Math.random() * seriesContainerPosition.width
194
- d.y = Math.random() * seriesContainerPosition.height
195
- }
196
- const r = radiusScale!(d.value ?? 0)! * scaleFactor * adjustmentFactor
197
- d.r = r
198
- d._originR = r
199
-
200
- return d
201
- })
202
- }
203
-
204
- function renderBubbles ({ selection, bubblesData, fullParams, sumSeries }: {
205
- selection: d3.Selection<SVGGElement, any, any, any>
206
- bubblesData: BubblesDatum[]
207
- fullParams: BubblesParams
208
- sumSeries: boolean
209
- }) {
210
- const bubblesSelection = selection.selectAll<SVGGElement, BubblesDatum>("g")
211
- .data(bubblesData, (d) => d.id)
212
- .join(
213
- enter => {
214
- const enterSelection = enter
215
- .append('g')
216
- .attr('cursor', 'pointer')
217
- .attr('font-size', 12)
218
- .style('fill', '#ffffff')
219
- .attr("text-anchor", "middle")
220
-
221
- enterSelection
222
- .append("circle")
223
- .attr("class", "node")
224
- .attr("cx", 0)
225
- .attr("cy", 0)
226
- // .attr("r", 1e-6)
227
- .attr('fill', (d) => d.color)
228
- // .transition()
229
- // .duration(500)
230
-
231
- enterSelection
232
- .append('text')
233
- .style('opacity', 0.8)
234
- .attr('pointer-events', 'none')
235
-
236
- return enterSelection
237
- },
238
- update => {
239
- return update
240
- },
241
- exit => {
242
- return exit
243
- .remove()
244
- }
245
- )
246
- .attr("transform", (d) => {
247
- return `translate(${d.x},${d.y})`
248
- })
249
-
250
- // 泡泡文字要使用的的資料欄位
251
- const textDataColumn = sumSeries ? 'seriesLabel' : 'label'// 如果有合併series則使用seriesLabel
252
-
253
- bubblesSelection.select('circle')
254
- .transition()
255
- .duration(200)
256
- .attr("r", (d) => d.r)
257
- .attr('fill', (d) => d.color)
258
- bubblesSelection
259
- .each((d,i,g) => {
260
- const gSelection = d3.select(g[i])
261
- let breakAll = true
262
- if (d[textDataColumn].length <= fullParams.bubbleText.lineLengthMin) {
263
- breakAll = false
264
- }
265
- gSelection.call(renderCircleText, {
266
- text: d[textDataColumn],
267
- radius: d.r * fullParams.bubbleText.fillRate,
268
- lineHeight: fullParams.bubbleText.lineHeight,
269
- isBreakAll: breakAll
270
- })
271
-
272
- })
273
-
274
- return bubblesSelection
275
- }
276
-
277
- function setHighlightData ({ data, highlightRIncrease, highlightIds }: {
278
- data: BubblesDatum[]
279
- // fullParams: BubblesParams
280
- highlightRIncrease: number
281
- highlightIds: string[]
282
- }) {
283
- if (highlightRIncrease == 0) {
284
- return
285
- }
286
- if (!highlightIds.length) {
287
- data.forEach(d => d.r = d._originR)
288
- return
289
- }
290
- data.forEach(d => {
291
- if (highlightIds.includes(d.id)) {
292
- d.r = d._originR + highlightRIncrease
293
- } else {
294
- d.r = d._originR
295
- }
296
- })
297
- }
298
-
299
- function drag (): d3.DragBehavior<Element, unknown, unknown> {
300
- return d3.drag()
301
- .on("start", (event, d: any) => {
302
- if (!event.active) {
303
- force!.alpha(1).restart()
304
- }
305
- d.fx = d.x
306
- d.fy = d.y
307
- })
308
- .on("drag", (event, d: any) => {
309
- if (!event.active) {
310
- force!.alphaTarget(0)
311
- }
312
- d.fx = event.x
313
- d.fy = event.y
314
- })
315
- .on("end", (event, d: any) => {
316
- d.fx = null
317
- d.fy = null
318
- })
319
- }
320
-
321
-
322
- // private nodeTypePos (d: any) {
323
- // console.log(d)
324
- // console.log(this.TypeCenters.get(d.type)!)
325
- // const typeCenter = this.TypeCenters.get(d.type)!
326
- // return typeCenter ? typeCenter.x : 0
327
- // }
328
-
329
- function groupBubbles ({ fullParams, SeriesContainerPositionMap }: {
330
- fullParams: BubblesParams
331
- // graphicWidth: number
332
- // graphicHeight: number
333
- SeriesContainerPositionMap: Map<string, ContainerPosition>
334
- }) {
335
- // console.log('groupBubbles')
336
- force!
337
- // .force('x', d3.forceX().strength(fullParams.force.strength).x(graphicWidth / 2))
338
- // .force('y', d3.forceY().strength(fullParams.force.strength).y(graphicHeight / 2))
339
- .force('x', d3.forceX().strength(fullParams.force.strength).x((data: BubblesSimulationDatum) => {
340
- return SeriesContainerPositionMap.get(data.seriesLabel)!.centerX
341
- }))
342
- .force('y', d3.forceY().strength(fullParams.force.strength).y((data: BubblesSimulationDatum) => {
343
- return SeriesContainerPositionMap.get(data.seriesLabel)!.centerY
344
- }))
345
-
346
- force!.alpha(1).restart()
347
- }
348
-
349
- function highlight ({ bubblesSelection, highlightIds, fullChartParams }: {
350
- bubblesSelection: d3.Selection<SVGGElement, BubblesDatum, any, any>
351
- fullChartParams: ChartParams
352
- highlightIds: string[]
353
- }) {
354
- bubblesSelection.interrupt('highlight')
355
-
356
- if (!highlightIds.length) {
357
- bubblesSelection
358
- .transition('highlight')
359
- .style('opacity', 1)
360
- return
361
- }
362
-
363
- bubblesSelection.each((d, i, n) => {
364
- const segment = d3.select(n[i])
365
-
366
- if (highlightIds.includes(d.id)) {
367
- segment
368
- .style('opacity', 1)
369
- .transition('highlight')
370
- .ease(d3.easeElastic)
371
- .duration(500)
372
- } else {
373
- // 取消放大
374
- segment
375
- .style('opacity', fullChartParams.styles.unhighlightedOpacity)
376
- }
377
- })
378
- }
379
-
380
-
381
- export const Bubbles = defineSeriesPlugin(pluginConfig)(({ selection, name, observer, subject }) => {
382
-
383
- const destroy$ = new Subject()
384
-
385
- // 紀錄前一次bubble data
386
- let LastBubbleDataMap: Map<string, BubblesDatum> = new Map()
387
-
388
-
389
- const sumSeries$ = observer.fullDataFormatter$.pipe(
390
- map(d => d.sumSeries),
391
- distinctUntilChanged()
392
- )
393
-
394
- const scaleType$ = observer.fullParams$.pipe(
395
- takeUntil(destroy$),
396
- map(d => d.arcScaleType),
397
- distinctUntilChanged()
398
- )
399
-
400
- const bubblesData$ = combineLatest({
401
- layout: observer.layout$,
402
- SeriesContainerPositionMap: observer.SeriesContainerPositionMap$,
403
- visibleComputedLayoutData: observer.visibleComputedLayoutData$,
404
- scaleType: scaleType$
405
- }).pipe(
406
- takeUntil(destroy$),
407
- switchMap(async (d) => d),
408
- map(data => {
409
- // console.log(data.visibleComputedLayoutData)
410
- return createBubblesData({
411
- visibleComputedLayoutData: data.visibleComputedLayoutData,
412
- LastBubbleDataMap,
413
- graphicWidth: data.layout.width,
414
- graphicHeight: data.layout.height,
415
- SeriesContainerPositionMap: data.SeriesContainerPositionMap,
416
- scaleType: data.scaleType
417
- })
418
- }),
419
- shareReplay(1)
420
- )
421
-
422
- // 紀錄前一次bubble data
423
- bubblesData$.subscribe(d => {
424
- LastBubbleDataMap = new Map(d.map(_d => [_d.id, _d])) // key: id, value: datum
425
- })
426
-
427
- const highlightTarget$ = observer.fullChartParams$.pipe(
428
- takeUntil(destroy$),
429
- map(d => d.highlightTarget),
430
- distinctUntilChanged()
431
- )
432
-
433
- const bubblesSelection$ = combineLatest({
434
- bubblesData: bubblesData$,
435
- fullParams: observer.fullParams$,
436
- SeriesContainerPositionMap: observer.SeriesContainerPositionMap$,
437
- sumSeries: sumSeries$
438
- }).pipe(
439
- takeUntil(destroy$),
440
- switchMap(async (d) => d),
441
- map(data => {
442
- if (force) {
443
- force.stop()
444
- }
445
-
446
- const bubblesSelection = renderBubbles({
447
- selection,
448
- bubblesData: data.bubblesData,
449
- fullParams: data.fullParams,
450
- sumSeries: data.sumSeries
451
- })
452
-
453
- force = makeForce(bubblesSelection, data.fullParams)
454
-
455
- force.nodes(data.bubblesData)
456
-
457
- groupBubbles({
458
- fullParams: data.fullParams,
459
- SeriesContainerPositionMap: data.SeriesContainerPositionMap
460
- // graphicWidth: data.layout.width,
461
- // graphicHeight: data.layout.height
462
- })
463
-
464
- // setTimeout(() => {
465
- // force!.alphaTarget(0)
466
- // force!.alpha(1).restart()
467
- // }, 2000)
468
-
469
- return bubblesSelection
470
- })
471
- )
472
-
473
- combineLatest({
474
- bubblesSelection: bubblesSelection$,
475
- computedData: observer.computedData$,
476
- SeriesDataMap: observer.SeriesDataMap$,
477
- highlightTarget: highlightTarget$,
478
- }).pipe(
479
- takeUntil(destroy$),
480
- switchMap(async (d) => d)
481
- ).subscribe(data => {
482
-
483
- data.bubblesSelection
484
- .on('mouseover', (event, datum) => {
485
- // this.tooltip!.setDatum({
486
- // data: d,
487
- // x: d3.event.clientX,
488
- // y: d3.event.clientY
489
- // })
490
-
491
- subject.event$.next({
492
- type: 'series',
493
- eventName: 'mouseover',
494
- pluginName: name,
495
- highlightTarget: data.highlightTarget,
496
- datum,
497
- series: data.SeriesDataMap.get(datum.seriesLabel)!,
498
- seriesIndex: datum.seriesIndex,
499
- seriesLabel: datum.seriesLabel,
500
- event,
501
- data: data.computedData
502
- })
503
- })
504
- .on('mousemove', (event, datum) => {
505
- // this.tooltip!.setDatum({
506
- // x: d3.event.clientX,
507
- // y: d3.event.clientY
508
- // })
509
-
510
- subject.event$.next({
511
- type: 'series',
512
- eventName: 'mousemove',
513
- pluginName: name,
514
- highlightTarget: data.highlightTarget,
515
- datum,
516
- series: data.SeriesDataMap.get(datum.seriesLabel)!,
517
- seriesIndex: datum.seriesIndex,
518
- seriesLabel: datum.seriesLabel,
519
- event,
520
- data: data.computedData
521
- })
522
- })
523
- .on('mouseout', (event, datum) => {
524
- // this.tooltip!.remove()
525
-
526
- subject.event$.next({
527
- type: 'series',
528
- eventName: 'mouseout',
529
- pluginName: name,
530
- highlightTarget: data.highlightTarget,
531
- datum,
532
- series: data.SeriesDataMap.get(datum.seriesLabel)!,
533
- seriesIndex: datum.seriesIndex,
534
- seriesLabel: datum.seriesLabel,
535
- event,
536
- data: data.computedData
537
- })
538
- })
539
- .on('click', (event, datum) => {
540
-
541
- subject.event$.next({
542
- type: 'series',
543
- eventName: 'click',
544
- pluginName: name,
545
- highlightTarget: data.highlightTarget,
546
- datum,
547
- series: data.SeriesDataMap.get(datum.seriesLabel)!,
548
- seriesIndex: datum.seriesIndex,
549
- seriesLabel: datum.seriesLabel,
550
- event,
551
- data: data.computedData
552
- })
553
- })
554
- .call(drag() as any)
555
-
556
-
557
- })
558
-
559
- combineLatest({
560
- bubblesSelection: bubblesSelection$,
561
- bubblesData: bubblesData$,
562
- highlight: observer.seriesHighlight$.pipe(
563
- map(data => data.map(d => d.id))
564
- ),
565
- fullChartParams: observer.fullChartParams$,
566
- fullParams: observer.fullParams$,
567
- sumSeries: sumSeries$,
568
- // layout: observer.layout$,
569
- SeriesContainerPositionMap: observer.SeriesContainerPositionMap$,
570
- }).pipe(
571
- takeUntil(destroy$),
572
- switchMap(async d => d)
573
- ).subscribe(data => {
574
- highlight({
575
- bubblesSelection: data.bubblesSelection,
576
- highlightIds: data.highlight,
577
- fullChartParams: data.fullChartParams
578
- })
579
-
580
- // if (data.fullParams.highlightRIncrease) {
581
- // setHighlightData ({
582
- // data: data.bubblesData,
583
- // highlightRIncrease: data.fullParams.highlightRIncrease,
584
- // highlightIds: data.highlight
585
- // })
586
- // data.bubblesSelection.select('circle')
587
- // // .transition()
588
- // // .duration(200)
589
- // .attr("r", (d) => d.r)
590
-
591
- // force!.nodes(data.bubblesData)
592
-
593
- // groupBubbles({
594
- // fullParams: data.fullParams,
595
- // SeriesContainerPositionMap: data.SeriesContainerPositionMap
596
- // })
597
- // }
598
-
599
- })
600
-
601
- return () => {
602
- destroy$.next(undefined)
603
- }
1
+ import * as d3 from 'd3'
2
+ import {
3
+ combineLatest,
4
+ map,
5
+ switchMap,
6
+ first,
7
+ takeUntil,
8
+ Subject,
9
+ Observable,
10
+ distinctUntilChanged,
11
+ shareReplay} from 'rxjs'
12
+ import type { DefinePluginConfig } from '../../../lib/core-types'
13
+ import type {
14
+ ChartParams,
15
+ DatumValue,
16
+ DataSeries,
17
+ EventName,
18
+ ComputedDataSeries,
19
+ ComputedDatumSeries,
20
+ ContainerPosition } from '../../../lib/core-types'
21
+ import {
22
+ defineSeriesPlugin } from '../../../lib/core'
23
+ import type { BubblesParams, ArcScaleType } from '../../../lib/plugins-basic-types'
24
+ import { DEFAULT_BUBBLES_PARAMS } from '../defaults'
25
+ import { renderCircleText } from '../../utils/d3Graphics'
26
+ import { LAYER_INDEX_OF_GRAPHIC } from '../../const'
27
+
28
+ interface BubblesDatum extends ComputedDatumSeries {
29
+ x: number
30
+ y: number
31
+ r: number
32
+ _originR: number // 紀錄變化前的r
33
+ }
34
+
35
+ type BubblesSimulationDatum = BubblesDatum & d3.SimulationNodeDatum
36
+
37
+ const pluginName = 'Bubbles'
38
+
39
+ const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_BUBBLES_PARAMS> = {
40
+ name: pluginName,
41
+ defaultParams: DEFAULT_BUBBLES_PARAMS,
42
+ layerIndex: LAYER_INDEX_OF_GRAPHIC,
43
+ validator: (params, { validateColumns }) => {
44
+ const result = validateColumns(params, {
45
+ force: {
46
+ toBeTypes: ['object']
47
+ },
48
+ bubbleLabel: {
49
+ toBeTypes: ['object']
50
+ },
51
+ arcScaleType: {
52
+ toBe: '"area" | "radius"',
53
+ test: (value) => value === 'area' || value === 'radius'
54
+ }
55
+ })
56
+ if (params.force) {
57
+ const forceResult = validateColumns(params.force, {
58
+ velocityDecay: {
59
+ toBeTypes: ['number']
60
+ },
61
+ collisionSpacing: {
62
+ toBeTypes: ['number']
63
+ },
64
+ strength: {
65
+ toBeTypes: ['number']
66
+ },
67
+ })
68
+ if (forceResult.status === 'error') {
69
+ return forceResult
70
+ }
71
+ }
72
+ if (params.bubbleLabel) {
73
+ const bubbleLabelResult = validateColumns(params.bubbleLabel, {
74
+ fillRate: {
75
+ toBeTypes: ['number']
76
+ },
77
+ lineHeight: {
78
+ toBeTypes: ['number']
79
+ },
80
+ lineLengthMin: {
81
+ toBeTypes: ['number']
82
+ },
83
+ })
84
+ if (bubbleLabelResult.status === 'error') {
85
+ return bubbleLabelResult
86
+ }
87
+ }
88
+ return result
89
+ }
90
+ }
91
+
92
+ let force: d3.Simulation<d3.SimulationNodeDatum, undefined> | undefined
93
+
94
+ function makeForce (bubblesSelection: d3.Selection<SVGGElement, BubblesDatum, any, any>, fullParams: BubblesParams) {
95
+ return d3.forceSimulation()
96
+ .velocityDecay(fullParams.force!.velocityDecay!)
97
+ // .alphaDecay(0.2)
98
+ .force(
99
+ "collision",
100
+ d3.forceCollide()
101
+ .radius((d: d3.SimulationNodeDatum & BubblesDatum) => {
102
+ return d.r + fullParams.force!.collisionSpacing
103
+ })
104
+ // .strength(0.01)
105
+ )
106
+ .force("charge", d3.forceManyBody().strength((d: d3.SimulationNodeDatum & BubblesDatum) => {
107
+ return - Math.pow(d.r, 2.0) * fullParams.force!.strength
108
+ }))
109
+ // .force("charge", d3.forceManyBody().strength(-2000))
110
+ // .force("collision", d3.forceCollide(60).strength(1)) // @Q@ 60為泡泡的R,暫時是先寫死的
111
+ // .force("x", d3.forceX().strength(forceStrength).x(this.graphicWidth / 2))
112
+ // .force("y", d3.forceY().strength(forceStrength).y(this.graphicHeight / 2))
113
+ .on("tick", () => {
114
+ // if (!bubblesSelection) {
115
+ // return
116
+ // }
117
+ bubblesSelection
118
+ .attr("transform", (d) => {
119
+ return `translate(${d.x},${d.y})`
120
+ })
121
+ // .attr("cx", (d) => d.x)
122
+ // .attr("cy", (d) => d.y)
123
+ })
124
+
125
+ }
126
+
127
+
128
+ // // 計算最大泡泡的半徑
129
+ // function getMaxR ({ data, totalR, maxValue, avgValue }: {
130
+ // data: DatumValue[]
131
+ // totalR: number
132
+ // maxValue: number
133
+ // avgValue: number
134
+ // }) {
135
+ // // 平均r(假想是正方型來計算的,比如說大正方型裡有4個正方型,則 r = width/Math.sqrt(4)/2)
136
+ // const avgR = totalR / Math.sqrt(data.length)
137
+ // const avgSize = avgR * avgR * Math.PI
138
+ // const sizeRate = avgSize / avgValue
139
+ // const maxSize = maxValue * sizeRate
140
+ // const maxR = Math.pow(maxSize / Math.PI, 0.5)
141
+
142
+ // const modifier = 0.785 // @Q@ 因為以下公式是假設泡泡是正方型來計算,所以畫出來的圖會偏大一些,這個數值是用來修正用的
143
+ // return maxR * modifier
144
+ // }
145
+
146
+ function createBubblesData ({ visibleComputedLayoutData, LastBubbleDataMap, graphicWidth, graphicHeight, SeriesContainerPositionMap, scaleType }: {
147
+ visibleComputedLayoutData: ComputedDataSeries
148
+ LastBubbleDataMap: Map<string, BubblesDatum>
149
+ graphicWidth: number
150
+ graphicHeight: number
151
+ SeriesContainerPositionMap: Map<string, ContainerPosition>
152
+ scaleType: ArcScaleType
153
+ // highlightIds: string[]
154
+ }): BubblesDatum[] {
155
+ // 虛擬大圓(所有小圓聚合起來的大圓)的半徑
156
+ const totalR = Math.min(...[graphicWidth, graphicHeight]) / 2
157
+
158
+ const data = visibleComputedLayoutData.flat()
159
+
160
+ const totalValue = data.reduce((acc, current) => acc + current.value, 0)
161
+
162
+ // 半徑比例尺
163
+ const radiusScale = d3.scalePow()
164
+ .domain([0, totalValue])
165
+ .range([0, totalR])
166
+ .exponent(scaleType === 'area'
167
+ ? 0.5 // 數值映射面積(0.5為取平方根)
168
+ : 1 // 數值映射半徑
169
+ )
170
+
171
+ // 縮放比例 - 確保多個小圓的總面積等於大圓的面積
172
+ const scaleFactor = scaleType === 'area'
173
+ ? 1
174
+ // 當數值映射半徑時,多個小圓的總面積會小於大圓的面積,所以要計算縮放比例
175
+ : (() => {
176
+ const totalArea = totalR * totalR * Math.PI
177
+ return Math.sqrt(totalArea / d3.sum(data, d => Math.PI * Math.pow(radiusScale(d.value), 2)))
178
+ })()
179
+
180
+ // 調整係數 - 因為圓和圓之間的空隙造成聚合起來的大圓會略大,所以稍作微調
181
+ const adjustmentFactor = 0.9
182
+
183
+ return data.map((_d) => {
184
+ const d: BubblesDatum = _d as BubblesDatum
185
+
186
+ const existDatum = LastBubbleDataMap.get(d.id)
187
+
188
+ if (existDatum) {
189
+ // 使用現有的座標
190
+ d.x = existDatum.x
191
+ d.y = existDatum.y
192
+ } else {
193
+ const seriesContainerPosition = SeriesContainerPositionMap.get(d.seriesLabel)!
194
+ d.x = Math.random() * seriesContainerPosition.width
195
+ d.y = Math.random() * seriesContainerPosition.height
196
+ }
197
+ const r = radiusScale!(d.value ?? 0)! * scaleFactor * adjustmentFactor
198
+ d.r = r
199
+ d._originR = r
200
+
201
+ return d
202
+ })
203
+ }
204
+
205
+ function renderBubbles ({ selection, bubblesData, fullParams, sumSeries }: {
206
+ selection: d3.Selection<SVGGElement, any, any, any>
207
+ bubblesData: BubblesDatum[]
208
+ fullParams: BubblesParams
209
+ sumSeries: boolean
210
+ }) {
211
+ const bubblesSelection = selection.selectAll<SVGGElement, BubblesDatum>("g")
212
+ .data(bubblesData, (d) => d.id)
213
+ .join(
214
+ enter => {
215
+ const enterSelection = enter
216
+ .append('g')
217
+ .attr('cursor', 'pointer')
218
+ .attr('font-size', 12)
219
+ .style('fill', '#ffffff')
220
+ .attr("text-anchor", "middle")
221
+
222
+ enterSelection
223
+ .append("circle")
224
+ .attr("class", "node")
225
+ .attr("cx", 0)
226
+ .attr("cy", 0)
227
+ // .attr("r", 1e-6)
228
+ .attr('fill', (d) => d.color)
229
+ // .transition()
230
+ // .duration(500)
231
+
232
+ enterSelection
233
+ .append('text')
234
+ .style('opacity', 0.8)
235
+ .attr('pointer-events', 'none')
236
+
237
+ return enterSelection
238
+ },
239
+ update => {
240
+ return update
241
+ },
242
+ exit => {
243
+ return exit
244
+ .remove()
245
+ }
246
+ )
247
+ .attr("transform", (d) => {
248
+ return `translate(${d.x},${d.y})`
249
+ })
250
+
251
+ // 泡泡文字要使用的的資料欄位
252
+ const textDataColumn = sumSeries ? 'seriesLabel' : 'label'// 如果有合併series則使用seriesLabel
253
+
254
+ bubblesSelection.select('circle')
255
+ .transition()
256
+ .duration(200)
257
+ .attr("r", (d) => d.r)
258
+ .attr('fill', (d) => d.color)
259
+ bubblesSelection
260
+ .each((d,i,g) => {
261
+ const gSelection = d3.select(g[i])
262
+ let breakAll = true
263
+ if (d[textDataColumn].length <= fullParams.bubbleLabel.lineLengthMin) {
264
+ breakAll = false
265
+ }
266
+ gSelection.call(renderCircleText, {
267
+ text: d[textDataColumn],
268
+ radius: d.r * fullParams.bubbleLabel.fillRate,
269
+ lineHeight: fullParams.bubbleLabel.lineHeight,
270
+ isBreakAll: breakAll
271
+ })
272
+
273
+ })
274
+
275
+ return bubblesSelection
276
+ }
277
+
278
+ function setHighlightData ({ data, highlightRIncrease, highlightIds }: {
279
+ data: BubblesDatum[]
280
+ // fullParams: BubblesParams
281
+ highlightRIncrease: number
282
+ highlightIds: string[]
283
+ }) {
284
+ if (highlightRIncrease == 0) {
285
+ return
286
+ }
287
+ if (!highlightIds.length) {
288
+ data.forEach(d => d.r = d._originR)
289
+ return
290
+ }
291
+ data.forEach(d => {
292
+ if (highlightIds.includes(d.id)) {
293
+ d.r = d._originR + highlightRIncrease
294
+ } else {
295
+ d.r = d._originR
296
+ }
297
+ })
298
+ }
299
+
300
+ function drag (): d3.DragBehavior<Element, unknown, unknown> {
301
+ return d3.drag()
302
+ .on("start", (event, d: any) => {
303
+ if (!event.active) {
304
+ force!.alpha(1).restart()
305
+ }
306
+ d.fx = d.x
307
+ d.fy = d.y
308
+ })
309
+ .on("drag", (event, d: any) => {
310
+ if (!event.active) {
311
+ force!.alphaTarget(0)
312
+ }
313
+ d.fx = event.x
314
+ d.fy = event.y
315
+ })
316
+ .on("end", (event, d: any) => {
317
+ d.fx = null
318
+ d.fy = null
319
+ })
320
+ }
321
+
322
+
323
+ // private nodeTypePos (d: any) {
324
+ // console.log(d)
325
+ // console.log(this.TypeCenters.get(d.type)!)
326
+ // const typeCenter = this.TypeCenters.get(d.type)!
327
+ // return typeCenter ? typeCenter.x : 0
328
+ // }
329
+
330
+ function groupBubbles ({ fullParams, SeriesContainerPositionMap }: {
331
+ fullParams: BubblesParams
332
+ // graphicWidth: number
333
+ // graphicHeight: number
334
+ SeriesContainerPositionMap: Map<string, ContainerPosition>
335
+ }) {
336
+ // console.log('groupBubbles')
337
+ force!
338
+ // .force('x', d3.forceX().strength(fullParams.force.strength).x(graphicWidth / 2))
339
+ // .force('y', d3.forceY().strength(fullParams.force.strength).y(graphicHeight / 2))
340
+ .force('x', d3.forceX().strength(fullParams.force.strength).x((data: BubblesSimulationDatum) => {
341
+ return SeriesContainerPositionMap.get(data.seriesLabel)!.centerX
342
+ }))
343
+ .force('y', d3.forceY().strength(fullParams.force.strength).y((data: BubblesSimulationDatum) => {
344
+ return SeriesContainerPositionMap.get(data.seriesLabel)!.centerY
345
+ }))
346
+
347
+ force!.alpha(1).restart()
348
+ }
349
+
350
+ function highlight ({ bubblesSelection, highlightIds, fullChartParams }: {
351
+ bubblesSelection: d3.Selection<SVGGElement, BubblesDatum, any, any>
352
+ fullChartParams: ChartParams
353
+ highlightIds: string[]
354
+ }) {
355
+ bubblesSelection.interrupt('highlight')
356
+
357
+ if (!highlightIds.length) {
358
+ bubblesSelection
359
+ .transition('highlight')
360
+ .style('opacity', 1)
361
+ return
362
+ }
363
+
364
+ bubblesSelection.each((d, i, n) => {
365
+ const segment = d3.select(n[i])
366
+
367
+ if (highlightIds.includes(d.id)) {
368
+ segment
369
+ .style('opacity', 1)
370
+ .transition('highlight')
371
+ .ease(d3.easeElastic)
372
+ .duration(500)
373
+ } else {
374
+ // 取消放大
375
+ segment
376
+ .style('opacity', fullChartParams.styles.unhighlightedOpacity)
377
+ }
378
+ })
379
+ }
380
+
381
+
382
+ export const Bubbles = defineSeriesPlugin(pluginConfig)(({ selection, name, observer, subject }) => {
383
+
384
+ const destroy$ = new Subject()
385
+
386
+ // 紀錄前一次bubble data
387
+ let LastBubbleDataMap: Map<string, BubblesDatum> = new Map()
388
+
389
+
390
+ const sumSeries$ = observer.fullDataFormatter$.pipe(
391
+ map(d => d.sumSeries),
392
+ distinctUntilChanged()
393
+ )
394
+
395
+ const scaleType$ = observer.fullParams$.pipe(
396
+ takeUntil(destroy$),
397
+ map(d => d.arcScaleType),
398
+ distinctUntilChanged()
399
+ )
400
+
401
+ const bubblesData$ = combineLatest({
402
+ layout: observer.layout$,
403
+ SeriesContainerPositionMap: observer.SeriesContainerPositionMap$,
404
+ visibleComputedLayoutData: observer.visibleComputedLayoutData$,
405
+ scaleType: scaleType$
406
+ }).pipe(
407
+ takeUntil(destroy$),
408
+ switchMap(async (d) => d),
409
+ map(data => {
410
+ // console.log(data.visibleComputedLayoutData)
411
+ return createBubblesData({
412
+ visibleComputedLayoutData: data.visibleComputedLayoutData,
413
+ LastBubbleDataMap,
414
+ graphicWidth: data.layout.width,
415
+ graphicHeight: data.layout.height,
416
+ SeriesContainerPositionMap: data.SeriesContainerPositionMap,
417
+ scaleType: data.scaleType
418
+ })
419
+ }),
420
+ shareReplay(1)
421
+ )
422
+
423
+ // 紀錄前一次bubble data
424
+ bubblesData$.subscribe(d => {
425
+ LastBubbleDataMap = new Map(d.map(_d => [_d.id, _d])) // key: id, value: datum
426
+ })
427
+
428
+ const highlightTarget$ = observer.fullChartParams$.pipe(
429
+ takeUntil(destroy$),
430
+ map(d => d.highlightTarget),
431
+ distinctUntilChanged()
432
+ )
433
+
434
+ const bubblesSelection$ = combineLatest({
435
+ bubblesData: bubblesData$,
436
+ fullParams: observer.fullParams$,
437
+ SeriesContainerPositionMap: observer.SeriesContainerPositionMap$,
438
+ sumSeries: sumSeries$
439
+ }).pipe(
440
+ takeUntil(destroy$),
441
+ switchMap(async (d) => d),
442
+ map(data => {
443
+ if (force) {
444
+ force.stop()
445
+ }
446
+
447
+ const bubblesSelection = renderBubbles({
448
+ selection,
449
+ bubblesData: data.bubblesData,
450
+ fullParams: data.fullParams,
451
+ sumSeries: data.sumSeries
452
+ })
453
+
454
+ force = makeForce(bubblesSelection, data.fullParams)
455
+
456
+ force.nodes(data.bubblesData)
457
+
458
+ groupBubbles({
459
+ fullParams: data.fullParams,
460
+ SeriesContainerPositionMap: data.SeriesContainerPositionMap
461
+ // graphicWidth: data.layout.width,
462
+ // graphicHeight: data.layout.height
463
+ })
464
+
465
+ // setTimeout(() => {
466
+ // force!.alphaTarget(0)
467
+ // force!.alpha(1).restart()
468
+ // }, 2000)
469
+
470
+ return bubblesSelection
471
+ })
472
+ )
473
+
474
+ combineLatest({
475
+ bubblesSelection: bubblesSelection$,
476
+ computedData: observer.computedData$,
477
+ SeriesDataMap: observer.SeriesDataMap$,
478
+ highlightTarget: highlightTarget$,
479
+ }).pipe(
480
+ takeUntil(destroy$),
481
+ switchMap(async (d) => d)
482
+ ).subscribe(data => {
483
+
484
+ data.bubblesSelection
485
+ .on('mouseover', (event, datum) => {
486
+ // this.tooltip!.setDatum({
487
+ // data: d,
488
+ // x: d3.event.clientX,
489
+ // y: d3.event.clientY
490
+ // })
491
+
492
+ subject.event$.next({
493
+ type: 'series',
494
+ eventName: 'mouseover',
495
+ pluginName: name,
496
+ highlightTarget: data.highlightTarget,
497
+ datum,
498
+ series: data.SeriesDataMap.get(datum.seriesLabel)!,
499
+ seriesIndex: datum.seriesIndex,
500
+ seriesLabel: datum.seriesLabel,
501
+ event,
502
+ data: data.computedData
503
+ })
504
+ })
505
+ .on('mousemove', (event, datum) => {
506
+ // this.tooltip!.setDatum({
507
+ // x: d3.event.clientX,
508
+ // y: d3.event.clientY
509
+ // })
510
+
511
+ subject.event$.next({
512
+ type: 'series',
513
+ eventName: 'mousemove',
514
+ pluginName: name,
515
+ highlightTarget: data.highlightTarget,
516
+ datum,
517
+ series: data.SeriesDataMap.get(datum.seriesLabel)!,
518
+ seriesIndex: datum.seriesIndex,
519
+ seriesLabel: datum.seriesLabel,
520
+ event,
521
+ data: data.computedData
522
+ })
523
+ })
524
+ .on('mouseout', (event, datum) => {
525
+ // this.tooltip!.remove()
526
+
527
+ subject.event$.next({
528
+ type: 'series',
529
+ eventName: 'mouseout',
530
+ pluginName: name,
531
+ highlightTarget: data.highlightTarget,
532
+ datum,
533
+ series: data.SeriesDataMap.get(datum.seriesLabel)!,
534
+ seriesIndex: datum.seriesIndex,
535
+ seriesLabel: datum.seriesLabel,
536
+ event,
537
+ data: data.computedData
538
+ })
539
+ })
540
+ .on('click', (event, datum) => {
541
+
542
+ subject.event$.next({
543
+ type: 'series',
544
+ eventName: 'click',
545
+ pluginName: name,
546
+ highlightTarget: data.highlightTarget,
547
+ datum,
548
+ series: data.SeriesDataMap.get(datum.seriesLabel)!,
549
+ seriesIndex: datum.seriesIndex,
550
+ seriesLabel: datum.seriesLabel,
551
+ event,
552
+ data: data.computedData
553
+ })
554
+ })
555
+ .call(drag() as any)
556
+
557
+
558
+ })
559
+
560
+ combineLatest({
561
+ bubblesSelection: bubblesSelection$,
562
+ bubblesData: bubblesData$,
563
+ highlight: observer.seriesHighlight$.pipe(
564
+ map(data => data.map(d => d.id))
565
+ ),
566
+ fullChartParams: observer.fullChartParams$,
567
+ fullParams: observer.fullParams$,
568
+ sumSeries: sumSeries$,
569
+ // layout: observer.layout$,
570
+ SeriesContainerPositionMap: observer.SeriesContainerPositionMap$,
571
+ }).pipe(
572
+ takeUntil(destroy$),
573
+ switchMap(async d => d)
574
+ ).subscribe(data => {
575
+ highlight({
576
+ bubblesSelection: data.bubblesSelection,
577
+ highlightIds: data.highlight,
578
+ fullChartParams: data.fullChartParams
579
+ })
580
+
581
+ // if (data.fullParams.highlightRIncrease) {
582
+ // setHighlightData ({
583
+ // data: data.bubblesData,
584
+ // highlightRIncrease: data.fullParams.highlightRIncrease,
585
+ // highlightIds: data.highlight
586
+ // })
587
+ // data.bubblesSelection.select('circle')
588
+ // // .transition()
589
+ // // .duration(200)
590
+ // .attr("r", (d) => d.r)
591
+
592
+ // force!.nodes(data.bubblesData)
593
+
594
+ // groupBubbles({
595
+ // fullParams: data.fullParams,
596
+ // SeriesContainerPositionMap: data.SeriesContainerPositionMap
597
+ // })
598
+ // }
599
+
600
+ })
601
+
602
+ return () => {
603
+ destroy$.next(undefined)
604
+ }
604
605
  })