@orbcharts/plugins-basic 3.0.0-beta.6 → 3.0.0-beta.8

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