Files
n8n-enterprise-unlocked/packages/frontend/editor-ui/src/components/ImportCurlModal.vue

149 lines
3.5 KiB
Vue

<script lang="ts" setup>
import Modal from '@/components/Modal.vue';
import { IMPORT_CURL_MODAL_KEY } from '@/constants';
import { onMounted, ref } from 'vue';
import { useUIStore } from '@/stores/ui.store';
import { createEventBus } from '@n8n/utils/event-bus';
import { useTelemetry } from '@/composables/useTelemetry';
import { useI18n } from '@n8n/i18n';
import { useNDVStore } from '@/stores/ndv.store';
const telemetry = useTelemetry();
const i18n = useI18n();
const uiStore = useUIStore();
const ndvStore = useNDVStore();
const curlCommand = ref('');
const modalBus = createEventBus();
const inputRef = ref<HTMLTextAreaElement | null>(null);
onMounted(() => {
const curlCommands = uiStore.modalsById[IMPORT_CURL_MODAL_KEY].data?.curlCommands as Record<
string,
string
>;
const nodeId = ndvStore.activeNode?.id ?? '';
const command = curlCommands?.[nodeId];
curlCommand.value = command ?? '';
setTimeout(() => {
inputRef.value?.focus();
});
});
function onInput(value: string): void {
curlCommand.value = value;
}
function closeDialog(): void {
modalBus.emit('close');
}
function onImportSuccess() {
sendTelemetry();
closeDialog();
}
function onImportFailure(data: { invalidProtocol: boolean; protocol?: string }) {
sendTelemetry({ success: false, ...data });
}
function onAfterImport() {
const nodeId = ndvStore.activeNode?.id as string;
const curlCommands =
(uiStore.modalsById[IMPORT_CURL_MODAL_KEY].data?.curlCommands as Record<string, string>) ?? {};
curlCommands[nodeId] = curlCommand.value;
uiStore.setModalData({
name: IMPORT_CURL_MODAL_KEY,
data: { curlCommands },
});
}
function sendTelemetry(
data: { success: boolean; invalidProtocol: boolean; protocol?: string } = {
success: true,
invalidProtocol: false,
protocol: '',
},
): void {
telemetry.track('User imported curl command', {
success: data.success,
invalidProtocol: data.invalidProtocol,
protocol: data.protocol,
});
}
async function onImport() {
const { useImportCurlCommand } = await import('@/composables/useImportCurlCommand');
const { importCurlCommand } = useImportCurlCommand({
onImportSuccess,
onImportFailure,
onAfterImport,
});
importCurlCommand(curlCommand);
}
</script>
<template>
<Modal
width="700px"
:title="i18n.baseText('importCurlModal.title')"
:event-bus="modalBus"
:name="IMPORT_CURL_MODAL_KEY"
:center="true"
>
<template #content>
<div :class="$style.container">
<N8nInputLabel :label="i18n.baseText('importCurlModal.input.label')" color="text-dark">
<N8nInput
ref="inputRef"
:model-value="curlCommand"
type="textarea"
:rows="5"
data-test-id="import-curl-modal-input"
:placeholder="i18n.baseText('importCurlModal.input.placeholder')"
@update:model-value="onInput"
@focus="$event.target.select()"
/>
</N8nInputLabel>
</div>
</template>
<template #footer>
<div :class="$style.modalFooter">
<N8nNotice
:class="$style.notice"
:content="i18n.baseText('ImportCurlModal.notice.content')"
/>
<div>
<N8nButton
float="right"
:label="i18n.baseText('importCurlModal.button.label')"
data-test-id="import-curl-modal-button"
@click="onImport"
/>
</div>
</div>
</template>
</Modal>
</template>
<style module lang="scss">
.modalFooter {
justify-content: space-between;
display: flex;
flex-direction: row;
}
.notice {
margin: 0;
}
.container > * {
margin-bottom: var(--spacing-s);
&:last-child {
margin-bottom: 0;
}
}
</style>