Translate user security page

pull/436/head
Lorenzo 2025-06-26 00:16:30 +02:00
parent 34ec6210c6
commit dfddfc3e06
4 changed files with 222 additions and 171 deletions

View File

@ -9,96 +9,103 @@ import { useSettingStore } from "/@/store/settings";
import PageFooter from "./components/footer/index.vue"; import PageFooter from "./components/footer/index.vue";
import { useRouter } from "vue-router"; import { useRouter } from "vue-router";
import MaxKBChat from "/@/components/ai/index.vue"; import MaxKBChat from "/@/components/ai/index.vue";
import { useI18n } from "vue-i18n";
const { t } = useI18n();
const userStore = useUserStore(); const userStore = useUserStore();
const router = useRouter(); const router = useRouter();
const menus = computed(() => [ const menus = computed(() => [
{ {
handler: () => { handler: () => {
router.push("/certd/mine/user-profile"); router.push("/certd/mine/user-profile");
}, },
icon: "fa-solid:book", icon: "fa-solid:book",
text: "账号信息", text: t("certd.accountInfo"),
}, },
{ {
handler: () => { handler: () => {
router.push("/certd/mine/security"); router.push("/certd/mine/security");
}, },
icon: "fluent:shield-keyhole-16-regular", icon: "fluent:shield-keyhole-16-regular",
text: "认证安全设置", text: t("certd.securitySettings"),
}, },
]); ]);
const avatar = computed(() => { const avatar = computed(() => {
const avt = userStore.getUserInfo?.avatar; const avt = userStore.getUserInfo?.avatar;
return avt ? `/api/basic/file/download?key=${avt}` : ""; return avt ? `/api/basic/file/download?key=${avt}` : "";
}); });
async function handleLogout() { async function handleLogout() {
await userStore.logout(true); await userStore.logout(true);
} }
const settingStore = useSettingStore(); const settingStore = useSettingStore();
const sysPublic = computed(() => { const sysPublic = computed(() => {
return settingStore.sysPublic; return settingStore.sysPublic;
}); });
const siteInfo = computed(() => { const siteInfo = computed(() => {
return settingStore.siteInfo; return settingStore.siteInfo;
}); });
onErrorCaptured(e => { onErrorCaptured(e => {
console.error("ErrorCaptured:", e); console.error("ErrorCaptured:", e);
// notification.error({ message: e.message }); // notification.error({ message: e.message });
// //
return false; return false;
}); });
onMounted(async () => { onMounted(async () => {
await settingStore.checkUrlBound(); await settingStore.checkUrlBound();
}); });
function goGithub() { function goGithub() {
window.open("https://github.com/certd/certd"); window.open("https://github.com/certd/certd");
} }
const settingsStore = useSettingStore(); const settingsStore = useSettingStore();
const chatBox = ref(); const chatBox = ref();
const openChat = (q: string) => { const openChat = (q: string) => {
chatBox.value.openChat({ q }); chatBox.value.openChat({ q });
}; };
provide("fn:ai.open", openChat); provide("fn:ai.open", openChat);
</script> </script>
<template> <template>
<BasicLayout @clear-preferences-and-logout="handleLogout"> <BasicLayout @clear-preferences-and-logout="handleLogout">
<template #user-dropdown> <template #user-dropdown>
<UserDropdown :avatar="avatar" :menus="menus" :text="userStore.userInfo?.nickName || userStore.userInfo?.username" description="" tag-text="" @logout="handleLogout" /> <UserDropdown :avatar="avatar" :menus="menus"
</template> :text="userStore.userInfo?.nickName || userStore.userInfo?.username" description="" tag-text=""
<template #lock-screen> @logout="handleLogout" />
<LockScreen :avatar @to-login="handleLogout" /> </template>
</template> <template #lock-screen>
<template #header-right-0> <LockScreen :avatar @to-login="handleLogout" />
<div v-if="!settingStore.isComm" class="hover:bg-accent ml-1 mr-2 cursor-pointer rounded-full hidden md:block"> </template>
<tutorial-button class="flex-center header-btn" /> <template #header-right-0>
</div> <div v-if="!settingStore.isComm"
<div class="hover:bg-accent ml-1 mr-2 cursor-pointer rounded-full"> class="hover:bg-accent ml-1 mr-2 cursor-pointer rounded-full hidden md:block">
<vip-button class="flex-center header-btn" mode="nav" /> <tutorial-button class="flex-center header-btn" />
</div> </div>
<div v-if="!settingStore.isComm" class="hover:bg-accent ml-1 mr-2 cursor-pointer rounded-full"> <div class="hover:bg-accent ml-1 mr-2 cursor-pointer rounded-full">
<fs-button shape="circle" type="text" icon="ion:logo-github" :text="null" @click="goGithub" /> <vip-button class="flex-center header-btn" mode="nav" />
</div> </div>
</template> <div v-if="!settingStore.isComm" class="hover:bg-accent ml-1 mr-2 cursor-pointer rounded-full">
<template #footer> <fs-button shape="circle" type="text" icon="ion:logo-github" :text="null" @click="goGithub" />
<PageFooter></PageFooter> </div>
<MaxKBChat v-if="settingsStore.sysPublic.aiChatEnabled !== false" ref="chatBox" /> </template>
</template> <template #footer>
</BasicLayout> <PageFooter></PageFooter>
<MaxKBChat v-if="settingsStore.sysPublic.aiChatEnabled !== false" ref="chatBox" />
</template>
</BasicLayout>
</template> </template>
<style lang="less"> <style lang="less">
.header-btn { .header-btn {
font-size: 14px; font-size: 14px;
padding: 5px; padding: 5px;
} }
</style> </style>

View File

@ -274,4 +274,19 @@ export default {
required: "Please enter domains to import", required: "Please enter domains to import",
placeholder: "www.baidu.com:443:Baidu\nwww.taobao.com::Taobao\nwww.google.com\n", placeholder: "www.baidu.com:443:Baidu\nwww.taobao.com::Taobao\nwww.google.com\n",
}, },
accountInfo: "Account Information",
securitySettings: "Security & Settings",
confirmDisable2FA: "Are you sure you want to disable two-factor authentication login?",
disabledSuccess: "Disabled successfully",
saveSuccess: "Saved successfully",
twoFactorAuth: "2FA Two-Factor Authentication Login",
rebind: "Rebind",
twoFactorAuthHelper: "Enable or disable two-factor authentication login",
bindDevice: "Bind Device",
step1: "1. Install any authenticator app, for example:",
tooltipGoogleServiceError: "If you get a Google service not found error, you can install KK Google Assistant",
step2: "2. Scan the QR code to add the account",
step3: "3. Enter the verification code",
inputVerifyCode: "Please enter the verification code",
cancel: "Cancel",
}; };

View File

@ -280,4 +280,19 @@ export default {
required: "请输入要导入的域名", required: "请输入要导入的域名",
placeholder: "www.baidu.com:443:百度\nwww.taobao.com::淘宝\nwww.google.com\n", placeholder: "www.baidu.com:443:百度\nwww.taobao.com::淘宝\nwww.google.com\n",
}, },
accountInfo: "账号信息",
securitySettings: "认证安全设置",
confirmDisable2FA: "确定要关闭多重验证登录吗?",
disabledSuccess: "关闭成功",
saveSuccess: "保存成功",
twoFactorAuth: "2FA多重验证登录",
rebind: "重新绑定",
twoFactorAuthHelper: "是否开启多重验证登录",
bindDevice: "绑定设备",
step1: "1. 安装任意一款支持Authenticator的验证APP比如",
tooltipGoogleServiceError: "如果报没有找到谷歌服务的错误您可以安装KK谷歌助手",
step2: "2. 扫描二维码添加账号",
step3: "3. 输入验证码",
inputVerifyCode: "请输入验证码",
cancel: "取消",
}; };

View File

@ -1,73 +1,82 @@
<template> <template>
<fs-page class="page-user-settings page-two-factor"> <fs-page class="page-user-settings page-two-factor">
<template #header> <template #header>
<div class="title">认证安全设置</div> <div class="title">{{ t("certd.securitySettings") }}</div>
</template> </template>
<div class="user-settings-form settings-form">
<a-form :model="formState" name="basic" :label-col="{ span: 8 }" :wrapper-col="{ span: 16 }" autocomplete="off">
<a-form-item label="2FA多重验证登录" :name="['authenticator', 'enabled']">
<div class="flex mt-5">
<a-switch v-model:checked="formState.authenticator.enabled" :disabled="!settingsStore.isPlus" @change="onAuthenticatorEnabledChanged" />
<a-button <div class="user-settings-form settings-form">
v-if="formState.authenticator.enabled && formState.authenticator.verified" <a-form :model="formState" name="basic" :label-col="{ span: 8 }" :wrapper-col="{ span: 16 }"
:disabled="authenticatorOpenRef || !settingsStore.isPlus" autocomplete="off">
size="small" <a-form-item :label="t('certd.twoFactorAuth')" :name="['authenticator', 'enabled']">
class="ml-5" <div class="flex mt-5">
type="primary" <a-switch v-model:checked="formState.authenticator.enabled" :disabled="!settingsStore.isPlus"
@click="authenticatorForm.open = true" @change="onAuthenticatorEnabledChanged" />
>
重新绑定
</a-button>
<vip-button class="ml-5" mode="button"></vip-button> <a-button v-if="formState.authenticator.enabled && formState.authenticator.verified"
</div> :disabled="authenticatorOpenRef || !settingsStore.isPlus" size="small" class="ml-5"
type="primary" @click="authenticatorForm.open = true">
{{ t('certd.rebind') }}
</a-button>
<div class="helper">是否开启多重验证登录</div> <vip-button class="ml-5" mode="button"></vip-button>
</a-form-item> </div>
<a-form-item v-if="authenticatorOpenRef" label="绑定设备" class="authenticator-config">
<h3 class="font-bold m-5">1. 安装任意一款支持Authenticator的验证APP比如</h3> <div class="helper">{{ t('certd.twoFactorAuthHelper') }}</div>
<div class="ml-20"> </a-form-item>
<ul>
<li> <a-form-item v-if="authenticatorOpenRef" :label="t('certd.bindDevice')" class="authenticator-config">
<a-tooltip title="如果报没有找到谷歌服务的错误您可以安装KK谷歌助手"> <h3 class="font-bold m-5">{{ t('certd.step1') }}</h3>
<a href="https://appgallery.huawei.com/app/C100262999" target="_blank"> Microsoft Authenticator</a> <div class="ml-20">
</a-tooltip> <ul>
</li> <li>
<li> <a-tooltip :title="t('certd.tooltipGoogleServiceError')">
<a href="https://sj.qq.com/appdetail/com.tencent.authenticator" target="_blank">腾讯身份验证器</a> <a href="https://appgallery.huawei.com/app/C100262999" target="_blank">Microsoft
</li> Authenticator</a>
<li> </a-tooltip>
<a href="https://www.synology.cn/zh-cn/dsm/feature/authentication" target="_blank">群晖身份验证器</a> </li>
</li> <li>
<li> <a href="https://sj.qq.com/appdetail/com.tencent.authenticator"
<a-tooltip title="如果报没有找到谷歌服务的错误您可以安装KK谷歌助手"> target="_blank">腾讯身份验证器</a>
<a href="https://play.google.com/store/apps/details?id=com.google.android.apps.authenticator2" target="_blank">Google Authenticator</a> </li>
</a-tooltip> <li>
</li> <a href="https://www.synology.cn/zh-cn/dsm/feature/authentication"
<li> target="_blank">群晖身份验证器</a>
<a href="https://play.google.com/store/apps/details?id=com.authy.authy" target="_blank">Authy</a> </li>
</li> <li>
</ul> <a-tooltip :title="t('certd.tooltipGoogleServiceError')">
</div> <a href="https://play.google.com/store/apps/details?id=com.google.android.apps.authenticator2"
<h3 class="font-bold m-10">2. 扫描二维码添加账号</h3> target="_blank">Google Authenticator</a>
<div v-if="authenticatorForm.qrcodeSrc" class="qrcode"> </a-tooltip>
<div class="ml-20"> </li>
<img class="full-w" :src="authenticatorForm.qrcodeSrc" /> <li>
</div> <a href="https://play.google.com/store/apps/details?id=com.authy.authy"
</div> target="_blank">Authy</a>
<h3 class="font-bold m-10">3. 输入验证码</h3> </li>
<div class="ml-20"> </ul>
<a-input v-model:value="authenticatorForm.verifyCode" placeholder="请输入验证码" /> </div>
</div> <h3 class="font-bold m-10">{{ t('certd.step2') }}</h3>
<div class="ml-20 flex mt-10"> <div v-if="authenticatorForm.qrcodeSrc" class="qrcode">
<loading-button type="primary" html-type="button" :click="doAuthenticatorSave">确认</loading-button> <div class="ml-20">
<a-button class="ml-1" @click="authenticatorForm.open = false">取消</a-button> <img class="full-w" :src="authenticatorForm.qrcodeSrc" />
</div> </div>
</a-form-item> </div>
</a-form> <h3 class="font-bold m-10">{{ t('certd.step3') }}</h3>
</div> <div class="ml-20">
</fs-page> <a-input v-model:value="authenticatorForm.verifyCode"
:placeholder="t('certd.inputVerifyCode')" />
</div>
<div class="ml-20 flex mt-10">
<loading-button type="primary" html-type="button" :click="doAuthenticatorSave">{{
t('certd.confirm')
}}</loading-button>
<a-button class="ml-1" @click="authenticatorForm.open = false">{{ t('certd.cancel')
}}</a-button>
</div>
</a-form-item>
</a-form>
</div>
</fs-page>
</template> </template>
<script setup lang="tsx"> <script setup lang="tsx">
@ -77,87 +86,92 @@ import { UserTwoFactorSetting } from "./api";
import { Modal, notification } from "ant-design-vue"; import { Modal, notification } from "ant-design-vue";
import { merge } from "lodash-es"; import { merge } from "lodash-es";
import { useSettingStore } from "/@/store/settings"; import { useSettingStore } from "/@/store/settings";
import { useI18n } from "vue-i18n";
const { t } = useI18n();
const settingsStore = useSettingStore(); const settingsStore = useSettingStore();
defineOptions({ defineOptions({
name: "UserSecurity", name: "UserSecurity",
}); });
const formState = reactive<Partial<UserTwoFactorSetting>>({ const formState = reactive<Partial<UserTwoFactorSetting>>({
authenticator: { authenticator: {
enabled: false, enabled: false,
verified: false, verified: false,
}, },
}); });
const authenticatorForm = reactive({ const authenticatorForm = reactive({
qrcodeSrc: "", qrcodeSrc: "",
verifyCode: "", verifyCode: "",
open: false, open: false,
}); });
const authenticatorOpenRef = computed(() => { const authenticatorOpenRef = computed(() => {
return formState.authenticator.enabled && (authenticatorForm.open || !formState.authenticator.verified); return formState.authenticator.enabled && (authenticatorForm.open || !formState.authenticator.verified);
}); });
watch( watch(
() => { () => {
return authenticatorOpenRef.value; return authenticatorOpenRef.value;
}, },
async open => { async open => {
if (open) { if (open) {
//base64 //base64
authenticatorForm.qrcodeSrc = await api.TwoFactorAuthenticatorGet(); authenticatorForm.qrcodeSrc = await api.TwoFactorAuthenticatorGet();
} else { } else {
authenticatorForm.qrcodeSrc = ""; authenticatorForm.qrcodeSrc = "";
authenticatorForm.verifyCode = ""; authenticatorForm.verifyCode = "";
} }
} }
); );
async function loadUserSettings() { async function loadUserSettings() {
const data: any = await api.TwoFactorSettingsGet(); const data: any = await api.TwoFactorSettingsGet();
merge(formState, data); merge(formState, data);
} }
loadUserSettings(); loadUserSettings();
const doAuthenticatorSave = async (form: any) => { const doAuthenticatorSave = async (form: any) => {
await api.TwoFactorAuthenticatorSave({ await api.TwoFactorAuthenticatorSave({
verifyCode: authenticatorForm.verifyCode, verifyCode: authenticatorForm.verifyCode,
}); });
notification.success({ notification.success({
message: "保存成功", message: t("certd.saveSuccess"),
}); });
authenticatorForm.open = false; authenticatorForm.open = false;
formState.authenticator.verified = true; formState.authenticator.verified = true;
}; };
function onAuthenticatorEnabledChanged(value: any) { function onAuthenticatorEnabledChanged(value: any) {
if (!value) { if (!value) {
// //
if (formState.authenticator.verified) { if (formState.authenticator.verified) {
Modal.confirm({ Modal.confirm({
title: "确认", title: t("certd.confirm"),
content: `确定要关闭多重验证登录吗?`, content: t("certd.confirmDisable2FA"),
async onOk() { async onOk() {
await api.TwoFactorAuthenticatorOff(); await api.TwoFactorAuthenticatorOff();
notification.success({ notification.success({
message: "关闭成功", message: t("certd.disabledSuccess"),
}); });
loadUserSettings(); loadUserSettings();
}, },
onCancel() { onCancel() {
formState.authenticator.enabled = true; formState.authenticator.enabled = true;
}, },
}); });
} }
} }
} }
</script> </script>
<style lang="less"> <style lang="less">
.page-user-settings { .page-user-settings {
.user-settings-form { .user-settings-form {
width: 600px; width: 600px;
margin: 20px; margin: 20px;
} }
} }
</style> </style>