feat(editor): Show the right editor in focus panel (#17062)

Co-authored-by: Charlie Kolb <charlie@n8n.io>
This commit is contained in:
Daria
2025-07-09 14:40:39 +03:00
committed by GitHub
parent c37397cb2b
commit 3aeb622978
14 changed files with 464 additions and 220 deletions

View File

@@ -2,23 +2,53 @@
import { useFocusPanelStore } from '@/stores/focusPanel.store';
import { useNodeTypesStore } from '@/stores/nodeTypes.store';
import { N8nText, N8nInput } from '@n8n/design-system';
import { computed } from 'vue';
import { computed, nextTick, ref } from 'vue';
import { useI18n } from '@n8n/i18n';
import {
formatAsExpression,
getParameterTypeOption,
isValidParameterOption,
parseFromExpression,
} from '@/utils/nodeSettingsUtils';
import { isValueExpression } from '@/utils/nodeTypesUtils';
import { useNodeHelpers } from '@/composables/useNodeHelpers';
import { useNodeSettingsParameters } from '@/composables/useNodeSettingsParameters';
import { useResolvedExpression } from '@/composables/useResolvedExpression';
import {
AI_TRANSFORM_NODE_TYPE,
type CodeExecutionMode,
type CodeNodeEditorLanguage,
type EditorType,
HTML_NODE_TYPE,
isResourceLocatorValue,
} from 'n8n-workflow';
import { useEnvironmentsStore } from '@/stores/environments.ee.store';
import { useDebounce } from '@/composables/useDebounce';
import { htmlEditorEventBus } from '@/event-bus';
import { hasFocusOnInput, isFocusableEl } from '@/utils/typesUtils';
import type { TargetNodeParameterContext } from '@/Interface';
defineOptions({ name: 'FocusPanel' });
const props = defineProps<{
executable: boolean;
isCanvasReadOnly: boolean;
}>();
const emit = defineEmits<{
focus: [];
}>();
// ESLint: false positive
// eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents
const inputField = ref<InstanceType<typeof N8nInput> | HTMLElement>();
const locale = useI18n();
const nodeHelpers = useNodeHelpers();
const focusPanelStore = useFocusPanelStore();
const nodeTypesStore = useNodeTypesStore();
const nodeSettingsParameters = useNodeSettingsParameters();
const environmentsStore = useEnvironmentsStore();
const { debounce } = useDebounce();
const focusedNodeParameter = computed(() => focusPanelStore.focusedNodeParameters[0]);
const resolvedParameter = computed(() =>
@@ -29,32 +59,119 @@ const resolvedParameter = computed(() =>
const focusPanelActive = computed(() => focusPanelStore.focusPanelActive);
const isDisabled = computed(() => {
if (!resolvedParameter.value) return false;
// shouldDisplayNodeParameter returns true if disabledOptions exists and matches, OR if disabledOptions doesn't exist
return (
!!resolvedParameter.value.parameter.disabledOptions &&
nodeSettingsParameters.shouldDisplayNodeParameter(
resolvedParameter.value.node.parameters,
resolvedParameter.value.node,
resolvedParameter.value.parameter,
'',
'disabledOptions',
)
);
});
const isDisplayed = computed(() => {
if (!resolvedParameter.value) return true;
return nodeSettingsParameters.shouldDisplayNodeParameter(
resolvedParameter.value.node.parameters,
resolvedParameter.value.node,
resolvedParameter.value.parameter,
'',
'displayOptions',
);
});
const isExecutable = computed(() => {
if (!resolvedParameter.value) return false;
if (!isDisplayed.value) return false;
const foreignCredentials = nodeHelpers.getForeignCredentialsIfSharingEnabled(
resolvedParameter.value.node.credentials,
);
return nodeHelpers.isNodeExecutable(
resolvedParameter.value.node,
props.executable,
!props.isCanvasReadOnly,
foreignCredentials,
);
});
function getTypeOption<T>(optionName: string): T | undefined {
return resolvedParameter.value
? getParameterTypeOption<T>(resolvedParameter.value.parameter, optionName)
: undefined;
}
const codeEditorMode = computed<CodeExecutionMode>(() => {
return resolvedParameter.value?.node.parameters.mode as CodeExecutionMode;
});
const editorType = computed<EditorType | 'json' | 'code' | 'cssEditor' | undefined>(() => {
return getTypeOption('editor') ?? undefined;
});
const editorLanguage = computed<CodeNodeEditorLanguage>(() => {
if (editorType.value === 'json' || resolvedParameter.value?.parameter.type === 'json')
return 'json' as CodeNodeEditorLanguage;
return getTypeOption('editorLanguage') ?? 'javaScript';
});
const editorRows = computed(() => getTypeOption<number>('rows'));
const isToolNode = computed(() =>
resolvedParameter.value ? nodeTypesStore.isToolNode(resolvedParameter.value?.node.type) : false,
);
const isHtmlNode = computed(
() => !!resolvedParameter.value && resolvedParameter.value.node.type === HTML_NODE_TYPE,
);
const expressionModeEnabled = computed(
() =>
resolvedParameter.value &&
isValueExpression(resolvedParameter.value.parameter, resolvedParameter.value.value),
);
function optionSelected() {
// TODO: Handle the option selected (command: string) from the dropdown
}
const expression = computed(() => {
if (!expressionModeEnabled.value) return '';
return isResourceLocatorValue(resolvedParameter.value)
? resolvedParameter.value.value
: resolvedParameter.value;
});
const shouldCaptureForPosthog = computed(
() => resolvedParameter.value?.node.type === AI_TRANSFORM_NODE_TYPE,
);
const isReadOnly = computed(() => props.isCanvasReadOnly || isDisabled.value);
const resolvedAdditionalExpressionData = computed(() => {
return {
$vars: environmentsStore.variablesAsObject,
};
});
const targetNodeParameterContext = computed<TargetNodeParameterContext | undefined>(() => {
if (!resolvedParameter.value) return undefined;
return {
nodeName: resolvedParameter.value.node.name,
parameterPath: resolvedParameter.value.parameterPath,
};
});
const { resolvedExpression } = useResolvedExpression({
expression,
additionalData: resolvedAdditionalExpressionData,
stringifyObject:
resolvedParameter.value && resolvedParameter.value.parameter.type !== 'multiOptions',
});
function valueChanged(value: string) {
if (resolvedParameter.value === undefined) {
@@ -68,6 +185,65 @@ function valueChanged(value: string) {
isToolNode.value,
);
}
async function setFocus() {
await nextTick();
if (inputField.value) {
if (hasFocusOnInput(inputField.value)) {
inputField.value.focusOnInput();
} else if (isFocusableEl(inputField.value)) {
inputField.value.focus();
}
}
emit('focus');
}
function optionSelected(command: string) {
if (!resolvedParameter.value) return;
switch (command) {
case 'resetValue':
return (
typeof resolvedParameter.value.parameter.default === 'string' &&
valueChanged(resolvedParameter.value.parameter.default)
);
case 'addExpression': {
const newValue = formatAsExpression(
resolvedParameter.value.value,
resolvedParameter.value.parameter.type,
);
valueChanged(typeof newValue === 'string' ? newValue : newValue.value);
void setFocus();
break;
}
case 'removeExpression': {
const newValue = parseFromExpression(
resolvedParameter.value.value,
resolvedExpression.value,
resolvedParameter.value.parameter.type,
resolvedParameter.value.parameter.default,
(resolvedParameter.value.parameter.options ?? []).filter(isValidParameterOption),
);
if (typeof newValue === 'string') {
valueChanged(newValue);
} else if (newValue && typeof (newValue as { value?: unknown }).value === 'string') {
valueChanged((newValue as { value: string }).value);
}
void setFocus();
break;
}
case 'formatHtml':
htmlEditorEventBus.emit('format-html');
return;
}
}
const valueChangedDebounced = debounce(valueChanged, { debounceTime: 0 });
</script>
<template>
@@ -104,34 +280,96 @@ function valueChanged(value: string) {
<div :class="$style.parameterOptionsWrapper">
<div></div>
<ParameterOptions
v-if="isDisplayed"
:parameter="resolvedParameter.parameter"
:value="resolvedParameter.value"
:is-read-only="false"
:is-read-only="isReadOnly"
@update:model-value="optionSelected"
/>
</div>
<div v-if="typeof resolvedParameter.value === 'string'" :class="$style.editorContainer">
<div v-if="!isDisplayed" :class="[$style.content, $style.emptyContent]">
<div :class="$style.emptyText">
<N8nText color="text-base">
{{ locale.baseText('nodeView.focusPanel.missingParameter') }}
</N8nText>
</div>
</div>
<ExpressionEditorModalInput
v-if="expressionModeEnabled"
v-else-if="expressionModeEnabled"
ref="inputField"
:model-value="resolvedParameter.value"
:class="$style.editor"
:is-read-only="false"
:is-read-only="isReadOnly"
:path="resolvedParameter.parameterPath"
data-test-id="expression-modal-input"
:target-node-parameter-context="{
nodeName: resolvedParameter.node.name,
parameterPath: resolvedParameter.parameterPath,
}"
@change="valueChanged($event.value)"
:target-node-parameter-context="targetNodeParameterContext"
@change="valueChangedDebounced($event.value)"
/>
<N8nInput
v-else
:model-value="resolvedParameter.value"
:class="$style.editor"
type="textarea"
resize="none"
@update:model-value="valueChanged($event)"
></N8nInput>
<template v-else-if="['json', 'string'].includes(resolvedParameter.parameter.type)">
<CodeNodeEditor
v-if="editorType === 'codeNodeEditor'"
:id="resolvedParameter.parameterPath"
:mode="codeEditorMode"
:model-value="resolvedParameter.value"
:default-value="resolvedParameter.parameter.default"
:language="editorLanguage"
:is-read-only="isReadOnly"
:target-node-parameter-context="targetNodeParameterContext"
fill-parent
:disable-ask-ai="true"
@update:model-value="valueChangedDebounced" />
<HtmlEditor
v-else-if="editorType === 'htmlEditor'"
:model-value="resolvedParameter.value"
:is-read-only="isReadOnly"
:rows="editorRows"
:disable-expression-coloring="!isHtmlNode"
:disable-expression-completions="!isHtmlNode"
fullscreen
@update:model-value="valueChangedDebounced" />
<CssEditor
v-else-if="editorType === 'cssEditor'"
:model-value="resolvedParameter.value"
:is-read-only="isReadOnly"
:rows="editorRows"
fullscreen
@update:model-value="valueChangedDebounced" />
<SqlEditor
v-else-if="editorType === 'sqlEditor'"
:model-value="resolvedParameter.value"
:dialect="getTypeOption('sqlDialect')"
:is-read-only="isReadOnly"
:rows="editorRows"
fullscreen
@update:model-value="valueChangedDebounced" />
<JsEditor
v-else-if="editorType === 'jsEditor'"
:model-value="resolvedParameter.value"
:is-read-only="isReadOnly"
:rows="editorRows"
:posthog-capture="shouldCaptureForPosthog"
fill-parent
@update:model-value="valueChangedDebounced" />
<JsonEditor
v-else-if="resolvedParameter.parameter.type === 'json'"
:model-value="resolvedParameter.value"
:is-read-only="isReadOnly"
:rows="editorRows"
fullscreen
fill-parent
@update:model-value="valueChangedDebounced" />
<N8nInput
v-else
ref="inputField"
:model-value="resolvedParameter.value"
:class="$style.editor"
:readonly="isReadOnly"
type="textarea"
resize="none"
@update:model-value="valueChangedDebounced"
></N8nInput
></template>
</div>
</div>
</div>
@@ -151,7 +389,7 @@ function valueChanged(value: string) {
flex-direction: column;
width: 528px;
border-left: 1px solid var(--color-foreground-base);
background: var(--color-background-base);
background: var(--color-foreground-light);
overflow-y: hidden;
}
@@ -164,7 +402,7 @@ function valueChanged(value: string) {
padding: var(--spacing-2xs);
justify-content: space-between;
border-bottom: 1px solid var(--color-foreground-base);
background: var(--color-background-xlight);
background: var(--color-foreground-xlight);
}
.content {
@@ -193,7 +431,7 @@ function valueChanged(value: string) {
.tabHeaderText {
display: flex;
gap: var(--spacing-4xs);
align-items: center;
align-items: baseline;
}
.buttonWrapper {
@@ -216,7 +454,6 @@ function valueChanged(value: string) {
}
.editorContainer {
display: flex;
height: 100%;
overflow-y: auto;
@@ -224,7 +461,7 @@ function valueChanged(value: string) {
display: flex;
height: 100%;
width: 100%;
font-size: var(--font-size-xs);
font-size: var(--font-size-2xs);
:global(.cm-editor) {
width: 100%;