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

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 (169) hide show
  1. package/LICENSE +200 -200
  2. package/dist/orbcharts-plugins-basic/lib/plugins-basic-types.d.ts +1 -1
  3. package/dist/orbcharts-plugins-basic/src/base/{BaseBarStack.d.ts → BaseStackedBar.d.ts} +4 -4
  4. package/dist/orbcharts-plugins-basic/src/grid/defaults.d.ts +3 -3
  5. package/dist/orbcharts-plugins-basic/src/grid/index.d.ts +2 -2
  6. package/dist/orbcharts-plugins-basic/src/grid/plugins/Bars.d.ts +1 -1
  7. package/dist/orbcharts-plugins-basic/src/grid/plugins/BarsPN.d.ts +1 -1
  8. package/dist/orbcharts-plugins-basic/src/grid/plugins/BarsTriangle.d.ts +1 -1
  9. package/dist/orbcharts-plugins-basic/src/grid/plugins/Dots.d.ts +1 -1
  10. package/dist/orbcharts-plugins-basic/src/grid/plugins/GridLegend.d.ts +1 -1
  11. package/dist/orbcharts-plugins-basic/src/grid/plugins/GridTooltip.d.ts +1 -1
  12. package/dist/orbcharts-plugins-basic/src/grid/plugins/GridZoom.d.ts +1 -1
  13. package/dist/orbcharts-plugins-basic/src/grid/plugins/GroupAux.d.ts +1 -1
  14. package/dist/orbcharts-plugins-basic/src/grid/plugins/GroupAxis.d.ts +1 -1
  15. package/dist/orbcharts-plugins-basic/src/grid/plugins/LineAreas.d.ts +1 -1
  16. package/dist/orbcharts-plugins-basic/src/grid/plugins/Lines.d.ts +1 -1
  17. package/dist/orbcharts-plugins-basic/src/grid/plugins/StackedBar.d.ts +1 -0
  18. package/dist/orbcharts-plugins-basic/src/grid/plugins/StackedValueAxis.d.ts +1 -0
  19. package/dist/orbcharts-plugins-basic/src/grid/plugins/ValueAxis.d.ts +1 -1
  20. package/dist/orbcharts-plugins-basic/src/index.d.ts +1 -0
  21. package/dist/orbcharts-plugins-basic/src/multiGrid/defaults.d.ts +4 -4
  22. package/dist/orbcharts-plugins-basic/src/multiGrid/index.d.ts +3 -3
  23. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiBars.d.ts +1 -1
  24. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiBarsTriangle.d.ts +1 -1
  25. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiDots.d.ts +1 -1
  26. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiGridLegend.d.ts +1 -1
  27. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiGridTooltip.d.ts +1 -1
  28. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiGroupAxis.d.ts +1 -1
  29. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiLineAreas.d.ts +1 -1
  30. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiLines.d.ts +1 -1
  31. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiStackedBar.d.ts +1 -0
  32. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiStackedValueAxis.d.ts +1 -0
  33. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiValueAxis.d.ts +1 -1
  34. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/OverlappingStackedValueAxes.d.ts +1 -0
  35. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/OverlappingValueAxes.d.ts +1 -1
  36. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/MultiValueLegend.d.ts +1 -1
  37. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/MultiValueTooltip.d.ts +1 -1
  38. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/Scatter.d.ts +1 -1
  39. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/ScatterBubbles.d.ts +1 -1
  40. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/XYAux.d.ts +1 -1
  41. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/XYAxes.d.ts +1 -1
  42. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/XYZoom.d.ts +1 -1
  43. package/dist/orbcharts-plugins-basic/src/noneData/plugins/Container.d.ts +0 -1
  44. package/dist/orbcharts-plugins-basic/src/noneData/plugins/Tooltip.d.ts +0 -3
  45. package/dist/orbcharts-plugins-basic/src/relationship/defaults.d.ts +6 -0
  46. package/dist/orbcharts-plugins-basic/src/relationship/index.d.ts +5 -0
  47. package/dist/orbcharts-plugins-basic/src/relationship/plugins/ForceDirected.d.ts +3 -0
  48. package/dist/orbcharts-plugins-basic/src/relationship/plugins/ForceDirectedBubbles.d.ts +3 -0
  49. package/dist/orbcharts-plugins-basic/src/relationship/plugins/RelationshipLegend.d.ts +1 -0
  50. package/dist/orbcharts-plugins-basic/src/relationship/plugins/RelationshipTooltip.d.ts +1 -0
  51. package/dist/orbcharts-plugins-basic/src/relationship/relationshipObservables.d.ts +1 -0
  52. package/dist/orbcharts-plugins-basic/src/series/plugins/Bubbles.d.ts +1 -1
  53. package/dist/orbcharts-plugins-basic/src/series/plugins/Pie.d.ts +1 -1
  54. package/dist/orbcharts-plugins-basic/src/series/plugins/PieEventTexts.d.ts +1 -1
  55. package/dist/orbcharts-plugins-basic/src/series/plugins/PieLabels.d.ts +1 -1
  56. package/dist/orbcharts-plugins-basic/src/series/plugins/Rose.d.ts +1 -1
  57. package/dist/orbcharts-plugins-basic/src/series/plugins/RoseLabels.d.ts +1 -1
  58. package/dist/orbcharts-plugins-basic/src/series/plugins/SeriesLegend.d.ts +1 -1
  59. package/dist/orbcharts-plugins-basic/src/series/plugins/SeriesTooltip.d.ts +1 -1
  60. package/dist/orbcharts-plugins-basic/src/tree/plugins/TreeLegend.d.ts +1 -1
  61. package/dist/orbcharts-plugins-basic/src/tree/plugins/TreeMap.d.ts +1 -1
  62. package/dist/orbcharts-plugins-basic/src/tree/plugins/TreeTooltip.d.ts +1 -1
  63. package/dist/orbcharts-plugins-basic/src/utils/commonUtils.d.ts +1 -1
  64. package/dist/orbcharts-plugins-basic/src/utils/orbchartsUtils.d.ts +1 -1
  65. package/dist/orbcharts-plugins-basic.es.js +9421 -8102
  66. package/dist/orbcharts-plugins-basic.umd.js +69 -51
  67. package/lib/core-types.ts +7 -7
  68. package/lib/core.ts +6 -6
  69. package/lib/plugins-basic-types.ts +6 -6
  70. package/package.json +44 -44
  71. package/src/base/BaseBars.ts +765 -765
  72. package/src/base/BaseBarsTriangle.ts +676 -676
  73. package/src/base/BaseDots.ts +464 -464
  74. package/src/base/BaseGroupAxis.ts +679 -679
  75. package/src/base/BaseLegend.ts +684 -684
  76. package/src/base/BaseLineAreas.ts +629 -629
  77. package/src/base/BaseLines.ts +706 -706
  78. package/src/base/{BaseBarStack.ts → BaseStackedBar.ts} +782 -782
  79. package/src/base/BaseTooltip.ts +385 -385
  80. package/src/base/BaseValueAxis.ts +583 -583
  81. package/src/base/types.ts +2 -2
  82. package/src/const.ts +30 -30
  83. package/src/grid/defaults.ts +246 -246
  84. package/src/grid/gridObservables.ts +554 -554
  85. package/src/grid/index.ts +16 -16
  86. package/src/grid/plugins/Bars.ts +69 -69
  87. package/src/grid/plugins/BarsPN.ts +66 -66
  88. package/src/grid/plugins/BarsTriangle.ts +73 -73
  89. package/src/grid/plugins/Dots.ts +68 -68
  90. package/src/grid/plugins/GridLegend.ts +107 -107
  91. package/src/grid/plugins/GridTooltip.ts +66 -66
  92. package/src/grid/plugins/GridZoom.ts +218 -218
  93. package/src/grid/plugins/GroupAux.ts +1103 -1103
  94. package/src/grid/plugins/GroupAxis.ts +97 -97
  95. package/src/grid/plugins/LineAreas.ts +65 -65
  96. package/src/grid/plugins/Lines.ts +59 -59
  97. package/src/grid/plugins/{BarStack.ts → StackedBar.ts} +64 -64
  98. package/src/grid/plugins/{ValueStackAxis.ts → StackedValueAxis.ts} +96 -96
  99. package/src/grid/plugins/ValueAxis.ts +94 -94
  100. package/src/index.ts +6 -10
  101. package/src/multiGrid/defaults.ts +224 -224
  102. package/src/multiGrid/index.ts +15 -15
  103. package/src/multiGrid/multiGridObservables.ts +49 -49
  104. package/src/multiGrid/plugins/MultiBars.ts +108 -108
  105. package/src/multiGrid/plugins/MultiBarsTriangle.ts +114 -114
  106. package/src/multiGrid/plugins/MultiDots.ts +102 -102
  107. package/src/multiGrid/plugins/MultiGridLegend.ts +159 -159
  108. package/src/multiGrid/plugins/MultiGridTooltip.ts +66 -66
  109. package/src/multiGrid/plugins/MultiGroupAxis.ts +137 -137
  110. package/src/multiGrid/plugins/MultiLineAreas.ts +107 -107
  111. package/src/multiGrid/plugins/MultiLines.ts +101 -101
  112. package/src/multiGrid/plugins/{MultiBarStack.ts → MultiStackedBar.ts} +106 -106
  113. package/src/multiGrid/plugins/{MultiValueStackAxis.ts → MultiStackedValueAxis.ts} +134 -134
  114. package/src/multiGrid/plugins/MultiValueAxis.ts +134 -134
  115. package/src/multiGrid/plugins/{OverlappingValueStackAxes.ts → OverlappingStackedValueAxes.ts} +299 -299
  116. package/src/multiGrid/plugins/OverlappingValueAxes.ts +300 -300
  117. package/src/multiValue/defaults.ts +166 -166
  118. package/src/multiValue/index.ts +8 -8
  119. package/src/multiValue/multiValueObservables.ts +297 -297
  120. package/src/multiValue/plugins/MultiValueLegend.ts +107 -107
  121. package/src/multiValue/plugins/MultiValueTooltip.ts +66 -66
  122. package/src/multiValue/plugins/Scatter.ts +426 -426
  123. package/src/multiValue/plugins/ScatterBubbles.ts +554 -554
  124. package/src/multiValue/plugins/XYAux.ts +681 -681
  125. package/src/multiValue/plugins/XYAxes.ts +684 -684
  126. package/src/multiValue/plugins/XYZoom.ts +299 -299
  127. package/src/noneData/defaults.ts +102 -102
  128. package/src/noneData/index.ts +3 -3
  129. package/src/noneData/plugins/Container.ts +28 -28
  130. package/src/noneData/plugins/Tooltip.ts +374 -374
  131. package/src/relationship/defaults.ts +196 -0
  132. package/src/relationship/index.ts +5 -0
  133. package/src/relationship/plugins/ForceDirected.ts +1167 -0
  134. package/src/relationship/plugins/ForceDirectedBubbles.ts +1391 -0
  135. package/src/relationship/plugins/RelationshipLegend.ts +100 -0
  136. package/src/relationship/plugins/RelationshipTooltip.ts +66 -0
  137. package/src/relationship/relationshipObservables.ts +50 -0
  138. package/src/series/defaults.ts +206 -206
  139. package/src/series/index.ts +9 -9
  140. package/src/series/plugins/Bubbles.ts +606 -603
  141. package/src/series/plugins/Pie.ts +623 -623
  142. package/src/series/plugins/PieEventTexts.ts +284 -283
  143. package/src/series/plugins/PieLabels.ts +640 -640
  144. package/src/series/plugins/Rose.ts +516 -516
  145. package/src/series/plugins/RoseLabels.ts +600 -600
  146. package/src/series/plugins/SeriesLegend.ts +107 -107
  147. package/src/series/plugins/SeriesTooltip.ts +66 -66
  148. package/src/series/seriesObservables.ts +145 -145
  149. package/src/series/seriesUtils.ts +51 -51
  150. package/src/tree/defaults.ts +78 -78
  151. package/src/tree/index.ts +4 -4
  152. package/src/tree/plugins/TreeLegend.ts +100 -100
  153. package/src/tree/plugins/TreeMap.ts +333 -333
  154. package/src/tree/plugins/TreeTooltip.ts +66 -66
  155. package/src/utils/commonUtils.ts +21 -21
  156. package/src/utils/d3Graphics.ts +174 -174
  157. package/src/utils/d3Utils.ts +74 -74
  158. package/src/utils/observables.ts +14 -14
  159. package/src/utils/orbchartsUtils.ts +115 -115
  160. package/tsconfig.base.json +13 -13
  161. package/tsconfig.json +2 -2
  162. package/vite.config.js +22 -22
  163. package/dist/orbcharts-plugins-basic/src/grid/plugins/BarStack.d.ts +0 -1
  164. package/dist/orbcharts-plugins-basic/src/grid/plugins/ValueStackAxis.d.ts +0 -1
  165. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiBarStack.d.ts +0 -1
  166. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiValueStackAxis.d.ts +0 -1
  167. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/OverlappingValueStackAxes.d.ts +0 -1
  168. package/dist/orbcharts-plugins-basic/src/relationship/plugins/Relationship.d.ts +0 -0
  169. package/src/relationship/plugins/Relationship.ts +0 -0
@@ -1,283 +1,284 @@
1
- import * as d3 from 'd3'
2
- import {
3
- combineLatest,
4
- switchMap,
5
- first,
6
- takeUntil,
7
- map,
8
- distinctUntilChanged,
9
- Observable,
10
- Subject } from 'rxjs'
11
- import type { DefinePluginConfig } from '../../../lib/core-types'
12
- import type { Subscription } from 'rxjs'
13
- import {
14
- defineSeriesPlugin} from '../../../lib/core'
15
- import type {
16
- ComputedDatumSeries,
17
- ChartParams,
18
- ContainerPosition,
19
- EventName,
20
- EventSeries } from '../../../lib/core-types'
21
- import type { PieEventTextsParams } from '../../../lib/plugins-basic-types'
22
- import { DEFAULT_PIE_EVENT_TEXTS_PARAMS } from '../defaults'
23
- import { getD3TransitionEase } from '../../utils/d3Utils'
24
- import { getClassName } from '../../utils/orbchartsUtils'
25
- import { seriesCenterSelectionObservable } from '../seriesObservables'
26
- import { LAYER_INDEX_OF_LABEL } from '../../const'
27
-
28
- type TextDatum = {
29
- text: string
30
- attr: { [key:string]: any }
31
- style: { [key:string]: any }
32
- }
33
-
34
- const pluginName = 'PieEventTexts'
35
- const textClassName = getClassName(pluginName, 'text')
36
-
37
- const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_PIE_EVENT_TEXTS_PARAMS> = {
38
- name: pluginName,
39
- defaultParams: DEFAULT_PIE_EVENT_TEXTS_PARAMS,
40
- layerIndex: LAYER_INDEX_OF_LABEL,
41
- validator: (params, { validateColumns }) => {
42
- const result = validateColumns(params, {
43
- eventFn: {
44
- toBeTypes: ['Function'],
45
- },
46
- textAttrs: {
47
- toBeTypes: ['object[]'],
48
- },
49
- textStyles: {
50
- toBeTypes: ['object[]'],
51
- }
52
- })
53
- return result
54
- }
55
- }
56
-
57
- function renderText (
58
- selection: d3.Selection<SVGGElement, unknown, any, any>,
59
- data: Array<TextDatum>
60
- ): d3.Selection<SVGTextElement, TextDatum, SVGGElement, unknown> {
61
- const textUpdate = selection
62
- .selectAll<SVGTextElement, TextDatum>(`text.${textClassName}`)
63
- .data(data)
64
- const textEnter = textUpdate.enter()
65
- .append('text')
66
- .classed(textClassName, true)
67
- const text = textUpdate.merge(textEnter)
68
- text
69
- .each((d, i, g) => {
70
- const t = d3.select(g[i])
71
- .text(d.text)
72
- Object.keys(d.attr)
73
- .forEach(key => {
74
- t.attr(key, d.attr[key])
75
- })
76
- Object.keys(d.style)
77
- .forEach(key => {
78
- t.style(key, d.style[key])
79
- })
80
- })
81
-
82
- textUpdate.exit().remove()
83
-
84
- return text
85
- }
86
-
87
- function makeTextData ({ eventData, eventName, t, eventFn, textAttrs, textStyles }: {
88
- eventData: EventSeries,
89
- eventName: EventName,
90
- t: number,
91
- eventFn: (d: EventSeries, eventName: EventName, t: number) => string[]
92
- textAttrs: Array<{ [key:string]: string | number }>
93
- textStyles: Array<{ [key:string]: string | number }>
94
- }): TextDatum[] {
95
- const callbackText = eventFn(eventData, eventName, t)
96
- return callbackText.map((d, i) => {
97
- return {
98
- text: d,
99
- attr: textAttrs[i],
100
- style: textStyles[i]
101
- }
102
- })
103
- }
104
-
105
- function createEachPieEventTexts (pluginName: string, context: {
106
- containerSelection: d3.Selection<SVGGElement, any, any, unknown>
107
- computedData$: Observable<ComputedDatumSeries[][]>
108
- containerComputedLayoutData$: Observable<ComputedDatumSeries[]>
109
- SeriesDataMap$: Observable<Map<string, ComputedDatumSeries[]>>
110
- fullParams$: Observable<PieEventTextsParams>
111
- fullChartParams$: Observable<ChartParams>
112
- seriesHighlight$: Observable<ComputedDatumSeries[]>
113
- seriesContainerPosition$: Observable<ContainerPosition>
114
- event$: Subject<EventSeries>
115
- }) {
116
- const destroy$ = new Subject()
117
-
118
- // const graphicSelection: d3.Selection<SVGGElement, any, any, any> = selection.append('g')
119
- let centerTextSelection: d3.Selection<SVGTextElement, TextDatum, SVGGElement, unknown> | undefined
120
- let storeEventSubscription: Subscription | undefined
121
-
122
- // context.layout$
123
- // .pipe(
124
- // first()
125
- // )
126
- // .subscribe(size => {
127
- // selection
128
- // .attr('transform', `translate(${size.width / 2}, ${size.height / 2})`)
129
- // context.layout$
130
- // .pipe(
131
- // takeUntil(destroy$)
132
- // )
133
- // .subscribe(size => {
134
- // selection
135
- // .transition()
136
- // .attr('transform', `translate(${size.width / 2}, ${size.height / 2})`)
137
- // })
138
- // })
139
-
140
- const highlightTarget$ = context.fullChartParams$.pipe(
141
- takeUntil(destroy$),
142
- map(d => d.highlightTarget),
143
- distinctUntilChanged()
144
- )
145
-
146
- combineLatest({
147
- computedData: context.computedData$,
148
- fullParams: context.fullParams$,
149
- fullChartParams: context.fullChartParams$,
150
- highlightTarget: highlightTarget$,
151
- }).pipe(
152
- takeUntil(destroy$),
153
- switchMap(async (d) => d),
154
- ).subscribe(data => {
155
-
156
- context.containerSelection
157
- .transition('move')
158
- .duration(data.fullChartParams.transitionDuration!)
159
- // .ease(getD3TransitionEase(data.fullChartParams.transitionEase!))
160
- .tween('move', (event, datum) => {
161
- return (t) => {
162
- const renderData = makeTextData({
163
- eventData: {
164
- type: 'series',
165
- pluginName,
166
- eventName: 'transitionMove',
167
- event,
168
- highlightTarget: data.highlightTarget,
169
- data: data.computedData,
170
- series: [],
171
- seriesIndex: -1,
172
- seriesLabel: '',
173
- datum: null
174
- },
175
- eventName: 'transitionMove',
176
- t,
177
- eventFn: data.fullParams.eventFn!,
178
- textAttrs: data.fullParams.textAttrs!,
179
- textStyles: data.fullParams.textStyles!
180
- })
181
- centerTextSelection = renderText(context.containerSelection, renderData)
182
- }
183
- })
184
- .on('end', (event, datum) => {
185
- const renderData = makeTextData({
186
- eventData: {
187
- type: 'series',
188
- pluginName,
189
- eventName: 'transitionEnd',
190
- event,
191
- highlightTarget: data.highlightTarget,
192
- data: data.computedData,
193
- series: [],
194
- seriesIndex: -1,
195
- seriesLabel: '',
196
- datum: null
197
- },
198
- eventName: 'transitionMove',
199
- t: 1,
200
- eventFn: data.fullParams.eventFn!,
201
- textAttrs: data.fullParams.textAttrs!,
202
- textStyles: data.fullParams.textStyles!
203
- })
204
- centerTextSelection = renderText(context.containerSelection, renderData)
205
-
206
- if (storeEventSubscription) {
207
- storeEventSubscription.unsubscribe()
208
- }
209
- storeEventSubscription = context.event$
210
- .subscribe(eventData => {
211
- const renderData = makeTextData({
212
- eventData,
213
- eventName: eventData.eventName,
214
- t: 1,
215
- eventFn: data.fullParams.eventFn!,
216
- textAttrs: data.fullParams.textAttrs!,
217
- textStyles: data.fullParams.textStyles!
218
- })
219
- centerTextSelection = renderText(context.containerSelection, renderData)
220
- })
221
- })
222
- })
223
-
224
- return () => {
225
- destroy$.next(undefined)
226
- }
227
- }
228
-
229
- export const PieEventTexts = defineSeriesPlugin(pluginConfig)(({ selection, name, observer, subject }) => {
230
- const destroy$ = new Subject()
231
-
232
- const { seriesCenterSelection$ } = seriesCenterSelectionObservable({
233
- selection: selection,
234
- pluginName,
235
- separateSeries$: observer.separateSeries$,
236
- seriesLabels$: observer.seriesLabels$,
237
- seriesContainerPosition$: observer.seriesContainerPosition$
238
- })
239
-
240
- const unsubscribeFnArr: (() => void)[] = []
241
-
242
- seriesCenterSelection$
243
- .pipe(
244
- takeUntil(destroy$)
245
- )
246
- .subscribe(seriesCenterSelection => {
247
- // 每次重新計算時,清除之前的訂閱
248
- unsubscribeFnArr.forEach(fn => fn())
249
-
250
- seriesCenterSelection.each((d, containerIndex, g) => {
251
-
252
- const containerSelection = d3.select(g[containerIndex])
253
-
254
- const containerComputedLayoutData$ = observer.computedLayoutData$.pipe(
255
- takeUntil(destroy$),
256
- map(data => data[containerIndex] ?? data[0])
257
- )
258
-
259
- const containerPosition$ = observer.seriesContainerPosition$.pipe(
260
- takeUntil(destroy$),
261
- map(data => data[containerIndex] ?? data[0])
262
- )
263
-
264
- unsubscribeFnArr[containerIndex] = createEachPieEventTexts(pluginName, {
265
- containerSelection: containerSelection,
266
- computedData$: observer.computedData$,
267
- containerComputedLayoutData$: containerComputedLayoutData$,
268
- SeriesDataMap$: observer.SeriesDataMap$,
269
- fullParams$: observer.fullParams$,
270
- fullChartParams$: observer.fullChartParams$,
271
- seriesHighlight$: observer.seriesHighlight$,
272
- seriesContainerPosition$: containerPosition$,
273
- event$: subject.event$,
274
- })
275
-
276
- })
277
- })
278
-
279
- return () => {
280
- destroy$.next(undefined)
281
- unsubscribeFnArr.forEach(fn => fn())
282
- }
283
- })
1
+ import * as d3 from 'd3'
2
+ import {
3
+ combineLatest,
4
+ switchMap,
5
+ first,
6
+ takeUntil,
7
+ map,
8
+ distinctUntilChanged,
9
+ Observable,
10
+ Subject } from 'rxjs'
11
+ import type { DefinePluginConfig } from '../../../lib/core-types'
12
+ import type { Subscription } from 'rxjs'
13
+ import {
14
+ defineSeriesPlugin} from '../../../lib/core'
15
+ import type {
16
+ ComputedDatumSeries,
17
+ ChartParams,
18
+ ContainerPosition,
19
+ EventName,
20
+ EventSeries } from '../../../lib/core-types'
21
+ import type { PieEventTextsParams } from '../../../lib/plugins-basic-types'
22
+ import { DEFAULT_PIE_EVENT_TEXTS_PARAMS } from '../defaults'
23
+ import { getD3TransitionEase } from '../../utils/d3Utils'
24
+ import { getClassName } from '../../utils/orbchartsUtils'
25
+ import { seriesCenterSelectionObservable } from '../seriesObservables'
26
+ import { LAYER_INDEX_OF_LABEL } from '../../const'
27
+
28
+ type TextDatum = {
29
+ text: string
30
+ attr: { [key:string]: any }
31
+ style: { [key:string]: any }
32
+ }
33
+
34
+ const pluginName = 'PieEventTexts'
35
+ const textClassName = getClassName(pluginName, 'text')
36
+
37
+ const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_PIE_EVENT_TEXTS_PARAMS> = {
38
+ name: pluginName,
39
+ defaultParams: DEFAULT_PIE_EVENT_TEXTS_PARAMS,
40
+ layerIndex: LAYER_INDEX_OF_LABEL,
41
+ validator: (params, { validateColumns }) => {
42
+ const result = validateColumns(params, {
43
+ renderFn: {
44
+ toBeTypes: ['Function'],
45
+ },
46
+ textAttrs: {
47
+ toBeTypes: ['object[]'],
48
+ },
49
+ textStyles: {
50
+ toBeTypes: ['object[]'],
51
+ }
52
+ })
53
+ return result
54
+ }
55
+ }
56
+
57
+ function renderText (
58
+ selection: d3.Selection<SVGGElement, unknown, any, any>,
59
+ data: Array<TextDatum>
60
+ ): d3.Selection<SVGTextElement, TextDatum, SVGGElement, unknown> {
61
+ const textUpdate = selection
62
+ .selectAll<SVGTextElement, TextDatum>(`text.${textClassName}`)
63
+ .data(data)
64
+ const textEnter = textUpdate.enter()
65
+ .append('text')
66
+ .classed(textClassName, true)
67
+ const text = textUpdate.merge(textEnter)
68
+ text
69
+ .each((d, i, g) => {
70
+ const t = d3.select(g[i])
71
+ .text(d.text)
72
+ Object.keys(d.attr)
73
+ .forEach(key => {
74
+ t.attr(key, d.attr[key])
75
+ })
76
+ Object.keys(d.style)
77
+ .forEach(key => {
78
+ t.style(key, d.style[key])
79
+ })
80
+ })
81
+
82
+ textUpdate.exit().remove()
83
+
84
+ return text
85
+ }
86
+
87
+ function createTextData ({ eventData, renderFn, textAttrs, textStyles }: {
88
+ eventData: EventSeries,
89
+ // t: number,
90
+ renderFn: (d: EventSeries) => string[] | string
91
+ textAttrs: Array<{ [key:string]: string | number }>
92
+ textStyles: Array<{ [key:string]: string | number }>
93
+ }): TextDatum[] {
94
+ const callbackText = renderFn(eventData)
95
+ const textArr = Array.isArray(callbackText) ? callbackText : [callbackText]
96
+ return textArr.map((d, i) => {
97
+ return {
98
+ text: d,
99
+ attr: textAttrs[i],
100
+ style: textStyles[i]
101
+ }
102
+ })
103
+ }
104
+
105
+ function createEachPieEventTexts (pluginName: string, context: {
106
+ containerSelection: d3.Selection<SVGGElement, any, any, unknown>
107
+ computedData$: Observable<ComputedDatumSeries[][]>
108
+ containerComputedLayoutData$: Observable<ComputedDatumSeries[]>
109
+ SeriesDataMap$: Observable<Map<string, ComputedDatumSeries[]>>
110
+ fullParams$: Observable<PieEventTextsParams>
111
+ fullChartParams$: Observable<ChartParams>
112
+ seriesHighlight$: Observable<ComputedDatumSeries[]>
113
+ seriesContainerPosition$: Observable<ContainerPosition>
114
+ event$: Subject<EventSeries>
115
+ }) {
116
+ const destroy$ = new Subject()
117
+
118
+ // const graphicSelection: d3.Selection<SVGGElement, any, any, any> = selection.append('g')
119
+ let centerTextSelection: d3.Selection<SVGTextElement, TextDatum, SVGGElement, unknown> | undefined
120
+ let storeEventSubscription: Subscription | undefined
121
+
122
+ // context.layout$
123
+ // .pipe(
124
+ // first()
125
+ // )
126
+ // .subscribe(size => {
127
+ // selection
128
+ // .attr('transform', `translate(${size.width / 2}, ${size.height / 2})`)
129
+ // context.layout$
130
+ // .pipe(
131
+ // takeUntil(destroy$)
132
+ // )
133
+ // .subscribe(size => {
134
+ // selection
135
+ // .transition()
136
+ // .attr('transform', `translate(${size.width / 2}, ${size.height / 2})`)
137
+ // })
138
+ // })
139
+
140
+ const highlightTarget$ = context.fullChartParams$.pipe(
141
+ takeUntil(destroy$),
142
+ map(d => d.highlightTarget),
143
+ distinctUntilChanged()
144
+ )
145
+
146
+ combineLatest({
147
+ computedData: context.computedData$,
148
+ fullParams: context.fullParams$,
149
+ fullChartParams: context.fullChartParams$,
150
+ highlightTarget: highlightTarget$,
151
+ }).pipe(
152
+ takeUntil(destroy$),
153
+ switchMap(async (d) => d),
154
+ ).subscribe(data => {
155
+
156
+ context.containerSelection
157
+ .transition('move')
158
+ .duration(data.fullChartParams.transitionDuration!)
159
+ // .ease(getD3TransitionEase(data.fullChartParams.transitionEase!))
160
+ .tween('move', (event, datum) => {
161
+ return (t) => {
162
+ const renderData = createTextData({
163
+ eventData: {
164
+ type: 'series',
165
+ pluginName,
166
+ eventName: 'transitionMove',
167
+ event,
168
+ tween: t,
169
+ highlightTarget: data.highlightTarget,
170
+ data: data.computedData,
171
+ series: [],
172
+ seriesIndex: -1,
173
+ seriesLabel: '',
174
+ datum: null
175
+ },
176
+ // eventName: 'transitionMove',
177
+ // t,
178
+ renderFn: data.fullParams.renderFn!,
179
+ textAttrs: data.fullParams.textAttrs!,
180
+ textStyles: data.fullParams.textStyles!
181
+ })
182
+ centerTextSelection = renderText(context.containerSelection, renderData)
183
+ }
184
+ })
185
+ .on('end', (event, datum) => {
186
+ const renderData = createTextData({
187
+ eventData: {
188
+ type: 'series',
189
+ pluginName,
190
+ eventName: 'transitionEnd',
191
+ event,
192
+ tween: 1,
193
+ highlightTarget: data.highlightTarget,
194
+ data: data.computedData,
195
+ series: [],
196
+ seriesIndex: -1,
197
+ seriesLabel: '',
198
+ datum: null
199
+ },
200
+ // eventName: 'transitionMove',
201
+ // t: 1,
202
+ renderFn: data.fullParams.renderFn!,
203
+ textAttrs: data.fullParams.textAttrs!,
204
+ textStyles: data.fullParams.textStyles!
205
+ })
206
+ centerTextSelection = renderText(context.containerSelection, renderData)
207
+
208
+ if (storeEventSubscription) {
209
+ storeEventSubscription.unsubscribe()
210
+ }
211
+ storeEventSubscription = context.event$
212
+ .subscribe(eventData => {
213
+ const renderData = createTextData({
214
+ eventData,
215
+ // t: 1,
216
+ renderFn: data.fullParams.renderFn!,
217
+ textAttrs: data.fullParams.textAttrs!,
218
+ textStyles: data.fullParams.textStyles!
219
+ })
220
+ centerTextSelection = renderText(context.containerSelection, renderData)
221
+ })
222
+ })
223
+ })
224
+
225
+ return () => {
226
+ destroy$.next(undefined)
227
+ }
228
+ }
229
+
230
+ export const PieEventTexts = defineSeriesPlugin(pluginConfig)(({ selection, name, observer, subject }) => {
231
+ const destroy$ = new Subject()
232
+
233
+ const { seriesCenterSelection$ } = seriesCenterSelectionObservable({
234
+ selection: selection,
235
+ pluginName,
236
+ separateSeries$: observer.separateSeries$,
237
+ seriesLabels$: observer.seriesLabels$,
238
+ seriesContainerPosition$: observer.seriesContainerPosition$
239
+ })
240
+
241
+ const unsubscribeFnArr: (() => void)[] = []
242
+
243
+ seriesCenterSelection$
244
+ .pipe(
245
+ takeUntil(destroy$)
246
+ )
247
+ .subscribe(seriesCenterSelection => {
248
+ // 每次重新計算時,清除之前的訂閱
249
+ unsubscribeFnArr.forEach(fn => fn())
250
+
251
+ seriesCenterSelection.each((d, containerIndex, g) => {
252
+
253
+ const containerSelection = d3.select(g[containerIndex])
254
+
255
+ const containerComputedLayoutData$ = observer.computedLayoutData$.pipe(
256
+ takeUntil(destroy$),
257
+ map(data => data[containerIndex] ?? data[0])
258
+ )
259
+
260
+ const containerPosition$ = observer.seriesContainerPosition$.pipe(
261
+ takeUntil(destroy$),
262
+ map(data => data[containerIndex] ?? data[0])
263
+ )
264
+
265
+ unsubscribeFnArr[containerIndex] = createEachPieEventTexts(pluginName, {
266
+ containerSelection: containerSelection,
267
+ computedData$: observer.computedData$,
268
+ containerComputedLayoutData$: containerComputedLayoutData$,
269
+ SeriesDataMap$: observer.SeriesDataMap$,
270
+ fullParams$: observer.fullParams$,
271
+ fullChartParams$: observer.fullChartParams$,
272
+ seriesHighlight$: observer.seriesHighlight$,
273
+ seriesContainerPosition$: containerPosition$,
274
+ event$: subject.event$,
275
+ })
276
+
277
+ })
278
+ })
279
+
280
+ return () => {
281
+ destroy$.next(undefined)
282
+ unsubscribeFnArr.forEach(fn => fn())
283
+ }
284
+ })