mirror of
https://github.com/n8n-io/n8n.git
synced 2025-03-05 20:50:17 -08:00
165 lines
3.6 KiB
Vue
165 lines
3.6 KiB
Vue
<script setup lang="ts">
|
|
import { computed, ref, onMounted, onUnmounted, type StyleValue, watch } from 'vue';
|
|
import { useI18n } from '@/composables/useI18n';
|
|
import type { IRunDataDisplayMode, NodePanelType } from '@/Interface';
|
|
import { useDebounce } from '@/composables/useDebounce';
|
|
|
|
type Props = {
|
|
modelValue: string;
|
|
paneType?: NodePanelType;
|
|
displayMode?: IRunDataDisplayMode;
|
|
isAreaActive?: boolean;
|
|
};
|
|
|
|
const COLLAPSED_WIDTH = '30px';
|
|
const OPEN_WIDTH = '204px';
|
|
const OPEN_MIN_WIDTH = '120px';
|
|
|
|
const emit = defineEmits<{
|
|
(event: 'update:modelValue', value: Props['modelValue']): void;
|
|
(event: 'focus'): void;
|
|
}>();
|
|
|
|
const props = withDefaults(defineProps<Props>(), {
|
|
paneType: 'output',
|
|
displayMode: 'schema',
|
|
isAreaActive: false,
|
|
});
|
|
|
|
const locale = useI18n();
|
|
const { debounce } = useDebounce();
|
|
|
|
const inputRef = ref<HTMLInputElement | null>(null);
|
|
const search = ref(props.modelValue ?? '');
|
|
const opened = ref(false);
|
|
const placeholder = computed(() => {
|
|
if (props.paneType === 'output') {
|
|
return locale.baseText('ndv.search.placeholder.output');
|
|
}
|
|
|
|
if (props.displayMode === 'schema') {
|
|
return locale.baseText('ndv.search.placeholder.input.schema');
|
|
}
|
|
|
|
return locale.baseText('ndv.search.placeholder.input');
|
|
});
|
|
|
|
const style = computed<StyleValue>(() =>
|
|
opened.value ? { maxWidth: OPEN_WIDTH, minWidth: OPEN_MIN_WIDTH } : { maxWidth: COLLAPSED_WIDTH },
|
|
);
|
|
|
|
const documentKeyHandler = (event: KeyboardEvent) => {
|
|
const isTargetFormElementOrEditable =
|
|
event.target instanceof HTMLInputElement ||
|
|
event.target instanceof HTMLTextAreaElement ||
|
|
event.target instanceof HTMLSelectElement ||
|
|
(event.target as HTMLElement)?.getAttribute?.('contentEditable') === 'true';
|
|
|
|
if (event.key === '/' && props.isAreaActive && !isTargetFormElementOrEditable) {
|
|
inputRef.value?.focus();
|
|
inputRef.value?.select();
|
|
}
|
|
};
|
|
|
|
const debouncedEmitUpdate = debounce(async (value: string) => emit('update:modelValue', value), {
|
|
debounceTime: 300,
|
|
trailing: true,
|
|
});
|
|
|
|
const onSearchUpdate = (value: string) => {
|
|
search.value = value;
|
|
void debouncedEmitUpdate(value);
|
|
};
|
|
|
|
const onFocus = () => {
|
|
opened.value = true;
|
|
inputRef.value?.select();
|
|
emit('focus');
|
|
};
|
|
|
|
const onBlur = () => {
|
|
if (!props.modelValue) {
|
|
opened.value = false;
|
|
}
|
|
};
|
|
|
|
onMounted(() => {
|
|
document.addEventListener('keyup', documentKeyHandler);
|
|
});
|
|
|
|
onUnmounted(() => {
|
|
document.removeEventListener('keyup', documentKeyHandler);
|
|
});
|
|
|
|
watch(
|
|
() => props.modelValue,
|
|
(value) => {
|
|
search.value = value;
|
|
},
|
|
);
|
|
</script>
|
|
|
|
<template>
|
|
<n8n-input
|
|
ref="inputRef"
|
|
data-test-id="ndv-search"
|
|
:class="{
|
|
[$style.ioSearch]: true,
|
|
[$style.ioSearchOpened]: opened,
|
|
}"
|
|
:style="style"
|
|
:model-value="search"
|
|
:placeholder="placeholder"
|
|
size="small"
|
|
@update:model-value="onSearchUpdate"
|
|
@focus="onFocus"
|
|
@blur="onBlur"
|
|
>
|
|
<template #prefix>
|
|
<n8n-icon :class="$style.ioSearchIcon" icon="search" />
|
|
</template>
|
|
</n8n-input>
|
|
</template>
|
|
|
|
<style lang="scss" module>
|
|
@import '@/styles/variables';
|
|
|
|
.ioSearch {
|
|
transition: max-width 0.3s $ease-out-expo;
|
|
|
|
.ioSearchIcon {
|
|
color: var(--color-foreground-xdark);
|
|
cursor: pointer;
|
|
}
|
|
|
|
:global(.el-input__prefix) {
|
|
left: 8px;
|
|
}
|
|
|
|
&:global(.el-input--prefix .el-input__inner) {
|
|
padding-left: 30px;
|
|
}
|
|
|
|
input {
|
|
border: 0;
|
|
opacity: 0;
|
|
background: transparent;
|
|
cursor: pointer;
|
|
}
|
|
}
|
|
|
|
.ioSearchOpened {
|
|
.ioSearchIcon {
|
|
cursor: default;
|
|
}
|
|
|
|
input {
|
|
border: var(--input-border-color, var(--border-color-base))
|
|
var(--input-border-style, var(--border-style-base)) var(--border-width-base);
|
|
background: var(--input-background-color, var(--color-foreground-xlight));
|
|
opacity: 1;
|
|
cursor: text;
|
|
}
|
|
}
|
|
</style>
|