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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (228) hide show
  1. package/LICENSE +200 -200
  2. package/dist/orbcharts-plugins-basic.es.js +11839 -10485
  3. package/dist/orbcharts-plugins-basic.umd.js +231 -171
  4. package/dist/{orbcharts-plugins-basic/src → src}/base/BaseBars.d.ts +3 -3
  5. package/dist/{orbcharts-plugins-basic/src → src}/base/BaseBarsTriangle.d.ts +3 -3
  6. package/dist/{orbcharts-plugins-basic/src → src}/base/BaseDots.d.ts +3 -3
  7. package/dist/{orbcharts-plugins-basic/src → src}/base/BaseLineAreas.d.ts +3 -3
  8. package/dist/{orbcharts-plugins-basic/src → src}/base/BaseLines.d.ts +3 -3
  9. package/dist/src/base/BaseRacingBars.d.ts +23 -0
  10. package/dist/src/base/BaseRacingLabels.d.ts +20 -0
  11. package/dist/src/base/BaseRacingValueLabels.d.ts +20 -0
  12. package/dist/{orbcharts-plugins-basic/src/base/BaseStackedBar.d.ts → src/base/BaseStackedBars.d.ts} +7 -7
  13. package/dist/{orbcharts-plugins-basic/src → src}/base/BaseValueAxis.d.ts +4 -7
  14. package/dist/src/base/BaseXAxis.d.ts +20 -0
  15. package/dist/src/base/BaseYAxis.d.ts +18 -0
  16. package/dist/{orbcharts-plugins-basic/src → src}/grid/defaults.d.ts +3 -3
  17. package/dist/{orbcharts-plugins-basic/src → src}/grid/gridObservables.d.ts +8 -1
  18. package/dist/{orbcharts-plugins-basic/src → src}/grid/index.d.ts +2 -2
  19. package/dist/src/grid/plugins/GroupZoom.d.ts +1 -0
  20. package/dist/src/grid/plugins/StackedBars.d.ts +1 -0
  21. package/dist/src/index.d.ts +7 -1
  22. package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/defaults.d.ts +2 -2
  23. package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/index.d.ts +1 -1
  24. package/dist/src/multiGrid/plugins/MultiStackedBars.d.ts +1 -0
  25. package/dist/src/multiValue/defaults.d.ts +13 -0
  26. package/dist/{orbcharts-plugins-basic/src → src}/multiValue/index.d.ts +5 -1
  27. package/dist/src/multiValue/multiValueObservables.d.ts +56 -0
  28. package/dist/src/multiValue/plugins/RacingBars.d.ts +1 -0
  29. package/dist/src/multiValue/plugins/RacingCounterTexts.d.ts +3 -0
  30. package/dist/src/multiValue/plugins/RacingValueAxis.d.ts +1 -0
  31. package/dist/src/multiValue/plugins/XAxis.d.ts +1 -0
  32. package/dist/src/multiValue/plugins/XYAxes.d.ts +1 -0
  33. package/dist/src/multiValue/plugins/XZoom.d.ts +1 -0
  34. package/dist/{orbcharts-plugins-basic/src → src}/utils/commonUtils.d.ts +1 -0
  35. package/dist/{orbcharts-plugins-basic/src → src}/utils/d3Graphics.d.ts +2 -1
  36. package/lib/core-types.ts +7 -7
  37. package/lib/core.ts +6 -6
  38. package/lib/plugins-basic-types.ts +6 -6
  39. package/package.json +44 -44
  40. package/src/base/BaseBars.ts +765 -765
  41. package/src/base/BaseBarsTriangle.ts +676 -676
  42. package/src/base/BaseDots.ts +464 -464
  43. package/src/base/BaseGroupAxis.ts +691 -679
  44. package/src/base/BaseLegend.ts +684 -684
  45. package/src/base/BaseLineAreas.ts +629 -629
  46. package/src/base/BaseLines.ts +706 -706
  47. package/src/base/BaseRacingBars.ts +552 -0
  48. package/src/base/BaseRacingLabels.ts +396 -0
  49. package/src/base/BaseRacingValueLabels.ts +403 -0
  50. package/src/base/{BaseStackedBar.ts → BaseStackedBars.ts} +782 -782
  51. package/src/base/BaseTooltip.ts +386 -385
  52. package/src/base/BaseValueAxis.ts +600 -583
  53. package/src/base/BaseXAxis.ts +427 -0
  54. package/src/base/BaseYAxis.ts +389 -0
  55. package/src/base/types.ts +2 -2
  56. package/src/const.ts +30 -30
  57. package/src/grid/defaults.ts +213 -250
  58. package/src/grid/gridObservables.ts +612 -554
  59. package/src/grid/index.ts +16 -16
  60. package/src/grid/plugins/Bars.ts +69 -69
  61. package/src/grid/plugins/BarsPN.ts +66 -66
  62. package/src/grid/plugins/BarsTriangle.ts +73 -73
  63. package/src/grid/plugins/Dots.ts +68 -68
  64. package/src/grid/plugins/GridLegend.ts +107 -107
  65. package/src/grid/plugins/GridTooltip.ts +66 -66
  66. package/src/grid/plugins/GroupAux.ts +1120 -1103
  67. package/src/grid/plugins/GroupAxis.ts +73 -97
  68. package/src/grid/plugins/{GridZoom.ts → GroupZoom.ts} +218 -218
  69. package/src/grid/plugins/LineAreas.ts +65 -65
  70. package/src/grid/plugins/Lines.ts +59 -59
  71. package/src/grid/plugins/{StackedBar.ts → StackedBars.ts} +64 -64
  72. package/src/grid/plugins/StackedValueAxis.ts +96 -96
  73. package/src/grid/plugins/ValueAxis.ts +94 -94
  74. package/src/index.ts +6 -6
  75. package/src/multiGrid/defaults.ts +244 -228
  76. package/src/multiGrid/index.ts +14 -14
  77. package/src/multiGrid/multiGridObservables.ts +50 -49
  78. package/src/multiGrid/plugins/MultiBars.ts +108 -108
  79. package/src/multiGrid/plugins/MultiBarsTriangle.ts +114 -114
  80. package/src/multiGrid/plugins/MultiDots.ts +102 -102
  81. package/src/multiGrid/plugins/MultiGridLegend.ts +169 -159
  82. package/src/multiGrid/plugins/MultiGridTooltip.ts +66 -66
  83. package/src/multiGrid/plugins/MultiGroupAxis.ts +137 -137
  84. package/src/multiGrid/plugins/MultiLineAreas.ts +107 -107
  85. package/src/multiGrid/plugins/MultiLines.ts +101 -101
  86. package/src/multiGrid/plugins/{MultiStackedBar.ts → MultiStackedBars.ts} +106 -106
  87. package/src/multiGrid/plugins/MultiStackedValueAxis.ts +134 -134
  88. package/src/multiGrid/plugins/MultiValueAxis.ts +134 -134
  89. package/src/multiGrid/plugins/OverlappingStackedValueAxes.ts +300 -299
  90. package/src/multiGrid/plugins/OverlappingValueAxes.ts +300 -300
  91. package/src/multiValue/defaults.ts +388 -166
  92. package/src/multiValue/index.ts +13 -9
  93. package/src/multiValue/multiValueObservables.ts +667 -297
  94. package/src/multiValue/plugins/MultiValueLegend.ts +107 -107
  95. package/src/multiValue/plugins/MultiValueTooltip.ts +66 -66
  96. package/src/multiValue/plugins/OrdinalBubbles.ts +0 -0
  97. package/src/multiValue/plugins/OrdinalXAxis.ts +0 -0
  98. package/src/multiValue/plugins/RacingBars.ts +373 -0
  99. package/src/multiValue/plugins/RacingCounterAxis.ts +0 -0
  100. package/src/multiValue/plugins/RacingCounterTexts.ts +300 -0
  101. package/src/multiValue/plugins/RacingValueAxis.ts +115 -0
  102. package/src/multiValue/plugins/RankingAxis_legacy.ts +109 -0
  103. package/src/multiValue/plugins/Scatter.ts +426 -426
  104. package/src/multiValue/plugins/ScatterBubbles.ts +554 -554
  105. package/src/multiValue/plugins/XAxis.ts +108 -0
  106. package/src/multiValue/plugins/XYAux.ts +682 -681
  107. package/src/multiValue/plugins/XYAxes.ts +194 -684
  108. package/src/multiValue/plugins/XYAxes_legacy.ts +684 -0
  109. package/src/multiValue/plugins/{XYZoom.ts → XZoom.ts} +299 -299
  110. package/src/noneData/defaults.ts +102 -102
  111. package/src/noneData/index.ts +3 -3
  112. package/src/noneData/plugins/Container.ts +27 -27
  113. package/src/noneData/plugins/Tooltip.ts +373 -373
  114. package/src/relationship/defaults.ts +220 -218
  115. package/src/relationship/index.ts +5 -5
  116. package/src/relationship/plugins/ForceDirected.ts +1168 -1168
  117. package/src/relationship/plugins/ForceDirectedBubbles.ts +1403 -1403
  118. package/src/relationship/plugins/RelationshipLegend.ts +100 -100
  119. package/src/relationship/plugins/RelationshipTooltip.ts +66 -66
  120. package/src/relationship/relationshipObservables.ts +49 -49
  121. package/src/series/defaults.ts +221 -230
  122. package/src/series/index.ts +9 -9
  123. package/src/series/plugins/Bubbles.ts +636 -620
  124. package/src/series/plugins/Pie.ts +623 -623
  125. package/src/series/plugins/PieEventTexts.ts +284 -284
  126. package/src/series/plugins/PieLabels.ts +640 -640
  127. package/src/series/plugins/Rose.ts +516 -516
  128. package/src/series/plugins/RoseLabels.ts +600 -600
  129. package/src/series/plugins/SeriesLegend.ts +107 -107
  130. package/src/series/plugins/SeriesTooltip.ts +66 -66
  131. package/src/series/seriesObservables.ts +145 -145
  132. package/src/series/seriesUtils.ts +51 -51
  133. package/src/tree/defaults.ts +102 -100
  134. package/src/tree/index.ts +4 -4
  135. package/src/tree/plugins/TreeLegend.ts +100 -100
  136. package/src/tree/plugins/TreeMap.ts +341 -341
  137. package/src/tree/plugins/TreeTooltip.ts +66 -66
  138. package/src/utils/commonUtils.ts +31 -22
  139. package/src/utils/d3Graphics.ts +176 -174
  140. package/src/utils/d3Utils.ts +92 -92
  141. package/src/utils/observables.ts +14 -14
  142. package/src/utils/orbchartsUtils.ts +129 -129
  143. package/tsconfig.base.json +13 -13
  144. package/tsconfig.json +2 -2
  145. package/vite.config.js +22 -22
  146. package/dist/orbcharts-plugins-basic/src/grid/plugins/GridZoom.d.ts +0 -1
  147. package/dist/orbcharts-plugins-basic/src/grid/plugins/StackedBar.d.ts +0 -1
  148. package/dist/orbcharts-plugins-basic/src/index.d.ts +0 -7
  149. package/dist/orbcharts-plugins-basic/src/multiGrid/plugins/MultiStackedBar.d.ts +0 -1
  150. package/dist/orbcharts-plugins-basic/src/multiValue/defaults.d.ts +0 -9
  151. package/dist/orbcharts-plugins-basic/src/multiValue/multiValueObservables.d.ts +0 -33
  152. package/dist/orbcharts-plugins-basic/src/multiValue/plugins/XYZoom.d.ts +0 -1
  153. /package/dist/{orbcharts-plugins-basic/lib → lib}/core-types.d.ts +0 -0
  154. /package/dist/{orbcharts-plugins-basic/lib → lib}/core.d.ts +0 -0
  155. /package/dist/{orbcharts-plugins-basic/lib → lib}/plugins-basic-types.d.ts +0 -0
  156. /package/dist/{orbcharts-plugins-basic/src → src}/base/BaseGroupAxis.d.ts +0 -0
  157. /package/dist/{orbcharts-plugins-basic/src → src}/base/BaseLegend.d.ts +0 -0
  158. /package/dist/{orbcharts-plugins-basic/src → src}/base/BaseTooltip.d.ts +0 -0
  159. /package/dist/{orbcharts-plugins-basic/src → src}/base/types.d.ts +0 -0
  160. /package/dist/{orbcharts-plugins-basic/src → src}/const.d.ts +0 -0
  161. /package/dist/{orbcharts-plugins-basic/src → src}/grid/plugins/Bars.d.ts +0 -0
  162. /package/dist/{orbcharts-plugins-basic/src → src}/grid/plugins/BarsPN.d.ts +0 -0
  163. /package/dist/{orbcharts-plugins-basic/src → src}/grid/plugins/BarsTriangle.d.ts +0 -0
  164. /package/dist/{orbcharts-plugins-basic/src → src}/grid/plugins/Dots.d.ts +0 -0
  165. /package/dist/{orbcharts-plugins-basic/src → src}/grid/plugins/GridLegend.d.ts +0 -0
  166. /package/dist/{orbcharts-plugins-basic/src → src}/grid/plugins/GridTooltip.d.ts +0 -0
  167. /package/dist/{orbcharts-plugins-basic/src → src}/grid/plugins/GroupAux.d.ts +0 -0
  168. /package/dist/{orbcharts-plugins-basic/src → src}/grid/plugins/GroupAxis.d.ts +0 -0
  169. /package/dist/{orbcharts-plugins-basic/src → src}/grid/plugins/LineAreas.d.ts +0 -0
  170. /package/dist/{orbcharts-plugins-basic/src → src}/grid/plugins/Lines.d.ts +0 -0
  171. /package/dist/{orbcharts-plugins-basic/src → src}/grid/plugins/StackedValueAxis.d.ts +0 -0
  172. /package/dist/{orbcharts-plugins-basic/src → src}/grid/plugins/ValueAxis.d.ts +0 -0
  173. /package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/multiGridObservables.d.ts +0 -0
  174. /package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/plugins/MultiBars.d.ts +0 -0
  175. /package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/plugins/MultiBarsTriangle.d.ts +0 -0
  176. /package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/plugins/MultiDots.d.ts +0 -0
  177. /package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/plugins/MultiGridLegend.d.ts +0 -0
  178. /package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/plugins/MultiGridTooltip.d.ts +0 -0
  179. /package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/plugins/MultiGroupAxis.d.ts +0 -0
  180. /package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/plugins/MultiLineAreas.d.ts +0 -0
  181. /package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/plugins/MultiLines.d.ts +0 -0
  182. /package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/plugins/MultiStackedValueAxis.d.ts +0 -0
  183. /package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/plugins/MultiValueAxis.d.ts +0 -0
  184. /package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/plugins/OverlappingStackedValueAxes.d.ts +0 -0
  185. /package/dist/{orbcharts-plugins-basic/src → src}/multiGrid/plugins/OverlappingValueAxes.d.ts +0 -0
  186. /package/dist/{orbcharts-plugins-basic/src → src}/multiValue/plugins/MultiValueLegend.d.ts +0 -0
  187. /package/dist/{orbcharts-plugins-basic/src → src}/multiValue/plugins/MultiValueTooltip.d.ts +0 -0
  188. /package/dist/{orbcharts-plugins-basic/src/multiValue/plugins/Ranking.d.ts → src/multiValue/plugins/OrdinalBubbles.d.ts} +0 -0
  189. /package/dist/{orbcharts-plugins-basic/src/multiValue/plugins/RankingAxis.d.ts → src/multiValue/plugins/OrdinalXAxis.d.ts} +0 -0
  190. /package/dist/{orbcharts-plugins-basic/src/noneData/plugins/Container.d.ts → src/multiValue/plugins/RacingCounterAxis.d.ts} +0 -0
  191. /package/dist/{orbcharts-plugins-basic/src/noneData/plugins/Tooltip.d.ts → src/multiValue/plugins/RankingAxis_legacy.d.ts} +0 -0
  192. /package/dist/{orbcharts-plugins-basic/src → src}/multiValue/plugins/Scatter.d.ts +0 -0
  193. /package/dist/{orbcharts-plugins-basic/src → src}/multiValue/plugins/ScatterBubbles.d.ts +0 -0
  194. /package/dist/{orbcharts-plugins-basic/src → src}/multiValue/plugins/XYAux.d.ts +0 -0
  195. /package/dist/{orbcharts-plugins-basic/src/multiValue/plugins/XYAxes.d.ts → src/multiValue/plugins/XYAxes_legacy.d.ts} +0 -0
  196. /package/dist/{orbcharts-plugins-basic/src → src}/noneData/defaults.d.ts +0 -0
  197. /package/dist/{orbcharts-plugins-basic/src → src}/noneData/index.d.ts +0 -0
  198. /package/dist/{orbcharts-plugins-basic/src/series/plugins/Waffle.d.ts → src/noneData/plugins/Container.d.ts} +0 -0
  199. /package/{src/multiValue/plugins/Ranking.ts → dist/src/noneData/plugins/Tooltip.d.ts} +0 -0
  200. /package/dist/{orbcharts-plugins-basic/src → src}/relationship/defaults.d.ts +0 -0
  201. /package/dist/{orbcharts-plugins-basic/src → src}/relationship/index.d.ts +0 -0
  202. /package/dist/{orbcharts-plugins-basic/src → src}/relationship/plugins/ForceDirected.d.ts +0 -0
  203. /package/dist/{orbcharts-plugins-basic/src → src}/relationship/plugins/ForceDirectedBubbles.d.ts +0 -0
  204. /package/dist/{orbcharts-plugins-basic/src → src}/relationship/plugins/RelationshipLegend.d.ts +0 -0
  205. /package/dist/{orbcharts-plugins-basic/src → src}/relationship/plugins/RelationshipTooltip.d.ts +0 -0
  206. /package/dist/{orbcharts-plugins-basic/src → src}/relationship/relationshipObservables.d.ts +0 -0
  207. /package/dist/{orbcharts-plugins-basic/src → src}/series/defaults.d.ts +0 -0
  208. /package/dist/{orbcharts-plugins-basic/src → src}/series/index.d.ts +0 -0
  209. /package/dist/{orbcharts-plugins-basic/src → src}/series/plugins/Bubbles.d.ts +0 -0
  210. /package/dist/{orbcharts-plugins-basic/src → src}/series/plugins/Pie.d.ts +0 -0
  211. /package/dist/{orbcharts-plugins-basic/src → src}/series/plugins/PieEventTexts.d.ts +0 -0
  212. /package/dist/{orbcharts-plugins-basic/src → src}/series/plugins/PieLabels.d.ts +0 -0
  213. /package/dist/{orbcharts-plugins-basic/src → src}/series/plugins/Rose.d.ts +0 -0
  214. /package/dist/{orbcharts-plugins-basic/src → src}/series/plugins/RoseLabels.d.ts +0 -0
  215. /package/dist/{orbcharts-plugins-basic/src → src}/series/plugins/SeriesLegend.d.ts +0 -0
  216. /package/dist/{orbcharts-plugins-basic/src → src}/series/plugins/SeriesTooltip.d.ts +0 -0
  217. /package/{src/multiValue/plugins/RankingAxis.ts → dist/src/series/plugins/Waffle.d.ts} +0 -0
  218. /package/dist/{orbcharts-plugins-basic/src → src}/series/seriesObservables.d.ts +0 -0
  219. /package/dist/{orbcharts-plugins-basic/src → src}/series/seriesUtils.d.ts +0 -0
  220. /package/dist/{orbcharts-plugins-basic/src → src}/tree/defaults.d.ts +0 -0
  221. /package/dist/{orbcharts-plugins-basic/src → src}/tree/index.d.ts +0 -0
  222. /package/dist/{orbcharts-plugins-basic/src → src}/tree/plugins/TreeLegend.d.ts +0 -0
  223. /package/dist/{orbcharts-plugins-basic/src → src}/tree/plugins/TreeMap.d.ts +0 -0
  224. /package/dist/{orbcharts-plugins-basic/src → src}/tree/plugins/TreeTooltip.d.ts +0 -0
  225. /package/dist/{orbcharts-plugins-basic/src → src}/utils/d3Utils.d.ts +0 -0
  226. /package/dist/{orbcharts-plugins-basic/src → src}/utils/observables.d.ts +0 -0
  227. /package/dist/{orbcharts-plugins-basic/src → src}/utils/orbchartsUtils.d.ts +0 -0
  228. /package/dist/{orbcharts-plugins-basic/vite.config.d.ts → vite.config.d.ts} +0 -0
@@ -1,1169 +1,1169 @@
1
- import * as d3 from 'd3'
2
- import {
3
- of,
4
- combineLatest,
5
- map,
6
- switchMap,
7
- first,
8
- takeUntil,
9
- Subject,
10
- BehaviorSubject,
11
- Observable,
12
- distinctUntilChanged,
13
- shareReplay,
14
- take,
15
- share,
16
- filter,
17
- iif,
18
- EMPTY
19
- } from 'rxjs'
20
- import type { DefinePluginConfig } from '../../../lib/core-types'
21
- import type {
22
- ChartParams,
23
- DatumValue,
24
- DataSeries,
25
- EventName,
26
- EventRelationship,
27
- ComputedDataSeries,
28
- ComputedNode,
29
- ComputedEdge,
30
- ContainerPosition,
31
- Layout
32
- } from '../../../lib/core-types'
33
- import {
34
- defineRelationshipPlugin } from '../../../lib/core'
35
- import type { BubblesParams, ArcScaleType, ForceDirectedParams } from '../../../lib/plugins-basic-types'
36
- import { getDatumColor, getClassName, getUniID } from '../../utils/orbchartsUtils'
37
- import { DEFAULT_FORCE_DIRECTED_PARAMS } from '../defaults'
38
- // import { renderCircleText } from '../../utils/d3Graphics'
39
- import { LAYER_INDEX_OF_GRAPHIC } from '../../const'
40
- import { d3EventObservable } from '../../utils/observables'
41
-
42
- // interface BubblesDatum extends ComputedNode {
43
- // x: number
44
- // y: number
45
- // r: number
46
- // _originR: number // 紀錄變化前的r
47
- // }
48
-
49
- type Zoom = {
50
- xOffset: number
51
- yOffset: number
52
- scaleExtent: {
53
- min: number
54
- max: number
55
- }
56
- }
57
-
58
- // d3 forceSimulation使用的node資料
59
- type RenderNode = d3.SimulationNodeDatum & ComputedNode
60
-
61
- // d3 forceSimulation使用的edge資料
62
- interface RenderEdge extends ComputedEdge {
63
- source: RenderNode
64
- target: RenderNode
65
- }
66
-
67
- // d3 forceSimulation使用的資料
68
- type RenderData = {
69
- nodes: (ComputedNode | RenderNode)[] // 經過d3 forceSimulation計算後的node才有座標資訊
70
- edges: RenderEdge[]
71
- }
72
-
73
- interface D3DragEvent {
74
- active: number
75
- dx: number
76
- dy: number
77
- identifier: string
78
- sourceEvent: MouseEvent
79
- subject: RenderNode
80
- target: any
81
- type: string
82
- x: number
83
- y: number
84
- }
85
-
86
- type DragStatus = 'start' | 'drag' | 'end'
87
-
88
- // type BubblesSimulationDatum = BubblesDatum & d3.SimulationNodeDatum
89
-
90
- const pluginName = 'ForceDirected'
91
-
92
- const gSelectionClassName = getClassName(pluginName, 'zoom-area')
93
- const defsArrowMarkerId = getUniID(pluginName, 'arrow')
94
- const defsArrowMarkerClassName = getClassName(pluginName, 'arrow-marker')
95
- const edgeListGClassName = getClassName(pluginName, 'edge-list-g')
96
- const edgeGClassName = getClassName(pluginName, 'edge-g')
97
- const edgeArrowPathClassName = getClassName(pluginName, 'edge-arrow-path')
98
- const edgeLabelGClassName = getClassName(pluginName, 'edge-label-g')
99
- const edgeLabelClassName = getClassName(pluginName, 'edge-label')
100
- const nodeListGClassName = getClassName(pluginName, 'node-list-g')
101
- const nodeGClassName = getClassName(pluginName, 'node-g')
102
- const nodeCircleClassName = getClassName(pluginName, 'node-circle')
103
- const nodeLabelGClassName = getClassName(pluginName, 'node-label-g')
104
- const nodeLabelClassName = getClassName(pluginName, 'node-label')
105
-
106
- const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_FORCE_DIRECTED_PARAMS> = {
107
- name: pluginName,
108
- defaultParams: DEFAULT_FORCE_DIRECTED_PARAMS,
109
- layerIndex: LAYER_INDEX_OF_GRAPHIC,
110
- validator: (params, { validateColumns }) => {
111
- const result = validateColumns(params, {
112
- dot: {
113
- toBeTypes: ['object']
114
- },
115
- dotLabel: {
116
- toBeTypes: ['object']
117
- },
118
- arrow: {
119
- toBeTypes: ['object']
120
- },
121
- arrowLabel: {
122
- toBeTypes: ['object']
123
- },
124
- force: {
125
- toBeTypes: ['object']
126
- },
127
- zoomable: {
128
- toBeTypes: ['boolean']
129
- },
130
- transform: {
131
- toBeTypes: ['object']
132
- },
133
- scaleExtent: {
134
- toBeTypes: ['object']
135
- }
136
- })
137
- if (params.dot) {
138
- const dotResult = validateColumns(params.dot, {
139
- radius: {
140
- toBeTypes: ['number']
141
- },
142
- fillColorType: {
143
- toBeOption: 'ColorType'
144
- },
145
- strokeColorType: {
146
- toBeOption: 'ColorType'
147
- },
148
- strokeWidth: {
149
- toBeTypes: ['number']
150
- },
151
- styleFn: {
152
- toBeTypes: ['Function']
153
- },
154
- })
155
- if (dotResult.status === 'error') {
156
- return dotResult
157
- }
158
- }
159
- if (params.dotLabel) {
160
- const dotLabelResult = validateColumns(params.dotLabel, {
161
- colorType: {
162
- toBeOption: 'ColorType'
163
- },
164
- sizeFixed: {
165
- toBeTypes: ['boolean']
166
- },
167
- styleFn: {
168
- toBeTypes: ['Function']
169
- },
170
- })
171
- if (dotLabelResult.status === 'error') {
172
- return dotLabelResult
173
- }
174
- }
175
- if (params.arrow) {
176
- const arrowResult = validateColumns(params.arrow, {
177
- colorType: {
178
- toBeOption: 'ColorType'
179
- },
180
- strokeWidth: {
181
- toBeTypes: ['number']
182
- },
183
- pointerWidth: {
184
- toBeTypes: ['number']
185
- },
186
- pointerHeight: {
187
- toBeTypes: ['number']
188
- },
189
- styleFn: {
190
- toBeTypes: ['Function']
191
- },
192
- })
193
- if (arrowResult.status === 'error') {
194
- return arrowResult
195
- }
196
- }
197
- if (params.arrowLabel) {
198
- const arrowLabelResult = validateColumns(params.arrowLabel, {
199
- colorType: {
200
- toBeOption: 'ColorType'
201
- },
202
- sizeFixed: {
203
- toBeTypes: ['boolean']
204
- },
205
- styleFn: {
206
- toBeTypes: ['Function']
207
- },
208
- })
209
- if (arrowLabelResult.status === 'error') {
210
- return arrowLabelResult
211
- }
212
- }
213
- if (params.force) {
214
- const forceResult = validateColumns(params.force, {
215
- nodeStrength: {
216
- toBeTypes: ['number']
217
- },
218
- linkDistance: {
219
- toBeTypes: ['number']
220
- },
221
- velocityDecay: {
222
- toBeTypes: ['number']
223
- },
224
- alphaDecay: {
225
- toBeTypes: ['number']
226
- },
227
- })
228
- if (forceResult.status === 'error') {
229
- return forceResult
230
- }
231
- }
232
- if (params.transform) {
233
- const transformResult = validateColumns(params.transform, {
234
- x: {
235
- toBeTypes: ['number']
236
- },
237
- y: {
238
- toBeTypes: ['number']
239
- },
240
- k: {
241
- toBeTypes: ['number']
242
- },
243
- })
244
- if (transformResult.status === 'error') {
245
- return transformResult
246
- }
247
- }
248
- if (params.scaleExtent) {
249
- const scaleExtentResult = validateColumns(params.scaleExtent, {
250
- min: {
251
- toBeTypes: ['number']
252
- },
253
- max: {
254
- toBeTypes: ['number']
255
- },
256
- })
257
- if (scaleExtentResult.status === 'error') {
258
- return scaleExtentResult
259
- }
260
- }
261
- return result
262
- }
263
- }
264
-
265
- // let force: d3.Simulation<d3.SimulationNodeDatum, undefined> | undefined
266
-
267
- function createSimulation (layout: Layout, fullParams: ForceDirectedParams) {
268
- return d3.forceSimulation()
269
- .velocityDecay(fullParams.force.velocityDecay)
270
- .alphaDecay(fullParams.force.alphaDecay)
271
- .force(
272
- "link",
273
- d3.forceLink()
274
- .id((d: d3.SimulationNodeDatum & ComputedNode) => d.id)
275
- .strength(1)
276
- .distance((d: d3.SimulationLinkDatum<d3.SimulationNodeDatum & ComputedNode>) => {
277
- // if (d.direction === 'top') {
278
- // return 200
279
- // } else {
280
- // return 250
281
- // }
282
- return fullParams.force.linkDistance
283
- })
284
- )
285
- .force("charge", d3.forceManyBody().strength(fullParams.force.nodeStrength))
286
- .force("collision", d3.forceCollide(fullParams.dot.radius).strength(1))
287
- .force("center", d3.forceCenter(layout.width / 2, layout.height / 2))
288
-
289
- }
290
-
291
- function translateFn (d: any): string {
292
- // console.log('translateFn', d)
293
- return "translate(" + d.x + "," + d.y + ")";
294
- }
295
-
296
- function translateCenterFn (d: any): string {
297
- // console.log('translateCenterFn', d)
298
- const x = d.source.x + ((d.target.x - d.source.x) / 2) // 置中的話除2
299
- const y = d.source.y + ((d.target.y - d.source.y) / 2) // 置中的話除2
300
- return "translate(" + x + "," + y + ")";
301
- }
302
-
303
- function linkArcFn (d: RenderEdge): string {
304
- // console.log('linkArcFn', d)
305
-
306
- // var dx = d.target.x - d.source.x,
307
- // dy = d.target.y - d.source.y
308
- // dr讓方向線變成有弧度的
309
- // dr = Math.sqrt(dx * dx + dy * dy);
310
- // return "M" + d.source.x + "," + d.source.y + "A" + dr + "," + dr + " 0 0,1 " + d.target.x + "," + d.target.y;
311
-
312
- // 直線
313
- return "M" + d.source.x + "," + d.source.y + " L" + d.target.x + "," + d.target.y;
314
-
315
-
316
- }
317
-
318
-
319
-
320
- function renderArrowMarker (defsSelection: d3.Selection<SVGDefsElement, any, any, unknown>, fullParams: ForceDirectedParams) {
321
- return defsSelection
322
- .selectAll<SVGMarkerElement, any>(`marker.${defsArrowMarkerClassName}`)
323
- .data([fullParams])
324
- .join(
325
- enter => {
326
- const enterSelection = enter
327
- .append("marker")
328
- .classed(defsArrowMarkerClassName, true)
329
- .attr('id', defsArrowMarkerId)
330
- .attr("viewBox", d => `-${d.arrow.pointerWidth} -${d.arrow.pointerHeight / 2} ${d.arrow.pointerWidth} ${d.arrow.pointerHeight}`)
331
- .attr("orient", "auto")
332
- enterSelection.append("path")
333
- .attr("d", d => `M${-d.arrow.pointerWidth},${-d.arrow.pointerHeight / 2}L0,0L${-d.arrow.pointerWidth},${d.arrow.pointerHeight / 2}`) // 箭頭的尖端為(0,0)
334
- return enterSelection
335
- },
336
- update => {
337
- return update
338
- },
339
- exit => {
340
- return exit.remove()
341
- }
342
- )
343
- .attr("markerWidth", d => d.arrow.pointerWidth)
344
- .attr("markerHeight", d => d.arrow.pointerHeight)
345
- /* refX:修正marker位置(計算出和circle半徑相等的寬度)
346
- (1)circle半徑需加上 strokeWidth/2 是因為框線是以 circle 的邊緣往內及往外擴展,所以 stroke 多出來的寬度是一半而已
347
- (2)circle半徑需除以 path 寬度是因為「marker 的位置會受到 path 的stroke-width影響」,所以要進行修正
348
- (3)- 1 是要修正奇怪的誤差(不知原因)
349
- */
350
- .attr('refX', d => ((d.dot.radius + (fullParams.dot.strokeWidth / 2)) / d.arrow.strokeWidth) - 1)
351
- .attr("refY", 0)
352
-
353
- }
354
-
355
- // function drag (): d3.DragBehavior<Element, unknown, unknown> {
356
- // let originHighlightLockMode: boolean // 拖拽前的highlightLockMode
357
-
358
- // return d3.drag()
359
- // .on("start", (event: D3DragEvent) => {
360
- // console.log('start', event.sourceEvent)
361
- // // if (this.params.lockMode) {
362
- // // return
363
- // // }
364
- // // if (!d3.event.active) {
365
- // // this.forceRestart()
366
- // // }
367
- // // d.fx = d.x
368
- // // d.fy = d.y
369
-
370
- // // // 鎖定模式才不會在拖拽過程式觸發到其他事件造成衝突
371
- // // originHighlightLockMode = this.highlightLockMode
372
- // // this.highlightLockMode = true
373
- // // this.noneStopMode = true
374
- // // // 動畫會有點卡住所以乾脆拿掉
375
- // // if(this.tooltip != null) {
376
- // // this.tooltip.remove()
377
- // // }
378
- // })
379
- // .on("drag", function (event: D3DragEvent) {
380
- // console.log('drag', event)
381
- // // if (this.params.lockMode) {
382
- // // return
383
- // // }
384
- // // if (!d3.event.active) {
385
- // // this.force.alphaTarget(0)
386
- // // }
387
- // // d.fx = d3.event.x
388
- // // d.fy = d3.event.y
389
- // // d3.select(this).attr({
390
- // // 'cx': event.x,
391
- // // 'cy': event.y,
392
- // // })
393
- // d3.select(this)
394
- // .attr('fx', event.x)
395
- // .attr('fy', event.y)
396
- // })
397
- // .on("end", (event: D3DragEvent) => {
398
- // console.log('end', event)
399
- // // if (this.params.lockMode) {
400
- // // return
401
- // // }
402
- // // d.fx = null
403
- // // d.fy = null
404
-
405
- // // this.highlightLockMode = originHighlightLockMode // 還原拖拽前的highlightLockMode
406
- // // this.noneStopMode = false
407
- // // if (this.highlightLockMode) {
408
- // // this.forceStop()
409
- // // }
410
- // })
411
- // }
412
-
413
- function drag (simulation: d3.Simulation<d3.SimulationNodeDatum, undefined>, dragStatus$: BehaviorSubject<DragStatus>) {
414
- function dragstarted (event: D3DragEvent, node: RenderNode) {
415
- if (!event.active) simulation.alphaTarget(0.3).restart()
416
- event.subject.fx = event.x
417
- event.subject.fy = event.y
418
-
419
- dragStatus$.next('start')
420
- }
421
-
422
- function dragged (event: D3DragEvent, node: RenderNode) {
423
- event.subject.fx = event.x
424
- event.subject.fy = event.y
425
-
426
- dragStatus$.next('drag')
427
- }
428
-
429
- function dragended (event: D3DragEvent, node: RenderNode) {
430
- if (!event.active) simulation.alphaTarget(0);
431
- event.subject.fx = null
432
- event.subject.fy = null
433
-
434
- dragStatus$.next('end')
435
- }
436
-
437
- return d3.drag()
438
- .on("start", dragstarted)
439
- .on("drag", dragged)
440
- .on("end", dragended)
441
- }
442
-
443
- function renderNodeG ({ nodeListGSelection, nodes }: {
444
- nodeListGSelection: d3.Selection<SVGGElement, any, any, unknown>
445
- nodes: RenderNode[]
446
- }) {
447
- return nodeListGSelection.selectAll<SVGGElement, RenderNode>('g')
448
- .data(nodes, d => d.id)
449
- .join(
450
- enter => {
451
- const enterSelection = enter
452
- .append('g')
453
- .classed(nodeGClassName, true)
454
- // .attr('cursor', 'pointer')
455
- return enterSelection
456
- },
457
- update => {
458
- return update
459
- },
460
- exit => {
461
- return exit.remove()
462
- }
463
- )
464
- }
465
-
466
- function renderNodeCircle ({ nodeGSelection, fullParams, fullChartParams }: {
467
- nodeGSelection: d3.Selection<SVGGElement, RenderNode, any, unknown>
468
- fullParams: ForceDirectedParams
469
- fullChartParams: ChartParams
470
- }) {
471
- nodeGSelection.each((data,i,g) => {
472
- const gSelection = d3.select(g[i])
473
- gSelection.selectAll<SVGCircleElement, ComputedEdge>('circle')
474
- .data([data])
475
- .join(
476
- enter => {
477
- const enterSelection = enter
478
- .append('circle')
479
- .classed(nodeCircleClassName, true)
480
- .attr('cursor', 'pointer')
481
- return enterSelection
482
- },
483
- update => {
484
- return update
485
- },
486
- exit => {
487
- return exit.remove()
488
- }
489
- )
490
- .attr('r', fullParams.dot.radius)
491
- .attr('fill', d => getDatumColor({ datum: d, colorType: fullParams.dot.fillColorType, fullChartParams }))
492
- .attr('stroke', d => getDatumColor({ datum: d, colorType: fullParams.dot.strokeColorType, fullChartParams }))
493
- .attr('stroke-width', fullParams.dot.strokeWidth)
494
- .attr('style', d => fullParams.dot.styleFn(d))
495
- })
496
-
497
- return nodeGSelection.select<SVGCircleElement>(`circle.${nodeCircleClassName}`)
498
- }
499
-
500
- function renderNodeLabelG ({ nodeGSelection, fullParams }: {
501
- nodeGSelection: d3.Selection<SVGGElement, any, any, unknown>
502
- fullParams: ForceDirectedParams
503
- }) {
504
- nodeGSelection.each((data,i,g) => {
505
- const gSelection = d3.select(g[i])
506
- gSelection.selectAll<SVGGElement, RenderNode>('g')
507
- .data([data])
508
- .join(
509
- enter => {
510
- const enterSelection = enter
511
- .append('g')
512
- .classed(nodeLabelGClassName, true)
513
- // .attr('cursor', 'pointer')
514
- return enterSelection
515
- },
516
- update => {
517
- return update
518
- },
519
- exit => {
520
- return exit.remove()
521
- }
522
- )
523
- .attr('transform', `translate(0, ${- fullParams.dot.radius - 10})`)
524
- })
525
-
526
- return nodeGSelection.select<SVGTextElement>(`g.${nodeLabelGClassName}`)
527
- }
528
-
529
- function renderNodeLabel ({ nodeLabelGSelection, fullParams, fullChartParams }: {
530
- nodeLabelGSelection: d3.Selection<SVGGElement, RenderNode, any, unknown>
531
- fullParams: ForceDirectedParams
532
- fullChartParams: ChartParams
533
- }) {
534
- nodeLabelGSelection.each((data,i,g) => {
535
- const gSelection = d3.select(g[i])
536
- gSelection.selectAll<SVGTextElement, RenderNode>('text')
537
- .data([data], d => d.id)
538
- .join(
539
- enter => {
540
- const enterSelection = enter
541
- .append('text')
542
- .classed(nodeLabelClassName, true)
543
- // .attr('cursor', 'pointer')
544
- .attr('text-anchor', 'middle')
545
- .attr('pointer-events', 'none')
546
- return enterSelection
547
- },
548
- update => {
549
- return update
550
- },
551
- exit => {
552
- return exit.remove()
553
- }
554
- )
555
- .text(d => d.label)
556
- .attr('fill', d => getDatumColor({ datum: d, colorType: fullParams.dotLabel.colorType, fullChartParams }))
557
- .attr('font-size', fullChartParams.styles.textSize)
558
- .attr('style', d => fullParams.dotLabel.styleFn(d))
559
- })
560
-
561
- return nodeLabelGSelection.select<SVGTextElement>(`text.${nodeLabelClassName}`)
562
- }
563
-
564
- function renderEdgeG ({ edgeListGSelection, edges }: {
565
- edgeListGSelection: d3.Selection<SVGGElement, any, any, unknown>
566
- edges: RenderEdge[]
567
- }) {
568
- return edgeListGSelection.selectAll<SVGGElement, RenderEdge>('g')
569
- .data(edges, d => d.id)
570
- .join(
571
- enter => {
572
- const enterSelection = enter
573
- .append('g')
574
- .classed(edgeGClassName, true)
575
- // .attr('cursor', 'pointer')
576
- return enterSelection
577
- },
578
- update => {
579
- return update
580
- },
581
- exit => {
582
- return exit.remove()
583
- }
584
- )
585
- }
586
-
587
- function renderEdgeArrowPath ({ edgeGSelection, fullParams, fullChartParams }: {
588
- edgeGSelection: d3.Selection<SVGGElement, RenderEdge, any, unknown>
589
- fullParams: ForceDirectedParams
590
- fullChartParams: ChartParams
591
- }) {
592
- edgeGSelection.each((data,i,g) => {
593
- const gSelection = d3.select(g[i])
594
- gSelection.selectAll<SVGPathElement, ComputedEdge>('path')
595
- .data([data])
596
- .join(
597
- enter => {
598
- return enter
599
- .append('path')
600
- .classed(edgeArrowPathClassName, true)
601
- .attr('marker-end', `url(#${defsArrowMarkerId})`)
602
- },
603
- update => {
604
- return update
605
- },
606
- exit => {
607
- return exit.remove()
608
- }
609
- )
610
- .attr('stroke', d => getDatumColor({ datum: d.data, colorType: fullParams.arrow.colorType, fullChartParams }))
611
- .attr('stroke-width', fullParams.arrow.strokeWidth)
612
- .attr('style', d => fullParams.arrow.styleFn(d))
613
- })
614
-
615
- return edgeGSelection.select<SVGPathElement>(`path.${edgeArrowPathClassName}`)
616
- }
617
-
618
- function renderEdgeLabelG ({ edgeGSelection }: {
619
- edgeGSelection: d3.Selection<SVGGElement, any, any, unknown>
620
- }) {
621
- edgeGSelection.each((data,i,g) => {
622
- const gSelection = d3.select(g[i])
623
- gSelection.selectAll<SVGGElement, RenderEdge>('g')
624
- .data([data])
625
- .join(
626
- enter => {
627
- const enterSelection = enter
628
- .append('g')
629
- .classed(edgeLabelGClassName, true)
630
- // .attr('cursor', 'pointer')
631
- return enterSelection
632
- },
633
- update => {
634
- return update
635
- },
636
- exit => {
637
- return exit.remove()
638
- }
639
- )
640
- })
641
-
642
- return edgeGSelection.select<SVGTextElement>(`g.${edgeLabelGClassName}`)
643
- }
644
-
645
- function renderEdgeLabel ({ edgeLabelGSelection, fullParams, fullChartParams }: {
646
- edgeLabelGSelection: d3.Selection<SVGGElement, RenderEdge, any, unknown>
647
- fullParams: ForceDirectedParams
648
- fullChartParams: ChartParams
649
- }) {
650
- edgeLabelGSelection.each((data,i,g) => {
651
- const gSelection = d3.select(g[i])
652
- gSelection.selectAll<SVGTextElement, RenderEdge>('text')
653
- .data([data], d => d.id)
654
- .join(
655
- enter => {
656
- const enterSelection = enter
657
- .append('text')
658
- .classed(edgeLabelClassName, true)
659
- // .attr('cursor', 'pointer')
660
- .attr('text-anchor', 'middle')
661
- .attr('pointer-events', 'none')
662
- return enterSelection
663
- },
664
- update => {
665
- return update
666
- },
667
- exit => {
668
- return exit.remove()
669
- }
670
- )
671
- .text(d => d.label)
672
- .attr('fill', d => getDatumColor({ datum: d, colorType: fullParams.arrowLabel.colorType, fullChartParams }))
673
- .attr('font-size', fullChartParams.styles.textSize)
674
- .attr('style', d => fullParams.arrowLabel.styleFn(d))
675
- })
676
-
677
- return edgeLabelGSelection.select<SVGTextElement>(`text.${edgeLabelClassName}`)
678
- }
679
-
680
-
681
- // function renderBubbles ({ selection, bubblesData, fullParams, sumSeries }: {
682
- // selection: d3.Selection<SVGGElement, any, any, any>
683
- // bubblesData: BubblesDatum[]
684
- // fullParams: BubblesParams
685
- // sumSeries: boolean
686
- // }) {
687
- // const bubblesSelection = selection.selectAll<SVGGElement, BubblesDatum>("g")
688
- // .data(bubblesData, (d) => d.id)
689
- // .join(
690
- // enter => {
691
- // const enterSelection = enter
692
- // .append('g')
693
- // .attr('cursor', 'pointer')
694
- // .attr('font-size', 12)
695
- // .style('fill', '#ffffff')
696
- // .attr("text-anchor", "middle")
697
-
698
- // enterSelection
699
- // .append("circle")
700
- // .attr("class", "node")
701
- // .attr("cx", 0)
702
- // .attr("cy", 0)
703
- // // .attr("r", 1e-6)
704
- // .attr('fill', (d) => d.color)
705
- // // .transition()
706
- // // .duration(500)
707
-
708
- // enterSelection
709
- // .append('text')
710
- // .style('opacity', 0.8)
711
- // .attr('pointer-events', 'none')
712
-
713
- // return enterSelection
714
- // },
715
- // update => {
716
- // return update
717
- // },
718
- // exit => {
719
- // return exit
720
- // .remove()
721
- // }
722
- // )
723
- // .attr("transform", (d) => {
724
- // return `translate(${d.x},${d.y})`
725
- // })
726
-
727
- // // 泡泡文字要使用的的資料欄位
728
- // const textDataColumn = sumSeries ? 'seriesLabel' : 'label'// 如果有合併series則使用seriesLabel
729
-
730
- // bubblesSelection.select('circle')
731
- // .transition()
732
- // .duration(200)
733
- // .attr("r", (d) => d.r)
734
- // .attr('fill', (d) => d.color)
735
- // bubblesSelection
736
- // .each((d,i,g) => {
737
- // const gSelection = d3.select(g[i])
738
- // let breakAll = true
739
- // if (d[textDataColumn].length <= fullParams.label.maxLineLength) {
740
- // breakAll = false
741
- // }
742
- // gSelection.call(renderCircleText, {
743
- // text: d[textDataColumn],
744
- // radius: d.r * fullParams.label.fillRate,
745
- // lineHeight: fullParams.label.lineHeight,
746
- // isBreakAll: breakAll
747
- // })
748
-
749
- // })
750
-
751
- // return bubblesSelection
752
- // }
753
-
754
- // function setHighlightData ({ data, highlightRIncrease, highlightIds }: {
755
- // data: BubblesDatum[]
756
- // // fullParams: BubblesParams
757
- // highlightRIncrease: number
758
- // highlightIds: string[]
759
- // }) {
760
- // if (highlightRIncrease == 0) {
761
- // return
762
- // }
763
- // if (!highlightIds.length) {
764
- // data.forEach(d => d.r = d._originR)
765
- // return
766
- // }
767
- // data.forEach(d => {
768
- // if (highlightIds.includes(d.id)) {
769
- // d.r = d._originR + highlightRIncrease
770
- // } else {
771
- // d.r = d._originR
772
- // }
773
- // })
774
- // }
775
-
776
-
777
- // function groupBubbles ({ fullParams, SeriesContainerPositionMap }: {
778
- // fullParams: BubblesParams
779
- // // graphicWidth: number
780
- // // graphicHeight: number
781
- // SeriesContainerPositionMap: Map<string, ContainerPosition>
782
- // }) {
783
- // // console.log('groupBubbles')
784
- // force!
785
- // // .force('x', d3.forceX().strength(fullParams.force.strength).x(graphicWidth / 2))
786
- // // .force('y', d3.forceY().strength(fullParams.force.strength).y(graphicHeight / 2))
787
- // .force('x', d3.forceX().strength(fullParams.force.strength).x((data: BubblesSimulationDatum) => {
788
- // return SeriesContainerPositionMap.get(data.seriesLabel)!.centerX
789
- // }))
790
- // .force('y', d3.forceY().strength(fullParams.force.strength).y((data: BubblesSimulationDatum) => {
791
- // return SeriesContainerPositionMap.get(data.seriesLabel)!.centerY
792
- // }))
793
-
794
- // force!.alpha(1).restart()
795
- // }
796
-
797
- function highlightNodes ({ nodeGSelection, edgeGSelection, highlightIds, fullChartParams }: {
798
- nodeGSelection: d3.Selection<SVGGElement, RenderNode, SVGGElement, any>
799
- edgeGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any>
800
- fullChartParams: ChartParams
801
- highlightIds: string[]
802
- }) {
803
- nodeGSelection.interrupt('highlight')
804
- edgeGSelection.interrupt('highlight')
805
- // console.log(highlightIds)
806
- if (!highlightIds.length) {
807
- nodeGSelection
808
- .transition('highlight')
809
- .style('opacity', 1)
810
- edgeGSelection
811
- .transition('highlight')
812
- .style('opacity', 1)
813
- return
814
- }
815
-
816
- edgeGSelection
817
- .style('opacity', fullChartParams.styles.unhighlightedOpacity)
818
-
819
- nodeGSelection.each((d, i, n) => {
820
- const segment = d3.select(n[i])
821
-
822
- if (highlightIds.includes(d.id)) {
823
- segment
824
- .style('opacity', 1)
825
- .transition('highlight')
826
- .ease(d3.easeElastic)
827
- .duration(500)
828
- } else {
829
- // 取消
830
- segment
831
- .style('opacity', fullChartParams.styles.unhighlightedOpacity)
832
- }
833
- })
834
- }
835
-
836
- // 暫不處理edge的highlight
837
- // function highlightEdges ({ edgeGSelection, highlightIds, fullChartParams }: {
838
- // edgeGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any>
839
- // fullChartParams: ChartParams
840
- // highlightIds: string[]
841
- // }) {
842
- // edgeGSelection.interrupt('highlight')
843
-
844
- // if (!highlightIds.length) {
845
- // edgeGSelection
846
- // .transition('highlight')
847
- // .style('opacity', 1)
848
- // return
849
- // }
850
-
851
- // edgeGSelection.each((d, i, n) => {
852
- // const segment = d3.select(n[i])
853
-
854
- // if (highlightIds.includes(d.id)) {
855
- // segment
856
- // .style('opacity', 1)
857
- // .transition('highlight')
858
- // .ease(d3.easeElastic)
859
- // .duration(500)
860
- // } else {
861
- // // 取消放大
862
- // segment
863
- // .style('opacity', fullChartParams.styles.unhighlightedOpacity)
864
- // }
865
- // })
866
- // }
867
-
868
- export const ForceDirected = defineRelationshipPlugin(pluginConfig)(({ selection, rootSelection, name, observer, subject }) => {
869
-
870
- const destroy$ = new Subject()
871
-
872
- const gSelection = selection.append('g').classed(gSelectionClassName, true)
873
- const defsSelection = gSelection.append('defs')
874
- const edgeListGSelection = gSelection.append('g').classed(edgeListGClassName, true)
875
- const nodeListGSelection = gSelection.append('g').classed(nodeListGClassName, true)
876
-
877
- let nodeGSelection: d3.Selection<SVGGElement, RenderNode, SVGGElement, any> | undefined
878
- let nodeCircleSelection: d3.Selection<SVGCircleElement, RenderNode, SVGGElement, any> | undefined
879
- let nodeLabelGSelection: d3.Selection<SVGGElement, RenderNode, SVGGElement, any> | undefined
880
- let nodeLabelSelection: d3.Selection<SVGTextElement, RenderNode, SVGGElement, any> | undefined
881
- let edgeGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any> | undefined
882
- let edgeArrowSelection: d3.Selection<SVGPathElement, RenderEdge, SVGGElement, any> | undefined
883
- let edgeLabelGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any> | undefined
884
- let edgeLabelSelection: d3.Selection<SVGTextElement, RenderEdge, SVGGElement, any> | undefined
885
-
886
- const dragStatus$ = new BehaviorSubject<DragStatus>('end') // start, drag, end
887
- const mouseEvent$ = new Subject<EventRelationship>()
888
-
889
- // <marker> marker selection
890
- observer.fullParams$.pipe(
891
- takeUntil(destroy$),
892
- map(fullParams => {
893
- return renderArrowMarker(defsSelection, fullParams)
894
- })
895
- ).subscribe()
896
-
897
- // init zoom
898
- const d3Zoom$ = observer.fullParams$.pipe(
899
- takeUntil(destroy$),
900
- // map(d => d.scaleExtent),
901
- // distinctUntilChanged((a, b) => String(a) === String(b)),
902
- // first(),
903
- map(data => {
904
- let d3Zoom = data.zoomable
905
- ? d3.zoom().on('zoom', (event) => {
906
- // console.log(event)
907
- // this.svgGroup.attr('transform', `translate(
908
- // ${event.transform.x + (this.zoom.xOffset * event.transform.k)},
909
- // ${event.transform.y + (this.zoom.yOffset * event.transform.k)}
910
- // ) scale(
911
- // ${event.transform.k}
912
- // )`)
913
- gSelection.attr('transform', `translate(
914
- ${event.transform.x},
915
- ${event.transform.y}
916
- ) scale(
917
- ${event.transform.k}
918
- )`)
919
-
920
- if (data.dotLabel.sizeFixed && nodeLabelSelection) {
921
- // 反向 scale 抵消掉放大縮小
922
- nodeLabelSelection.attr('transform', `scale(${1 / event.transform.k})`)
923
- }
924
- if (data.arrowLabel.sizeFixed && edgeLabelSelection) {
925
- // 反向 scale 抵消掉放大縮小
926
- edgeLabelSelection.attr('transform', `scale(${1 / event.transform.k})`)
927
- }
928
- })
929
- : d3.zoom().on('zoom', null)
930
- if (data.scaleExtent) {
931
- d3Zoom.scaleExtent([data.scaleExtent.min, data.scaleExtent.max])
932
- }
933
- rootSelection.call(d3Zoom)
934
-
935
- return d3Zoom
936
- }),
937
- // shareReplay(1)
938
- )
939
-
940
- // zoom transform
941
- combineLatest({
942
- d3Zoom: d3Zoom$,
943
- transform: observer.fullParams$.pipe(
944
- takeUntil(destroy$),
945
- map(d => d.transform),
946
- )
947
- }).pipe(
948
- takeUntil(destroy$),
949
- switchMap(async d => d)
950
- ).subscribe(data => {
951
- // console.log('call')
952
- selection.call(
953
- data.d3Zoom.transform, d3.zoomIdentity
954
- .translate(data.transform.x, data.transform.y)
955
- .scale(data.transform.k)
956
- )
957
- })
958
-
959
-
960
- const simulation$: Observable<d3.Simulation<d3.SimulationNodeDatum, undefined>> = combineLatest({
961
- layout: observer.layout$.pipe(
962
- first() // 只使用第一次的尺寸(置中)
963
- ),
964
- fullParams: observer.fullParams$
965
- }).pipe(
966
- takeUntil(destroy$),
967
- switchMap(async d => d),
968
- map(data => createSimulation(data.layout, data.fullParams)),
969
- shareReplay(1)
970
- )
971
-
972
- const renderData$: Observable<RenderData> = observer.visibleComputedData$.pipe(
973
- takeUntil(destroy$),
974
- map(data => {
975
- return {
976
- nodes: data.nodes,
977
- edges: data.edges.map(_d => {
978
- let d: RenderEdge = _d as RenderEdge
979
- d.source = _d.startNode // reference
980
- d.target = _d.endNode
981
- return d
982
- })
983
- }
984
- }),
985
- shareReplay(1)
986
- )
987
-
988
- combineLatest({
989
- renderData: renderData$,
990
- computedData: observer.computedData$,
991
- CategoryNodeMap: observer.CategoryNodeMap$,
992
- simulation: simulation$,
993
- fullParams: observer.fullParams$,
994
- fullChartParams: observer.fullChartParams$
995
- }).pipe(
996
- takeUntil(destroy$),
997
- switchMap(async d => d),
998
- ).subscribe(data => {
999
-
1000
- nodeGSelection = renderNodeG({
1001
- nodeListGSelection: nodeListGSelection,
1002
- nodes: data.renderData.nodes,
1003
- })
1004
-
1005
- nodeCircleSelection = renderNodeCircle({
1006
- nodeGSelection: nodeGSelection,
1007
- fullParams: data.fullParams,
1008
- fullChartParams: data.fullChartParams
1009
- })
1010
- nodeGSelection.call(drag(data.simulation, dragStatus$))
1011
-
1012
- nodeLabelGSelection = renderNodeLabelG({
1013
- nodeGSelection: nodeGSelection,
1014
- fullParams: data.fullParams
1015
- })
1016
-
1017
- nodeLabelSelection = renderNodeLabel({
1018
- nodeLabelGSelection: nodeLabelGSelection,
1019
- fullParams: data.fullParams,
1020
- fullChartParams: data.fullChartParams
1021
- })
1022
-
1023
- edgeGSelection = renderEdgeG({
1024
- edgeListGSelection: edgeListGSelection,
1025
- edges: data.renderData.edges
1026
- })
1027
-
1028
- edgeArrowSelection = renderEdgeArrowPath({
1029
- edgeGSelection: edgeGSelection,
1030
- fullParams: data.fullParams,
1031
- fullChartParams: data.fullChartParams
1032
- })
1033
-
1034
- edgeLabelGSelection = renderEdgeLabelG({
1035
- edgeGSelection: edgeGSelection,
1036
- })
1037
-
1038
- edgeLabelSelection = renderEdgeLabel({
1039
- edgeLabelGSelection: edgeLabelGSelection,
1040
- fullParams: data.fullParams,
1041
- fullChartParams: data.fullChartParams
1042
- })
1043
-
1044
- data.simulation.nodes(data.renderData.nodes)
1045
- .on('tick', () => {
1046
- edgeArrowSelection.attr('d', linkArcFn)
1047
- nodeGSelection.attr('transform', translateFn)
1048
- // nodeLabelGSelection.attr('transform', d => translateFn({
1049
- // x: d.x,
1050
- // y: d.y - data.fullParams.dot.radius - 10
1051
- // }))
1052
- edgeLabelGSelection.attr('transform', d => translateCenterFn(d))
1053
- })
1054
- ;(data.simulation.force("link") as any).links(data.renderData.edges)
1055
-
1056
- data.simulation.alpha(0.3).restart()
1057
-
1058
- nodeCircleSelection
1059
- .on('mouseover', (event, datum) => {
1060
- event.stopPropagation()
1061
-
1062
- mouseEvent$.next({
1063
- type: 'relationship',
1064
- eventName: 'mouseover',
1065
- pluginName,
1066
- highlightTarget: data.fullChartParams.highlightTarget,
1067
- datum: datum,
1068
- category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1069
- categoryIndex: datum.categoryIndex,
1070
- categoryLabel: datum.categoryLabel,
1071
- event,
1072
- data: data.computedData
1073
- })
1074
- })
1075
- .on('mousemove', (event, datum) => {
1076
- event.stopPropagation()
1077
-
1078
- mouseEvent$.next({
1079
- type: 'relationship',
1080
- eventName: 'mousemove',
1081
- pluginName,
1082
- highlightTarget: data.fullChartParams.highlightTarget,
1083
- datum: datum,
1084
- category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1085
- categoryIndex: datum.categoryIndex,
1086
- categoryLabel: datum.categoryLabel,
1087
- event,
1088
- data: data.computedData
1089
- })
1090
- })
1091
- .on('mouseout', (event, datum) => {
1092
- event.stopPropagation()
1093
-
1094
- mouseEvent$.next({
1095
- type: 'relationship',
1096
- eventName: 'mouseout',
1097
- pluginName,
1098
- highlightTarget: data.fullChartParams.highlightTarget,
1099
- datum: datum,
1100
- category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1101
- categoryIndex: datum.categoryIndex,
1102
- categoryLabel: datum.categoryLabel,
1103
- event,
1104
- data: data.computedData
1105
- })
1106
- })
1107
- .on('click', (event, datum) => {
1108
- event.stopPropagation()
1109
-
1110
- mouseEvent$.next({
1111
- type: 'relationship',
1112
- eventName: 'click',
1113
- pluginName,
1114
- highlightTarget: data.fullChartParams.highlightTarget,
1115
- datum: datum,
1116
- category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1117
- categoryIndex: datum.categoryIndex,
1118
- categoryLabel: datum.categoryLabel,
1119
- event,
1120
- data: data.computedData
1121
- })
1122
- })
1123
- })
1124
-
1125
- dragStatus$.pipe(
1126
- distinctUntilChanged((a, b) => a === b),
1127
- // 只有沒有托曳時才執行
1128
- switchMap(d => iif(() => d === 'end', mouseEvent$, EMPTY))
1129
- ).subscribe(data => {
1130
- subject.event$.next(data)
1131
- })
1132
-
1133
- combineLatest({
1134
- renderData: renderData$,
1135
- highlightNodes: observer.relationshipHighlightNodes$.pipe(
1136
- map(data => data.map(d => d.id))
1137
- ),
1138
- highlightEdges: observer.relationshipHighlightEdges$.pipe(
1139
- map(data => data.map(d => d.id))
1140
- ),
1141
- fullChartParams: observer.fullChartParams$,
1142
- fullParams: observer.fullParams$,
1143
- }).pipe(
1144
- takeUntil(destroy$),
1145
- switchMap(async d => d)
1146
- ).subscribe(data => {
1147
- if (!nodeGSelection || !edgeGSelection) {
1148
- return
1149
- }
1150
-
1151
- highlightNodes({
1152
- nodeGSelection,
1153
- edgeGSelection,
1154
- highlightIds: data.highlightNodes,
1155
- fullChartParams: data.fullChartParams
1156
- })
1157
- // highlightEdges({
1158
- // edgeGSelection,
1159
- // highlightIds: data.highlightEdges,
1160
- // fullChartParams: data.fullChartParams
1161
- // })
1162
- })
1163
-
1164
-
1165
-
1166
- return () => {
1167
- destroy$.next(undefined)
1168
- }
1
+ import * as d3 from 'd3'
2
+ import {
3
+ of,
4
+ combineLatest,
5
+ map,
6
+ switchMap,
7
+ first,
8
+ takeUntil,
9
+ Subject,
10
+ BehaviorSubject,
11
+ Observable,
12
+ distinctUntilChanged,
13
+ shareReplay,
14
+ take,
15
+ share,
16
+ filter,
17
+ iif,
18
+ EMPTY
19
+ } from 'rxjs'
20
+ import type { DefinePluginConfig } from '../../../lib/core-types'
21
+ import type {
22
+ ChartParams,
23
+ DatumValue,
24
+ DataSeries,
25
+ EventName,
26
+ EventRelationship,
27
+ ComputedDataSeries,
28
+ ComputedNode,
29
+ ComputedEdge,
30
+ ContainerPosition,
31
+ Layout
32
+ } from '../../../lib/core-types'
33
+ import {
34
+ defineRelationshipPlugin } from '../../../lib/core'
35
+ import type { BubblesParams, ArcScaleType, ForceDirectedParams } from '../../../lib/plugins-basic-types'
36
+ import { getDatumColor, getClassName, getUniID } from '../../utils/orbchartsUtils'
37
+ import { DEFAULT_FORCE_DIRECTED_PARAMS } from '../defaults'
38
+ // import { renderCircleText } from '../../utils/d3Graphics'
39
+ import { LAYER_INDEX_OF_GRAPHIC } from '../../const'
40
+ import { d3EventObservable } from '../../utils/observables'
41
+
42
+ // interface BubblesDatum extends ComputedNode {
43
+ // x: number
44
+ // y: number
45
+ // r: number
46
+ // _originR: number // 紀錄變化前的r
47
+ // }
48
+
49
+ type Zoom = {
50
+ xOffset: number
51
+ yOffset: number
52
+ scaleExtent: {
53
+ min: number
54
+ max: number
55
+ }
56
+ }
57
+
58
+ // d3 forceSimulation使用的node資料
59
+ type RenderNode = d3.SimulationNodeDatum & ComputedNode
60
+
61
+ // d3 forceSimulation使用的edge資料
62
+ interface RenderEdge extends ComputedEdge {
63
+ source: RenderNode
64
+ target: RenderNode
65
+ }
66
+
67
+ // d3 forceSimulation使用的資料
68
+ type RenderData = {
69
+ nodes: (ComputedNode | RenderNode)[] // 經過d3 forceSimulation計算後的node才有座標資訊
70
+ edges: RenderEdge[]
71
+ }
72
+
73
+ interface D3DragEvent {
74
+ active: number
75
+ dx: number
76
+ dy: number
77
+ identifier: string
78
+ sourceEvent: MouseEvent
79
+ subject: RenderNode
80
+ target: any
81
+ type: string
82
+ x: number
83
+ y: number
84
+ }
85
+
86
+ type DragStatus = 'start' | 'drag' | 'end'
87
+
88
+ // type BubblesSimulationDatum = BubblesDatum & d3.SimulationNodeDatum
89
+
90
+ const pluginName = 'ForceDirected'
91
+
92
+ const gSelectionClassName = getClassName(pluginName, 'zoom-area')
93
+ const defsArrowMarkerId = getUniID(pluginName, 'arrow')
94
+ const defsArrowMarkerClassName = getClassName(pluginName, 'arrow-marker')
95
+ const edgeListGClassName = getClassName(pluginName, 'edge-list-g')
96
+ const edgeGClassName = getClassName(pluginName, 'edge-g')
97
+ const edgeArrowPathClassName = getClassName(pluginName, 'edge-arrow-path')
98
+ const edgeLabelGClassName = getClassName(pluginName, 'edge-label-g')
99
+ const edgeLabelClassName = getClassName(pluginName, 'edge-label')
100
+ const nodeListGClassName = getClassName(pluginName, 'node-list-g')
101
+ const nodeGClassName = getClassName(pluginName, 'node-g')
102
+ const nodeCircleClassName = getClassName(pluginName, 'node-circle')
103
+ const nodeLabelGClassName = getClassName(pluginName, 'node-label-g')
104
+ const nodeLabelClassName = getClassName(pluginName, 'node-label')
105
+
106
+ const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_FORCE_DIRECTED_PARAMS> = {
107
+ name: pluginName,
108
+ defaultParams: DEFAULT_FORCE_DIRECTED_PARAMS,
109
+ layerIndex: LAYER_INDEX_OF_GRAPHIC,
110
+ validator: (params, { validateColumns }) => {
111
+ const result = validateColumns(params, {
112
+ dot: {
113
+ toBeTypes: ['object']
114
+ },
115
+ dotLabel: {
116
+ toBeTypes: ['object']
117
+ },
118
+ arrow: {
119
+ toBeTypes: ['object']
120
+ },
121
+ arrowLabel: {
122
+ toBeTypes: ['object']
123
+ },
124
+ force: {
125
+ toBeTypes: ['object']
126
+ },
127
+ zoomable: {
128
+ toBeTypes: ['boolean']
129
+ },
130
+ transform: {
131
+ toBeTypes: ['object']
132
+ },
133
+ scaleExtent: {
134
+ toBeTypes: ['object']
135
+ }
136
+ })
137
+ if (params.dot) {
138
+ const dotResult = validateColumns(params.dot, {
139
+ radius: {
140
+ toBeTypes: ['number']
141
+ },
142
+ fillColorType: {
143
+ toBeOption: 'ColorType'
144
+ },
145
+ strokeColorType: {
146
+ toBeOption: 'ColorType'
147
+ },
148
+ strokeWidth: {
149
+ toBeTypes: ['number']
150
+ },
151
+ styleFn: {
152
+ toBeTypes: ['Function']
153
+ },
154
+ })
155
+ if (dotResult.status === 'error') {
156
+ return dotResult
157
+ }
158
+ }
159
+ if (params.dotLabel) {
160
+ const dotLabelResult = validateColumns(params.dotLabel, {
161
+ colorType: {
162
+ toBeOption: 'ColorType'
163
+ },
164
+ sizeFixed: {
165
+ toBeTypes: ['boolean']
166
+ },
167
+ styleFn: {
168
+ toBeTypes: ['Function']
169
+ },
170
+ })
171
+ if (dotLabelResult.status === 'error') {
172
+ return dotLabelResult
173
+ }
174
+ }
175
+ if (params.arrow) {
176
+ const arrowResult = validateColumns(params.arrow, {
177
+ colorType: {
178
+ toBeOption: 'ColorType'
179
+ },
180
+ strokeWidth: {
181
+ toBeTypes: ['number']
182
+ },
183
+ pointerWidth: {
184
+ toBeTypes: ['number']
185
+ },
186
+ pointerHeight: {
187
+ toBeTypes: ['number']
188
+ },
189
+ styleFn: {
190
+ toBeTypes: ['Function']
191
+ },
192
+ })
193
+ if (arrowResult.status === 'error') {
194
+ return arrowResult
195
+ }
196
+ }
197
+ if (params.arrowLabel) {
198
+ const arrowLabelResult = validateColumns(params.arrowLabel, {
199
+ colorType: {
200
+ toBeOption: 'ColorType'
201
+ },
202
+ sizeFixed: {
203
+ toBeTypes: ['boolean']
204
+ },
205
+ styleFn: {
206
+ toBeTypes: ['Function']
207
+ },
208
+ })
209
+ if (arrowLabelResult.status === 'error') {
210
+ return arrowLabelResult
211
+ }
212
+ }
213
+ if (params.force) {
214
+ const forceResult = validateColumns(params.force, {
215
+ nodeStrength: {
216
+ toBeTypes: ['number']
217
+ },
218
+ linkDistance: {
219
+ toBeTypes: ['number']
220
+ },
221
+ velocityDecay: {
222
+ toBeTypes: ['number']
223
+ },
224
+ alphaDecay: {
225
+ toBeTypes: ['number']
226
+ },
227
+ })
228
+ if (forceResult.status === 'error') {
229
+ return forceResult
230
+ }
231
+ }
232
+ if (params.transform) {
233
+ const transformResult = validateColumns(params.transform, {
234
+ x: {
235
+ toBeTypes: ['number']
236
+ },
237
+ y: {
238
+ toBeTypes: ['number']
239
+ },
240
+ k: {
241
+ toBeTypes: ['number']
242
+ },
243
+ })
244
+ if (transformResult.status === 'error') {
245
+ return transformResult
246
+ }
247
+ }
248
+ if (params.scaleExtent) {
249
+ const scaleExtentResult = validateColumns(params.scaleExtent, {
250
+ min: {
251
+ toBeTypes: ['number']
252
+ },
253
+ max: {
254
+ toBeTypes: ['number']
255
+ },
256
+ })
257
+ if (scaleExtentResult.status === 'error') {
258
+ return scaleExtentResult
259
+ }
260
+ }
261
+ return result
262
+ }
263
+ }
264
+
265
+ // let force: d3.Simulation<d3.SimulationNodeDatum, undefined> | undefined
266
+
267
+ function createSimulation (layout: Layout, fullParams: ForceDirectedParams) {
268
+ return d3.forceSimulation()
269
+ .velocityDecay(fullParams.force.velocityDecay)
270
+ .alphaDecay(fullParams.force.alphaDecay)
271
+ .force(
272
+ "link",
273
+ d3.forceLink()
274
+ .id((d: d3.SimulationNodeDatum & ComputedNode) => d.id)
275
+ .strength(1)
276
+ .distance((d: d3.SimulationLinkDatum<d3.SimulationNodeDatum & ComputedNode>) => {
277
+ // if (d.direction === 'top') {
278
+ // return 200
279
+ // } else {
280
+ // return 250
281
+ // }
282
+ return fullParams.force.linkDistance
283
+ })
284
+ )
285
+ .force("charge", d3.forceManyBody().strength(fullParams.force.nodeStrength))
286
+ .force("collision", d3.forceCollide(fullParams.dot.radius).strength(1))
287
+ .force("center", d3.forceCenter(layout.width / 2, layout.height / 2))
288
+
289
+ }
290
+
291
+ function translateFn (d: any): string {
292
+ // console.log('translateFn', d)
293
+ return "translate(" + d.x + "," + d.y + ")";
294
+ }
295
+
296
+ function translateCenterFn (d: any): string {
297
+ // console.log('translateCenterFn', d)
298
+ const x = d.source.x + ((d.target.x - d.source.x) / 2) // 置中的話除2
299
+ const y = d.source.y + ((d.target.y - d.source.y) / 2) // 置中的話除2
300
+ return "translate(" + x + "," + y + ")";
301
+ }
302
+
303
+ function linkArcFn (d: RenderEdge): string {
304
+ // console.log('linkArcFn', d)
305
+
306
+ // var dx = d.target.x - d.source.x,
307
+ // dy = d.target.y - d.source.y
308
+ // dr讓方向線變成有弧度的
309
+ // dr = Math.sqrt(dx * dx + dy * dy);
310
+ // return "M" + d.source.x + "," + d.source.y + "A" + dr + "," + dr + " 0 0,1 " + d.target.x + "," + d.target.y;
311
+
312
+ // 直線
313
+ return "M" + d.source.x + "," + d.source.y + " L" + d.target.x + "," + d.target.y;
314
+
315
+
316
+ }
317
+
318
+
319
+
320
+ function renderArrowMarker (defsSelection: d3.Selection<SVGDefsElement, any, any, unknown>, fullParams: ForceDirectedParams) {
321
+ return defsSelection
322
+ .selectAll<SVGMarkerElement, any>(`marker.${defsArrowMarkerClassName}`)
323
+ .data([fullParams])
324
+ .join(
325
+ enter => {
326
+ const enterSelection = enter
327
+ .append("marker")
328
+ .classed(defsArrowMarkerClassName, true)
329
+ .attr('id', defsArrowMarkerId)
330
+ .attr("viewBox", d => `-${d.arrow.pointerWidth} -${d.arrow.pointerHeight / 2} ${d.arrow.pointerWidth} ${d.arrow.pointerHeight}`)
331
+ .attr("orient", "auto")
332
+ enterSelection.append("path")
333
+ .attr("d", d => `M${-d.arrow.pointerWidth},${-d.arrow.pointerHeight / 2}L0,0L${-d.arrow.pointerWidth},${d.arrow.pointerHeight / 2}`) // 箭頭的尖端為(0,0)
334
+ return enterSelection
335
+ },
336
+ update => {
337
+ return update
338
+ },
339
+ exit => {
340
+ return exit.remove()
341
+ }
342
+ )
343
+ .attr("markerWidth", d => d.arrow.pointerWidth)
344
+ .attr("markerHeight", d => d.arrow.pointerHeight)
345
+ /* refX:修正marker位置(計算出和circle半徑相等的寬度)
346
+ (1)circle半徑需加上 strokeWidth/2 是因為框線是以 circle 的邊緣往內及往外擴展,所以 stroke 多出來的寬度是一半而已
347
+ (2)circle半徑需除以 path 寬度是因為「marker 的位置會受到 path 的stroke-width影響」,所以要進行修正
348
+ (3)- 1 是要修正奇怪的誤差(不知原因)
349
+ */
350
+ .attr('refX', d => ((d.dot.radius + (fullParams.dot.strokeWidth / 2)) / d.arrow.strokeWidth) - 1)
351
+ .attr("refY", 0)
352
+
353
+ }
354
+
355
+ // function drag (): d3.DragBehavior<Element, unknown, unknown> {
356
+ // let originHighlightLockMode: boolean // 拖拽前的highlightLockMode
357
+
358
+ // return d3.drag()
359
+ // .on("start", (event: D3DragEvent) => {
360
+ // console.log('start', event.sourceEvent)
361
+ // // if (this.params.lockMode) {
362
+ // // return
363
+ // // }
364
+ // // if (!d3.event.active) {
365
+ // // this.forceRestart()
366
+ // // }
367
+ // // d.fx = d.x
368
+ // // d.fy = d.y
369
+
370
+ // // // 鎖定模式才不會在拖拽過程式觸發到其他事件造成衝突
371
+ // // originHighlightLockMode = this.highlightLockMode
372
+ // // this.highlightLockMode = true
373
+ // // this.noneStopMode = true
374
+ // // // 動畫會有點卡住所以乾脆拿掉
375
+ // // if(this.tooltip != null) {
376
+ // // this.tooltip.remove()
377
+ // // }
378
+ // })
379
+ // .on("drag", function (event: D3DragEvent) {
380
+ // console.log('drag', event)
381
+ // // if (this.params.lockMode) {
382
+ // // return
383
+ // // }
384
+ // // if (!d3.event.active) {
385
+ // // this.force.alphaTarget(0)
386
+ // // }
387
+ // // d.fx = d3.event.x
388
+ // // d.fy = d3.event.y
389
+ // // d3.select(this).attr({
390
+ // // 'cx': event.x,
391
+ // // 'cy': event.y,
392
+ // // })
393
+ // d3.select(this)
394
+ // .attr('fx', event.x)
395
+ // .attr('fy', event.y)
396
+ // })
397
+ // .on("end", (event: D3DragEvent) => {
398
+ // console.log('end', event)
399
+ // // if (this.params.lockMode) {
400
+ // // return
401
+ // // }
402
+ // // d.fx = null
403
+ // // d.fy = null
404
+
405
+ // // this.highlightLockMode = originHighlightLockMode // 還原拖拽前的highlightLockMode
406
+ // // this.noneStopMode = false
407
+ // // if (this.highlightLockMode) {
408
+ // // this.forceStop()
409
+ // // }
410
+ // })
411
+ // }
412
+
413
+ function drag (simulation: d3.Simulation<d3.SimulationNodeDatum, undefined>, dragStatus$: BehaviorSubject<DragStatus>) {
414
+ function dragstarted (event: D3DragEvent, node: RenderNode) {
415
+ if (!event.active) simulation.alphaTarget(0.3).restart()
416
+ event.subject.fx = event.x
417
+ event.subject.fy = event.y
418
+
419
+ dragStatus$.next('start')
420
+ }
421
+
422
+ function dragged (event: D3DragEvent, node: RenderNode) {
423
+ event.subject.fx = event.x
424
+ event.subject.fy = event.y
425
+
426
+ dragStatus$.next('drag')
427
+ }
428
+
429
+ function dragended (event: D3DragEvent, node: RenderNode) {
430
+ if (!event.active) simulation.alphaTarget(0);
431
+ event.subject.fx = null
432
+ event.subject.fy = null
433
+
434
+ dragStatus$.next('end')
435
+ }
436
+
437
+ return d3.drag()
438
+ .on("start", dragstarted)
439
+ .on("drag", dragged)
440
+ .on("end", dragended)
441
+ }
442
+
443
+ function renderNodeG ({ nodeListGSelection, nodes }: {
444
+ nodeListGSelection: d3.Selection<SVGGElement, any, any, unknown>
445
+ nodes: RenderNode[]
446
+ }) {
447
+ return nodeListGSelection.selectAll<SVGGElement, RenderNode>('g')
448
+ .data(nodes, d => d.id)
449
+ .join(
450
+ enter => {
451
+ const enterSelection = enter
452
+ .append('g')
453
+ .classed(nodeGClassName, true)
454
+ // .attr('cursor', 'pointer')
455
+ return enterSelection
456
+ },
457
+ update => {
458
+ return update
459
+ },
460
+ exit => {
461
+ return exit.remove()
462
+ }
463
+ )
464
+ }
465
+
466
+ function renderNodeCircle ({ nodeGSelection, fullParams, fullChartParams }: {
467
+ nodeGSelection: d3.Selection<SVGGElement, RenderNode, any, unknown>
468
+ fullParams: ForceDirectedParams
469
+ fullChartParams: ChartParams
470
+ }) {
471
+ nodeGSelection.each((data,i,g) => {
472
+ const gSelection = d3.select(g[i])
473
+ gSelection.selectAll<SVGCircleElement, ComputedEdge>('circle')
474
+ .data([data])
475
+ .join(
476
+ enter => {
477
+ const enterSelection = enter
478
+ .append('circle')
479
+ .classed(nodeCircleClassName, true)
480
+ .attr('cursor', 'pointer')
481
+ return enterSelection
482
+ },
483
+ update => {
484
+ return update
485
+ },
486
+ exit => {
487
+ return exit.remove()
488
+ }
489
+ )
490
+ .attr('r', fullParams.dot.radius)
491
+ .attr('fill', d => getDatumColor({ datum: d, colorType: fullParams.dot.fillColorType, fullChartParams }))
492
+ .attr('stroke', d => getDatumColor({ datum: d, colorType: fullParams.dot.strokeColorType, fullChartParams }))
493
+ .attr('stroke-width', fullParams.dot.strokeWidth)
494
+ .attr('style', d => fullParams.dot.styleFn(d))
495
+ })
496
+
497
+ return nodeGSelection.select<SVGCircleElement>(`circle.${nodeCircleClassName}`)
498
+ }
499
+
500
+ function renderNodeLabelG ({ nodeGSelection, fullParams }: {
501
+ nodeGSelection: d3.Selection<SVGGElement, any, any, unknown>
502
+ fullParams: ForceDirectedParams
503
+ }) {
504
+ nodeGSelection.each((data,i,g) => {
505
+ const gSelection = d3.select(g[i])
506
+ gSelection.selectAll<SVGGElement, RenderNode>('g')
507
+ .data([data])
508
+ .join(
509
+ enter => {
510
+ const enterSelection = enter
511
+ .append('g')
512
+ .classed(nodeLabelGClassName, true)
513
+ // .attr('cursor', 'pointer')
514
+ return enterSelection
515
+ },
516
+ update => {
517
+ return update
518
+ },
519
+ exit => {
520
+ return exit.remove()
521
+ }
522
+ )
523
+ .attr('transform', `translate(0, ${- fullParams.dot.radius - 10})`)
524
+ })
525
+
526
+ return nodeGSelection.select<SVGTextElement>(`g.${nodeLabelGClassName}`)
527
+ }
528
+
529
+ function renderNodeLabel ({ nodeLabelGSelection, fullParams, fullChartParams }: {
530
+ nodeLabelGSelection: d3.Selection<SVGGElement, RenderNode, any, unknown>
531
+ fullParams: ForceDirectedParams
532
+ fullChartParams: ChartParams
533
+ }) {
534
+ nodeLabelGSelection.each((data,i,g) => {
535
+ const gSelection = d3.select(g[i])
536
+ gSelection.selectAll<SVGTextElement, RenderNode>('text')
537
+ .data([data], d => d.id)
538
+ .join(
539
+ enter => {
540
+ const enterSelection = enter
541
+ .append('text')
542
+ .classed(nodeLabelClassName, true)
543
+ // .attr('cursor', 'pointer')
544
+ .attr('text-anchor', 'middle')
545
+ .attr('pointer-events', 'none')
546
+ return enterSelection
547
+ },
548
+ update => {
549
+ return update
550
+ },
551
+ exit => {
552
+ return exit.remove()
553
+ }
554
+ )
555
+ .text(d => d.label)
556
+ .attr('fill', d => getDatumColor({ datum: d, colorType: fullParams.dotLabel.colorType, fullChartParams }))
557
+ .attr('font-size', fullChartParams.styles.textSize)
558
+ .attr('style', d => fullParams.dotLabel.styleFn(d))
559
+ })
560
+
561
+ return nodeLabelGSelection.select<SVGTextElement>(`text.${nodeLabelClassName}`)
562
+ }
563
+
564
+ function renderEdgeG ({ edgeListGSelection, edges }: {
565
+ edgeListGSelection: d3.Selection<SVGGElement, any, any, unknown>
566
+ edges: RenderEdge[]
567
+ }) {
568
+ return edgeListGSelection.selectAll<SVGGElement, RenderEdge>('g')
569
+ .data(edges, d => d.id)
570
+ .join(
571
+ enter => {
572
+ const enterSelection = enter
573
+ .append('g')
574
+ .classed(edgeGClassName, true)
575
+ // .attr('cursor', 'pointer')
576
+ return enterSelection
577
+ },
578
+ update => {
579
+ return update
580
+ },
581
+ exit => {
582
+ return exit.remove()
583
+ }
584
+ )
585
+ }
586
+
587
+ function renderEdgeArrowPath ({ edgeGSelection, fullParams, fullChartParams }: {
588
+ edgeGSelection: d3.Selection<SVGGElement, RenderEdge, any, unknown>
589
+ fullParams: ForceDirectedParams
590
+ fullChartParams: ChartParams
591
+ }) {
592
+ edgeGSelection.each((data,i,g) => {
593
+ const gSelection = d3.select(g[i])
594
+ gSelection.selectAll<SVGPathElement, ComputedEdge>('path')
595
+ .data([data])
596
+ .join(
597
+ enter => {
598
+ return enter
599
+ .append('path')
600
+ .classed(edgeArrowPathClassName, true)
601
+ .attr('marker-end', `url(#${defsArrowMarkerId})`)
602
+ },
603
+ update => {
604
+ return update
605
+ },
606
+ exit => {
607
+ return exit.remove()
608
+ }
609
+ )
610
+ .attr('stroke', d => getDatumColor({ datum: d.data, colorType: fullParams.arrow.colorType, fullChartParams }))
611
+ .attr('stroke-width', fullParams.arrow.strokeWidth)
612
+ .attr('style', d => fullParams.arrow.styleFn(d))
613
+ })
614
+
615
+ return edgeGSelection.select<SVGPathElement>(`path.${edgeArrowPathClassName}`)
616
+ }
617
+
618
+ function renderEdgeLabelG ({ edgeGSelection }: {
619
+ edgeGSelection: d3.Selection<SVGGElement, any, any, unknown>
620
+ }) {
621
+ edgeGSelection.each((data,i,g) => {
622
+ const gSelection = d3.select(g[i])
623
+ gSelection.selectAll<SVGGElement, RenderEdge>('g')
624
+ .data([data])
625
+ .join(
626
+ enter => {
627
+ const enterSelection = enter
628
+ .append('g')
629
+ .classed(edgeLabelGClassName, true)
630
+ // .attr('cursor', 'pointer')
631
+ return enterSelection
632
+ },
633
+ update => {
634
+ return update
635
+ },
636
+ exit => {
637
+ return exit.remove()
638
+ }
639
+ )
640
+ })
641
+
642
+ return edgeGSelection.select<SVGTextElement>(`g.${edgeLabelGClassName}`)
643
+ }
644
+
645
+ function renderEdgeLabel ({ edgeLabelGSelection, fullParams, fullChartParams }: {
646
+ edgeLabelGSelection: d3.Selection<SVGGElement, RenderEdge, any, unknown>
647
+ fullParams: ForceDirectedParams
648
+ fullChartParams: ChartParams
649
+ }) {
650
+ edgeLabelGSelection.each((data,i,g) => {
651
+ const gSelection = d3.select(g[i])
652
+ gSelection.selectAll<SVGTextElement, RenderEdge>('text')
653
+ .data([data], d => d.id)
654
+ .join(
655
+ enter => {
656
+ const enterSelection = enter
657
+ .append('text')
658
+ .classed(edgeLabelClassName, true)
659
+ // .attr('cursor', 'pointer')
660
+ .attr('text-anchor', 'middle')
661
+ .attr('pointer-events', 'none')
662
+ return enterSelection
663
+ },
664
+ update => {
665
+ return update
666
+ },
667
+ exit => {
668
+ return exit.remove()
669
+ }
670
+ )
671
+ .text(d => d.label)
672
+ .attr('fill', d => getDatumColor({ datum: d, colorType: fullParams.arrowLabel.colorType, fullChartParams }))
673
+ .attr('font-size', fullChartParams.styles.textSize)
674
+ .attr('style', d => fullParams.arrowLabel.styleFn(d))
675
+ })
676
+
677
+ return edgeLabelGSelection.select<SVGTextElement>(`text.${edgeLabelClassName}`)
678
+ }
679
+
680
+
681
+ // function renderBubbles ({ selection, bubblesData, fullParams, sumSeries }: {
682
+ // selection: d3.Selection<SVGGElement, any, any, any>
683
+ // bubblesData: BubblesDatum[]
684
+ // fullParams: BubblesParams
685
+ // sumSeries: boolean
686
+ // }) {
687
+ // const bubblesSelection = selection.selectAll<SVGGElement, BubblesDatum>("g")
688
+ // .data(bubblesData, (d) => d.id)
689
+ // .join(
690
+ // enter => {
691
+ // const enterSelection = enter
692
+ // .append('g')
693
+ // .attr('cursor', 'pointer')
694
+ // .attr('font-size', 12)
695
+ // .style('fill', '#ffffff')
696
+ // .attr("text-anchor", "middle")
697
+
698
+ // enterSelection
699
+ // .append("circle")
700
+ // .attr("class", "node")
701
+ // .attr("cx", 0)
702
+ // .attr("cy", 0)
703
+ // // .attr("r", 1e-6)
704
+ // .attr('fill', (d) => d.color)
705
+ // // .transition()
706
+ // // .duration(500)
707
+
708
+ // enterSelection
709
+ // .append('text')
710
+ // .style('opacity', 0.8)
711
+ // .attr('pointer-events', 'none')
712
+
713
+ // return enterSelection
714
+ // },
715
+ // update => {
716
+ // return update
717
+ // },
718
+ // exit => {
719
+ // return exit
720
+ // .remove()
721
+ // }
722
+ // )
723
+ // .attr("transform", (d) => {
724
+ // return `translate(${d.x},${d.y})`
725
+ // })
726
+
727
+ // // 泡泡文字要使用的的資料欄位
728
+ // const textDataColumn = sumSeries ? 'seriesLabel' : 'label'// 如果有合併series則使用seriesLabel
729
+
730
+ // bubblesSelection.select('circle')
731
+ // .transition()
732
+ // .duration(200)
733
+ // .attr("r", (d) => d.r)
734
+ // .attr('fill', (d) => d.color)
735
+ // bubblesSelection
736
+ // .each((d,i,g) => {
737
+ // const gSelection = d3.select(g[i])
738
+ // let breakAll = true
739
+ // if (d[textDataColumn].length <= fullParams.label.maxLineLength) {
740
+ // breakAll = false
741
+ // }
742
+ // gSelection.call(renderCircleText, {
743
+ // text: d[textDataColumn],
744
+ // radius: d.r * fullParams.label.fillRate,
745
+ // lineHeight: fullParams.label.lineHeight,
746
+ // isBreakAll: breakAll
747
+ // })
748
+
749
+ // })
750
+
751
+ // return bubblesSelection
752
+ // }
753
+
754
+ // function setHighlightData ({ data, highlightRIncrease, highlightIds }: {
755
+ // data: BubblesDatum[]
756
+ // // fullParams: BubblesParams
757
+ // highlightRIncrease: number
758
+ // highlightIds: string[]
759
+ // }) {
760
+ // if (highlightRIncrease == 0) {
761
+ // return
762
+ // }
763
+ // if (!highlightIds.length) {
764
+ // data.forEach(d => d.r = d._originR)
765
+ // return
766
+ // }
767
+ // data.forEach(d => {
768
+ // if (highlightIds.includes(d.id)) {
769
+ // d.r = d._originR + highlightRIncrease
770
+ // } else {
771
+ // d.r = d._originR
772
+ // }
773
+ // })
774
+ // }
775
+
776
+
777
+ // function groupBubbles ({ fullParams, SeriesContainerPositionMap }: {
778
+ // fullParams: BubblesParams
779
+ // // graphicWidth: number
780
+ // // graphicHeight: number
781
+ // SeriesContainerPositionMap: Map<string, ContainerPosition>
782
+ // }) {
783
+ // // console.log('groupBubbles')
784
+ // force!
785
+ // // .force('x', d3.forceX().strength(fullParams.force.strength).x(graphicWidth / 2))
786
+ // // .force('y', d3.forceY().strength(fullParams.force.strength).y(graphicHeight / 2))
787
+ // .force('x', d3.forceX().strength(fullParams.force.strength).x((data: BubblesSimulationDatum) => {
788
+ // return SeriesContainerPositionMap.get(data.seriesLabel)!.centerX
789
+ // }))
790
+ // .force('y', d3.forceY().strength(fullParams.force.strength).y((data: BubblesSimulationDatum) => {
791
+ // return SeriesContainerPositionMap.get(data.seriesLabel)!.centerY
792
+ // }))
793
+
794
+ // force!.alpha(1).restart()
795
+ // }
796
+
797
+ function highlightNodes ({ nodeGSelection, edgeGSelection, highlightIds, fullChartParams }: {
798
+ nodeGSelection: d3.Selection<SVGGElement, RenderNode, SVGGElement, any>
799
+ edgeGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any>
800
+ fullChartParams: ChartParams
801
+ highlightIds: string[]
802
+ }) {
803
+ nodeGSelection.interrupt('highlight')
804
+ edgeGSelection.interrupt('highlight')
805
+ // console.log(highlightIds)
806
+ if (!highlightIds.length) {
807
+ nodeGSelection
808
+ .transition('highlight')
809
+ .style('opacity', 1)
810
+ edgeGSelection
811
+ .transition('highlight')
812
+ .style('opacity', 1)
813
+ return
814
+ }
815
+
816
+ edgeGSelection
817
+ .style('opacity', fullChartParams.styles.unhighlightedOpacity)
818
+
819
+ nodeGSelection.each((d, i, n) => {
820
+ const segment = d3.select(n[i])
821
+
822
+ if (highlightIds.includes(d.id)) {
823
+ segment
824
+ .style('opacity', 1)
825
+ .transition('highlight')
826
+ .ease(d3.easeElastic)
827
+ .duration(500)
828
+ } else {
829
+ // 取消
830
+ segment
831
+ .style('opacity', fullChartParams.styles.unhighlightedOpacity)
832
+ }
833
+ })
834
+ }
835
+
836
+ // 暫不處理edge的highlight
837
+ // function highlightEdges ({ edgeGSelection, highlightIds, fullChartParams }: {
838
+ // edgeGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any>
839
+ // fullChartParams: ChartParams
840
+ // highlightIds: string[]
841
+ // }) {
842
+ // edgeGSelection.interrupt('highlight')
843
+
844
+ // if (!highlightIds.length) {
845
+ // edgeGSelection
846
+ // .transition('highlight')
847
+ // .style('opacity', 1)
848
+ // return
849
+ // }
850
+
851
+ // edgeGSelection.each((d, i, n) => {
852
+ // const segment = d3.select(n[i])
853
+
854
+ // if (highlightIds.includes(d.id)) {
855
+ // segment
856
+ // .style('opacity', 1)
857
+ // .transition('highlight')
858
+ // .ease(d3.easeElastic)
859
+ // .duration(500)
860
+ // } else {
861
+ // // 取消放大
862
+ // segment
863
+ // .style('opacity', fullChartParams.styles.unhighlightedOpacity)
864
+ // }
865
+ // })
866
+ // }
867
+
868
+ export const ForceDirected = defineRelationshipPlugin(pluginConfig)(({ selection, rootSelection, name, observer, subject }) => {
869
+
870
+ const destroy$ = new Subject()
871
+
872
+ const gSelection = selection.append('g').classed(gSelectionClassName, true)
873
+ const defsSelection = gSelection.append('defs')
874
+ const edgeListGSelection = gSelection.append('g').classed(edgeListGClassName, true)
875
+ const nodeListGSelection = gSelection.append('g').classed(nodeListGClassName, true)
876
+
877
+ let nodeGSelection: d3.Selection<SVGGElement, RenderNode, SVGGElement, any> | undefined
878
+ let nodeCircleSelection: d3.Selection<SVGCircleElement, RenderNode, SVGGElement, any> | undefined
879
+ let nodeLabelGSelection: d3.Selection<SVGGElement, RenderNode, SVGGElement, any> | undefined
880
+ let nodeLabelSelection: d3.Selection<SVGTextElement, RenderNode, SVGGElement, any> | undefined
881
+ let edgeGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any> | undefined
882
+ let edgeArrowSelection: d3.Selection<SVGPathElement, RenderEdge, SVGGElement, any> | undefined
883
+ let edgeLabelGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any> | undefined
884
+ let edgeLabelSelection: d3.Selection<SVGTextElement, RenderEdge, SVGGElement, any> | undefined
885
+
886
+ const dragStatus$ = new BehaviorSubject<DragStatus>('end') // start, drag, end
887
+ const mouseEvent$ = new Subject<EventRelationship>()
888
+
889
+ // <marker> marker selection
890
+ observer.fullParams$.pipe(
891
+ takeUntil(destroy$),
892
+ map(fullParams => {
893
+ return renderArrowMarker(defsSelection, fullParams)
894
+ })
895
+ ).subscribe()
896
+
897
+ // init zoom
898
+ const d3Zoom$ = observer.fullParams$.pipe(
899
+ takeUntil(destroy$),
900
+ // map(d => d.scaleExtent),
901
+ // distinctUntilChanged((a, b) => String(a) === String(b)),
902
+ // first(),
903
+ map(data => {
904
+ let d3Zoom = data.zoomable
905
+ ? d3.zoom().on('zoom', (event) => {
906
+ // console.log(event)
907
+ // this.svgGroup.attr('transform', `translate(
908
+ // ${event.transform.x + (this.zoom.xOffset * event.transform.k)},
909
+ // ${event.transform.y + (this.zoom.yOffset * event.transform.k)}
910
+ // ) scale(
911
+ // ${event.transform.k}
912
+ // )`)
913
+ gSelection.attr('transform', `translate(
914
+ ${event.transform.x},
915
+ ${event.transform.y}
916
+ ) scale(
917
+ ${event.transform.k}
918
+ )`)
919
+
920
+ if (data.dotLabel.sizeFixed && nodeLabelSelection) {
921
+ // 反向 scale 抵消掉放大縮小
922
+ nodeLabelSelection.attr('transform', `scale(${1 / event.transform.k})`)
923
+ }
924
+ if (data.arrowLabel.sizeFixed && edgeLabelSelection) {
925
+ // 反向 scale 抵消掉放大縮小
926
+ edgeLabelSelection.attr('transform', `scale(${1 / event.transform.k})`)
927
+ }
928
+ })
929
+ : d3.zoom().on('zoom', null)
930
+ if (data.scaleExtent) {
931
+ d3Zoom.scaleExtent([data.scaleExtent.min, data.scaleExtent.max])
932
+ }
933
+ rootSelection.call(d3Zoom)
934
+
935
+ return d3Zoom
936
+ }),
937
+ // shareReplay(1)
938
+ )
939
+
940
+ // zoom transform
941
+ combineLatest({
942
+ d3Zoom: d3Zoom$,
943
+ transform: observer.fullParams$.pipe(
944
+ takeUntil(destroy$),
945
+ map(d => d.transform),
946
+ )
947
+ }).pipe(
948
+ takeUntil(destroy$),
949
+ switchMap(async d => d)
950
+ ).subscribe(data => {
951
+ // console.log('call')
952
+ selection.call(
953
+ data.d3Zoom.transform, d3.zoomIdentity
954
+ .translate(data.transform.x, data.transform.y)
955
+ .scale(data.transform.k)
956
+ )
957
+ })
958
+
959
+
960
+ const simulation$: Observable<d3.Simulation<d3.SimulationNodeDatum, undefined>> = combineLatest({
961
+ layout: observer.layout$.pipe(
962
+ first() // 只使用第一次的尺寸(置中)
963
+ ),
964
+ fullParams: observer.fullParams$
965
+ }).pipe(
966
+ takeUntil(destroy$),
967
+ switchMap(async d => d),
968
+ map(data => createSimulation(data.layout, data.fullParams)),
969
+ shareReplay(1)
970
+ )
971
+
972
+ const renderData$: Observable<RenderData> = observer.visibleComputedData$.pipe(
973
+ takeUntil(destroy$),
974
+ map(data => {
975
+ return {
976
+ nodes: data.nodes,
977
+ edges: data.edges.map(_d => {
978
+ let d: RenderEdge = _d as RenderEdge
979
+ d.source = _d.startNode // reference
980
+ d.target = _d.endNode
981
+ return d
982
+ })
983
+ }
984
+ }),
985
+ shareReplay(1)
986
+ )
987
+
988
+ combineLatest({
989
+ renderData: renderData$,
990
+ computedData: observer.computedData$,
991
+ CategoryNodeMap: observer.CategoryNodeMap$,
992
+ simulation: simulation$,
993
+ fullParams: observer.fullParams$,
994
+ fullChartParams: observer.fullChartParams$
995
+ }).pipe(
996
+ takeUntil(destroy$),
997
+ switchMap(async d => d),
998
+ ).subscribe(data => {
999
+
1000
+ nodeGSelection = renderNodeG({
1001
+ nodeListGSelection: nodeListGSelection,
1002
+ nodes: data.renderData.nodes,
1003
+ })
1004
+
1005
+ nodeCircleSelection = renderNodeCircle({
1006
+ nodeGSelection: nodeGSelection,
1007
+ fullParams: data.fullParams,
1008
+ fullChartParams: data.fullChartParams
1009
+ })
1010
+ nodeGSelection.call(drag(data.simulation, dragStatus$))
1011
+
1012
+ nodeLabelGSelection = renderNodeLabelG({
1013
+ nodeGSelection: nodeGSelection,
1014
+ fullParams: data.fullParams
1015
+ })
1016
+
1017
+ nodeLabelSelection = renderNodeLabel({
1018
+ nodeLabelGSelection: nodeLabelGSelection,
1019
+ fullParams: data.fullParams,
1020
+ fullChartParams: data.fullChartParams
1021
+ })
1022
+
1023
+ edgeGSelection = renderEdgeG({
1024
+ edgeListGSelection: edgeListGSelection,
1025
+ edges: data.renderData.edges
1026
+ })
1027
+
1028
+ edgeArrowSelection = renderEdgeArrowPath({
1029
+ edgeGSelection: edgeGSelection,
1030
+ fullParams: data.fullParams,
1031
+ fullChartParams: data.fullChartParams
1032
+ })
1033
+
1034
+ edgeLabelGSelection = renderEdgeLabelG({
1035
+ edgeGSelection: edgeGSelection,
1036
+ })
1037
+
1038
+ edgeLabelSelection = renderEdgeLabel({
1039
+ edgeLabelGSelection: edgeLabelGSelection,
1040
+ fullParams: data.fullParams,
1041
+ fullChartParams: data.fullChartParams
1042
+ })
1043
+
1044
+ data.simulation.nodes(data.renderData.nodes)
1045
+ .on('tick', () => {
1046
+ edgeArrowSelection.attr('d', linkArcFn)
1047
+ nodeGSelection.attr('transform', translateFn)
1048
+ // nodeLabelGSelection.attr('transform', d => translateFn({
1049
+ // x: d.x,
1050
+ // y: d.y - data.fullParams.dot.radius - 10
1051
+ // }))
1052
+ edgeLabelGSelection.attr('transform', d => translateCenterFn(d))
1053
+ })
1054
+ ;(data.simulation.force("link") as any).links(data.renderData.edges)
1055
+
1056
+ data.simulation.alpha(0.3).restart()
1057
+
1058
+ nodeCircleSelection
1059
+ .on('mouseover', (event, datum) => {
1060
+ event.stopPropagation()
1061
+
1062
+ mouseEvent$.next({
1063
+ type: 'relationship',
1064
+ eventName: 'mouseover',
1065
+ pluginName,
1066
+ highlightTarget: data.fullChartParams.highlightTarget,
1067
+ datum: datum,
1068
+ category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1069
+ categoryIndex: datum.categoryIndex,
1070
+ categoryLabel: datum.categoryLabel,
1071
+ event,
1072
+ data: data.computedData
1073
+ })
1074
+ })
1075
+ .on('mousemove', (event, datum) => {
1076
+ event.stopPropagation()
1077
+
1078
+ mouseEvent$.next({
1079
+ type: 'relationship',
1080
+ eventName: 'mousemove',
1081
+ pluginName,
1082
+ highlightTarget: data.fullChartParams.highlightTarget,
1083
+ datum: datum,
1084
+ category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1085
+ categoryIndex: datum.categoryIndex,
1086
+ categoryLabel: datum.categoryLabel,
1087
+ event,
1088
+ data: data.computedData
1089
+ })
1090
+ })
1091
+ .on('mouseout', (event, datum) => {
1092
+ event.stopPropagation()
1093
+
1094
+ mouseEvent$.next({
1095
+ type: 'relationship',
1096
+ eventName: 'mouseout',
1097
+ pluginName,
1098
+ highlightTarget: data.fullChartParams.highlightTarget,
1099
+ datum: datum,
1100
+ category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1101
+ categoryIndex: datum.categoryIndex,
1102
+ categoryLabel: datum.categoryLabel,
1103
+ event,
1104
+ data: data.computedData
1105
+ })
1106
+ })
1107
+ .on('click', (event, datum) => {
1108
+ event.stopPropagation()
1109
+
1110
+ mouseEvent$.next({
1111
+ type: 'relationship',
1112
+ eventName: 'click',
1113
+ pluginName,
1114
+ highlightTarget: data.fullChartParams.highlightTarget,
1115
+ datum: datum,
1116
+ category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1117
+ categoryIndex: datum.categoryIndex,
1118
+ categoryLabel: datum.categoryLabel,
1119
+ event,
1120
+ data: data.computedData
1121
+ })
1122
+ })
1123
+ })
1124
+
1125
+ dragStatus$.pipe(
1126
+ distinctUntilChanged((a, b) => a === b),
1127
+ // 只有沒有托曳時才執行
1128
+ switchMap(d => iif(() => d === 'end', mouseEvent$, EMPTY))
1129
+ ).subscribe(data => {
1130
+ subject.event$.next(data)
1131
+ })
1132
+
1133
+ combineLatest({
1134
+ renderData: renderData$,
1135
+ highlightNodes: observer.relationshipHighlightNodes$.pipe(
1136
+ map(data => data.map(d => d.id))
1137
+ ),
1138
+ highlightEdges: observer.relationshipHighlightEdges$.pipe(
1139
+ map(data => data.map(d => d.id))
1140
+ ),
1141
+ fullChartParams: observer.fullChartParams$,
1142
+ fullParams: observer.fullParams$,
1143
+ }).pipe(
1144
+ takeUntil(destroy$),
1145
+ switchMap(async d => d)
1146
+ ).subscribe(data => {
1147
+ if (!nodeGSelection || !edgeGSelection) {
1148
+ return
1149
+ }
1150
+
1151
+ highlightNodes({
1152
+ nodeGSelection,
1153
+ edgeGSelection,
1154
+ highlightIds: data.highlightNodes,
1155
+ fullChartParams: data.fullChartParams
1156
+ })
1157
+ // highlightEdges({
1158
+ // edgeGSelection,
1159
+ // highlightIds: data.highlightEdges,
1160
+ // fullChartParams: data.fullChartParams
1161
+ // })
1162
+ })
1163
+
1164
+
1165
+
1166
+ return () => {
1167
+ destroy$.next(undefined)
1168
+ }
1169
1169
  })