diff --git a/packages/editor-ui/src/components/Node.vue b/packages/editor-ui/src/components/Node.vue index 1425659d1a..4914be06b6 100644 --- a/packages/editor-ui/src/components/Node.vue +++ b/packages/editor-ui/src/components/Node.vue @@ -50,13 +50,9 @@
-
+

- {{ this.$headerText({ - key: `headers.${shortNodeType}.displayName`, - fallback: data.name, - }) - }} + {{ nodeTitle }}

({{ $baseText('node.disabled') }}}

@@ -164,6 +160,23 @@ export default mixins(externalHooks, nodeBase, nodeHelpers, renderText, workflow shortNodeType (): string { return this.$shortNodeType(this.data.type); }, + nodeTitle (): string { + const node = this.data; + + const nodeName = this.$headerText({ + key: `headers.${this.$shortNodeType(node.type)}.displayName`, + fallback: node.name, + }); + + if (!/\d$/.test(node.name)) return nodeName; + + const nativeDuplicateSuffix = this.getDuplicateSuffix(node, { fromNative: true }); + + if (nativeDuplicateSuffix) return nodeName + nativeDuplicateSuffix; + + return nodeName + this.getDuplicateSuffix(node, { fromStandard: true }); + + }, waiting (): string | undefined { const workflowExecution = this.$store.getters.getWorkflowExecution; @@ -272,6 +285,35 @@ export default mixins(externalHooks, nodeBase, nodeHelpers, renderText, workflow this.$emit('duplicateNode', this.data.name); }); }, + + /** + * Extract the duplicate number suffix from a node name: + * - from a node name natively ending in a number, e.g. `'S31'` → `'1'` + * - from a standard node name, e.g. `'GitHub1'` → `'1'` + */ + getDuplicateSuffix( + node: INodeUi, + { fromNative, fromStandard }: { fromNative?: true; fromStandard?: true; }, + ) { + if (fromNative) { + const { nativelyNumberSuffixedNodeNames: natives } = this.$store.getters; + const found = natives.find((native: string) => node.name.includes(native)); + + if (!found) return null; + + return node.name.split(found).pop()!; + } + + if (fromStandard) { + const match = node.name.match(/(.*)(?\d)$/); + if (!match || !match.groups || !match.groups.duplicateSuffix) return null; + + return match.groups.duplicateSuffix; + } + + throw new Error('Either "fromNative" or "fromStandard" must be specified'); + }, + setNodeActive () { this.$store.commit('setActiveNode', this.data.name); }, diff --git a/packages/editor-ui/src/store.ts b/packages/editor-ui/src/store.ts index 4a6224037b..611c659f80 100644 --- a/packages/editor-ui/src/store.ts +++ b/packages/editor-ui/src/store.ts @@ -818,6 +818,19 @@ export const store = new Vuex.Store({ allNodeTypes: (state): INodeTypeDescription[] => { return state.nodeTypes; }, + + /** + * Getter for node display names ending with a number: `'S3'`, `'MSG91'`, etc. + */ + nativelyNumberSuffixedNodeNames: (_, getters): string[] => { + const allNodeTypes: INodeTypeDescription[] = getters.allNodeTypes; + + return allNodeTypes.reduce((acc, cur) => { + if (/\d$/.test(cur.displayName)) acc.push(cur.displayName); + return acc; + }, []); + }, + nodeType: (state, getters) => (nodeType: string, typeVersion?: number): INodeTypeDescription | null => { const foundType = state.nodeTypes.find(typeData => { return typeData.name === nodeType && typeData.version === (typeVersion || typeData.defaultVersion || DEFAULT_NODETYPE_VERSION);