This repository has been archived on 2024-07-22. You can view files and clone it, but cannot push or open issues or pull requests.
Zaimki/components/Account.vue

377 lines
15 KiB
Vue
Raw Normal View History

2020-10-15 11:29:56 -07:00
<template>
<section v-if="logoutInProgress">
<p class="text-center">
<Spinner size="5rem"/>
</p>
<div v-if="!impersonationActive">
<iframe v-for="domain in universalDomains"
:src="`${domain}/api/user/logout-universal`"
style="width: 1px; height: 1px; opacity: .01"
>
</iframe>
</div>
</section>
<section v-else>
2021-12-11 04:19:29 -08:00
<div v-if="showTermsUpdate" class="alert alert-info container my-4 small">
<h4 class="mb-3">
<Icon v="info-circle"/>
<T>terms.update.header</T>
</h4>
<p>
<T>terms.update.intro</T>
</p>
<ul>
<li v-for="change in $t('terms.update.changes')">
{{ change }}
</li>
</ul>
<p class="text-center">
<button class="btn btn-primary" @click.prevent="dismissTermsUpdate">
<Icon v="shield-check"/>
<T>confirm.ok</T>
</button>
</p>
</div>
2020-10-15 15:41:41 -07:00
<div class="card mb-3">
<div class="card-body d-flex flex-column flex-md-row">
2020-10-16 01:08:48 -07:00
<div class="mx-2 text-center">
<p class="mb-0">
2020-10-15 15:41:41 -07:00
<Avatar :user="$user()"/>
</p>
2020-12-24 04:00:37 -08:00
<div class="mb-2">
2021-01-08 06:14:53 -08:00
<div v-if="$user().avatarSource === 'gravatar'" class="mt-3">
2020-12-24 04:00:37 -08:00
<a href="https://gravatar.com" target="_blank" rel="noopener" class="small">
<Icon v="external-link"/>
<T>user.avatar.change</T>
2021-01-08 06:14:53 -08:00
Gravatar
</a>
</div>
<div v-else class="mt-3">
Gravatar:
<a href="#" @click.prevent="setAvatar('gravatar')">
<Avatar :user="$user()" :src="gravatar($user())" dsize="2rem"/>
2020-12-24 04:00:37 -08:00
</a>
</div>
<div v-if="$user().avatarSource">
<a href="#" @click.prevent="setAvatar(null)" class="small">
<Icon v="trash"/>
<T>crud.remove</T>
</a>
</div>
<ImageUploader small @uploaded="uploaded" sizes="avatar"/>
2020-12-24 04:00:37 -08:00
</div>
2020-12-30 15:03:30 -08:00
<p v-if="$isGranted('panel') || $isGranted('users')">
2021-04-01 09:24:47 -07:00
<nuxt-link to="/admin" class="badge bg-primary text-white"><T>user.account.admin</T></nuxt-link>
2020-10-16 01:08:48 -07:00
</p>
2020-10-15 15:41:41 -07:00
</div>
<div class="mx-2 flex-grow-1">
<Alert type="danger" :message="error"/>
2021-04-13 08:24:55 -07:00
<div v-if="message" class="alert alert-success">
2020-10-27 10:15:18 -07:00
<p class="mb-0 narrow-message">
2021-04-13 08:24:55 -07:00
<Icon :v="messageIcon"/>
2022-03-19 13:46:34 -07:00
<T :params="messageParams">{{message}}</T>
2020-10-27 10:15:18 -07:00
</p>
</div>
<form @submit.prevent="changeUsername" :disabled="savingUsername">
2020-10-15 15:41:41 -07:00
<h3 class="h6"><T>user.account.changeUsername.header</T></h3>
<input type="text" class="form-control" v-model="username"
required minlength="4" maxlength="16"/>
<div class="d-none d-md-block mt-3">
<button class="btn btn-outline-primary" :disabled="username === user.username">
2021-01-22 14:54:24 -08:00
<T>user.account.changeUsername.action</T>
</button>
2020-10-15 15:41:41 -07:00
</div>
<div class="d-block d-md-none mt-3">
<button class="btn btn-outline-primary w-100" :disabled="username === user.username">
<T>user.account.changeUsername.action</T>
</button>
</div>
2020-10-15 15:41:41 -07:00
</form>
<hr/>
<form @submit.prevent="changeEmail" :disabled="savingEmail">
2020-10-15 15:41:41 -07:00
<h3 class="h6"><T>user.account.changeEmail.header</T></h3>
<div v-if="!changeEmailAuthId" class="">
<input type="email" class="form-control mb-3" v-model="email" required/>
2021-08-07 04:47:03 -07:00
<div class="d-flex flex-column flex-md-row">
<Captcha v-if="showCaptcha" v-model="captchaToken"/>
<div :class="['d-none', 'd-md-block', showCaptcha ? 'ms-3' : '']">
<button class="btn btn-outline-primary" :disabled="!canChangeEmail">
<T>user.account.changeEmail.action</T>
</button>
</div>
2021-08-07 04:47:03 -07:00
<div class="d-block d-md-none mt-3">
<button class="btn btn-outline-primary w-100" :disabled="!canChangeEmail">
<T>user.account.changeEmail.action</T>
</button>
</div>
</div>
2020-10-27 10:15:18 -07:00
</div>
<div v-else class="input-group mb-3">
<input type="text" class="form-control text-center" v-model="code"
placeholder="000000" autofocus required minlength="0" maxlength="6"
inputmode="numeric" pattern="[0-9]{6}" autocomplete="one-time-code"
ref="code"
/>
2021-01-22 14:54:24 -08:00
<button class="btn btn-outline-primary">
<Icon v="key"/>
<T>user.code.action</T>
</button>
2020-10-27 10:15:18 -07:00
</div>
</form>
2020-10-15 11:29:56 -07:00
</div>
</div>
</div>
<Loading :value="profiles">
2020-11-02 10:31:05 -08:00
<template v-slot:header>
2021-12-02 08:18:25 -08:00
<h3 class="h4"><T>profile.list</T><T>quotation.colon</T></h3>
2020-11-02 10:31:05 -08:00
</template>
<ul v-if="profiles !== undefined" class="list-group">
<li v-for="(options, locale) in locales" :key="locale" :class="['list-group-item', locale === config.locale ? 'profile-current' : '']">
2021-07-24 10:18:39 -07:00
<ProfileOverview :username="username" :profile="profiles[locale]" :locale="locale" @update="setProfiles"/>
2020-11-02 10:31:05 -08:00
</li>
</ul>
</Loading>
<Loading :value="socialConnections">
2020-11-02 10:31:05 -08:00
<template v-slot:header>
2021-12-02 08:18:25 -08:00
<h3 class="h4"><T>user.socialConnection.list</T><T>quotation.colon</T></h3>
2020-11-02 10:31:05 -08:00
</template>
<ul v-if="socialConnections !== undefined" class="list-group">
2020-11-02 12:25:48 -08:00
<li v-for="(providerOptions, provider) in socialProviders" :key="provider" :class="['list-group-item', socialConnections[provider] !== undefined ? 'profile-current' : '']">
<SocialConnection :provider="provider" :providerOptions="providerOptions" :connection="socialConnections[provider]"
@disconnected="socialConnections[provider] = undefined" @setAvatar="setAvatar"/>
</li>
2021-12-18 10:54:36 -08:00
<li :class="['list-group-item', $user().mfa ? 'profile-current' : '']">
<MfaConnection/>
</li>
</ul>
2020-11-02 10:31:05 -08:00
</Loading>
2020-10-27 08:33:45 -07:00
<section>
2021-01-22 14:54:24 -08:00
<a href="#" class="badge bg-light text-dark border" @click.prevent="logout">
2020-10-15 15:41:41 -07:00
<Icon v="sign-out"/>
<T>user.logout</T>
2020-10-27 08:33:45 -07:00
</a>
2021-01-22 14:54:24 -08:00
<a href="#" class="badge bg-light text-dark border" @click.prevent="deleteAccount">
2020-10-27 08:33:45 -07:00
<Icon v="trash-alt"/>
<T>user.deleteAccount</T>
</a>
2021-11-28 03:19:37 -08:00
<a v-if="impersonationActive" href="#" class="badge bg-light text-dark border border-primary" @click.prevent="stopImpersonation">
<Icon v="user-secret"/>
Stop impersonation
</a>
2020-10-27 08:33:45 -07:00
</section>
<div>
<iframe v-for="domain in universalDomains"
:src="`${domain}/api/user/init-universal/${$cookies.get('token')}`"
style="width: 1px; height: 1px; opacity: .01"
>
</iframe>
</div>
2020-10-15 11:29:56 -07:00
</section>
</template>
<script>
2021-12-14 06:15:27 -08:00
import {socialProviders} from "../src/socialProviders";
2020-12-24 04:00:37 -08:00
import {gravatar} from "../src/helpers";
import cookieSettings from "../src/cookieSettings";
import {mapState} from "vuex";
2020-11-02 10:31:05 -08:00
2020-10-15 11:29:56 -07:00
export default {
data() {
return {
username: this.$user().username,
email: this.$user().email,
2021-04-13 08:24:55 -07:00
message: '',
2022-03-19 13:46:34 -07:00
messageParams: {},
2021-04-13 08:24:55 -07:00
messageIcon: null,
2020-10-15 11:29:56 -07:00
error: '',
2020-10-27 10:15:18 -07:00
changeEmailAuthId: null,
code: '',
profiles: undefined,
2020-11-02 10:31:05 -08:00
socialProviders,
socialConnections: undefined,
2020-12-24 04:00:37 -08:00
savingUsername: false,
savingEmail: false,
2021-04-13 08:24:55 -07:00
2020-12-24 04:00:37 -08:00
gravatar,
showCaptcha: false,
captchaToken: null,
universalDomains: process.env.ALL_LOCALES_URLS.split(',').filter(x => x !== process.env.BASE_URL),
logoutInProgress: false,
2021-11-28 03:19:37 -08:00
2021-12-11 04:19:29 -08:00
impersonationActive: !!this.$cookies.get('impersonator'),
2022-04-18 02:23:56 -07:00
showTermsUpdate: false,
// this.$ulidTime(this.$user().id) < new Date(2021, 11, 13) / 1000
// && !this.$cookies.get('termsUpdateDismissed')
// && (!this.$user().lastActive || this.$user().lastActive < +new Date(2021, 11, 18, 0, 0, 0))
}
},
async mounted() {
this.profiles = (await this.$axios.$get(`/profile/get/${this.$user().username}`)).profiles;
2020-11-02 10:31:05 -08:00
this.socialConnections = await this.$axios.$get(`/user/social-connections`);
if (process.client) {
const redirectTo = window.sessionStorage.getItem('after-login');
if (this.$user() && redirectTo) {
window.sessionStorage.removeItem('after-login')
await this.$router.push(redirectTo);
}
}
2020-10-15 11:29:56 -07:00
},
methods: {
async changeUsername() {
this.error = '';
if (this.savingUsername) { return; }
this.savingUsername = true;
try {
const response = await this.$post(`/user/change-username`, {
username: this.username,
});
2020-10-15 11:29:56 -07:00
if (response.error) {
this.error = response.error;
return;
}
this.$store.commit('setToken', response.token);
this.username = this.$user().username;
this.$cookies.set('token', this.$store.state.token, cookieSettings);
this.message = 'crud.saved';
2022-03-19 13:46:34 -07:00
this.messageParams = {};
this.messageIcon = 'check-circle';
setTimeout(() => this.message = '', 3000);
} finally {
this.savingUsername = false;
}
2020-10-15 11:29:56 -07:00
},
2020-10-27 10:15:18 -07:00
async changeEmail() {
this.error = '';
if (this.savingEmail) { return; }
this.savingEmail = true;
try {
const response = await this.$post(`/user/change-email`, {
email: this.email,
authId: this.changeEmailAuthId,
code: this.code,
captchaToken: this.captchaToken,
2020-10-27 10:15:18 -07:00
});
if (response.error) {
this.error = response.error;
return;
}
if (!this.changeEmailAuthId) {
this.changeEmailAuthId = response.authId;
this.message = 'user.login.emailSent';
2022-03-19 13:46:34 -07:00
this.messageParams = {'email': this.addBrackets(this.email)};
this.messageIcon = 'envelope-open-text';
this.$nextTick(_ => {
this.$refs.code.focus();
});
} else {
this.changeEmailAuthId = null;
this.message = '';
2022-03-19 13:46:34 -07:00
this.messageParams = {};
this.code = null;
this.$store.commit('setToken', response.token);
this.$cookies.set('token', this.$store.state.token, cookieSettings);
this.message = 'crud.saved';
2022-03-19 13:46:34 -07:00
this.messageParams = {};
this.messageIcon = 'check-circle';
setTimeout(() => this.message = '', 3000);
}
} finally {
this.savingEmail = false;
2020-10-27 10:15:18 -07:00
}
},
2020-10-15 11:29:56 -07:00
logout() {
this.logoutInProgress = true;
setTimeout(this.doLogout, 3000);
},
doLogout() {
2020-10-15 11:29:56 -07:00
this.$store.commit('setToken', null);
this.$cookies.removeAll();
this.logoutInProgress = false;
2020-10-15 15:41:41 -07:00
},
2020-10-24 13:32:12 -07:00
setProfiles(profiles) {
this.profiles = profiles;
},
2020-10-27 08:33:45 -07:00
async deleteAccount() {
await this.$confirm(this.$t('user.deleteAccountConfirm'), 'danger');
const response = await this.$post(`/user/delete`);
2020-10-27 08:33:45 -07:00
this.logout();
},
2020-11-02 12:12:15 -08:00
async setAvatar(source) {
const response = await this.$post(`/user/set-avatar`, {source});
2020-11-02 12:12:15 -08:00
this.$store.commit('setToken', response.token);
this.$cookies.set('token', this.$store.state.token, cookieSettings);
2020-11-02 12:12:15 -08:00
},
2021-01-08 06:14:53 -08:00
async uploaded(ids) {
2022-04-01 07:22:53 -07:00
await this.setAvatar(`${process.env.CLOUDFRONT}/images/${ids[0]}-avatar.png`);
2021-01-08 06:14:53 -08:00
},
2021-11-28 03:19:37 -08:00
async stopImpersonation() {
this.$cookies.set('token', this.$cookies.get('impersonator'));
this.$cookies.remove('impersonator');
window.location.reload();
},
2021-12-11 04:19:29 -08:00
dismissTermsUpdate() {
this.$cookies.set('termsUpdateDismissed', true);
this.showTermsUpdate = false;
},
2022-03-19 13:46:34 -07:00
addBrackets(str) {
return str ? `(${str})` : '';
},
2020-10-15 11:29:56 -07:00
},
computed: {
...mapState([
'user',
]),
canChangeEmail() {
return this.email && this.captchaToken;
}
},
watch: {
email(v) {
if (v !== this.$user().email) {
this.showCaptcha = true;
}
}
}
2020-10-15 11:29:56 -07:00
}
</script>
<style lang="scss" scoped>
@import "assets/variables";
.profile-current {
2021-03-01 13:35:03 -08:00
border-inline-start: 3px solid $primary;
}
2020-10-27 10:15:18 -07:00
.narrow-message {
max-width: 56ch;
}
</style>