mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-09 20:07:31 -08:00
411 lines
10 KiB
Vue
411 lines
10 KiB
Vue
<script lang="ts" setup>
|
|
import { ref, computed, onMounted, onBeforeUnmount } from 'vue';
|
|
import { useI18n } from '@/composables/useI18n';
|
|
import { useToast } from '@/composables/useToast';
|
|
import type { IFormInputs, IUser, ThemeOption } from '@/Interface';
|
|
import {
|
|
CHANGE_PASSWORD_MODAL_KEY,
|
|
MFA_DOCS_URL,
|
|
MFA_SETUP_MODAL_KEY,
|
|
PROMPT_MFA_CODE_MODAL_KEY,
|
|
} from '@/constants';
|
|
import { useUIStore } from '@/stores/ui.store';
|
|
import { useUsersStore } from '@/stores/users.store';
|
|
import { useSettingsStore } from '@/stores/settings.store';
|
|
import { createFormEventBus } from 'n8n-design-system/utils';
|
|
import type { MfaModalEvents } from '@/event-bus/mfa';
|
|
import { promptMfaCodeBus } from '@/event-bus/mfa';
|
|
|
|
type UserBasicDetailsForm = {
|
|
firstName: string;
|
|
lastName: string;
|
|
email: string;
|
|
};
|
|
|
|
type UserBasicDetailsWithMfa = UserBasicDetailsForm & {
|
|
mfaCode?: string;
|
|
};
|
|
|
|
const i18n = useI18n();
|
|
const { showToast, showError } = useToast();
|
|
|
|
const hasAnyBasicInfoChanges = ref<boolean>(false);
|
|
const formInputs = ref<null | IFormInputs>(null);
|
|
const formBus = createFormEventBus();
|
|
const readyToSubmit = ref(false);
|
|
const currentSelectedTheme = ref(useUIStore().theme);
|
|
const themeOptions = ref<Array<{ name: ThemeOption; label: string }>>([
|
|
{
|
|
name: 'system',
|
|
label: 'settings.personal.theme.systemDefault',
|
|
},
|
|
{
|
|
name: 'light',
|
|
label: 'settings.personal.theme.light',
|
|
},
|
|
{
|
|
name: 'dark',
|
|
label: 'settings.personal.theme.dark',
|
|
},
|
|
]);
|
|
|
|
const uiStore = useUIStore();
|
|
const usersStore = useUsersStore();
|
|
const settingsStore = useSettingsStore();
|
|
|
|
const currentUser = computed((): IUser | null => {
|
|
return usersStore.currentUser;
|
|
});
|
|
const isExternalAuthEnabled = computed((): boolean => {
|
|
const isLdapEnabled =
|
|
settingsStore.settings.enterprise.ldap && currentUser.value?.signInType === 'ldap';
|
|
const isSamlEnabled =
|
|
settingsStore.isSamlLoginEnabled && settingsStore.isDefaultAuthenticationSaml;
|
|
return isLdapEnabled || isSamlEnabled;
|
|
});
|
|
const isPersonalSecurityEnabled = computed((): boolean => {
|
|
return usersStore.isInstanceOwner || !isExternalAuthEnabled.value;
|
|
});
|
|
const mfaDisabled = computed((): boolean => {
|
|
return !usersStore.mfaEnabled;
|
|
});
|
|
const isMfaFeatureEnabled = computed((): boolean => {
|
|
return settingsStore.isMfaFeatureEnabled;
|
|
});
|
|
const hasAnyPersonalisationChanges = computed((): boolean => {
|
|
return currentSelectedTheme.value !== uiStore.theme;
|
|
});
|
|
const hasAnyChanges = computed(() => {
|
|
return hasAnyBasicInfoChanges.value || hasAnyPersonalisationChanges.value;
|
|
});
|
|
|
|
onMounted(() => {
|
|
formInputs.value = [
|
|
{
|
|
name: 'firstName',
|
|
initialValue: currentUser.value?.firstName,
|
|
properties: {
|
|
label: i18n.baseText('auth.firstName'),
|
|
maxlength: 32,
|
|
required: true,
|
|
autocomplete: 'given-name',
|
|
capitalize: true,
|
|
disabled: isExternalAuthEnabled.value,
|
|
},
|
|
},
|
|
{
|
|
name: 'lastName',
|
|
initialValue: currentUser.value?.lastName,
|
|
properties: {
|
|
label: i18n.baseText('auth.lastName'),
|
|
maxlength: 32,
|
|
required: true,
|
|
autocomplete: 'family-name',
|
|
capitalize: true,
|
|
disabled: isExternalAuthEnabled.value,
|
|
},
|
|
},
|
|
{
|
|
name: 'email',
|
|
initialValue: currentUser.value?.email,
|
|
properties: {
|
|
label: i18n.baseText('auth.email'),
|
|
type: 'email',
|
|
required: true,
|
|
validationRules: [{ name: 'VALID_EMAIL' }],
|
|
autocomplete: 'email',
|
|
capitalize: true,
|
|
disabled: !isPersonalSecurityEnabled.value,
|
|
},
|
|
},
|
|
];
|
|
});
|
|
|
|
function onInput() {
|
|
hasAnyBasicInfoChanges.value = true;
|
|
}
|
|
|
|
function onReadyToSubmit(ready: boolean) {
|
|
readyToSubmit.value = ready;
|
|
}
|
|
|
|
/** Saves users basic info and personalization settings */
|
|
async function saveUserSettings(params: UserBasicDetailsWithMfa) {
|
|
try {
|
|
// The MFA code might be invalid so we update the user's basic info first
|
|
await updateUserBasicInfo(params);
|
|
await updatePersonalisationSettings();
|
|
|
|
showToast({
|
|
title: i18n.baseText('settings.personal.personalSettingsUpdated'),
|
|
message: '',
|
|
type: 'success',
|
|
});
|
|
} catch (e) {
|
|
showError(e, i18n.baseText('settings.personal.personalSettingsUpdatedError'));
|
|
}
|
|
}
|
|
|
|
async function onSubmit(form: UserBasicDetailsForm) {
|
|
if (!usersStore.currentUser?.mfaEnabled) {
|
|
await saveUserSettings(form);
|
|
return;
|
|
}
|
|
|
|
uiStore.openModal(PROMPT_MFA_CODE_MODAL_KEY);
|
|
|
|
promptMfaCodeBus.once('closed', async (payload: MfaModalEvents['closed']) => {
|
|
if (!payload) {
|
|
// User closed the modal without submitting the form
|
|
return;
|
|
}
|
|
|
|
await saveUserSettings({
|
|
...form,
|
|
mfaCode: payload.mfaCode,
|
|
});
|
|
});
|
|
}
|
|
|
|
async function updateUserBasicInfo(userBasicInfo: UserBasicDetailsWithMfa) {
|
|
if (!hasAnyBasicInfoChanges.value || !usersStore.currentUserId) {
|
|
return;
|
|
}
|
|
|
|
await usersStore.updateUser({
|
|
firstName: userBasicInfo.firstName,
|
|
lastName: userBasicInfo.lastName,
|
|
email: userBasicInfo.email,
|
|
mfaCode: userBasicInfo.mfaCode,
|
|
});
|
|
hasAnyBasicInfoChanges.value = false;
|
|
}
|
|
|
|
async function updatePersonalisationSettings() {
|
|
if (!hasAnyPersonalisationChanges.value) {
|
|
return;
|
|
}
|
|
|
|
uiStore.setTheme(currentSelectedTheme.value);
|
|
}
|
|
|
|
function onSaveClick() {
|
|
formBus.emit('submit');
|
|
}
|
|
|
|
function openPasswordModal() {
|
|
uiStore.openModal(CHANGE_PASSWORD_MODAL_KEY);
|
|
}
|
|
|
|
function onMfaEnableClick() {
|
|
uiStore.openModal(MFA_SETUP_MODAL_KEY);
|
|
}
|
|
|
|
async function disableMfa(payload: MfaModalEvents['closed']) {
|
|
if (!payload) {
|
|
// User closed the modal without submitting the form
|
|
return;
|
|
}
|
|
|
|
try {
|
|
await usersStore.disableMfa(payload.mfaCode);
|
|
|
|
showToast({
|
|
title: i18n.baseText('settings.personal.mfa.toast.disabledMfa.title'),
|
|
message: i18n.baseText('settings.personal.mfa.toast.disabledMfa.message'),
|
|
type: 'success',
|
|
duration: 0,
|
|
});
|
|
} catch (e) {
|
|
showError(e, i18n.baseText('settings.personal.mfa.toast.disabledMfa.error.message'));
|
|
}
|
|
}
|
|
|
|
async function onMfaDisableClick() {
|
|
uiStore.openModal(PROMPT_MFA_CODE_MODAL_KEY);
|
|
|
|
promptMfaCodeBus.once('closed', disableMfa);
|
|
}
|
|
|
|
onBeforeUnmount(() => {
|
|
promptMfaCodeBus.off('closed', disableMfa);
|
|
});
|
|
</script>
|
|
|
|
<template>
|
|
<div :class="$style.container" data-test-id="personal-settings-container">
|
|
<div :class="$style.header">
|
|
<n8n-heading size="2xlarge">{{
|
|
i18n.baseText('settings.personal.personalSettings')
|
|
}}</n8n-heading>
|
|
<div v-if="currentUser" :class="$style.user">
|
|
<span :class="$style.username" data-test-id="current-user-name">
|
|
<n8n-text color="text-light">{{ currentUser.fullName }}</n8n-text>
|
|
</span>
|
|
<n8n-avatar
|
|
:first-name="currentUser.firstName"
|
|
:last-name="currentUser.lastName"
|
|
size="large"
|
|
/>
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<div class="mb-s">
|
|
<n8n-heading size="large">{{
|
|
i18n.baseText('settings.personal.basicInformation')
|
|
}}</n8n-heading>
|
|
</div>
|
|
<div data-test-id="personal-data-form">
|
|
<n8n-form-inputs
|
|
v-if="formInputs"
|
|
:inputs="formInputs"
|
|
:event-bus="formBus"
|
|
@update="onInput"
|
|
@ready="onReadyToSubmit"
|
|
@submit="onSubmit"
|
|
/>
|
|
</div>
|
|
</div>
|
|
<div v-if="isPersonalSecurityEnabled">
|
|
<div class="mb-s">
|
|
<n8n-heading size="large">{{ i18n.baseText('settings.personal.security') }}</n8n-heading>
|
|
</div>
|
|
<div class="mb-s">
|
|
<n8n-input-label :label="i18n.baseText('auth.password')">
|
|
<n8n-link data-test-id="change-password-link" @click="openPasswordModal">{{
|
|
i18n.baseText('auth.changePassword')
|
|
}}</n8n-link>
|
|
</n8n-input-label>
|
|
</div>
|
|
<div v-if="isMfaFeatureEnabled" data-test-id="mfa-section">
|
|
<div class="mb-xs">
|
|
<n8n-input-label :label="i18n.baseText('settings.personal.mfa.section.title')" />
|
|
<n8n-text :bold="false" :class="$style.infoText">
|
|
{{
|
|
mfaDisabled
|
|
? i18n.baseText('settings.personal.mfa.button.disabled.infobox')
|
|
: i18n.baseText('settings.personal.mfa.button.enabled.infobox')
|
|
}}
|
|
<n8n-link :to="MFA_DOCS_URL" size="small" :bold="true">
|
|
{{ i18n.baseText('generic.learnMore') }}
|
|
</n8n-link>
|
|
</n8n-text>
|
|
</div>
|
|
<n8n-button
|
|
v-if="mfaDisabled"
|
|
:class="$style.button"
|
|
type="tertiary"
|
|
:label="i18n.baseText('settings.personal.mfa.button.enabled')"
|
|
data-test-id="enable-mfa-button"
|
|
@click="onMfaEnableClick"
|
|
/>
|
|
<n8n-button
|
|
v-else
|
|
:class="$style.disableMfaButton"
|
|
type="tertiary"
|
|
:label="i18n.baseText('settings.personal.mfa.button.disabled')"
|
|
data-test-id="disable-mfa-button"
|
|
@click="onMfaDisableClick"
|
|
/>
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<div class="mb-s">
|
|
<n8n-heading size="large">{{
|
|
i18n.baseText('settings.personal.personalisation')
|
|
}}</n8n-heading>
|
|
</div>
|
|
<div>
|
|
<n8n-input-label :label="i18n.baseText('settings.personal.theme')">
|
|
<n8n-select
|
|
v-model="currentSelectedTheme"
|
|
:class="$style.themeSelect"
|
|
data-test-id="theme-select"
|
|
size="small"
|
|
filterable
|
|
>
|
|
<n8n-option
|
|
v-for="item in themeOptions"
|
|
:key="item.name"
|
|
:label="$t(item.label)"
|
|
:value="item.name"
|
|
>
|
|
</n8n-option>
|
|
</n8n-select>
|
|
</n8n-input-label>
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<n8n-button
|
|
float="right"
|
|
:label="i18n.baseText('settings.personal.save')"
|
|
size="large"
|
|
:disabled="!hasAnyChanges || !readyToSubmit"
|
|
data-test-id="save-settings-button"
|
|
@click="onSaveClick"
|
|
/>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<style lang="scss" module>
|
|
.container {
|
|
> * {
|
|
margin-bottom: var(--spacing-2xl);
|
|
}
|
|
padding-bottom: 100px;
|
|
}
|
|
|
|
.header {
|
|
display: flex;
|
|
align-items: center;
|
|
white-space: nowrap;
|
|
*:first-child {
|
|
flex-grow: 1;
|
|
}
|
|
}
|
|
|
|
.user {
|
|
display: flex;
|
|
align-items: center;
|
|
|
|
@media (max-width: $breakpoint-2xs) {
|
|
display: none;
|
|
}
|
|
}
|
|
|
|
.username {
|
|
margin-right: var(--spacing-s);
|
|
text-align: right;
|
|
|
|
@media (max-width: $breakpoint-sm) {
|
|
max-width: 100px;
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
}
|
|
}
|
|
|
|
.disableMfaButton {
|
|
--button-color: var(--color-danger);
|
|
> span {
|
|
font-weight: var(--font-weight-bold);
|
|
}
|
|
}
|
|
|
|
.button {
|
|
font-size: var(--spacing-xs);
|
|
> span {
|
|
font-weight: var(--font-weight-bold);
|
|
}
|
|
}
|
|
|
|
.infoText {
|
|
font-size: var(--font-size-2xs);
|
|
color: var(--color-text-light);
|
|
}
|
|
|
|
.themeSelect {
|
|
max-width: 50%;
|
|
}
|
|
</style>
|