mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-13 22:07:35 -08:00
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:
parent
69684fc4f7
commit
0c78df61ea
131
packages/editor-ui/src/components/Node/NodeCreation.vue
Normal file
131
packages/editor-ui/src/components/Node/NodeCreation.vue
Normal 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>
|
|
@ -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({
|
|
@ -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[] {
|
|
@ -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);
|
|
@ -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';
|
|
@ -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');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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);
|
||||||
|
|
Loading…
Reference in a new issue