n8n/packages/editor-ui/src/components/mixins/nodeBase.ts

380 lines
11 KiB
TypeScript
Raw Normal View History

2019-06-23 03:35:23 -07:00
import { IConnectionsUi, IEndpointOptions, INodeUi, XYPositon } from '@/Interface';
import mixins from 'vue-typed-mixins';
import { deviceSupportHelpers } from '@/components/mixins/deviceSupportHelpers';
2019-06-23 03:35:23 -07:00
import { nodeIndex } from '@/components/mixins/nodeIndex';
:sparkles: Introduce telemetry (#2099) * introduce analytics * add user survey backend * add user survey backend * set answers on survey submit Co-authored-by: Mutasem Aldmour <4711238+mutdmour@users.noreply.github.com> * change name to personalization * lint Co-authored-by: Mutasem Aldmour <4711238+mutdmour@users.noreply.github.com> * N8n 2495 add personalization modal (#2280) * update modals * add onboarding modal * implement questions * introduce analytics * simplify impl * implement survey handling * add personalized cateogry * update modal behavior * add thank you view * handle empty cases * rename modal * standarize modal names * update image, add tags to headings * remove unused file * remove unused interfaces * clean up footer spacing * introduce analytics * refactor to fix bug * update endpoint * set min height * update stories * update naming from questions to survey * remove spacing after core categories * fix bug in logic * sort nodes * rename types * merge with be * rename userSurvey * clean up rest api * use constants for keys * use survey keys * clean up types * move personalization to its own file Co-authored-by: ahsan-virani <ahsan.virani@gmail.com> * Survey new options (#2300) * split up options * fix quotes * remove unused import * add user created workflow event (#2301) * simplify env vars * fix versionCli on FE * update personalization env * fix event User opened Credentials panel * fix select modal spacing * fix nodes panel event * fix workflow id in workflow execute event * improve telemetry error logging * fix config and stop process events * add flush call on n8n stop * ready for release * improve telemetry process exit * fix merge * improve n8n stop events Co-authored-by: Mutasem Aldmour <4711238+mutdmour@users.noreply.github.com> Co-authored-by: Mutasem <mutdmour@gmail.com> Co-authored-by: Jan Oberhauser <jan.oberhauser@gmail.com>
2021-10-18 20:57:49 -07:00
import { NODE_NAME_PREFIX, NO_OP_NODE_TYPE } from '@/constants';
2021-10-22 03:28:38 -07:00
import { getStyleTokenValue } from '../helpers';
2019-06-23 03:35:23 -07:00
export const nodeBase = mixins(
deviceSupportHelpers,
nodeIndex,
).extend({
2019-06-23 03:35:23 -07:00
mounted () {
// Initialize the node
if (this.data !== null) {
this.__addNode(this.data);
}
},
computed: {
data (): INodeUi {
2021-10-15 08:42:12 -07:00
return this.$store.getters.getNodeByName(this.name);
2019-06-23 03:35:23 -07:00
},
hasIssues (): boolean {
if (this.data.issues !== undefined && Object.keys(this.data.issues).length) {
return true;
}
return false;
},
nodeName (): string {
return NODE_NAME_PREFIX + this.nodeIndex;
},
nodeIndex (): string {
return this.$store.getters.getNodeIndex(this.data.name).toString();
},
nodePosition (): object {
const node = this.$store.getters.nodesByName[this.name]; // position responsive to store changes
2019-06-23 03:35:23 -07:00
const returnStyles: {
[key: string]: string;
} = {
2021-10-15 08:42:12 -07:00
left: node.position[0] + 'px',
top: node.position[1] + 'px',
};
2019-06-23 03:35:23 -07:00
return returnStyles;
},
},
props: [
'name',
'nodeId',
'instance',
2020-05-24 05:06:22 -07:00
'isReadOnly',
'isActive',
2019-06-23 03:35:23 -07:00
],
methods: {
__addNode (node: INodeUi) {
// TODO: Later move the node-connection definitions to a special file
let nodeTypeData = this.$store.getters.nodeType(node.type);
2019-06-23 03:35:23 -07:00
const nodeConnectors: IConnectionsUi = {
main: {
input: {
uuid: '-input',
2019-06-23 03:35:23 -07:00
maxConnections: -1,
endpoint: 'Rectangle',
endpointStyle: {
2021-10-28 07:26:15 -07:00
width: 8,
height: nodeTypeData && nodeTypeData.outputs.length > 2 ? 18 : 20,
fill: getStyleTokenValue('--color-foreground-xdark'),
stroke: getStyleTokenValue('--color-foreground-xdark'),
lineWidth: 0,
},
2021-10-22 03:28:38 -07:00
endpointHoverStyle: {
2021-10-28 07:26:15 -07:00
width: 8,
height: nodeTypeData && nodeTypeData.outputs.length > 2 ? 18 : 20,
2021-10-22 03:28:38 -07:00
fill: getStyleTokenValue('--color-primary'),
stroke: getStyleTokenValue('--color-primary'),
lineWidth: 0,
},
2019-06-23 03:35:23 -07:00
dragAllowedWhenFull: true,
},
output: {
uuid: '-output',
2019-06-23 03:35:23 -07:00
maxConnections: -1,
endpoint: 'Dot',
endpointStyle: {
2021-10-28 07:26:15 -07:00
radius: nodeTypeData && nodeTypeData.outputs.length > 2 ? 7 : 9,
fill: getStyleTokenValue('--color-foreground-xdark'),
outlineStroke: 'none',
},
2021-10-22 03:28:38 -07:00
endpointHoverStyle: {
2021-10-28 07:26:15 -07:00
radius: nodeTypeData && nodeTypeData.outputs.length > 2 ? 7 : 9,
2021-10-22 03:28:38 -07:00
fill: getStyleTokenValue('--color-primary'),
outlineStroke: 'none',
},
2019-06-23 03:35:23 -07:00
dragAllowedWhenFull: true,
},
},
};
if (!nodeTypeData) {
// If node type is not know use by default the base.noOp data to display it
:sparkles: Introduce telemetry (#2099) * introduce analytics * add user survey backend * add user survey backend * set answers on survey submit Co-authored-by: Mutasem Aldmour <4711238+mutdmour@users.noreply.github.com> * change name to personalization * lint Co-authored-by: Mutasem Aldmour <4711238+mutdmour@users.noreply.github.com> * N8n 2495 add personalization modal (#2280) * update modals * add onboarding modal * implement questions * introduce analytics * simplify impl * implement survey handling * add personalized cateogry * update modal behavior * add thank you view * handle empty cases * rename modal * standarize modal names * update image, add tags to headings * remove unused file * remove unused interfaces * clean up footer spacing * introduce analytics * refactor to fix bug * update endpoint * set min height * update stories * update naming from questions to survey * remove spacing after core categories * fix bug in logic * sort nodes * rename types * merge with be * rename userSurvey * clean up rest api * use constants for keys * use survey keys * clean up types * move personalization to its own file Co-authored-by: ahsan-virani <ahsan.virani@gmail.com> * Survey new options (#2300) * split up options * fix quotes * remove unused import * add user created workflow event (#2301) * simplify env vars * fix versionCli on FE * update personalization env * fix event User opened Credentials panel * fix select modal spacing * fix nodes panel event * fix workflow id in workflow execute event * improve telemetry error logging * fix config and stop process events * add flush call on n8n stop * ready for release * improve telemetry process exit * fix merge * improve n8n stop events Co-authored-by: Mutasem Aldmour <4711238+mutdmour@users.noreply.github.com> Co-authored-by: Mutasem <mutdmour@gmail.com> Co-authored-by: Jan Oberhauser <jan.oberhauser@gmail.com>
2021-10-18 20:57:49 -07:00
nodeTypeData = this.$store.getters.nodeType(NO_OP_NODE_TYPE);
2019-06-23 03:35:23 -07:00
}
const anchorPositions: {
[key: string]: {
[key: number]: string[] | number[][];
}
} = {
input: {
1: [
2021-10-28 07:26:15 -07:00
[0.01, 0.5, -1, 0],
2019-06-23 03:35:23 -07:00
],
2: [
2021-10-28 07:26:15 -07:00
[0.01, 0.3, -1, 0],
[0.01, 0.7, -1, 0],
2019-06-23 03:35:23 -07:00
],
3: [
2021-10-28 07:26:15 -07:00
[0.01, 0.25, -1, 0],
[0.01, 0.5, -1, 0],
[0.01, 0.75, -1, 0],
2019-06-23 03:35:23 -07:00
],
4: [
2021-10-28 07:26:15 -07:00
[0.01, 0.2, -1, 0],
[0.01, 0.4, -1, 0],
[0.01, 0.6, -1, 0],
[0.01, 0.8, -1, 0],
],
2019-06-23 03:35:23 -07:00
},
output: {
1: [
2021-10-28 07:26:15 -07:00
[.99, 0.5, 1, 0],
2019-06-23 03:35:23 -07:00
],
2: [
2021-10-28 07:26:15 -07:00
[.99, 0.3, 1, 0],
[.99, 0.7, 1, 0],
2019-06-23 03:35:23 -07:00
],
3: [
2021-10-28 07:26:15 -07:00
[.99, 0.25, 1, 0],
[.99, 0.5, 1, 0],
[.99, 0.75, 1, 0],
2019-06-23 03:35:23 -07:00
],
4: [
2021-10-28 07:26:15 -07:00
[.99, 0.2, 1, 0],
[.99, 0.4, 1, 0],
[.99, 0.6, 1, 0],
[.99, 0.8, 1, 0],
],
2019-06-23 03:35:23 -07:00
},
};
// Add Inputs
let index, inputData, anchorPosition;
let newEndpointData: IEndpointOptions;
let indexData: {
[key: string]: number;
} = {};
nodeTypeData.inputs.forEach((inputName: string) => {
// @ts-ignore
inputData = nodeConnectors[inputName].input;
// Increment the index for inputs with current name
if (indexData.hasOwnProperty(inputName)) {
indexData[inputName]++;
} else {
indexData[inputName] = 0;
}
index = indexData[inputName];
// Get the position of the anchor depending on how many it has
anchorPosition = anchorPositions.input[nodeTypeData.inputs.length][index];
newEndpointData = {
uuid: `${this.nodeIndex}` + inputData.uuid + index,
anchor: anchorPosition,
maxConnections: inputData.maxConnections,
endpoint: inputData.endpoint,
endpointStyle: inputData.endpointStyle,
2021-10-22 03:28:38 -07:00
endpointHoverStyle: inputData.endpointHoverStyle,
2019-06-23 03:35:23 -07:00
isSource: false,
2020-05-24 05:06:22 -07:00
isTarget: !this.isReadOnly,
2019-06-23 03:35:23 -07:00
parameters: {
nodeIndex: this.nodeIndex,
type: inputName,
index,
},
dragAllowedWhenFull: inputData.dragAllowedWhenFull,
dropOptions: {
tolerance: 'touch',
hoverClass: 'dropHover',
},
};
if (nodeTypeData.inputNames) {
// Apply input names if they got set
newEndpointData.overlays = [
['Label',
{
id: 'input-name-label',
2021-10-28 08:17:50 -07:00
location: [-3, 0.5],
label: nodeTypeData.inputNames[index],
cssClass: 'node-input-endpoint-label',
visible: true,
},
],
];
}
2019-06-23 03:35:23 -07:00
this.instance.addEndpoint(this.nodeName, newEndpointData);
// TODO: Activate again if it makes sense. Currently makes problems when removing
// connection on which the input has a name. It does not get hidden because
// the endpoint to which it connects when letting it go over the node is
// different to the regular one (have different ids). So that seems to make
// problems when hiding the input-name.
// if (index === 0 && inputName === 'main') {
// // Make the first main-input the default one to connect to when connection gets dropped on node
// this.instance.makeTarget(this.nodeName, newEndpointData);
// }
2019-06-23 03:35:23 -07:00
});
// Add Outputs
indexData = {};
nodeTypeData.outputs.forEach((inputName: string) => {
inputData = nodeConnectors[inputName].output;
// Increment the index for outputs with current name
if (indexData.hasOwnProperty(inputName)) {
indexData[inputName]++;
} else {
indexData[inputName] = 0;
}
index = indexData[inputName];
// Get the position of the anchor depending on how many it has
anchorPosition = anchorPositions.output[nodeTypeData.outputs.length][index];
newEndpointData = {
uuid: `${this.nodeIndex}` + inputData.uuid + index,
anchor: anchorPosition,
maxConnections: inputData.maxConnections,
endpoint: inputData.endpoint,
endpointStyle: inputData.endpointStyle,
2021-10-22 03:28:38 -07:00
endpointHoverStyle: inputData.endpointHoverStyle,
2020-05-24 05:06:22 -07:00
isSource: !this.isReadOnly,
2019-06-23 03:35:23 -07:00
isTarget: false,
parameters: {
nodeIndex: this.nodeIndex,
type: inputName,
index,
},
dragAllowedWhenFull: inputData.dragAllowedWhenFull,
dragProxy: ['Rectangle', { width: 1, height: 1, strokeWidth: 0 }],
};
if (nodeTypeData.outputNames) {
// Apply output names if they got set
newEndpointData.overlays = [
['Label',
{
id: 'output-name-label',
2021-10-28 08:21:31 -07:00
location: [1.9, 0.5],
2019-06-23 03:35:23 -07:00
label: nodeTypeData.outputNames[index],
cssClass: 'node-output-endpoint-label',
2019-06-23 03:35:23 -07:00
visible: true,
},
],
];
}
this.instance.addEndpoint(this.nodeName, newEndpointData);
});
// TODO: This caused problems with displaying old information
// https://github.com/jsplumb/katavorio/wiki
// https://jsplumb.github.io/jsplumb/home.html
// Make nodes draggable
this.instance.draggable(this.nodeName, {
grid: [10, 10],
start: (params: { e: MouseEvent }) => {
if (this.isReadOnly === true) {
// Do not allow to move nodes in readOnly mode
return false;
}
if (params.e && !this.$store.getters.isNodeSelected(this.data.name)) {
// Only the node which gets dragged directly gets an event, for all others it is
// undefined. So check if the currently dragged node is selected and if not clear
// the drag-selection.
this.instance.clearDragSelection();
this.$store.commit('resetSelectedNodes');
}
this.$store.commit('addActiveAction', 'dragActive');
return true;
},
stop: (params: { e: MouseEvent }) => {
if (this.$store.getters.isActionActive('dragActive')) {
const moveNodes = this.$store.getters.getSelectedNodes.slice();
const selectedNodeNames = moveNodes.map((node: INodeUi) => node.name);
if (!selectedNodeNames.includes(this.data.name)) {
// If the current node is not in selected add it to the nodes which
// got moved manually
moveNodes.push(this.data);
2019-06-23 03:35:23 -07:00
}
// This does for some reason just get called once for the node that got clicked
// even though "start" and "drag" gets called for all. So lets do for now
// some dirty DOM query to get the new positions till I have more time to
// create a proper solution
let newNodePositon: XYPositon;
moveNodes.forEach((node: INodeUi) => {
const nodeElement = `node-${this.getNodeIndex(node.name)}`;
const element = document.getElementById(nodeElement);
if (element === null) {
return;
2019-06-23 03:35:23 -07:00
}
newNodePositon = [
parseInt(element.style.left!.slice(0, -2), 10),
parseInt(element.style.top!.slice(0, -2), 10),
];
const updateInformation = {
name: node.name,
properties: {
// @ts-ignore, draggable does not have definitions
position: newNodePositon,
},
};
this.$store.commit('updateNodeProperties', updateInformation);
});
2021-10-13 03:45:41 -07:00
this.$emit('moved', node);
}
},
filter: '.node-description, .node-description .node-name, .node-description .node-subtitle',
});
2019-06-23 03:35:23 -07:00
},
touchEnd(e: MouseEvent) {
if (this.isTouchDevice) {
if (this.$store.getters.isActionActive('dragActive')) {
this.$store.commit('removeActiveAction', 'dragActive');
}
}
},
mouseLeftClick (e: MouseEvent) {
// @ts-ignore
const path = e.path || (e.composedPath && e.composedPath());
for (let index = 0; index < path.length; index++) {
if (path[index].className && typeof path[index].className === 'string' && path[index].className.includes('no-select-on-click')) {
return;
}
}
if (!this.isTouchDevice) {
if (this.$store.getters.isActionActive('dragActive')) {
this.$store.commit('removeActiveAction', 'dragActive');
} else {
if (this.isCtrlKeyPressed(e) === false) {
this.$emit('deselectAllNodes');
}
if (this.$store.getters.isNodeSelected(this.data.name)) {
this.$emit('deselectNode', this.name);
} else {
this.$emit('nodeSelected', this.name);
}
}
2019-06-23 03:35:23 -07:00
}
},
},
});