refactor(editor): encapsulate node creation actions (#4287)

* refactor(editor): encapsulate node creation actions

* fix(editor): add sticky node event name

* refactor(editor): move node creation and load it dynamically

* refactor(editor): move node creator

* refactor(editor): move node creator from node view to node creation

* fix(editor): fix node creator opening
This commit is contained in:
Csaba Tuncsik 2022-10-11 10:06:33 +02:00 committed by GitHub
parent 69684fc4f7
commit 0c78df61ea
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
15 changed files with 176 additions and 152 deletions

View file

@ -0,0 +1,131 @@
<template>
<div>
<div v-if="!createNodeActive" :class="[$style.nodeButtonsWrapper, showStickyButton ? $style.noEvents : '']" @mouseenter="onCreateMenuHoverIn">
<div :class="$style.nodeCreatorButton">
<n8n-icon-button size="xlarge" icon="plus" @click="openNodeCreator" :title="$locale.baseText('nodeView.addNode')"/>
<div :class="[$style.addStickyButton, showStickyButton ? $style.visibleButton : '']" @click="addStickyNote">
<n8n-icon-button size="medium" type="secondary" :icon="['far', 'note-sticky']" :title="$locale.baseText('nodeView.addSticky')"/>
</div>
</div>
</div>
<node-creator :active="createNodeActive" @nodeTypeSelected="nodeTypeSelected" @closeNodeCreator="closeNodeCreator" />
</div>
</template>
<script lang="ts">
import Vue from "vue";
import * as CanvasHelpers from "@/views/canvasHelpers";
import {DEFAULT_STICKY_HEIGHT, DEFAULT_STICKY_WIDTH, STICKY_NODE_TYPE} from "@/constants";
export default Vue.extend({
name: 'node-creation',
components: {
NodeCreator: () => import('@/components/Node/NodeCreator/NodeCreator.vue'),
},
props: {
nodeViewScale: {
type: Number,
required: true,
},
createNodeActive: {
type: Boolean,
default: false,
},
},
data() {
return {
showStickyButton: false,
};
},
methods: {
onCreateMenuHoverIn(mouseinEvent: MouseEvent) {
const buttonsWrapper = mouseinEvent.target as Element;
// Once the popup menu is hovered, it's pointer events are disabled so it's not interfering with element underneath it.
this.showStickyButton = true;
const moveCallback = (mousemoveEvent: MouseEvent) => {
if (buttonsWrapper) {
const wrapperBounds = buttonsWrapper.getBoundingClientRect();
const wrapperH = wrapperBounds.height;
const wrapperW = wrapperBounds.width;
const wrapperLeftNear = wrapperBounds.left;
const wrapperLeftFar = wrapperLeftNear + wrapperW;
const wrapperTopNear = wrapperBounds.top;
const wrapperTopFar = wrapperTopNear + wrapperH;
const inside = ((mousemoveEvent.pageX > wrapperLeftNear && mousemoveEvent.pageX < wrapperLeftFar) && (mousemoveEvent.pageY > wrapperTopNear && mousemoveEvent.pageY < wrapperTopFar));
if (!inside) {
this.showStickyButton = false;
document.removeEventListener('mousemove', moveCallback, false);
}
}
};
document.addEventListener('mousemove', moveCallback, false);
},
openNodeCreator() {
this.$emit('toggleNodeCreator', { source: 'add_node_button', createNodeActive: true });
},
addStickyNote() {
if (document.activeElement) {
(document.activeElement as HTMLElement).blur();
}
const offset: [number, number] = [...(this.$store.getters.getNodeViewOffsetPosition as [number, number])];
const position = CanvasHelpers.getMidCanvasPosition(this.nodeViewScale, offset);
position[0] -= DEFAULT_STICKY_WIDTH / 2;
position[1] -= DEFAULT_STICKY_HEIGHT / 2;
this.$emit('addNode', {
nodeTypeName: STICKY_NODE_TYPE,
position,
});
},
closeNodeCreator() {
this.$emit('toggleNodeCreator', { createNodeActive: false });
},
nodeTypeSelected(nodeTypeName: string) {
this.$emit('addNode', { nodeTypeName });
this.closeNodeCreator();
},
},
});
</script>
<style lang="scss" module>
.nodeButtonsWrapper {
position: fixed;
width: 150px;
height: 200px;
top: 0;
right: 0;
display: flex;
}
.addStickyButton {
margin-top: var(--spacing-2xs);
opacity: 0;
transition: .1s;
transition-timing-function: linear;
}
.visibleButton {
opacity: 1;
pointer-events: all;
}
.noEvents {
pointer-events: none;
}
.nodeCreatorButton {
position: fixed;
text-align: center;
top: 80px;
right: 20px;
pointer-events: all !important;
button {
position: relative;
}
}
</style>

View file

@ -67,7 +67,7 @@ import SearchBar from './SearchBar.vue';
import SubcategoryPanel from './SubcategoryPanel.vue'; import SubcategoryPanel from './SubcategoryPanel.vue';
import { INodeCreateElement, INodeItemProps, ISubcategoryItemProps } from '@/Interface'; import { INodeCreateElement, INodeItemProps, ISubcategoryItemProps } from '@/Interface';
import { ALL_NODE_FILTER, CORE_NODES_CATEGORY, REGULAR_NODE_FILTER, TRIGGER_NODE_FILTER } from '@/constants'; import { ALL_NODE_FILTER, CORE_NODES_CATEGORY, REGULAR_NODE_FILTER, TRIGGER_NODE_FILTER } from '@/constants';
import SlideTransition from '../transitions/SlideTransition.vue'; import SlideTransition from '../../transitions/SlideTransition.vue';
import { matchesNodeType, matchesSelectType } from './helpers'; import { matchesNodeType, matchesSelectType } from './helpers';
export default mixins(externalHooks).extend({ export default mixins(externalHooks).extend({

View file

@ -1,23 +1,21 @@
<template> <template>
<div> <SlideTransition>
<SlideTransition> <div
<div v-if="active"
v-if="active" class="node-creator"
class="node-creator" ref="nodeCreator"
ref="nodeCreator" v-click-outside="onClickOutside"
v-click-outside="onClickOutside" @dragover="onDragOver"
@dragover="onDragOver" @drop="onDrop"
@drop="onDrop" >
> <MainPanel
<MainPanel @nodeTypeSelected="nodeTypeSelected"
@nodeTypeSelected="nodeTypeSelected" :categorizedItems="categorizedItems"
:categorizedItems="categorizedItems" :categoriesWithNodes="categoriesWithNodes"
:categoriesWithNodes="categoriesWithNodes" :searchItems="searchItems"
:searchItems="searchItems" />
/> </div>
</div> </SlideTransition>
</SlideTransition>
</div>
</template> </template>
<script lang="ts"> <script lang="ts">
@ -26,7 +24,7 @@ import Vue from 'vue';
import { ICategoriesWithNodes, INodeCreateElement } from '@/Interface'; import { ICategoriesWithNodes, INodeCreateElement } from '@/Interface';
import { INodeTypeDescription } from 'n8n-workflow'; import { INodeTypeDescription } from 'n8n-workflow';
import SlideTransition from '../transitions/SlideTransition.vue'; import SlideTransition from '../../transitions/SlideTransition.vue';
import MainPanel from './MainPanel.vue'; import MainPanel from './MainPanel.vue';
import { getCategoriesWithNodes, getCategorizedList } from './helpers'; import { getCategoriesWithNodes, getCategorizedList } from './helpers';
@ -38,9 +36,11 @@ export default Vue.extend({
MainPanel, MainPanel,
SlideTransition, SlideTransition,
}, },
props: [ props: {
'active', active: {
], type: Boolean,
},
},
computed: { computed: {
...mapGetters('users', ['personalizedNodeTypes']), ...mapGetters('users', ['personalizedNodeTypes']),
allLatestNodeTypes(): INodeTypeDescription[] { allLatestNodeTypes(): INodeTypeDescription[] {

View file

@ -57,11 +57,11 @@
import {getNewNodePosition, NODE_SIZE} from '@/views/canvasHelpers'; import {getNewNodePosition, NODE_SIZE} from '@/views/canvasHelpers';
import Vue from 'vue'; import Vue from 'vue';
import NodeIcon from '../NodeIcon.vue'; import NodeIcon from '../../NodeIcon.vue';
import TriggerIcon from '../TriggerIcon.vue'; import TriggerIcon from '../../TriggerIcon.vue';
import { COMMUNITY_NODES_INSTALLATION_DOCS_URL } from '../../constants'; import { COMMUNITY_NODES_INSTALLATION_DOCS_URL } from '@/constants';
import { isCommunityPackageName } from '../helpers'; import { isCommunityPackageName } from '../../helpers';
Vue.component('NodeIcon', NodeIcon); Vue.component('NodeIcon', NodeIcon);
Vue.component('TriggerIcon', TriggerIcon); Vue.component('TriggerIcon', TriggerIcon);

View file

@ -23,7 +23,6 @@
<script lang="ts"> <script lang="ts">
import camelcase from 'lodash.camelcase'; import camelcase from 'lodash.camelcase';
import { INodeCreateElement } from '@/Interface';
import Vue from 'vue'; import Vue from 'vue';
import ItemIterator from './ItemIterator.vue'; import ItemIterator from './ItemIterator.vue';

View file

@ -311,7 +311,7 @@ export const nodeBase = mixins(
if (this.$store.getters.isActionActive('dragActive')) { if (this.$store.getters.isActionActive('dragActive')) {
this.$store.commit('removeActiveAction', 'dragActive'); this.$store.commit('removeActiveAction', 'dragActive');
} else { } else {
if (this.isCtrlKeyPressed(e) === false) { if (!this.isCtrlKeyPressed(e)) {
this.$emit('deselectAllNodes'); this.$emit('deselectAllNodes');
} }

View file

@ -19,19 +19,7 @@
</div> </div>
</div> </div>
<NodeDetailsView :readOnly="isReadOnly" :renaming="renamingActive" @valueChanged="valueChanged" /> <NodeDetailsView :readOnly="isReadOnly" :renaming="renamingActive" @valueChanged="valueChanged" />
<div :class="['node-buttons-wrapper', showStickyButton ? 'no-events' : '']" v-if="!createNodeActive && !isReadOnly" <node-creation v-if="!isReadOnly" :create-node-active="createNodeActive" :node-view-scale="nodeViewScale" @toggleNodeCreator="onToggleNodeCreator" @addNode="onAddNode"/>
@mouseenter="onCreateMenuHoverIn">
<div class="node-creator-button">
<n8n-icon-button size="xlarge" icon="plus" @click="() => openNodeCreator('add_node_button')"
:title="$locale.baseText('nodeView.addNode')" />
<div :class="['add-sticky-button', showStickyButton ? 'visible-button' : '']" @click="addStickyNote">
<n8n-icon-button size="medium" type="secondary" :icon="['far', 'note-sticky']"
:title="$locale.baseText('nodeView.addSticky')" />
</div>
</div>
</div>
<node-creator :active="createNodeActive" @nodeTypeSelected="nodeTypeSelected"
@closeNodeCreator="closeNodeCreator" />
<div <div
:class="{ 'zoom-menu': true, 'regular-zoom-menu': !isDemo, 'demo-zoom-menu': isDemo, expanded: !sidebarMenuCollapsed }"> :class="{ 'zoom-menu': true, 'regular-zoom-menu': !isDemo, 'demo-zoom-menu': isDemo, expanded: !sidebarMenuCollapsed }">
<n8n-icon-button @click="zoomToFit" type="tertiary" size="large" :title="$locale.baseText('nodeView.zoomToFit')" <n8n-icon-button @click="zoomToFit" type="tertiary" size="large" :title="$locale.baseText('nodeView.zoomToFit')"
@ -73,8 +61,6 @@ import {
import type { MessageBoxInputData } from 'element-ui/types/message-box'; import type { MessageBoxInputData } from 'element-ui/types/message-box';
import { jsPlumb, OnConnectionBindInfo } from 'jsplumb'; import { jsPlumb, OnConnectionBindInfo } from 'jsplumb';
import { import {
DEFAULT_STICKY_HEIGHT,
DEFAULT_STICKY_WIDTH,
FIRST_ONBOARDING_PROMPT_TIMEOUT, FIRST_ONBOARDING_PROMPT_TIMEOUT,
MODAL_CANCEL, MODAL_CANCEL,
MODAL_CLOSE, MODAL_CLOSE,
@ -105,7 +91,6 @@ import { workflowRun } from '@/components/mixins/workflowRun';
import NodeDetailsView from '@/components/NodeDetailsView.vue'; import NodeDetailsView from '@/components/NodeDetailsView.vue';
import Node from '@/components/Node.vue'; import Node from '@/components/Node.vue';
import NodeCreator from '@/components/NodeCreator/NodeCreator.vue';
import NodeSettings from '@/components/NodeSettings.vue'; import NodeSettings from '@/components/NodeSettings.vue';
import Sticky from '@/components/Sticky.vue'; import Sticky from '@/components/Sticky.vue';
@ -146,17 +131,11 @@ import {
IWorkflowTemplate, IWorkflowTemplate,
IExecutionsSummary, IExecutionsSummary,
IWorkflowToShare, IWorkflowToShare,
} from '../Interface'; } from '@/Interface';
import { mapGetters } from 'vuex'; import { mapGetters } from 'vuex';
import {
addNodeTranslation,
} from '@/plugins/i18n';
import '../plugins/N8nCustomConnectorType'; import '../plugins/N8nCustomConnectorType';
import '../plugins/PlusEndpointType'; import '../plugins/PlusEndpointType';
import { getAccountAge } from '@/modules/userHelpers'; import { getAccountAge } from '@/modules/userHelpers';
import { IUser } from 'n8n-design-system';
import { dataPinningEventBus } from "@/event-bus/data-pinning-event-bus"; import { dataPinningEventBus } from "@/event-bus/data-pinning-event-bus";
import { debounceHelper } from '@/components/mixins/debounce'; import { debounceHelper } from '@/components/mixins/debounce';
@ -185,9 +164,10 @@ export default mixins(
components: { components: {
NodeDetailsView, NodeDetailsView,
Node, Node,
NodeCreator, NodeCreator: () => import('@/components/Node/NodeCreator/NodeCreator.vue'),
NodeSettings, NodeSettings,
Sticky, Sticky,
NodeCreation: () => import('@/components/Node/NodeCreation.vue'),
}, },
errorCaptured: (err, vm, info) => { errorCaptured: (err, vm, info) => {
console.error('errorCaptured'); // eslint-disable-line no-console console.error('errorCaptured'); // eslint-disable-line no-console
@ -377,29 +357,6 @@ export default mixins(
this.runWorkflow(); this.runWorkflow();
}, },
onCreateMenuHoverIn(mouseinEvent: MouseEvent) {
const buttonsWrapper = mouseinEvent.target as Element;
// Once the popup menu is hovered, it's pointer events are disabled so it's not interfering with element underneath it.
this.showStickyButton = true;
const moveCallback = (mousemoveEvent: MouseEvent) => {
if (buttonsWrapper) {
const wrapperBounds = buttonsWrapper.getBoundingClientRect();
const wrapperH = wrapperBounds.height;
const wrapperW = wrapperBounds.width;
const wrapperLeftNear = wrapperBounds.left;
const wrapperLeftFar = wrapperLeftNear + wrapperW;
const wrapperTopNear = wrapperBounds.top;
const wrapperTopFar = wrapperTopNear + wrapperH;
const inside = ((mousemoveEvent.pageX > wrapperLeftNear && mousemoveEvent.pageX < wrapperLeftFar) && (mousemoveEvent.pageY > wrapperTopNear && mousemoveEvent.pageY < wrapperTopFar));
if (!inside) {
this.showStickyButton = false;
document.removeEventListener('mousemove', moveCallback, false);
}
}
};
document.addEventListener('mousemove', moveCallback, false);
},
clearExecutionData() { clearExecutionData() {
this.$store.commit('setWorkflowExecutionData', null); this.$store.commit('setWorkflowExecutionData', null);
this.updateNodesExecutionIssues(); this.updateNodesExecutionIssues();
@ -478,11 +435,6 @@ export default mixins(
const saved = await this.saveCurrentWorkflow(); const saved = await this.saveCurrentWorkflow();
if (saved) this.$store.dispatch('settings/fetchPromptsData'); if (saved) this.$store.dispatch('settings/fetchPromptsData');
}, },
openNodeCreator(source: string) {
this.createNodeActive = true;
this.$externalHooks().run('nodeView.createNodeActiveChanged', { source, createNodeActive: this.createNodeActive });
this.$telemetry.trackNodesPanel('nodeView.createNodeActiveChanged', { source, workflow_id: this.$store.getters.workflowId, createNodeActive: this.createNodeActive });
},
async openExecution(executionId: string) { async openExecution(executionId: string) {
this.resetWorkspace(); this.resetWorkspace();
@ -743,10 +695,7 @@ export default mixins(
this.callDebounced('deleteSelectedNodes', { debounceTime: 500 }); this.callDebounced('deleteSelectedNodes', { debounceTime: 500 });
} else if (e.key === 'Tab') { } else if (e.key === 'Tab') {
this.createNodeActive = !this.createNodeActive && !this.isReadOnly; this.onToggleNodeCreator({ source: 'tab', createNodeActive: !this.createNodeActive && !this.isReadOnly });
this.$externalHooks().run('nodeView.createNodeActiveChanged', { source: 'tab', createNodeActive: this.createNodeActive });
this.$telemetry.trackNodesPanel('nodeView.createNodeActiveChanged', { source: 'tab', workflow_id: this.$store.getters.workflowId, createNodeActive: this.createNodeActive });
} else if (e.key === this.controlKeyCode) { } else if (e.key === this.controlKeyCode) {
this.ctrlKeyPressed = true; this.ctrlKeyPressed = true;
} else if (e.key === 'F2' && !this.isReadOnly) { } else if (e.key === 'F2' && !this.isReadOnly) {
@ -1351,32 +1300,6 @@ export default mixins(
); );
} }
}, },
closeNodeCreator() {
this.createNodeActive = false;
},
addStickyNote() {
if (document.activeElement) {
(document.activeElement as HTMLElement).blur();
}
const offset: [number, number] = [...(this.$store.getters.getNodeViewOffsetPosition as [number, number])];
const position = CanvasHelpers.getMidCanvasPosition(this.nodeViewScale, offset);
position[0] -= DEFAULT_STICKY_WIDTH / 2;
position[1] -= DEFAULT_STICKY_HEIGHT / 2;
this.addNodeButton(STICKY_NODE_TYPE, {
position,
});
},
nodeTypeSelected(nodeTypeName: string) {
this.addNodeButton(nodeTypeName);
this.createNodeActive = false;
},
onDragOver(event: DragEvent) { onDragOver(event: DragEvent) {
event.preventDefault(); event.preventDefault();
}, },
@ -1391,7 +1314,7 @@ export default mixins(
const mousePosition = this.getMousePositionWithinNodeView(event); const mousePosition = this.getMousePositionWithinNodeView(event);
const sidebarOffset = this.sidebarMenuCollapsed ? CanvasHelpers.SIDEBAR_WIDTH : CanvasHelpers.SIDEBAR_WIDTH_EXPANDED; const sidebarOffset = this.sidebarMenuCollapsed ? CanvasHelpers.SIDEBAR_WIDTH : CanvasHelpers.SIDEBAR_WIDTH_EXPANDED;
this.addNodeButton(nodeTypeName, { this.addNode(nodeTypeName, {
position: [ position: [
mousePosition[0] - CanvasHelpers.NODE_SIZE / 2, mousePosition[0] - CanvasHelpers.NODE_SIZE / 2,
mousePosition[1] - CanvasHelpers.NODE_SIZE / 2, mousePosition[1] - CanvasHelpers.NODE_SIZE / 2,
@ -1593,7 +1516,7 @@ export default mixins(
this.__addConnection(connectionData, true); this.__addConnection(connectionData, true);
}, },
async addNodeButton(nodeTypeName: string, options: AddNodeOptions = {}) { async addNode(nodeTypeName: string, options: AddNodeOptions = {}) {
if (this.editAllowedCheck() === false) { if (this.editAllowedCheck() === false) {
return; return;
} }
@ -1653,7 +1576,7 @@ export default mixins(
this.lastSelectedConnection = info.connection; this.lastSelectedConnection = info.connection;
} }
this.openNodeCreator(info.eventSource); this.onToggleNodeCreator({ source: info.eventSource, createNodeActive: true});
}; };
this.instance.bind('connectionAborted', (connection) => { this.instance.bind('connectionAborted', (connection) => {
@ -3029,6 +2952,14 @@ export default mixins(
connections.forEach(CanvasHelpers.resetConnection); connections.forEach(CanvasHelpers.resetConnection);
}); });
}, },
onToggleNodeCreator({ source, createNodeActive }: { source?: string; createNodeActive: boolean }) {
this.createNodeActive = createNodeActive;
this.$externalHooks().run('nodeView.createNodeActiveChanged', { source, createNodeActive });
this.$telemetry.trackNodesPanel('nodeView.createNodeActiveChanged', { source, createNodeActive, workflow_id: this.$store.getters.workflowId });
},
onAddNode({ nodeTypeName, position }: { nodeTypeName: string; position?: [number, number] }) {
this.addNode(nodeTypeName, { position });
},
}, },
async mounted() { async mounted() {
@ -3175,43 +3106,6 @@ export default mixins(
bottom: 10px; bottom: 10px;
} }
.no-events {
pointer-events: none;
}
.node-buttons-wrapper {
position: fixed;
width: 150px;
height: 200px;
top: 0;
right: 0;
display: flex;
.add-sticky-button {
margin-top: var(--spacing-2xs);
opacity: 0;
transition: .1s;
transition-timing-function: linear;
}
.visible-button {
opacity: 1;
pointer-events: all;
}
}
.node-creator-button {
position: fixed;
text-align: center;
top: 80px;
right: 20px;
pointer-events: all !important;
}
.node-creator-button button {
position: relative;
}
.node-view-root { .node-view-root {
overflow: hidden; overflow: hidden;
background-color: var(--color-canvas-background); background-color: var(--color-canvas-background);