123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331 |
- <script setup lang="ts">
- import {
- ref,
- toRaw,
- reactive,
- watch,
- computed,
- onMounted,
- onBeforeUnmount
- } from "vue";
- import { useI18n } from "vue-i18n";
- import Motion from "./utils/motion";
- // import { useRouter } from "vue-router";
- // import { router } from "@/router";
- import { message } from "@/utils/message";
- import { loginRules } from "./utils/rule";
- import phone from "./components/phone.vue";
- import TypeIt from "@/components/ReTypeit";
- import qrCode from "./components/qrCode.vue";
- import regist from "./components/regist.vue";
- import update from "./components/update.vue";
- import { initRouter } from "@/router/utils";
- import { Router } from "vue-router";
- import { useNav } from "@/layout/hooks/useNav";
- import type { FormInstance } from "element-plus";
- import { $t, transformI18n } from "@/plugins/i18n";
- import { operates, thirdParty } from "./utils/enums";
- import { useLayout } from "@/layout/hooks/useLayout";
- import { useUserStoreHook } from "@/store/modules/user";
- import { loginicon, avatar, illustration } from "./utils/static";
- import { ReImageVerify } from "@/components/ReImageVerify";
- import { useRenderIcon } from "@/components/ReIcon/src/hooks";
- import { useTranslationLang } from "@/layout/hooks/useTranslationLang";
- import { useDataThemeChange } from "@/layout/hooks/useDataThemeChange";
- import Cookies from "js-cookie";
- import { ShakeDownEncrypt, ShakeDownDecrypt } from "@/utils/CryptoJS";
- import dayIcon from "@/assets/svg/day.svg?component";
- import darkIcon from "@/assets/svg/dark.svg?component";
- import globalization from "@/assets/svg/globalization.svg?component";
- import Lock from "@iconify-icons/ri/lock-fill";
- import Check from "@iconify-icons/ep/check";
- import User from "@iconify-icons/ri/user-3-fill";
- defineOptions({
- name: "Login"
- });
- const imgCode = ref("");
- // const router = useRouter();
- const loading = ref(false);
- const checked = ref(false);
- const ruleFormRef = ref<FormInstance>();
- const currentPage = computed(() => {
- return useUserStoreHook().currentPage;
- });
- const { t } = useI18n();
- const { initStorage } = useLayout();
- initStorage();
- const { dataTheme, dataThemeChange } = useDataThemeChange();
- dataThemeChange();
- const { title, getDropdownItemStyle, getDropdownItemClass } = useNav();
- const { locale, translationCh, translationEn } = useTranslationLang();
- const ruleForm = reactive({
- username: "",
- password: "",
- verifyCode: ""
- });
- const onLogin = async (formEl: FormInstance | undefined) => {
- loading.value = true;
- const value = {
- username: ruleForm.username,
- password: ruleForm.password
- };
- checked.value
- ? Cookies.set("isRememberPassword", "true") &&
- Cookies.set("RememberAccount", ShakeDownEncrypt(ruleForm))
- : Cookies.set("isRememberPassword", "") &&
- Cookies.set(
- "RememberAccount",
- ShakeDownEncrypt({ username: ruleForm.username })
- );
- if (!formEl) return;
- await formEl.validate((valid, fields) => {
- if (valid) {
- useUserStoreHook()
- .loginByUsername(value)
- .then((res: any) => {
- if (res.status === 1) {
- // 获取后端路
- initRouter().then((router: Router) => {
- router.push("/");
- message("登录成功", { type: "success" });
- });
- } else {
- loading.value = false;
- message(res.msg, { type: "error" });
- }
- });
- } else {
- loading.value = false;
- return fields;
- }
- });
- };
- /** 使用公共函数,避免`removeEventListener`失效 */
- function onkeypress({ code }: KeyboardEvent) {
- if (code === "Enter" || code === "NumpadEnter") {
- onLogin(ruleFormRef.value);
- }
- }
- onMounted(() => {
- Cookies.get("isRememberPassword") &&
- (checked.value = true) &&
- (ruleForm.password = ShakeDownDecrypt(
- Cookies.get("RememberAccount")
- ).password);
- Cookies.get("RememberAccount") &&
- (ruleForm.username = ShakeDownDecrypt(
- Cookies.get("RememberAccount")
- ).username);
- window.document.addEventListener("keypress", onkeypress);
- });
- onBeforeUnmount(() => {
- window.document.removeEventListener("keypress", onkeypress);
- });
- watch(imgCode, value => {
- useUserStoreHook().SET_VERIFYCODE(value);
- });
- </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"
- /> -->
- <!-- 国际化 -->
- <!-- <el-dropdown trigger="click">
- <globalization
- class="hover:text-primary hover:!bg-[transparent] w-[20px] h-[20px] ml-1.5 cursor-pointer outline-none duration-300"
- />
- <template #dropdown>
- <el-dropdown-menu class="translation">
- <el-dropdown-item
- :style="getDropdownItemStyle(locale, 'zh')"
- :class="['dark:!text-white', getDropdownItemClass(locale, 'zh')]"
- @click="translationCh"
- >
- <IconifyIconOffline
- class="check-zh"
- v-show="locale === 'zh'"
- :icon="Check"
- />
- 简体中文
- </el-dropdown-item>
- <el-dropdown-item
- :style="getDropdownItemStyle(locale, 'en')"
- :class="['dark:!text-white', getDropdownItemClass(locale, 'en')]"
- @click="translationEn"
- >
- <span class="check-en" v-show="locale === 'en'">
- <IconifyIconOffline :icon="Check" />
- </span>
- English
- </el-dropdown-item>
- </el-dropdown-menu>
- </template>
- </el-dropdown> -->
- </div>
- <div class="login-container">
- <div class="img">
- <!-- <component :is="toRaw(illustration)" /> -->
- <img :src="loginicon" class="loginicon" />
- </div>
- <div class="login-box">
- <div class="login-form">
- <!-- <avatar class="avatar" /> -->
- <img :src="avatar" class="avatar" />
- <Motion>
- <h2 class="outline-none">
- <TypeIt :values="[title]" :cursor="false" :speed="150" />
- </h2>
- </Motion>
- <el-form v-if="currentPage === 0" ref="ruleFormRef" :model="ruleForm" :rules="loginRules" size="large">
- <Motion :delay="100">
- <el-form-item :rules="[
- {
- required: true,
- message: transformI18n($t('login.usernameReg')),
- trigger: 'blur'
- }
- ]" prop="username">
- <el-input clearable v-model="ruleForm.username" :placeholder="t('login.username')"
- :prefix-icon="useRenderIcon(User)" />
- </el-form-item>
- </Motion>
- <Motion :delay="150">
- <el-form-item prop="password">
- <el-input clearable show-password v-model="ruleForm.password" :placeholder="t('login.password')"
- :prefix-icon="useRenderIcon(Lock)" />
- </el-form-item>
- </Motion>
- <Motion :delay="200">
- <el-form-item prop="verifyCode">
- <el-input clearable v-model="ruleForm.verifyCode" :placeholder="t('login.verifyCode')"
- :prefix-icon="useRenderIcon('ri:shield-keyhole-line')">
- <template v-slot:append>
- <ReImageVerify v-model:code="imgCode" />
- </template>
- </el-input>
- </el-form-item>
- </Motion>
- <Motion :delay="250">
- <el-form-item>
- <div class="w-full h-[20px] flex justify-between items-center">
- <el-checkbox v-model="checked">
- {{ t("login.remember") }}
- </el-checkbox>
- <!-- <el-button
- link
- type="primary"
- @click="useUserStoreHook().SET_CURRENTPAGE(4)"
- >
- {{ t("login.forget") }}
- </el-button> -->
- </div>
- <el-button class="w-full mt-4" size="default" type="primary" :loading="loading"
- @click="onLogin(ruleFormRef)">
- {{ t("login.login") }}
- </el-button>
- </el-form-item>
- </Motion>
- <!-- <Motion :delay="300">
- <el-form-item>
- <div class="w-full h-[20px] flex justify-between items-center">
- <el-button
- v-for="(item, index) in operates"
- :key="index"
- class="w-full mt-4"
- size="default"
- @click="useUserStoreHook().SET_CURRENTPAGE(index + 1)"
- >
- {{ t(item.title) }}
- </el-button>
- </div>
- </el-form-item>
- </Motion> -->
- </el-form>
- <!-- <Motion v-if="currentPage === 0" :delay="350">
- <el-form-item>
- <el-divider>
- <p class="text-gray-500 text-xs">{{ t("login.thirdLogin") }}</p>
- </el-divider>
- <div class="w-full flex justify-evenly">
- <span
- v-for="(item, index) in thirdParty"
- :key="index"
- :title="t(item.title)"
- >
- <IconifyIconOnline
- :icon="`ri:${item.icon}-fill`"
- width="20"
- class="cursor-pointer text-gray-500 hover:text-blue-400"
- />
- </span>
- </div>
- </el-form-item>
- </Motion> -->
- <!-- 手机号登录 -->
- <phone v-if="currentPage === 1" />
- <!-- 二维码登录 -->
- <qrCode v-if="currentPage === 2" />
- <!-- 注册 -->
- <regist v-if="currentPage === 3" />
- <!-- 忘记密码 -->
- <update v-if="currentPage === 4" />
- </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;
- }
- .translation {
- ::v-deep(.el-dropdown-menu__item) {
- padding: 5px 40px;
- }
- .check-zh {
- position: absolute;
- left: 20px;
- }
- .check-en {
- position: absolute;
- left: 20px;
- }
- }
- .avatar {
- margin: 0 auto;
- }
- </style>
|