support item labels and success path

This commit is contained in:
Mutasem 2021-10-18 14:53:56 +02:00
parent 0eae86874b
commit 571140deee
3 changed files with 188 additions and 52 deletions

View file

@ -61,6 +61,7 @@ import { workflowHelpers } from '@/components/mixins/workflowHelpers';
import { import {
INodeTypeDescription, INodeTypeDescription,
ITaskData,
NodeHelpers, NodeHelpers,
} from 'n8n-workflow'; } from 'n8n-workflow';
@ -76,8 +77,11 @@ export default mixins(externalHooks, nodeBase, nodeHelpers, workflowHelpers).ext
NodeIcon, NodeIcon,
}, },
computed: { computed: {
workflowDataItems () { nodeRunData(): ITaskData[] {
const workflowResultDataNode = this.$store.getters.getWorkflowResultDataByNodeName(this.data.name); return this.$store.getters.getWorkflowResultDataByNodeName(this.data.name);
},
workflowDataItems (): Number {
const workflowResultDataNode = this.nodeRunData;
if (workflowResultDataNode === null) { if (workflowResultDataNode === null) {
return 0; return 0;
} }
@ -161,9 +165,15 @@ export default mixins(externalHooks, nodeBase, nodeHelpers, workflowHelpers).ext
this.setSubtitle(); this.setSubtitle();
} }
}, },
nodeRunData(newValue) {
this.$emit('run', {name: this.data.name, data: newValue});
},
}, },
mounted() { mounted() {
this.setSubtitle(); this.setSubtitle();
setTimeout(() => {
this.$emit('run', {name: this.data.name, data: this.nodeRunData});
}, 0);
}, },
data () { data () {
return { return {
@ -402,6 +412,10 @@ export default mixins(externalHooks, nodeBase, nodeHelpers, workflowHelpers).ext
z-index: 5; z-index: 5;
} }
.jtk-connector.jtk-success {
z-index: 5;
}
.jtk-endpoint { .jtk-endpoint {
z-index:5; z-index:5;
} }

View file

@ -22,6 +22,7 @@
@removeNode="removeNode" @removeNode="removeNode"
@runWorkflow="runWorkflow" @runWorkflow="runWorkflow"
@moved="onNodeMoved" @moved="onNodeMoved"
@run="onNodeRun"
:id="'node-' + getNodeIndex(nodeData.name)" :id="'node-' + getNodeIndex(nodeData.name)"
:key="getNodeIndex(nodeData.name)" :key="getNodeIndex(nodeData.name)"
:name="nodeData.name" :name="nodeData.name"
@ -132,7 +133,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, addOrRemoveMidpointArrow, addEndpointArrow, getDefaultOverlays, getIcon, getNewNodePosition } from './helpers'; import { getLeftmostTopNode, getWorkflowCorners, scaleSmaller, scaleBigger, scaleReset, showOrHideMidpointArrow, addEndpointArrow, getDefaultOverlays, getIcon, getNewNodePosition, hideMidpointArrow } 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';
@ -145,14 +146,14 @@ import {
INodeIssues, INodeIssues,
INodeTypeDescription, INodeTypeDescription,
INodeTypeNameVersion, INodeTypeNameVersion,
NodeInputConnections,
NodeHelpers, NodeHelpers,
Workflow, Workflow,
IRun, IRun,
ITaskData,
INodeCredentialsDetails, INodeCredentialsDetails,
INodeExecutionData,
} from 'n8n-workflow'; } from 'n8n-workflow';
import { import {
IConnectionsUi,
ICredentialsResponse, ICredentialsResponse,
IExecutionResponse, IExecutionResponse,
IN8nUISettings, IN8nUISettings,
@ -1329,7 +1330,7 @@ export default mixins(
info.connection.setConnector(['Flowchart', { cornerRadius: 8, stub: JSPLUMB_FLOWCHART_STUB, gap: 5, alwaysRespectStubs: false}]); info.connection.setConnector(['Flowchart', { cornerRadius: 8, stub: JSPLUMB_FLOWCHART_STUB, gap: 5, alwaysRespectStubs: false}]);
addEndpointArrow(info.connection); addEndpointArrow(info.connection);
addOrRemoveMidpointArrow(info.connection); showOrHideMidpointArrow(info.connection);
// @ts-ignore // @ts-ignore
const sourceInfo = info.sourceEndpoint.getParameters(); const sourceInfo = info.sourceEndpoint.getParameters();
@ -1355,9 +1356,11 @@ export default mixins(
const overlay = info.connection.getOverlay('connection-actions'); const overlay = info.connection.getOverlay('connection-actions');
overlay.setVisible(true); overlay.setVisible(true);
const arrow = info.connection.getOverlay('midpoint-arrow'); hideMidpointArrow(info.connection);
if (arrow) {
arrow.setVisible(false); const itemsOverlay = info.connection.getOverlay('output-items-label');
if (itemsOverlay) {
itemsOverlay.setVisible(false);
} }
}); });
info.connection.bind('mouseout', (connection: IConnection) => { info.connection.bind('mouseout', (connection: IConnection) => {
@ -1366,9 +1369,11 @@ export default mixins(
overlay.setVisible(false); overlay.setVisible(false);
timer = undefined; timer = undefined;
const arrow = info.connection.getOverlay('midpoint-arrow'); showOrHideMidpointArrow(info.connection);
if (arrow) {
arrow.setVisible(true); const itemsOverlay = info.connection.getOverlay('output-items-label');
if (itemsOverlay) {
itemsOverlay.setVisible(true);
} }
}, 500); }, 500);
}); });
@ -1722,7 +1727,128 @@ export default mixins(
}) as Connection[]; }) as Connection[];
[...incoming, ...outgoing].forEach((connection: Connection) => { [...incoming, ...outgoing].forEach((connection: Connection) => {
addOrRemoveMidpointArrow(connection); showOrHideMidpointArrow(connection);
});
},
onNodeRun ({name, data}: {name: string, data: ITaskData[] | null}) {
const sourceIndex = this.$store.getters.getNodeIndex(name);
const sourceId = `${NODE_NAME_PREFIX}${sourceIndex}`;
if (data === null || data.length === 0) {
// @ts-ignore
const outgoing = this.instance.getConnections({
source: sourceId,
}) as Connection[];
outgoing.forEach((connection: Connection) => {
const arrow = connection.getOverlay('midpoint-arrow');
if (arrow) {
// @ts-ignore
arrow.setLocation(0.5);
}
connection.removeOverlay('output-items-label');
connection.setPaintStyle({stroke: getStyleTokenValue('--color-foreground-dark')});
});
return;
}
const nodeConnections = (this.$store.getters.outgoingConnectionsByNodeName(name) as INodeConnections).main;
if (!nodeConnections) {
return;
}
const outputMap: {[sourceEndpoint: string]: {[targetId: string]: {[targetEndpoint: string]: {total: number, iterations: number}}}} = {};
data.forEach((run: ITaskData) => {
if (!run.data) {
return;
}
run.data.main.forEach((output: INodeExecutionData[] | null, i: number) => {
nodeConnections[i]
.map((conn: IConnection) => {
const targetIndex = this.getNodeIndex(conn.node);
const targetId = `${NODE_NAME_PREFIX}${targetIndex}`;
const sourceEndpoint = `${sourceIndex}-output${i}`;
const targetEndpoint = `${targetIndex}-input${conn.index}`;
if (!outputMap[sourceEndpoint]) {
outputMap[sourceEndpoint] = {};
}
if (!outputMap[sourceEndpoint][targetId]) {
outputMap[sourceEndpoint][targetId] = {};
}
if (!outputMap[sourceEndpoint][targetId][targetEndpoint]) {
outputMap[sourceEndpoint][targetId][targetEndpoint] = {
total: 0,
iterations: 0,
};
}
outputMap[sourceEndpoint][targetId][targetEndpoint].total += output ? output.length : 0;
outputMap[sourceEndpoint][targetId][targetEndpoint].iterations += output ? 1 : 0;
});
});
});
Object.keys(outputMap).forEach((sourceEndpoint: string) => {
Object.keys(outputMap[sourceEndpoint]).forEach((targetId: string) => {
Object.keys(outputMap[sourceEndpoint][targetId]).forEach((targetEndpoint: string) => {
// @ts-ignore
const connections = this.instance.getConnections({
source: sourceId,
target: targetId,
}) as Connection[];
const conn = connections.find((connection: Connection) => {
// @ts-ignore
const uuids = connection.getUuids();
return uuids[0] === sourceEndpoint && uuids[1] === targetEndpoint;
});
if (!conn) {
return;
}
const output = outputMap[sourceEndpoint][targetId][targetEndpoint];
if (!output || !output.total) {
conn.setPaintStyle({stroke: getStyleTokenValue('--color-foreground-dark')});
conn.removeOverlay('output-items-label');
return;
}
conn.setPaintStyle({stroke: getStyleTokenValue('--color-success')});
if (conn.getOverlay('output-items-label')) {
conn.removeOverlay('output-items-label');
}
let label = `${output.total}`;
label = output.total > 1 ? `${label} items` : `${label} item`;
label = output.iterations > 1 ? `${label} total` : label;
conn.addOverlay([
'Label',
{
id: 'output-items-label',
label,
cssClass: 'connection-output-name-label',
location: .5,
},
]);
const arrow = connections[0].getOverlay('midpoint-arrow');
if (arrow) {
// @ts-ignore
arrow.setLocation(0.6);
}
});
});
}); });
}, },
removeNode (nodeName: string) { removeNode (nodeName: string) {

View file

@ -1,4 +1,3 @@
import { JSPLUMB_FLOWCHART_STUB } from "@/constants";
import { INodeUi, IZoomConfig, XYPositon } from "@/Interface"; import { INodeUi, IZoomConfig, XYPositon } from "@/Interface";
import { Connection, OverlaySpec } from "jsplumb"; import { Connection, OverlaySpec } from "jsplumb";
@ -95,6 +94,7 @@ export const getDefaultOverlays = (): OverlaySpec[] => ([
width: 12, width: 12,
foldback: 1, foldback: 1,
length: 10, length: 10,
visible: true,
}, },
], ],
[ [
@ -104,46 +104,10 @@ export const getDefaultOverlays = (): OverlaySpec[] => ([
label: 'Drop connection<br />to create node', label: 'Drop connection<br />to create node',
cssClass: 'drop-add-node-label', cssClass: 'drop-add-node-label',
location: 0.5, location: 0.5,
visible: false,
}, },
], ],
]); [
export const addEndpointArrow = (connection: Connection) => {
const hasArrow = !!connection.getOverlay('midpoint-arrow');
if (!hasArrow) {
connection.addOverlay([
'Arrow',
{
id: 'endpoint-arrow',
location: 1,
width: 12,
foldback: 1,
length: 10,
},
]);
}
};
export const addOrRemoveMidpointArrow = (connection: Connection) => {
const sourceEndpoint = connection.endpoints[0];
const targetEndpoint = connection.endpoints[1];
const requiresArrow = sourceEndpoint.anchor.lastReturnValue[0] >= targetEndpoint.anchor.lastReturnValue[0];
const hasArrow = !!connection.getOverlay('midpoint-arrow');
if (!requiresArrow) {
if (hasArrow) {
connection.removeOverlay('midpoint-arrow');
}
return;
}
if (hasArrow) {
return;
}
connection.addOverlay([
'Arrow', 'Arrow',
{ {
id: 'midpoint-arrow', id: 'midpoint-arrow',
@ -151,10 +115,42 @@ export const addOrRemoveMidpointArrow = (connection: Connection) => {
width: 12, width: 12,
foldback: 1, foldback: 1,
length: 10, length: 10,
visible: false,
},
],
]);
export const addEndpointArrow = (connection: Connection) => {
connection.addOverlay([
'Arrow',
{
id: 'endpoint-arrow',
location: 1,
width: 12,
foldback: 1,
length: 10,
}, },
]); ]);
}; };
export const hideMidpointArrow = (connection: Connection) => {
const arrow = connection.getOverlay('midpoint-arrow');
if (arrow) {
arrow.setVisible(false);
}
};
export const showOrHideMidpointArrow = (connection: Connection) => {
const sourceEndpoint = connection.endpoints[0];
const targetEndpoint = connection.endpoints[1];
const requiresArrow = sourceEndpoint.anchor.lastReturnValue[0] >= targetEndpoint.anchor.lastReturnValue[0];
const arrow = connection.getOverlay('midpoint-arrow');
if (arrow) {
arrow.setVisible(requiresArrow);
}
};
export const getIcon = (name: string): string => { export const getIcon = (name: string): string => {
if (name === 'trash') { if (name === 'trash') {
return `<svg data-v-66d5c7e2="" aria-hidden="true" focusable="false" data-prefix="fas" data-icon="trash" role="img" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 448 512" class="svg-inline--fa fa-trash fa-w-14 Icon__medium_ctPPJ"><path data-v-66d5c7e2="" fill="currentColor" d="M432 32H312l-9.4-18.7A24 24 0 0 0 281.1 0H166.8a23.72 23.72 0 0 0-21.4 13.3L136 32H16A16 16 0 0 0 0 48v32a16 16 0 0 0 16 16h416a16 16 0 0 0 16-16V48a16 16 0 0 0-16-16zM53.2 467a48 48 0 0 0 47.9 45h245.8a48 48 0 0 0 47.9-45L416 128H32z" class=""></path></svg>`; return `<svg data-v-66d5c7e2="" aria-hidden="true" focusable="false" data-prefix="fas" data-icon="trash" role="img" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 448 512" class="svg-inline--fa fa-trash fa-w-14 Icon__medium_ctPPJ"><path data-v-66d5c7e2="" fill="currentColor" d="M432 32H312l-9.4-18.7A24 24 0 0 0 281.1 0H166.8a23.72 23.72 0 0 0-21.4 13.3L136 32H16A16 16 0 0 0 0 48v32a16 16 0 0 0 16 16h416a16 16 0 0 0 16-16V48a16 16 0 0 0-16-16zM53.2 467a48 48 0 0 0 47.9 45h245.8a48 48 0 0 0 47.9-45L416 128H32z" class=""></path></svg>`;