@flowgram.ai/free-node-panel-plugin 0.1.0

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.
@@ -0,0 +1,513 @@
1
+ var __defProp = Object.defineProperty;
2
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
3
+ var __decorateClass = (decorators, target, key, kind) => {
4
+ var result = kind > 1 ? void 0 : kind ? __getOwnPropDesc(target, key) : target;
5
+ for (var i = decorators.length - 1, decorator; i >= 0; i--)
6
+ if (decorator = decorators[i])
7
+ result = (kind ? decorator(target, key, result) : decorator(result)) || result;
8
+ if (kind && result) __defProp(target, key, result);
9
+ return result;
10
+ };
11
+
12
+ // src/create-plugin.ts
13
+ import {
14
+ definePluginCreator
15
+ } from "@flowgram.ai/core";
16
+
17
+ // src/service.ts
18
+ import { inject, injectable } from "inversify";
19
+ import { delay, DisposableCollection, Rectangle } from "@flowgram.ai/utils";
20
+ import {
21
+ WorkflowDocument,
22
+ WorkflowDragService,
23
+ WorkflowLinesManager,
24
+ WorkflowNodePortsData
25
+ } from "@flowgram.ai/free-layout-core";
26
+ import { WorkflowSelectService } from "@flowgram.ai/free-layout-core";
27
+ import { FreeOperationType, HistoryService } from "@flowgram.ai/free-history-plugin";
28
+ import { FlowNodeTransformData } from "@flowgram.ai/document";
29
+ import { FlowNodeBaseType } from "@flowgram.ai/document";
30
+ import { PlaygroundConfigEntity } from "@flowgram.ai/core";
31
+ import { TransformData } from "@flowgram.ai/core";
32
+
33
+ // src/utils.ts
34
+ var isGreaterThan = (a, b) => {
35
+ if (a === void 0 || b === void 0) {
36
+ return false;
37
+ }
38
+ const EPSILON = 1e-5;
39
+ return a - b > EPSILON;
40
+ };
41
+ var isLessThan = (a, b) => {
42
+ if (a === void 0 || b === void 0) {
43
+ return false;
44
+ }
45
+ const EPSILON = 1e-5;
46
+ return b - a > EPSILON;
47
+ };
48
+
49
+ // src/service.ts
50
+ var WorkflowNodePanelService = class {
51
+ constructor() {
52
+ this.toDispose = new DisposableCollection();
53
+ this.callNodePanel = async () => void 0;
54
+ }
55
+ /** 销毁 */
56
+ dispose() {
57
+ this.toDispose.dispose();
58
+ }
59
+ setCallNodePanel(callNodePanel) {
60
+ this.callNodePanel = callNodePanel;
61
+ }
62
+ /** 唤起节点面板 */
63
+ async call(callParams) {
64
+ const {
65
+ panelPosition,
66
+ fromPort,
67
+ enableMultiAdd = false,
68
+ panelProps = {},
69
+ containerNode,
70
+ afterAddNode
71
+ } = callParams;
72
+ if (!panelPosition || this.playgroundConfig.readonly) {
73
+ return;
74
+ }
75
+ const nodes = [];
76
+ return new Promise((resolve) => {
77
+ this.callNodePanel({
78
+ position: panelPosition,
79
+ enableMultiAdd,
80
+ panelProps,
81
+ containerNode: this.getContainerNode({
82
+ fromPort,
83
+ containerNode
84
+ }),
85
+ onSelect: async (panelParams) => {
86
+ const node = await this.addNode(callParams, panelParams);
87
+ afterAddNode?.(node);
88
+ if (!enableMultiAdd) {
89
+ resolve(node);
90
+ } else if (node) {
91
+ nodes.push(node);
92
+ }
93
+ },
94
+ onClose: () => {
95
+ resolve(enableMultiAdd ? nodes : void 0);
96
+ }
97
+ });
98
+ });
99
+ }
100
+ /** 添加节点 */
101
+ async addNode(callParams, panelParams) {
102
+ const {
103
+ panelPosition,
104
+ fromPort,
105
+ toPort,
106
+ canAddNode,
107
+ autoOffsetPadding = {
108
+ x: 100,
109
+ y: 100
110
+ },
111
+ enableBuildLine = false,
112
+ enableSelectPosition = false,
113
+ enableAutoOffset = false,
114
+ enableDragNode = false
115
+ } = callParams;
116
+ if (!panelPosition || !panelParams) {
117
+ return;
118
+ }
119
+ const { nodeType, selectEvent, nodeJSON } = panelParams;
120
+ const containerNode = this.getContainerNode({
121
+ fromPort,
122
+ containerNode: callParams.containerNode
123
+ });
124
+ if (canAddNode) {
125
+ const canAdd = canAddNode({ nodeType, containerNode });
126
+ if (!canAdd) {
127
+ return;
128
+ }
129
+ }
130
+ const selectPosition = this.playgroundConfig.getPosFromMouseEvent(selectEvent);
131
+ const nodePosition = callParams.customPosition ? callParams.customPosition({ nodeType, selectPosition }) : this.adjustNodePosition({
132
+ nodeType,
133
+ position: enableSelectPosition ? selectPosition : panelPosition,
134
+ fromPort,
135
+ toPort,
136
+ containerNode
137
+ });
138
+ const node = await this.document.createWorkflowNodeByType(
139
+ nodeType,
140
+ nodePosition,
141
+ nodeJSON ?? {},
142
+ containerNode?.id
143
+ );
144
+ if (!node) {
145
+ return;
146
+ }
147
+ if (enableAutoOffset && fromPort && toPort) {
148
+ const subOffset = this.subPositionOffset({
149
+ node,
150
+ fromPort,
151
+ toPort,
152
+ padding: autoOffsetPadding
153
+ });
154
+ const subsequentNodes = this.getSubsequentNodes(toPort.node);
155
+ this.updateSubSequentNodesPosition({
156
+ node,
157
+ subsequentNodes,
158
+ fromPort,
159
+ toPort,
160
+ containerNode,
161
+ offset: subOffset
162
+ });
163
+ }
164
+ if (!enableBuildLine && !enableDragNode) {
165
+ return node;
166
+ }
167
+ await delay(20);
168
+ if (enableBuildLine) {
169
+ this.buildLine({
170
+ fromPort,
171
+ node,
172
+ toPort
173
+ });
174
+ }
175
+ if (enableDragNode) {
176
+ this.selectService.selectNode(node);
177
+ this.dragService.startDragSelectedNodes(selectEvent);
178
+ }
179
+ return node;
180
+ }
181
+ /** 建立连线 */
182
+ buildLine(params) {
183
+ const { fromPort, node, toPort } = params;
184
+ const portsData = node.getData(WorkflowNodePortsData);
185
+ if (!portsData) {
186
+ return;
187
+ }
188
+ const shouldBuildFromLine = portsData.inputPorts?.length > 0;
189
+ if (fromPort && shouldBuildFromLine) {
190
+ const toTargetPort = portsData.inputPorts[0];
191
+ const isSingleInput = portsData.inputPorts.length === 1;
192
+ this.linesManager.createLine({
193
+ from: fromPort.node.id,
194
+ fromPort: fromPort.portID,
195
+ to: node.id,
196
+ toPort: isSingleInput ? void 0 : toTargetPort.id
197
+ });
198
+ }
199
+ const shouldBuildToLine = portsData.outputPorts?.length > 0;
200
+ if (toPort && shouldBuildToLine) {
201
+ const fromTargetPort = portsData.outputPorts[0];
202
+ this.linesManager.createLine({
203
+ from: node.id,
204
+ fromPort: fromTargetPort.portID,
205
+ to: toPort.node.id,
206
+ toPort: toPort.portID
207
+ });
208
+ }
209
+ }
210
+ /** 调整节点坐标 */
211
+ adjustNodePosition(params) {
212
+ const { nodeType, position, fromPort, toPort, containerNode } = params;
213
+ const register = this.document.getNodeRegistry(nodeType);
214
+ const size = register?.meta?.size;
215
+ let adjustedPosition = position;
216
+ if (!size) {
217
+ adjustedPosition = position;
218
+ } else if (fromPort && toPort) {
219
+ adjustedPosition = {
220
+ x: position.x,
221
+ y: position.y - size.height / 2
222
+ };
223
+ } else if (fromPort && !toPort) {
224
+ adjustedPosition = {
225
+ x: position.x + size.width / 2,
226
+ y: position.y - size.height / 2
227
+ };
228
+ } else if (!fromPort && toPort) {
229
+ adjustedPosition = {
230
+ x: position.x - size.width / 2,
231
+ y: position.y - size.height / 2
232
+ };
233
+ } else {
234
+ adjustedPosition = position;
235
+ }
236
+ return this.dragService.adjustSubNodePosition(nodeType, containerNode, adjustedPosition);
237
+ }
238
+ getContainerNode(params) {
239
+ const { fromPort, containerNode } = params;
240
+ if (containerNode) {
241
+ return containerNode;
242
+ }
243
+ const fromNode = fromPort?.node;
244
+ const fromContainer = fromNode?.parent;
245
+ if (fromNode?.flowNodeType === FlowNodeBaseType.SUB_CANVAS) {
246
+ return fromNode;
247
+ }
248
+ return fromContainer;
249
+ }
250
+ /** 获取端口矩形 */
251
+ getPortBox(port, offset = { x: 0, y: 0 }) {
252
+ const node = port.node;
253
+ if (node.flowNodeType === FlowNodeBaseType.SUB_CANVAS) {
254
+ const { point } = port;
255
+ if (port.portType === "input") {
256
+ return new Rectangle(point.x + offset.x, point.y - 50 + offset.y, 300, 100);
257
+ }
258
+ return new Rectangle(point.x - 300, point.y - 50, 300, 100);
259
+ }
260
+ const box = node.getData(FlowNodeTransformData).bounds;
261
+ return box;
262
+ }
263
+ /** 后续节点位置偏移 */
264
+ subPositionOffset(params) {
265
+ const { node, fromPort, toPort, padding } = params;
266
+ const fromBox = this.getPortBox(fromPort);
267
+ const toBox = this.getPortBox(toPort);
268
+ const nodeTrans = node.getData(FlowNodeTransformData);
269
+ const nodeSize = node.getNodeMeta()?.size ?? {
270
+ width: nodeTrans.bounds.width,
271
+ height: nodeTrans.bounds.height
272
+ };
273
+ const minDistance = {
274
+ x: nodeSize.width + padding.x,
275
+ y: nodeSize.height + padding.y
276
+ };
277
+ const boxDistance = this.rectDistance(fromBox, toBox);
278
+ const neededOffset = {
279
+ x: isGreaterThan(boxDistance.x, minDistance.x) ? 0 : minDistance.x - boxDistance.x,
280
+ y: isGreaterThan(boxDistance.y, minDistance.y) ? 0 : minDistance.y - boxDistance.y
281
+ };
282
+ if (neededOffset.x === 0 || neededOffset.y === 0) {
283
+ return;
284
+ }
285
+ const intersection = {
286
+ // 这里没有写反,Rectangle内置的算法是反的
287
+ vertical: Rectangle.intersects(fromBox, toBox, "horizontal"),
288
+ horizontal: Rectangle.intersects(fromBox, toBox, "vertical")
289
+ };
290
+ let offsetX = 0;
291
+ let offsetY = 0;
292
+ if (!intersection.horizontal) {
293
+ if (isGreaterThan(toBox.center.y, fromBox.center.y)) {
294
+ offsetY = neededOffset.y;
295
+ } else if (isLessThan(toBox.center.y, fromBox.center.y)) {
296
+ offsetY = -neededOffset.y;
297
+ }
298
+ }
299
+ if (!intersection.vertical) {
300
+ if (isGreaterThan(toBox.center.x, fromBox.center.x)) {
301
+ offsetX = neededOffset.x;
302
+ } else if (isLessThan(toBox.center.x, fromBox.center.x)) {
303
+ offsetX = -neededOffset.x;
304
+ }
305
+ }
306
+ return {
307
+ x: offsetX,
308
+ y: offsetY
309
+ };
310
+ }
311
+ /** 矩形间距 */
312
+ rectDistance(rectA, rectB) {
313
+ const distanceX = Math.abs(
314
+ Math.min(rectA.right, rectB.right) - Math.max(rectA.left, rectB.left)
315
+ );
316
+ const distanceY = Math.abs(
317
+ Math.min(rectA.bottom, rectB.bottom) - Math.max(rectA.top, rectB.top)
318
+ );
319
+ if (Rectangle.intersects(rectA, rectB)) {
320
+ return {
321
+ x: -distanceX,
322
+ y: -distanceY
323
+ };
324
+ }
325
+ return {
326
+ x: distanceX,
327
+ y: distanceY
328
+ };
329
+ }
330
+ /** 获取后续节点 */
331
+ getSubsequentNodes(node) {
332
+ if (node.flowNodeType === FlowNodeBaseType.SUB_CANVAS) {
333
+ return [];
334
+ }
335
+ const brothers = node.parent?.collapsedChildren ?? [];
336
+ const linkedBrothers = /* @__PURE__ */ new Set();
337
+ const linesMap = /* @__PURE__ */ new Map();
338
+ this.linesManager.getAllLines().forEach((line) => {
339
+ if (!linesMap.has(line.from.id)) {
340
+ linesMap.set(line.from.id, []);
341
+ }
342
+ if (!line.to?.id || line.to.flowNodeType === FlowNodeBaseType.SUB_CANVAS) {
343
+ return;
344
+ }
345
+ linesMap.get(line.from.id)?.push(line.to.id);
346
+ });
347
+ const bfs = (nodeId) => {
348
+ if (linkedBrothers.has(nodeId)) {
349
+ return;
350
+ }
351
+ linkedBrothers.add(nodeId);
352
+ const nextNodes = linesMap.get(nodeId) ?? [];
353
+ nextNodes.forEach(bfs);
354
+ };
355
+ bfs(node.id);
356
+ const subsequentNodes = brothers.filter((node2) => linkedBrothers.has(node2.id));
357
+ return subsequentNodes;
358
+ }
359
+ /** 更新后续节点位置 */
360
+ updateSubSequentNodesPosition(params) {
361
+ const { node, subsequentNodes, fromPort, toPort, containerNode, offset } = params;
362
+ if (!offset || !toPort) {
363
+ return;
364
+ }
365
+ const subsequentNodesPositions = subsequentNodes.map((node2) => {
366
+ const nodeTrans2 = node2.getData(TransformData);
367
+ return {
368
+ x: nodeTrans2.position.x,
369
+ y: nodeTrans2.position.y
370
+ };
371
+ });
372
+ this.historyService.pushOperation({
373
+ type: FreeOperationType.dragNodes,
374
+ value: {
375
+ ids: subsequentNodes.map((node2) => node2.id),
376
+ value: subsequentNodesPositions.map((position) => ({
377
+ x: position.x + offset.x,
378
+ y: position.y + offset.y
379
+ })),
380
+ oldValue: subsequentNodesPositions
381
+ }
382
+ });
383
+ const fromBox = this.getPortBox(fromPort);
384
+ const toBox = this.getPortBox(toPort, offset);
385
+ const nodeTrans = node.getData(TransformData);
386
+ let nodePos = {
387
+ x: (fromBox.center.x + toBox.center.x) / 2,
388
+ y: (fromBox.y + toBox.y) / 2
389
+ };
390
+ if (containerNode) {
391
+ nodePos = this.dragService.adjustSubNodePosition(
392
+ node.flowNodeType,
393
+ containerNode,
394
+ nodePos
395
+ );
396
+ }
397
+ this.historyService.pushOperation({
398
+ type: FreeOperationType.dragNodes,
399
+ value: {
400
+ ids: [node.id],
401
+ value: [nodePos],
402
+ oldValue: [
403
+ {
404
+ x: nodeTrans.position.x,
405
+ y: nodeTrans.position.y
406
+ }
407
+ ]
408
+ }
409
+ });
410
+ }
411
+ };
412
+ __decorateClass([
413
+ inject(WorkflowDocument)
414
+ ], WorkflowNodePanelService.prototype, "document", 2);
415
+ __decorateClass([
416
+ inject(WorkflowDragService)
417
+ ], WorkflowNodePanelService.prototype, "dragService", 2);
418
+ __decorateClass([
419
+ inject(WorkflowSelectService)
420
+ ], WorkflowNodePanelService.prototype, "selectService", 2);
421
+ __decorateClass([
422
+ inject(WorkflowLinesManager)
423
+ ], WorkflowNodePanelService.prototype, "linesManager", 2);
424
+ __decorateClass([
425
+ inject(PlaygroundConfigEntity)
426
+ ], WorkflowNodePanelService.prototype, "playgroundConfig", 2);
427
+ __decorateClass([
428
+ inject(HistoryService)
429
+ ], WorkflowNodePanelService.prototype, "historyService", 2);
430
+ WorkflowNodePanelService = __decorateClass([
431
+ injectable()
432
+ ], WorkflowNodePanelService);
433
+
434
+ // src/layer.tsx
435
+ import React from "react";
436
+ import { inject as inject2 } from "inversify";
437
+ import { Layer } from "@flowgram.ai/core";
438
+ import { nanoid } from "@flowgram.ai/free-layout-core";
439
+ import { domUtils } from "@flowgram.ai/utils";
440
+ var WorkflowNodePanelLayer = class extends Layer {
441
+ constructor() {
442
+ super();
443
+ this.node = domUtils.createDivWithClass("gedit-playground-layer gedit-node-panel-layer");
444
+ this.node.style.zIndex = "9999";
445
+ this.renderList = /* @__PURE__ */ new Map();
446
+ }
447
+ onReady() {
448
+ this.service.setCallNodePanel(this.call.bind(this));
449
+ }
450
+ onZoom(zoom) {
451
+ this.node.style.transform = `scale(${zoom})`;
452
+ }
453
+ render() {
454
+ const NodePanelRender = this.options.renderer;
455
+ return /* @__PURE__ */ React.createElement(React.Fragment, null, Array.from(this.renderList.keys()).map((taskId) => {
456
+ const renderProps = this.renderList.get(taskId);
457
+ return /* @__PURE__ */ React.createElement(NodePanelRender, { key: taskId, ...renderProps });
458
+ }));
459
+ }
460
+ async call(params) {
461
+ const taskId = nanoid();
462
+ const { enableMultiAdd, onSelect, onClose } = params;
463
+ return new Promise((resolve) => {
464
+ const unmount = () => {
465
+ this.renderList.delete(taskId);
466
+ this.render();
467
+ resolve();
468
+ };
469
+ const handleClose = () => {
470
+ unmount();
471
+ onClose();
472
+ };
473
+ const handleSelect = (params2) => {
474
+ onSelect(params2);
475
+ if (!enableMultiAdd) {
476
+ unmount();
477
+ }
478
+ };
479
+ const renderProps = {
480
+ ...params,
481
+ onSelect: handleSelect,
482
+ onClose: handleClose
483
+ };
484
+ this.renderList.set(taskId, renderProps);
485
+ this.render();
486
+ });
487
+ }
488
+ };
489
+ WorkflowNodePanelLayer.type = "WorkflowNodePanelLayer";
490
+ __decorateClass([
491
+ inject2(WorkflowNodePanelService)
492
+ ], WorkflowNodePanelLayer.prototype, "service", 2);
493
+
494
+ // src/create-plugin.ts
495
+ var createFreeNodePanelPlugin = definePluginCreator({
496
+ onBind({ bind }) {
497
+ bind(WorkflowNodePanelService).toSelf().inSingletonScope();
498
+ },
499
+ onInit: (ctx, opts) => {
500
+ ctx.playground.registerLayer(WorkflowNodePanelLayer, {
501
+ renderer: opts.renderer
502
+ });
503
+ },
504
+ onDispose: (ctx) => {
505
+ const nodePanelService = ctx.get(WorkflowNodePanelService);
506
+ nodePanelService.dispose();
507
+ }
508
+ });
509
+ export {
510
+ WorkflowNodePanelService,
511
+ createFreeNodePanelPlugin
512
+ };
513
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/create-plugin.ts","../../src/service.ts","../../src/utils.ts","../../src/layer.tsx"],"sourcesContent":["import {\n definePluginCreator,\n type PluginBindConfig,\n type PluginContext,\n} from '@flowgram.ai/core';\n\nimport { NodePanelPluginOptions } from './type';\nimport { WorkflowNodePanelService } from './service';\nimport { WorkflowNodePanelLayer } from './layer';\n\nexport const createFreeNodePanelPlugin = definePluginCreator({\n onBind({ bind }: PluginBindConfig) {\n bind(WorkflowNodePanelService).toSelf().inSingletonScope();\n },\n onInit: (ctx: PluginContext, opts: NodePanelPluginOptions) => {\n ctx.playground.registerLayer(WorkflowNodePanelLayer, {\n renderer: opts.renderer,\n });\n },\n onDispose: (ctx: PluginContext) => {\n const nodePanelService = ctx.get(WorkflowNodePanelService);\n nodePanelService.dispose();\n },\n});\n","import { inject, injectable } from 'inversify';\nimport { delay, DisposableCollection, Rectangle } from '@flowgram.ai/utils';\nimport type { IPoint, PositionSchema } from '@flowgram.ai/utils';\nimport {\n WorkflowDocument,\n WorkflowDragService,\n WorkflowLinesManager,\n WorkflowPortEntity,\n WorkflowNodePortsData,\n WorkflowNodeEntity,\n} from '@flowgram.ai/free-layout-core';\nimport { WorkflowSelectService } from '@flowgram.ai/free-layout-core';\nimport { WorkflowNodeJSON } from '@flowgram.ai/free-layout-core';\nimport { FreeOperationType, HistoryService } from '@flowgram.ai/free-history-plugin';\nimport { FlowNodeTransformData } from '@flowgram.ai/document';\nimport { FlowNodeBaseType } from '@flowgram.ai/document';\nimport { PlaygroundConfigEntity } from '@flowgram.ai/core';\nimport { TransformData } from '@flowgram.ai/core';\n\nimport { isGreaterThan, isLessThan } from './utils';\nimport type { CallNodePanel, NodePanelCallParams, NodePanelResult } from './type';\n\n/**\n * 添加节点面板服务\n */\n@injectable()\nexport class WorkflowNodePanelService {\n @inject(WorkflowDocument) private readonly document: WorkflowDocument;\n\n @inject(WorkflowDragService)\n private readonly dragService: WorkflowDragService;\n\n @inject(WorkflowSelectService)\n private readonly selectService: WorkflowSelectService;\n\n @inject(WorkflowLinesManager)\n private readonly linesManager: WorkflowLinesManager;\n\n @inject(PlaygroundConfigEntity)\n private readonly playgroundConfig: PlaygroundConfigEntity;\n\n @inject(HistoryService) private readonly historyService: HistoryService;\n\n private readonly toDispose = new DisposableCollection();\n\n private callNodePanel: CallNodePanel = async () => undefined;\n\n /** 销毁 */\n public dispose(): void {\n this.toDispose.dispose();\n }\n\n public setCallNodePanel(callNodePanel: CallNodePanel) {\n this.callNodePanel = callNodePanel;\n }\n\n /** 唤起节点面板 */\n public async call(\n callParams: NodePanelCallParams\n ): Promise<WorkflowNodeEntity | WorkflowNodeEntity[] | undefined> {\n const {\n panelPosition,\n fromPort,\n enableMultiAdd = false,\n panelProps = {},\n containerNode,\n afterAddNode,\n } = callParams;\n\n if (!panelPosition || this.playgroundConfig.readonly) {\n return;\n }\n\n const nodes: WorkflowNodeEntity[] = [];\n\n return new Promise((resolve) => {\n this.callNodePanel({\n position: panelPosition,\n enableMultiAdd,\n panelProps,\n containerNode: this.getContainerNode({\n fromPort,\n containerNode,\n }),\n onSelect: async (panelParams?: NodePanelResult) => {\n const node = await this.addNode(callParams, panelParams);\n afterAddNode?.(node);\n if (!enableMultiAdd) {\n resolve(node);\n } else if (node) {\n nodes.push(node);\n }\n },\n onClose: () => {\n resolve(enableMultiAdd ? nodes : undefined);\n },\n });\n });\n }\n\n /** 添加节点 */\n private async addNode(\n callParams: NodePanelCallParams,\n panelParams: NodePanelResult\n ): Promise<WorkflowNodeEntity | undefined> {\n const {\n panelPosition,\n fromPort,\n toPort,\n canAddNode,\n autoOffsetPadding = {\n x: 100,\n y: 100,\n },\n enableBuildLine = false,\n enableSelectPosition = false,\n enableAutoOffset = false,\n enableDragNode = false,\n } = callParams;\n\n if (!panelPosition || !panelParams) {\n return;\n }\n\n const { nodeType, selectEvent, nodeJSON } = panelParams;\n\n const containerNode = this.getContainerNode({\n fromPort,\n containerNode: callParams.containerNode,\n });\n\n // 判断是否可以添加节点\n if (canAddNode) {\n const canAdd = canAddNode({ nodeType, containerNode });\n if (!canAdd) {\n return;\n }\n }\n\n // 鼠标选择坐标\n const selectPosition = this.playgroundConfig.getPosFromMouseEvent(selectEvent);\n\n // 自定义坐标\n const nodePosition: PositionSchema = callParams.customPosition\n ? callParams.customPosition({ nodeType, selectPosition })\n : this.adjustNodePosition({\n nodeType,\n position: enableSelectPosition ? selectPosition : panelPosition,\n fromPort,\n toPort,\n containerNode,\n });\n\n // 创建节点\n const node: WorkflowNodeEntity = await this.document.createWorkflowNodeByType(\n nodeType,\n nodePosition,\n nodeJSON ?? ({} as WorkflowNodeJSON),\n containerNode?.id\n );\n\n if (!node) {\n return;\n }\n\n // 后续节点偏移\n if (enableAutoOffset && fromPort && toPort) {\n const subOffset = this.subPositionOffset({\n node,\n fromPort,\n toPort,\n padding: autoOffsetPadding,\n });\n const subsequentNodes = this.getSubsequentNodes(toPort.node);\n this.updateSubSequentNodesPosition({\n node,\n subsequentNodes,\n fromPort,\n toPort,\n containerNode,\n offset: subOffset,\n });\n }\n\n if (!enableBuildLine && !enableDragNode) {\n return node;\n }\n\n // 等待节点渲染\n await delay(20);\n\n // 重建连线(需先让端口完成渲染)\n if (enableBuildLine) {\n this.buildLine({\n fromPort,\n node,\n toPort,\n });\n }\n\n // 开始拖拽节点\n if (enableDragNode) {\n this.selectService.selectNode(node);\n this.dragService.startDragSelectedNodes(selectEvent);\n }\n\n return node;\n }\n\n /** 建立连线 */\n private buildLine(params: {\n node: WorkflowNodeEntity;\n fromPort?: WorkflowPortEntity;\n toPort?: WorkflowPortEntity;\n }): void {\n const { fromPort, node, toPort } = params;\n const portsData = node.getData(WorkflowNodePortsData);\n if (!portsData) {\n return;\n }\n\n const shouldBuildFromLine = portsData.inputPorts?.length > 0;\n if (fromPort && shouldBuildFromLine) {\n const toTargetPort = portsData.inputPorts[0];\n const isSingleInput = portsData.inputPorts.length === 1;\n this.linesManager.createLine({\n from: fromPort.node.id,\n fromPort: fromPort.portID,\n to: node.id,\n toPort: isSingleInput ? undefined : toTargetPort.id,\n });\n }\n const shouldBuildToLine = portsData.outputPorts?.length > 0;\n if (toPort && shouldBuildToLine) {\n const fromTargetPort = portsData.outputPorts[0];\n this.linesManager.createLine({\n from: node.id,\n fromPort: fromTargetPort.portID,\n to: toPort.node.id,\n toPort: toPort.portID,\n });\n }\n }\n\n /** 调整节点坐标 */\n private adjustNodePosition(params: {\n nodeType: string;\n position: PositionSchema;\n fromPort?: WorkflowPortEntity;\n toPort?: WorkflowPortEntity;\n containerNode?: WorkflowNodeEntity;\n }): PositionSchema {\n const { nodeType, position, fromPort, toPort, containerNode } = params;\n const register = this.document.getNodeRegistry(nodeType);\n const size = register?.meta?.size;\n let adjustedPosition = position;\n if (!size) {\n adjustedPosition = position;\n }\n // 计算坐标偏移\n else if (fromPort && toPort) {\n // 输入输出\n adjustedPosition = {\n x: position.x,\n y: position.y - size.height / 2,\n };\n } else if (fromPort && !toPort) {\n // 仅输入\n adjustedPosition = {\n x: position.x + size.width / 2,\n y: position.y - size.height / 2,\n };\n } else if (!fromPort && toPort) {\n // 仅输出\n adjustedPosition = {\n x: position.x - size.width / 2,\n y: position.y - size.height / 2,\n };\n } else {\n adjustedPosition = position;\n }\n return this.dragService.adjustSubNodePosition(nodeType, containerNode, adjustedPosition);\n }\n\n private getContainerNode(params: {\n containerNode?: WorkflowNodeEntity;\n fromPort?: WorkflowPortEntity;\n toPort?: WorkflowPortEntity;\n }): WorkflowNodeEntity | undefined {\n const { fromPort, containerNode } = params;\n if (containerNode) {\n return containerNode;\n }\n const fromNode = fromPort?.node;\n const fromContainer = fromNode?.parent;\n if (fromNode?.flowNodeType === FlowNodeBaseType.SUB_CANVAS) {\n // 子画布内部输入连线\n return fromNode;\n }\n return fromContainer;\n }\n\n /** 获取端口矩形 */\n private getPortBox(port: WorkflowPortEntity, offset: IPoint = { x: 0, y: 0 }): Rectangle {\n const node = port.node;\n if (node.flowNodeType === FlowNodeBaseType.SUB_CANVAS) {\n // 子画布内部端口需要虚拟节点\n const { point } = port;\n if (port.portType === 'input') {\n return new Rectangle(point.x + offset.x, point.y - 50 + offset.y, 300, 100);\n }\n return new Rectangle(point.x - 300, point.y - 50, 300, 100);\n }\n const box = node.getData(FlowNodeTransformData).bounds;\n return box;\n }\n\n /** 后续节点位置偏移 */\n private subPositionOffset(params: {\n node: WorkflowNodeEntity;\n fromPort: WorkflowPortEntity;\n toPort: WorkflowPortEntity;\n padding: {\n x: number;\n y: number;\n };\n }):\n | {\n x: number;\n y: number;\n }\n | undefined {\n const { node, fromPort, toPort, padding } = params;\n\n const fromBox = this.getPortBox(fromPort);\n const toBox = this.getPortBox(toPort);\n\n const nodeTrans = node.getData(FlowNodeTransformData);\n const nodeSize = node.getNodeMeta()?.size ?? {\n width: nodeTrans.bounds.width,\n height: nodeTrans.bounds.height,\n };\n\n // 最小距离\n const minDistance: IPoint = {\n x: nodeSize.width + padding.x,\n y: nodeSize.height + padding.y,\n };\n // from 与 to 的距离\n const boxDistance = this.rectDistance(fromBox, toBox);\n\n // 需要的偏移量\n const neededOffset: IPoint = {\n x: isGreaterThan(boxDistance.x, minDistance.x) ? 0 : minDistance.x - boxDistance.x,\n y: isGreaterThan(boxDistance.y, minDistance.y) ? 0 : minDistance.y - boxDistance.y,\n };\n\n // 至少有一个方向满足要求,无需偏移\n if (neededOffset.x === 0 || neededOffset.y === 0) {\n return;\n }\n\n // 是否存在相交\n const intersection = {\n // 这里没有写反,Rectangle内置的算法是反的\n vertical: Rectangle.intersects(fromBox, toBox, 'horizontal'),\n horizontal: Rectangle.intersects(fromBox, toBox, 'vertical'),\n };\n\n // 初始化偏移量\n let offsetX: number = 0;\n let offsetY: number = 0;\n\n if (!intersection.horizontal) {\n // 水平不相交,需要加垂直方向的偏移\n if (isGreaterThan(toBox.center.y, fromBox.center.y)) {\n // B在A下方\n offsetY = neededOffset.y;\n } else if (isLessThan(toBox.center.y, fromBox.center.y)) {\n // B在A上方\n offsetY = -neededOffset.y;\n }\n }\n\n if (!intersection.vertical) {\n // 垂直不相交,需要加水平方向的偏移\n if (isGreaterThan(toBox.center.x, fromBox.center.x)) {\n // B在A右侧\n offsetX = neededOffset.x;\n } else if (isLessThan(toBox.center.x, fromBox.center.x)) {\n // B在A左侧\n offsetX = -neededOffset.x;\n }\n }\n\n return {\n x: offsetX,\n y: offsetY,\n };\n }\n\n /** 矩形间距 */\n private rectDistance(rectA: Rectangle, rectB: Rectangle): IPoint {\n // 计算 x 轴距离\n const distanceX = Math.abs(\n Math.min(rectA.right, rectB.right) - Math.max(rectA.left, rectB.left)\n );\n // 计算 y 轴距离\n const distanceY = Math.abs(\n Math.min(rectA.bottom, rectB.bottom) - Math.max(rectA.top, rectB.top)\n );\n if (Rectangle.intersects(rectA, rectB)) {\n // 相交距离为负\n return {\n x: -distanceX,\n y: -distanceY,\n };\n }\n return {\n x: distanceX,\n y: distanceY,\n };\n }\n\n /** 获取后续节点 */\n private getSubsequentNodes(node: WorkflowNodeEntity): WorkflowNodeEntity[] {\n if (node.flowNodeType === FlowNodeBaseType.SUB_CANVAS) {\n return [];\n }\n const brothers = node.parent?.collapsedChildren ?? [];\n const linkedBrothers = new Set();\n const linesMap = new Map<string, string[]>();\n this.linesManager.getAllLines().forEach((line) => {\n if (!linesMap.has(line.from.id)) {\n linesMap.set(line.from.id, []);\n }\n if (\n !line.to?.id ||\n line.to.flowNodeType === FlowNodeBaseType.SUB_CANVAS // 子画布内部成环\n ) {\n return;\n }\n linesMap.get(line.from.id)?.push(line.to.id);\n });\n\n const bfs = (nodeId: string) => {\n if (linkedBrothers.has(nodeId)) {\n return;\n }\n linkedBrothers.add(nodeId);\n const nextNodes = linesMap.get(nodeId) ?? [];\n nextNodes.forEach(bfs);\n };\n\n bfs(node.id);\n\n const subsequentNodes = brothers.filter((node) => linkedBrothers.has(node.id));\n return subsequentNodes;\n }\n\n /** 更新后续节点位置 */\n private updateSubSequentNodesPosition(params: {\n node: WorkflowNodeEntity;\n subsequentNodes: WorkflowNodeEntity[];\n fromPort: WorkflowPortEntity;\n toPort: WorkflowPortEntity;\n containerNode?: WorkflowNodeEntity;\n offset?: IPoint;\n }): void {\n const { node, subsequentNodes, fromPort, toPort, containerNode, offset } = params;\n if (!offset || !toPort) {\n return;\n }\n // 更新后续节点位置\n const subsequentNodesPositions = subsequentNodes.map((node) => {\n const nodeTrans = node.getData(TransformData);\n return {\n x: nodeTrans.position.x,\n y: nodeTrans.position.y,\n };\n });\n this.historyService.pushOperation({\n type: FreeOperationType.dragNodes,\n value: {\n ids: subsequentNodes.map((node) => node.id),\n value: subsequentNodesPositions.map((position) => ({\n x: position.x + offset.x,\n y: position.y + offset.y,\n })),\n oldValue: subsequentNodesPositions,\n },\n });\n // 新增节点坐标需重新计算\n const fromBox = this.getPortBox(fromPort);\n const toBox = this.getPortBox(toPort, offset);\n const nodeTrans = node.getData(TransformData);\n let nodePos: PositionSchema = {\n x: (fromBox.center.x + toBox.center.x) / 2,\n y: (fromBox.y + toBox.y) / 2,\n };\n if (containerNode) {\n nodePos = this.dragService.adjustSubNodePosition(\n node.flowNodeType as string,\n containerNode,\n nodePos\n );\n }\n this.historyService.pushOperation({\n type: FreeOperationType.dragNodes,\n value: {\n ids: [node.id],\n value: [nodePos],\n oldValue: [\n {\n x: nodeTrans.position.x,\n y: nodeTrans.position.y,\n },\n ],\n },\n });\n }\n}\n","/**\n * 检查 a 是否大于 b,考虑浮点数精度问题\n * @param a 第一个数\n * @param b 第二个数\n * @returns 如果 a 大于 b 则返回 true,否则返回 false\n */\nexport const isGreaterThan = (a: number | undefined, b: number | undefined): boolean => {\n // 如果任一参数为 undefined,返回 false\n if (a === undefined || b === undefined) {\n return false;\n }\n\n // 定义一个很小的误差值\n const EPSILON: number = 0.00001;\n\n // 检查 a 是否显著大于 b\n return a - b > EPSILON;\n};\n\n/**\n * 检查 a 是否小于 b,考虑浮点数精度问题\n * @param a 第一个数\n * @param b 第二个数\n * @returns 如果 a 小于 b 则返回 true,否则返回 false\n */\nexport const isLessThan = (a: number | undefined, b: number | undefined): boolean => {\n // 如果任一参数为 undefined,返回 false\n if (a === undefined || b === undefined) {\n return false;\n }\n\n // 定义一个很小的误差值\n const EPSILON: number = 0.00001;\n\n // 检查 a 是否显著小于 b\n return b - a > EPSILON;\n};\n","/* eslint-disable react/no-deprecated */\nimport React from 'react';\n\nimport { inject } from 'inversify';\nimport { Layer } from '@flowgram.ai/core';\nimport { nanoid } from '@flowgram.ai/free-layout-core';\nimport { domUtils } from '@flowgram.ai/utils';\n\nimport type {\n CallNodePanelParams,\n NodePanelLayerOptions,\n NodePanelRenderProps,\n NodePanelResult,\n} from './type';\nimport { WorkflowNodePanelService } from './service';\n\nexport class WorkflowNodePanelLayer extends Layer<NodePanelLayerOptions> {\n public static type = 'WorkflowNodePanelLayer';\n\n @inject(WorkflowNodePanelService) private service: WorkflowNodePanelService;\n\n public node: HTMLDivElement;\n\n private renderList: Map<string, NodePanelRenderProps>;\n\n constructor() {\n super();\n this.node = domUtils.createDivWithClass('gedit-playground-layer gedit-node-panel-layer');\n this.node.style.zIndex = '9999';\n this.renderList = new Map();\n }\n\n public onReady(): void {\n this.service.setCallNodePanel(this.call.bind(this));\n }\n\n public onZoom(zoom: number): void {\n this.node.style.transform = `scale(${zoom})`;\n }\n\n public render(): JSX.Element {\n const NodePanelRender = this.options.renderer;\n return (\n <>\n {Array.from(this.renderList.keys()).map(taskId => {\n const renderProps = this.renderList.get(taskId)!;\n return <NodePanelRender key={taskId} {...renderProps} />;\n })}\n </>\n );\n }\n\n private async call(params: CallNodePanelParams): Promise<void> {\n const taskId = nanoid();\n const { enableMultiAdd, onSelect, onClose } = params;\n return new Promise(resolve => {\n const unmount = () => {\n // 清理挂载的组件\n this.renderList.delete(taskId);\n this.render();\n resolve();\n };\n const handleClose = () => {\n unmount();\n onClose();\n };\n const handleSelect = (params?: NodePanelResult) => {\n onSelect(params);\n if (!enableMultiAdd) {\n unmount();\n }\n };\n const renderProps: NodePanelRenderProps = {\n ...params,\n onSelect: handleSelect,\n onClose: handleClose,\n };\n this.renderList.set(taskId, renderProps);\n this.render();\n });\n }\n}\n"],"mappings":"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","names":["node","nodeTrans","inject","params","inject"]}
@@ -0,0 +1,103 @@
1
+ import * as _flowgram_ai_core from '@flowgram.ai/core';
2
+ import React from 'react';
3
+ import { WorkflowNodeJSON, WorkflowNodeEntity, WorkflowPortEntity } from '@flowgram.ai/free-layout-core';
4
+ import { PositionSchema } from '@flowgram.ai/utils';
5
+
6
+ interface NodePanelCallParams {
7
+ /** 唤起节点面板的位置 */
8
+ panelPosition: PositionSchema;
9
+ /** 自定义传给面板组件的props */
10
+ panelProps?: Record<string, any>;
11
+ /** 指定父节点 */
12
+ containerNode?: WorkflowNodeEntity;
13
+ /** 创建节点后需连接的前序端口 */
14
+ fromPort?: WorkflowPortEntity;
15
+ /** 创建节点后需连接的后序端口 */
16
+ toPort?: WorkflowPortEntity;
17
+ /** 偏移后续节点默认间距 */
18
+ autoOffsetPadding?: PositionSchema;
19
+ /** 自定义节点位置 */
20
+ customPosition?: (params: {
21
+ nodeType: string;
22
+ selectPosition: PositionSchema;
23
+ }) => PositionSchema;
24
+ /** 是否可以添加节点 */
25
+ canAddNode?: (params: {
26
+ nodeType: string;
27
+ containerNode?: WorkflowNodeEntity;
28
+ }) => boolean;
29
+ /** 创建节点后 */
30
+ afterAddNode?: (node?: WorkflowNodeEntity) => void;
31
+ /** 是否创建线条 */
32
+ enableBuildLine?: boolean;
33
+ /** 是否使用选中位置作为节点位置 */
34
+ enableSelectPosition?: boolean;
35
+ /** 是否偏移后续节点 */
36
+ enableAutoOffset?: boolean;
37
+ /** 是否触发节点拖拽 */
38
+ enableDragNode?: boolean;
39
+ /** 是否可以添加多个节点 */
40
+ enableMultiAdd?: boolean;
41
+ }
42
+ type NodePanelResult = {
43
+ nodeType: string;
44
+ nodeJSON?: WorkflowNodeJSON;
45
+ selectEvent: React.MouseEvent;
46
+ } | undefined;
47
+ interface CallNodePanelParams {
48
+ onSelect: (params?: NodePanelResult) => void;
49
+ position: PositionSchema;
50
+ onClose: () => void;
51
+ enableMultiAdd: boolean;
52
+ panelProps: Record<string, any>;
53
+ containerNode?: WorkflowNodeEntity;
54
+ }
55
+ type CallNodePanel = (params: CallNodePanelParams) => Promise<void>;
56
+ interface NodePanelRenderProps extends CallNodePanelParams {
57
+ }
58
+ type NodePanelRender = React.FC<NodePanelRenderProps>;
59
+ interface NodePanelLayerOptions {
60
+ renderer: NodePanelRender;
61
+ }
62
+ interface NodePanelPluginOptions extends NodePanelLayerOptions {
63
+ }
64
+
65
+ declare const createFreeNodePanelPlugin: _flowgram_ai_core.PluginCreator<NodePanelPluginOptions>;
66
+
67
+ /**
68
+ * 添加节点面板服务
69
+ */
70
+ declare class WorkflowNodePanelService {
71
+ private readonly document;
72
+ private readonly dragService;
73
+ private readonly selectService;
74
+ private readonly linesManager;
75
+ private readonly playgroundConfig;
76
+ private readonly historyService;
77
+ private readonly toDispose;
78
+ private callNodePanel;
79
+ /** 销毁 */
80
+ dispose(): void;
81
+ setCallNodePanel(callNodePanel: CallNodePanel): void;
82
+ /** 唤起节点面板 */
83
+ call(callParams: NodePanelCallParams): Promise<WorkflowNodeEntity | WorkflowNodeEntity[] | undefined>;
84
+ /** 添加节点 */
85
+ private addNode;
86
+ /** 建立连线 */
87
+ private buildLine;
88
+ /** 调整节点坐标 */
89
+ private adjustNodePosition;
90
+ private getContainerNode;
91
+ /** 获取端口矩形 */
92
+ private getPortBox;
93
+ /** 后续节点位置偏移 */
94
+ private subPositionOffset;
95
+ /** 矩形间距 */
96
+ private rectDistance;
97
+ /** 获取后续节点 */
98
+ private getSubsequentNodes;
99
+ /** 更新后续节点位置 */
100
+ private updateSubSequentNodesPosition;
101
+ }
102
+
103
+ export { type NodePanelPluginOptions, type NodePanelRender, type NodePanelRenderProps, type NodePanelResult, type NodePanelLayerOptions as NodePanelServiceOptions, WorkflowNodePanelService, createFreeNodePanelPlugin };