change flow state on drop

This commit is contained in:
Mutasem 2021-10-20 15:27:12 +02:00
parent d8178b96f1
commit 8d703a4b6c
3 changed files with 100 additions and 54 deletions

View file

@ -21,11 +21,15 @@ import {
WorkflowExecuteMode, WorkflowExecuteMode,
} from 'n8n-workflow'; } from 'n8n-workflow';
import {
PaintStyle,
} from 'jsplumb';
declare module 'jsplumb' { declare module 'jsplumb' {
interface PaintStyle {
stroke?: string;
fill?: string;
strokeWidth?: number;
outlineStroke?: string;
outlineWidth?: number;
}
interface Anchor { interface Anchor {
lastReturnValue: number[]; lastReturnValue: number[];
} }

View file

@ -109,6 +109,8 @@ import Vue from 'vue';
import { import {
Connection, Connection,
Overlay, Overlay,
OverlaySpec,
PaintStyle,
} from 'jsplumb'; } from 'jsplumb';
import { MessageBoxInputData } from 'element-ui/types/message-box'; import { MessageBoxInputData } from 'element-ui/types/message-box';
import { jsPlumb, Endpoint, OnConnectionBindInfo } from 'jsplumb'; import { jsPlumb, Endpoint, OnConnectionBindInfo } from 'jsplumb';
@ -133,7 +135,7 @@ import NodeCreator from '@/components/NodeCreator/NodeCreator.vue';
import NodeSettings from '@/components/NodeSettings.vue'; import NodeSettings from '@/components/NodeSettings.vue';
import RunData from '@/components/RunData.vue'; import RunData from '@/components/RunData.vue';
import { getLeftmostTopNode, getWorkflowCorners, scaleSmaller, scaleBigger, scaleReset, showOrHideMidpointArrow, addEndpointArrow, getDefaultOverlays, getIcon, getNewNodePosition, hideMidpointArrow, showOrHideItemsLabel } from './helpers'; import { getLeftmostTopNode, getWorkflowCorners, scaleSmaller, scaleBigger, scaleReset, showOrHideMidpointArrow, addEndpointArrow, getIcon, getNewNodePosition, hideMidpointArrow, showOrHideItemsLabel } from './helpers';
import mixins from 'vue-typed-mixins'; import mixins from 'vue-typed-mixins';
import { v4 as uuidv4} from 'uuid'; import { v4 as uuidv4} from 'uuid';
@ -218,6 +220,72 @@ if (!window.localStorage.getItem('PUSH_NODES_LENGTH')) {
// @ts-ignore // @ts-ignore
const _PUSH_NODES_LENGTH = parseInt(window.localStorage.getItem('PUSH_NODES_LENGTH'), 10); const _PUSH_NODES_LENGTH = parseInt(window.localStorage.getItem('PUSH_NODES_LENGTH'), 10);
const CONNECTOR_PAINT_STYLE_DEFAULT: PaintStyle = {
stroke: getStyleTokenValue('--color-foreground-dark'),
strokeWidth: 2,
outlineWidth: 12,
outlineStroke: 'transparent',
};
const CONNECTOR_PAINT_STYLE_PRIMARY = {
...CONNECTOR_PAINT_STYLE_DEFAULT,
stroke: getStyleTokenValue('--color-primary'),
};
const CONNECTOR_PAINT_STYLE_SUCCESS = {
...CONNECTOR_PAINT_STYLE_DEFAULT,
stroke: getStyleTokenValue('--color-success'),
};
const CONNECTOR_TYPE_BEZIER = ['Bezier', { curviness: _CURVINESS }];
const CONNECTOR_TYPE_FLOWCHART = ['Flowchart', { cornerRadius: 8, stub: JSPLUMB_FLOWCHART_STUB, gap: 5, alwaysRespectStubs: _ALWAYS_RESPECT_STUB}];
const OVERLAY_DROP_NODE_ID = 'drop-add-node';
const CONNECTOR_ARROW_OVERLAYS: OverlaySpec[] = [
[
'Arrow',
{
id: 'endpoint-arrow',
location: 1,
width: 12,
foldback: 1,
length: 10,
visible: true,
},
],
[
'Arrow',
{
id: 'midpoint-arrow',
location: 0.5,
width: 12,
foldback: 1,
length: 10,
visible: false,
},
],
];
const CONNECTOR_DROP_NODE_OVERLAY: OverlaySpec[] = [
[
'Label',
{
id: OVERLAY_DROP_NODE_ID,
label: 'Drop connection<br />to create node',
cssClass: 'drop-add-node-label',
location: 0.5,
visible: true,
},
],
];
const addOverlays = (connection: Connection, overlays: OverlaySpec[]) => {
overlays.forEach((overlay: OverlaySpec) => {
connection.addOverlay(overlay);
});
};
export default mixins( export default mixins(
copyPaste, copyPaste,
externalHooks, externalHooks,
@ -1326,14 +1394,14 @@ export default mixins(
}, },
initNodeView () { initNodeView () {
this.instance.importDefaults({ this.instance.importDefaults({
Connector: ['Bezier', { curviness: _CURVINESS }], Connector: CONNECTOR_TYPE_BEZIER,
Endpoint: ['Dot', { radius: 5 }], Endpoint: ['Dot', { radius: 5 }],
DragOptions: { cursor: 'pointer', zIndex: 5000 }, DragOptions: { cursor: 'pointer', zIndex: 5000 },
PaintStyle: { strokeWidth: 2, stroke: getStyleTokenValue('--color-foreground-dark'), outlineStroke: _OUTLINE_STROKE_COLOR, outlineWidth: _OUTLINE_STROKE_WIDTH}, PaintStyle: { strokeWidth: 2, stroke: getStyleTokenValue('--color-foreground-dark')},
EndpointStyle: { radius: 9, fill: '#acd', stroke: 'red' }, EndpointStyle: { radius: 9, fill: '#acd', stroke: 'red' },
HoverPaintStyle: { stroke: '#ff6d5a', lineWidth: 4 }, HoverPaintStyle: { stroke: '#ff6d5a', lineWidth: 4 },
EndpointHoverStyle: { fill: '#ff6d5a', stroke: '#acd' }, EndpointHoverStyle: { fill: '#ff6d5a', stroke: '#acd' },
ConnectionOverlays: getDefaultOverlays(), ConnectionOverlays: CONNECTOR_ARROW_OVERLAYS,
Container: '#node-view', Container: '#node-view',
}); });
@ -1357,9 +1425,9 @@ export default mixins(
})); }));
this.instance.bind('connection', (info: OnConnectionBindInfo) => { this.instance.bind('connection', (info: OnConnectionBindInfo) => {
info.connection.setConnector(['Flowchart', { cornerRadius: 8, stub: JSPLUMB_FLOWCHART_STUB, gap: 5, alwaysRespectStubs: _ALWAYS_RESPECT_STUB}]); info.connection.setConnector(CONNECTOR_TYPE_FLOWCHART);
// @ts-ignore info.connection.setPaintStyle(CONNECTOR_PAINT_STYLE_DEFAULT);
info.connection.setPaintStyle({stroke: getStyleTokenValue('--color-foreground-dark'), strokeWidth: 2, outlineStroke: _OUTLINE_STROKE_COLOR, outlineWidth: _OUTLINE_STROKE_WIDTH}); addOverlays(info.connection, CONNECTOR_ARROW_OVERLAYS);
addEndpointArrow(info.connection); addEndpointArrow(info.connection);
showOrHideMidpointArrow(info.connection); showOrHideMidpointArrow(info.connection);
@ -1375,8 +1443,7 @@ export default mixins(
const sourceNode = this.$store.getters.getNodeByName(sourceNodeName); const sourceNode = this.$store.getters.getNodeByName(sourceNodeName);
const targetNode = this.$store.getters.getNodeByName(targetNodeName); const targetNode = this.$store.getters.getNodeByName(targetNodeName);
// @ts-ignore info.connection.removeOverlay(OVERLAY_DROP_NODE_ID);
info.connection.removeOverlay('drop-add-node');
if (this.isReadOnly === false) { if (this.isReadOnly === false) {
// Display the connection-delete button only on hover // Display the connection-delete button only on hover
@ -1579,17 +1646,26 @@ export default mixins(
// @ts-ignore // @ts-ignore
this.instance.bind('connectionDrag', (connection: Connection) => { this.instance.bind('connectionDrag', (connection: Connection) => {
addOverlays(connection, CONNECTOR_DROP_NODE_OVERLAY);
let droppable = false;
const onMouseMove = () => { const onMouseMove = () => {
if (!connection) { if (!connection) {
return; return;
} }
const elements = document.querySelector('div.jtk-endpoint.dropHover'); const elements = document.querySelector('div.jtk-endpoint.dropHover');
if (elements) { if (elements && !droppable) {
connection.setPaintStyle({stroke: getStyleTokenValue('--color-primary')}); droppable = true;
connection.setConnector(CONNECTOR_TYPE_FLOWCHART);
connection.setPaintStyle(CONNECTOR_PAINT_STYLE_PRIMARY);
addOverlays(connection, CONNECTOR_ARROW_OVERLAYS);
} }
else { else if (!elements && droppable) {
connection.setPaintStyle({stroke: getStyleTokenValue('--color-foreground-dark')}); droppable = false;
connection.setConnector(CONNECTOR_TYPE_BEZIER);
connection.setPaintStyle(CONNECTOR_PAINT_STYLE_DEFAULT);
addOverlays(connection, CONNECTOR_ARROW_OVERLAYS);
} }
}; };
@ -1804,7 +1880,7 @@ export default mixins(
outgoing.forEach((connection: Connection) => { outgoing.forEach((connection: Connection) => {
connection.removeOverlay('output-items-label'); connection.removeOverlay('output-items-label');
connection.setPaintStyle({stroke: getStyleTokenValue('--color-foreground-dark')}); connection.setPaintStyle(CONNECTOR_PAINT_STYLE_DEFAULT);
showOrHideMidpointArrow(connection); showOrHideMidpointArrow(connection);
}); });
@ -1874,12 +1950,12 @@ export default mixins(
const output = outputMap[sourceEndpoint][targetId][targetEndpoint]; const output = outputMap[sourceEndpoint][targetId][targetEndpoint];
if (!output || !output.total) { if (!output || !output.total) {
conn.setPaintStyle({stroke: getStyleTokenValue('--color-foreground-dark')}); conn.setPaintStyle(CONNECTOR_PAINT_STYLE_DEFAULT);
conn.removeOverlay('output-items-label'); conn.removeOverlay('output-items-label');
return; return;
} }
conn.setPaintStyle({stroke: getStyleTokenValue('--color-success')}); conn.setPaintStyle(CONNECTOR_PAINT_STYLE_SUCCESS);
if (conn.getOverlay('output-items-label')) { if (conn.getOverlay('output-items-label')) {
conn.removeOverlay('output-items-label'); conn.removeOverlay('output-items-label');

View file

@ -85,40 +85,6 @@ export const scaleReset = (config: IZoomConfig): IZoomConfig => {
return config; return config;
}; };
export const getDefaultOverlays = (): OverlaySpec[] => ([
[
'Arrow',
{
id: 'endpoint-arrow',
location: 1,
width: 12,
foldback: 1,
length: 10,
visible: true,
},
],
[
'Label',
{
id: 'drop-add-node',
label: 'Drop connection<br />to create node',
cssClass: 'drop-add-node-label',
location: 0.5,
},
],
[
'Arrow',
{
id: 'midpoint-arrow',
location: 0.5,
width: 12,
foldback: 1,
length: 10,
visible: false,
},
],
]);
export const addEndpointArrow = (connection: Connection) => { export const addEndpointArrow = (connection: Connection) => {
connection.addOverlay([ connection.addOverlay([
'Arrow', 'Arrow',