@orbcharts/plugins-basic 3.0.0-beta.15 → 3.0.0-beta.17

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