2024-07-08 03:25:18 -07:00
|
|
|
/**
|
|
|
|
* Canvas V2 Only
|
|
|
|
* @TODO Remove this notice when Canvas V2 is the only one in use
|
|
|
|
*/
|
|
|
|
|
2024-06-04 05:36:27 -07:00
|
|
|
import type { CanvasElement } from '@/types';
|
2024-07-08 03:25:18 -07:00
|
|
|
import { CanvasConnectionMode } from '@/types';
|
2024-06-25 02:11:44 -07:00
|
|
|
import type {
|
|
|
|
AddedNodesAndConnections,
|
|
|
|
INodeUi,
|
|
|
|
INodeUpdatePropertiesInformation,
|
|
|
|
XYPosition,
|
|
|
|
} from '@/Interface';
|
2024-07-10 01:53:27 -07:00
|
|
|
import {
|
|
|
|
FORM_TRIGGER_NODE_TYPE,
|
|
|
|
QUICKSTART_NOTE_NAME,
|
|
|
|
STICKY_NODE_TYPE,
|
|
|
|
WEBHOOK_NODE_TYPE,
|
|
|
|
} from '@/constants';
|
2024-06-04 05:36:27 -07:00
|
|
|
import { useWorkflowsStore } from '@/stores/workflows.store';
|
|
|
|
import { useHistoryStore } from '@/stores/history.store';
|
|
|
|
import { useUIStore } from '@/stores/ui.store';
|
|
|
|
import { useTelemetry } from '@/composables/useTelemetry';
|
|
|
|
import { useExternalHooks } from '@/composables/useExternalHooks';
|
2024-06-17 05:46:55 -07:00
|
|
|
import {
|
2024-07-10 01:53:27 -07:00
|
|
|
AddNodeCommand,
|
2024-06-17 05:46:55 -07:00
|
|
|
MoveNodeCommand,
|
|
|
|
RemoveConnectionCommand,
|
|
|
|
RemoveNodeCommand,
|
|
|
|
RenameNodeCommand,
|
|
|
|
} from '@/models/history';
|
2024-06-04 05:36:27 -07:00
|
|
|
import type { Connection } from '@vue-flow/core';
|
2024-06-25 02:11:44 -07:00
|
|
|
import {
|
|
|
|
getUniqueNodeName,
|
|
|
|
mapCanvasConnectionToLegacyConnection,
|
|
|
|
parseCanvasConnectionHandleString,
|
|
|
|
} from '@/utils/canvasUtilsV2';
|
|
|
|
import type {
|
|
|
|
ConnectionTypes,
|
|
|
|
IConnection,
|
|
|
|
INodeInputConfiguration,
|
|
|
|
INodeOutputConfiguration,
|
|
|
|
INodeTypeDescription,
|
|
|
|
INodeTypeNameVersion,
|
|
|
|
ITelemetryTrackProperties,
|
|
|
|
} from 'n8n-workflow';
|
|
|
|
import { NodeConnectionType, NodeHelpers } from 'n8n-workflow';
|
2024-06-17 05:46:55 -07:00
|
|
|
import { useNDVStore } from '@/stores/ndv.store';
|
2024-06-25 02:11:44 -07:00
|
|
|
import { useNodeTypesStore } from '@/stores/nodeTypes.store';
|
|
|
|
import { useI18n } from '@/composables/useI18n';
|
|
|
|
import { useToast } from '@/composables/useToast';
|
|
|
|
import * as NodeViewUtils from '@/utils/nodeViewUtils';
|
|
|
|
import { v4 as uuid } from 'uuid';
|
|
|
|
import type { Ref } from 'vue';
|
|
|
|
import { computed } from 'vue';
|
|
|
|
import { useCredentialsStore } from '@/stores/credentials.store';
|
|
|
|
import { useWorkflowHelpers } from '@/composables/useWorkflowHelpers';
|
|
|
|
import type { useRouter } from 'vue-router';
|
|
|
|
import { useCanvasStore } from '@/stores/canvas.store';
|
2024-06-26 06:56:58 -07:00
|
|
|
import { useNodeHelpers } from '@/composables/useNodeHelpers';
|
2024-06-25 02:11:44 -07:00
|
|
|
|
2024-07-10 01:53:27 -07:00
|
|
|
type AddNodeData = Partial<INodeUi> & {
|
2024-06-25 02:11:44 -07:00
|
|
|
type: string;
|
|
|
|
};
|
|
|
|
|
|
|
|
type AddNodeOptions = {
|
|
|
|
dragAndDrop?: boolean;
|
|
|
|
openNDV?: boolean;
|
|
|
|
trackHistory?: boolean;
|
|
|
|
isAutoAdd?: boolean;
|
|
|
|
};
|
|
|
|
|
|
|
|
export function useCanvasOperations({
|
|
|
|
router,
|
|
|
|
lastClickPosition,
|
|
|
|
}: {
|
|
|
|
router: ReturnType<typeof useRouter>;
|
|
|
|
lastClickPosition: Ref<XYPosition>;
|
|
|
|
}) {
|
2024-06-04 05:36:27 -07:00
|
|
|
const workflowsStore = useWorkflowsStore();
|
2024-06-25 02:11:44 -07:00
|
|
|
const credentialsStore = useCredentialsStore();
|
2024-06-04 05:36:27 -07:00
|
|
|
const historyStore = useHistoryStore();
|
|
|
|
const uiStore = useUIStore();
|
2024-06-17 05:46:55 -07:00
|
|
|
const ndvStore = useNDVStore();
|
2024-06-25 02:11:44 -07:00
|
|
|
const nodeTypesStore = useNodeTypesStore();
|
|
|
|
const canvasStore = useCanvasStore();
|
2024-06-04 05:36:27 -07:00
|
|
|
|
2024-06-25 02:11:44 -07:00
|
|
|
const i18n = useI18n();
|
|
|
|
const toast = useToast();
|
|
|
|
const workflowHelpers = useWorkflowHelpers({ router });
|
2024-06-26 06:56:58 -07:00
|
|
|
const nodeHelpers = useNodeHelpers();
|
2024-06-04 05:36:27 -07:00
|
|
|
const telemetry = useTelemetry();
|
|
|
|
const externalHooks = useExternalHooks();
|
|
|
|
|
2024-06-25 02:11:44 -07:00
|
|
|
const editableWorkflow = computed(() => workflowsStore.workflow);
|
|
|
|
const editableWorkflowObject = computed(() => workflowsStore.getCurrentWorkflow());
|
|
|
|
|
|
|
|
const triggerNodes = computed<INodeUi[]>(() => {
|
|
|
|
return workflowsStore.workflowTriggerNodes;
|
|
|
|
});
|
|
|
|
|
2024-06-04 05:36:27 -07:00
|
|
|
/**
|
|
|
|
* Node operations
|
|
|
|
*/
|
|
|
|
|
|
|
|
function updateNodePosition(
|
|
|
|
id: string,
|
|
|
|
position: CanvasElement['position'],
|
|
|
|
{ trackHistory = false, trackBulk = true } = {},
|
|
|
|
) {
|
|
|
|
const node = workflowsStore.getNodeById(id);
|
|
|
|
if (!node) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (trackHistory && trackBulk) {
|
|
|
|
historyStore.startRecordingUndo();
|
|
|
|
}
|
|
|
|
|
|
|
|
const oldPosition: XYPosition = [...node.position];
|
|
|
|
const newPosition: XYPosition = [position.x, position.y];
|
|
|
|
|
|
|
|
workflowsStore.setNodePositionById(id, newPosition);
|
|
|
|
|
|
|
|
if (trackHistory) {
|
|
|
|
historyStore.pushCommandToUndo(new MoveNodeCommand(node.name, oldPosition, newPosition));
|
|
|
|
|
|
|
|
if (trackBulk) {
|
|
|
|
historyStore.stopRecordingUndo();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-17 05:46:55 -07:00
|
|
|
async function renameNode(currentName: string, newName: string, { trackHistory = false } = {}) {
|
|
|
|
if (currentName === newName) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (trackHistory) {
|
|
|
|
historyStore.startRecordingUndo();
|
|
|
|
}
|
|
|
|
|
|
|
|
newName = getUniqueNodeName(newName, workflowsStore.canvasNames);
|
|
|
|
|
|
|
|
// Rename the node and update the connections
|
|
|
|
const workflow = workflowsStore.getCurrentWorkflow(true);
|
|
|
|
workflow.renameNode(currentName, newName);
|
|
|
|
|
|
|
|
if (trackHistory) {
|
|
|
|
historyStore.pushCommandToUndo(new RenameNodeCommand(currentName, newName));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update also last selected node and execution data
|
|
|
|
workflowsStore.renameNodeSelectedAndExecution({ old: currentName, new: newName });
|
|
|
|
|
|
|
|
workflowsStore.setNodes(Object.values(workflow.nodes));
|
|
|
|
workflowsStore.setConnections(workflow.connectionsBySourceNode);
|
|
|
|
|
|
|
|
const isRenamingActiveNode = ndvStore.activeNodeName === currentName;
|
|
|
|
if (isRenamingActiveNode) {
|
|
|
|
ndvStore.activeNodeName = newName;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (trackHistory) {
|
|
|
|
historyStore.stopRecordingUndo();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function revertRenameNode(currentName: string, previousName: string) {
|
|
|
|
await renameNode(currentName, previousName);
|
|
|
|
}
|
|
|
|
|
2024-06-04 05:36:27 -07:00
|
|
|
function deleteNode(id: string, { trackHistory = false, trackBulk = true } = {}) {
|
|
|
|
const node = workflowsStore.getNodeById(id);
|
|
|
|
if (!node) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (trackHistory && trackBulk) {
|
|
|
|
historyStore.startRecordingUndo();
|
|
|
|
}
|
|
|
|
|
|
|
|
workflowsStore.removeNodeConnectionsById(id);
|
|
|
|
workflowsStore.removeNodeExecutionDataById(id);
|
2024-07-08 03:25:18 -07:00
|
|
|
workflowsStore.removeNodeById(id);
|
2024-06-04 05:36:27 -07:00
|
|
|
|
|
|
|
if (trackHistory) {
|
|
|
|
historyStore.pushCommandToUndo(new RemoveNodeCommand(node));
|
|
|
|
|
|
|
|
if (trackBulk) {
|
|
|
|
historyStore.stopRecordingUndo();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
trackDeleteNode(id);
|
|
|
|
}
|
|
|
|
|
|
|
|
function revertDeleteNode(node: INodeUi) {
|
|
|
|
workflowsStore.addNode(node);
|
|
|
|
}
|
|
|
|
|
|
|
|
function trackDeleteNode(id: string) {
|
|
|
|
const node = workflowsStore.getNodeById(id);
|
|
|
|
if (!node) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (node.type === STICKY_NODE_TYPE) {
|
|
|
|
telemetry.track('User deleted workflow note', {
|
|
|
|
workflow_id: workflowsStore.workflowId,
|
|
|
|
is_welcome_note: node.name === QUICKSTART_NOTE_NAME,
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
void externalHooks.run('node.deleteNode', { node });
|
|
|
|
telemetry.track('User deleted node', {
|
|
|
|
node_type: node.type,
|
|
|
|
workflow_id: workflowsStore.workflowId,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-17 05:46:55 -07:00
|
|
|
function setNodeActive(id: string) {
|
|
|
|
const node = workflowsStore.getNodeById(id);
|
|
|
|
if (!node) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-07-08 03:25:18 -07:00
|
|
|
setNodeActiveByName(node.name);
|
2024-06-17 05:46:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
function setNodeActiveByName(name: string) {
|
|
|
|
ndvStore.activeNodeName = name;
|
|
|
|
}
|
|
|
|
|
2024-06-25 02:11:44 -07:00
|
|
|
function setNodeSelected(id?: string) {
|
|
|
|
if (!id) {
|
|
|
|
uiStore.lastSelectedNode = '';
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const node = workflowsStore.getNodeById(id);
|
|
|
|
if (!node) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
uiStore.lastSelectedNode = node.name;
|
|
|
|
}
|
|
|
|
|
2024-06-26 06:56:58 -07:00
|
|
|
function toggleNodeDisabled(
|
|
|
|
id: string,
|
|
|
|
{ trackHistory = true }: { trackHistory?: boolean } = {},
|
|
|
|
) {
|
|
|
|
const node = workflowsStore.getNodeById(id);
|
|
|
|
if (!node) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nodeHelpers.disableNodes([node], trackHistory);
|
|
|
|
}
|
|
|
|
|
2024-06-25 02:11:44 -07:00
|
|
|
async function addNodes(
|
|
|
|
nodes: AddedNodesAndConnections['nodes'],
|
|
|
|
{
|
|
|
|
dragAndDrop,
|
|
|
|
position,
|
|
|
|
}: {
|
|
|
|
dragAndDrop?: boolean;
|
|
|
|
position?: XYPosition;
|
|
|
|
} = {},
|
|
|
|
) {
|
|
|
|
let currentPosition = position;
|
|
|
|
let lastAddedNode: INodeUi | undefined;
|
2024-07-10 01:53:27 -07:00
|
|
|
for (const { isAutoAdd, openDetail, ...nodeData } of nodes) {
|
2024-06-25 02:11:44 -07:00
|
|
|
try {
|
|
|
|
await createNode(
|
|
|
|
{
|
2024-07-10 01:53:27 -07:00
|
|
|
...nodeData,
|
|
|
|
position: nodeData.position ?? currentPosition,
|
2024-06-25 02:11:44 -07:00
|
|
|
},
|
|
|
|
{
|
|
|
|
dragAndDrop,
|
|
|
|
openNDV: openDetail ?? false,
|
|
|
|
trackHistory: true,
|
|
|
|
isAutoAdd,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
} catch (error) {
|
|
|
|
toast.showError(error, i18n.baseText('error'));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
lastAddedNode = editableWorkflow.value.nodes[editableWorkflow.value.nodes.length - 1];
|
|
|
|
currentPosition = [
|
|
|
|
lastAddedNode.position[0] + NodeViewUtils.NODE_SIZE * 2 + NodeViewUtils.GRID_SIZE,
|
|
|
|
lastAddedNode.position[1],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the last added node has multiple inputs, move them down
|
|
|
|
if (!lastAddedNode) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const lastNodeInputs = editableWorkflowObject.value.getParentNodesByDepth(
|
|
|
|
lastAddedNode.name,
|
|
|
|
1,
|
|
|
|
);
|
|
|
|
if (lastNodeInputs.length > 1) {
|
|
|
|
lastNodeInputs.slice(1).forEach((node, index) => {
|
|
|
|
const nodeUi = workflowsStore.getNodeByName(node.name);
|
|
|
|
if (!nodeUi) return;
|
|
|
|
|
|
|
|
updateNodePosition(nodeUi.id, {
|
|
|
|
x: nodeUi.position[0],
|
|
|
|
y: nodeUi.position[1] + 100 * (index + 1),
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function createNode(node: AddNodeData, options: AddNodeOptions = {}): Promise<INodeUi> {
|
|
|
|
const newNodeData = await resolveNodeData(node, options);
|
|
|
|
if (!newNodeData) {
|
|
|
|
throw new Error(i18n.baseText('nodeViewV2.showError.failedToCreateNode'));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @TODO Check if maximum node type limit reached
|
|
|
|
*/
|
|
|
|
|
|
|
|
newNodeData.name = getUniqueNodeName(newNodeData.name, workflowsStore.canvasNames);
|
|
|
|
|
|
|
|
workflowsStore.addNode(newNodeData);
|
|
|
|
|
2024-07-10 01:53:27 -07:00
|
|
|
nodeHelpers.matchCredentials(newNodeData);
|
2024-06-25 02:11:44 -07:00
|
|
|
|
|
|
|
const lastSelectedNode = uiStore.getLastSelectedNode;
|
|
|
|
const lastSelectedNodeOutputIndex = uiStore.lastSelectedNodeOutputIndex;
|
|
|
|
const lastSelectedNodeEndpointUuid = uiStore.lastSelectedNodeEndpointUuid;
|
|
|
|
|
|
|
|
historyStore.startRecordingUndo();
|
2024-07-10 01:53:27 -07:00
|
|
|
if (options.trackHistory) {
|
|
|
|
historyStore.pushCommandToUndo(new AddNodeCommand(newNodeData));
|
|
|
|
}
|
2024-06-25 02:11:44 -07:00
|
|
|
|
|
|
|
const outputIndex = lastSelectedNodeOutputIndex ?? 0;
|
|
|
|
const targetEndpoint = lastSelectedNodeEndpointUuid ?? '';
|
|
|
|
|
2024-07-08 03:25:18 -07:00
|
|
|
// Create a connection between the last selected node and the new one
|
2024-06-25 02:11:44 -07:00
|
|
|
if (lastSelectedNode && !options.isAutoAdd) {
|
2024-07-08 03:25:18 -07:00
|
|
|
// If we have a specific endpoint to connect to
|
2024-06-25 02:11:44 -07:00
|
|
|
if (lastSelectedNodeEndpointUuid) {
|
2024-07-08 03:25:18 -07:00
|
|
|
const { type: connectionType, mode } = parseCanvasConnectionHandleString(
|
2024-06-25 02:11:44 -07:00
|
|
|
lastSelectedNodeEndpointUuid,
|
|
|
|
);
|
2024-07-08 03:25:18 -07:00
|
|
|
|
|
|
|
const newNodeId = newNodeData.id;
|
|
|
|
const newNodeHandle = `${CanvasConnectionMode.Input}/${connectionType}/0`;
|
|
|
|
const lasSelectedNodeId = lastSelectedNode.id;
|
|
|
|
const lastSelectedNodeHandle = targetEndpoint;
|
|
|
|
|
|
|
|
if (mode === CanvasConnectionMode.Input) {
|
|
|
|
createConnection({
|
|
|
|
source: newNodeId,
|
|
|
|
sourceHandle: newNodeHandle,
|
|
|
|
target: lasSelectedNodeId,
|
|
|
|
targetHandle: lastSelectedNodeHandle,
|
|
|
|
});
|
|
|
|
} else {
|
2024-06-25 02:11:44 -07:00
|
|
|
createConnection({
|
2024-07-08 03:25:18 -07:00
|
|
|
source: lasSelectedNodeId,
|
|
|
|
sourceHandle: lastSelectedNodeHandle,
|
|
|
|
target: newNodeId,
|
|
|
|
targetHandle: newNodeHandle,
|
2024-06-25 02:11:44 -07:00
|
|
|
});
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// If a node is last selected then connect between the active and its child ones
|
|
|
|
// Connect active node to the newly created one
|
|
|
|
createConnection({
|
|
|
|
source: lastSelectedNode.id,
|
|
|
|
sourceHandle: `outputs/${NodeConnectionType.Main}/${outputIndex}`,
|
|
|
|
target: newNodeData.id,
|
|
|
|
targetHandle: `inputs/${NodeConnectionType.Main}/0`,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
historyStore.stopRecordingUndo();
|
|
|
|
|
|
|
|
return newNodeData;
|
|
|
|
}
|
|
|
|
|
|
|
|
async function initializeNodeDataWithDefaultCredentials(node: AddNodeData) {
|
|
|
|
const nodeTypeDescription = nodeTypesStore.getNodeType(node.type);
|
|
|
|
if (!nodeTypeDescription) {
|
|
|
|
throw new Error(i18n.baseText('nodeViewV2.showError.failedToCreateNode'));
|
|
|
|
}
|
|
|
|
|
|
|
|
let nodeVersion = nodeTypeDescription.defaultVersion;
|
|
|
|
if (typeof nodeVersion === 'undefined') {
|
|
|
|
nodeVersion = Array.isArray(nodeTypeDescription.version)
|
|
|
|
? nodeTypeDescription.version.slice(-1)[0]
|
|
|
|
: nodeTypeDescription.version;
|
|
|
|
}
|
|
|
|
|
|
|
|
const newNodeData: INodeUi = {
|
2024-07-10 01:53:27 -07:00
|
|
|
...node,
|
|
|
|
id: node.id ?? uuid(),
|
2024-06-25 02:11:44 -07:00
|
|
|
name: node.name ?? (nodeTypeDescription.defaults.name as string),
|
|
|
|
type: nodeTypeDescription.name,
|
|
|
|
typeVersion: nodeVersion,
|
|
|
|
position: node.position ?? [0, 0],
|
2024-07-10 01:53:27 -07:00
|
|
|
disabled: node.disabled ?? false,
|
|
|
|
parameters: node.parameters ?? {},
|
2024-06-25 02:11:44 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
await loadNodeTypesProperties([{ name: newNodeData.type, version: newNodeData.typeVersion }]);
|
|
|
|
|
|
|
|
const nodeType = nodeTypesStore.getNodeType(newNodeData.type, newNodeData.typeVersion);
|
|
|
|
const nodeParameters = NodeHelpers.getNodeParameters(
|
|
|
|
nodeType?.properties ?? [],
|
|
|
|
{},
|
|
|
|
true,
|
|
|
|
false,
|
|
|
|
newNodeData,
|
|
|
|
);
|
|
|
|
|
|
|
|
newNodeData.parameters = nodeParameters ?? {};
|
|
|
|
|
|
|
|
const credentialPerType = nodeTypeDescription.credentials
|
|
|
|
?.map((type) => credentialsStore.getUsableCredentialByType(type.name))
|
|
|
|
.flat();
|
|
|
|
|
|
|
|
if (credentialPerType?.length === 1) {
|
|
|
|
const defaultCredential = credentialPerType[0];
|
|
|
|
|
|
|
|
const selectedCredentials = credentialsStore.getCredentialById(defaultCredential.id);
|
|
|
|
const selected = { id: selectedCredentials.id, name: selectedCredentials.name };
|
|
|
|
const credentials = {
|
|
|
|
[defaultCredential.type]: selected,
|
|
|
|
};
|
|
|
|
|
|
|
|
if (nodeTypeDescription.credentials) {
|
|
|
|
const authentication = nodeTypeDescription.credentials.find(
|
|
|
|
(type) => type.name === defaultCredential.type,
|
|
|
|
);
|
|
|
|
if (authentication?.displayOptions?.hide) {
|
|
|
|
return newNodeData;
|
|
|
|
}
|
|
|
|
|
|
|
|
const authDisplayOptions = authentication?.displayOptions?.show;
|
|
|
|
if (!authDisplayOptions) {
|
|
|
|
newNodeData.credentials = credentials;
|
|
|
|
return newNodeData;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Object.keys(authDisplayOptions).length === 1 && authDisplayOptions.authentication) {
|
|
|
|
// ignore complex case when there's multiple dependencies
|
|
|
|
newNodeData.credentials = credentials;
|
|
|
|
|
|
|
|
let parameters: { [key: string]: string } = {};
|
|
|
|
for (const displayOption of Object.keys(authDisplayOptions)) {
|
|
|
|
if (nodeParameters && !nodeParameters[displayOption]) {
|
|
|
|
parameters = {};
|
|
|
|
newNodeData.credentials = undefined;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
const optionValue = authDisplayOptions[displayOption]?.[0];
|
|
|
|
if (optionValue && typeof optionValue === 'string') {
|
|
|
|
parameters[displayOption] = optionValue;
|
|
|
|
}
|
|
|
|
newNodeData.parameters = {
|
|
|
|
...newNodeData.parameters,
|
|
|
|
...parameters,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return newNodeData;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resolves the data for a new node
|
|
|
|
*/
|
|
|
|
async function resolveNodeData(node: AddNodeData, options: AddNodeOptions = {}) {
|
|
|
|
const nodeTypeDescription: INodeTypeDescription | null = nodeTypesStore.getNodeType(node.type);
|
|
|
|
if (nodeTypeDescription === null) {
|
|
|
|
toast.showMessage({
|
|
|
|
title: i18n.baseText('nodeView.showMessage.addNodeButton.title'),
|
|
|
|
message: i18n.baseText('nodeView.showMessage.addNodeButton.message', {
|
|
|
|
interpolate: { nodeTypeName: node.type },
|
|
|
|
}),
|
|
|
|
type: 'error',
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (
|
|
|
|
nodeTypeDescription.maxNodes !== undefined &&
|
|
|
|
workflowHelpers.getNodeTypeCount(node.type) >= nodeTypeDescription.maxNodes
|
|
|
|
) {
|
|
|
|
showMaxNodeTypeError(nodeTypeDescription);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const newNodeData = await initializeNodeDataWithDefaultCredentials(node);
|
|
|
|
|
|
|
|
// When pulling new connection from node or injecting into a connection
|
|
|
|
const lastSelectedNode = uiStore.getLastSelectedNode;
|
|
|
|
|
|
|
|
if (node.position) {
|
|
|
|
newNodeData.position = NodeViewUtils.getNewNodePosition(
|
|
|
|
canvasStore.getNodesWithPlaceholderNode(),
|
|
|
|
node.position,
|
|
|
|
);
|
|
|
|
} else if (lastSelectedNode) {
|
|
|
|
// @TODO Implement settings lastSelectedConnection for new canvas
|
|
|
|
const lastSelectedConnection = canvasStore.lastSelectedConnection;
|
|
|
|
if (lastSelectedConnection) {
|
|
|
|
// set when injecting into a connection
|
|
|
|
const [diffX] = NodeViewUtils.getConnectorLengths(lastSelectedConnection);
|
|
|
|
if (diffX <= NodeViewUtils.MAX_X_TO_PUSH_DOWNSTREAM_NODES) {
|
|
|
|
pushDownstreamNodes(lastSelectedNode.name, NodeViewUtils.PUSH_NODES_OFFSET, {
|
|
|
|
trackHistory: options.trackHistory,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// This position is set in `onMouseUp` when pulling connections
|
|
|
|
if (canvasStore.newNodeInsertPosition) {
|
|
|
|
newNodeData.position = NodeViewUtils.getNewNodePosition(workflowsStore.allNodes, [
|
|
|
|
canvasStore.newNodeInsertPosition[0] + NodeViewUtils.GRID_SIZE,
|
|
|
|
canvasStore.newNodeInsertPosition[1] - NodeViewUtils.NODE_SIZE / 2,
|
|
|
|
]);
|
|
|
|
canvasStore.newNodeInsertPosition = null;
|
|
|
|
} else {
|
|
|
|
let yOffset = 0;
|
|
|
|
if (lastSelectedConnection) {
|
|
|
|
const sourceNodeType = nodeTypesStore.getNodeType(
|
|
|
|
lastSelectedNode.type,
|
|
|
|
lastSelectedNode.typeVersion,
|
|
|
|
);
|
|
|
|
|
|
|
|
if (sourceNodeType) {
|
|
|
|
const offsets = [
|
|
|
|
[-100, 100],
|
|
|
|
[-140, 0, 140],
|
|
|
|
[-240, -100, 100, 240],
|
|
|
|
];
|
|
|
|
|
|
|
|
const sourceNodeOutputs = NodeHelpers.getNodeOutputs(
|
2024-07-08 03:25:18 -07:00
|
|
|
editableWorkflowObject.value,
|
2024-06-25 02:11:44 -07:00
|
|
|
lastSelectedNode,
|
|
|
|
sourceNodeType,
|
|
|
|
);
|
|
|
|
const sourceNodeOutputTypes = NodeHelpers.getConnectionTypes(sourceNodeOutputs);
|
|
|
|
|
|
|
|
const sourceNodeOutputMainOutputs = sourceNodeOutputTypes.filter(
|
|
|
|
(output) => output === NodeConnectionType.Main,
|
|
|
|
);
|
|
|
|
|
|
|
|
if (sourceNodeOutputMainOutputs.length > 1) {
|
|
|
|
const offset = offsets[sourceNodeOutputMainOutputs.length - 2];
|
|
|
|
const sourceOutputIndex = lastSelectedConnection.__meta
|
|
|
|
? lastSelectedConnection.__meta.sourceOutputIndex
|
|
|
|
: 0;
|
|
|
|
yOffset = offset[sourceOutputIndex];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let outputs: Array<ConnectionTypes | INodeOutputConfiguration> = [];
|
|
|
|
try {
|
|
|
|
// It fails when the outputs are an expression. As those nodes have
|
|
|
|
// normally no outputs by default and the only reason we need the
|
|
|
|
// outputs here is to calculate the position, it is fine to assume
|
|
|
|
// that they have no outputs and are so treated as a regular node
|
|
|
|
// with only "main" outputs.
|
2024-07-08 03:25:18 -07:00
|
|
|
outputs = NodeHelpers.getNodeOutputs(
|
|
|
|
editableWorkflowObject.value,
|
|
|
|
newNodeData,
|
|
|
|
nodeTypeDescription,
|
|
|
|
);
|
2024-06-25 02:11:44 -07:00
|
|
|
} catch (e) {}
|
|
|
|
const outputTypes = NodeHelpers.getConnectionTypes(outputs);
|
|
|
|
const lastSelectedNodeType = nodeTypesStore.getNodeType(
|
|
|
|
lastSelectedNode.type,
|
|
|
|
lastSelectedNode.typeVersion,
|
|
|
|
);
|
|
|
|
|
|
|
|
// If node has only scoped outputs, position it below the last selected node
|
|
|
|
if (
|
|
|
|
outputTypes.length > 0 &&
|
|
|
|
outputTypes.every((outputName) => outputName !== NodeConnectionType.Main)
|
|
|
|
) {
|
2024-07-08 03:25:18 -07:00
|
|
|
const lastSelectedNodeWorkflow = editableWorkflowObject.value.getNode(
|
|
|
|
lastSelectedNode.name,
|
|
|
|
);
|
2024-06-25 02:11:44 -07:00
|
|
|
if (!lastSelectedNodeWorkflow || !lastSelectedNodeType) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const lastSelectedInputs = NodeHelpers.getNodeInputs(
|
2024-07-08 03:25:18 -07:00
|
|
|
editableWorkflowObject.value,
|
2024-06-25 02:11:44 -07:00
|
|
|
lastSelectedNodeWorkflow,
|
|
|
|
lastSelectedNodeType,
|
|
|
|
);
|
|
|
|
const lastSelectedInputTypes = NodeHelpers.getConnectionTypes(lastSelectedInputs);
|
|
|
|
|
|
|
|
const scopedConnectionIndex = (lastSelectedInputTypes || [])
|
|
|
|
.filter((input) => input !== NodeConnectionType.Main)
|
|
|
|
.findIndex((inputType) => outputs[0] === inputType);
|
|
|
|
|
|
|
|
newNodeData.position = NodeViewUtils.getNewNodePosition(
|
|
|
|
workflowsStore.allNodes,
|
|
|
|
[
|
|
|
|
lastSelectedNode.position[0] +
|
|
|
|
(NodeViewUtils.NODE_SIZE /
|
|
|
|
(Math.max(lastSelectedNodeType?.inputs?.length ?? 1), 1)) *
|
|
|
|
scopedConnectionIndex,
|
|
|
|
lastSelectedNode.position[1] + NodeViewUtils.PUSH_NODES_OFFSET,
|
|
|
|
],
|
|
|
|
[100, 0],
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
if (!lastSelectedNodeType) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Has only main outputs or no outputs at all
|
|
|
|
const inputs = NodeHelpers.getNodeInputs(
|
2024-07-08 03:25:18 -07:00
|
|
|
editableWorkflowObject.value,
|
2024-06-25 02:11:44 -07:00
|
|
|
lastSelectedNode,
|
|
|
|
lastSelectedNodeType,
|
|
|
|
);
|
|
|
|
const inputsTypes = NodeHelpers.getConnectionTypes(inputs);
|
|
|
|
|
|
|
|
let pushOffset = NodeViewUtils.PUSH_NODES_OFFSET;
|
|
|
|
if (!!inputsTypes.find((input) => input !== NodeConnectionType.Main)) {
|
|
|
|
// If the node has scoped inputs, push it down a bit more
|
|
|
|
pushOffset += 150;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If a node is active then add the new node directly after the current one
|
|
|
|
newNodeData.position = NodeViewUtils.getNewNodePosition(
|
|
|
|
workflowsStore.allNodes,
|
|
|
|
[lastSelectedNode.position[0] + pushOffset, lastSelectedNode.position[1] + yOffset],
|
|
|
|
[100, 0],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// If added node is a trigger and it's the first one added to the canvas
|
|
|
|
// we place it at canvasAddButtonPosition to replace the canvas add button
|
|
|
|
const position =
|
|
|
|
nodeTypesStore.isTriggerNode(node.type) && triggerNodes.value.length === 0
|
|
|
|
? canvasStore.canvasAddButtonPosition
|
|
|
|
: // If no node is active find a free spot
|
|
|
|
(lastClickPosition.value as XYPosition);
|
|
|
|
|
|
|
|
newNodeData.position = NodeViewUtils.getNewNodePosition(workflowsStore.allNodes, position);
|
|
|
|
}
|
|
|
|
|
|
|
|
const localizedName = i18n.localizeNodeName(newNodeData.name, newNodeData.type);
|
|
|
|
|
|
|
|
newNodeData.name = getUniqueNodeName(localizedName, workflowsStore.canvasNames);
|
|
|
|
|
|
|
|
if (nodeTypeDescription.webhooks?.length) {
|
|
|
|
newNodeData.webhookId = uuid();
|
|
|
|
}
|
|
|
|
|
2024-07-10 01:53:27 -07:00
|
|
|
// if it's a webhook and the path is empty set the UUID as the default path
|
|
|
|
if (
|
|
|
|
[WEBHOOK_NODE_TYPE, FORM_TRIGGER_NODE_TYPE].includes(newNodeData.type) &&
|
|
|
|
newNodeData.parameters.path === ''
|
|
|
|
) {
|
|
|
|
newNodeData.parameters.path = newNodeData.webhookId as string;
|
|
|
|
}
|
|
|
|
|
2024-06-25 02:11:44 -07:00
|
|
|
workflowsStore.setNodePristine(newNodeData.name, true);
|
|
|
|
uiStore.stateIsDirty = true;
|
|
|
|
|
|
|
|
if (node.type === STICKY_NODE_TYPE) {
|
|
|
|
telemetry.trackNodesPanel('nodeView.addSticky', {
|
|
|
|
workflow_id: workflowsStore.workflowId,
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
void externalHooks.run('nodeView.addNodeButton', { nodeTypeName: node.type });
|
|
|
|
const trackProperties: ITelemetryTrackProperties = {
|
|
|
|
node_type: node.type,
|
|
|
|
node_version: newNodeData.typeVersion,
|
|
|
|
is_auto_add: options.isAutoAdd,
|
|
|
|
workflow_id: workflowsStore.workflowId,
|
|
|
|
drag_and_drop: options.dragAndDrop,
|
|
|
|
};
|
|
|
|
|
|
|
|
if (lastSelectedNode) {
|
|
|
|
trackProperties.input_node_type = lastSelectedNode.type;
|
|
|
|
}
|
|
|
|
|
|
|
|
telemetry.trackNodesPanel('nodeView.addNodeButton', trackProperties);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Automatically deselect all nodes and select the current one and also active
|
|
|
|
// current node. But only if it's added manually by the user (not by undo/redo mechanism)
|
|
|
|
// @TODO
|
|
|
|
// if (trackHistory) {
|
|
|
|
// this.deselectAllNodes();
|
|
|
|
// setTimeout(() => {
|
|
|
|
// this.nodeSelectedByName(newNodeData.name, showDetail && nodeTypeName !== STICKY_NODE_TYPE);
|
|
|
|
// });
|
|
|
|
// }
|
|
|
|
|
|
|
|
return newNodeData;
|
|
|
|
}
|
|
|
|
|
|
|
|
function pushDownstreamNodes(
|
|
|
|
sourceNodeName: string,
|
|
|
|
margin: number,
|
|
|
|
{ trackHistory = false }: { trackHistory?: boolean },
|
|
|
|
) {
|
|
|
|
const sourceNode = workflowsStore.nodesByName[sourceNodeName];
|
2024-07-08 03:25:18 -07:00
|
|
|
const checkNodes = workflowHelpers.getConnectedNodes(
|
|
|
|
'downstream',
|
|
|
|
editableWorkflowObject.value,
|
|
|
|
sourceNodeName,
|
|
|
|
);
|
2024-06-25 02:11:44 -07:00
|
|
|
for (const nodeName of checkNodes) {
|
|
|
|
const node = workflowsStore.nodesByName[nodeName];
|
|
|
|
const oldPosition = node.position;
|
|
|
|
|
|
|
|
if (node.position[0] < sourceNode.position[0]) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const updateInformation: INodeUpdatePropertiesInformation = {
|
|
|
|
name: nodeName,
|
|
|
|
properties: {
|
|
|
|
position: [node.position[0] + margin, node.position[1]],
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
workflowsStore.updateNodeProperties(updateInformation);
|
|
|
|
updateNodePosition(node.id, { x: node.position[0], y: node.position[1] });
|
|
|
|
|
|
|
|
if (
|
|
|
|
(trackHistory && oldPosition[0] !== updateInformation.properties.position[0]) ||
|
|
|
|
oldPosition[1] !== updateInformation.properties.position[1]
|
|
|
|
) {
|
|
|
|
historyStore.pushCommandToUndo(
|
|
|
|
new MoveNodeCommand(nodeName, oldPosition, updateInformation.properties.position),
|
|
|
|
trackHistory,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function loadNodeTypesProperties(nodeInfos: INodeTypeNameVersion[]): Promise<void> {
|
|
|
|
const allNodeTypeDescriptions: INodeTypeDescription[] = nodeTypesStore.allNodeTypes;
|
|
|
|
|
|
|
|
const nodesToBeFetched: INodeTypeNameVersion[] = [];
|
|
|
|
allNodeTypeDescriptions.forEach((nodeTypeDescription) => {
|
|
|
|
const nodeVersions = Array.isArray(nodeTypeDescription.version)
|
|
|
|
? nodeTypeDescription.version
|
|
|
|
: [nodeTypeDescription.version];
|
|
|
|
if (
|
|
|
|
!!nodeInfos.find(
|
|
|
|
(n) => n.name === nodeTypeDescription.name && nodeVersions.includes(n.version),
|
|
|
|
) &&
|
|
|
|
!nodeTypeDescription.hasOwnProperty('properties')
|
|
|
|
) {
|
|
|
|
nodesToBeFetched.push({
|
|
|
|
name: nodeTypeDescription.name,
|
|
|
|
version: Array.isArray(nodeTypeDescription.version)
|
|
|
|
? nodeTypeDescription.version.slice(-1)[0]
|
|
|
|
: nodeTypeDescription.version,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
if (nodesToBeFetched.length > 0) {
|
|
|
|
// Only call API if node information is actually missing
|
|
|
|
await nodeTypesStore.getNodesInformation(nodesToBeFetched);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function showMaxNodeTypeError(nodeTypeDescription: INodeTypeDescription) {
|
|
|
|
const maxNodes = nodeTypeDescription.maxNodes;
|
|
|
|
toast.showMessage({
|
|
|
|
title: i18n.baseText('nodeView.showMessage.showMaxNodeTypeError.title'),
|
|
|
|
message: i18n.baseText('nodeView.showMessage.showMaxNodeTypeError.message', {
|
|
|
|
adjustToNumber: maxNodes,
|
|
|
|
interpolate: { nodeTypeDataDisplayName: nodeTypeDescription.displayName },
|
|
|
|
}),
|
|
|
|
type: 'error',
|
|
|
|
duration: 0,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-06-04 05:36:27 -07:00
|
|
|
/**
|
|
|
|
* Connection operations
|
|
|
|
*/
|
|
|
|
|
|
|
|
function createConnection(connection: Connection) {
|
|
|
|
const sourceNode = workflowsStore.getNodeById(connection.source);
|
|
|
|
const targetNode = workflowsStore.getNodeById(connection.target);
|
2024-06-25 02:11:44 -07:00
|
|
|
if (!sourceNode || !targetNode) {
|
2024-06-04 05:36:27 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const mappedConnection = mapCanvasConnectionToLegacyConnection(
|
|
|
|
sourceNode,
|
|
|
|
targetNode,
|
|
|
|
connection,
|
|
|
|
);
|
2024-06-25 02:11:44 -07:00
|
|
|
|
|
|
|
if (!isConnectionAllowed(sourceNode, targetNode, mappedConnection[1].type)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-06-04 05:36:27 -07:00
|
|
|
workflowsStore.addConnection({
|
|
|
|
connection: mappedConnection,
|
|
|
|
});
|
|
|
|
|
2024-07-08 03:25:18 -07:00
|
|
|
nodeHelpers.updateNodeInputIssues(sourceNode);
|
|
|
|
nodeHelpers.updateNodeInputIssues(targetNode);
|
|
|
|
|
2024-06-04 05:36:27 -07:00
|
|
|
uiStore.stateIsDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
function deleteConnection(
|
|
|
|
connection: Connection,
|
|
|
|
{ trackHistory = false, trackBulk = true } = {},
|
|
|
|
) {
|
|
|
|
const sourceNode = workflowsStore.getNodeById(connection.source);
|
|
|
|
const targetNode = workflowsStore.getNodeById(connection.target);
|
|
|
|
if (!sourceNode || !targetNode) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const mappedConnection = mapCanvasConnectionToLegacyConnection(
|
|
|
|
sourceNode,
|
|
|
|
targetNode,
|
|
|
|
connection,
|
|
|
|
);
|
|
|
|
|
|
|
|
if (trackHistory && trackBulk) {
|
|
|
|
historyStore.startRecordingUndo();
|
|
|
|
}
|
|
|
|
|
|
|
|
workflowsStore.removeConnection({
|
|
|
|
connection: mappedConnection,
|
|
|
|
});
|
|
|
|
|
|
|
|
if (trackHistory) {
|
|
|
|
historyStore.pushCommandToUndo(new RemoveConnectionCommand(mappedConnection));
|
|
|
|
|
|
|
|
if (trackBulk) {
|
|
|
|
historyStore.stopRecordingUndo();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function revertDeleteConnection(connection: [IConnection, IConnection]) {
|
|
|
|
workflowsStore.addConnection({
|
|
|
|
connection,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-06-25 02:11:44 -07:00
|
|
|
function isConnectionAllowed(
|
|
|
|
sourceNode: INodeUi,
|
|
|
|
targetNode: INodeUi,
|
2024-07-08 03:25:18 -07:00
|
|
|
connectionType: NodeConnectionType,
|
2024-06-25 02:11:44 -07:00
|
|
|
): boolean {
|
2024-07-08 03:25:18 -07:00
|
|
|
if (sourceNode.id === targetNode.id) {
|
|
|
|
return false;
|
|
|
|
}
|
2024-06-25 02:11:44 -07:00
|
|
|
|
2024-07-08 03:25:18 -07:00
|
|
|
const targetNodeType = nodeTypesStore.getNodeType(targetNode.type, targetNode.typeVersion);
|
2024-06-25 02:11:44 -07:00
|
|
|
if (targetNodeType?.inputs?.length) {
|
2024-07-08 03:25:18 -07:00
|
|
|
const workflowNode = editableWorkflowObject.value.getNode(targetNode.name);
|
2024-06-25 02:11:44 -07:00
|
|
|
if (!workflowNode) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
let inputs: Array<ConnectionTypes | INodeInputConfiguration> = [];
|
|
|
|
if (targetNodeType) {
|
2024-07-08 03:25:18 -07:00
|
|
|
inputs =
|
|
|
|
NodeHelpers.getNodeInputs(editableWorkflowObject.value, workflowNode, targetNodeType) ||
|
|
|
|
[];
|
2024-06-25 02:11:44 -07:00
|
|
|
}
|
|
|
|
|
2024-07-08 03:25:18 -07:00
|
|
|
let targetHasConnectionTypeAsInput = false;
|
2024-06-25 02:11:44 -07:00
|
|
|
for (const input of inputs) {
|
2024-07-08 03:25:18 -07:00
|
|
|
const inputType = typeof input === 'string' ? input : input.type;
|
|
|
|
if (inputType === connectionType) {
|
|
|
|
if (typeof input === 'object' && 'filter' in input && input.filter?.nodes.length) {
|
|
|
|
if (!input.filter.nodes.includes(sourceNode.type)) {
|
|
|
|
// this.dropPrevented = true;
|
|
|
|
toast.showToast({
|
|
|
|
title: i18n.baseText('nodeView.showError.nodeNodeCompatible.title'),
|
|
|
|
message: i18n.baseText('nodeView.showError.nodeNodeCompatible.message', {
|
|
|
|
interpolate: { sourceNodeName: sourceNode.name, targetNodeName: targetNode.name },
|
|
|
|
}),
|
|
|
|
type: 'error',
|
|
|
|
duration: 5000,
|
|
|
|
});
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2024-06-25 02:11:44 -07:00
|
|
|
}
|
2024-07-08 03:25:18 -07:00
|
|
|
|
|
|
|
targetHasConnectionTypeAsInput = true;
|
2024-06-25 02:11:44 -07:00
|
|
|
}
|
|
|
|
}
|
2024-07-08 03:25:18 -07:00
|
|
|
|
|
|
|
return targetHasConnectionTypeAsInput;
|
2024-06-25 02:11:44 -07:00
|
|
|
}
|
|
|
|
|
2024-07-08 03:25:18 -07:00
|
|
|
return false;
|
2024-06-04 05:36:27 -07:00
|
|
|
}
|
|
|
|
|
2024-06-25 02:11:44 -07:00
|
|
|
async function addConnections(
|
|
|
|
connections: AddedNodesAndConnections['connections'],
|
|
|
|
{ offsetIndex }: { offsetIndex: number },
|
|
|
|
) {
|
|
|
|
for (const { from, to } of connections) {
|
|
|
|
const fromNode = editableWorkflow.value.nodes[offsetIndex + from.nodeIndex];
|
|
|
|
const toNode = editableWorkflow.value.nodes[offsetIndex + to.nodeIndex];
|
|
|
|
|
|
|
|
createConnection({
|
|
|
|
source: fromNode.id,
|
|
|
|
sourceHandle: `outputs/${NodeConnectionType.Main}/${from.outputIndex ?? 0}`,
|
|
|
|
target: toNode.id,
|
|
|
|
targetHandle: `inputs/${NodeConnectionType.Main}/${to.inputIndex ?? 0}`,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-04 05:36:27 -07:00
|
|
|
return {
|
2024-06-25 02:11:44 -07:00
|
|
|
editableWorkflow,
|
|
|
|
editableWorkflowObject,
|
|
|
|
triggerNodes,
|
|
|
|
initializeNodeDataWithDefaultCredentials,
|
|
|
|
addNodes,
|
2024-06-04 05:36:27 -07:00
|
|
|
updateNodePosition,
|
2024-06-17 05:46:55 -07:00
|
|
|
setNodeActive,
|
|
|
|
setNodeActiveByName,
|
2024-06-25 02:11:44 -07:00
|
|
|
setNodeSelected,
|
2024-06-26 06:56:58 -07:00
|
|
|
toggleNodeDisabled,
|
2024-06-17 05:46:55 -07:00
|
|
|
renameNode,
|
|
|
|
revertRenameNode,
|
2024-06-04 05:36:27 -07:00
|
|
|
deleteNode,
|
|
|
|
revertDeleteNode,
|
2024-06-25 02:11:44 -07:00
|
|
|
addConnections,
|
2024-06-04 05:36:27 -07:00
|
|
|
createConnection,
|
|
|
|
deleteConnection,
|
|
|
|
revertDeleteConnection,
|
2024-07-08 03:25:18 -07:00
|
|
|
isConnectionAllowed,
|
2024-06-04 05:36:27 -07:00
|
|
|
};
|
|
|
|
}
|