@orbcharts/plugins-basic 3.0.0-beta.10 → 3.0.0-beta.11

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 (99) hide show
  1. package/LICENSE +200 -200
  2. package/dist/orbcharts-plugins-basic.es.js +177 -177
  3. package/dist/orbcharts-plugins-basic.umd.js +55 -55
  4. package/lib/core-types.ts +7 -7
  5. package/lib/core.ts +6 -6
  6. package/lib/plugins-basic-types.ts +6 -6
  7. package/package.json +44 -44
  8. package/src/base/BaseBars.ts +765 -765
  9. package/src/base/BaseBarsTriangle.ts +676 -676
  10. package/src/base/BaseDots.ts +464 -464
  11. package/src/base/BaseGroupAxis.ts +679 -679
  12. package/src/base/BaseLegend.ts +684 -684
  13. package/src/base/BaseLineAreas.ts +629 -629
  14. package/src/base/BaseLines.ts +706 -706
  15. package/src/base/BaseStackedBar.ts +782 -782
  16. package/src/base/BaseTooltip.ts +385 -385
  17. package/src/base/BaseValueAxis.ts +583 -583
  18. package/src/base/types.ts +2 -2
  19. package/src/const.ts +30 -30
  20. package/src/grid/defaults.ts +246 -246
  21. package/src/grid/gridObservables.ts +554 -554
  22. package/src/grid/index.ts +16 -16
  23. package/src/grid/plugins/Bars.ts +69 -69
  24. package/src/grid/plugins/BarsPN.ts +66 -66
  25. package/src/grid/plugins/BarsTriangle.ts +73 -73
  26. package/src/grid/plugins/Dots.ts +68 -68
  27. package/src/grid/plugins/GridLegend.ts +107 -107
  28. package/src/grid/plugins/GridTooltip.ts +66 -66
  29. package/src/grid/plugins/GridZoom.ts +218 -218
  30. package/src/grid/plugins/GroupAux.ts +1103 -1103
  31. package/src/grid/plugins/GroupAxis.ts +97 -97
  32. package/src/grid/plugins/LineAreas.ts +65 -65
  33. package/src/grid/plugins/Lines.ts +59 -59
  34. package/src/grid/plugins/StackedBar.ts +64 -64
  35. package/src/grid/plugins/StackedValueAxis.ts +96 -96
  36. package/src/grid/plugins/ValueAxis.ts +94 -94
  37. package/src/index.ts +6 -6
  38. package/src/multiGrid/defaults.ts +224 -224
  39. package/src/multiGrid/index.ts +14 -14
  40. package/src/multiGrid/multiGridObservables.ts +49 -49
  41. package/src/multiGrid/plugins/MultiBars.ts +108 -108
  42. package/src/multiGrid/plugins/MultiBarsTriangle.ts +114 -114
  43. package/src/multiGrid/plugins/MultiDots.ts +102 -102
  44. package/src/multiGrid/plugins/MultiGridLegend.ts +159 -159
  45. package/src/multiGrid/plugins/MultiGridTooltip.ts +66 -66
  46. package/src/multiGrid/plugins/MultiGroupAxis.ts +137 -137
  47. package/src/multiGrid/plugins/MultiLineAreas.ts +107 -107
  48. package/src/multiGrid/plugins/MultiLines.ts +101 -101
  49. package/src/multiGrid/plugins/MultiStackedBar.ts +106 -106
  50. package/src/multiGrid/plugins/MultiStackedValueAxis.ts +134 -134
  51. package/src/multiGrid/plugins/MultiValueAxis.ts +134 -134
  52. package/src/multiGrid/plugins/OverlappingStackedValueAxes.ts +299 -299
  53. package/src/multiGrid/plugins/OverlappingValueAxes.ts +300 -300
  54. package/src/multiValue/defaults.ts +166 -166
  55. package/src/multiValue/index.ts +8 -8
  56. package/src/multiValue/multiValueObservables.ts +297 -297
  57. package/src/multiValue/plugins/MultiValueLegend.ts +107 -107
  58. package/src/multiValue/plugins/MultiValueTooltip.ts +66 -66
  59. package/src/multiValue/plugins/Scatter.ts +426 -426
  60. package/src/multiValue/plugins/ScatterBubbles.ts +554 -554
  61. package/src/multiValue/plugins/XYAux.ts +681 -681
  62. package/src/multiValue/plugins/XYAxes.ts +684 -684
  63. package/src/multiValue/plugins/XYZoom.ts +299 -299
  64. package/src/noneData/defaults.ts +102 -102
  65. package/src/noneData/index.ts +3 -3
  66. package/src/noneData/plugins/Container.ts +27 -27
  67. package/src/noneData/plugins/Tooltip.ts +373 -373
  68. package/src/relationship/defaults.ts +196 -195
  69. package/src/relationship/index.ts +5 -5
  70. package/src/relationship/plugins/ForceDirected.ts +1168 -1166
  71. package/src/relationship/plugins/ForceDirectedBubbles.ts +1393 -1390
  72. package/src/relationship/plugins/RelationshipLegend.ts +100 -100
  73. package/src/relationship/plugins/RelationshipTooltip.ts +66 -66
  74. package/src/relationship/relationshipObservables.ts +49 -49
  75. package/src/series/defaults.ts +207 -206
  76. package/src/series/index.ts +9 -9
  77. package/src/series/plugins/Bubbles.ts +603 -606
  78. package/src/series/plugins/Pie.ts +623 -623
  79. package/src/series/plugins/PieEventTexts.ts +284 -284
  80. package/src/series/plugins/PieLabels.ts +640 -640
  81. package/src/series/plugins/Rose.ts +516 -516
  82. package/src/series/plugins/RoseLabels.ts +600 -600
  83. package/src/series/plugins/SeriesLegend.ts +107 -107
  84. package/src/series/plugins/SeriesTooltip.ts +66 -66
  85. package/src/series/seriesObservables.ts +145 -145
  86. package/src/series/seriesUtils.ts +51 -51
  87. package/src/tree/defaults.ts +78 -78
  88. package/src/tree/index.ts +4 -4
  89. package/src/tree/plugins/TreeLegend.ts +100 -100
  90. package/src/tree/plugins/TreeMap.ts +333 -333
  91. package/src/tree/plugins/TreeTooltip.ts +66 -66
  92. package/src/utils/commonUtils.ts +21 -21
  93. package/src/utils/d3Graphics.ts +174 -174
  94. package/src/utils/d3Utils.ts +74 -74
  95. package/src/utils/observables.ts +14 -14
  96. package/src/utils/orbchartsUtils.ts +115 -115
  97. package/tsconfig.base.json +13 -13
  98. package/tsconfig.json +2 -2
  99. package/vite.config.js +22 -22
@@ -1,1391 +1,1394 @@
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
- lineLengthMin: {
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) {
454
- if (!event.active) simulation.alphaTarget(0.3).restart();
455
- event.subject.fx = event.subject.x;
456
- event.subject.fy = event.subject.y;
457
-
458
- dragStatus$.next('start')
459
- }
460
-
461
- function dragged (event: D3DragEvent) {
462
- event.subject.fx = event.x;
463
- event.subject.fy = event.y;
464
-
465
- dragStatus$.next('drag')
466
- }
467
-
468
- function dragended (event: D3DragEvent) {
469
- if (!event.active) simulation.alphaTarget(0);
470
- event.subject.fx = null;
471
- event.subject.fy = null;
472
-
473
- dragStatus$.next('end')
474
- }
475
-
476
- return d3.drag()
477
- .on("start", dragstarted)
478
- .on("drag", dragged)
479
- .on("end", dragended);
480
- }
481
-
482
- function renderNodeG ({ nodeListGSelection, nodes }: {
483
- nodeListGSelection: d3.Selection<SVGGElement, any, any, unknown>
484
- nodes: RenderNode[]
485
- }) {
486
- return nodeListGSelection.selectAll<SVGGElement, RenderNode>('g')
487
- .data(nodes, d => d.id)
488
- .join(
489
- enter => {
490
- const enterSelection = enter
491
- .append('g')
492
- .classed(nodeGClassName, true)
493
- // .attr('cursor', 'pointer')
494
- return enterSelection
495
- },
496
- update => {
497
- return update
498
- },
499
- exit => {
500
- return exit.remove()
501
- }
502
- )
503
- }
504
-
505
- function renderNodeCircle ({ nodeGSelection, fullParams, fullChartParams }: {
506
- nodeGSelection: d3.Selection<SVGGElement, RenderNode, any, unknown>
507
- fullParams: ForceDirectedBubblesParams
508
- fullChartParams: ChartParams
509
- }) {
510
- nodeGSelection
511
- .each((data,i,g) => {
512
- const gSelection = d3.select(g[i])
513
- gSelection.selectAll<SVGCircleElement, ComputedEdge>('circle')
514
- .data([data])
515
- .join(
516
- enter => {
517
- const enterSelection = enter
518
- .append('circle')
519
- .classed(nodeCircleClassName, true)
520
- .attr('cursor', 'pointer')
521
- return enterSelection
522
- },
523
- update => {
524
- return update
525
- },
526
- exit => {
527
- return exit.remove()
528
- }
529
- )
530
- .attr('r', d => d.r)
531
- .attr('fill', d => getDatumColor({ datum: d, colorType: fullParams.bubble.fillColorType, fullChartParams }))
532
- .attr('stroke', d => getDatumColor({ datum: d, colorType: fullParams.bubble.strokeColorType, fullChartParams }))
533
- .attr('stroke-width', fullParams.bubble.strokeWidth)
534
- .attr('style', d => fullParams.bubble.styleFn(d))
535
-
536
- })
537
- .attr("text-anchor", "middle")
538
- .attr('font-size', baseLineHeight)
539
- .each((d,i,g) => {
540
- const gSelection = d3.select(g[i])
541
- let breakAll = true
542
- if (d.label.length <= fullParams.bubbleLabel.lineLengthMin) {
543
- breakAll = false
544
- }
545
- gSelection.call(renderCircleText, {
546
- text: d.label,
547
- radius: d.r * fullParams.bubbleLabel.fillRate,
548
- lineHeight: baseLineHeight * fullParams.bubbleLabel.lineHeight,
549
- isBreakAll: breakAll
550
- })
551
-
552
- })
553
-
554
- nodeGSelection.select('text')
555
- .attr('pointer-events', 'none')
556
- .attr('style', d => fullParams.bubbleLabel.styleFn(d))
557
-
558
- return nodeGSelection.select<SVGCircleElement>(`circle.${nodeCircleClassName}`)
559
- }
560
-
561
- // function renderNodeLabelG ({ nodeGSelection }: {
562
- // nodeGSelection: d3.Selection<SVGGElement, any, any, unknown>
563
- // }) {
564
- // nodeGSelection.each((data,i,g) => {
565
- // const gSelection = d3.select(g[i])
566
- // gSelection.selectAll<SVGGElement, RenderNode>('g')
567
- // .data([data])
568
- // .join(
569
- // enter => {
570
- // const enterSelection = enter
571
- // .append('g')
572
- // .classed(nodeLabelGClassName, true)
573
- // // .attr('cursor', 'pointer')
574
- // return enterSelection
575
- // },
576
- // update => {
577
- // return update
578
- // },
579
- // exit => {
580
- // return exit.remove()
581
- // }
582
- // )
583
- // })
584
-
585
- // return nodeGSelection.select<SVGTextElement>(`g.${nodeLabelGClassName}`)
586
- // }
587
-
588
- // function renderNodeLabel ({ nodeLabelGSelection, fullParams, fullChartParams }: {
589
- // nodeLabelGSelection: d3.Selection<SVGGElement, RenderNode, any, unknown>
590
- // fullParams: ForceDirectedBubblesParams
591
- // fullChartParams: ChartParams
592
- // }) {
593
- // nodeLabelGSelection.each((data,i,g) => {
594
- // const gSelection = d3.select(g[i])
595
- // gSelection.selectAll<SVGTextElement, RenderNode>('text')
596
- // .data([data], d => d.id)
597
- // .join(
598
- // enter => {
599
- // const enterSelection = enter
600
- // .append('text')
601
- // .classed(nodeLabelClassName, true)
602
- // // .attr('cursor', 'pointer')
603
- // .attr('text-anchor', 'middle')
604
- // .attr('pointer-events', 'none')
605
- // return enterSelection
606
- // },
607
- // update => {
608
- // return update
609
- // },
610
- // exit => {
611
- // return exit.remove()
612
- // }
613
- // )
614
- // .text(d => d.label)
615
- // .attr('transform', d => `translate(0, ${- d.r - 10})`)
616
- // .attr('fill', d => getDatumColor({ datum: d, colorType: fullParams.node.labelColorType, fullChartParams }))
617
- // .attr('font-size', fullChartParams.styles.textSize)
618
- // .attr('style', d => fullParams.node.labelStyleFn(d))
619
- // })
620
-
621
- // return nodeLabelGSelection.select<SVGTextElement>(`text.${nodeLabelClassName}`)
622
- // }
623
-
624
- function renderEdgeG ({ edgeListGSelection, edges }: {
625
- edgeListGSelection: d3.Selection<SVGGElement, any, any, unknown>
626
- edges: RenderEdge[]
627
- }) {
628
- return edgeListGSelection.selectAll<SVGGElement, RenderEdge>('g')
629
- .data(edges, d => d.id)
630
- .join(
631
- enter => {
632
- const enterSelection = enter
633
- .append('g')
634
- .classed(edgeGClassName, true)
635
- // .attr('cursor', 'pointer')
636
- return enterSelection
637
- },
638
- update => {
639
- return update
640
- },
641
- exit => {
642
- return exit.remove()
643
- }
644
- )
645
- }
646
-
647
- function renderEdgeArrowPath ({ edgeGSelection, fullParams, fullChartParams }: {
648
- edgeGSelection: d3.Selection<SVGGElement, RenderEdge, any, unknown>
649
- fullParams: ForceDirectedBubblesParams
650
- fullChartParams: ChartParams
651
- }) {
652
- edgeGSelection.each((data,i,g) => {
653
- const gSelection = d3.select(g[i])
654
- gSelection.selectAll<SVGPathElement, ComputedEdge>('path')
655
- .data([data])
656
- .join(
657
- enter => {
658
- return enter
659
- .append('path')
660
- .classed(edgeArrowPathClassName, true)
661
- },
662
- update => {
663
- return update
664
- },
665
- exit => {
666
- return exit.remove()
667
- }
668
- )
669
- .attr('marker-end', d => `url(#${d.markerId})`)
670
- .attr('stroke', d => getDatumColor({ datum: d.data, colorType: fullParams.arrow.colorType, fullChartParams }))
671
- .attr('stroke-width', d => d.strokeWidth)
672
- .attr('style', d => fullParams.arrow.styleFn(d))
673
- })
674
-
675
- return edgeGSelection.select<SVGPathElement>(`path.${edgeArrowPathClassName}`)
676
- }
677
-
678
- function renderEdgeLabelG ({ edgeGSelection }: {
679
- edgeGSelection: d3.Selection<SVGGElement, any, any, unknown>
680
- }) {
681
- edgeGSelection.each((data,i,g) => {
682
- const gSelection = d3.select(g[i])
683
- gSelection.selectAll<SVGGElement, RenderEdge>('g')
684
- .data([data])
685
- .join(
686
- enter => {
687
- const enterSelection = enter
688
- .append('g')
689
- .classed(edgeLabelGClassName, true)
690
- // .attr('cursor', 'pointer')
691
- return enterSelection
692
- },
693
- update => {
694
- return update
695
- },
696
- exit => {
697
- return exit.remove()
698
- }
699
- )
700
- })
701
-
702
- return edgeGSelection.select<SVGTextElement>(`g.${edgeLabelGClassName}`)
703
- }
704
-
705
- function renderEdgeLabel ({ edgeLabelGSelection, fullParams, fullChartParams }: {
706
- edgeLabelGSelection: d3.Selection<SVGGElement, RenderEdge, any, unknown>
707
- fullParams: ForceDirectedBubblesParams
708
- fullChartParams: ChartParams
709
- }) {
710
- edgeLabelGSelection.each((data,i,g) => {
711
- const gSelection = d3.select(g[i])
712
- gSelection.selectAll<SVGTextElement, RenderEdge>('text')
713
- .data([data], d => d.id)
714
- .join(
715
- enter => {
716
- const enterSelection = enter
717
- .append('text')
718
- .classed(edgeLabelClassName, true)
719
- // .attr('cursor', 'pointer')
720
- .attr('text-anchor', 'middle')
721
- .attr('pointer-events', 'none')
722
- return enterSelection
723
- },
724
- update => {
725
- return update
726
- },
727
- exit => {
728
- return exit.remove()
729
- }
730
- )
731
- .text(d => d.label)
732
- .attr('fill', d => getDatumColor({ datum: d, colorType: fullParams.arrowLabel.colorType, fullChartParams }))
733
- .attr('font-size', fullChartParams.styles.textSize)
734
- .attr('style', d => fullParams.arrowLabel.styleFn(d))
735
- })
736
-
737
- return edgeLabelGSelection.select<SVGTextElement>(`text.${edgeLabelClassName}`)
738
- }
739
-
740
-
741
- // function renderBubbles ({ selection, bubblesData, fullParams, sumSeries }: {
742
- // selection: d3.Selection<SVGGElement, any, any, any>
743
- // bubblesData: BubblesDatum[]
744
- // fullParams: BubblesParams
745
- // sumSeries: boolean
746
- // }) {
747
- // const bubblesSelection = selection.selectAll<SVGGElement, BubblesDatum>("g")
748
- // .data(bubblesData, (d) => d.id)
749
- // .join(
750
- // enter => {
751
- // const enterSelection = enter
752
- // .append('g')
753
- // .attr('cursor', 'pointer')
754
- // .attr('font-size', 12)
755
- // .style('fill', '#ffffff')
756
- // .attr("text-anchor", "middle")
757
-
758
- // enterSelection
759
- // .append("circle")
760
- // .attr("class", "node")
761
- // .attr("cx", 0)
762
- // .attr("cy", 0)
763
- // // .attr("r", 1e-6)
764
- // .attr('fill', (d) => d.color)
765
- // // .transition()
766
- // // .duration(500)
767
-
768
- // enterSelection
769
- // .append('text')
770
- // .style('opacity', 0.8)
771
- // .attr('pointer-events', 'none')
772
-
773
- // return enterSelection
774
- // },
775
- // update => {
776
- // return update
777
- // },
778
- // exit => {
779
- // return exit
780
- // .remove()
781
- // }
782
- // )
783
- // .attr("transform", (d) => {
784
- // return `translate(${d.x},${d.y})`
785
- // })
786
-
787
- // // 泡泡文字要使用的的資料欄位
788
- // const textDataColumn = sumSeries ? 'seriesLabel' : 'label'// 如果有合併series則使用seriesLabel
789
-
790
- // bubblesSelection.select('circle')
791
- // .transition()
792
- // .duration(200)
793
- // .attr("r", (d) => d.r)
794
- // .attr('fill', (d) => d.color)
795
- // bubblesSelection
796
- // .each((d,i,g) => {
797
- // const gSelection = d3.select(g[i])
798
- // let breakAll = true
799
- // if (d[textDataColumn].length <= fullParams.label.lineLengthMin) {
800
- // breakAll = false
801
- // }
802
- // gSelection.call(renderCircleText, {
803
- // text: d[textDataColumn],
804
- // radius: d.r * fullParams.label.fillRate,
805
- // lineHeight: fullParams.label.lineHeight,
806
- // isBreakAll: breakAll
807
- // })
808
-
809
- // })
810
-
811
- // return bubblesSelection
812
- // }
813
-
814
- // function setHighlightData ({ data, highlightRIncrease, highlightIds }: {
815
- // data: BubblesDatum[]
816
- // // fullParams: BubblesParams
817
- // highlightRIncrease: number
818
- // highlightIds: string[]
819
- // }) {
820
- // if (highlightRIncrease == 0) {
821
- // return
822
- // }
823
- // if (!highlightIds.length) {
824
- // data.forEach(d => d.r = d._originR)
825
- // return
826
- // }
827
- // data.forEach(d => {
828
- // if (highlightIds.includes(d.id)) {
829
- // d.r = d._originR + highlightRIncrease
830
- // } else {
831
- // d.r = d._originR
832
- // }
833
- // })
834
- // }
835
-
836
-
837
- // function groupBubbles ({ fullParams, SeriesContainerPositionMap }: {
838
- // fullParams: BubblesParams
839
- // // graphicWidth: number
840
- // // graphicHeight: number
841
- // SeriesContainerPositionMap: Map<string, ContainerPosition>
842
- // }) {
843
- // // console.log('groupBubbles')
844
- // force!
845
- // // .force('x', d3.forceX().strength(fullParams.force.strength).x(graphicWidth / 2))
846
- // // .force('y', d3.forceY().strength(fullParams.force.strength).y(graphicHeight / 2))
847
- // .force('x', d3.forceX().strength(fullParams.force.strength).x((data: BubblesSimulationDatum) => {
848
- // return SeriesContainerPositionMap.get(data.seriesLabel)!.centerX
849
- // }))
850
- // .force('y', d3.forceY().strength(fullParams.force.strength).y((data: BubblesSimulationDatum) => {
851
- // return SeriesContainerPositionMap.get(data.seriesLabel)!.centerY
852
- // }))
853
-
854
- // force!.alpha(1).restart()
855
- // }
856
-
857
- function highlightNodes ({ nodeGSelection, edgeGSelection, highlightIds, fullChartParams }: {
858
- nodeGSelection: d3.Selection<SVGGElement, RenderNode, SVGGElement, any>
859
- edgeGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any>
860
- fullChartParams: ChartParams
861
- highlightIds: string[]
862
- }) {
863
- nodeGSelection.interrupt('highlight')
864
- // console.log(highlightIds)
865
- if (!highlightIds.length) {
866
- nodeGSelection
867
- .transition('highlight')
868
- .style('opacity', 1)
869
- edgeGSelection
870
- .transition('highlight')
871
- .style('opacity', 1)
872
- return
873
- }
874
-
875
- edgeGSelection
876
- .style('opacity', fullChartParams.styles.unhighlightedOpacity)
877
-
878
- nodeGSelection.each((d, i, n) => {
879
- const segment = d3.select(n[i])
880
-
881
- if (highlightIds.includes(d.id)) {
882
- segment
883
- .style('opacity', 1)
884
- .transition('highlight')
885
- .ease(d3.easeElastic)
886
- .duration(500)
887
- } else {
888
- // 取消
889
- segment
890
- .style('opacity', fullChartParams.styles.unhighlightedOpacity)
891
- }
892
- })
893
- }
894
-
895
- // 暫不處理edge的highlight
896
- // function highlightEdges ({ edgeGSelection, highlightIds, fullChartParams }: {
897
- // edgeGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any>
898
- // fullChartParams: ChartParams
899
- // highlightIds: string[]
900
- // }) {
901
- // edgeGSelection.interrupt('highlight')
902
-
903
- // if (!highlightIds.length) {
904
- // edgeGSelection
905
- // .transition('highlight')
906
- // .style('opacity', 1)
907
- // return
908
- // }
909
-
910
- // edgeGSelection.each((d, i, n) => {
911
- // const segment = d3.select(n[i])
912
-
913
- // if (highlightIds.includes(d.id)) {
914
- // segment
915
- // .style('opacity', 1)
916
- // .transition('highlight')
917
- // .ease(d3.easeElastic)
918
- // .duration(500)
919
- // } else {
920
- // // 取消放大
921
- // segment
922
- // .style('opacity', fullChartParams.styles.unhighlightedOpacity)
923
- // }
924
- // })
925
- // }
926
-
927
- export const ForceDirectedBubbles = defineRelationshipPlugin(pluginConfig)(({ selection, rootSelection, name, observer, subject }) => {
928
-
929
- const destroy$ = new Subject()
930
-
931
- const gSelection = selection.append('g').classed(gSelectionClassName, true)
932
- const defsSelection = gSelection.append('defs')
933
- const edgeListGSelection = gSelection.append('g').classed(edgeListGClassName, true)
934
- const nodeListGSelection = gSelection.append('g').classed(nodeListGClassName, true)
935
-
936
- let nodeGSelection: d3.Selection<SVGGElement, RenderNode, SVGGElement, any> | undefined
937
- let nodeCircleSelection: d3.Selection<SVGCircleElement, RenderNode, SVGGElement, any> | undefined
938
- // let nodeLabelGSelection: d3.Selection<SVGGElement, RenderNode, SVGGElement, any> | undefined
939
- // let nodeLabelSelection: d3.Selection<SVGTextElement, RenderNode, SVGGElement, any> | undefined
940
- let edgeGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any> | undefined
941
- let edgeArrowSelection: d3.Selection<SVGPathElement, RenderEdge, SVGGElement, any> | undefined
942
- let edgeLabelGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any> | undefined
943
- let edgeLabelSelection: d3.Selection<SVGTextElement, RenderEdge, SVGGElement, any> | undefined
944
-
945
- const dragStatus$ = new BehaviorSubject<DragStatus>('end') // start, drag, end
946
- const mouseEvent$ = new Subject<EventRelationship>()
947
-
948
-
949
- // init zoom
950
- const d3Zoom$ = observer.fullParams$.pipe(
951
- takeUntil(destroy$),
952
- // map(d => d.scaleExtent),
953
- // distinctUntilChanged((a, b) => String(a) === String(b)),
954
- // first(),
955
- map(data => {
956
- let d3Zoom = data.zoomable
957
- ? d3.zoom().on('zoom', (event) => {
958
- // console.log(event)
959
- // this.svgGroup.attr('transform', `translate(
960
- // ${event.transform.x + (this.zoom.xOffset * event.transform.k)},
961
- // ${event.transform.y + (this.zoom.yOffset * event.transform.k)}
962
- // ) scale(
963
- // ${event.transform.k}
964
- // )`)
965
- gSelection.attr('transform', `translate(
966
- ${event.transform.x},
967
- ${event.transform.y}
968
- ) scale(
969
- ${event.transform.k}
970
- )`)
971
-
972
- // if (data.node.labelSizeFixed && nodeLabelSelection) {
973
- // // 反向 scale 抵消掉放大縮小
974
- // nodeLabelSelection.attr('transform', `scale(${1 / event.transform.k})`)
975
- // }
976
- if (data.arrowLabel.sizeFixed && edgeLabelSelection) {
977
- // 反向 scale 抵消掉放大縮小
978
- edgeLabelSelection.attr('transform', `scale(${1 / event.transform.k})`)
979
- }
980
- })
981
- : d3.zoom().on('zoom', null)
982
- if (data.scaleExtent) {
983
- d3Zoom.scaleExtent([data.scaleExtent.min, data.scaleExtent.max])
984
- }
985
- rootSelection.call(d3Zoom)
986
-
987
- return d3Zoom
988
- }),
989
- // shareReplay(1)
990
- )
991
-
992
- // zoom transform
993
- combineLatest({
994
- d3Zoom: d3Zoom$,
995
- transform: observer.fullParams$.pipe(
996
- takeUntil(destroy$),
997
- map(d => d.transform),
998
- )
999
- }).pipe(
1000
- takeUntil(destroy$),
1001
- switchMap(async d => d)
1002
- ).subscribe(data => {
1003
- // console.log('call')
1004
- selection.call(
1005
- data.d3Zoom.transform, d3.zoomIdentity
1006
- .translate(data.transform.x, data.transform.y)
1007
- .scale(data.transform.k)
1008
- )
1009
- })
1010
-
1011
-
1012
- const simulation$: Observable<d3.Simulation<d3.SimulationNodeDatum, undefined>> = combineLatest({
1013
- layout: observer.layout$.pipe(
1014
- first() // 只使用第一次的尺寸(置中)
1015
- ),
1016
- fullParams: observer.fullParams$
1017
- }).pipe(
1018
- takeUntil(destroy$),
1019
- switchMap(async d => d),
1020
- map(data => createSimulation(data.layout, data.fullParams)),
1021
- shareReplay(1)
1022
- )
1023
-
1024
- const nodeMinMaxValue$ = observer.computedData$.pipe(
1025
- takeUntil(destroy$),
1026
- map(data => {
1027
- const hadValueData = data.nodes.filter(d => d.value != undefined)
1028
- if (!hadValueData.length) {
1029
- return [0, 2] // 給預設值
1030
- }
1031
- const minMax = getMinMax(data.nodes.map(d => d.value))
1032
- if (hadValueData.length == 1 || minMax[0] === minMax[1]) {
1033
- return [minMax[0] - 1, minMax[1] + 1] // 避免最大最小值相同
1034
- }
1035
- return minMax
1036
- }),
1037
- shareReplay(1)
1038
- )
1039
-
1040
- const edgeMinMaxValue$ = observer.computedData$.pipe(
1041
- takeUntil(destroy$),
1042
- map(data => {
1043
- const hadValueData = data.edges.filter(d => d.value != undefined)
1044
- if (!hadValueData.length) {
1045
- return [0, 2] // 給預設值
1046
- }
1047
- const minMax = getMinMax(data.edges.map(d => d.value))
1048
- if (hadValueData.length == 1 || minMax[0] === minMax[1]) {
1049
- return [minMax[0] - 1, minMax[1] + 1] // 避免最大最小值相同
1050
- }
1051
- return minMax
1052
- }),
1053
- shareReplay(1)
1054
- )
1055
-
1056
- // 當無value時給的預設值
1057
- const defaultNodeValue$ = nodeMinMaxValue$.pipe(
1058
- takeUntil(destroy$),
1059
- map(data => (data[1] - data[0]) / 2) // 預設值為最大及最小的中間值
1060
- )
1061
-
1062
- // 當無value時給的預設值
1063
- const defaultEdgeValue$ = edgeMinMaxValue$.pipe(
1064
- takeUntil(destroy$),
1065
- map(data => (data[1] - data[0]) / 2) // 預設值為最大及最小的中間值
1066
- )
1067
-
1068
- const radiusScale$ = combineLatest({
1069
- nodeMinMaxValue: nodeMinMaxValue$,
1070
- fullParams: observer.fullParams$
1071
- }).pipe(
1072
- takeUntil(destroy$),
1073
- switchMap(async (d) => d),
1074
- map(data => {
1075
- // console.log({ totalR: data.totalR, totalValue: data.totalValue })
1076
- return d3.scalePow()
1077
- .domain(data.nodeMinMaxValue)
1078
- .range([data.fullParams.bubble.radiusMin, data.fullParams.bubble.radiusMax])
1079
- .exponent(data.fullParams.bubble.arcScaleType === 'area'
1080
- ? 0.5 // 數值映射面積(0.5為取平方根)
1081
- : 1 // 數值映射半徑
1082
- )
1083
- })
1084
- )
1085
-
1086
- const strokeWidthScale$ = combineLatest({
1087
- edgeMinMaxValue: edgeMinMaxValue$,
1088
- fullParams: observer.fullParams$
1089
- }).pipe(
1090
- takeUntil(destroy$),
1091
- switchMap(async (d) => d),
1092
- map(data => {
1093
- return d3.scaleLinear()
1094
- .domain(data.edgeMinMaxValue)
1095
- .range([data.fullParams.arrow.strokeWidthMin, data.fullParams.arrow.strokeWidthMax])
1096
- })
1097
- )
1098
-
1099
- // 先將未篩選的資料全部儲起來,就不會因為 visibleFilter 而重新計算
1100
- const RenderNodeMap$: Observable<Map<string, RenderNode>> = combineLatest({
1101
- computedData: observer.computedData$,
1102
- radiusScale: radiusScale$,
1103
- defaultNodeValue: defaultNodeValue$,
1104
- }).pipe(
1105
- takeUntil(destroy$),
1106
- switchMap(async (d) => d),
1107
- map(data => {
1108
- return new Map(
1109
- data.computedData.nodes.map(_d => {
1110
- let d: RenderNode = _d as RenderNode
1111
- d.r = data.radiusScale(d.value ?? data.defaultNodeValue)
1112
- return [d.id, d]
1113
- })
1114
- )
1115
- }),
1116
- )
1117
-
1118
- // 先將未篩選的資料全部儲起來,就不會因為 visibleFilter 而重新計算
1119
- const RenderEdgeMap$: Observable<Map<string, RenderEdge>> = combineLatest({
1120
- computedData: observer.computedData$,
1121
- strokeWidthScale: strokeWidthScale$,
1122
- defaultEdgeValue: defaultEdgeValue$
1123
- }).pipe(
1124
- takeUntil(destroy$),
1125
- switchMap(async (d) => d),
1126
- map(data => {
1127
- return new Map(
1128
- data.computedData.edges.map(_d => {
1129
- let d: any = _d as RenderEdge
1130
- d.source = _d.startNode // reference
1131
- d.target = _d.endNode
1132
- d.strokeWidth = data.strokeWidthScale(d.value ?? data.defaultEdgeValue)
1133
- d.markerId = `${defsArrowMarkerId}__${d.id}`
1134
- return [d.id, d]
1135
- })
1136
- )
1137
- }),
1138
- )
1139
-
1140
- const renderData$: Observable<RenderData> = combineLatest({
1141
- visibleComputedData: observer.visibleComputedData$,
1142
- RenderNodeMap: RenderNodeMap$,
1143
- RenderEdgeMap: RenderEdgeMap$,
1144
- }).pipe(
1145
- takeUntil(destroy$),
1146
- switchMap(async (d) => d),
1147
- map(data => {
1148
- return {
1149
- nodes: data.visibleComputedData.nodes.map(d => data.RenderNodeMap.get(d.id)!),
1150
- edges: data.visibleComputedData.edges.map(d => data.RenderEdgeMap.get(d.id)!),
1151
- }
1152
- }),
1153
- shareReplay(1)
1154
- )
1155
-
1156
- const markerParams$: Observable<MarkerParams> = observer.fullParams$.pipe(
1157
- takeUntil(destroy$),
1158
- map(fullParams => {
1159
- return {
1160
- viewBox: `-${fullParams.arrow.pointerWidth} -${fullParams.arrow.pointerHeight / 2} ${fullParams.arrow.pointerWidth} ${fullParams.arrow.pointerHeight}`,
1161
- d: `M${-fullParams.arrow.pointerWidth},${-fullParams.arrow.pointerHeight / 2}L0,0L${-fullParams.arrow.pointerWidth},${fullParams.arrow.pointerHeight / 2}`, // 箭頭的尖端為(0,0)
1162
- pointerWidth: fullParams.arrow.pointerWidth,
1163
- pointerHeight: fullParams.arrow.pointerHeight,
1164
- }
1165
- })
1166
- )
1167
-
1168
- const markerData$: Observable<MarkerDatum[]> = combineLatest({
1169
- computedData: observer.computedData$,
1170
- fullParams: observer.fullParams$,
1171
- RenderNodeMap: RenderNodeMap$,
1172
- RenderEdgeMap: RenderEdgeMap$,
1173
- }).pipe(
1174
- takeUntil(destroy$),
1175
- switchMap(async d => d),
1176
- map(data => {
1177
- return data.computedData.edges.map(d => {
1178
- const renderEdge = data.RenderEdgeMap.get(d.id)!
1179
- const renderEndNode = data.RenderNodeMap.get(d.endNode.id)!
1180
- return {
1181
- id: renderEdge.markerId,
1182
- edgeId: d.id,
1183
- strokeWidth: renderEdge.strokeWidth,
1184
- /* refX:修正marker位置(計算出和circle半徑相等的寬度)
1185
- (1)circle半徑需加上 strokeWidth/2 是因為框線是以 circle 的邊緣往內及往外擴展,所以 stroke 多出來的寬度是一半而已
1186
- (2)circle半徑需除以 path 寬度是因為「marker 的位置會受到 path 的stroke-width影響」,所以要進行修正
1187
- (3)- 1 是要修正奇怪的誤差(不知原因)
1188
- */
1189
- refX: ((renderEndNode.r + (data.fullParams.bubble.strokeWidth / 2)) / renderEdge.strokeWidth) - 1
1190
- }
1191
- })
1192
- }),
1193
- )
1194
-
1195
- // <marker> marker selection
1196
- combineLatest({
1197
- defsSelection,
1198
- markerParams: markerParams$,
1199
- markerData: markerData$
1200
- }).pipe(
1201
- takeUntil(destroy$),
1202
- map(data => {
1203
- return renderArrowMarker({
1204
- defsSelection,
1205
- markerParams: data.markerParams,
1206
- markerData: data.markerData
1207
- })
1208
- })
1209
- ).subscribe()
1210
-
1211
-
1212
- combineLatest({
1213
- renderData: renderData$,
1214
- computedData: observer.computedData$,
1215
- CategoryNodeMap: observer.CategoryNodeMap$,
1216
- simulation: simulation$,
1217
- fullParams: observer.fullParams$,
1218
- fullChartParams: observer.fullChartParams$
1219
- }).pipe(
1220
- takeUntil(destroy$),
1221
- switchMap(async d => d),
1222
- ).subscribe(data => {
1223
-
1224
- nodeGSelection = renderNodeG({
1225
- nodeListGSelection: nodeListGSelection,
1226
- nodes: data.renderData.nodes as RenderNode[],
1227
- })
1228
-
1229
- nodeCircleSelection = renderNodeCircle({
1230
- nodeGSelection: nodeGSelection,
1231
- fullParams: data.fullParams,
1232
- fullChartParams: data.fullChartParams
1233
- })
1234
- nodeCircleSelection.call(drag(data.simulation, dragStatus$))
1235
-
1236
- // nodeLabelGSelection = renderNodeLabelG({
1237
- // nodeGSelection: nodeGSelection,
1238
- // })
1239
-
1240
- // nodeLabelSelection = renderNodeLabel({
1241
- // nodeLabelGSelection: nodeLabelGSelection,
1242
- // fullParams: data.fullParams,
1243
- // fullChartParams: data.fullChartParams
1244
- // })
1245
-
1246
- edgeGSelection = renderEdgeG({
1247
- edgeListGSelection: edgeListGSelection,
1248
- edges: data.renderData.edges
1249
- })
1250
-
1251
- edgeArrowSelection = renderEdgeArrowPath({
1252
- edgeGSelection: edgeGSelection,
1253
- fullParams: data.fullParams,
1254
- fullChartParams: data.fullChartParams
1255
- })
1256
-
1257
- edgeLabelGSelection = renderEdgeLabelG({
1258
- edgeGSelection: edgeGSelection,
1259
- })
1260
-
1261
- edgeLabelSelection = renderEdgeLabel({
1262
- edgeLabelGSelection: edgeLabelGSelection,
1263
- fullParams: data.fullParams,
1264
- fullChartParams: data.fullChartParams
1265
- })
1266
-
1267
- data.simulation.nodes(data.renderData.nodes)
1268
- .on('tick', () => {
1269
- edgeArrowSelection.attr('d', linkArcFn)
1270
- nodeGSelection.attr('transform', translateFn)
1271
- // nodeLabelGSelection.attr('transform', d => translateFn({
1272
- // x: d.x,
1273
- // y: d.y - d.r - 10
1274
- // }))
1275
- edgeLabelGSelection.attr('transform', d => translateCenterFn(d))
1276
- })
1277
- ;(data.simulation.force("link") as any).links(data.renderData.edges)
1278
-
1279
- data.simulation.alpha(0.3).restart()
1280
-
1281
- nodeCircleSelection
1282
- .on('mouseover', (event, datum) => {
1283
- event.stopPropagation()
1284
-
1285
- mouseEvent$.next({
1286
- type: 'relationship',
1287
- eventName: 'mouseover',
1288
- pluginName,
1289
- highlightTarget: data.fullChartParams.highlightTarget,
1290
- datum: datum,
1291
- category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1292
- categoryIndex: datum.categoryIndex,
1293
- categoryLabel: datum.categoryLabel,
1294
- event,
1295
- data: data.computedData
1296
- })
1297
- })
1298
- .on('mousemove', (event, datum) => {
1299
- event.stopPropagation()
1300
-
1301
- mouseEvent$.next({
1302
- type: 'relationship',
1303
- eventName: 'mousemove',
1304
- pluginName,
1305
- highlightTarget: data.fullChartParams.highlightTarget,
1306
- datum: datum,
1307
- category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1308
- categoryIndex: datum.categoryIndex,
1309
- categoryLabel: datum.categoryLabel,
1310
- event,
1311
- data: data.computedData
1312
- })
1313
- })
1314
- .on('mouseout', (event, datum) => {
1315
- event.stopPropagation()
1316
-
1317
- mouseEvent$.next({
1318
- type: 'relationship',
1319
- eventName: 'mouseout',
1320
- pluginName,
1321
- highlightTarget: data.fullChartParams.highlightTarget,
1322
- datum: datum,
1323
- category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1324
- categoryIndex: datum.categoryIndex,
1325
- categoryLabel: datum.categoryLabel,
1326
- event,
1327
- data: data.computedData
1328
- })
1329
- })
1330
- .on('click', (event, datum) => {
1331
- event.stopPropagation()
1332
-
1333
- mouseEvent$.next({
1334
- type: 'relationship',
1335
- eventName: 'click',
1336
- pluginName,
1337
- highlightTarget: data.fullChartParams.highlightTarget,
1338
- datum: datum,
1339
- category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1340
- categoryIndex: datum.categoryIndex,
1341
- categoryLabel: datum.categoryLabel,
1342
- event,
1343
- data: data.computedData
1344
- })
1345
- })
1346
- })
1347
-
1348
- dragStatus$.pipe(
1349
- // 只有沒有托曳時才執行
1350
- switchMap(d => iif(() => d === 'end', mouseEvent$, EMPTY))
1351
- ).subscribe(data => {
1352
- subject.event$.next(data)
1353
- })
1354
-
1355
- combineLatest({
1356
- renderData: renderData$,
1357
- highlightNodes: observer.relationshipHighlightNodes$.pipe(
1358
- map(data => data.map(d => d.id))
1359
- ),
1360
- highlightEdges: observer.relationshipHighlightEdges$.pipe(
1361
- map(data => data.map(d => d.id))
1362
- ),
1363
- fullChartParams: observer.fullChartParams$,
1364
- fullParams: observer.fullParams$,
1365
- }).pipe(
1366
- takeUntil(destroy$),
1367
- switchMap(async d => d)
1368
- ).subscribe(data => {
1369
- if (!nodeGSelection || !edgeGSelection) {
1370
- return
1371
- }
1372
-
1373
- highlightNodes({
1374
- nodeGSelection,
1375
- edgeGSelection,
1376
- highlightIds: data.highlightNodes,
1377
- fullChartParams: data.fullChartParams
1378
- })
1379
- // highlightEdges({
1380
- // edgeGSelection,
1381
- // highlightIds: data.highlightEdges,
1382
- // fullChartParams: data.fullChartParams
1383
- // })
1384
- })
1385
-
1386
-
1387
-
1388
- return () => {
1389
- destroy$.next(undefined)
1390
- }
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: fullParams.bubbleLabel.wordBreakAll
551
+ })
552
+
553
+ })
554
+
555
+ nodeGSelection.select('text')
556
+ .attr('pointer-events', 'none')
557
+ .attr('style', d => fullParams.bubbleLabel.styleFn(d))
558
+
559
+ return nodeGSelection.select<SVGCircleElement>(`circle.${nodeCircleClassName}`)
560
+ }
561
+
562
+ // function renderNodeLabelG ({ nodeGSelection }: {
563
+ // nodeGSelection: d3.Selection<SVGGElement, any, any, unknown>
564
+ // }) {
565
+ // nodeGSelection.each((data,i,g) => {
566
+ // const gSelection = d3.select(g[i])
567
+ // gSelection.selectAll<SVGGElement, RenderNode>('g')
568
+ // .data([data])
569
+ // .join(
570
+ // enter => {
571
+ // const enterSelection = enter
572
+ // .append('g')
573
+ // .classed(nodeLabelGClassName, true)
574
+ // // .attr('cursor', 'pointer')
575
+ // return enterSelection
576
+ // },
577
+ // update => {
578
+ // return update
579
+ // },
580
+ // exit => {
581
+ // return exit.remove()
582
+ // }
583
+ // )
584
+ // })
585
+
586
+ // return nodeGSelection.select<SVGTextElement>(`g.${nodeLabelGClassName}`)
587
+ // }
588
+
589
+ // function renderNodeLabel ({ nodeLabelGSelection, fullParams, fullChartParams }: {
590
+ // nodeLabelGSelection: d3.Selection<SVGGElement, RenderNode, any, unknown>
591
+ // fullParams: ForceDirectedBubblesParams
592
+ // fullChartParams: ChartParams
593
+ // }) {
594
+ // nodeLabelGSelection.each((data,i,g) => {
595
+ // const gSelection = d3.select(g[i])
596
+ // gSelection.selectAll<SVGTextElement, RenderNode>('text')
597
+ // .data([data], d => d.id)
598
+ // .join(
599
+ // enter => {
600
+ // const enterSelection = enter
601
+ // .append('text')
602
+ // .classed(nodeLabelClassName, true)
603
+ // // .attr('cursor', 'pointer')
604
+ // .attr('text-anchor', 'middle')
605
+ // .attr('pointer-events', 'none')
606
+ // return enterSelection
607
+ // },
608
+ // update => {
609
+ // return update
610
+ // },
611
+ // exit => {
612
+ // return exit.remove()
613
+ // }
614
+ // )
615
+ // .text(d => d.label)
616
+ // .attr('transform', d => `translate(0, ${- d.r - 10})`)
617
+ // .attr('fill', d => getDatumColor({ datum: d, colorType: fullParams.node.labelColorType, fullChartParams }))
618
+ // .attr('font-size', fullChartParams.styles.textSize)
619
+ // .attr('style', d => fullParams.node.labelStyleFn(d))
620
+ // })
621
+
622
+ // return nodeLabelGSelection.select<SVGTextElement>(`text.${nodeLabelClassName}`)
623
+ // }
624
+
625
+ function renderEdgeG ({ edgeListGSelection, edges }: {
626
+ edgeListGSelection: d3.Selection<SVGGElement, any, any, unknown>
627
+ edges: RenderEdge[]
628
+ }) {
629
+ return edgeListGSelection.selectAll<SVGGElement, RenderEdge>('g')
630
+ .data(edges, d => d.id)
631
+ .join(
632
+ enter => {
633
+ const enterSelection = enter
634
+ .append('g')
635
+ .classed(edgeGClassName, true)
636
+ // .attr('cursor', 'pointer')
637
+ return enterSelection
638
+ },
639
+ update => {
640
+ return update
641
+ },
642
+ exit => {
643
+ return exit.remove()
644
+ }
645
+ )
646
+ }
647
+
648
+ function renderEdgeArrowPath ({ edgeGSelection, fullParams, fullChartParams }: {
649
+ edgeGSelection: d3.Selection<SVGGElement, RenderEdge, any, unknown>
650
+ fullParams: ForceDirectedBubblesParams
651
+ fullChartParams: ChartParams
652
+ }) {
653
+ edgeGSelection.each((data,i,g) => {
654
+ const gSelection = d3.select(g[i])
655
+ gSelection.selectAll<SVGPathElement, ComputedEdge>('path')
656
+ .data([data])
657
+ .join(
658
+ enter => {
659
+ return enter
660
+ .append('path')
661
+ .classed(edgeArrowPathClassName, true)
662
+ },
663
+ update => {
664
+ return update
665
+ },
666
+ exit => {
667
+ return exit.remove()
668
+ }
669
+ )
670
+ .attr('marker-end', d => `url(#${d.markerId})`)
671
+ .attr('stroke', d => getDatumColor({ datum: d.data, colorType: fullParams.arrow.colorType, fullChartParams }))
672
+ .attr('stroke-width', d => d.strokeWidth)
673
+ .attr('style', d => fullParams.arrow.styleFn(d))
674
+ })
675
+
676
+ return edgeGSelection.select<SVGPathElement>(`path.${edgeArrowPathClassName}`)
677
+ }
678
+
679
+ function renderEdgeLabelG ({ edgeGSelection }: {
680
+ edgeGSelection: d3.Selection<SVGGElement, any, any, unknown>
681
+ }) {
682
+ edgeGSelection.each((data,i,g) => {
683
+ const gSelection = d3.select(g[i])
684
+ gSelection.selectAll<SVGGElement, RenderEdge>('g')
685
+ .data([data])
686
+ .join(
687
+ enter => {
688
+ const enterSelection = enter
689
+ .append('g')
690
+ .classed(edgeLabelGClassName, true)
691
+ // .attr('cursor', 'pointer')
692
+ return enterSelection
693
+ },
694
+ update => {
695
+ return update
696
+ },
697
+ exit => {
698
+ return exit.remove()
699
+ }
700
+ )
701
+ })
702
+
703
+ return edgeGSelection.select<SVGTextElement>(`g.${edgeLabelGClassName}`)
704
+ }
705
+
706
+ function renderEdgeLabel ({ edgeLabelGSelection, fullParams, fullChartParams }: {
707
+ edgeLabelGSelection: d3.Selection<SVGGElement, RenderEdge, any, unknown>
708
+ fullParams: ForceDirectedBubblesParams
709
+ fullChartParams: ChartParams
710
+ }) {
711
+ edgeLabelGSelection.each((data,i,g) => {
712
+ const gSelection = d3.select(g[i])
713
+ gSelection.selectAll<SVGTextElement, RenderEdge>('text')
714
+ .data([data], d => d.id)
715
+ .join(
716
+ enter => {
717
+ const enterSelection = enter
718
+ .append('text')
719
+ .classed(edgeLabelClassName, true)
720
+ // .attr('cursor', 'pointer')
721
+ .attr('text-anchor', 'middle')
722
+ .attr('pointer-events', 'none')
723
+ return enterSelection
724
+ },
725
+ update => {
726
+ return update
727
+ },
728
+ exit => {
729
+ return exit.remove()
730
+ }
731
+ )
732
+ .text(d => d.label)
733
+ .attr('fill', d => getDatumColor({ datum: d, colorType: fullParams.arrowLabel.colorType, fullChartParams }))
734
+ .attr('font-size', fullChartParams.styles.textSize)
735
+ .attr('style', d => fullParams.arrowLabel.styleFn(d))
736
+ })
737
+
738
+ return edgeLabelGSelection.select<SVGTextElement>(`text.${edgeLabelClassName}`)
739
+ }
740
+
741
+
742
+ // function renderBubbles ({ selection, bubblesData, fullParams, sumSeries }: {
743
+ // selection: d3.Selection<SVGGElement, any, any, any>
744
+ // bubblesData: BubblesDatum[]
745
+ // fullParams: BubblesParams
746
+ // sumSeries: boolean
747
+ // }) {
748
+ // const bubblesSelection = selection.selectAll<SVGGElement, BubblesDatum>("g")
749
+ // .data(bubblesData, (d) => d.id)
750
+ // .join(
751
+ // enter => {
752
+ // const enterSelection = enter
753
+ // .append('g')
754
+ // .attr('cursor', 'pointer')
755
+ // .attr('font-size', 12)
756
+ // .style('fill', '#ffffff')
757
+ // .attr("text-anchor", "middle")
758
+
759
+ // enterSelection
760
+ // .append("circle")
761
+ // .attr("class", "node")
762
+ // .attr("cx", 0)
763
+ // .attr("cy", 0)
764
+ // // .attr("r", 1e-6)
765
+ // .attr('fill', (d) => d.color)
766
+ // // .transition()
767
+ // // .duration(500)
768
+
769
+ // enterSelection
770
+ // .append('text')
771
+ // .style('opacity', 0.8)
772
+ // .attr('pointer-events', 'none')
773
+
774
+ // return enterSelection
775
+ // },
776
+ // update => {
777
+ // return update
778
+ // },
779
+ // exit => {
780
+ // return exit
781
+ // .remove()
782
+ // }
783
+ // )
784
+ // .attr("transform", (d) => {
785
+ // return `translate(${d.x},${d.y})`
786
+ // })
787
+
788
+ // // 泡泡文字要使用的的資料欄位
789
+ // const textDataColumn = sumSeries ? 'seriesLabel' : 'label'// 如果有合併series則使用seriesLabel
790
+
791
+ // bubblesSelection.select('circle')
792
+ // .transition()
793
+ // .duration(200)
794
+ // .attr("r", (d) => d.r)
795
+ // .attr('fill', (d) => d.color)
796
+ // bubblesSelection
797
+ // .each((d,i,g) => {
798
+ // const gSelection = d3.select(g[i])
799
+ // let breakAll = true
800
+ // if (d[textDataColumn].length <= fullParams.label.maxLineLength) {
801
+ // breakAll = false
802
+ // }
803
+ // gSelection.call(renderCircleText, {
804
+ // text: d[textDataColumn],
805
+ // radius: d.r * fullParams.label.fillRate,
806
+ // lineHeight: fullParams.label.lineHeight,
807
+ // isBreakAll: breakAll
808
+ // })
809
+
810
+ // })
811
+
812
+ // return bubblesSelection
813
+ // }
814
+
815
+ // function setHighlightData ({ data, highlightRIncrease, highlightIds }: {
816
+ // data: BubblesDatum[]
817
+ // // fullParams: BubblesParams
818
+ // highlightRIncrease: number
819
+ // highlightIds: string[]
820
+ // }) {
821
+ // if (highlightRIncrease == 0) {
822
+ // return
823
+ // }
824
+ // if (!highlightIds.length) {
825
+ // data.forEach(d => d.r = d._originR)
826
+ // return
827
+ // }
828
+ // data.forEach(d => {
829
+ // if (highlightIds.includes(d.id)) {
830
+ // d.r = d._originR + highlightRIncrease
831
+ // } else {
832
+ // d.r = d._originR
833
+ // }
834
+ // })
835
+ // }
836
+
837
+
838
+ // function groupBubbles ({ fullParams, SeriesContainerPositionMap }: {
839
+ // fullParams: BubblesParams
840
+ // // graphicWidth: number
841
+ // // graphicHeight: number
842
+ // SeriesContainerPositionMap: Map<string, ContainerPosition>
843
+ // }) {
844
+ // // console.log('groupBubbles')
845
+ // force!
846
+ // // .force('x', d3.forceX().strength(fullParams.force.strength).x(graphicWidth / 2))
847
+ // // .force('y', d3.forceY().strength(fullParams.force.strength).y(graphicHeight / 2))
848
+ // .force('x', d3.forceX().strength(fullParams.force.strength).x((data: BubblesSimulationDatum) => {
849
+ // return SeriesContainerPositionMap.get(data.seriesLabel)!.centerX
850
+ // }))
851
+ // .force('y', d3.forceY().strength(fullParams.force.strength).y((data: BubblesSimulationDatum) => {
852
+ // return SeriesContainerPositionMap.get(data.seriesLabel)!.centerY
853
+ // }))
854
+
855
+ // force!.alpha(1).restart()
856
+ // }
857
+
858
+ function highlightNodes ({ nodeGSelection, edgeGSelection, highlightIds, fullChartParams }: {
859
+ nodeGSelection: d3.Selection<SVGGElement, RenderNode, SVGGElement, any>
860
+ edgeGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any>
861
+ fullChartParams: ChartParams
862
+ highlightIds: string[]
863
+ }) {
864
+ nodeGSelection.interrupt('highlight')
865
+ edgeGSelection.interrupt('highlight')
866
+ // console.log(highlightIds)
867
+ if (!highlightIds.length) {
868
+ nodeGSelection
869
+ .transition('highlight')
870
+ .style('opacity', 1)
871
+ edgeGSelection
872
+ .transition('highlight')
873
+ .style('opacity', 1)
874
+ return
875
+ }
876
+
877
+ edgeGSelection
878
+ .style('opacity', fullChartParams.styles.unhighlightedOpacity)
879
+
880
+ nodeGSelection.each((d, i, n) => {
881
+ const segment = d3.select(n[i])
882
+
883
+ if (highlightIds.includes(d.id)) {
884
+ segment
885
+ .style('opacity', 1)
886
+ .transition('highlight')
887
+ .ease(d3.easeElastic)
888
+ .duration(500)
889
+ } else {
890
+ // 取消
891
+ segment
892
+ .style('opacity', fullChartParams.styles.unhighlightedOpacity)
893
+ }
894
+ })
895
+ }
896
+
897
+ // 暫不處理edge的highlight
898
+ // function highlightEdges ({ edgeGSelection, highlightIds, fullChartParams }: {
899
+ // edgeGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any>
900
+ // fullChartParams: ChartParams
901
+ // highlightIds: string[]
902
+ // }) {
903
+ // edgeGSelection.interrupt('highlight')
904
+
905
+ // if (!highlightIds.length) {
906
+ // edgeGSelection
907
+ // .transition('highlight')
908
+ // .style('opacity', 1)
909
+ // return
910
+ // }
911
+
912
+ // edgeGSelection.each((d, i, n) => {
913
+ // const segment = d3.select(n[i])
914
+
915
+ // if (highlightIds.includes(d.id)) {
916
+ // segment
917
+ // .style('opacity', 1)
918
+ // .transition('highlight')
919
+ // .ease(d3.easeElastic)
920
+ // .duration(500)
921
+ // } else {
922
+ // // 取消放大
923
+ // segment
924
+ // .style('opacity', fullChartParams.styles.unhighlightedOpacity)
925
+ // }
926
+ // })
927
+ // }
928
+
929
+ export const ForceDirectedBubbles = defineRelationshipPlugin(pluginConfig)(({ selection, rootSelection, name, observer, subject }) => {
930
+
931
+ const destroy$ = new Subject()
932
+
933
+ const gSelection = selection.append('g').classed(gSelectionClassName, true)
934
+ const defsSelection = gSelection.append('defs')
935
+ const edgeListGSelection = gSelection.append('g').classed(edgeListGClassName, true)
936
+ const nodeListGSelection = gSelection.append('g').classed(nodeListGClassName, true)
937
+
938
+ let nodeGSelection: d3.Selection<SVGGElement, RenderNode, SVGGElement, any> | undefined
939
+ let nodeCircleSelection: d3.Selection<SVGCircleElement, RenderNode, SVGGElement, any> | undefined
940
+ // let nodeLabelGSelection: d3.Selection<SVGGElement, RenderNode, SVGGElement, any> | undefined
941
+ // let nodeLabelSelection: d3.Selection<SVGTextElement, RenderNode, SVGGElement, any> | undefined
942
+ let edgeGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any> | undefined
943
+ let edgeArrowSelection: d3.Selection<SVGPathElement, RenderEdge, SVGGElement, any> | undefined
944
+ let edgeLabelGSelection: d3.Selection<SVGGElement, RenderEdge, SVGGElement, any> | undefined
945
+ let edgeLabelSelection: d3.Selection<SVGTextElement, RenderEdge, SVGGElement, any> | undefined
946
+
947
+ const dragStatus$ = new BehaviorSubject<DragStatus>('end') // start, drag, end
948
+ const mouseEvent$ = new Subject<EventRelationship>()
949
+
950
+
951
+ // init zoom
952
+ const d3Zoom$ = observer.fullParams$.pipe(
953
+ takeUntil(destroy$),
954
+ // map(d => d.scaleExtent),
955
+ // distinctUntilChanged((a, b) => String(a) === String(b)),
956
+ // first(),
957
+ map(data => {
958
+ let d3Zoom = data.zoomable
959
+ ? d3.zoom().on('zoom', (event) => {
960
+ // console.log(event)
961
+ // this.svgGroup.attr('transform', `translate(
962
+ // ${event.transform.x + (this.zoom.xOffset * event.transform.k)},
963
+ // ${event.transform.y + (this.zoom.yOffset * event.transform.k)}
964
+ // ) scale(
965
+ // ${event.transform.k}
966
+ // )`)
967
+ gSelection.attr('transform', `translate(
968
+ ${event.transform.x},
969
+ ${event.transform.y}
970
+ ) scale(
971
+ ${event.transform.k}
972
+ )`)
973
+
974
+ // if (data.node.labelSizeFixed && nodeLabelSelection) {
975
+ // // 反向 scale 抵消掉放大縮小
976
+ // nodeLabelSelection.attr('transform', `scale(${1 / event.transform.k})`)
977
+ // }
978
+ if (data.arrowLabel.sizeFixed && edgeLabelSelection) {
979
+ // 反向 scale 抵消掉放大縮小
980
+ edgeLabelSelection.attr('transform', `scale(${1 / event.transform.k})`)
981
+ }
982
+ })
983
+ : d3.zoom().on('zoom', null)
984
+ if (data.scaleExtent) {
985
+ d3Zoom.scaleExtent([data.scaleExtent.min, data.scaleExtent.max])
986
+ }
987
+ rootSelection.call(d3Zoom)
988
+
989
+ return d3Zoom
990
+ }),
991
+ // shareReplay(1)
992
+ )
993
+
994
+ // zoom transform
995
+ combineLatest({
996
+ d3Zoom: d3Zoom$,
997
+ transform: observer.fullParams$.pipe(
998
+ takeUntil(destroy$),
999
+ map(d => d.transform),
1000
+ )
1001
+ }).pipe(
1002
+ takeUntil(destroy$),
1003
+ switchMap(async d => d)
1004
+ ).subscribe(data => {
1005
+ // console.log('call')
1006
+ selection.call(
1007
+ data.d3Zoom.transform, d3.zoomIdentity
1008
+ .translate(data.transform.x, data.transform.y)
1009
+ .scale(data.transform.k)
1010
+ )
1011
+ })
1012
+
1013
+
1014
+ const simulation$: Observable<d3.Simulation<d3.SimulationNodeDatum, undefined>> = combineLatest({
1015
+ layout: observer.layout$.pipe(
1016
+ first() // 只使用第一次的尺寸(置中)
1017
+ ),
1018
+ fullParams: observer.fullParams$
1019
+ }).pipe(
1020
+ takeUntil(destroy$),
1021
+ switchMap(async d => d),
1022
+ map(data => createSimulation(data.layout, data.fullParams)),
1023
+ shareReplay(1)
1024
+ )
1025
+
1026
+ const nodeMinMaxValue$ = observer.computedData$.pipe(
1027
+ takeUntil(destroy$),
1028
+ map(data => {
1029
+ const hadValueData = data.nodes.filter(d => d.value != undefined)
1030
+ if (!hadValueData.length) {
1031
+ return [0, 2] // 給預設值
1032
+ }
1033
+ const minMax = getMinMax(data.nodes.map(d => d.value))
1034
+ if (hadValueData.length == 1 || minMax[0] === minMax[1]) {
1035
+ return [minMax[0] - 1, minMax[1] + 1] // 避免最大最小值相同
1036
+ }
1037
+ return minMax
1038
+ }),
1039
+ shareReplay(1)
1040
+ )
1041
+
1042
+ const edgeMinMaxValue$ = observer.computedData$.pipe(
1043
+ takeUntil(destroy$),
1044
+ map(data => {
1045
+ const hadValueData = data.edges.filter(d => d.value != undefined)
1046
+ if (!hadValueData.length) {
1047
+ return [0, 2] // 給預設值
1048
+ }
1049
+ const minMax = getMinMax(data.edges.map(d => d.value))
1050
+ if (hadValueData.length == 1 || minMax[0] === minMax[1]) {
1051
+ return [minMax[0] - 1, minMax[1] + 1] // 避免最大最小值相同
1052
+ }
1053
+ return minMax
1054
+ }),
1055
+ shareReplay(1)
1056
+ )
1057
+
1058
+ // 當無value時給的預設值
1059
+ const defaultNodeValue$ = nodeMinMaxValue$.pipe(
1060
+ takeUntil(destroy$),
1061
+ map(data => (data[1] - data[0]) / 2) // 預設值為最大及最小的中間值
1062
+ )
1063
+
1064
+ // 當無value時給的預設值
1065
+ const defaultEdgeValue$ = edgeMinMaxValue$.pipe(
1066
+ takeUntil(destroy$),
1067
+ map(data => (data[1] - data[0]) / 2) // 預設值為最大及最小的中間值
1068
+ )
1069
+
1070
+ const radiusScale$ = combineLatest({
1071
+ nodeMinMaxValue: nodeMinMaxValue$,
1072
+ fullParams: observer.fullParams$
1073
+ }).pipe(
1074
+ takeUntil(destroy$),
1075
+ switchMap(async (d) => d),
1076
+ map(data => {
1077
+ // console.log({ totalR: data.totalR, totalValue: data.totalValue })
1078
+ return d3.scalePow()
1079
+ .domain(data.nodeMinMaxValue)
1080
+ .range([data.fullParams.bubble.radiusMin, data.fullParams.bubble.radiusMax])
1081
+ .exponent(data.fullParams.bubble.arcScaleType === 'area'
1082
+ ? 0.5 // 數值映射面積(0.5為取平方根)
1083
+ : 1 // 數值映射半徑
1084
+ )
1085
+ })
1086
+ )
1087
+
1088
+ const strokeWidthScale$ = combineLatest({
1089
+ edgeMinMaxValue: edgeMinMaxValue$,
1090
+ fullParams: observer.fullParams$
1091
+ }).pipe(
1092
+ takeUntil(destroy$),
1093
+ switchMap(async (d) => d),
1094
+ map(data => {
1095
+ return d3.scaleLinear()
1096
+ .domain(data.edgeMinMaxValue)
1097
+ .range([data.fullParams.arrow.strokeWidthMin, data.fullParams.arrow.strokeWidthMax])
1098
+ })
1099
+ )
1100
+
1101
+ // 先將未篩選的資料全部儲起來,就不會因為 visibleFilter 而重新計算
1102
+ const RenderNodeMap$: Observable<Map<string, RenderNode>> = combineLatest({
1103
+ computedData: observer.computedData$,
1104
+ radiusScale: radiusScale$,
1105
+ defaultNodeValue: defaultNodeValue$,
1106
+ }).pipe(
1107
+ takeUntil(destroy$),
1108
+ switchMap(async (d) => d),
1109
+ map(data => {
1110
+ return new Map(
1111
+ data.computedData.nodes.map(_d => {
1112
+ let d: RenderNode = _d as RenderNode
1113
+ d.r = data.radiusScale(d.value ?? data.defaultNodeValue)
1114
+ return [d.id, d]
1115
+ })
1116
+ )
1117
+ }),
1118
+ )
1119
+
1120
+ // 先將未篩選的資料全部儲起來,就不會因為 visibleFilter 而重新計算
1121
+ const RenderEdgeMap$: Observable<Map<string, RenderEdge>> = combineLatest({
1122
+ computedData: observer.computedData$,
1123
+ strokeWidthScale: strokeWidthScale$,
1124
+ defaultEdgeValue: defaultEdgeValue$
1125
+ }).pipe(
1126
+ takeUntil(destroy$),
1127
+ switchMap(async (d) => d),
1128
+ map(data => {
1129
+ return new Map(
1130
+ data.computedData.edges.map(_d => {
1131
+ let d: any = _d as RenderEdge
1132
+ d.source = _d.startNode // reference
1133
+ d.target = _d.endNode
1134
+ d.strokeWidth = data.strokeWidthScale(d.value ?? data.defaultEdgeValue)
1135
+ d.markerId = `${defsArrowMarkerId}__${d.id}`
1136
+ return [d.id, d]
1137
+ })
1138
+ )
1139
+ }),
1140
+ )
1141
+
1142
+ const renderData$: Observable<RenderData> = combineLatest({
1143
+ visibleComputedData: observer.visibleComputedData$,
1144
+ RenderNodeMap: RenderNodeMap$,
1145
+ RenderEdgeMap: RenderEdgeMap$,
1146
+ }).pipe(
1147
+ takeUntil(destroy$),
1148
+ switchMap(async (d) => d),
1149
+ map(data => {
1150
+ return {
1151
+ nodes: data.visibleComputedData.nodes.map(d => data.RenderNodeMap.get(d.id)!),
1152
+ edges: data.visibleComputedData.edges.map(d => data.RenderEdgeMap.get(d.id)!),
1153
+ }
1154
+ }),
1155
+ shareReplay(1)
1156
+ )
1157
+
1158
+ const markerParams$: Observable<MarkerParams> = observer.fullParams$.pipe(
1159
+ takeUntil(destroy$),
1160
+ map(fullParams => {
1161
+ return {
1162
+ viewBox: `-${fullParams.arrow.pointerWidth} -${fullParams.arrow.pointerHeight / 2} ${fullParams.arrow.pointerWidth} ${fullParams.arrow.pointerHeight}`,
1163
+ d: `M${-fullParams.arrow.pointerWidth},${-fullParams.arrow.pointerHeight / 2}L0,0L${-fullParams.arrow.pointerWidth},${fullParams.arrow.pointerHeight / 2}`, // 箭頭的尖端為(0,0)
1164
+ pointerWidth: fullParams.arrow.pointerWidth,
1165
+ pointerHeight: fullParams.arrow.pointerHeight,
1166
+ }
1167
+ })
1168
+ )
1169
+
1170
+ const markerData$: Observable<MarkerDatum[]> = combineLatest({
1171
+ computedData: observer.computedData$,
1172
+ fullParams: observer.fullParams$,
1173
+ RenderNodeMap: RenderNodeMap$,
1174
+ RenderEdgeMap: RenderEdgeMap$,
1175
+ }).pipe(
1176
+ takeUntil(destroy$),
1177
+ switchMap(async d => d),
1178
+ map(data => {
1179
+ return data.computedData.edges.map(d => {
1180
+ const renderEdge = data.RenderEdgeMap.get(d.id)!
1181
+ const renderEndNode = data.RenderNodeMap.get(d.endNode.id)!
1182
+ return {
1183
+ id: renderEdge.markerId,
1184
+ edgeId: d.id,
1185
+ strokeWidth: renderEdge.strokeWidth,
1186
+ /* refX:修正marker位置(計算出和circle半徑相等的寬度)
1187
+ (1)circle半徑需加上 strokeWidth/2 是因為框線是以 circle 的邊緣往內及往外擴展,所以 stroke 多出來的寬度是一半而已
1188
+ (2)circle半徑需除以 path 寬度是因為「marker 的位置會受到 path 的stroke-width影響」,所以要進行修正
1189
+ (3)- 1 是要修正奇怪的誤差(不知原因)
1190
+ */
1191
+ refX: ((renderEndNode.r + (data.fullParams.bubble.strokeWidth / 2)) / renderEdge.strokeWidth) - 1
1192
+ }
1193
+ })
1194
+ }),
1195
+ )
1196
+
1197
+ // <marker> marker selection
1198
+ combineLatest({
1199
+ defsSelection,
1200
+ markerParams: markerParams$,
1201
+ markerData: markerData$
1202
+ }).pipe(
1203
+ takeUntil(destroy$),
1204
+ map(data => {
1205
+ return renderArrowMarker({
1206
+ defsSelection,
1207
+ markerParams: data.markerParams,
1208
+ markerData: data.markerData
1209
+ })
1210
+ })
1211
+ ).subscribe()
1212
+
1213
+
1214
+ combineLatest({
1215
+ renderData: renderData$,
1216
+ computedData: observer.computedData$,
1217
+ CategoryNodeMap: observer.CategoryNodeMap$,
1218
+ simulation: simulation$,
1219
+ fullParams: observer.fullParams$,
1220
+ fullChartParams: observer.fullChartParams$
1221
+ }).pipe(
1222
+ takeUntil(destroy$),
1223
+ switchMap(async d => d),
1224
+ ).subscribe(data => {
1225
+
1226
+ nodeGSelection = renderNodeG({
1227
+ nodeListGSelection: nodeListGSelection,
1228
+ nodes: data.renderData.nodes as RenderNode[],
1229
+ })
1230
+
1231
+ nodeCircleSelection = renderNodeCircle({
1232
+ nodeGSelection: nodeGSelection,
1233
+ fullParams: data.fullParams,
1234
+ fullChartParams: data.fullChartParams
1235
+ })
1236
+ nodeGSelection.call(drag(data.simulation, dragStatus$))
1237
+
1238
+ // nodeLabelGSelection = renderNodeLabelG({
1239
+ // nodeGSelection: nodeGSelection,
1240
+ // })
1241
+
1242
+ // nodeLabelSelection = renderNodeLabel({
1243
+ // nodeLabelGSelection: nodeLabelGSelection,
1244
+ // fullParams: data.fullParams,
1245
+ // fullChartParams: data.fullChartParams
1246
+ // })
1247
+
1248
+ edgeGSelection = renderEdgeG({
1249
+ edgeListGSelection: edgeListGSelection,
1250
+ edges: data.renderData.edges
1251
+ })
1252
+
1253
+ edgeArrowSelection = renderEdgeArrowPath({
1254
+ edgeGSelection: edgeGSelection,
1255
+ fullParams: data.fullParams,
1256
+ fullChartParams: data.fullChartParams
1257
+ })
1258
+
1259
+ edgeLabelGSelection = renderEdgeLabelG({
1260
+ edgeGSelection: edgeGSelection,
1261
+ })
1262
+
1263
+ edgeLabelSelection = renderEdgeLabel({
1264
+ edgeLabelGSelection: edgeLabelGSelection,
1265
+ fullParams: data.fullParams,
1266
+ fullChartParams: data.fullChartParams
1267
+ })
1268
+
1269
+ data.simulation.nodes(data.renderData.nodes)
1270
+ .on('tick', () => {
1271
+ edgeArrowSelection.attr('d', linkArcFn)
1272
+ nodeGSelection.attr('transform', translateFn)
1273
+ // nodeLabelGSelection.attr('transform', d => translateFn({
1274
+ // x: d.x,
1275
+ // y: d.y - d.r - 10
1276
+ // }))
1277
+ edgeLabelGSelection.attr('transform', d => translateCenterFn(d))
1278
+ })
1279
+ ;(data.simulation.force("link") as any).links(data.renderData.edges)
1280
+
1281
+ data.simulation.alpha(0.3).restart()
1282
+
1283
+ nodeCircleSelection
1284
+ .on('mouseover', (event, datum) => {
1285
+ event.stopPropagation()
1286
+
1287
+ mouseEvent$.next({
1288
+ type: 'relationship',
1289
+ eventName: 'mouseover',
1290
+ pluginName,
1291
+ highlightTarget: data.fullChartParams.highlightTarget,
1292
+ datum: datum,
1293
+ category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1294
+ categoryIndex: datum.categoryIndex,
1295
+ categoryLabel: datum.categoryLabel,
1296
+ event,
1297
+ data: data.computedData
1298
+ })
1299
+ })
1300
+ .on('mousemove', (event, datum) => {
1301
+ event.stopPropagation()
1302
+
1303
+ mouseEvent$.next({
1304
+ type: 'relationship',
1305
+ eventName: 'mousemove',
1306
+ pluginName,
1307
+ highlightTarget: data.fullChartParams.highlightTarget,
1308
+ datum: datum,
1309
+ category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1310
+ categoryIndex: datum.categoryIndex,
1311
+ categoryLabel: datum.categoryLabel,
1312
+ event,
1313
+ data: data.computedData
1314
+ })
1315
+ })
1316
+ .on('mouseout', (event, datum) => {
1317
+ event.stopPropagation()
1318
+
1319
+ mouseEvent$.next({
1320
+ type: 'relationship',
1321
+ eventName: 'mouseout',
1322
+ pluginName,
1323
+ highlightTarget: data.fullChartParams.highlightTarget,
1324
+ datum: datum,
1325
+ category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1326
+ categoryIndex: datum.categoryIndex,
1327
+ categoryLabel: datum.categoryLabel,
1328
+ event,
1329
+ data: data.computedData
1330
+ })
1331
+ })
1332
+ .on('click', (event, datum) => {
1333
+ event.stopPropagation()
1334
+
1335
+ mouseEvent$.next({
1336
+ type: 'relationship',
1337
+ eventName: 'click',
1338
+ pluginName,
1339
+ highlightTarget: data.fullChartParams.highlightTarget,
1340
+ datum: datum,
1341
+ category: data.CategoryNodeMap.get(datum.categoryLabel)!,
1342
+ categoryIndex: datum.categoryIndex,
1343
+ categoryLabel: datum.categoryLabel,
1344
+ event,
1345
+ data: data.computedData
1346
+ })
1347
+ })
1348
+ })
1349
+
1350
+ dragStatus$.pipe(
1351
+ distinctUntilChanged((a, b) => a === b),
1352
+ // 只有沒有托曳時才執行
1353
+ switchMap(d => iif(() => d === 'end', mouseEvent$, EMPTY))
1354
+ ).subscribe(data => {
1355
+ subject.event$.next(data)
1356
+ })
1357
+
1358
+ combineLatest({
1359
+ renderData: renderData$,
1360
+ highlightNodes: observer.relationshipHighlightNodes$.pipe(
1361
+ map(data => data.map(d => d.id))
1362
+ ),
1363
+ highlightEdges: observer.relationshipHighlightEdges$.pipe(
1364
+ map(data => data.map(d => d.id))
1365
+ ),
1366
+ fullChartParams: observer.fullChartParams$,
1367
+ fullParams: observer.fullParams$,
1368
+ }).pipe(
1369
+ takeUntil(destroy$),
1370
+ switchMap(async d => d)
1371
+ ).subscribe(data => {
1372
+ if (!nodeGSelection || !edgeGSelection) {
1373
+ return
1374
+ }
1375
+
1376
+ highlightNodes({
1377
+ nodeGSelection,
1378
+ edgeGSelection,
1379
+ highlightIds: data.highlightNodes,
1380
+ fullChartParams: data.fullChartParams
1381
+ })
1382
+ // highlightEdges({
1383
+ // edgeGSelection,
1384
+ // highlightIds: data.highlightEdges,
1385
+ // fullChartParams: data.fullChartParams
1386
+ // })
1387
+ })
1388
+
1389
+
1390
+
1391
+ return () => {
1392
+ destroy$.next(undefined)
1393
+ }
1391
1394
  })