@orbcharts/plugins-basic 3.0.0-alpha.44 → 3.0.0-alpha.45

Sign up to get free protection for your applications and to get access to all the features.
Files changed (155) hide show
  1. package/LICENSE +200 -200
  2. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseBarStack.d.ts +6 -4
  3. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseBars.d.ts +6 -4
  4. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseBarsTriangle.d.ts +7 -4
  5. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseDots.d.ts +5 -3
  6. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseGroupAxis.d.ts +3 -3
  7. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseLineAreas.d.ts +6 -3
  8. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseLines.d.ts +6 -3
  9. package/dist/{src → orbcharts-plugins-basic/src}/base/BaseValueAxis.d.ts +3 -3
  10. package/dist/{src → orbcharts-plugins-basic/src}/grid/gridObservables.d.ts +4 -4
  11. package/dist/orbcharts-plugins-basic/src/index.d.ts +5 -0
  12. package/dist/orbcharts-plugins-basic/src/multiGrid/multiGridObservables.d.ts +8 -0
  13. package/dist/orbcharts-plugins-basic/src/series/plugins/PieEventTexts.d.ts +3 -0
  14. package/dist/orbcharts-plugins-basic/src/series/seriesObservables.d.ts +21 -0
  15. package/dist/{src → orbcharts-plugins-basic/src}/series/seriesUtils.d.ts +3 -3
  16. package/dist/orbcharts-plugins-basic.es.js +7696 -7529
  17. package/dist/orbcharts-plugins-basic.umd.js +8 -8
  18. package/dist/src/index.d.ts +1 -5
  19. package/package.json +42 -42
  20. package/src/base/BaseBarStack.ts +778 -881
  21. package/src/base/BaseBars.ts +764 -750
  22. package/src/base/BaseBarsTriangle.ts +672 -659
  23. package/src/base/BaseDots.ts +502 -639
  24. package/src/base/BaseGroupAxis.ts +496 -496
  25. package/src/base/BaseLegend.ts +641 -641
  26. package/src/base/BaseLineAreas.ts +625 -621
  27. package/src/base/BaseLines.ts +699 -692
  28. package/src/base/BaseValueAxis.ts +478 -479
  29. package/src/base/types.ts +2 -2
  30. package/src/grid/defaults.ts +121 -121
  31. package/src/grid/gridObservables.ts +247 -263
  32. package/src/grid/index.ts +15 -15
  33. package/src/grid/plugins/BarStack.ts +50 -37
  34. package/src/grid/plugins/Bars.ts +51 -37
  35. package/src/grid/plugins/BarsDiverging.ts +41 -39
  36. package/src/grid/plugins/BarsTriangle.ts +50 -34
  37. package/src/grid/plugins/Dots.ts +37 -35
  38. package/src/grid/plugins/GridLegend.ts +59 -59
  39. package/src/grid/plugins/GroupAux.ts +645 -646
  40. package/src/grid/plugins/GroupAxis.ts +42 -30
  41. package/src/grid/plugins/LineAreas.ts +39 -36
  42. package/src/grid/plugins/Lines.ts +38 -35
  43. package/src/grid/plugins/ScalingArea.ts +173 -174
  44. package/src/grid/plugins/ValueAxis.ts +43 -31
  45. package/src/grid/plugins/ValueStackAxis.ts +79 -70
  46. package/src/grid/types.ts +120 -120
  47. package/src/index.ts +9 -9
  48. package/src/multiGrid/defaults.ts +147 -147
  49. package/src/multiGrid/index.ts +11 -11
  50. package/src/multiGrid/multiGridObservables.ts +42 -289
  51. package/src/multiGrid/plugins/MultiBarStack.ts +74 -60
  52. package/src/multiGrid/plugins/MultiBars.ts +73 -59
  53. package/src/multiGrid/plugins/MultiBarsTriangle.ts +73 -58
  54. package/src/multiGrid/plugins/MultiDots.ts +60 -58
  55. package/src/multiGrid/plugins/MultiGridLegend.ts +89 -89
  56. package/src/multiGrid/plugins/MultiGroupAxis.ts +65 -53
  57. package/src/multiGrid/plugins/MultiLineAreas.ts +62 -59
  58. package/src/multiGrid/plugins/MultiLines.ts +61 -58
  59. package/src/multiGrid/plugins/MultiValueAxis.ts +65 -53
  60. package/src/multiGrid/plugins/OverlappingValueAxes.ts +169 -164
  61. package/src/multiGrid/types.ts +67 -67
  62. package/src/noneData/defaults.ts +64 -61
  63. package/src/noneData/index.ts +3 -3
  64. package/src/noneData/plugins/Container.ts +10 -10
  65. package/src/noneData/plugins/Tooltip.ts +310 -310
  66. package/src/noneData/types.ts +26 -26
  67. package/src/series/defaults.ts +109 -99
  68. package/src/series/index.ts +6 -6
  69. package/src/series/plugins/Bubbles.ts +571 -551
  70. package/src/series/plugins/Pie.ts +548 -600
  71. package/src/series/plugins/PieEventTexts.ts +258 -194
  72. package/src/series/plugins/PieLabels.ts +335 -288
  73. package/src/series/plugins/SeriesLegend.ts +59 -59
  74. package/src/series/seriesObservables.ts +145 -0
  75. package/src/series/seriesUtils.ts +50 -50
  76. package/src/series/types.ts +67 -67
  77. package/src/tree/defaults.ts +22 -22
  78. package/src/tree/index.ts +3 -3
  79. package/src/tree/plugins/TreeLegend.ts +59 -59
  80. package/src/tree/plugins/TreeMap.ts +305 -305
  81. package/src/tree/types.ts +23 -23
  82. package/src/utils/commonUtils.ts +21 -21
  83. package/src/utils/d3Graphics.ts +124 -124
  84. package/src/utils/d3Utils.ts +73 -73
  85. package/src/utils/observables.ts +14 -14
  86. package/src/utils/orbchartsUtils.ts +100 -100
  87. package/tsconfig.dev.json +16 -16
  88. package/tsconfig.json +16 -13
  89. package/tsconfig.prod.json +13 -13
  90. package/vite.config.js +49 -49
  91. package/dist/src/multiGrid/multiGridObservables.d.ts +0 -12
  92. package/dist/src/series/plugins/PieEventTexts.d.ts +0 -1
  93. /package/dist/{src → orbcharts-plugins-basic/src}/base/BaseGroupArea.d.ts +0 -0
  94. /package/dist/{src → orbcharts-plugins-basic/src}/base/BaseLegend.d.ts +0 -0
  95. /package/dist/{src → orbcharts-plugins-basic/src}/base/types.d.ts +0 -0
  96. /package/dist/{src → orbcharts-plugins-basic/src}/grid/defaults.d.ts +0 -0
  97. /package/dist/{src → orbcharts-plugins-basic/src}/grid/index.d.ts +0 -0
  98. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/BarStack.d.ts +0 -0
  99. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/Bars.d.ts +0 -0
  100. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/BarsDiverging.d.ts +0 -0
  101. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/BarsTriangle.d.ts +0 -0
  102. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/Dots.d.ts +0 -0
  103. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/GridLegend.d.ts +0 -0
  104. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/GroupAux.d.ts +0 -0
  105. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/GroupAxis.d.ts +0 -0
  106. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/LineAreas.d.ts +0 -0
  107. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/Lines.d.ts +0 -0
  108. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/Ranking.d.ts +0 -0
  109. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/RankingAxis.d.ts +0 -0
  110. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/ScalingArea.d.ts +0 -0
  111. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/ValueAxis.d.ts +0 -0
  112. /package/dist/{src → orbcharts-plugins-basic/src}/grid/plugins/ValueStackAxis.d.ts +0 -0
  113. /package/dist/{src → orbcharts-plugins-basic/src}/grid/types.d.ts +0 -0
  114. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/defaults.d.ts +0 -0
  115. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/index.d.ts +0 -0
  116. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiBarStack.d.ts +0 -0
  117. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiBars.d.ts +0 -0
  118. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiBarsTriangle.d.ts +0 -0
  119. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiDots.d.ts +0 -0
  120. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiGridLegend.d.ts +0 -0
  121. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiGroupAxis.d.ts +0 -0
  122. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiLineAreas.d.ts +0 -0
  123. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiLines.d.ts +0 -0
  124. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/MultiValueAxis.d.ts +0 -0
  125. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/plugins/OverlappingValueAxes.d.ts +0 -0
  126. /package/dist/{src → orbcharts-plugins-basic/src}/multiGrid/types.d.ts +0 -0
  127. /package/dist/{src → orbcharts-plugins-basic/src}/multiValue/index.d.ts +0 -0
  128. /package/dist/{src → orbcharts-plugins-basic/src}/multiValue/plugins/Scatter.d.ts +0 -0
  129. /package/dist/{src → orbcharts-plugins-basic/src}/multiValue/plugins/ScatterAxes.d.ts +0 -0
  130. /package/dist/{src → orbcharts-plugins-basic/src}/noneData/defaults.d.ts +0 -0
  131. /package/dist/{src → orbcharts-plugins-basic/src}/noneData/index.d.ts +0 -0
  132. /package/dist/{src → orbcharts-plugins-basic/src}/noneData/plugins/Container.d.ts +0 -0
  133. /package/dist/{src → orbcharts-plugins-basic/src}/noneData/plugins/Tooltip.d.ts +0 -0
  134. /package/dist/{src → orbcharts-plugins-basic/src}/noneData/types.d.ts +0 -0
  135. /package/dist/{src → orbcharts-plugins-basic/src}/relationship/index.d.ts +0 -0
  136. /package/dist/{src → orbcharts-plugins-basic/src}/relationship/plugins/Relationship.d.ts +0 -0
  137. /package/dist/{src → orbcharts-plugins-basic/src}/series/defaults.d.ts +0 -0
  138. /package/dist/{src → orbcharts-plugins-basic/src}/series/index.d.ts +0 -0
  139. /package/dist/{src → orbcharts-plugins-basic/src}/series/plugins/Bubbles.d.ts +0 -0
  140. /package/dist/{src → orbcharts-plugins-basic/src}/series/plugins/Pie.d.ts +0 -0
  141. /package/dist/{src → orbcharts-plugins-basic/src}/series/plugins/PieLabels.d.ts +0 -0
  142. /package/dist/{src → orbcharts-plugins-basic/src}/series/plugins/SeriesLegend.d.ts +0 -0
  143. /package/dist/{src → orbcharts-plugins-basic/src}/series/plugins/Waffle.d.ts +0 -0
  144. /package/dist/{src → orbcharts-plugins-basic/src}/series/types.d.ts +0 -0
  145. /package/dist/{src → orbcharts-plugins-basic/src}/tree/defaults.d.ts +0 -0
  146. /package/dist/{src → orbcharts-plugins-basic/src}/tree/index.d.ts +0 -0
  147. /package/dist/{src → orbcharts-plugins-basic/src}/tree/plugins/TreeLegend.d.ts +0 -0
  148. /package/dist/{src → orbcharts-plugins-basic/src}/tree/plugins/TreeMap.d.ts +0 -0
  149. /package/dist/{src → orbcharts-plugins-basic/src}/tree/types.d.ts +0 -0
  150. /package/dist/{src → orbcharts-plugins-basic/src}/utils/commonUtils.d.ts +0 -0
  151. /package/dist/{src → orbcharts-plugins-basic/src}/utils/d3Graphics.d.ts +0 -0
  152. /package/dist/{src → orbcharts-plugins-basic/src}/utils/d3Utils.d.ts +0 -0
  153. /package/dist/{src → orbcharts-plugins-basic/src}/utils/observables.d.ts +0 -0
  154. /package/dist/{src → orbcharts-plugins-basic/src}/utils/orbchartsUtils.d.ts +0 -0
  155. /package/dist/{vite.config.d.ts → orbcharts-plugins-basic/vite.config.d.ts} +0 -0
@@ -1,305 +1,305 @@
1
- import * as d3 from 'd3'
2
- import {
3
- Subject,
4
- Observable,
5
- of,
6
- takeUntil,
7
- map,
8
- switchMap,
9
- combineLatest,
10
- debounceTime,
11
- distinctUntilChanged } from 'rxjs'
12
- import {
13
- defineTreePlugin } from '@orbcharts/core'
14
- import type { Layout, ComputedDataTree, DataFormatterTree, ChartParams } from '@orbcharts/core'
15
- import type { TreeMapParams } from '../types'
16
- import { DEFAULT_TREE_MAP_PARAMS } from '../defaults'
17
- import { getClassName, getColor } from '../../utils/orbchartsUtils'
18
-
19
- const pluginName = 'TreeMap'
20
- const treeClassName = getClassName(pluginName, 'tree')
21
- const tileClassName = getClassName(pluginName, 'tile')
22
-
23
- function renderTree ({ selection, treeData, fullParams, fullChartParams, textSizePx }: {
24
- selection: d3.Selection<any, any, any, any>
25
- treeData: d3.HierarchyRectangularNode<ComputedDataTree>[]
26
- fullParams: TreeMapParams
27
- fullChartParams: ChartParams
28
- textSizePx: number
29
- }) {
30
- const padding = textSizePx / 2
31
- const lineHeight = textSizePx // 行高
32
-
33
- const cell = selection.selectAll<SVGGElement, d3.HierarchyRectangularNode<ComputedDataTree>>(`g.${treeClassName}`)
34
- .data(treeData, d => d.data.id)
35
- .join('g')
36
- .attr('class', treeClassName)
37
-
38
- cell
39
- // .transition()
40
- // .duration(fullChartParams.transitionDuration)
41
- .attr('transform', (d) => !d.x0 || !d.y0 ? null : `translate(${d.x0},${d.y0})`)
42
- .each((d, i, nodes) => {
43
- const eachCell = d3.select(nodes[i])
44
-
45
- const tile = eachCell
46
- .selectAll<SVGRectElement, d3.HierarchyRectangularNode<ComputedDataTree>>(`rect.${tileClassName}`)
47
- .data([d], d => d.data.id)
48
- .join('rect')
49
- .attr("id", d => d.data.id)
50
- .attr("class", tileClassName)
51
- .attr('cursor', 'pointer')
52
- .attr("width", (d) => d.x1 - d.x0)
53
- .attr("height", (d) => d.y1 - d.y0)
54
- .attr('fill', d => d.data.color)
55
- .attr('data-name', d => d.data.label)
56
- .attr('data-category', d => d.data.categoryLabel)
57
- .attr('data-value', d => d.data.value)
58
-
59
- const label = eachCell
60
- .selectAll('g')
61
- .data([d])
62
- .join('g')
63
- .each((d, i, nodes) => {
64
- const eachLabel = d3.select(nodes[i])
65
- const text = eachLabel
66
- .selectAll('text')
67
- .data([d])
68
- .join('text')
69
- .text(d => d.data.label)
70
- .attr('dominant-baseline', 'hanging')
71
- .attr("x", padding)
72
- .attr("y", padding)
73
- .attr('font-size', fullChartParams.styles.textSize)
74
- .each(function(d) {
75
- // -- tspan(自動斷行) --
76
- const textElement = d3.select(this);
77
- const words = d.data.label.split(/\s+/).reverse() // 以空隔分割字串
78
- let word;
79
- let line: string[] = []
80
- const x = textElement.attr("x")
81
- let y = textElement.attr("y")
82
- let dy = 0
83
- let tspan = textElement
84
- .text(null)
85
- .append("tspan")
86
- .attr('cursor', 'pointer')
87
- .attr('fill', getColor(fullParams.labelColorType, fullChartParams))
88
- .attr('font-size', fullChartParams.styles.textSize)
89
- .attr("x", x)
90
- .attr("y", y)
91
-
92
- while (word = words.pop()) {
93
- line.push(word)
94
- tspan.text(line.join(" "))
95
- if (tspan.node().getComputedTextLength() > (d.x1 - d.x0 - padding)) {
96
- line.pop()
97
- tspan.text(line.join(" "))
98
- line = [word]
99
- dy += lineHeight
100
- tspan = textElement
101
- .append("tspan")
102
- .attr('cursor', 'pointer')
103
- .attr('fill', getColor(fullParams.labelColorType, fullChartParams))
104
- .attr('font-size', fullChartParams.styles.textSize)
105
- .attr("x", x)
106
- .attr("y", y)
107
- .attr("dy", dy + "px")
108
- .text(word)
109
- }
110
- }
111
- })
112
- })
113
-
114
- })
115
-
116
- return cell
117
- }
118
-
119
- function highlight ({ selection, ids, fullChartParams }: {
120
- selection: d3.Selection<any, d3.HierarchyRectangularNode<ComputedDataTree>, any, any>
121
- ids: string[]
122
- fullChartParams: ChartParams
123
- }) {
124
- selection.interrupt('highlight')
125
-
126
- if (!ids.length) {
127
- // remove highlight
128
- selection
129
- .transition('highlight')
130
- .duration(200)
131
- .style('opacity', 1)
132
- return
133
- }
134
-
135
- selection
136
- .each((d, i, n) => {
137
- if (ids.includes(d.data.id)) {
138
- d3.select(n[i])
139
- .style('opacity', 1)
140
- } else {
141
- d3.select(n[i])
142
- .style('opacity', fullChartParams.styles.unhighlightedOpacity)
143
- }
144
- })
145
- }
146
-
147
- export const TreeMap = defineTreePlugin(pluginName, DEFAULT_TREE_MAP_PARAMS)(({ selection, name, subject, observer }) => {
148
- const destroy$ = new Subject()
149
-
150
- const treeData$ = combineLatest({
151
- layout: observer.layout$,
152
- visibleComputedData: observer.visibleComputedData$,
153
- fullParams: observer.fullParams$,
154
- fullDataFormatter: observer.fullDataFormatter$,
155
- fullChartParams: observer.fullChartParams$
156
- }).pipe(
157
- takeUntil(destroy$),
158
- switchMap(async d => d),
159
- map(data => {
160
- const treemap = d3.treemap()
161
- .size([data.layout.width, data.layout.height])
162
- .paddingInner(data.fullParams.paddingInner)
163
- .paddingOuter(data.fullParams.paddingOuter)
164
- .round(true)
165
- .tile(d3.treemapSquarify.ratio(data.fullParams.squarifyRatio))
166
-
167
- const root = d3.hierarchy(data.visibleComputedData)
168
- .sum(d => d.value)
169
- .sort(data.fullParams.sort as (a: any, b: any) => number)
170
-
171
- //call treemap
172
- treemap(root)
173
-
174
- const treeData: d3.HierarchyRectangularNode<ComputedDataTree>[] = root.leaves() as any
175
-
176
- return treeData
177
- })
178
- )
179
-
180
- const cellSelection$ = combineLatest({
181
- selection: of(selection),
182
- treeData: treeData$,
183
- fullParams: observer.fullParams$,
184
- fullChartParams: observer.fullChartParams$,
185
- textSizePx: observer.textSizePx$
186
- }).pipe(
187
- takeUntil(destroy$),
188
- switchMap(async d => d),
189
- map(data => {
190
- return renderTree({
191
- selection,
192
- treeData: data.treeData,
193
- fullParams: data.fullParams,
194
- fullChartParams: data.fullChartParams,
195
- textSizePx: data.textSizePx
196
- })
197
- })
198
- )
199
-
200
- const highlightTarget$ = observer.fullChartParams$.pipe(
201
- takeUntil(destroy$),
202
- map(d => d.highlightTarget),
203
- distinctUntilChanged()
204
- )
205
-
206
- combineLatest({
207
- cellSelection: cellSelection$,
208
- computedData: observer.computedData$,
209
- treeData: treeData$,
210
- fullParams: observer.fullParams$,
211
- fullChartParams: observer.fullChartParams$,
212
- highlightTarget: highlightTarget$,
213
- CategoryDataMap: observer.CategoryDataMap$,
214
- }).pipe(
215
- takeUntil(destroy$),
216
- switchMap(async d => d)
217
- ).subscribe(data => {
218
- data.cellSelection
219
- .on('mouseover', (event, datum) => {
220
- event.stopPropagation()
221
-
222
- subject.event$.next({
223
- type: 'tree',
224
- eventName: 'mouseover',
225
- pluginName,
226
- highlightTarget: data.highlightTarget,
227
- datum: datum.data,
228
- category: data.CategoryDataMap.get(datum.data.categoryLabel)!,
229
- categoryIndex: datum.data.categoryIndex,
230
- categoryLabel: datum.data.categoryLabel,
231
- event,
232
- data: data.computedData
233
- })
234
- })
235
- .on('mousemove', (event, datum) => {
236
- event.stopPropagation()
237
-
238
- subject.event$.next({
239
- type: 'tree',
240
- eventName: 'mousemove',
241
- pluginName,
242
- highlightTarget: data.highlightTarget,
243
- datum: datum.data,
244
- category: data.CategoryDataMap.get(datum.data.categoryLabel)!,
245
- categoryIndex: datum.data.categoryIndex,
246
- categoryLabel: datum.data.categoryLabel,
247
- event,
248
- data: data.computedData
249
- })
250
- })
251
- .on('mouseout', (event, datum) => {
252
- event.stopPropagation()
253
-
254
- subject.event$.next({
255
- type: 'tree',
256
- eventName: 'mouseout',
257
- pluginName,
258
- highlightTarget: data.highlightTarget,
259
- datum: datum.data,
260
- category: data.CategoryDataMap.get(datum.data.categoryLabel)!,
261
- categoryIndex: datum.data.categoryIndex,
262
- categoryLabel: datum.data.categoryLabel,
263
- event,
264
- data: data.computedData
265
- })
266
- })
267
- .on('click', (event, datum) => {
268
- event.stopPropagation()
269
-
270
- subject.event$.next({
271
- type: 'tree',
272
- eventName: 'click',
273
- pluginName,
274
- highlightTarget: data.highlightTarget,
275
- datum: datum.data,
276
- category: data.CategoryDataMap.get(datum.data.categoryLabel)!,
277
- categoryIndex: datum.data.categoryIndex,
278
- categoryLabel: datum.data.categoryLabel,
279
- event,
280
- data: data.computedData
281
- })
282
- })
283
- })
284
-
285
- combineLatest({
286
- cellSelection: cellSelection$,
287
- highlight: observer.treeHighlight$.pipe(
288
- map(data => data.map(d => d.id))
289
- ),
290
- fullChartParams: observer.fullChartParams$
291
- }).pipe(
292
- takeUntil(destroy$),
293
- switchMap(async d => d)
294
- ).subscribe(data => {
295
- highlight({
296
- selection: data.cellSelection,
297
- ids: data.highlight,
298
- fullChartParams: data.fullChartParams
299
- })
300
- })
301
-
302
- return () => {
303
- destroy$.next(undefined)
304
- }
305
- })
1
+ import * as d3 from 'd3'
2
+ import {
3
+ Subject,
4
+ Observable,
5
+ of,
6
+ takeUntil,
7
+ map,
8
+ switchMap,
9
+ combineLatest,
10
+ debounceTime,
11
+ distinctUntilChanged } from 'rxjs'
12
+ import {
13
+ defineTreePlugin } from '@orbcharts/core'
14
+ import type { Layout, ComputedDataTree, DataFormatterTree, ChartParams } from '@orbcharts/core'
15
+ import type { TreeMapParams } from '../types'
16
+ import { DEFAULT_TREE_MAP_PARAMS } from '../defaults'
17
+ import { getClassName, getColor } from '../../utils/orbchartsUtils'
18
+
19
+ const pluginName = 'TreeMap'
20
+ const treeClassName = getClassName(pluginName, 'tree')
21
+ const tileClassName = getClassName(pluginName, 'tile')
22
+
23
+ function renderTree ({ selection, treeData, fullParams, fullChartParams, textSizePx }: {
24
+ selection: d3.Selection<any, any, any, any>
25
+ treeData: d3.HierarchyRectangularNode<ComputedDataTree>[]
26
+ fullParams: TreeMapParams
27
+ fullChartParams: ChartParams
28
+ textSizePx: number
29
+ }) {
30
+ const padding = textSizePx / 2
31
+ const lineHeight = textSizePx // 行高
32
+
33
+ const cell = selection.selectAll<SVGGElement, d3.HierarchyRectangularNode<ComputedDataTree>>(`g.${treeClassName}`)
34
+ .data(treeData, d => d.data.id)
35
+ .join('g')
36
+ .attr('class', treeClassName)
37
+
38
+ cell
39
+ // .transition()
40
+ // .duration(fullChartParams.transitionDuration)
41
+ .attr('transform', (d) => !d.x0 || !d.y0 ? null : `translate(${d.x0},${d.y0})`)
42
+ .each((d, i, nodes) => {
43
+ const eachCell = d3.select(nodes[i])
44
+
45
+ const tile = eachCell
46
+ .selectAll<SVGRectElement, d3.HierarchyRectangularNode<ComputedDataTree>>(`rect.${tileClassName}`)
47
+ .data([d], d => d.data.id)
48
+ .join('rect')
49
+ .attr("id", d => d.data.id)
50
+ .attr("class", tileClassName)
51
+ .attr('cursor', 'pointer')
52
+ .attr("width", (d) => d.x1 - d.x0)
53
+ .attr("height", (d) => d.y1 - d.y0)
54
+ .attr('fill', d => d.data.color)
55
+ .attr('data-name', d => d.data.label)
56
+ .attr('data-category', d => d.data.categoryLabel)
57
+ .attr('data-value', d => d.data.value)
58
+
59
+ const label = eachCell
60
+ .selectAll('g')
61
+ .data([d])
62
+ .join('g')
63
+ .each((d, i, nodes) => {
64
+ const eachLabel = d3.select(nodes[i])
65
+ const text = eachLabel
66
+ .selectAll('text')
67
+ .data([d])
68
+ .join('text')
69
+ .text(d => d.data.label)
70
+ .attr('dominant-baseline', 'hanging')
71
+ .attr("x", padding)
72
+ .attr("y", padding)
73
+ .attr('font-size', fullChartParams.styles.textSize)
74
+ .each(function(d) {
75
+ // -- tspan(自動斷行) --
76
+ const textElement = d3.select(this);
77
+ const words = d.data.label.split(/\s+/).reverse() // 以空隔分割字串
78
+ let word;
79
+ let line: string[] = []
80
+ const x = textElement.attr("x")
81
+ let y = textElement.attr("y")
82
+ let dy = 0
83
+ let tspan = textElement
84
+ .text(null)
85
+ .append("tspan")
86
+ .attr('cursor', 'pointer')
87
+ .attr('fill', getColor(fullParams.labelColorType, fullChartParams))
88
+ .attr('font-size', fullChartParams.styles.textSize)
89
+ .attr("x", x)
90
+ .attr("y", y)
91
+
92
+ while (word = words.pop()) {
93
+ line.push(word)
94
+ tspan.text(line.join(" "))
95
+ if (tspan.node().getComputedTextLength() > (d.x1 - d.x0 - padding)) {
96
+ line.pop()
97
+ tspan.text(line.join(" "))
98
+ line = [word]
99
+ dy += lineHeight
100
+ tspan = textElement
101
+ .append("tspan")
102
+ .attr('cursor', 'pointer')
103
+ .attr('fill', getColor(fullParams.labelColorType, fullChartParams))
104
+ .attr('font-size', fullChartParams.styles.textSize)
105
+ .attr("x", x)
106
+ .attr("y", y)
107
+ .attr("dy", dy + "px")
108
+ .text(word)
109
+ }
110
+ }
111
+ })
112
+ })
113
+
114
+ })
115
+
116
+ return cell
117
+ }
118
+
119
+ function highlight ({ selection, ids, fullChartParams }: {
120
+ selection: d3.Selection<any, d3.HierarchyRectangularNode<ComputedDataTree>, any, any>
121
+ ids: string[]
122
+ fullChartParams: ChartParams
123
+ }) {
124
+ selection.interrupt('highlight')
125
+
126
+ if (!ids.length) {
127
+ // remove highlight
128
+ selection
129
+ .transition('highlight')
130
+ .duration(200)
131
+ .style('opacity', 1)
132
+ return
133
+ }
134
+
135
+ selection
136
+ .each((d, i, n) => {
137
+ if (ids.includes(d.data.id)) {
138
+ d3.select(n[i])
139
+ .style('opacity', 1)
140
+ } else {
141
+ d3.select(n[i])
142
+ .style('opacity', fullChartParams.styles.unhighlightedOpacity)
143
+ }
144
+ })
145
+ }
146
+
147
+ export const TreeMap = defineTreePlugin(pluginName, DEFAULT_TREE_MAP_PARAMS)(({ selection, name, subject, observer }) => {
148
+ const destroy$ = new Subject()
149
+
150
+ const treeData$ = combineLatest({
151
+ layout: observer.layout$,
152
+ visibleComputedData: observer.visibleComputedData$,
153
+ fullParams: observer.fullParams$,
154
+ fullDataFormatter: observer.fullDataFormatter$,
155
+ fullChartParams: observer.fullChartParams$
156
+ }).pipe(
157
+ takeUntil(destroy$),
158
+ switchMap(async d => d),
159
+ map(data => {
160
+ const treemap = d3.treemap()
161
+ .size([data.layout.width, data.layout.height])
162
+ .paddingInner(data.fullParams.paddingInner)
163
+ .paddingOuter(data.fullParams.paddingOuter)
164
+ .round(true)
165
+ .tile(d3.treemapSquarify.ratio(data.fullParams.squarifyRatio))
166
+
167
+ const root = d3.hierarchy(data.visibleComputedData)
168
+ .sum(d => d.value)
169
+ .sort(data.fullParams.sort as (a: any, b: any) => number)
170
+
171
+ //call treemap
172
+ treemap(root)
173
+
174
+ const treeData: d3.HierarchyRectangularNode<ComputedDataTree>[] = root.leaves() as any
175
+
176
+ return treeData
177
+ })
178
+ )
179
+
180
+ const cellSelection$ = combineLatest({
181
+ selection: of(selection),
182
+ treeData: treeData$,
183
+ fullParams: observer.fullParams$,
184
+ fullChartParams: observer.fullChartParams$,
185
+ textSizePx: observer.textSizePx$
186
+ }).pipe(
187
+ takeUntil(destroy$),
188
+ switchMap(async d => d),
189
+ map(data => {
190
+ return renderTree({
191
+ selection,
192
+ treeData: data.treeData,
193
+ fullParams: data.fullParams,
194
+ fullChartParams: data.fullChartParams,
195
+ textSizePx: data.textSizePx
196
+ })
197
+ })
198
+ )
199
+
200
+ const highlightTarget$ = observer.fullChartParams$.pipe(
201
+ takeUntil(destroy$),
202
+ map(d => d.highlightTarget),
203
+ distinctUntilChanged()
204
+ )
205
+
206
+ combineLatest({
207
+ cellSelection: cellSelection$,
208
+ computedData: observer.computedData$,
209
+ treeData: treeData$,
210
+ fullParams: observer.fullParams$,
211
+ fullChartParams: observer.fullChartParams$,
212
+ highlightTarget: highlightTarget$,
213
+ CategoryDataMap: observer.CategoryDataMap$,
214
+ }).pipe(
215
+ takeUntil(destroy$),
216
+ switchMap(async d => d)
217
+ ).subscribe(data => {
218
+ data.cellSelection
219
+ .on('mouseover', (event, datum) => {
220
+ event.stopPropagation()
221
+
222
+ subject.event$.next({
223
+ type: 'tree',
224
+ eventName: 'mouseover',
225
+ pluginName,
226
+ highlightTarget: data.highlightTarget,
227
+ datum: datum.data,
228
+ category: data.CategoryDataMap.get(datum.data.categoryLabel)!,
229
+ categoryIndex: datum.data.categoryIndex,
230
+ categoryLabel: datum.data.categoryLabel,
231
+ event,
232
+ data: data.computedData
233
+ })
234
+ })
235
+ .on('mousemove', (event, datum) => {
236
+ event.stopPropagation()
237
+
238
+ subject.event$.next({
239
+ type: 'tree',
240
+ eventName: 'mousemove',
241
+ pluginName,
242
+ highlightTarget: data.highlightTarget,
243
+ datum: datum.data,
244
+ category: data.CategoryDataMap.get(datum.data.categoryLabel)!,
245
+ categoryIndex: datum.data.categoryIndex,
246
+ categoryLabel: datum.data.categoryLabel,
247
+ event,
248
+ data: data.computedData
249
+ })
250
+ })
251
+ .on('mouseout', (event, datum) => {
252
+ event.stopPropagation()
253
+
254
+ subject.event$.next({
255
+ type: 'tree',
256
+ eventName: 'mouseout',
257
+ pluginName,
258
+ highlightTarget: data.highlightTarget,
259
+ datum: datum.data,
260
+ category: data.CategoryDataMap.get(datum.data.categoryLabel)!,
261
+ categoryIndex: datum.data.categoryIndex,
262
+ categoryLabel: datum.data.categoryLabel,
263
+ event,
264
+ data: data.computedData
265
+ })
266
+ })
267
+ .on('click', (event, datum) => {
268
+ event.stopPropagation()
269
+
270
+ subject.event$.next({
271
+ type: 'tree',
272
+ eventName: 'click',
273
+ pluginName,
274
+ highlightTarget: data.highlightTarget,
275
+ datum: datum.data,
276
+ category: data.CategoryDataMap.get(datum.data.categoryLabel)!,
277
+ categoryIndex: datum.data.categoryIndex,
278
+ categoryLabel: datum.data.categoryLabel,
279
+ event,
280
+ data: data.computedData
281
+ })
282
+ })
283
+ })
284
+
285
+ combineLatest({
286
+ cellSelection: cellSelection$,
287
+ highlight: observer.treeHighlight$.pipe(
288
+ map(data => data.map(d => d.id))
289
+ ),
290
+ fullChartParams: observer.fullChartParams$
291
+ }).pipe(
292
+ takeUntil(destroy$),
293
+ switchMap(async d => d)
294
+ ).subscribe(data => {
295
+ highlight({
296
+ selection: data.cellSelection,
297
+ ids: data.highlight,
298
+ fullChartParams: data.fullChartParams
299
+ })
300
+ })
301
+
302
+ return () => {
303
+ destroy$.next(undefined)
304
+ }
305
+ })
package/src/tree/types.ts CHANGED
@@ -1,23 +1,23 @@
1
- import type { ColorType, ComputedDataTree } from '@orbcharts/core'
2
-
3
- export interface TreeMapParams {
4
- paddingInner: number
5
- paddingOuter: number
6
- labelColorType: ColorType
7
- squarifyRatio: number
8
- sort: (a: ComputedDataTree, b: ComputedDataTree) => number
9
- }
10
-
11
- export interface TreeLegendParams {
12
- position: 'top' | 'bottom' | 'left' | 'right'
13
- justify: 'start' | 'center' | 'end'
14
- padding: number
15
- backgroundFill: ColorType
16
- backgroundStroke: ColorType
17
- gap: number
18
- listRectWidth: number
19
- listRectHeight: number
20
- listRectRadius: number
21
- textColorType: ColorType
22
- }
23
-
1
+ import type { ColorType, ComputedDataTree } from '@orbcharts/core'
2
+
3
+ export interface TreeMapParams {
4
+ paddingInner: number
5
+ paddingOuter: number
6
+ labelColorType: ColorType
7
+ squarifyRatio: number
8
+ sort: (a: ComputedDataTree, b: ComputedDataTree) => number
9
+ }
10
+
11
+ export interface TreeLegendParams {
12
+ position: 'top' | 'bottom' | 'left' | 'right'
13
+ justify: 'start' | 'center' | 'end'
14
+ padding: number
15
+ backgroundFill: ColorType
16
+ backgroundStroke: ColorType
17
+ gap: number
18
+ listRectWidth: number
19
+ listRectHeight: number
20
+ listRectRadius: number
21
+ textColorType: ColorType
22
+ }
23
+