2025-01-24 04:38:54 -08:00
|
|
|
import { getCanvasNodes } from '../composables/workflow';
|
2024-06-10 06:49:50 -07:00
|
|
|
import {
|
|
|
|
SCHEDULE_TRIGGER_NODE_NAME,
|
|
|
|
CODE_NODE_NAME,
|
|
|
|
SET_NODE_NAME,
|
2025-01-24 04:38:54 -08:00
|
|
|
MANUAL_TRIGGER_NODE_NAME,
|
|
|
|
MANUAL_TRIGGER_NODE_DISPLAY_NAME,
|
2024-06-10 06:49:50 -07:00
|
|
|
} from '../constants';
|
2023-12-21 23:42:53 -08:00
|
|
|
import { MessageBox as MessageBoxClass } from '../pages/modals/message-box';
|
2023-01-18 06:48:36 -08:00
|
|
|
import { NDV } from '../pages/ndv';
|
2024-09-18 00:19:33 -07:00
|
|
|
import { WorkflowPage as WorkflowPageClass } from '../pages/workflow';
|
2022-12-14 01:33:44 -08:00
|
|
|
|
|
|
|
const WorkflowPage = new WorkflowPageClass();
|
2023-12-21 23:42:53 -08:00
|
|
|
const messageBox = new MessageBoxClass();
|
2023-01-18 06:48:36 -08:00
|
|
|
const ndv = new NDV();
|
2022-12-14 01:33:44 -08:00
|
|
|
|
|
|
|
describe('Undo/Redo', () => {
|
2023-02-24 09:07:35 -08:00
|
|
|
beforeEach(() => {
|
2022-12-14 01:33:44 -08:00
|
|
|
WorkflowPage.actions.visit();
|
|
|
|
});
|
|
|
|
|
2023-11-20 05:37:12 -08:00
|
|
|
it('should undo/redo deleting node using context menu', () => {
|
2022-12-14 01:33:44 -08:00
|
|
|
WorkflowPage.actions.addNodeToCanvas(SCHEDULE_TRIGGER_NODE_NAME);
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(CODE_NODE_NAME);
|
2023-11-20 05:37:12 -08:00
|
|
|
WorkflowPage.actions.deleteNodeFromContextMenu(CODE_NODE_NAME);
|
2022-12-14 01:33:44 -08:00
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 1);
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 0);
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 2);
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 1);
|
|
|
|
WorkflowPage.actions.hitRedo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 1);
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 0);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should undo/redo deleting node using keyboard shortcut', () => {
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(SCHEDULE_TRIGGER_NODE_NAME);
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(CODE_NODE_NAME);
|
|
|
|
WorkflowPage.getters.canvasNodeByName(CODE_NODE_NAME).click();
|
|
|
|
cy.get('body').type('{backspace}');
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 1);
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 0);
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 2);
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 1);
|
|
|
|
WorkflowPage.actions.hitRedo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 1);
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 0);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should undo/redo deleting node between two connected nodes', () => {
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(SCHEDULE_TRIGGER_NODE_NAME);
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(CODE_NODE_NAME);
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(SET_NODE_NAME);
|
|
|
|
WorkflowPage.getters.canvasNodeByName(CODE_NODE_NAME).click();
|
|
|
|
WorkflowPage.actions.zoomToFit();
|
|
|
|
cy.get('body').type('{backspace}');
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 2);
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 1);
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 3);
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 2);
|
|
|
|
WorkflowPage.actions.hitRedo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 2);
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 1);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should undo/redo deleting whole workflow', () => {
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(SCHEDULE_TRIGGER_NODE_NAME);
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(CODE_NODE_NAME);
|
|
|
|
cy.get('body').type('{esc}');
|
|
|
|
cy.get('body').type('{esc}');
|
2024-06-11 05:45:15 -07:00
|
|
|
WorkflowPage.actions.hitDeleteAllNodes();
|
2022-12-14 01:33:44 -08:00
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 0);
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 2);
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 1);
|
|
|
|
WorkflowPage.actions.hitRedo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 0);
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 0);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should undo/redo moving nodes', () => {
|
2025-01-24 04:38:54 -08:00
|
|
|
WorkflowPage.actions.addNodeToCanvas(MANUAL_TRIGGER_NODE_NAME);
|
|
|
|
WorkflowPage.getters.canvasNodeByName(MANUAL_TRIGGER_NODE_DISPLAY_NAME).click();
|
2022-12-14 01:33:44 -08:00
|
|
|
WorkflowPage.actions.addNodeToCanvas(CODE_NODE_NAME);
|
2024-06-25 02:14:02 -07:00
|
|
|
|
2025-01-24 04:38:54 -08:00
|
|
|
WorkflowPage.actions.zoomToFit();
|
2023-10-25 05:34:47 -07:00
|
|
|
|
2025-01-24 04:38:54 -08:00
|
|
|
getCanvasNodes()
|
|
|
|
.last()
|
|
|
|
.then(($node) => {
|
|
|
|
const { x: x1, y: y1 } = $node[0].getBoundingClientRect();
|
|
|
|
|
|
|
|
cy.ifCanvasVersion(
|
|
|
|
() => {
|
|
|
|
cy.drag('[data-test-id="canvas-node"].jtk-drag-selected', [50, 150], {
|
|
|
|
clickToFinish: true,
|
|
|
|
});
|
|
|
|
},
|
|
|
|
() => {
|
|
|
|
cy.drag(getCanvasNodes().last(), [50, 150], {
|
|
|
|
realMouse: true,
|
|
|
|
abs: true,
|
|
|
|
});
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
getCanvasNodes()
|
|
|
|
.last()
|
|
|
|
.then(($node) => {
|
|
|
|
const { x: x2, y: y2 } = $node[0].getBoundingClientRect();
|
|
|
|
expect(x2).to.be.greaterThan(x1);
|
|
|
|
expect(y2).to.be.greaterThan(y1);
|
|
|
|
});
|
|
|
|
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
|
|
|
|
getCanvasNodes()
|
|
|
|
.last()
|
|
|
|
.then(($node) => {
|
|
|
|
const { x: x3, y: y3 } = $node[0].getBoundingClientRect();
|
|
|
|
expect(x3).to.equal(x1);
|
|
|
|
expect(y3).to.equal(y1);
|
|
|
|
});
|
|
|
|
|
|
|
|
WorkflowPage.actions.hitRedo();
|
|
|
|
|
|
|
|
getCanvasNodes()
|
|
|
|
.last()
|
|
|
|
.then(($node) => {
|
|
|
|
const { x: x4, y: y4 } = $node[0].getBoundingClientRect();
|
|
|
|
expect(x4).to.be.greaterThan(x1);
|
|
|
|
expect(y4).to.be.greaterThan(y1);
|
|
|
|
});
|
2024-06-25 02:14:02 -07:00
|
|
|
});
|
2022-12-14 01:33:44 -08:00
|
|
|
});
|
|
|
|
|
2023-11-20 05:37:12 -08:00
|
|
|
it('should undo/redo deleting a connection using context menu', () => {
|
2022-12-14 01:33:44 -08:00
|
|
|
WorkflowPage.actions.addNodeToCanvas(SCHEDULE_TRIGGER_NODE_NAME);
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(CODE_NODE_NAME);
|
2024-10-21 06:13:09 -07:00
|
|
|
WorkflowPage.actions.deleteNodeBetweenNodes(SCHEDULE_TRIGGER_NODE_NAME, CODE_NODE_NAME);
|
2022-12-14 01:33:44 -08:00
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 0);
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 1);
|
|
|
|
WorkflowPage.actions.hitRedo();
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 0);
|
|
|
|
});
|
|
|
|
|
2023-11-20 05:37:12 -08:00
|
|
|
it('should undo/redo disabling a node using context menu', () => {
|
2022-12-14 01:33:44 -08:00
|
|
|
WorkflowPage.actions.addNodeToCanvas(SCHEDULE_TRIGGER_NODE_NAME);
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(CODE_NODE_NAME);
|
2023-11-20 05:37:12 -08:00
|
|
|
WorkflowPage.actions.disableNode(CODE_NODE_NAME);
|
2022-12-14 01:33:44 -08:00
|
|
|
WorkflowPage.getters.disabledNodes().should('have.length', 1);
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.disabledNodes().should('have.length', 0);
|
|
|
|
WorkflowPage.actions.hitRedo();
|
|
|
|
WorkflowPage.getters.disabledNodes().should('have.length', 1);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should undo/redo disabling a node using keyboard shortcut', () => {
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(SCHEDULE_TRIGGER_NODE_NAME);
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(CODE_NODE_NAME);
|
|
|
|
WorkflowPage.getters.canvasNodes().last().click();
|
|
|
|
WorkflowPage.actions.hitDisableNodeShortcut();
|
|
|
|
WorkflowPage.getters.disabledNodes().should('have.length', 1);
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.disabledNodes().should('have.length', 0);
|
|
|
|
WorkflowPage.actions.hitRedo();
|
|
|
|
WorkflowPage.getters.disabledNodes().should('have.length', 1);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should undo/redo disabling multiple nodes', () => {
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(SCHEDULE_TRIGGER_NODE_NAME);
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(CODE_NODE_NAME);
|
|
|
|
cy.get('body').type('{esc}');
|
|
|
|
cy.get('body').type('{esc}');
|
2024-06-11 05:45:15 -07:00
|
|
|
WorkflowPage.actions.hitSelectAll();
|
2022-12-14 01:33:44 -08:00
|
|
|
WorkflowPage.actions.hitDisableNodeShortcut();
|
|
|
|
WorkflowPage.getters.disabledNodes().should('have.length', 2);
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.disabledNodes().should('have.length', 0);
|
|
|
|
WorkflowPage.actions.hitRedo();
|
|
|
|
WorkflowPage.getters.disabledNodes().should('have.length', 2);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should undo/redo duplicating a node', () => {
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(SCHEDULE_TRIGGER_NODE_NAME);
|
|
|
|
WorkflowPage.actions.addNodeToCanvas(CODE_NODE_NAME);
|
2023-11-20 05:37:12 -08:00
|
|
|
WorkflowPage.actions.duplicateNode(CODE_NODE_NAME);
|
2022-12-14 01:33:44 -08:00
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.length', 2);
|
|
|
|
WorkflowPage.actions.hitRedo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.length', 3);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should undo/redo pasting nodes', () => {
|
2022-12-15 07:39:59 -08:00
|
|
|
cy.fixture('Test_workflow-actions_paste-data.json').then((data) => {
|
2022-12-14 01:33:44 -08:00
|
|
|
cy.get('body').paste(JSON.stringify(data));
|
|
|
|
WorkflowPage.actions.zoomToFit();
|
2024-01-26 06:09:50 -08:00
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 5);
|
2022-12-14 01:33:44 -08:00
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 0);
|
|
|
|
WorkflowPage.actions.hitRedo();
|
2024-01-26 06:09:50 -08:00
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 5);
|
2022-12-14 01:33:44 -08:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2023-12-20 01:13:59 -08:00
|
|
|
it('should be able to copy and paste pinned data nodes in workflows with dynamic Switch node', () => {
|
|
|
|
cy.fixture('Test_workflow_form_switch.json').then((data) => {
|
|
|
|
cy.get('body').paste(JSON.stringify(data));
|
|
|
|
});
|
|
|
|
WorkflowPage.actions.zoomToFit();
|
|
|
|
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.length', 2);
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 1);
|
|
|
|
cy.get(WorkflowPage.getters.getEndpointSelector('input', 'Switch')).should('have.length', 1);
|
|
|
|
|
|
|
|
cy.fixture('Test_workflow_form_switch.json').then((data) => {
|
|
|
|
cy.get('body').paste(JSON.stringify(data));
|
|
|
|
});
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.length', 4);
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 2);
|
|
|
|
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.length', 2);
|
|
|
|
WorkflowPage.getters.nodeConnections().should('have.length', 1);
|
|
|
|
cy.get(WorkflowPage.getters.getEndpointSelector('input', 'Switch')).should('have.length', 1);
|
|
|
|
});
|
2023-12-21 23:42:53 -08:00
|
|
|
|
|
|
|
it('should not undo/redo when NDV or a modal is open', () => {
|
|
|
|
WorkflowPage.actions.addInitialNodeToCanvas(SCHEDULE_TRIGGER_NODE_NAME, { keepNdvOpen: true });
|
|
|
|
// Try while NDV is open
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 1);
|
|
|
|
ndv.getters.backToCanvas().click();
|
|
|
|
// Try while modal is open
|
|
|
|
cy.getByTestId('menu-item').contains('About n8n').click({ force: true });
|
|
|
|
cy.getByTestId('about-modal').should('be.visible');
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 1);
|
|
|
|
cy.getByTestId('close-about-modal-button').click();
|
|
|
|
// Should work now
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 0);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not undo/redo when NDV or a prompt is open', () => {
|
|
|
|
WorkflowPage.actions.addInitialNodeToCanvas(SCHEDULE_TRIGGER_NODE_NAME, { keepNdvOpen: false });
|
|
|
|
WorkflowPage.getters.workflowMenu().click();
|
|
|
|
WorkflowPage.getters.workflowMenuItemImportFromURLItem().should('be.visible');
|
|
|
|
WorkflowPage.getters.workflowMenuItemImportFromURLItem().click();
|
|
|
|
// Try while prompt is open
|
|
|
|
messageBox.getters.header().click();
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 1);
|
|
|
|
// Close prompt and try again
|
|
|
|
messageBox.actions.cancel();
|
|
|
|
WorkflowPage.actions.hitUndo();
|
|
|
|
WorkflowPage.getters.canvasNodes().should('have.have.length', 0);
|
|
|
|
});
|
2022-12-14 01:33:44 -08:00
|
|
|
});
|