@orbcharts/plugins-basic 3.0.0-beta.13 → 3.0.0-beta.14

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