@orbcharts/plugins-basic 3.0.0-beta.4 → 3.0.0-beta.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (192) hide show
  1. package/LICENSE +200 -200
  2. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseBarStack.d.ts +2 -2
  3. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseBars.d.ts +2 -2
  4. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseBarsTriangle.d.ts +2 -2
  5. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseDots.d.ts +2 -2
  6. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseGroupAxis.d.ts +2 -2
  7. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseLegend.d.ts +1 -1
  8. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseLineAreas.d.ts +3 -3
  9. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseLines.d.ts +3 -3
  10. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseValueAxis.d.ts +3 -2
  11. package/dist/{src → orbcharts-plugins-basic/src}/grid/defaults.d.ts +1 -1
  12. package/dist/{src → orbcharts-plugins-basic/src}/grid/gridObservables.d.ts +5 -5
  13. package/dist/orbcharts-plugins-basic/src/index.d.ts +6 -0
  14. package/dist/orbcharts-plugins-basic/src/multiValue/defaults.d.ts +9 -0
  15. package/dist/orbcharts-plugins-basic/src/multiValue/index.d.ts +8 -0
  16. package/dist/orbcharts-plugins-basic/src/multiValue/multiValueObservables.d.ts +33 -0
  17. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/MultiValueLegend.d.ts +1 -0
  18. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/MultiValueTooltip.d.ts +1 -0
  19. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/Scatter.d.ts +3 -0
  20. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/ScatterBubbles.d.ts +3 -0
  21. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/XYAux.d.ts +3 -0
  22. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/XYAxes.d.ts +3 -0
  23. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/XYZoom.d.ts +1 -0
  24. package/dist/{src → orbcharts-plugins-basic/src}/series/seriesObservables.d.ts +3 -3
  25. package/dist/{src → orbcharts-plugins-basic/src}/utils/observables.d.ts +1 -1
  26. package/dist/{src → orbcharts-plugins-basic/src}/utils/orbchartsUtils.d.ts +3 -3
  27. package/dist/orbcharts-plugins-basic.es.js +11809 -10185
  28. package/dist/orbcharts-plugins-basic.umd.js +66 -34
  29. package/dist/src/index.d.ts +1 -5
  30. package/lib/core-types.ts +7 -7
  31. package/lib/core.ts +6 -6
  32. package/lib/plugins-basic-types.ts +6 -6
  33. package/package.json +44 -44
  34. package/src/base/BaseBarStack.ts +780 -780
  35. package/src/base/BaseBars.ts +765 -765
  36. package/src/base/BaseBarsTriangle.ts +674 -674
  37. package/src/base/BaseDots.ts +464 -515
  38. package/src/base/BaseGroupAxis.ts +679 -679
  39. package/src/base/BaseLegend.ts +684 -680
  40. package/src/base/BaseLineAreas.ts +629 -629
  41. package/src/base/BaseLines.ts +706 -706
  42. package/src/base/BaseTooltip.ts +385 -385
  43. package/src/base/BaseValueAxis.ts +583 -580
  44. package/src/base/types.ts +2 -2
  45. package/src/const.ts +30 -30
  46. package/src/grid/defaults.ts +244 -244
  47. package/src/grid/gridObservables.ts +554 -545
  48. package/src/grid/index.ts +16 -16
  49. package/src/grid/plugins/BarStack.ts +64 -64
  50. package/src/grid/plugins/Bars.ts +69 -69
  51. package/src/grid/plugins/BarsPN.ts +66 -66
  52. package/src/grid/plugins/BarsTriangle.ts +73 -73
  53. package/src/grid/plugins/Dots.ts +68 -68
  54. package/src/grid/plugins/GridLegend.ts +107 -107
  55. package/src/grid/plugins/GridTooltip.ts +66 -66
  56. package/src/grid/plugins/GridZoom.ts +218 -218
  57. package/src/grid/plugins/GroupAux.ts +1103 -1098
  58. package/src/grid/plugins/GroupAxis.ts +97 -97
  59. package/src/grid/plugins/LineAreas.ts +65 -65
  60. package/src/grid/plugins/Lines.ts +59 -59
  61. package/src/grid/plugins/ValueAxis.ts +94 -93
  62. package/src/grid/plugins/ValueStackAxis.ts +96 -95
  63. package/src/index.ts +10 -9
  64. package/src/multiGrid/defaults.ts +224 -224
  65. package/src/multiGrid/index.ts +14 -14
  66. package/src/multiGrid/multiGridObservables.ts +49 -49
  67. package/src/multiGrid/plugins/MultiBarStack.ts +106 -106
  68. package/src/multiGrid/plugins/MultiBars.ts +108 -108
  69. package/src/multiGrid/plugins/MultiBarsTriangle.ts +114 -114
  70. package/src/multiGrid/plugins/MultiDots.ts +102 -102
  71. package/src/multiGrid/plugins/MultiGridLegend.ts +159 -159
  72. package/src/multiGrid/plugins/MultiGridTooltip.ts +66 -66
  73. package/src/multiGrid/plugins/MultiGroupAxis.ts +137 -137
  74. package/src/multiGrid/plugins/MultiLineAreas.ts +107 -107
  75. package/src/multiGrid/plugins/MultiLines.ts +101 -101
  76. package/src/multiGrid/plugins/MultiValueAxis.ts +134 -133
  77. package/src/multiGrid/plugins/MultiValueStackAxis.ts +134 -133
  78. package/src/multiGrid/plugins/OverlappingValueAxes.ts +300 -299
  79. package/src/multiGrid/plugins/OverlappingValueStackAxes.ts +299 -298
  80. package/src/multiValue/defaults.ts +163 -0
  81. package/src/multiValue/index.ts +9 -0
  82. package/src/multiValue/multiValueObservables.ts +258 -0
  83. package/src/multiValue/plugins/MultiValueLegend.ts +107 -0
  84. package/src/multiValue/plugins/MultiValueTooltip.ts +66 -0
  85. package/src/multiValue/plugins/Scatter.ts +426 -0
  86. package/src/multiValue/plugins/ScatterBubbles.ts +551 -0
  87. package/src/multiValue/plugins/XYAux.ts +682 -0
  88. package/src/multiValue/plugins/XYAxes.ts +677 -0
  89. package/src/multiValue/plugins/XYZoom.ts +300 -0
  90. package/src/noneData/defaults.ts +102 -102
  91. package/src/noneData/index.ts +3 -3
  92. package/src/noneData/plugins/Container.ts +27 -27
  93. package/src/noneData/plugins/Tooltip.ts +373 -373
  94. package/src/series/defaults.ts +206 -206
  95. package/src/series/index.ts +9 -9
  96. package/src/series/plugins/Bubbles.ts +603 -603
  97. package/src/series/plugins/Pie.ts +623 -623
  98. package/src/series/plugins/PieEventTexts.ts +283 -283
  99. package/src/series/plugins/PieLabels.ts +640 -640
  100. package/src/series/plugins/Rose.ts +516 -516
  101. package/src/series/plugins/RoseLabels.ts +600 -600
  102. package/src/series/plugins/SeriesLegend.ts +107 -107
  103. package/src/series/plugins/SeriesTooltip.ts +66 -66
  104. package/src/series/seriesObservables.ts +145 -145
  105. package/src/series/seriesUtils.ts +51 -51
  106. package/src/tree/defaults.ts +78 -78
  107. package/src/tree/index.ts +4 -4
  108. package/src/tree/plugins/TreeLegend.ts +100 -107
  109. package/src/tree/plugins/TreeMap.ts +333 -333
  110. package/src/tree/plugins/TreeTooltip.ts +66 -66
  111. package/src/utils/commonUtils.ts +21 -21
  112. package/src/utils/d3Graphics.ts +174 -174
  113. package/src/utils/d3Utils.ts +74 -73
  114. package/src/utils/observables.ts +14 -14
  115. package/src/utils/orbchartsUtils.ts +101 -100
  116. package/tsconfig.base.json +13 -13
  117. package/tsconfig.json +2 -2
  118. package/vite.config.js +22 -22
  119. package/dist/src/base/BaseGroupArea.d.ts +0 -0
  120. package/dist/src/multiValue/plugins/Scatter.d.ts +0 -0
  121. package/dist/src/multiValue/plugins/ScatterAxes.d.ts +0 -0
  122. package/dist/src/relationship/index.d.ts +0 -0
  123. package/src/base/BaseGroupArea.ts +0 -0
  124. package/src/multiValue/plugins/ScatterAxes.ts +0 -0
  125. /package/dist/{lib → orbcharts-plugins-basic/lib}/core-types.d.ts +0 -0
  126. /package/dist/{lib → orbcharts-plugins-basic/lib}/core.d.ts +0 -0
  127. /package/dist/{lib → orbcharts-plugins-basic/lib}/plugins-basic-types.d.ts +0 -0
  128. /package/dist/{src → orbcharts-plugins-basic/src}/base/BaseTooltip.d.ts +0 -0
  129. /package/dist/{src → orbcharts-plugins-basic/src}/base/types.d.ts +0 -0
  130. /package/dist/{src → orbcharts-plugins-basic/src}/const.d.ts +0 -0
  131. /package/dist/{src → orbcharts-plugins-basic/src}/grid/index.d.ts +0 -0
  132. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/BarStack.d.ts +0 -0
  133. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/Bars.d.ts +0 -0
  134. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/BarsPN.d.ts +0 -0
  135. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/BarsTriangle.d.ts +0 -0
  136. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/Dots.d.ts +0 -0
  137. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/GridLegend.d.ts +0 -0
  138. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/GridTooltip.d.ts +0 -0
  139. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/GridZoom.d.ts +0 -0
  140. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/GroupAux.d.ts +0 -0
  141. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/GroupAxis.d.ts +0 -0
  142. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/LineAreas.d.ts +0 -0
  143. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/Lines.d.ts +0 -0
  144. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/ValueAxis.d.ts +0 -0
  145. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/ValueStackAxis.d.ts +0 -0
  146. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/defaults.d.ts +0 -0
  147. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/index.d.ts +0 -0
  148. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/multiGridObservables.d.ts +0 -0
  149. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiBarStack.d.ts +0 -0
  150. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiBars.d.ts +0 -0
  151. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiBarsTriangle.d.ts +0 -0
  152. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiDots.d.ts +0 -0
  153. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiGridLegend.d.ts +0 -0
  154. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiGridTooltip.d.ts +0 -0
  155. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiGroupAxis.d.ts +0 -0
  156. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiLineAreas.d.ts +0 -0
  157. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiLines.d.ts +0 -0
  158. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiValueAxis.d.ts +0 -0
  159. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiValueStackAxis.d.ts +0 -0
  160. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/OverlappingValueAxes.d.ts +0 -0
  161. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/OverlappingValueStackAxes.d.ts +0 -0
  162. /package/dist/{src/grid → orbcharts-plugins-basic/src/multiValue}/plugins/Ranking.d.ts +0 -0
  163. /package/dist/{src/grid → orbcharts-plugins-basic/src/multiValue}/plugins/RankingAxis.d.ts +0 -0
  164. /package/dist/{src → orbcharts-plugins-basic/src}/noneData/defaults.d.ts +0 -0
  165. /package/dist/{src → orbcharts-plugins-basic/src}/noneData/index.d.ts +0 -0
  166. /package/dist/{src → orbcharts-plugins-basic/src}/noneData/plugins/Container.d.ts +0 -0
  167. /package/dist/{src → orbcharts-plugins-basic/src}/noneData/plugins/Tooltip.d.ts +0 -0
  168. /package/dist/{src/multiValue → orbcharts-plugins-basic/src/relationship}/index.d.ts +0 -0
  169. /package/dist/{src → orbcharts-plugins-basic/src}/relationship/plugins/Relationship.d.ts +0 -0
  170. /package/dist/{src → orbcharts-plugins-basic/src}/series/defaults.d.ts +0 -0
  171. /package/dist/{src → orbcharts-plugins-basic/src}/series/index.d.ts +0 -0
  172. /package/dist/{src → orbcharts-plugins-basic/src}/series/plugins/Bubbles.d.ts +0 -0
  173. /package/dist/{src → orbcharts-plugins-basic/src}/series/plugins/Pie.d.ts +0 -0
  174. /package/dist/{src → orbcharts-plugins-basic/src}/series/plugins/PieEventTexts.d.ts +0 -0
  175. /package/dist/{src → orbcharts-plugins-basic/src}/series/plugins/PieLabels.d.ts +0 -0
  176. /package/dist/{src → orbcharts-plugins-basic/src}/series/plugins/Rose.d.ts +0 -0
  177. /package/dist/{src → orbcharts-plugins-basic/src}/series/plugins/RoseLabels.d.ts +0 -0
  178. /package/dist/{src → orbcharts-plugins-basic/src}/series/plugins/SeriesLegend.d.ts +0 -0
  179. /package/dist/{src → orbcharts-plugins-basic/src}/series/plugins/SeriesTooltip.d.ts +0 -0
  180. /package/dist/{src → orbcharts-plugins-basic/src}/series/plugins/Waffle.d.ts +0 -0
  181. /package/dist/{src → orbcharts-plugins-basic/src}/series/seriesUtils.d.ts +0 -0
  182. /package/dist/{src → orbcharts-plugins-basic/src}/tree/defaults.d.ts +0 -0
  183. /package/dist/{src → orbcharts-plugins-basic/src}/tree/index.d.ts +0 -0
  184. /package/dist/{src → orbcharts-plugins-basic/src}/tree/plugins/TreeLegend.d.ts +0 -0
  185. /package/dist/{src → orbcharts-plugins-basic/src}/tree/plugins/TreeMap.d.ts +0 -0
  186. /package/dist/{src → orbcharts-plugins-basic/src}/tree/plugins/TreeTooltip.d.ts +0 -0
  187. /package/dist/{src → orbcharts-plugins-basic/src}/utils/commonUtils.d.ts +0 -0
  188. /package/dist/{src → orbcharts-plugins-basic/src}/utils/d3Graphics.d.ts +0 -0
  189. /package/dist/{src → orbcharts-plugins-basic/src}/utils/d3Utils.d.ts +0 -0
  190. /package/dist/{vite.config.d.ts → orbcharts-plugins-basic/vite.config.d.ts} +0 -0
  191. /package/src/{grid → multiValue}/plugins/Ranking.ts +0 -0
  192. /package/src/{grid → multiValue}/plugins/RankingAxis.ts +0 -0
@@ -0,0 +1,551 @@
1
+ import * as d3 from 'd3'
2
+ import {
3
+ combineLatest,
4
+ map,
5
+ switchMap,
6
+ takeUntil,
7
+ distinctUntilChanged,
8
+ Observable,
9
+ Subject,
10
+ share,
11
+ shareReplay
12
+ } from 'rxjs'
13
+ import type {
14
+ ComputedDatumMultiValue,
15
+ ComputedDataMultiValue,
16
+ ComputedLayoutDatumMultiValue,
17
+ ComputedLayoutDataMultiValue,
18
+ DefinePluginConfig,
19
+ EventMultiValue,
20
+ ChartParams,
21
+ ContainerPositionScaled,
22
+ Layout,
23
+ TransformData,
24
+ ColorType
25
+ } from '../../../lib/core-types'
26
+ import {
27
+ defineMultiValuePlugin,
28
+ getMinAndMax
29
+ } from '../../../lib/core'
30
+ import type { ScatterBubblesParams } from '../../../lib/plugins-basic-types'
31
+ import { DEFAULT_SCATTER_BUBBLES_PARAMS } from '../defaults'
32
+ import { LAYER_INDEX_OF_GRAPHIC_COVER } from '../../const'
33
+ import { getDatumColor, getClassName, getUniID } from '../../utils/orbchartsUtils'
34
+ import { multiValueSelectionsObservable } from '../multiValueObservables'
35
+
36
+ type ClipPathDatum = {
37
+ id: string;
38
+ // x: number;
39
+ // y: number;
40
+ width: number;
41
+ height: number;
42
+ }
43
+
44
+ interface BubbleDatum extends ComputedLayoutDatumMultiValue {
45
+ r: number
46
+ opacity: number
47
+ }
48
+
49
+ const pluginName = 'ScatterBubbles'
50
+
51
+ // 調整係數 - 因為圓和圓之間的空隙造成聚合起來的大圓會略大,所以稍作微調
52
+ const adjustmentFactor = 0.9
53
+
54
+ const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_SCATTER_BUBBLES_PARAMS> = {
55
+ name: pluginName,
56
+ defaultParams: DEFAULT_SCATTER_BUBBLES_PARAMS,
57
+ layerIndex: LAYER_INDEX_OF_GRAPHIC_COVER,
58
+ validator: (params, { validateColumns }) => {
59
+ const result = validateColumns(params, {
60
+ // radius: {
61
+ // toBeTypes: ['number']
62
+ // },
63
+ fillColorType: {
64
+ toBeOption: 'ColorType',
65
+ },
66
+ strokeColorType: {
67
+ toBeOption: 'ColorType',
68
+ },
69
+ strokeWidth: {
70
+ toBeTypes: ['number']
71
+ },
72
+ valueLinearOpacity: {
73
+ toBeTypes: ['number[]']
74
+ },
75
+ arcScaleType: {
76
+ toBe: 'ArcScaleType',
77
+ test: (value) => {
78
+ return value === 'area' || value === 'radius'
79
+ }
80
+ },
81
+ sizeAdjust: {
82
+ toBeTypes: ['number']
83
+ }
84
+ })
85
+ return result
86
+ }
87
+ }
88
+
89
+ function renderDots ({ graphicGSelection, circleGClassName, circleClassName, bubbleData, fullParams, fullChartParams, graphicReverseScale }: {
90
+ graphicGSelection: d3.Selection<SVGGElement, any, any, any>
91
+ circleGClassName: string
92
+ circleClassName: string
93
+ // visibleComputedLayoutData: ComputedLayoutDataMultiValue
94
+ bubbleData: BubbleDatum[][]
95
+ fullParams: ScatterBubblesParams
96
+ fullChartParams: ChartParams
97
+ graphicReverseScale: [number, number][]
98
+ }) {
99
+ const createEnterDuration = (enter: d3.Selection<d3.EnterElement, ComputedDatumMultiValue, SVGGElement, any>) => {
100
+ const enterSize = enter.size()
101
+ const eachDuration = fullChartParams.transitionDuration / enterSize
102
+ return eachDuration
103
+ }
104
+ // enterDuration
105
+ let enterDuration = 0
106
+
107
+ graphicGSelection
108
+ .each((categoryData, categoryIndex, g) => {
109
+ d3.select(g[categoryIndex])
110
+ .selectAll<SVGGElement, ComputedDatumMultiValue>('g')
111
+ .data(bubbleData[categoryIndex], d => d.id)
112
+ .join(
113
+ enter => {
114
+ // enterDuration
115
+ enterDuration = createEnterDuration(enter)
116
+
117
+ return enter
118
+ .append('g')
119
+ .classed(circleGClassName, true)
120
+ },
121
+ update => update,
122
+ exit => exit.remove()
123
+ )
124
+ .attr('transform', d => `translate(${d.axisX}, ${d.axisY})`)
125
+ .each((d, i, g) => {
126
+ const circle = d3.select(g[i])
127
+ .selectAll('circle')
128
+ .data([d])
129
+ .join(
130
+ enter => {
131
+ return enter
132
+ .append('circle')
133
+ .style('cursor', 'pointer')
134
+ .style('vector-effect', 'non-scaling-stroke')
135
+ .classed(circleClassName, true)
136
+ .attr('opacity', 0)
137
+ .transition()
138
+ .delay((_d, _i) => {
139
+ return i * enterDuration
140
+ })
141
+ .attr('opacity', 0.8)
142
+ },
143
+ update => {
144
+ return update
145
+ .transition()
146
+ .duration(50)
147
+ // .attr('cx', d => d.axisX)
148
+ // .attr('cy', d => d.axisY)
149
+ .attr('opacity', 0.8)
150
+ },
151
+ exit => exit.remove()
152
+ )
153
+ .attr('r', d => d.r)
154
+ .attr('fill', (d, i) => getDatumColor({ datum: d, colorType: fullParams.fillColorType, fullChartParams }))
155
+ .attr('stroke', (d, i) => getDatumColor({ datum: d, colorType: fullParams.strokeColorType, fullChartParams }))
156
+ .attr('stroke-width', fullParams.strokeWidth)
157
+ .attr('transform', `scale(${graphicReverseScale[categoryIndex][0] ?? 1}, ${graphicReverseScale[categoryIndex][1] ?? 1})`)
158
+ })
159
+ })
160
+
161
+ const graphicCircleSelection: d3.Selection<SVGRectElement, BubbleDatum, SVGGElement, unknown> = graphicGSelection.selectAll(`circle.${circleClassName}`)
162
+
163
+ return graphicCircleSelection
164
+ }
165
+
166
+
167
+ function highlightBubbles ({ selection, ids, fullChartParams }: {
168
+ selection: d3.Selection<SVGGElement, BubbleDatum, any, any>
169
+ ids: string[]
170
+ fullChartParams: ChartParams
171
+ }) {
172
+ selection.interrupt('highlight')
173
+ if (!ids.length) {
174
+ // remove highlight
175
+ selection
176
+ .transition('highlight')
177
+ .duration(200)
178
+ .style('opacity', d => d.opacity)
179
+ // selection
180
+ // .attr('stroke-width', fullParams.strokeWidth)
181
+
182
+ return
183
+ }
184
+
185
+ selection
186
+ .each((d, i, n) => {
187
+ if (ids.includes(d.id)) {
188
+ const dot = d3.select<SVGGElement, BubbleDatum>(n[i])
189
+ dot
190
+ .style('opacity', d => d.opacity)
191
+ .transition('highlight')
192
+ .duration(200)
193
+ // dot
194
+ // .attr('stroke-width', fullParams.strokeWidthWhileHighlight)
195
+ } else {
196
+ const dot = d3.select(n[i])
197
+ dot
198
+ .style('opacity', fullChartParams.styles.unhighlightedOpacity)
199
+ .transition('highlight')
200
+ .duration(200)
201
+ // dot
202
+ // .attr('stroke-width', fullParams.strokeWidth)
203
+
204
+ }
205
+ })
206
+ }
207
+
208
+ function renderClipPath ({ defsSelection, clipPathData }: {
209
+ defsSelection: d3.Selection<SVGDefsElement, any, any, any>
210
+ clipPathData: ClipPathDatum[]
211
+ }) {
212
+ const clipPath = defsSelection
213
+ .selectAll<SVGClipPathElement, Layout>('clipPath')
214
+ .data(clipPathData)
215
+ .join(
216
+ enter => {
217
+ return enter
218
+ .append('clipPath')
219
+ },
220
+ update => update,
221
+ exit => exit.remove()
222
+ )
223
+ .attr('id', d => d.id)
224
+ .each((d, i, g) => {
225
+ const rect = d3.select(g[i])
226
+ .selectAll<SVGRectElement, typeof d>('rect')
227
+ .data([d])
228
+ .join('rect')
229
+ .attr('x', 0)
230
+ .attr('y', 0)
231
+ .attr('width', _d => _d.width)
232
+ .attr('height', _d => _d.height)
233
+ })
234
+
235
+ }
236
+
237
+
238
+ export const ScatterBubbles = defineMultiValuePlugin(pluginConfig)(({ selection, name, subject, observer }) => {
239
+
240
+ const destroy$ = new Subject()
241
+
242
+ const clipPathID = getUniID(pluginName, 'clipPath-box')
243
+ const circleGClassName = getClassName(pluginName, 'circleG')
244
+ const circleClassName = getClassName(pluginName, 'circle')
245
+
246
+ const {
247
+ categorySelection$,
248
+ axesSelection$,
249
+ defsSelection$,
250
+ graphicGSelection$
251
+ } = multiValueSelectionsObservable({
252
+ selection,
253
+ pluginName,
254
+ clipPathID,
255
+ categoryLabels$: observer.categoryLabels$,
256
+ multiValueContainerPosition$: observer.multiValueContainerPosition$,
257
+ multiValueGraphicTransform$: observer.multiValueGraphicTransform$
258
+ })
259
+
260
+ const graphicReverseScale$: Observable<[number, number][]> = combineLatest({
261
+ computedData: observer.computedData$,
262
+ multiValueGraphicReverseScale: observer.multiValueGraphicReverseScale$
263
+ }).pipe(
264
+ takeUntil(destroy$),
265
+ switchMap(async data => data),
266
+ map(data => {
267
+ return data.computedData.map((series, categoryIndex) => {
268
+ return data.multiValueGraphicReverseScale[categoryIndex]
269
+ })
270
+ })
271
+ )
272
+
273
+ const clipPathSubscription = combineLatest({
274
+ defsSelection: defsSelection$,
275
+ layout: observer.layout$,
276
+ }).pipe(
277
+ takeUntil(destroy$),
278
+ switchMap(async (d) => d),
279
+ ).subscribe(data => {
280
+ // 外層的遮罩
281
+ const clipPathData = [{
282
+ id: clipPathID,
283
+ width: data.layout.width,
284
+ height: data.layout.height
285
+ }]
286
+ renderClipPath({
287
+ defsSelection: data.defsSelection,
288
+ clipPathData,
289
+ })
290
+ })
291
+
292
+ const filteredValueList$ = observer.filteredMinMaxXYData$.pipe(
293
+ takeUntil(destroy$),
294
+ map(data => data.datumList.flat().map(d => d.value[2] ?? 0)),
295
+ shareReplay(1)
296
+ )
297
+
298
+ const filteredMinMaxValue$ = observer.filteredMinMaxXYData$.pipe(
299
+ takeUntil(destroy$),
300
+ map(data => {
301
+ return getMinAndMax(data.datumList.flat().map(d => d.value[2] ?? 0))
302
+ })
303
+ )
304
+
305
+ const opacityScale$ = combineLatest({
306
+ filteredMinMaxValue: filteredMinMaxValue$,
307
+ fullParams: observer.fullParams$
308
+ }).pipe(
309
+ takeUntil(destroy$),
310
+ switchMap(async (d) => d),
311
+ map(data => {
312
+ return d3.scaleLinear()
313
+ .domain(data.filteredMinMaxValue)
314
+ .range(data.fullParams.valueLinearOpacity)
315
+ })
316
+ )
317
+
318
+ // 虛擬大圓(所有小圓聚合起來的大圓)的半徑
319
+ const totalR$ = combineLatest({
320
+ layout: observer.layout$,
321
+ fullParams: observer.fullParams$
322
+ }).pipe(
323
+ takeUntil(destroy$),
324
+ map(data => {
325
+ // 場景最短邊
326
+ const fullRadius = Math.min(...[data.layout.width, data.layout.height]) / 2
327
+ return fullRadius * data.fullParams.sizeAdjust
328
+ })
329
+ )
330
+
331
+ const totalValue$ = filteredValueList$.pipe(
332
+ takeUntil(destroy$),
333
+ map(data => {
334
+ return data.reduce((acc, current) => acc + current, 0)
335
+ })
336
+ )
337
+
338
+ const radiusScale$ = combineLatest({
339
+ totalR: totalR$,
340
+ totalValue: totalValue$,
341
+ fullParams: observer.fullParams$
342
+ }).pipe(
343
+ takeUntil(destroy$),
344
+ switchMap(async (d) => d),
345
+ map(data => {
346
+ return d3.scalePow()
347
+ .domain([0, data.totalValue])
348
+ .range([0, data.totalR])
349
+ .exponent(data.fullParams.arcScaleType === 'area'
350
+ ? 0.5 // 數值映射面積(0.5為取平方根)
351
+ : 1 // 數值映射半徑
352
+ )
353
+ })
354
+ )
355
+
356
+ const scaleFactor$ = combineLatest({
357
+ radiusScale: radiusScale$,
358
+ fullParams: observer.fullParams$,
359
+ totalR: totalR$,
360
+ filteredValueList: filteredValueList$
361
+ }).pipe(
362
+ takeUntil(destroy$),
363
+ switchMap(async (d) => d),
364
+ map(data => {
365
+ return data.fullParams.arcScaleType === 'area'
366
+ ? 1
367
+ // 當數值映射半徑時,多個小圓的總面積會小於大圓的面積,所以要計算縮放比例
368
+ : (() => {
369
+ const totalArea = data.totalR * data.totalR * Math.PI
370
+ return Math.sqrt(totalArea / d3.sum(data.filteredValueList, d => Math.PI * Math.pow(data.radiusScale(d), 2)))
371
+ })()
372
+ })
373
+ )
374
+
375
+ const bubbleData$ = combineLatest({
376
+ computedLayoutData: observer.computedLayoutData$,
377
+ opacityScale: opacityScale$,
378
+ radiusScale: radiusScale$,
379
+ scaleFactor: scaleFactor$,
380
+ fullParams: observer.fullParams$
381
+ }).pipe(
382
+ takeUntil(destroy$),
383
+ switchMap(async (d) => d),
384
+ map(data => {
385
+ return data.computedLayoutData.map(category => {
386
+ return category.map(_d => {
387
+ const d: BubbleDatum = _d as BubbleDatum
388
+ d.r = data.radiusScale(d.value[2]) * data.scaleFactor * adjustmentFactor
389
+ d.opacity = data.opacityScale(d.value[2])
390
+ return d
391
+ })
392
+ })
393
+ })
394
+ )
395
+
396
+
397
+ const graphicSelection$ = combineLatest({
398
+ graphicGSelection: graphicGSelection$,
399
+ // visibleComputedLayoutData: observer.visibleComputedLayoutData$,
400
+ bubbleData: bubbleData$,
401
+ graphicReverseScale: graphicReverseScale$,
402
+ fullChartParams: observer.fullChartParams$,
403
+ fullParams: observer.fullParams$,
404
+ }).pipe(
405
+ takeUntil(destroy$),
406
+ switchMap(async (d) => d),
407
+ map(data => {
408
+ return renderDots({
409
+ graphicGSelection: data.graphicGSelection,
410
+ circleGClassName,
411
+ circleClassName,
412
+ bubbleData: data.bubbleData,
413
+ fullParams: data.fullParams,
414
+ fullChartParams: data.fullChartParams,
415
+ graphicReverseScale: data.graphicReverseScale
416
+ })
417
+ })
418
+ )
419
+
420
+ const highlightTarget$ = observer.fullChartParams$.pipe(
421
+ takeUntil(destroy$),
422
+ map(d => d.highlightTarget),
423
+ distinctUntilChanged()
424
+ )
425
+
426
+ combineLatest({
427
+ graphicSelection: graphicSelection$,
428
+ computedData: observer.computedData$,
429
+ CategoryDataMap: observer.CategoryDataMap$,
430
+ highlightTarget: highlightTarget$
431
+ }).pipe(
432
+ takeUntil(destroy$),
433
+ switchMap(async (d) => d),
434
+ ).subscribe(data => {
435
+
436
+ data.graphicSelection
437
+ .on('mouseover', (event, datum) => {
438
+ // event.stopPropagation()
439
+ // console.log({
440
+ // type: 'multiValue',
441
+ // eventName: 'mouseover',
442
+ // pluginName,
443
+ // highlightTarget: data.highlightTarget,
444
+ // datum,
445
+ // category: data.CategoryDataMap.get(datum.categoryLabel)!,
446
+ // categoryIndex: datum.categoryIndex,
447
+ // categoryLabel: datum.categoryLabel,
448
+ // data: data.computedData,
449
+ // event,
450
+ // })
451
+ subject.event$.next({
452
+ type: 'multiValue',
453
+ eventName: 'mouseover',
454
+ pluginName,
455
+ highlightTarget: data.highlightTarget,
456
+ datum,
457
+ category: data.CategoryDataMap.get(datum.categoryLabel)!,
458
+ categoryIndex: datum.categoryIndex,
459
+ categoryLabel: datum.categoryLabel,
460
+ data: data.computedData,
461
+ event,
462
+ })
463
+ })
464
+ .on('mousemove', (event, datum) => {
465
+ // event.stopPropagation()
466
+
467
+ subject.event$.next({
468
+ type: 'multiValue',
469
+ eventName: 'mousemove',
470
+ pluginName,
471
+ highlightTarget: data.highlightTarget,
472
+ datum,
473
+ category: data.CategoryDataMap.get(datum.categoryLabel)!,
474
+ categoryIndex: datum.categoryIndex,
475
+ categoryLabel: datum.categoryLabel,
476
+ data: data.computedData,
477
+ event,
478
+ })
479
+ })
480
+ .on('mouseout', (event, datum) => {
481
+ // event.stopPropagation()
482
+
483
+ subject.event$.next({
484
+ type: 'multiValue',
485
+ eventName: 'mouseout',
486
+ pluginName,
487
+ highlightTarget: data.highlightTarget,
488
+ datum,
489
+ category: data.CategoryDataMap.get(datum.categoryLabel)!,
490
+ categoryIndex: datum.categoryIndex,
491
+ categoryLabel: datum.categoryLabel,
492
+ data: data.computedData,
493
+ event,
494
+ })
495
+ })
496
+ .on('click', (event, datum) => {
497
+ // event.stopPropagation()
498
+
499
+ subject.event$.next({
500
+ type: 'multiValue',
501
+ eventName: 'click',
502
+ pluginName,
503
+ highlightTarget: data.highlightTarget,
504
+ datum,
505
+ category: data.CategoryDataMap.get(datum.categoryLabel)!,
506
+ categoryIndex: datum.categoryIndex,
507
+ categoryLabel: datum.categoryLabel,
508
+ data: data.computedData,
509
+ event,
510
+ })
511
+ })
512
+
513
+ })
514
+
515
+ combineLatest({
516
+ graphicSelection: graphicSelection$,
517
+ highlight: observer.multiValueHighlight$.pipe(
518
+ map(data => data.map(d => d.id))
519
+ ),
520
+ fullChartParams: observer.fullChartParams$
521
+ }).pipe(
522
+ takeUntil(destroy$),
523
+ switchMap(async d => d)
524
+ ).subscribe(data => {
525
+ highlightBubbles({
526
+ selection: data.graphicSelection,
527
+ ids: data.highlight,
528
+ fullChartParams: data.fullChartParams
529
+ })
530
+ })
531
+
532
+ // graphicGSelection$.subscribe(data => {
533
+ // console.log('graphicGSelection$', data)
534
+ // })
535
+
536
+ // observer.visibleComputedLayoutData$.subscribe(data => {
537
+ // console.log('visibleComputedLayoutData$', data)
538
+ // })
539
+
540
+ // observer.fullChartParams$.subscribe(data => {
541
+ // console.log('fullChartParams$', data)
542
+ // })
543
+
544
+ // observer.fullParams$.subscribe(data => {
545
+ // console.log('fullParams$', data)
546
+ // })
547
+
548
+ return () => {
549
+ destroy$.next(undefined)
550
+ }
551
+ })