Files
n8n-enterprise-unlocked/packages/editor-ui/src/views/SetupWorkflowFromTemplateView/SetupWorkflowFromTemplateView.vue
Tomi Turtiainen a3ca7c7acd fix: Disable continue button unless at least 1 cred has been filled (no-changelog) (#7959)
Disable the continue button in template credential setup unless at least
one credential has been (pre-)filled
2023-12-07 19:35:34 +02:00

188 lines
4.7 KiB
Vue

<script setup lang="ts">
import { computed, onMounted, watch } from 'vue';
import { useRoute, useRouter } from 'vue-router';
import { useSetupTemplateStore } from './setupTemplate.store';
import N8nHeading from 'n8n-design-system/components/N8nHeading';
import N8nLink from 'n8n-design-system/components/N8nLink';
import AppsRequiringCredsNotice from './AppsRequiringCredsNotice.vue';
import SetupTemplateFormStep from './SetupTemplateFormStep.vue';
import TemplatesView from '../TemplatesView.vue';
import { VIEWS } from '@/constants';
import { useI18n } from '@/composables/useI18n';
import { useTelemetry } from '@/composables/useTelemetry';
// Store
const setupTemplateStore = useSetupTemplateStore();
const i18n = useI18n();
const telemetry = useTelemetry();
// Router
const route = useRoute();
const router = useRouter();
//#region Computed
const templateId = computed(() =>
Array.isArray(route.params.id) ? route.params.id[0] : route.params.id,
);
const title = computed(() => setupTemplateStore.template?.name ?? 'unknown');
const isReady = computed(() => !setupTemplateStore.isLoading);
const skipSetupUrl = computed(() => {
const resolvedRoute = router.resolve({
name: VIEWS.TEMPLATE_IMPORT,
params: { id: templateId.value },
});
return resolvedRoute.fullPath;
});
//#endregion Computed
//#region Watchers
watch(templateId, async (newTemplateId) => {
setupTemplateStore.setTemplateId(newTemplateId);
await setupTemplateStore.loadTemplateIfNeeded();
});
//#endregion Watchers
//#region Methods
const onSkipSetup = async (event: MouseEvent) => {
event.preventDefault();
await setupTemplateStore.skipSetup({
router,
});
};
const skipIfTemplateHasNoCreds = async () => {
const isTemplateLoaded = !!setupTemplateStore.template;
if (!isTemplateLoaded) {
return false;
}
if (setupTemplateStore.credentialUsages.length === 0) {
await setupTemplateStore.skipSetup({
router,
});
return true;
}
return false;
};
//#endregion Methods
//#region Lifecycle hooks
setupTemplateStore.setTemplateId(templateId.value);
onMounted(async () => {
await setupTemplateStore.init();
const wasSkipped = await skipIfTemplateHasNoCreds();
if (!wasSkipped) {
telemetry.track('User opened cred setup', undefined, {
withPostHog: true,
});
}
});
//#endregion Lifecycle hooks
</script>
<template>
<TemplatesView :goBackEnabled="true">
<template #header>
<n8n-heading v-if="isReady" tag="h1" size="2xlarge"
>{{ i18n.baseText('templateSetup.title', { interpolate: { name: title } }) }}
</n8n-heading>
<n8n-loading v-else variant="h1" />
</template>
<template #content>
<div :class="$style.grid">
<div :class="$style.notice" data-test-id="info-callout">
<AppsRequiringCredsNotice v-if="isReady" />
<n8n-loading v-else variant="p" />
</div>
<div>
<ol v-if="isReady" :class="$style.appCredentialsContainer">
<SetupTemplateFormStep
:class="$style.appCredential"
:key="credentials.key"
v-for="(credentials, index) in setupTemplateStore.credentialUsages"
:order="index + 1"
:credentials="credentials"
/>
</ol>
<div v-else :class="$style.appCredentialsContainer">
<n8n-loading :class="$style.appCredential" variant="p" :rows="3" />
<n8n-loading :class="$style.appCredential" variant="p" :rows="3" />
</div>
</div>
<div :class="$style.actions">
<n8n-link :href="skipSetupUrl" :newWindow="false" @click="onSkipSetup($event)">{{
i18n.baseText('templateSetup.skip')
}}</n8n-link>
<n8n-tooltip
v-if="isReady"
:content="i18n.baseText('templateSetup.continue.button.fillRemaining')"
:disabled="setupTemplateStore.numFilledCredentials > 0"
>
<n8n-button
size="large"
:label="i18n.baseText('templateSetup.continue.button')"
:disabled="
setupTemplateStore.isSaving || setupTemplateStore.numFilledCredentials === 0
"
@click="setupTemplateStore.createWorkflow({ router })"
data-test-id="continue-button"
/>
</n8n-tooltip>
<div v-else>
<n8n-loading variant="button" />
</div>
</div>
</div>
</template>
</TemplatesView>
</template>
<style lang="scss" module>
.grid {
margin: 0 auto;
display: flex;
flex-direction: column;
justify-content: center;
max-width: 768px;
}
.notice {
margin-bottom: var(--spacing-2xl);
}
.appCredentialsContainer {
display: flex;
flex-direction: column;
gap: var(--spacing-2xl);
}
.appCredential:not(:last-of-type) {
padding-bottom: var(--spacing-2xl);
border-bottom: 1px solid var(--color-foreground-light);
}
.actions {
display: flex;
justify-content: space-between;
align-items: center;
margin-top: var(--spacing-3xl);
margin-bottom: var(--spacing-3xl);
}
</style>