2024-05-18 01:11:46 +08:00

189 lines
5.5 KiB
Vue

<script setup lang="ts">
import Motion from "./utils/motion";
import { useRouter } from "vue-router";
import { message } from "@/utils/message";
import { loginRules } from "./utils/rule";
import { useNav } from "@/layout/hooks/useNav";
import type { FormInstance } from "element-plus";
import { useLayout } from "@/layout/hooks/useLayout";
import { useUserStoreHook } from "@/store/modules/user";
import { addPathMatch } from "@/router/utils";
import { bg, avatar, illustration } from "./utils/static";
import { useRenderIcon } from "@/components/ReIcon/src/hooks";
import { ref, reactive, toRaw, onMounted, onBeforeUnmount } from "vue";
import { useDataThemeChange } from "@/layout/hooks/useDataThemeChange";
import dayIcon from "@/assets/svg/day.svg?component";
import darkIcon from "@/assets/svg/dark.svg?component";
import Lock from "@iconify-icons/ri/lock-fill";
import User from "@iconify-icons/ri/user-3-fill";
import ReImageVerify from "@/components/ReImageVerify/src/index.vue";
import { usePermissionStoreHook } from "@/store/modules/permission";
import useGetGlobalProperties from "@/hooks/useGetGlobalProperties";
defineOptions({
name: "Login"
});
const { $bus } = useGetGlobalProperties();
const router = useRouter();
const loading = ref(false);
const loginRuleFormRef = ref<FormInstance>();
const { initStorage } = useLayout();
initStorage();
const { dataTheme, overallStyle, dataThemeChange } = useDataThemeChange();
dataThemeChange(overallStyle.value);
const { title } = useNav();
const imgCode = ref("");
const imgCodeId = ref("");
const loginRuleForm = reactive({
account: "", // 账号
password: "", // 密码
captchaId: "", // 验证码id
captchaAnswer: "" // 验证码
});
// 登陆接口
const onLogin = async (formEl: FormInstance | undefined) => {
if (!formEl) return;
await formEl.validate((valid, fields) => {
if (valid) {
loading.value = true;
loginRuleForm.captchaId = imgCodeId.value;
useUserStoreHook()
.loginByUsername(loginRuleForm)
.then(res => {
if (res.code === 200) {
// 获取后端路由
usePermissionStoreHook().handleWholeMenus([]);
addPathMatch();
router.push("/welcome");
message("登录成功", { type: "success" });
} else {
message("登录失败", { type: "error" });
}
})
.catch(e => {
$bus.emit("refreshCode", true);
})
.finally(() => (loading.value = false));
} else {
return fields;
}
});
};
/** 使用公共函数,避免`removeEventListener`失效 */
function onkeypress({ code }: KeyboardEvent) {
if (code === "Enter") {
onLogin(loginRuleFormRef.value);
}
}
onMounted(() => {
window.document.addEventListener("keypress", onkeypress);
});
onBeforeUnmount(() => {
window.document.removeEventListener("keypress", onkeypress);
});
</script>
<template>
<div class="select-none">
<img :src="bg" class="wave" />
<div class="flex-c absolute right-5 top-3">
<!-- 主题 -->
<el-switch
v-model="dataTheme"
inline-prompt
:active-icon="dayIcon"
:inactive-icon="darkIcon"
@change="dataThemeChange"
/>
</div>
<div class="login-container">
<div class="img">
<component :is="toRaw(illustration)" />
</div>
<div class="login-box">
<div class="login-form">
<avatar class="avatar" />
<Motion>
<h2 class="outline-none">{{ title }} LOGIN</h2>
</Motion>
<el-form
ref="loginRuleFormRef"
:model="loginRuleForm"
:rules="loginRules"
size="large"
>
<Motion :delay="100">
<el-form-item prop="account">
<el-input
v-model="loginRuleForm.account"
clearable
placeholder="账号"
:prefix-icon="useRenderIcon(User)"
/>
</el-form-item>
</Motion>
<Motion :delay="150">
<el-form-item prop="password">
<el-input
v-model="loginRuleForm.password"
clearable
show-password
placeholder="密码"
:prefix-icon="useRenderIcon(Lock)"
/>
</el-form-item>
</Motion>
<Motion :delay="200">
<el-form-item prop="captchaAnswer">
<el-input
v-model="loginRuleForm.captchaAnswer"
clearable
:prefix-icon="useRenderIcon('ri:shield-keyhole-line')"
>
<template v-slot:append>
<ReImageVerify
v-model:code="imgCode"
v-model:codeId="imgCodeId"
/>
</template>
</el-input>
</el-form-item>
</Motion>
<Motion :delay="250">
<el-button
class="w-full mt-4"
size="default"
type="primary"
:loading="loading"
@click="onLogin(loginRuleFormRef)"
>
登录
</el-button>
</Motion>
</el-form>
</div>
</div>
</div>
</div>
</template>
<style scoped>
@import url("@/style/login.css");
</style>
<style lang="scss" scoped>
:deep(.el-input-group__append, .el-input-group__prepend) {
padding: 0;
}
</style>