mirror of
https://github.com/n8n-io/n8n.git
synced 2025-03-05 20:50:17 -08:00
fix(editor): Fix save keybind in expression editor and unfocused node details view (#13640)
This commit is contained in:
parent
6d7e346e4f
commit
9ba9443460
|
@ -1,4 +1,6 @@
|
||||||
|
import { META_KEY } from '../constants';
|
||||||
import { NDV } from '../pages/ndv';
|
import { NDV } from '../pages/ndv';
|
||||||
|
import { successToast } from '../pages/notifications';
|
||||||
import { WorkflowPage as WorkflowPageClass } from '../pages/workflow';
|
import { WorkflowPage as WorkflowPageClass } from '../pages/workflow';
|
||||||
|
|
||||||
const WorkflowPage = new WorkflowPageClass();
|
const WorkflowPage = new WorkflowPageClass();
|
||||||
|
@ -11,6 +13,23 @@ describe('Expression editor modal', () => {
|
||||||
cy.on('uncaught:exception', (error) => error.name !== 'ExpressionError');
|
cy.on('uncaught:exception', (error) => error.name !== 'ExpressionError');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('Keybinds', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
WorkflowPage.actions.addNodeToCanvas('Hacker News');
|
||||||
|
WorkflowPage.actions.zoomToFit();
|
||||||
|
WorkflowPage.actions.openNode('Hacker News');
|
||||||
|
WorkflowPage.actions.openExpressionEditorModal();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should save the workflow with save keybind', () => {
|
||||||
|
WorkflowPage.getters.expressionModalInput().clear();
|
||||||
|
WorkflowPage.getters.expressionModalInput().click().type('{{ "hello"');
|
||||||
|
WorkflowPage.getters.expressionModalOutput().contains('hello');
|
||||||
|
WorkflowPage.getters.expressionModalInput().click().type(`{${META_KEY}+s}`);
|
||||||
|
successToast().should('be.visible');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe('Static data', () => {
|
describe('Static data', () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
WorkflowPage.actions.addNodeToCanvas('Hacker News');
|
WorkflowPage.actions.addNodeToCanvas('Hacker News');
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { createPinia, setActivePinia } from 'pinia';
|
import { createPinia, setActivePinia } from 'pinia';
|
||||||
import { waitFor } from '@testing-library/vue';
|
import { waitFor, waitForElementToBeRemoved, fireEvent } from '@testing-library/vue';
|
||||||
import { mock } from 'vitest-mock-extended';
|
import { mock } from 'vitest-mock-extended';
|
||||||
|
|
||||||
import NodeDetailsView from '@/components/NodeDetailsView.vue';
|
import NodeDetailsView from '@/components/NodeDetailsView.vue';
|
||||||
|
@ -24,7 +24,7 @@ vi.mock('vue-router', () => {
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
async function createPiniaWithActiveNode() {
|
async function createPiniaStore(isActiveNode: boolean) {
|
||||||
const node = mockNodes[0];
|
const node = mockNodes[0];
|
||||||
const workflow = mock<IWorkflowDb>({
|
const workflow = mock<IWorkflowDb>({
|
||||||
connections: {},
|
connections: {},
|
||||||
|
@ -42,12 +42,19 @@ async function createPiniaWithActiveNode() {
|
||||||
nodeTypesStore.setNodeTypes(defaultNodeDescriptions);
|
nodeTypesStore.setNodeTypes(defaultNodeDescriptions);
|
||||||
workflowsStore.workflow = workflow;
|
workflowsStore.workflow = workflow;
|
||||||
workflowsStore.nodeMetadata[node.name] = { pristine: true };
|
workflowsStore.nodeMetadata[node.name] = { pristine: true };
|
||||||
ndvStore.activeNodeName = node.name;
|
|
||||||
|
if (isActiveNode) {
|
||||||
|
ndvStore.activeNodeName = node.name;
|
||||||
|
}
|
||||||
|
|
||||||
await useSettingsStore().getSettings();
|
await useSettingsStore().getSettings();
|
||||||
await useUsersStore().loginWithCookie();
|
await useUsersStore().loginWithCookie();
|
||||||
|
|
||||||
return { pinia, currentWorkflow: workflowsStore.getCurrentWorkflow() };
|
return {
|
||||||
|
pinia,
|
||||||
|
currentWorkflow: workflowsStore.getCurrentWorkflow(),
|
||||||
|
nodeName: node.name,
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
describe('NodeDetailsView', () => {
|
describe('NodeDetailsView', () => {
|
||||||
|
@ -71,7 +78,7 @@ describe('NodeDetailsView', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should render correctly', async () => {
|
it('should render correctly', async () => {
|
||||||
const { pinia, currentWorkflow } = await createPiniaWithActiveNode();
|
const { pinia, currentWorkflow } = await createPiniaStore(true);
|
||||||
|
|
||||||
const renderComponent = createComponentRenderer(NodeDetailsView, {
|
const renderComponent = createComponentRenderer(NodeDetailsView, {
|
||||||
props: {
|
props: {
|
||||||
|
@ -94,4 +101,134 @@ describe('NodeDetailsView', () => {
|
||||||
|
|
||||||
await waitFor(() => expect(getByTestId('ndv')).toBeInTheDocument());
|
await waitFor(() => expect(getByTestId('ndv')).toBeInTheDocument());
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('keyboard listener', () => {
|
||||||
|
test('should register and unregister keydown listener based on modal open state', async () => {
|
||||||
|
const { pinia, currentWorkflow, nodeName } = await createPiniaStore(false);
|
||||||
|
const ndvStore = useNDVStore();
|
||||||
|
|
||||||
|
const renderComponent = createComponentRenderer(NodeDetailsView, {
|
||||||
|
props: {
|
||||||
|
teleported: false,
|
||||||
|
appendToBody: false,
|
||||||
|
workflowObject: currentWorkflow,
|
||||||
|
},
|
||||||
|
global: {
|
||||||
|
mocks: {
|
||||||
|
$route: {
|
||||||
|
name: VIEWS.WORKFLOW,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const { getByTestId, queryByTestId } = renderComponent({
|
||||||
|
pinia,
|
||||||
|
});
|
||||||
|
|
||||||
|
const addEventListenerSpy = vi.spyOn(document, 'addEventListener');
|
||||||
|
const removeEventListenerSpy = vi.spyOn(document, 'removeEventListener');
|
||||||
|
|
||||||
|
ndvStore.activeNodeName = nodeName;
|
||||||
|
|
||||||
|
await waitFor(() => expect(getByTestId('ndv')).toBeInTheDocument());
|
||||||
|
await waitFor(() => expect(queryByTestId('ndv-modal')).toBeInTheDocument());
|
||||||
|
|
||||||
|
expect(addEventListenerSpy).toHaveBeenCalledWith('keydown', expect.any(Function), true);
|
||||||
|
expect(removeEventListenerSpy).not.toHaveBeenCalledWith(
|
||||||
|
'keydown',
|
||||||
|
expect.any(Function),
|
||||||
|
true,
|
||||||
|
);
|
||||||
|
|
||||||
|
ndvStore.activeNodeName = null;
|
||||||
|
|
||||||
|
await waitForElementToBeRemoved(queryByTestId('ndv-modal'));
|
||||||
|
|
||||||
|
expect(removeEventListenerSpy).toHaveBeenCalledWith('keydown', expect.any(Function), true);
|
||||||
|
|
||||||
|
addEventListenerSpy.mockRestore();
|
||||||
|
removeEventListenerSpy.mockRestore();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should unregister keydown listener on unmount', async () => {
|
||||||
|
const { pinia, currentWorkflow, nodeName } = await createPiniaStore(false);
|
||||||
|
const ndvStore = useNDVStore();
|
||||||
|
|
||||||
|
const renderComponent = createComponentRenderer(NodeDetailsView, {
|
||||||
|
props: {
|
||||||
|
teleported: false,
|
||||||
|
appendToBody: false,
|
||||||
|
workflowObject: currentWorkflow,
|
||||||
|
},
|
||||||
|
global: {
|
||||||
|
mocks: {
|
||||||
|
$route: {
|
||||||
|
name: VIEWS.WORKFLOW,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const { getByTestId, queryByTestId, unmount } = renderComponent({
|
||||||
|
pinia,
|
||||||
|
});
|
||||||
|
|
||||||
|
ndvStore.activeNodeName = nodeName;
|
||||||
|
|
||||||
|
await waitFor(() => expect(getByTestId('ndv')).toBeInTheDocument());
|
||||||
|
await waitFor(() => expect(queryByTestId('ndv-modal')).toBeInTheDocument());
|
||||||
|
|
||||||
|
const removeEventListenerSpy = vi.spyOn(document, 'removeEventListener');
|
||||||
|
expect(removeEventListenerSpy).not.toHaveBeenCalledWith(
|
||||||
|
'keydown',
|
||||||
|
expect.any(Function),
|
||||||
|
true,
|
||||||
|
);
|
||||||
|
|
||||||
|
unmount();
|
||||||
|
|
||||||
|
expect(removeEventListenerSpy).toHaveBeenCalledWith('keydown', expect.any(Function), true);
|
||||||
|
|
||||||
|
removeEventListenerSpy.mockRestore();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should emit 'saveKeyboardShortcut' when save shortcut keybind is pressed", async () => {
|
||||||
|
const { pinia, currentWorkflow, nodeName } = await createPiniaStore(false);
|
||||||
|
const ndvStore = useNDVStore();
|
||||||
|
|
||||||
|
const renderComponent = createComponentRenderer(NodeDetailsView, {
|
||||||
|
props: {
|
||||||
|
teleported: false,
|
||||||
|
appendToBody: false,
|
||||||
|
workflowObject: currentWorkflow,
|
||||||
|
},
|
||||||
|
global: {
|
||||||
|
mocks: {
|
||||||
|
$route: {
|
||||||
|
name: VIEWS.WORKFLOW,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const { getByTestId, queryByTestId, emitted } = renderComponent({
|
||||||
|
pinia,
|
||||||
|
});
|
||||||
|
|
||||||
|
ndvStore.activeNodeName = nodeName;
|
||||||
|
|
||||||
|
await waitFor(() => expect(getByTestId('ndv')).toBeInTheDocument());
|
||||||
|
await waitFor(() => expect(queryByTestId('ndv-modal')).toBeInTheDocument());
|
||||||
|
|
||||||
|
await fireEvent.keyDown(getByTestId('ndv'), {
|
||||||
|
key: 's',
|
||||||
|
ctrlKey: true,
|
||||||
|
bubbles: true,
|
||||||
|
cancelable: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(emitted().saveKeyboardShortcut).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -352,15 +352,19 @@ const setIsTooltipVisible = ({ isTooltipVisible }: DataPinningDiscoveryEvent) =>
|
||||||
|
|
||||||
const onKeyDown = (e: KeyboardEvent) => {
|
const onKeyDown = (e: KeyboardEvent) => {
|
||||||
if (e.key === 's' && deviceSupport.isCtrlKeyPressed(e)) {
|
if (e.key === 's' && deviceSupport.isCtrlKeyPressed(e)) {
|
||||||
e.stopPropagation();
|
onSaveWorkflow(e);
|
||||||
e.preventDefault();
|
|
||||||
|
|
||||||
if (props.readOnly) return;
|
|
||||||
|
|
||||||
emit('saveKeyboardShortcut', e);
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const onSaveWorkflow = (e: KeyboardEvent) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
if (props.readOnly) return;
|
||||||
|
|
||||||
|
emit('saveKeyboardShortcut', e);
|
||||||
|
};
|
||||||
|
|
||||||
const onInputItemHover = (e: { itemIndex: number; outputIndex: number } | null) => {
|
const onInputItemHover = (e: { itemIndex: number; outputIndex: number } | null) => {
|
||||||
if (e === null || !inputNodeName.value || !isPairedItemHoveringEnabled.value) {
|
if (e === null || !inputNodeName.value || !isPairedItemHoveringEnabled.value) {
|
||||||
ndvStore.setHoveringItem(null);
|
ndvStore.setHoveringItem(null);
|
||||||
|
@ -597,11 +601,25 @@ const onSearch = (search: string) => {
|
||||||
isPairedItemHoveringEnabled.value = !search;
|
isPairedItemHoveringEnabled.value = !search;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const registerKeyboardListener = () => {
|
||||||
|
document.addEventListener('keydown', onKeyDown, true);
|
||||||
|
};
|
||||||
|
|
||||||
|
const unregisterKeyboardListener = () => {
|
||||||
|
document.removeEventListener('keydown', onKeyDown, true);
|
||||||
|
};
|
||||||
|
|
||||||
//watchers
|
//watchers
|
||||||
|
|
||||||
watch(
|
watch(
|
||||||
activeNode,
|
activeNode,
|
||||||
(node, oldNode) => {
|
(node, oldNode) => {
|
||||||
|
if (node && !oldNode) {
|
||||||
|
registerKeyboardListener();
|
||||||
|
} else if (!node) {
|
||||||
|
unregisterKeyboardListener();
|
||||||
|
}
|
||||||
|
|
||||||
if (node && node.name !== oldNode?.name && !isActiveStickyNode.value) {
|
if (node && node.name !== oldNode?.name && !isActiveStickyNode.value) {
|
||||||
runInputIndex.value = -1;
|
runInputIndex.value = -1;
|
||||||
runOutputIndex.value = -1;
|
runOutputIndex.value = -1;
|
||||||
|
@ -655,6 +673,7 @@ watch(
|
||||||
},
|
},
|
||||||
{ immediate: true },
|
{ immediate: true },
|
||||||
);
|
);
|
||||||
|
|
||||||
watch(maxOutputRun, () => {
|
watch(maxOutputRun, () => {
|
||||||
runOutputIndex.value = -1;
|
runOutputIndex.value = -1;
|
||||||
});
|
});
|
||||||
|
@ -681,6 +700,7 @@ onMounted(() => {
|
||||||
|
|
||||||
onBeforeUnmount(() => {
|
onBeforeUnmount(() => {
|
||||||
dataPinningEventBus.off('data-pinning-discovery', setIsTooltipVisible);
|
dataPinningEventBus.off('data-pinning-discovery', setIsTooltipVisible);
|
||||||
|
unregisterKeyboardListener();
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
@ -719,8 +739,8 @@ onBeforeUnmount(() => {
|
||||||
v-if="activeNode"
|
v-if="activeNode"
|
||||||
ref="container"
|
ref="container"
|
||||||
class="data-display"
|
class="data-display"
|
||||||
|
data-test-id="ndv-modal"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
@keydown.capture="onKeyDown"
|
|
||||||
>
|
>
|
||||||
<div :class="$style.modalBackground" @click="close"></div>
|
<div :class="$style.modalBackground" @click="close"></div>
|
||||||
<NDVDraggablePanels
|
<NDVDraggablePanels
|
||||||
|
|
Loading…
Reference in a new issue