@orbcharts/plugin-basic 4.0.0-pre-alpha.0

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 (208) hide show
  1. package/LICENSE +201 -0
  2. package/dist/orbcharts-plugin-basic.es.js +25335 -0
  3. package/dist/orbcharts-plugin-basic.umd.js +341 -0
  4. package/dist/plugin-basic/src/baseLayers/BaseBars.d.ts +38 -0
  5. package/dist/plugin-basic/src/baseLayers/BaseBarsTriangle.d.ts +37 -0
  6. package/dist/plugin-basic/src/baseLayers/BaseCategoryAxis.d.ts +42 -0
  7. package/dist/plugin-basic/src/baseLayers/BaseDots.d.ts +38 -0
  8. package/dist/plugin-basic/src/baseLayers/BaseLegend.d.ts +31 -0
  9. package/dist/plugin-basic/src/baseLayers/BaseLineAreas.d.ts +36 -0
  10. package/dist/plugin-basic/src/baseLayers/BaseLines.d.ts +36 -0
  11. package/dist/plugin-basic/src/baseLayers/BaseStackedBars.d.ts +41 -0
  12. package/dist/plugin-basic/src/baseLayers/BaseTooltip.d.ts +47 -0
  13. package/dist/plugin-basic/src/baseLayers/BaseValueAxis.d.ts +38 -0
  14. package/dist/plugin-basic/src/baseLayers/BaseXAxis.d.ts +25 -0
  15. package/dist/plugin-basic/src/baseLayers/BaseXZoom.d.ts +22 -0
  16. package/dist/plugin-basic/src/baseLayers/BaseYAxis.d.ts +23 -0
  17. package/dist/plugin-basic/src/baseLayers/types.d.ts +171 -0
  18. package/dist/plugin-basic/src/const/layerIndex.d.ts +10 -0
  19. package/dist/plugin-basic/src/const/sharedPluginParams.d.ts +6 -0
  20. package/dist/plugin-basic/src/index.d.ts +2 -0
  21. package/dist/plugin-basic/src/plugins/CompositionPlot/CompositionPlot.d.ts +22 -0
  22. package/dist/plugin-basic/src/plugins/CompositionPlot/contextObservables.d.ts +40 -0
  23. package/dist/plugin-basic/src/plugins/CompositionPlot/defaults.d.ts +10 -0
  24. package/dist/plugin-basic/src/plugins/CompositionPlot/index.d.ts +3 -0
  25. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/Bubbles.d.ts +16 -0
  26. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/Indicator.d.ts +0 -0
  27. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/Pie.d.ts +16 -0
  28. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/PieEventTexts.d.ts +16 -0
  29. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/PieLabels.d.ts +16 -0
  30. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/Rose.d.ts +16 -0
  31. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/RoseLabels.d.ts +16 -0
  32. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/Waffle.d.ts +0 -0
  33. package/dist/plugin-basic/src/plugins/CompositionPlot/types.d.ts +110 -0
  34. package/dist/plugin-basic/src/plugins/CompositionPlot/utils.d.ts +19 -0
  35. package/dist/plugin-basic/src/plugins/HierarchyPlot/HierarchyPlot.d.ts +22 -0
  36. package/dist/plugin-basic/src/plugins/HierarchyPlot/contextObservables.d.ts +16 -0
  37. package/dist/plugin-basic/src/plugins/HierarchyPlot/defaults.d.ts +4 -0
  38. package/dist/plugin-basic/src/plugins/HierarchyPlot/index.d.ts +3 -0
  39. package/dist/plugin-basic/src/plugins/HierarchyPlot/layers/TreeMap.d.ts +16 -0
  40. package/dist/plugin-basic/src/plugins/HierarchyPlot/types.d.ts +29 -0
  41. package/dist/plugin-basic/src/plugins/Legend/Legend.d.ts +22 -0
  42. package/dist/plugin-basic/src/plugins/Legend/contextObservables.d.ts +9 -0
  43. package/dist/plugin-basic/src/plugins/Legend/defaults.d.ts +4 -0
  44. package/dist/plugin-basic/src/plugins/Legend/index.d.ts +3 -0
  45. package/dist/plugin-basic/src/plugins/Legend/layers/Legend.d.ts +16 -0
  46. package/dist/plugin-basic/src/plugins/Legend/types.d.ts +31 -0
  47. package/dist/plugin-basic/src/plugins/Legend/utils.d.ts +19 -0
  48. package/dist/plugin-basic/src/plugins/NetworkPlot/NetworkPlot.d.ts +22 -0
  49. package/dist/plugin-basic/src/plugins/NetworkPlot/contextObservables.d.ts +19 -0
  50. package/dist/plugin-basic/src/plugins/NetworkPlot/defaults.d.ts +5 -0
  51. package/dist/plugin-basic/src/plugins/NetworkPlot/index.d.ts +3 -0
  52. package/dist/plugin-basic/src/plugins/NetworkPlot/layers/ForceDirected.d.ts +16 -0
  53. package/dist/plugin-basic/src/plugins/NetworkPlot/layers/ForceDirectedBubbles.d.ts +16 -0
  54. package/dist/plugin-basic/src/plugins/NetworkPlot/types.d.ts +117 -0
  55. package/dist/plugin-basic/src/plugins/ScatterPlot/ScatterPlot.d.ts +22 -0
  56. package/dist/plugin-basic/src/plugins/ScatterPlot/contextObservables.d.ts +140 -0
  57. package/dist/plugin-basic/src/plugins/ScatterPlot/defaults.d.ts +8 -0
  58. package/dist/plugin-basic/src/plugins/ScatterPlot/index.d.ts +3 -0
  59. package/dist/plugin-basic/src/plugins/ScatterPlot/layers/Scatter.d.ts +16 -0
  60. package/dist/plugin-basic/src/plugins/ScatterPlot/layers/ScatterBubbles.d.ts +16 -0
  61. package/dist/plugin-basic/src/plugins/ScatterPlot/layers/XYAux.d.ts +16 -0
  62. package/dist/plugin-basic/src/plugins/ScatterPlot/layers/XYAxes.d.ts +16 -0
  63. package/dist/plugin-basic/src/plugins/ScatterPlot/layers/XZoom.d.ts +16 -0
  64. package/dist/plugin-basic/src/plugins/ScatterPlot/types.d.ts +146 -0
  65. package/dist/plugin-basic/src/plugins/SeriesPlot/SeriesPlot.d.ts +22 -0
  66. package/dist/plugin-basic/src/plugins/SeriesPlot/contextObservables.d.ts +77 -0
  67. package/dist/plugin-basic/src/plugins/SeriesPlot/defaults.d.ts +15 -0
  68. package/dist/plugin-basic/src/plugins/SeriesPlot/index.d.ts +3 -0
  69. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/Bars.d.ts +16 -0
  70. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/BarsPN.d.ts +16 -0
  71. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/BarsTriangle.d.ts +16 -0
  72. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/CategoryAux.d.ts +16 -0
  73. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/CategoryAxis.d.ts +16 -0
  74. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/CategoryZoom.d.ts +16 -0
  75. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/Dots.d.ts +16 -0
  76. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/LineAreas.d.ts +16 -0
  77. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/Lines.d.ts +16 -0
  78. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/StackedBars.d.ts +16 -0
  79. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/StackedValueAxis.d.ts +16 -0
  80. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/ValueAxis.d.ts +16 -0
  81. package/dist/plugin-basic/src/plugins/SeriesPlot/types.d.ts +140 -0
  82. package/dist/plugin-basic/src/plugins/Tooltip/Tooltip.d.ts +22 -0
  83. package/dist/plugin-basic/src/plugins/Tooltip/contextObservables.d.ts +9 -0
  84. package/dist/plugin-basic/src/plugins/Tooltip/defaults.d.ts +4 -0
  85. package/dist/plugin-basic/src/plugins/Tooltip/index.d.ts +3 -0
  86. package/dist/plugin-basic/src/plugins/Tooltip/layers/Tooltip.d.ts +16 -0
  87. package/dist/plugin-basic/src/plugins/Tooltip/types.d.ts +35 -0
  88. package/dist/plugin-basic/src/plugins/Tooltip/utils.d.ts +19 -0
  89. package/dist/plugin-basic/src/plugins/index.d.ts +7 -0
  90. package/dist/plugin-basic/src/types/BaseLayer.d.ts +3 -0
  91. package/dist/plugin-basic/src/types/Common.d.ts +14 -0
  92. package/dist/plugin-basic/src/types/ComputedData.d.ts +27 -0
  93. package/dist/plugin-basic/src/types/PluginParams.d.ts +66 -0
  94. package/dist/plugin-basic/src/types/index.d.ts +3 -0
  95. package/dist/plugin-basic/src/utils/commonUtils.d.ts +3 -0
  96. package/dist/plugin-basic/src/utils/d3Graphics.d.ts +24 -0
  97. package/dist/plugin-basic/src/utils/d3Scale.d.ts +28 -0
  98. package/dist/plugin-basic/src/utils/d3Utils.d.ts +14 -0
  99. package/dist/plugin-basic/src/utils/graphObservables.d.ts +0 -0
  100. package/dist/plugin-basic/src/utils/gridObservables.d.ts +51 -0
  101. package/dist/plugin-basic/src/utils/multivariateObservables.d.ts +74 -0
  102. package/dist/plugin-basic/src/utils/observables.d.ts +34 -0
  103. package/dist/plugin-basic/src/utils/orbchartsUtils.d.ts +26 -0
  104. package/dist/plugin-basic/src/utils/seriesObservables.d.ts +22 -0
  105. package/dist/plugin-basic/vite.config.d.ts +2 -0
  106. package/dist/src/index.d.ts +1 -0
  107. package/package.json +62 -0
  108. package/src/baseLayers/BaseBars.ts +783 -0
  109. package/src/baseLayers/BaseBarsTriangle.ts +692 -0
  110. package/src/baseLayers/BaseCategoryAxis.ts +708 -0
  111. package/src/baseLayers/BaseDots.ts +495 -0
  112. package/src/baseLayers/BaseLegend.ts +684 -0
  113. package/src/baseLayers/BaseLineAreas.ts +644 -0
  114. package/src/baseLayers/BaseLines.ts +721 -0
  115. package/src/baseLayers/BaseStackedBars.ts +818 -0
  116. package/src/baseLayers/BaseTooltip.ts +435 -0
  117. package/src/baseLayers/BaseValueAxis.ts +612 -0
  118. package/src/baseLayers/BaseXAxis.ts +412 -0
  119. package/src/baseLayers/BaseXZoom.ts +250 -0
  120. package/src/baseLayers/BaseYAxis.ts +371 -0
  121. package/src/baseLayers/types.ts +174 -0
  122. package/src/const/layerIndex.ts +36 -0
  123. package/src/const/sharedPluginParams.ts +29 -0
  124. package/src/index.ts +3 -0
  125. package/src/plugins/CompositionPlot/CompositionPlot.ts +308 -0
  126. package/src/plugins/CompositionPlot/contextObservables.ts +251 -0
  127. package/src/plugins/CompositionPlot/defaults.ts +162 -0
  128. package/src/plugins/CompositionPlot/index.ts +3 -0
  129. package/src/plugins/CompositionPlot/layers/Bubbles.ts +808 -0
  130. package/src/plugins/CompositionPlot/layers/Indicator.ts +0 -0
  131. package/src/plugins/CompositionPlot/layers/Pie.ts +776 -0
  132. package/src/plugins/CompositionPlot/layers/PieEventTexts.ts +326 -0
  133. package/src/plugins/CompositionPlot/layers/PieLabels.ts +651 -0
  134. package/src/plugins/CompositionPlot/layers/Rose.ts +546 -0
  135. package/src/plugins/CompositionPlot/layers/RoseLabels.ts +616 -0
  136. package/src/plugins/CompositionPlot/layers/Waffle.ts +0 -0
  137. package/src/plugins/CompositionPlot/types.ts +129 -0
  138. package/src/plugins/CompositionPlot/utils.ts +53 -0
  139. package/src/plugins/HierarchyPlot/HierarchyPlot.ts +190 -0
  140. package/src/plugins/HierarchyPlot/contextObservables.ts +136 -0
  141. package/src/plugins/HierarchyPlot/defaults.ts +31 -0
  142. package/src/plugins/HierarchyPlot/index.ts +3 -0
  143. package/src/plugins/HierarchyPlot/layers/TreeMap.ts +371 -0
  144. package/src/plugins/HierarchyPlot/types.ts +36 -0
  145. package/src/plugins/Legend/Legend.ts +151 -0
  146. package/src/plugins/Legend/contextObservables.ts +55 -0
  147. package/src/plugins/Legend/defaults.ts +37 -0
  148. package/src/plugins/Legend/index.ts +3 -0
  149. package/src/plugins/Legend/layers/Legend.ts +114 -0
  150. package/src/plugins/Legend/types.ts +45 -0
  151. package/src/plugins/Legend/utils.ts +53 -0
  152. package/src/plugins/NetworkPlot/NetworkPlot.ts +228 -0
  153. package/src/plugins/NetworkPlot/contextObservables.ts +123 -0
  154. package/src/plugins/NetworkPlot/defaults.ts +147 -0
  155. package/src/plugins/NetworkPlot/index.ts +3 -0
  156. package/src/plugins/NetworkPlot/layers/ForceDirected.ts +1048 -0
  157. package/src/plugins/NetworkPlot/layers/ForceDirectedBubbles.ts +1318 -0
  158. package/src/plugins/NetworkPlot/types.ts +146 -0
  159. package/src/plugins/ScatterPlot/ScatterPlot.ts +569 -0
  160. package/src/plugins/ScatterPlot/contextObservables.ts +901 -0
  161. package/src/plugins/ScatterPlot/defaults.ts +212 -0
  162. package/src/plugins/ScatterPlot/index.ts +3 -0
  163. package/src/plugins/ScatterPlot/layers/Scatter.ts +518 -0
  164. package/src/plugins/ScatterPlot/layers/ScatterBubbles.ts +670 -0
  165. package/src/plugins/ScatterPlot/layers/XYAux.ts +686 -0
  166. package/src/plugins/ScatterPlot/layers/XYAxes.ts +205 -0
  167. package/src/plugins/ScatterPlot/layers/XZoom.ts +48 -0
  168. package/src/plugins/ScatterPlot/types.ts +179 -0
  169. package/src/plugins/SeriesPlot/SeriesPlot.ts +494 -0
  170. package/src/plugins/SeriesPlot/contextObservables.ts +726 -0
  171. package/src/plugins/SeriesPlot/defaults.ts +142 -0
  172. package/src/plugins/SeriesPlot/index.ts +3 -0
  173. package/src/plugins/SeriesPlot/layers/Bars.ts +84 -0
  174. package/src/plugins/SeriesPlot/layers/BarsPN.ts +85 -0
  175. package/src/plugins/SeriesPlot/layers/BarsTriangle.ts +89 -0
  176. package/src/plugins/SeriesPlot/layers/CategoryAux.ts +1131 -0
  177. package/src/plugins/SeriesPlot/layers/CategoryAxis.ts +92 -0
  178. package/src/plugins/SeriesPlot/layers/CategoryZoom.ts +233 -0
  179. package/src/plugins/SeriesPlot/layers/Dots.ts +91 -0
  180. package/src/plugins/SeriesPlot/layers/LineAreas.ts +82 -0
  181. package/src/plugins/SeriesPlot/layers/Lines.ts +75 -0
  182. package/src/plugins/SeriesPlot/layers/StackedBars.ts +85 -0
  183. package/src/plugins/SeriesPlot/layers/StackedValueAxis.ts +111 -0
  184. package/src/plugins/SeriesPlot/layers/ValueAxis.ts +111 -0
  185. package/src/plugins/SeriesPlot/types.ts +201 -0
  186. package/src/plugins/Tooltip/Tooltip.ts +159 -0
  187. package/src/plugins/Tooltip/contextObservables.ts +55 -0
  188. package/src/plugins/Tooltip/defaults.ts +458 -0
  189. package/src/plugins/Tooltip/index.ts +3 -0
  190. package/src/plugins/Tooltip/layers/Tooltip.ts +90 -0
  191. package/src/plugins/Tooltip/types.ts +55 -0
  192. package/src/plugins/Tooltip/utils.ts +53 -0
  193. package/src/plugins/index.ts +8 -0
  194. package/src/types/BaseLayer.ts +3 -0
  195. package/src/types/Common.ts +20 -0
  196. package/src/types/ComputedData.ts +55 -0
  197. package/src/types/PluginParams.ts +81 -0
  198. package/src/types/index.ts +3 -0
  199. package/src/utils/commonUtils.ts +31 -0
  200. package/src/utils/d3Graphics.ts +177 -0
  201. package/src/utils/d3Scale.ts +198 -0
  202. package/src/utils/d3Utils.ts +92 -0
  203. package/src/utils/graphObservables.ts +0 -0
  204. package/src/utils/gridObservables.ts +637 -0
  205. package/src/utils/multivariateObservables.ts +790 -0
  206. package/src/utils/observables.ts +357 -0
  207. package/src/utils/orbchartsUtils.ts +335 -0
  208. package/src/utils/seriesObservables.ts +172 -0
@@ -0,0 +1,651 @@
1
+ import * as d3 from 'd3'
2
+ import {
3
+ combineLatest,
4
+ map,
5
+ switchMap,
6
+ takeUntil,
7
+ distinctUntilChanged,
8
+ shareReplay,
9
+ debounceTime,
10
+ Observable,
11
+ Subject } from 'rxjs'
12
+ import type { Theme, EventData } from '@orbcharts/core'
13
+ import type { CompositionPlotExtendContext, CompositionPlotPluginParams, PieLabelsParams } from "../types"
14
+ import type { PieDatum } from '../utils'
15
+ import { defineSVGLayer } from '@orbcharts/core'
16
+ import { validateObject } from '@orbcharts/core'
17
+ import { DEFAULT_PIE_LABELS_PARAMS } from "../defaults"
18
+ import { seriesCenterSelectionObservable } from "../../../utils/seriesObservables"
19
+ import { getDatumColor } from '../../../utils/orbchartsUtils'
20
+ import { createClassName } from '../../../utils/orbchartsUtils'
21
+ import { makeD3Arc } from '../../../utils/d3Utils'
22
+ import { makePieData } from '../utils'
23
+ import type { ComputedDatumSeries } from '../../../types/ComputedData'
24
+ import type { ContainerPosition } from '../../../types/PluginParams'
25
+ import { LAYER_INDEX_OF_LABEL } from '../../../const/layerIndex'
26
+ import { renderTspansOnQuadrant } from '../../../utils/d3Graphics'
27
+
28
+ interface RenderDatum {
29
+ pieDatum: PieDatum
30
+ arcIndex: number
31
+ arcLabels: string[]
32
+ lineStartX: number
33
+ lineStartY: number
34
+ lineStartMouseoverX: number
35
+ lineStartMouseoverY: number
36
+ x: number
37
+ y: number
38
+ mouseoverX: number
39
+ mouseoverY: number
40
+ textWidth: number, // 文字寬度
41
+ collisionShiftX: number // 避免碰撞的位移
42
+ collisionShiftY: number
43
+ quadrant: number // 第幾象限
44
+ }
45
+
46
+ const pluginName = 'CompositionPlot'
47
+ const layerName = 'PieLabels'
48
+ const labelGClassName = createClassName(pluginName, layerName, 'label-g')
49
+ const lineGClassName = createClassName(pluginName, layerName, 'line-g')
50
+ const textClassName = createClassName(pluginName, layerName, 'text')
51
+ const pieOuterCentroid = 2
52
+
53
+ function makeRenderData ({ pieData, arc, arcMouseover, labelCentroid, lineStartCentroid, layerParams }: {
54
+ pieData: PieDatum[]
55
+ arc: d3.Arc<any, d3.DefaultArcObject>
56
+ arcMouseover: d3.Arc<any, d3.DefaultArcObject>
57
+ labelCentroid: number
58
+ lineStartCentroid: number
59
+ layerParams: PieLabelsParams
60
+ }): RenderDatum[] {
61
+ return pieData
62
+ .map((d, i) => {
63
+ const [_x, _y] = arc!.centroid(d as any)
64
+ const [_mouseoverX, _mouseoverY] = arcMouseover!.centroid(d as any)
65
+ const arcLabel = layerParams.labelFn(d.data)
66
+ return {
67
+ pieDatum: d,
68
+ arcIndex: i,
69
+ arcLabels: arcLabel.split('\n'),
70
+ lineStartX: _x * lineStartCentroid,
71
+ lineStartY: _y * lineStartCentroid,
72
+ lineStartMouseoverX: _mouseoverX * lineStartCentroid,
73
+ lineStartMouseoverY: _mouseoverY * lineStartCentroid,
74
+ x: _x * labelCentroid!,
75
+ y: _y * labelCentroid!,
76
+ mouseoverX: _mouseoverX * labelCentroid!,
77
+ mouseoverY: _mouseoverY * labelCentroid!,
78
+ textWidth: 0, // 後面再做計算
79
+ collisionShiftX: 0, // 後面再做計算
80
+ collisionShiftY: 0, // 後面再做計算
81
+ quadrant: _x >= 0 && _y <= 0
82
+ ? 1
83
+ : _x < 0 && _y <= 0
84
+ ? 2
85
+ : _x < 0 && _y > 0
86
+ ? 3
87
+ : 4
88
+ }
89
+ })
90
+ .filter(d => d.pieDatum.data.visible)
91
+ }
92
+
93
+ // 繪製圓餅圖
94
+ function renderLabel ({ labelGSelection, data, layerParams, pluginParams, theme, fontSizePx }: {
95
+ labelGSelection: d3.Selection<SVGGElement, undefined, any, any>
96
+ data: RenderDatum[]
97
+ layerParams: PieLabelsParams
98
+ pluginParams: CompositionPlotPluginParams
99
+ theme: Theme
100
+ fontSizePx: number
101
+ }) {
102
+ // console.log(data)
103
+ // let update = this.gSelection.selectAll('g').data(pieData)
104
+ const textSelection = labelGSelection
105
+ .selectAll<SVGTextElement, RenderDatum>('text')
106
+ .data(data, d => d.pieDatum.id)
107
+ .join('text')
108
+ .classed(textClassName, true)
109
+ .attr('font-weight', 'bold')
110
+ .attr('text-anchor', d => d.quadrant == 1 || d.quadrant == 4 ? 'start' : 'end')
111
+ .style('dominant-baseline', d => d.quadrant == 1 || d.quadrant == 2 ? 'auto' : 'hanging')
112
+ // .style('pointer-events', 'none')
113
+ .style('cursor', d => pluginParams.styles.highlightTarget && pluginParams.styles.highlightTarget != 'none'
114
+ ? 'pointer'
115
+ : 'none')
116
+ // .text(d => layerParams.labelFn(d.pieDatum.data))
117
+ .attr('font-size', fontSizePx)
118
+ .attr('x', 0)
119
+ .attr('y', 0)
120
+ .attr('fill', (d, i) => getDatumColor({ datum: d.pieDatum.data, colorType: layerParams.labelColorType, theme }))
121
+ .each((d, i, n) => {
122
+ // const textNode = d3.select<SVGTextElement, RenderDatum>(n[i])
123
+ // .selectAll('tspan')
124
+ // .data(d.arcLabels)
125
+ // .join('tspan')
126
+ // .attr('x', 0)
127
+ // .attr('y', (_d, _i) => d.quadrant == 1 || d.quadrant == 2
128
+ // ? - (d.arcLabels.length - 1 - _i) * fontSizePx
129
+ // : _i * fontSizePx)
130
+ // .text(d => d)
131
+ renderTspansOnQuadrant(d3.select<SVGTextElement, RenderDatum>(n[i]), {
132
+ textArr: d.arcLabels,
133
+ fontSizePx,
134
+ quadrant: d.quadrant
135
+ })
136
+ })
137
+ textSelection
138
+ .transition()
139
+ .attr('transform', (d) => {
140
+ // console.log('transform', d)
141
+ return 'translate(' + d.x + ',' + d.y + ')'
142
+ })
143
+ // .on('end', () => initHighlight({ labelSelection, data, fullChartParams }))
144
+
145
+ // 如無新增資料則不用等動畫
146
+ // if (enter.size() == 0) {
147
+ // this.initHighlight()
148
+ // }
149
+
150
+ return textSelection
151
+ }
152
+
153
+ // // 獲取每個文字元素的邊界框並檢查是否重疊
154
+ // function resolveCollisions(labelSelection: d3.Selection<SVGPathElement, RenderDatum, any, any>, data: RenderDatum[]) {
155
+ // const textArray = labelSelection.nodes();
156
+ // const padding = 10; // 調整文字間的間距
157
+
158
+ // // 存儲每個標籤的當前位置
159
+ // const positions = textArray.map((textNode, i) => {
160
+ // const bbox = textNode.getBBox();
161
+ // // const arcCentroid = arc.centroid(data[i]);
162
+ // const arcCentroid = [data[i].x, data[i].y];
163
+ // return {
164
+ // node: textNode,
165
+ // x: arcCentroid[0],
166
+ // y: arcCentroid[1],
167
+ // width: bbox.width,
168
+ // height: bbox.height
169
+ // };
170
+ // });
171
+ // // console.log('positions', positions)
172
+
173
+ // for (let i = 0; i < positions.length; i++) {
174
+ // const a = positions[i];
175
+
176
+ // for (let j = i + 1; j < positions.length; j++) {
177
+ // const b = positions[j];
178
+
179
+ // // 檢查是否重疊
180
+ // if (!(a.x + a.width / 2 < b.x - b.width / 2 ||
181
+ // a.x - a.width / 2 > b.x + b.width / 2 ||
182
+ // a.y + a.height / 2 < b.y - b.height / 2 ||
183
+ // a.y - a.height / 2 > b.y + b.height / 2)) {
184
+
185
+ // // 如果有重疊,則位移其中一個文字,這裡我們進行上下位移
186
+ // const moveDown = (b.y > a.y) ? padding : -padding; // 決定方向
187
+ // b.y += moveDown; // 更新 b 的 y 座標
188
+
189
+ // // 更新 b 的 x 座標,根據與 a 的位置差異進行微調
190
+ // const moveRight = (b.x > a.x) ? padding : -padding;
191
+ // b.x += moveRight;
192
+
193
+ // // // 重新設置 b 的 transform 來移動
194
+ // d3.select(b.node)
195
+ // .transition()
196
+ // .attr("transform", `translate(${b.x},${b.y})`);
197
+
198
+ // data[j].collisionShiftX = moveRight
199
+ // data[j].collisionShiftY = moveDown
200
+ // }
201
+ // }
202
+ // }
203
+ // }
204
+
205
+ // 獲取每個文字元素的邊界框並檢查是否重疊
206
+ function resolveCollisions(textSelection: d3.Selection<SVGTextElement, RenderDatum, any, any>, data: RenderDatum[], fontSizePx: number) {
207
+ const textArray = textSelection.nodes();
208
+ const padding = fontSizePx // 調整文字間的間距
209
+
210
+ // 存儲每個標籤的當前位置
211
+ const positions = textArray.map((textNode, i) => {
212
+ const bbox = textNode.getBBox();
213
+ // const arcCentroid = arc.centroid(data[i]);
214
+ const arcCentroid = [data[i].x, data[i].y];
215
+ return {
216
+ node: textNode,
217
+ x: arcCentroid[0],
218
+ y: arcCentroid[1],
219
+ width: bbox.width,
220
+ height: bbox.height
221
+ }
222
+ })
223
+
224
+ // 順時針碰撞檢測(只處理 2、4 象限,將較後面的文字碰撞時往外偏移)
225
+ for (let i = 0; i < positions.length; i++) {
226
+ const a = positions[i]
227
+
228
+ for (let j = i + 1; j < positions.length; j++) {
229
+ const b = positions[j]
230
+
231
+ // 記錄文字寬度
232
+ data[i].textWidth = a.width
233
+
234
+ const ax = a.x + data[i].collisionShiftX
235
+ const ay = a.y + data[i].collisionShiftY
236
+ const bx = b.x + data[j].collisionShiftX
237
+ const by = b.y + data[j].collisionShiftY
238
+
239
+ // 檢查是否重疊
240
+ if (!(ax + a.width / 2 < bx - b.width / 2 ||
241
+ ax - a.width / 2 > bx + b.width / 2 ||
242
+ ay + a.height / 2 < by - b.height / 2 ||
243
+ ay - a.height / 2 > by + b.height / 2)) {
244
+
245
+ if (data[j].quadrant == 2) {
246
+ const moveDown = (by > ay)
247
+ ? -padding * 2
248
+ : -padding
249
+ data[j].collisionShiftY += moveDown // 由後一個累加高度
250
+ } else if (data[j].quadrant == 4) {
251
+ const moveDown = (by > ay)
252
+ ? padding
253
+ : padding * 2
254
+ data[j].collisionShiftY += moveDown // 由後一個累加高度
255
+ }
256
+ }
257
+ }
258
+ }
259
+
260
+ // 逆時針碰撞檢測(只處理 1、3 象限,將較前面的文字碰撞時往外偏移)
261
+ for (let i = positions.length - 1; i >= 0; i--) {
262
+ const a = positions[i]
263
+
264
+ for (let j = i - 1; j >= 0; j--) {
265
+ const b = positions[j]
266
+
267
+ // 記錄文字寬度
268
+ data[i].textWidth = a.width
269
+
270
+ const ax = a.x + data[i].collisionShiftX
271
+ const ay = a.y + data[i].collisionShiftY
272
+ const bx = b.x + data[j].collisionShiftX
273
+ const by = b.y + data[j].collisionShiftY
274
+
275
+ // 檢查是否重疊
276
+ if (!(ax + a.width / 2 < bx - b.width / 2 ||
277
+ ax - a.width / 2 > bx + b.width / 2 ||
278
+ ay + a.height / 2 < by - b.height / 2 ||
279
+ ay - a.height / 2 > by + b.height / 2)) {
280
+
281
+ if (data[j].quadrant == 1) {
282
+ const moveDown = (by > ay)
283
+ ? -padding * 2
284
+ : -padding
285
+ data[j].collisionShiftY += moveDown // 由前一個累加高度
286
+ } else if (data[j].quadrant == 3) {
287
+ const moveDown = (by > ay)
288
+ ? padding
289
+ : padding * 2
290
+ data[j].collisionShiftY += moveDown // 由前一個累加高度
291
+ }
292
+ }
293
+ }
294
+ }
295
+
296
+ // 全部算完再來 render
297
+ textSelection
298
+ .data(data)
299
+ .transition()
300
+ .attr('transform', (d) => {
301
+ return `translate(${d.x + d.collisionShiftX},${d.y + d.collisionShiftY})`
302
+ })
303
+ }
304
+
305
+ function renderLine ({ lineGSelection, data, layerParams, theme }: {
306
+ lineGSelection: d3.Selection<SVGGElement, undefined, any, any>
307
+ data: RenderDatum[]
308
+ layerParams: PieLabelsParams
309
+ theme: Theme
310
+ }) {
311
+
312
+ // 只顯示在有偏移的標籤
313
+ const filteredData = data.filter(d => d.collisionShiftX || d.collisionShiftY)
314
+
315
+ // 添加標籤的連接線
316
+ const lines = lineGSelection.selectAll<SVGPolylineElement, RenderDatum>("polyline")
317
+ .data(filteredData, d => d.pieDatum.id)
318
+ .join("polyline")
319
+ .attr("stroke", d => getDatumColor({ datum: d.pieDatum.data, colorType: layerParams.labelColorType, theme }))
320
+ .attr("stroke-width", 1)
321
+ .attr("fill", "none")
322
+ .attr("points", (d) => {
323
+ return [[d.lineStartX, d.lineStartY], [d.lineStartX, d.lineStartY]] as any // 畫出從弧線中心到延伸點的線
324
+ })
325
+ lines
326
+ .transition()
327
+ .attr("points", (d) => {
328
+ // const pos = arc.centroid(d) // 起點:弧線的中心點
329
+ // const outerPos = [pos[0] * 2.5, pos[1] * 2.5] // 外部延伸的點(乘以倍率來延長線段)
330
+
331
+ let lineEndX = d.x + d.collisionShiftX
332
+ let lineEndY = d.y + d.collisionShiftY
333
+ // if (d.lineStartX >= Math.abs(d.lineStartY)) {
334
+ // lineEndX -= d.textWidth / 2
335
+ // } else if (d.lineStartX <= - Math.abs(d.lineStartY)) {
336
+ // lineEndX += d.textWidth / 2
337
+ // }
338
+
339
+ return [[lineEndX, lineEndY], [d.lineStartX, d.lineStartY]] as any // 畫出從弧線中心到延伸點的線
340
+ })
341
+
342
+ return lines
343
+ }
344
+
345
+ function highlight ({ textSelection, lineSelection, ids, pluginParams }: {
346
+ textSelection: d3.Selection<SVGTextElement, RenderDatum, any, any>
347
+ lineSelection: d3.Selection<SVGPolylineElement, RenderDatum, any, any>
348
+ ids: string[]
349
+ pluginParams: CompositionPlotPluginParams
350
+ }) {
351
+ textSelection.interrupt('highlight')
352
+ lineSelection.interrupt('highlight')
353
+
354
+ if (!ids.length) {
355
+ textSelection
356
+ .transition('highlight')
357
+ .duration(200)
358
+ .attr('transform', (d) => {
359
+ return `translate(${d.x + d.collisionShiftX},${d.y + d.collisionShiftY})`
360
+ })
361
+ .style('opacity', 1)
362
+ lineSelection
363
+ .transition('highlight')
364
+ .duration(200)
365
+ .style('opacity', 1)
366
+ return
367
+ }
368
+
369
+ textSelection.each((d, i, n) => {
370
+ const segment = d3.select<SVGTextElement, RenderDatum>(n[i])
371
+
372
+ if (ids.includes(d.pieDatum.id)) {
373
+ segment
374
+ .style('opacity', 1)
375
+ .transition('highlight')
376
+ .duration(200)
377
+ .attr('transform', (d) => {
378
+ // 如果已經有偏移過,則使用偏移後的位置(如果再改變的話很容易造成文字重疊)
379
+ if (d.collisionShiftX || d.collisionShiftY) {
380
+ return `translate(${d.x + d.collisionShiftX},${d.y + d.collisionShiftY})`
381
+ }
382
+ return `translate(${d.mouseoverX + d.collisionShiftX},${d.mouseoverY + d.collisionShiftY})`
383
+ })
384
+ } else {
385
+ segment
386
+ .style('opacity', pluginParams.styles.unhighlightedOpacity)
387
+ .transition('highlight')
388
+ .duration(200)
389
+ .attr('transform', (d) => {
390
+ return `translate(${d.x + d.collisionShiftX},${d.y + d.collisionShiftY})`
391
+ })
392
+ }
393
+ })
394
+ lineSelection.each((d, i, n) => {
395
+ const segment = d3.select<SVGPolylineElement, RenderDatum>(n[i])
396
+
397
+ if (ids.includes(d.pieDatum.data.id)) {
398
+ segment
399
+ .style('opacity', 1)
400
+ .transition('highlight')
401
+ .duration(200)
402
+ } else {
403
+ segment
404
+ .style('opacity', pluginParams.styles.unhighlightedOpacity)
405
+ .transition('highlight')
406
+ .duration(200)
407
+ }
408
+ })
409
+ }
410
+
411
+
412
+ function createEachPieLabel (context: {
413
+ containerSelection: d3.Selection<SVGGElement, any, any, unknown>
414
+ // computedData$: Observable<ComputedDatumSeries[][]>
415
+ containerVisibleComputedSortedData$: Observable<ComputedDatumSeries[]>
416
+ // SeriesDataMap$: Observable<Map<string, ComputedDatumSeries[]>>
417
+ layerParams$: Observable<PieLabelsParams>
418
+ pluginParams$: Observable<CompositionPlotPluginParams>
419
+ theme$: Observable<Theme>
420
+ fontSizePx$: Observable<number>
421
+ seriesHighlight$: Observable<ComputedDatumSeries[]>
422
+ seriesContainerPosition$: Observable<ContainerPosition>
423
+ eventTrigger$: Subject<EventData>
424
+ }) {
425
+ const destroy$ = new Subject()
426
+
427
+ context.containerSelection.selectAll('g').remove()
428
+
429
+ const lineGSelection: d3.Selection<SVGGElement, any, any, unknown> = context.containerSelection.append('g')
430
+ lineGSelection.classed(lineGClassName, true)
431
+ const labelGSelection: d3.Selection<SVGGElement, any, any, unknown> = context.containerSelection.append('g')
432
+ labelGSelection.classed(labelGClassName, true)
433
+
434
+ // const graphicSelection: d3.Selection<SVGGElement, any, any, any> = selection.append('g')
435
+ const textSelection$: Subject<d3.Selection<SVGTextElement, RenderDatum, any, any>> = new Subject()
436
+ const lineSelection$: Subject<d3.Selection<SVGPolylineElement, RenderDatum, any, any>> = new Subject()
437
+ let renderData: RenderDatum[] = []
438
+
439
+ const shorterSideWith$ = context.seriesContainerPosition$.pipe(
440
+ takeUntil(destroy$),
441
+ map(d => d.width < d.height ? d.width : d.height),
442
+ distinctUntilChanged()
443
+ )
444
+
445
+ const lineStartCentroid$ = context.layerParams$.pipe(
446
+ takeUntil(destroy$),
447
+ map(d => {
448
+ return d.labelCentroid >= pieOuterCentroid
449
+ ? pieOuterCentroid // 當 label在 pie的外側時,線條從 pie的邊緣開始
450
+ : d.labelCentroid // 當 label在 pie的內側時,線條從 label未偏移前的位置開始
451
+
452
+ })
453
+ )
454
+
455
+ combineLatest({
456
+ shorterSideWith: shorterSideWith$,
457
+ containerVisibleComputedSortedData: context.containerVisibleComputedSortedData$,
458
+ layerParams: context.layerParams$,
459
+ pluginParams: context.pluginParams$,
460
+ theme: context.theme$,
461
+ fontSizePx: context.fontSizePx$,
462
+ lineStartCentroid: lineStartCentroid$
463
+ }).pipe(
464
+ takeUntil(destroy$),
465
+ switchMap(async (d) => d),
466
+ ).subscribe(data => {
467
+
468
+ // 弧產生器 (d3.arc())
469
+ const arc = makeD3Arc({
470
+ axisWidth: data.shorterSideWith,
471
+ innerRadius: 0,
472
+ outerRadius: data.layerParams.outerRadius,
473
+ padAngle: 0,
474
+ cornerRadius: 0
475
+ })
476
+
477
+ const arcMouseover = makeD3Arc({
478
+ axisWidth: data.shorterSideWith,
479
+ innerRadius: 0,
480
+ outerRadius: data.layerParams.outerRadiusWhileHighlight, // 外半徑變化
481
+ padAngle: 0,
482
+ cornerRadius: 0
483
+ })
484
+
485
+ const pieData = makePieData({
486
+ data: data.containerVisibleComputedSortedData,
487
+ startAngle: data.layerParams.startAngle,
488
+ endAngle: data.layerParams.endAngle
489
+ })
490
+
491
+ renderData = makeRenderData({
492
+ pieData,
493
+ arc,
494
+ arcMouseover,
495
+ labelCentroid: data.layerParams.labelCentroid,
496
+ lineStartCentroid: data.lineStartCentroid,
497
+ layerParams: data.layerParams
498
+ })
499
+
500
+ // 先移除線條,等偏移後再重新繪製
501
+ lineGSelection.selectAll('polyline').remove()
502
+
503
+ const textSelection = renderLabel({
504
+ labelGSelection,
505
+ data: renderData,
506
+ layerParams: data.layerParams,
507
+ pluginParams: data.pluginParams,
508
+ theme: data.theme,
509
+ fontSizePx: data.fontSizePx
510
+ })
511
+
512
+ // 等 label 本身的 transition 結束後再進行碰撞檢測
513
+ setTimeout(() => {
514
+ // resolveCollisions(labelSelection, renderData)
515
+ // 偏移 label
516
+ resolveCollisions(textSelection, renderData, data.fontSizePx)
517
+
518
+ const lineSelection = renderLine({ lineGSelection, data: renderData, layerParams: data.layerParams, theme: data.theme })
519
+
520
+ lineSelection$.next(lineSelection)
521
+
522
+ }, 1000)
523
+
524
+ textSelection$.next(textSelection)
525
+
526
+ })
527
+
528
+ combineLatest({
529
+ textSelection: textSelection$,
530
+ lineSelection: lineSelection$,
531
+ highlight: context.seriesHighlight$.pipe(
532
+ map(data => data.map(d => d.id))
533
+ ),
534
+ pluginParams: context.pluginParams$,
535
+ }).pipe(
536
+ takeUntil(destroy$),
537
+ debounceTime(0)
538
+ ).subscribe(data => {
539
+ highlight({
540
+ textSelection: data.textSelection,
541
+ lineSelection: data.lineSelection,
542
+ ids: data.highlight,
543
+ pluginParams: data.pluginParams,
544
+ })
545
+ })
546
+
547
+ return () => {
548
+ destroy$.next(undefined)
549
+ }
550
+ }
551
+
552
+ export const PieLabels = defineSVGLayer<CompositionPlotExtendContext, CompositionPlotPluginParams, PieLabelsParams>({
553
+ name: layerName,
554
+ defaultParams: DEFAULT_PIE_LABELS_PARAMS,
555
+ layerIndex: LAYER_INDEX_OF_LABEL,
556
+ initShow: false,
557
+ validator: (params) => {
558
+ const result = validateObject(params, {
559
+ outerRadius: {
560
+ toBeTypes: ['number'],
561
+ },
562
+ outerRadiusWhileHighlight: {
563
+ toBeTypes: ['number'],
564
+ },
565
+ startAngle: {
566
+ toBeTypes: ['number'],
567
+ },
568
+ endAngle: {
569
+ toBeTypes: ['number'],
570
+ },
571
+ labelCentroid: {
572
+ toBeTypes: ['number'],
573
+ },
574
+ labelFn: {
575
+ toBeTypes: ['Function'],
576
+ },
577
+ labelColorType: {
578
+ toBeOption: 'ColorType'
579
+ }
580
+ })
581
+ return result
582
+ },
583
+ setup: ({ svgG, pluginParams$, layerParams$, context }) => {
584
+
585
+ const destroy$ = new Subject()
586
+
587
+ context.layout$
588
+ .pipe(
589
+ takeUntil(destroy$)
590
+ )
591
+ .subscribe(layout => {
592
+ d3.select(svgG)
593
+ .attr('transform', `translate(${layout.left}, ${layout.top})`)
594
+ })
595
+
596
+ const { seriesCenterSelection$ } = seriesCenterSelectionObservable({
597
+ selection: d3.select(svgG),
598
+ pluginName,
599
+ layerName,
600
+ visibleComputedSortedData$: context.visibleComputedSortedData$,
601
+ seriesContainerPosition$: context.seriesContainerPosition$
602
+ })
603
+
604
+ const unsubscribeFnArr: (() => void)[] = []
605
+
606
+ seriesCenterSelection$
607
+ .pipe(
608
+ takeUntil(destroy$)
609
+ )
610
+ .subscribe(seriesCenterSelection => {
611
+ // 每次重新計算時,清除之前的訂閱
612
+ unsubscribeFnArr.forEach(fn => fn())
613
+
614
+ seriesCenterSelection.each((d, containerIndex, g) => {
615
+
616
+ const containerSelection = d3.select(g[containerIndex])
617
+
618
+ const containerVisibleComputedSortedData$ = context.visibleComputedSortedData$.pipe(
619
+ takeUntil(destroy$),
620
+ map(data => data[containerIndex] ?? data[0])
621
+ )
622
+
623
+ const containerPosition$ = context.seriesContainerPosition$.pipe(
624
+ takeUntil(destroy$),
625
+ map(data => data[containerIndex] ?? data[0])
626
+ )
627
+
628
+ unsubscribeFnArr[containerIndex] = createEachPieLabel({
629
+ containerSelection: containerSelection,
630
+ // computedData$: observer.computedData$,
631
+ containerVisibleComputedSortedData$: containerVisibleComputedSortedData$,
632
+ // SeriesDataMap$: observer.SeriesDataMap$,
633
+ layerParams$: layerParams$,
634
+ pluginParams$: pluginParams$,
635
+ theme$: context.theme$,
636
+ fontSizePx$: context.fontSizePx$,
637
+ seriesHighlight$: context.seriesHighlight$,
638
+ seriesContainerPosition$: containerPosition$,
639
+ eventTrigger$: context.eventTrigger$,
640
+ })
641
+
642
+ })
643
+ })
644
+
645
+ return () => {
646
+ // subscription.unsubscribe()
647
+ destroy$.next(undefined)
648
+ unsubscribeFnArr.forEach(fn => fn())
649
+ }
650
+ }
651
+ })