2023-11-20 05:37:12 -08:00
|
|
|
import type { XYPosition } from '@/Interface';
|
|
|
|
import {
|
|
|
|
NOT_DUPLICATABE_NODE_TYPES,
|
|
|
|
PIN_DATA_NODE_TYPES_DENYLIST,
|
|
|
|
STICKY_NODE_TYPE,
|
|
|
|
} from '@/constants';
|
2023-11-28 03:15:08 -08:00
|
|
|
import { useNodeTypesStore } from '@/stores/nodeTypes.store';
|
|
|
|
import { useSourceControlStore } from '@/stores/sourceControl.store';
|
|
|
|
import { useUIStore } from '@/stores/ui.store';
|
|
|
|
import { useWorkflowsStore } from '@/stores/workflows.store';
|
2023-11-20 05:37:12 -08:00
|
|
|
import type { IActionDropdownItem } from 'n8n-design-system/src/components/N8nActionDropdown/ActionDropdown.vue';
|
|
|
|
import type { INode, INodeTypeDescription } from 'n8n-workflow';
|
|
|
|
import { computed, ref, watch } from 'vue';
|
|
|
|
import { getMousePosition } from '../utils/nodeViewUtils';
|
|
|
|
import { useI18n } from './useI18n';
|
|
|
|
import { useDataSchema } from './useDataSchema';
|
|
|
|
|
|
|
|
export type ContextMenuTarget =
|
|
|
|
| { source: 'canvas' }
|
|
|
|
| { source: 'node-right-click'; node: INode }
|
|
|
|
| { source: 'node-button'; node: INode };
|
2023-11-23 08:10:45 -08:00
|
|
|
export type ContextMenuActionCallback = (action: ContextMenuAction, targets: INode[]) => void;
|
2023-11-20 05:37:12 -08:00
|
|
|
export type ContextMenuAction =
|
|
|
|
| 'open'
|
|
|
|
| 'copy'
|
|
|
|
| 'toggle_activation'
|
|
|
|
| 'duplicate'
|
|
|
|
| 'execute'
|
|
|
|
| 'rename'
|
|
|
|
| 'toggle_pin'
|
|
|
|
| 'delete'
|
|
|
|
| 'select_all'
|
|
|
|
| 'deselect_all'
|
|
|
|
| 'add_node'
|
2023-11-23 08:10:45 -08:00
|
|
|
| 'add_sticky'
|
|
|
|
| 'change_color';
|
2023-11-20 05:37:12 -08:00
|
|
|
|
|
|
|
const position = ref<XYPosition>([0, 0]);
|
|
|
|
const isOpen = ref(false);
|
|
|
|
const target = ref<ContextMenuTarget>({ source: 'canvas' });
|
|
|
|
const actions = ref<IActionDropdownItem[]>([]);
|
2023-11-23 08:10:45 -08:00
|
|
|
const actionCallback = ref<ContextMenuActionCallback>(() => {});
|
2023-11-20 05:37:12 -08:00
|
|
|
|
2023-11-23 08:10:45 -08:00
|
|
|
export const useContextMenu = (onAction: ContextMenuActionCallback = () => {}) => {
|
2023-11-20 05:37:12 -08:00
|
|
|
const uiStore = useUIStore();
|
|
|
|
const nodeTypesStore = useNodeTypesStore();
|
|
|
|
const workflowsStore = useWorkflowsStore();
|
|
|
|
const sourceControlStore = useSourceControlStore();
|
|
|
|
const { getInputDataWithPinned } = useDataSchema();
|
|
|
|
const i18n = useI18n();
|
|
|
|
|
|
|
|
const isReadOnly = computed(
|
|
|
|
() => sourceControlStore.preferences.branchReadOnly || uiStore.isReadOnlyView,
|
|
|
|
);
|
|
|
|
|
|
|
|
const targetNodes = computed(() => {
|
|
|
|
if (!isOpen.value) return [];
|
|
|
|
const selectedNodes = uiStore.selectedNodes.map((node) =>
|
|
|
|
workflowsStore.getNodeByName(node.name),
|
|
|
|
) as INode[];
|
|
|
|
const currentTarget = target.value;
|
|
|
|
if (currentTarget.source === 'canvas') {
|
|
|
|
return selectedNodes;
|
|
|
|
} else if (currentTarget.source === 'node-right-click') {
|
|
|
|
const isNodeInSelection = selectedNodes.some((node) => node.name === currentTarget.node.name);
|
|
|
|
return isNodeInSelection ? selectedNodes : [currentTarget.node];
|
|
|
|
}
|
|
|
|
|
|
|
|
return [currentTarget.node];
|
|
|
|
});
|
|
|
|
|
|
|
|
const canAddNodeOfType = (nodeType: INodeTypeDescription) => {
|
|
|
|
const sameTypeNodes = workflowsStore.allNodes.filter((n) => n.type === nodeType.name);
|
|
|
|
return nodeType.maxNodes === undefined || sameTypeNodes.length < nodeType.maxNodes;
|
|
|
|
};
|
|
|
|
|
|
|
|
const canDuplicateNode = (node: INode): boolean => {
|
|
|
|
const nodeType = nodeTypesStore.getNodeType(node.type, node.typeVersion);
|
|
|
|
if (!nodeType) return false;
|
|
|
|
if (NOT_DUPLICATABE_NODE_TYPES.includes(nodeType.name)) return false;
|
|
|
|
|
|
|
|
return canAddNodeOfType(nodeType);
|
|
|
|
};
|
|
|
|
|
|
|
|
const canPinNode = (node: INode): boolean => {
|
|
|
|
const nodeType = nodeTypesStore.getNodeType(node.type, node.typeVersion);
|
|
|
|
const dataToPin = getInputDataWithPinned(node);
|
|
|
|
if (!nodeType || dataToPin.length === 0) return false;
|
|
|
|
return nodeType.outputs.length === 1 && !PIN_DATA_NODE_TYPES_DENYLIST.includes(node.type);
|
|
|
|
};
|
|
|
|
|
|
|
|
const hasPinData = (node: INode): boolean => {
|
|
|
|
return !!workflowsStore.pinDataByNodeName(node.name);
|
|
|
|
};
|
|
|
|
const close = () => {
|
|
|
|
target.value = { source: 'canvas' };
|
|
|
|
isOpen.value = false;
|
|
|
|
actions.value = [];
|
|
|
|
position.value = [0, 0];
|
|
|
|
};
|
|
|
|
|
|
|
|
const open = (event: MouseEvent, menuTarget: ContextMenuTarget = { source: 'canvas' }) => {
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
|
|
if (isOpen.value && menuTarget.source === target.value.source) {
|
|
|
|
// Close context menu, let browser open native context menu
|
|
|
|
close();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
2023-11-23 08:10:45 -08:00
|
|
|
actionCallback.value = onAction;
|
2023-11-20 05:37:12 -08:00
|
|
|
target.value = menuTarget;
|
|
|
|
position.value = getMousePosition(event);
|
|
|
|
isOpen.value = true;
|
|
|
|
|
|
|
|
const nodes = targetNodes.value;
|
|
|
|
const onlyStickies = nodes.every((node) => node.type === STICKY_NODE_TYPE);
|
|
|
|
const i18nOptions = {
|
|
|
|
adjustToNumber: nodes.length,
|
|
|
|
interpolate: {
|
|
|
|
subject: onlyStickies
|
|
|
|
? i18n.baseText('contextMenu.sticky', { adjustToNumber: nodes.length })
|
|
|
|
: i18n.baseText('contextMenu.node', { adjustToNumber: nodes.length }),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
const selectionActions = [
|
|
|
|
{
|
|
|
|
id: 'select_all',
|
|
|
|
divided: true,
|
|
|
|
label: i18n.baseText('contextMenu.selectAll'),
|
|
|
|
shortcut: { metaKey: true, keys: ['A'] },
|
|
|
|
disabled: nodes.length === workflowsStore.allNodes.length,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 'deselect_all',
|
|
|
|
label: i18n.baseText('contextMenu.deselectAll'),
|
|
|
|
disabled: nodes.length === 0,
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
if (nodes.length === 0) {
|
|
|
|
actions.value = [
|
|
|
|
{
|
|
|
|
id: 'add_node',
|
|
|
|
shortcut: { keys: ['Tab'] },
|
|
|
|
label: i18n.baseText('contextMenu.addNode'),
|
|
|
|
disabled: isReadOnly.value,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 'add_sticky',
|
|
|
|
shortcut: { shiftKey: true, keys: ['s'] },
|
|
|
|
label: i18n.baseText('contextMenu.addSticky'),
|
|
|
|
disabled: isReadOnly.value,
|
|
|
|
},
|
|
|
|
...selectionActions,
|
|
|
|
];
|
|
|
|
} else {
|
|
|
|
const menuActions: IActionDropdownItem[] = [
|
|
|
|
!onlyStickies && {
|
|
|
|
id: 'toggle_activation',
|
|
|
|
label: nodes.every((node) => node.disabled)
|
|
|
|
? i18n.baseText('contextMenu.activate', i18nOptions)
|
|
|
|
: i18n.baseText('contextMenu.deactivate', i18nOptions),
|
|
|
|
shortcut: { keys: ['D'] },
|
|
|
|
disabled: isReadOnly.value,
|
|
|
|
},
|
|
|
|
!onlyStickies && {
|
|
|
|
id: 'toggle_pin',
|
|
|
|
label: nodes.every((node) => hasPinData(node))
|
|
|
|
? i18n.baseText('contextMenu.unpin', i18nOptions)
|
|
|
|
: i18n.baseText('contextMenu.pin', i18nOptions),
|
|
|
|
shortcut: { keys: ['p'] },
|
|
|
|
disabled: isReadOnly.value || !nodes.every(canPinNode),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 'copy',
|
|
|
|
label: i18n.baseText('contextMenu.copy', i18nOptions),
|
|
|
|
shortcut: { metaKey: true, keys: ['C'] },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 'duplicate',
|
|
|
|
label: i18n.baseText('contextMenu.duplicate', i18nOptions),
|
|
|
|
shortcut: { metaKey: true, keys: ['D'] },
|
|
|
|
disabled: isReadOnly.value || !nodes.every(canDuplicateNode),
|
|
|
|
},
|
|
|
|
...selectionActions,
|
|
|
|
{
|
|
|
|
id: 'delete',
|
|
|
|
divided: true,
|
|
|
|
label: i18n.baseText('contextMenu.delete', i18nOptions),
|
|
|
|
shortcut: { keys: ['Del'] },
|
|
|
|
disabled: isReadOnly.value,
|
|
|
|
},
|
|
|
|
].filter(Boolean) as IActionDropdownItem[];
|
|
|
|
|
|
|
|
if (nodes.length === 1) {
|
|
|
|
const singleNodeActions = onlyStickies
|
|
|
|
? [
|
|
|
|
{
|
|
|
|
id: 'open',
|
|
|
|
label: i18n.baseText('contextMenu.editSticky'),
|
|
|
|
shortcut: { keys: ['↵'] },
|
2023-11-23 08:10:45 -08:00
|
|
|
disabled: isReadOnly.value,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 'change_color',
|
|
|
|
label: i18n.baseText('contextMenu.changeColor'),
|
|
|
|
disabled: isReadOnly.value,
|
2023-11-20 05:37:12 -08:00
|
|
|
},
|
|
|
|
]
|
|
|
|
: [
|
|
|
|
{
|
|
|
|
id: 'open',
|
|
|
|
label: i18n.baseText('contextMenu.open'),
|
|
|
|
shortcut: { keys: ['↵'] },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 'execute',
|
2024-01-05 07:23:51 -08:00
|
|
|
label: i18n.baseText('contextMenu.test'),
|
2023-11-23 08:10:45 -08:00
|
|
|
disabled: isReadOnly.value,
|
2023-11-20 05:37:12 -08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 'rename',
|
|
|
|
label: i18n.baseText('contextMenu.rename'),
|
|
|
|
shortcut: { keys: ['F2'] },
|
|
|
|
disabled: isReadOnly.value,
|
|
|
|
},
|
|
|
|
];
|
|
|
|
// Add actions only available for a single node
|
|
|
|
menuActions.unshift(...singleNodeActions);
|
|
|
|
}
|
|
|
|
|
|
|
|
actions.value = menuActions;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2023-11-23 08:10:45 -08:00
|
|
|
const _dispatchAction = (action: ContextMenuAction) => {
|
|
|
|
actionCallback.value(action, targetNodes.value);
|
|
|
|
};
|
|
|
|
|
2023-11-20 05:37:12 -08:00
|
|
|
watch(
|
|
|
|
() => uiStore.nodeViewOffsetPosition,
|
|
|
|
() => {
|
|
|
|
close();
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
return {
|
|
|
|
isOpen,
|
|
|
|
position,
|
|
|
|
target,
|
|
|
|
actions,
|
|
|
|
targetNodes,
|
|
|
|
open,
|
|
|
|
close,
|
2023-11-23 08:10:45 -08:00
|
|
|
_dispatchAction,
|
2023-11-20 05:37:12 -08:00
|
|
|
};
|
|
|
|
};
|