@logicflow/extension 2.2.0-alpha.7 → 2.2.1
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.
- package/README.md +16 -0
- package/package.json +10 -7
- package/.turbo/turbo-build.log +0 -38
- package/CHANGELOG.md +0 -1829
- package/__test__/bpmn-adapter.test.js +0 -227
- package/es/materials/curved-edge/__test__/curved-edge.test.d.ts +0 -1
- package/es/materials/curved-edge/__test__/curved-edge.test.js +0 -18
- package/jest.config.js +0 -198
- package/lib/materials/curved-edge/__test__/curved-edge.test.d.ts +0 -1
- package/lib/materials/curved-edge/__test__/curved-edge.test.js +0 -20
- package/rollup.config.js +0 -52
- package/src/NodeResize/BasicShape/Ellipse.tsx +0 -22
- package/src/NodeResize/BasicShape/Polygon.tsx +0 -24
- package/src/NodeResize/BasicShape/Rect.tsx +0 -44
- package/src/NodeResize/control/Control.tsx +0 -537
- package/src/NodeResize/control/ControlGroup.tsx +0 -76
- package/src/NodeResize/control/Util.ts +0 -206
- package/src/NodeResize/index.ts +0 -26
- package/src/NodeResize/node/DiamondResize.tsx +0 -149
- package/src/NodeResize/node/EllipseResize.tsx +0 -140
- package/src/NodeResize/node/HtmlResize.tsx +0 -125
- package/src/NodeResize/node/RectResize.tsx +0 -126
- package/src/NodeResize/node/index.ts +0 -4
- package/src/bpmn/constant.ts +0 -56
- package/src/bpmn/events/EndEvent.ts +0 -73
- package/src/bpmn/events/StartEvent.ts +0 -52
- package/src/bpmn/events/index.ts +0 -2
- package/src/bpmn/flow/SequenceFlow.ts +0 -25
- package/src/bpmn/flow/index.ts +0 -1
- package/src/bpmn/gateways/ExclusiveGateway.ts +0 -71
- package/src/bpmn/gateways/index.ts +0 -1
- package/src/bpmn/getBpmnId.ts +0 -31
- package/src/bpmn/index.ts +0 -60
- package/src/bpmn/tasks/ServiceTask.ts +0 -63
- package/src/bpmn/tasks/UserTask.ts +0 -64
- package/src/bpmn/tasks/index.ts +0 -2
- package/src/bpmn-adapter/bpmnIds.ts +0 -31
- package/src/bpmn-adapter/index.ts +0 -835
- package/src/bpmn-adapter/json2xml.ts +0 -127
- package/src/bpmn-adapter/xml2json.ts +0 -544
- package/src/bpmn-elements/README.md +0 -223
- package/src/bpmn-elements/__tests__/definition.test.js +0 -72
- package/src/bpmn-elements/index.d.ts +0 -26
- package/src/bpmn-elements/index.ts +0 -107
- package/src/bpmn-elements/presets/Event/EndEventFactory.ts +0 -114
- package/src/bpmn-elements/presets/Event/IntermediateCatchEvent.ts +0 -108
- package/src/bpmn-elements/presets/Event/IntermediateThrowEvent.ts +0 -109
- package/src/bpmn-elements/presets/Event/StartEventFactory.ts +0 -114
- package/src/bpmn-elements/presets/Event/boundaryEventFactory.ts +0 -117
- package/src/bpmn-elements/presets/Event/index.ts +0 -14
- package/src/bpmn-elements/presets/Flow/flow.d.ts +0 -6
- package/src/bpmn-elements/presets/Flow/index.ts +0 -8
- package/src/bpmn-elements/presets/Flow/manhattan.ts +0 -691
- package/src/bpmn-elements/presets/Flow/sequenceFlow.ts +0 -65
- package/src/bpmn-elements/presets/Gateway/gateway.ts +0 -107
- package/src/bpmn-elements/presets/Gateway/index.ts +0 -23
- package/src/bpmn-elements/presets/Pool/Lane.ts +0 -211
- package/src/bpmn-elements/presets/Pool/Pool.ts +0 -284
- package/src/bpmn-elements/presets/Pool/index.ts +0 -89
- package/src/bpmn-elements/presets/Task/index.ts +0 -122
- package/src/bpmn-elements/presets/Task/subProcess.ts +0 -189
- package/src/bpmn-elements/presets/Task/task.ts +0 -193
- package/src/bpmn-elements/presets/icons.ts +0 -155
- package/src/bpmn-elements/utils.ts +0 -52
- package/src/bpmn-elements-adapter/README.md +0 -293
- package/src/bpmn-elements-adapter/__tests__/adapter_in.test.js +0 -528
- package/src/bpmn-elements-adapter/__tests__/adapter_out.test.js +0 -569
- package/src/bpmn-elements-adapter/constant.ts +0 -76
- package/src/bpmn-elements-adapter/index.ts +0 -1134
- package/src/bpmn-elements-adapter/json2xml.ts +0 -105
- package/src/bpmn-elements-adapter/xml2json.ts +0 -542
- package/src/components/context-menu/index.ts +0 -253
- package/src/components/control/index.ts +0 -155
- package/src/components/dnd-panel/index.ts +0 -137
- package/src/components/highlight/index.ts +0 -227
- package/src/components/menu/index.ts +0 -748
- package/src/components/mini-map/index.ts +0 -686
- package/src/components/selection-select/index.ts +0 -387
- package/src/dynamic-group/index.ts +0 -774
- package/src/dynamic-group/model.ts +0 -580
- package/src/dynamic-group/node.ts +0 -288
- package/src/dynamic-group/utils.ts +0 -46
- package/src/index.less +0 -1
- package/src/index.ts +0 -47
- package/src/insert-node-in-polyline/edge.ts +0 -175
- package/src/insert-node-in-polyline/index.ts +0 -193
- package/src/materials/curved-edge/__test__/curved-edge.test.ts +0 -46
- package/src/materials/curved-edge/index.ts +0 -217
- package/src/materials/group/GroupNode.ts +0 -437
- package/src/materials/group/index.ts +0 -542
- package/src/materials/node-selection/index.ts +0 -380
- package/src/mindmap/fakerRoot.ts +0 -19
- package/src/mindmap/index.ts +0 -328
- package/src/mindmap/markContent.ts +0 -81
- package/src/mindmap/markContentOption.ts +0 -81
- package/src/mindmap/markEntity.ts +0 -82
- package/src/mindmap/markRoot.ts +0 -83
- package/src/mindmap/theme.ts +0 -11
- package/src/pool/LaneModel.ts +0 -226
- package/src/pool/LaneView.ts +0 -220
- package/src/pool/PoolModel.ts +0 -631
- package/src/pool/PoolView.ts +0 -75
- package/src/pool/constant.ts +0 -19
- package/src/pool/index.ts +0 -621
- package/src/pool/utils.ts +0 -46
- package/src/rect-label-node/RectLabelNodeView.ts +0 -33
- package/src/rect-label-node/index.ts +0 -15
- package/src/style/index.less +0 -381
- package/src/style/raw.ts +0 -328
- package/src/tools/auto-layout/index.ts +0 -282
- package/src/tools/flow-path/index.ts +0 -233
- package/src/tools/label/Label.tsx +0 -357
- package/src/tools/label/LabelModel.ts +0 -83
- package/src/tools/label/LabelOverlay.tsx +0 -162
- package/src/tools/label/algorithm.ts +0 -42
- package/src/tools/label/index.ts +0 -479
- package/src/tools/label/mediumEditor.ts +0 -121
- package/src/tools/label/utils.ts +0 -395
- package/src/tools/proximity-connect/index.ts +0 -435
- package/src/tools/snapshot/README.md +0 -145
- package/src/tools/snapshot/index.ts +0 -701
- package/src/tools/snapshot/utils.ts +0 -163
- package/src/turbo-adapter/index.ts +0 -212
- package/stats.html +0 -4842
- package/tsconfig.json +0 -18
|
@@ -1,774 +0,0 @@
|
|
|
1
|
-
import LogicFlow, {
|
|
2
|
-
BaseEdgeModel,
|
|
3
|
-
BaseNodeModel,
|
|
4
|
-
EventType,
|
|
5
|
-
CallbackArgs,
|
|
6
|
-
Model,
|
|
7
|
-
transformNodeData,
|
|
8
|
-
transformEdgeData,
|
|
9
|
-
} from '@logicflow/core'
|
|
10
|
-
import { assign, cloneDeep, filter, forEach, has, map, sortBy } from 'lodash-es'
|
|
11
|
-
import { DynamicGroupNode } from './node'
|
|
12
|
-
import { DynamicGroupNodeModel } from './model'
|
|
13
|
-
import { isAllowMoveTo, isBoundsInGroup } from './utils'
|
|
14
|
-
|
|
15
|
-
import GraphConfigData = LogicFlow.GraphConfigData
|
|
16
|
-
import GraphElements = LogicFlow.GraphElements
|
|
17
|
-
import EdgeConfig = LogicFlow.EdgeConfig
|
|
18
|
-
import EdgeData = LogicFlow.EdgeData
|
|
19
|
-
import NodeData = LogicFlow.NodeData
|
|
20
|
-
import BoxBoundsPoint = Model.BoxBoundsPoint
|
|
21
|
-
import ElementsInfoInGroup = DynamicGroup.ElementsInfoInGroup
|
|
22
|
-
|
|
23
|
-
export * from './node'
|
|
24
|
-
export * from './model'
|
|
25
|
-
|
|
26
|
-
export const dynamicGroup = {
|
|
27
|
-
type: 'dynamic-group',
|
|
28
|
-
view: DynamicGroupNode,
|
|
29
|
-
model: DynamicGroupNodeModel,
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
const DEFAULT_TOP_Z_INDEX = -1000
|
|
33
|
-
const DEFAULT_BOTTOM_Z_INDEX = -10000
|
|
34
|
-
|
|
35
|
-
export class DynamicGroup {
|
|
36
|
-
static pluginName = 'dynamicGroup'
|
|
37
|
-
|
|
38
|
-
private lf: LogicFlow
|
|
39
|
-
topGroupZIndex: number = DEFAULT_BOTTOM_Z_INDEX
|
|
40
|
-
// 激活态的 group 节点
|
|
41
|
-
activeGroup?: DynamicGroupNodeModel
|
|
42
|
-
// 存储节点与 group 的映射关系
|
|
43
|
-
nodeGroupMap: Map<string, string> = new Map()
|
|
44
|
-
|
|
45
|
-
constructor({ lf, options }: LogicFlow.IExtensionProps) {
|
|
46
|
-
lf.register(dynamicGroup)
|
|
47
|
-
this.lf = lf
|
|
48
|
-
assign(this, options)
|
|
49
|
-
// 初始化插件,从监听事件开始及设置规则开始
|
|
50
|
-
this.init()
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
/**
|
|
54
|
-
* 获取节点所属的分组
|
|
55
|
-
* @param nodeId
|
|
56
|
-
*/
|
|
57
|
-
getGroupByNodeId(nodeId: string) {
|
|
58
|
-
const groupId = this.nodeGroupMap.get(nodeId)
|
|
59
|
-
if (groupId) {
|
|
60
|
-
return this.lf.getNodeModelById(groupId)
|
|
61
|
-
}
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
/**
|
|
65
|
-
* 获取自定位置及其所属分组
|
|
66
|
-
* 当分组重合时,优先返回最上层的分组
|
|
67
|
-
* @param bounds
|
|
68
|
-
* @param nodeData
|
|
69
|
-
*/
|
|
70
|
-
getGroupByBounds(
|
|
71
|
-
bounds: BoxBoundsPoint,
|
|
72
|
-
nodeData: NodeData,
|
|
73
|
-
): DynamicGroupNodeModel | undefined {
|
|
74
|
-
const { nodes } = this.lf.graphModel
|
|
75
|
-
const groups = filter(nodes, (node) => {
|
|
76
|
-
return (
|
|
77
|
-
!!node.isGroup &&
|
|
78
|
-
isBoundsInGroup(bounds, node) &&
|
|
79
|
-
node.id !== nodeData.id
|
|
80
|
-
)
|
|
81
|
-
})
|
|
82
|
-
|
|
83
|
-
const count = groups.length
|
|
84
|
-
if (count <= 1) {
|
|
85
|
-
return groups[0] as DynamicGroupNodeModel
|
|
86
|
-
} else {
|
|
87
|
-
let topZIndexGroup = groups[count - 1]
|
|
88
|
-
for (let i = count - 2; i >= 0; i--) {
|
|
89
|
-
if (groups[i].zIndex > topZIndexGroup.zIndex) {
|
|
90
|
-
topZIndexGroup = groups[i]
|
|
91
|
-
}
|
|
92
|
-
}
|
|
93
|
-
return topZIndexGroup as DynamicGroupNodeModel
|
|
94
|
-
}
|
|
95
|
-
}
|
|
96
|
-
|
|
97
|
-
/**
|
|
98
|
-
* 提高元素的层级,如果是 group,同时提高其子元素的层级
|
|
99
|
-
* @param model
|
|
100
|
-
*/
|
|
101
|
-
sendNodeToFront(model?: BaseNodeModel) {
|
|
102
|
-
if (!model || !model.isGroup) return
|
|
103
|
-
|
|
104
|
-
this.topGroupZIndex++
|
|
105
|
-
model.setZIndex(this.topGroupZIndex)
|
|
106
|
-
if (model.children) {
|
|
107
|
-
const { children } = model as DynamicGroupNodeModel
|
|
108
|
-
forEach(Array.from(children), (nodeId) => {
|
|
109
|
-
const node = this.lf.getNodeModelById(nodeId)
|
|
110
|
-
this.sendNodeToFront(node)
|
|
111
|
-
})
|
|
112
|
-
}
|
|
113
|
-
}
|
|
114
|
-
|
|
115
|
-
/**
|
|
116
|
-
* 递归计算某个分组内最高的 zIndex 值
|
|
117
|
-
* TODO: 这块儿有点疑问❓如果 node 不是 group,这块儿返回的 maxZIndex 是最小值,但 node 的 zIndex 不一定是这个值
|
|
118
|
-
* @param node
|
|
119
|
-
*/
|
|
120
|
-
getMaxZIndex(node: BaseNodeModel) {
|
|
121
|
-
let maxZIndex = DEFAULT_BOTTOM_Z_INDEX
|
|
122
|
-
if (node.isGroup) {
|
|
123
|
-
maxZIndex = Math.max(maxZIndex, node.zIndex)
|
|
124
|
-
}
|
|
125
|
-
if (node.children) {
|
|
126
|
-
const { children } = node as DynamicGroupNodeModel
|
|
127
|
-
forEach(Array.from(children), (childId) => {
|
|
128
|
-
const child = this.lf.getNodeModelById(childId)
|
|
129
|
-
if (child?.isGroup) {
|
|
130
|
-
const childMaxZIndex = this.getMaxZIndex(child)
|
|
131
|
-
maxZIndex = Math.max(maxZIndex, childMaxZIndex)
|
|
132
|
-
}
|
|
133
|
-
})
|
|
134
|
-
}
|
|
135
|
-
return maxZIndex
|
|
136
|
-
}
|
|
137
|
-
|
|
138
|
-
/**
|
|
139
|
-
* 校准当前 topGroupZIndex 的值
|
|
140
|
-
* @param nodes
|
|
141
|
-
*/
|
|
142
|
-
calibrateTopGroupZIndex(nodes: NodeData[]) {
|
|
143
|
-
// 初始化时 or 增加新节点时,找出所有 nodes 的最大 zIndex
|
|
144
|
-
let maxZIndex = DEFAULT_BOTTOM_Z_INDEX
|
|
145
|
-
forEach(nodes, (node) => {
|
|
146
|
-
const nodeModel = this.lf.getNodeModelById(node.id)
|
|
147
|
-
if (nodeModel) {
|
|
148
|
-
const currNodeMaxZIndex = this.getMaxZIndex(nodeModel)
|
|
149
|
-
if (currNodeMaxZIndex > maxZIndex) {
|
|
150
|
-
maxZIndex = currNodeMaxZIndex
|
|
151
|
-
}
|
|
152
|
-
}
|
|
153
|
-
})
|
|
154
|
-
|
|
155
|
-
// TODO: 不是很理解这块儿的代码逻辑,需要整理一下
|
|
156
|
-
if (this.topGroupZIndex >= maxZIndex) {
|
|
157
|
-
// 一般是初始化时/增加新节点时发生,因为外部强行设置了一个很大的 zIndex
|
|
158
|
-
// 删除节点不会影响目前最高 zIndex 的赋值
|
|
159
|
-
return
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
// 新增 nodes 中如果存在 zIndex 比 this.topGroupZIndex 大
|
|
163
|
-
// 说明 this.topGroupZIndex 已经失去意义,代表不了目前最高 zIndex 的 group,需要重新校准
|
|
164
|
-
|
|
165
|
-
// https://github.com/didi/LogicFlow/issues/1535
|
|
166
|
-
// 当外部直接设置多个 BaseNode.zIndex = 1 时
|
|
167
|
-
// 当点击某一个 node 时,由于这个 this.topGroupZIndex 是从 -10000 开始计算的,
|
|
168
|
-
// this.topGroupZIndex + 1 也就是-9999,这就造成当前点击的 node 的 zIndex 远远
|
|
169
|
-
// 比其它 node 的 zIndex 小,因此造成 zIndex 错乱的问题
|
|
170
|
-
// TODO: 这儿的 nodes 能否直接用传参进来的 nodes 呢???
|
|
171
|
-
const allNodes = this.lf.graphModel.nodes
|
|
172
|
-
const allGroups = filter(allNodes, (node) => !!node.isGroup)
|
|
173
|
-
|
|
174
|
-
let max = this.topGroupZIndex
|
|
175
|
-
forEach(allGroups, (group) => {
|
|
176
|
-
if (group.zIndex > max) max = group.zIndex
|
|
177
|
-
})
|
|
178
|
-
this.topGroupZIndex = max
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
onSelectionDrop = () => {
|
|
182
|
-
const { nodes: selectedNodes } = this.lf.graphModel.getSelectElements()
|
|
183
|
-
selectedNodes.forEach((node) => {
|
|
184
|
-
this.addNodeToGroup(node)
|
|
185
|
-
})
|
|
186
|
-
}
|
|
187
|
-
|
|
188
|
-
onNodeAddOrDrop = ({ data: node }: CallbackArgs<'node:add'>) => {
|
|
189
|
-
this.addNodeToGroup(node)
|
|
190
|
-
}
|
|
191
|
-
|
|
192
|
-
addNodeToGroup = (node: LogicFlow.NodeData) => {
|
|
193
|
-
// 1. 如果该节点之前已经在 group 中了,则将其从之前的 group 移除
|
|
194
|
-
const preGroupId = this.nodeGroupMap.get(node.id)
|
|
195
|
-
|
|
196
|
-
if (preGroupId) {
|
|
197
|
-
const group = this.lf.getNodeModelById(
|
|
198
|
-
preGroupId,
|
|
199
|
-
) as DynamicGroupNodeModel
|
|
200
|
-
|
|
201
|
-
group.removeChild(node.id)
|
|
202
|
-
this.nodeGroupMap.delete(node.id)
|
|
203
|
-
group.setAllowAppendChild(false)
|
|
204
|
-
}
|
|
205
|
-
|
|
206
|
-
// 2. 然后再判断这个节点是否在某个 group 范围内,如果是,则将其添加到对应的 group 中
|
|
207
|
-
const nodeModel = this.lf.getNodeModelById(node.id)
|
|
208
|
-
const bounds = nodeModel?.getBounds()
|
|
209
|
-
|
|
210
|
-
if (nodeModel && bounds) {
|
|
211
|
-
// TODO: 确认下面的注释内容
|
|
212
|
-
// https://github.com/didi/LogicFlow/issues/1261
|
|
213
|
-
// 当使用 SelectionSelect 框选后触发 lf.addNode(Group)
|
|
214
|
-
// 会触发 appendNodeToGroup() 的执行
|
|
215
|
-
// 由于 this.getGroup() 会判断 node.id !== nodeData.id
|
|
216
|
-
// 因此当 addNode 是 Group 类型时,this.getGroup() 会一直返回空
|
|
217
|
-
// 导致了下面这段代码无法执行,也就是无法将当前添加的 Group 添加到 this.nodeGroupMap 中
|
|
218
|
-
// 这导致了折叠分组时触发的 foldEdge() 无法正确通过 getNodeGroup() 拿到正确的 groupId
|
|
219
|
-
// 从而导致折叠分组时一直都会创建一个虚拟边
|
|
220
|
-
// 而初始化分组时由于正确设置了nodeGroupMap的数据,因此不会产生虚拟边的错误情况
|
|
221
|
-
if (nodeModel.isGroup) {
|
|
222
|
-
const group = nodeModel as DynamicGroupNodeModel
|
|
223
|
-
forEach(Array.from(group.children), (childId) => {
|
|
224
|
-
this.nodeGroupMap.set(childId, node.id)
|
|
225
|
-
})
|
|
226
|
-
// 新增 node 时进行 this.topGroupZIndex 的校准更新
|
|
227
|
-
this.calibrateTopGroupZIndex([node])
|
|
228
|
-
this.onNodeSelect({
|
|
229
|
-
data: node,
|
|
230
|
-
isSelected: false,
|
|
231
|
-
isMultiple: false,
|
|
232
|
-
})
|
|
233
|
-
}
|
|
234
|
-
|
|
235
|
-
// TODO: 找到这个范围内的 groupModel, 并加 node 添加到该 group
|
|
236
|
-
const group = this.getGroupByBounds(bounds, node)
|
|
237
|
-
if (group) {
|
|
238
|
-
const isAllowAppendIn = group.isAllowAppendIn(node)
|
|
239
|
-
if (isAllowAppendIn) {
|
|
240
|
-
group.addChild(node.id)
|
|
241
|
-
// 建立节点与 group 的映射关系放在了 group.addChild 触发的事件中,与直接调用 addChild 的行为保持一致
|
|
242
|
-
group.setAllowAppendChild(false)
|
|
243
|
-
} else {
|
|
244
|
-
// 抛出不允许插入的事件
|
|
245
|
-
this.lf.emit('group:not-allowed', {
|
|
246
|
-
group: group.getData(),
|
|
247
|
-
node,
|
|
248
|
-
})
|
|
249
|
-
}
|
|
250
|
-
}
|
|
251
|
-
}
|
|
252
|
-
}
|
|
253
|
-
|
|
254
|
-
onGroupAddNode = ({
|
|
255
|
-
data: groupData,
|
|
256
|
-
childId,
|
|
257
|
-
}: CallbackArgs<'group:add-node'>) => {
|
|
258
|
-
this.nodeGroupMap.set(childId, groupData.id)
|
|
259
|
-
}
|
|
260
|
-
|
|
261
|
-
removeNodeFromGroup = ({
|
|
262
|
-
data: node,
|
|
263
|
-
model,
|
|
264
|
-
}: CallbackArgs<'node:delete'>) => {
|
|
265
|
-
if (model.isGroup && node.children) {
|
|
266
|
-
forEach(
|
|
267
|
-
Array.from((node as DynamicGroupNodeModel).children),
|
|
268
|
-
(childId) => {
|
|
269
|
-
this.nodeGroupMap.delete(childId)
|
|
270
|
-
this.lf.deleteNode(childId)
|
|
271
|
-
},
|
|
272
|
-
)
|
|
273
|
-
}
|
|
274
|
-
|
|
275
|
-
const groupId = this.nodeGroupMap.get(node.id)
|
|
276
|
-
if (groupId) {
|
|
277
|
-
const group = this.lf.getNodeModelById(groupId)
|
|
278
|
-
group && (group as DynamicGroupNodeModel).removeChild(node.id)
|
|
279
|
-
this.nodeGroupMap.delete(node.id)
|
|
280
|
-
}
|
|
281
|
-
}
|
|
282
|
-
|
|
283
|
-
onSelectionDrag = () => {
|
|
284
|
-
const { nodes: selectedNodes } = this.lf.graphModel.getSelectElements()
|
|
285
|
-
selectedNodes.forEach((node) => {
|
|
286
|
-
this.setActiveGroup(node)
|
|
287
|
-
})
|
|
288
|
-
}
|
|
289
|
-
onNodeDrag = ({ data: node }: CallbackArgs<'node:drag'>) => {
|
|
290
|
-
this.setActiveGroup(node)
|
|
291
|
-
}
|
|
292
|
-
|
|
293
|
-
setActiveGroup = (node: LogicFlow.NodeData) => {
|
|
294
|
-
const nodeModel = this.lf.getNodeModelById(node.id)
|
|
295
|
-
const bounds = nodeModel?.getBounds()
|
|
296
|
-
|
|
297
|
-
if (nodeModel && bounds) {
|
|
298
|
-
const targetGroup = this.getGroupByBounds(bounds, node)
|
|
299
|
-
if (this.activeGroup) {
|
|
300
|
-
this.activeGroup.setAllowAppendChild(false)
|
|
301
|
-
}
|
|
302
|
-
|
|
303
|
-
if (!targetGroup || (nodeModel.isGroup && targetGroup.id === node.id)) {
|
|
304
|
-
return
|
|
305
|
-
}
|
|
306
|
-
|
|
307
|
-
const isAllowAppendIn = targetGroup.isAllowAppendIn(node)
|
|
308
|
-
if (!isAllowAppendIn) return
|
|
309
|
-
|
|
310
|
-
this.activeGroup = targetGroup
|
|
311
|
-
this.activeGroup.setAllowAppendChild(true)
|
|
312
|
-
}
|
|
313
|
-
}
|
|
314
|
-
|
|
315
|
-
/**
|
|
316
|
-
* 1. 分组节点默认在普通节点下面
|
|
317
|
-
* 2. 分组节点被选中后,会将分组节点以及其内部的其它分组节点放到其余分组节点的上面
|
|
318
|
-
* 3. 分组节点取消选中后,不会将分组节点重置为原来的高度
|
|
319
|
-
* 4. 由于 LogicFlow 核心目标是支持用户手动绘制流程图,所以暂时不支持一张流程图超过 1000 个分组节点的情况
|
|
320
|
-
* @param node
|
|
321
|
-
* @param isMultiple
|
|
322
|
-
* @param isSelected
|
|
323
|
-
*/
|
|
324
|
-
onNodeSelect = ({
|
|
325
|
-
data: node,
|
|
326
|
-
isMultiple,
|
|
327
|
-
isSelected,
|
|
328
|
-
}: Omit<CallbackArgs<'node:click'>, 'e' | 'position'>) => {
|
|
329
|
-
const nodeModel = this.lf.getNodeModelById(node.id)
|
|
330
|
-
this.sendNodeToFront(nodeModel)
|
|
331
|
-
// 重置所有 group 的 zIndex,防止 group 节点 zIndex 增长为正数(目的是保持 group 节点在最底层)
|
|
332
|
-
if (this.topGroupZIndex > DEFAULT_TOP_Z_INDEX) {
|
|
333
|
-
const { nodes } = this.lf.graphModel
|
|
334
|
-
this.topGroupZIndex = DEFAULT_BOTTOM_Z_INDEX
|
|
335
|
-
const groups = sortBy(
|
|
336
|
-
filter(nodes, (node) => !!node.isGroup),
|
|
337
|
-
'zIndex',
|
|
338
|
-
)
|
|
339
|
-
|
|
340
|
-
let preZIndex = 0
|
|
341
|
-
forEach(groups, (group) => {
|
|
342
|
-
if (group.zIndex !== preZIndex) {
|
|
343
|
-
this.topGroupZIndex++
|
|
344
|
-
preZIndex = group.zIndex
|
|
345
|
-
}
|
|
346
|
-
group.setZIndex(this.topGroupZIndex)
|
|
347
|
-
})
|
|
348
|
-
}
|
|
349
|
-
|
|
350
|
-
// FIX #1004
|
|
351
|
-
// 如果节点被多选,
|
|
352
|
-
// 这个节点是分组,则将分组的所有子节点取消选中
|
|
353
|
-
// 这个节点是分组的子节点,且其所属分组节点已选,则取消选中
|
|
354
|
-
if (isMultiple && isSelected) {
|
|
355
|
-
if (nodeModel?.isGroup) {
|
|
356
|
-
const { children } = nodeModel as DynamicGroupNodeModel
|
|
357
|
-
forEach(Array.from(children), (childId) => {
|
|
358
|
-
const childModel = this.lf.getNodeModelById(childId)
|
|
359
|
-
childModel?.setSelected(false)
|
|
360
|
-
})
|
|
361
|
-
} else {
|
|
362
|
-
const groupId = this.nodeGroupMap.get(node.id)
|
|
363
|
-
if (groupId) {
|
|
364
|
-
const graphModel = this.lf.getNodeModelById(groupId)
|
|
365
|
-
graphModel?.isSelected && nodeModel?.setSelected(false)
|
|
366
|
-
}
|
|
367
|
-
}
|
|
368
|
-
}
|
|
369
|
-
}
|
|
370
|
-
|
|
371
|
-
onNodeMove = ({
|
|
372
|
-
deltaX,
|
|
373
|
-
deltaY,
|
|
374
|
-
data,
|
|
375
|
-
}: Omit<CallbackArgs<'node:mousemove'>, 'e' | 'position'>) => {
|
|
376
|
-
const { id, x, y, properties } = data
|
|
377
|
-
if (!properties) {
|
|
378
|
-
return
|
|
379
|
-
}
|
|
380
|
-
const { width, height } = properties
|
|
381
|
-
const groupId = this.nodeGroupMap.get(id)
|
|
382
|
-
if (!groupId) {
|
|
383
|
-
return
|
|
384
|
-
}
|
|
385
|
-
const groupModel = this.lf.getNodeModelById(
|
|
386
|
-
groupId,
|
|
387
|
-
) as DynamicGroupNodeModel
|
|
388
|
-
|
|
389
|
-
if (!groupModel || !groupModel.isRestrict || !groupModel.autoResize) {
|
|
390
|
-
return
|
|
391
|
-
}
|
|
392
|
-
// 当父节点isRestrict=true & autoResize=true
|
|
393
|
-
// 子节点在父节点中移动时,父节点会自动调整大小
|
|
394
|
-
|
|
395
|
-
// step1: 计算出当前child的bounds
|
|
396
|
-
const newX = x + deltaX / 2
|
|
397
|
-
const newY = y + deltaY / 2
|
|
398
|
-
const minX = newX - width! / 2
|
|
399
|
-
const minY = newY - height! / 2
|
|
400
|
-
const maxX = newX + width! / 2
|
|
401
|
-
const maxY = newY + height! / 2
|
|
402
|
-
// step2:比较当前child.bounds与parent.bounds的差异,比如child.minX<parent.minX,那么parent.minX=child.minX
|
|
403
|
-
let hasChange = false
|
|
404
|
-
const groupBounds = groupModel.getBounds()
|
|
405
|
-
const newGroupBounds = Object.assign({}, groupBounds)
|
|
406
|
-
if (minX < newGroupBounds.minX) {
|
|
407
|
-
newGroupBounds.minX = minX
|
|
408
|
-
hasChange = true
|
|
409
|
-
}
|
|
410
|
-
if (minY < newGroupBounds.minY) {
|
|
411
|
-
newGroupBounds.minY = minY
|
|
412
|
-
hasChange = true
|
|
413
|
-
}
|
|
414
|
-
if (maxX > newGroupBounds.maxX) {
|
|
415
|
-
newGroupBounds.maxX = maxX
|
|
416
|
-
hasChange = true
|
|
417
|
-
}
|
|
418
|
-
if (maxY > newGroupBounds.maxY) {
|
|
419
|
-
newGroupBounds.maxY = maxY
|
|
420
|
-
hasChange = true
|
|
421
|
-
}
|
|
422
|
-
if (!hasChange) {
|
|
423
|
-
return
|
|
424
|
-
}
|
|
425
|
-
// step3: 根据当前parent.bounds去计算出最新的x、y、width、height
|
|
426
|
-
const newGroupX =
|
|
427
|
-
newGroupBounds.minX + (newGroupBounds.maxX - newGroupBounds.minX) / 2
|
|
428
|
-
const newGroupY =
|
|
429
|
-
newGroupBounds.minY + (newGroupBounds.maxY - newGroupBounds.minY) / 2
|
|
430
|
-
const newGroupWidth = newGroupBounds.maxX - newGroupBounds.minX
|
|
431
|
-
const newGroupHeight = newGroupBounds.maxY - newGroupBounds.minY
|
|
432
|
-
groupModel.moveTo(newGroupX, newGroupY)
|
|
433
|
-
groupModel.width = newGroupWidth
|
|
434
|
-
groupModel.height = newGroupHeight
|
|
435
|
-
}
|
|
436
|
-
|
|
437
|
-
onGraphRendered = ({ data }: CallbackArgs<'graph:rendered'>) => {
|
|
438
|
-
forEach(data.nodes, (node) => {
|
|
439
|
-
if (node.children) {
|
|
440
|
-
forEach(node.children, (childId) => {
|
|
441
|
-
this.nodeGroupMap.set(childId, node.id)
|
|
442
|
-
})
|
|
443
|
-
}
|
|
444
|
-
})
|
|
445
|
-
|
|
446
|
-
// TODO: 确认一下下面方法的必要性及合理性
|
|
447
|
-
// 初始化 nodes 时进行 this.topGroupZIndex 的校准更新
|
|
448
|
-
this.calibrateTopGroupZIndex(data.nodes)
|
|
449
|
-
}
|
|
450
|
-
|
|
451
|
-
removeChildrenInGroupNodeData<
|
|
452
|
-
T extends LogicFlow.NodeData | LogicFlow.NodeConfig,
|
|
453
|
-
>(nodeData: T) {
|
|
454
|
-
const newNodeData = cloneDeep(nodeData)
|
|
455
|
-
delete newNodeData.children
|
|
456
|
-
if (newNodeData.properties?.children) {
|
|
457
|
-
delete newNodeData.properties.children
|
|
458
|
-
}
|
|
459
|
-
return newNodeData
|
|
460
|
-
}
|
|
461
|
-
|
|
462
|
-
/**
|
|
463
|
-
* 创建一个 Group 类型节点内部所有子节点的副本
|
|
464
|
-
* 并且在遍历所有 nodes 的过程中,顺便拿到所有 edges (只在 Group 范围的 edges)
|
|
465
|
-
*/
|
|
466
|
-
initGroupChildNodes(
|
|
467
|
-
nodeIdMap: Record<string, string>,
|
|
468
|
-
children: Set<string>,
|
|
469
|
-
curGroup: DynamicGroupNodeModel,
|
|
470
|
-
distance: number,
|
|
471
|
-
): ElementsInfoInGroup {
|
|
472
|
-
// Group 中所有子节点
|
|
473
|
-
const allChildNodes: BaseNodeModel[] = []
|
|
474
|
-
// 属于 Group 内部边的 EdgeData
|
|
475
|
-
const edgesDataArr: EdgeData[] = []
|
|
476
|
-
// 所有有关联的连线
|
|
477
|
-
const allRelatedEdges: BaseEdgeModel[] = []
|
|
478
|
-
|
|
479
|
-
forEach(Array.from(children), (childId: string) => {
|
|
480
|
-
const childNode = this.lf.getNodeModelById(childId)
|
|
481
|
-
if (childNode) {
|
|
482
|
-
const childNodeChildren = childNode.children
|
|
483
|
-
const childNodeData = childNode.getData()
|
|
484
|
-
const eventType = EventType.NODE_GROUP_COPY || 'node:group-copy-add'
|
|
485
|
-
|
|
486
|
-
const newNodeConfig = transformNodeData(
|
|
487
|
-
this.removeChildrenInGroupNodeData(childNodeData),
|
|
488
|
-
distance,
|
|
489
|
-
)
|
|
490
|
-
const tempChildNode = this.lf.addNode(newNodeConfig, eventType)
|
|
491
|
-
curGroup.addChild(tempChildNode.id)
|
|
492
|
-
|
|
493
|
-
nodeIdMap[childId] = tempChildNode.id // id 同 childId,做映射存储
|
|
494
|
-
allChildNodes.push(tempChildNode)
|
|
495
|
-
|
|
496
|
-
// 1. 存储 children 内部节点相关的输入边(incoming)
|
|
497
|
-
allRelatedEdges.push(
|
|
498
|
-
...[...tempChildNode.incoming.edges, ...tempChildNode.outgoing.edges],
|
|
499
|
-
)
|
|
500
|
-
|
|
501
|
-
if (childNodeChildren instanceof Set) {
|
|
502
|
-
const { childNodes, edgesData } = this.initGroupChildNodes(
|
|
503
|
-
nodeIdMap,
|
|
504
|
-
childNodeChildren,
|
|
505
|
-
tempChildNode as DynamicGroupNodeModel,
|
|
506
|
-
distance,
|
|
507
|
-
)
|
|
508
|
-
|
|
509
|
-
allChildNodes.push(...childNodes)
|
|
510
|
-
edgesDataArr.push(...edgesData)
|
|
511
|
-
}
|
|
512
|
-
}
|
|
513
|
-
})
|
|
514
|
-
|
|
515
|
-
// 1. 判断每一条边的开始节点、目标节点是否在 Group 中
|
|
516
|
-
const edgesInnerGroup = filter(allRelatedEdges, (edge) => {
|
|
517
|
-
return (
|
|
518
|
-
has(nodeIdMap, edge.sourceNodeId) && has(nodeIdMap, edge.targetNodeId)
|
|
519
|
-
)
|
|
520
|
-
})
|
|
521
|
-
// 2. 为「每一条 Group 的内部边」构建出 EdgeData 数据,得到 EdgeConfig,生成新的线
|
|
522
|
-
const edgesDataInnerGroup = map(edgesInnerGroup, (edge) => {
|
|
523
|
-
return edge.getData()
|
|
524
|
-
})
|
|
525
|
-
|
|
526
|
-
return {
|
|
527
|
-
childNodes: allChildNodes,
|
|
528
|
-
edgesData: edgesDataArr.concat(edgesDataInnerGroup),
|
|
529
|
-
}
|
|
530
|
-
}
|
|
531
|
-
|
|
532
|
-
/**
|
|
533
|
-
* 根据参数 edge 选择是新建边还是基于已有边,复制一条边出来
|
|
534
|
-
* @param edge
|
|
535
|
-
* @param nodeIdMap
|
|
536
|
-
* @param distance
|
|
537
|
-
*/
|
|
538
|
-
createEdge(
|
|
539
|
-
edge: EdgeConfig | EdgeData,
|
|
540
|
-
nodeIdMap: Record<string, string>,
|
|
541
|
-
distance: number,
|
|
542
|
-
) {
|
|
543
|
-
const { sourceNodeId, targetNodeId } = edge
|
|
544
|
-
const sourceId = nodeIdMap[sourceNodeId] ?? sourceNodeId
|
|
545
|
-
const targetId = nodeIdMap[targetNodeId] ?? targetNodeId
|
|
546
|
-
|
|
547
|
-
// 如果是有 id 且 text 是对象的边,需要重新计算位置,否则直接用 edgeConfig 生成边
|
|
548
|
-
let newEdgeConfig = cloneDeep(edge)
|
|
549
|
-
if (edge.id && typeof edge.text === 'object' && edge.text !== null) {
|
|
550
|
-
newEdgeConfig = transformEdgeData(edge as EdgeData, distance)
|
|
551
|
-
}
|
|
552
|
-
|
|
553
|
-
return this.lf.graphModel.addEdge({
|
|
554
|
-
...newEdgeConfig,
|
|
555
|
-
sourceNodeId: sourceId,
|
|
556
|
-
targetNodeId: targetId,
|
|
557
|
-
})
|
|
558
|
-
}
|
|
559
|
-
|
|
560
|
-
/**
|
|
561
|
-
* 检测group:resize后的bounds是否会小于children的bounds
|
|
562
|
-
* 限制group进行resize时不能小于内部的占地面积
|
|
563
|
-
* @param groupModel
|
|
564
|
-
* @param deltaX
|
|
565
|
-
* @param deltaY
|
|
566
|
-
* @param newWidth
|
|
567
|
-
* @param newHeight
|
|
568
|
-
*/
|
|
569
|
-
checkGroupBoundsWithChildren(
|
|
570
|
-
groupModel: DynamicGroupNodeModel,
|
|
571
|
-
deltaX: number,
|
|
572
|
-
deltaY: number,
|
|
573
|
-
newWidth: number,
|
|
574
|
-
newHeight: number,
|
|
575
|
-
) {
|
|
576
|
-
if (groupModel.children) {
|
|
577
|
-
const { children, x, y } = groupModel
|
|
578
|
-
// 根据deltaX和deltaY计算出当前model的bounds
|
|
579
|
-
const newX = x + deltaX / 2
|
|
580
|
-
const newY = y + deltaY / 2
|
|
581
|
-
const groupMinX = newX - newWidth / 2
|
|
582
|
-
const groupMinY = newY - newHeight / 2
|
|
583
|
-
const groupMaxX = newX + newWidth / 2
|
|
584
|
-
const groupMaxY = newY + newHeight / 2
|
|
585
|
-
|
|
586
|
-
const childrenArray = Array.from(children)
|
|
587
|
-
for (let i = 0; i < childrenArray.length; i++) {
|
|
588
|
-
const childId = childrenArray[i]
|
|
589
|
-
const child = this.lf.getNodeModelById(childId)
|
|
590
|
-
if (!child) {
|
|
591
|
-
continue
|
|
592
|
-
}
|
|
593
|
-
const childBounds = child.getBounds()
|
|
594
|
-
const { minX, minY, maxX, maxY } = childBounds
|
|
595
|
-
// parent:resize后的bounds不能小于child:bounds,否则阻止其resize
|
|
596
|
-
const canResize =
|
|
597
|
-
groupMinX <= minX &&
|
|
598
|
-
groupMinY <= minY &&
|
|
599
|
-
groupMaxX >= maxX &&
|
|
600
|
-
groupMaxY >= maxY
|
|
601
|
-
if (!canResize) {
|
|
602
|
-
return false
|
|
603
|
-
}
|
|
604
|
-
}
|
|
605
|
-
}
|
|
606
|
-
|
|
607
|
-
return true
|
|
608
|
-
}
|
|
609
|
-
|
|
610
|
-
/**
|
|
611
|
-
* Group 插件的初始化方法
|
|
612
|
-
* TODO:1. 待讨论,可能之前插件分类是有意义的 components, material, tools
|
|
613
|
-
* 区别是:1. 有些插件就是自定义节点,可能会有初始化方法 init,但不必要有 render (比如 Group)
|
|
614
|
-
* 2. 有些插件是渲染一些部件(比如 MiniMap、Control、Menu 等)必须要有 render
|
|
615
|
-
* 3. 工具类的,init 、 render
|
|
616
|
-
* 该如何分类呢?并如何完善插件的类型
|
|
617
|
-
*
|
|
618
|
-
* TODO: 2. 插件的 destroy 方法该做些什么,是否应该加 destroy 方法
|
|
619
|
-
* TODO: 3. 是否应该定义一个 Extension 的基类,所有插件基于这个基类来开发,这样在初始化的时候就可以确认执行什么方法
|
|
620
|
-
*/
|
|
621
|
-
init() {
|
|
622
|
-
const { lf } = this
|
|
623
|
-
const { graphModel } = lf
|
|
624
|
-
// 添加分组节点移动规则
|
|
625
|
-
// 1. 移动分组节点时,同时移动分组内所有节点
|
|
626
|
-
// 2. 移动子节点时,判断是否有限制规则(isRestrict)
|
|
627
|
-
graphModel.addNodeMoveRules((model, deltaX, deltaY) => {
|
|
628
|
-
// 判断如果是 group,移动时需要同时移动组内的所有节点
|
|
629
|
-
if (model.isGroup) {
|
|
630
|
-
// https://github.com/didi/LogicFlow/issues/1826
|
|
631
|
-
// 这里不应该触发移动子节点的逻辑,这里是判断是否可以移动,而不是触发移动逻辑
|
|
632
|
-
// 而且这里触发移动,会导致resize操作的this.x变动也会触发子item的this.x变动
|
|
633
|
-
// resize时的deltaX跟正常移动的deltaX是不同的
|
|
634
|
-
|
|
635
|
-
// const nodeIds = this.getNodesInGroup(model as DynamicGroupNodeModel)
|
|
636
|
-
// graphModel.moveNodes(nodeIds, deltaX, deltaY, true)
|
|
637
|
-
return true
|
|
638
|
-
}
|
|
639
|
-
|
|
640
|
-
const groupId = this.nodeGroupMap.get(model.id)!
|
|
641
|
-
const groupModel = this.lf.getNodeModelById(
|
|
642
|
-
groupId,
|
|
643
|
-
) as DynamicGroupNodeModel
|
|
644
|
-
|
|
645
|
-
if (groupModel && groupModel.isRestrict) {
|
|
646
|
-
if (groupModel.autoResize) {
|
|
647
|
-
// 子节点在父节点中移动时,父节点会自动调整大小
|
|
648
|
-
// 在node:mousemove中进行父节点的调整
|
|
649
|
-
return true
|
|
650
|
-
} else {
|
|
651
|
-
// 如果移动的节点存在于某个分组中,且这个分组禁止子节点移出去
|
|
652
|
-
const groupBounds = groupModel.getBounds()
|
|
653
|
-
return isAllowMoveTo(groupBounds, model, deltaX, deltaY)
|
|
654
|
-
}
|
|
655
|
-
}
|
|
656
|
-
|
|
657
|
-
return true
|
|
658
|
-
})
|
|
659
|
-
|
|
660
|
-
// https://github.com/didi/LogicFlow/issues/1442
|
|
661
|
-
// https://github.com/didi/LogicFlow/issues/937
|
|
662
|
-
// 添加分组节点resize规则
|
|
663
|
-
// isRestrict限制模式下,当前model resize时不能小于children的占地面积
|
|
664
|
-
// 并且在isRestrict限制模式下,transformWidthContainer即使设置为true,也无效
|
|
665
|
-
graphModel.addNodeResizeRules((model, deltaX, deltaY, width, height) => {
|
|
666
|
-
if (model.isGroup && model.isRestrict) {
|
|
667
|
-
return this.checkGroupBoundsWithChildren(
|
|
668
|
-
model as DynamicGroupNodeModel,
|
|
669
|
-
deltaX,
|
|
670
|
-
deltaY,
|
|
671
|
-
width,
|
|
672
|
-
height,
|
|
673
|
-
)
|
|
674
|
-
}
|
|
675
|
-
return true
|
|
676
|
-
})
|
|
677
|
-
|
|
678
|
-
graphModel.dynamicGroup = this
|
|
679
|
-
lf.on('node:add,node:drop,node:dnd-add', this.onNodeAddOrDrop)
|
|
680
|
-
lf.on('selection:drop', this.onSelectionDrop)
|
|
681
|
-
lf.on('node:delete', this.removeNodeFromGroup)
|
|
682
|
-
lf.on('node:drag,node:dnd-drag', this.onNodeDrag)
|
|
683
|
-
lf.on('selection:drag', this.onSelectionDrag)
|
|
684
|
-
lf.on('node:click', this.onNodeSelect)
|
|
685
|
-
lf.on('node:mousemove', this.onNodeMove)
|
|
686
|
-
lf.on('graph:rendered', this.onGraphRendered)
|
|
687
|
-
|
|
688
|
-
lf.on('group:add-node', this.onGroupAddNode)
|
|
689
|
-
|
|
690
|
-
// https://github.com/didi/LogicFlow/issues/1346
|
|
691
|
-
// 重写 addElements() 方法,在 addElements() 原有基础上增加对 group 内部所有 nodes 和 edges 的复制功能
|
|
692
|
-
// 使用场景:addElements api 项目内部目前只在快捷键粘贴时使用(此处解决的也应该是粘贴场景的问题)
|
|
693
|
-
lf.addElements = (
|
|
694
|
-
{ nodes: selectedNodes, edges: selectedEdges }: GraphConfigData,
|
|
695
|
-
distance = 40,
|
|
696
|
-
): GraphElements => {
|
|
697
|
-
// oldNodeId -> newNodeId 映射 Map
|
|
698
|
-
const nodeIdMap: Record<string, string> = {}
|
|
699
|
-
// 本次添加的所有节点和边
|
|
700
|
-
const elements: GraphElements = {
|
|
701
|
-
nodes: [],
|
|
702
|
-
edges: [],
|
|
703
|
-
}
|
|
704
|
-
// 所有属于分组内的边 -> sourceNodeId 和 targetNodeId 都在 Group 内
|
|
705
|
-
const edgesInnerGroup: EdgeData[] = []
|
|
706
|
-
|
|
707
|
-
forEach(selectedNodes, (node) => {
|
|
708
|
-
const originId = node.id
|
|
709
|
-
const children = node.properties?.children ?? node.children
|
|
710
|
-
|
|
711
|
-
const model = lf.addNode(this.removeChildrenInGroupNodeData(node))
|
|
712
|
-
|
|
713
|
-
if (originId) nodeIdMap[originId] = model.id
|
|
714
|
-
elements.nodes.push(model) // 此时为 group 的 nodeModel
|
|
715
|
-
|
|
716
|
-
// TODO: 递归创建 group 的 nodeModel 的 children
|
|
717
|
-
if (model.isGroup) {
|
|
718
|
-
const { edgesData } = this.initGroupChildNodes(
|
|
719
|
-
nodeIdMap,
|
|
720
|
-
children,
|
|
721
|
-
model as DynamicGroupNodeModel,
|
|
722
|
-
distance,
|
|
723
|
-
)
|
|
724
|
-
edgesInnerGroup.push(...edgesData)
|
|
725
|
-
}
|
|
726
|
-
})
|
|
727
|
-
|
|
728
|
-
forEach(edgesInnerGroup, (edge) => {
|
|
729
|
-
this.createEdge(edge, nodeIdMap, distance)
|
|
730
|
-
})
|
|
731
|
-
forEach(selectedEdges, (edge) => {
|
|
732
|
-
elements.edges.push(this.createEdge(edge, nodeIdMap, distance))
|
|
733
|
-
})
|
|
734
|
-
|
|
735
|
-
// 返回 elements 进行选中效果,即触发 element.selectElementById()
|
|
736
|
-
// shortcut.ts 也会对最外层的 nodes 和 edges 进行偏移,即 translationNodeData()
|
|
737
|
-
return elements
|
|
738
|
-
}
|
|
739
|
-
|
|
740
|
-
this.render()
|
|
741
|
-
}
|
|
742
|
-
|
|
743
|
-
render() {}
|
|
744
|
-
|
|
745
|
-
destroy() {
|
|
746
|
-
// 销毁监听的事件,并移除渲染的 dom 内容
|
|
747
|
-
this.lf.off('node:add,node:drop,node:dnd-add', this.onNodeAddOrDrop)
|
|
748
|
-
this.lf.off('selection:drop', this.onSelectionDrop)
|
|
749
|
-
this.lf.off('node:delete', this.removeNodeFromGroup)
|
|
750
|
-
this.lf.off('node:drag,node:dnd-drag', this.onNodeDrag)
|
|
751
|
-
this.lf.off('selection:drag', this.onSelectionDrag)
|
|
752
|
-
this.lf.off('node:click', this.onNodeSelect)
|
|
753
|
-
this.lf.off('node:mousemove', this.onNodeMove)
|
|
754
|
-
this.lf.off('graph:rendered', this.onGraphRendered)
|
|
755
|
-
this.lf.off('group:add-node', this.onGroupAddNode)
|
|
756
|
-
|
|
757
|
-
// 还原 lf.addElements 方法?
|
|
758
|
-
// 移除 graphModel 上重写的 addNodeMoveRules 方法?
|
|
759
|
-
// TODO: 讨论一下插件该具体做些什么
|
|
760
|
-
}
|
|
761
|
-
}
|
|
762
|
-
|
|
763
|
-
export namespace DynamicGroup {
|
|
764
|
-
export type ElementsInfoInGroup = {
|
|
765
|
-
childNodes: BaseNodeModel[] // 分组节点的所有子节点 model
|
|
766
|
-
edgesData: EdgeData[] // 属于分组内的线的 EdgeData (即开始节点和结束节点都在 Group 内)
|
|
767
|
-
}
|
|
768
|
-
|
|
769
|
-
export type DynamicGroupOptions = Partial<{
|
|
770
|
-
isCollapsed: boolean
|
|
771
|
-
}>
|
|
772
|
-
}
|
|
773
|
-
|
|
774
|
-
export default DynamicGroup
|