WholeProcessPlatform/frontend/src/views/login/index.vue

725 lines
19 KiB
Vue
Raw Normal View History

2026-03-25 10:02:19 +08:00
<template>
<div class="login-container">
2026-03-27 14:50:35 +08:00
<h1>
<div>
<img :src="loginImg" alt="Logo" class="logo-img" />
<a class="system-title">{{ $t("login.title") }}</a>
2026-03-25 10:02:19 +08:00
</div>
2026-03-27 14:50:35 +08:00
</h1>
<div class="login-wrapper">
<!-- 左侧背景图区域 -->
<div class="left-section">
<div class="slogan">
<p>绿水青山就是金山银山</p>
</div>
</div>
<!-- 右侧登录表单区域 -->
<div class="right-section">
<!-- 忘记密码页面 -->
<div v-if="showForgotPassword" class="forgot-password-container">
<div class="forgot-password-header">
<a-button type="link" @click="backToLogin" class="back-button">
<span> 返回</span>
</a-button>
<h3 class="forgot-password-title">忘记密码</h3>
2026-03-25 10:02:19 +08:00
</div>
2026-03-27 14:50:35 +08:00
<a-form
:model="forgotPasswordForm"
:rules="forgotPasswordRules"
layout="vertical"
class="form-container"
2026-03-25 10:02:19 +08:00
>
2026-03-27 14:50:35 +08:00
<a-form-item label="" name="phone">
<a-input placeholder="请输入手机号" size="large">
<template #prefix>
<MobileOutlined />
</template>
</a-input>
</a-form-item>
<a-form-item label="" name="captcha">
<div class="captcha-row">
<a-input
v-model="forgotPasswordForm.captcha"
placeholder="请输入验证码"
size="large"
class="captcha-input"
>
<template #prefix>
<LockOutlined />
</template>
<template #suffix>
<a-button
type="text"
size="small"
@click="sendForgotPasswordSms"
:disabled="smsButtonDisabled"
:loading="loading"
>
{{
smsCountdown > 0 ? `${smsCountdown}秒后重新获取` : "获取验证码"
}}
</a-button>
</template>
</a-input>
</div>
</a-form-item>
<a-form-item label="" name="newPassword">
<a-input-password
v-model="forgotPasswordForm.newPassword"
placeholder="请输入新密码"
size="large"
2026-03-25 10:02:19 +08:00
>
2026-03-27 14:50:35 +08:00
<template #prefix>
<LockOutlined />
</template>
</a-input-password>
</a-form-item>
<a-button
type="primary"
2026-03-25 10:02:19 +08:00
size="large"
2026-03-27 14:50:35 +08:00
block
@click="handleResetPassword"
:loading="loading"
>
<span>提交</span>
</a-button>
</a-form>
</div>
<!-- 登录页面 -->
<div v-else>
<!-- Tabs 切换账号登录 / 短信登录 -->
<a-tabs v-model:activeKey="activeTab" class="login-tabs">
<a-tab-pane key="account" tab="账号登录">
<a-form
:model="loginData"
:rules="loginRules"
layout="vertical"
class="form-container"
@finish="onFinish"
>
<!-- 用户名/账号/手机号输入框 -->
<a-form-item label="" name="username">
<a-input
ref="username"
v-model:value="loginData.username"
clearable
type="text"
placeholder="请输入用户账号/身份证号/手机号"
size="large"
>
<template #prefix>
<UserOutlined />
</template>
</a-input>
</a-form-item>
<!-- 密码输入框 -->
<a-form-item label="" name="password">
<a-input-password
type="password"
v-model:value="loginData.password"
placeholder="请输入密码"
size="large"
>
<template #prefix>
<LockOutlined />
</template>
</a-input-password>
</a-form-item>
<!-- 验证码区域 -->
<a-form-item label="" name="captcha">
<div class="captcha-row">
<a-row :gutter="24" align="middle">
<a-col :span="10"
><a-input
v-model:value="loginData.code"
placeholder="请输入验证码"
size="large"
class="captcha-input"
/></a-col>
<a-col :span="10"
><img :src="codeUrl" alt="验证码" class="captcha-img"
/></a-col>
<a-col :span="4"> <a @click="getCode">换一张</a></a-col>
</a-row>
</div>
</a-form-item>
<a-button
type="primary"
size="large"
block
htmlType="submit"
:loading="loading"
>
<span>登录</span>
</a-button>
<a-button
type="link"
size="mini"
block
@click="showForgotPasswordPage"
:style="{ marginTop: '10px', border: 'none' }"
>
忘记密码
</a-button>
<!-- 忘记密码 -->
</a-form>
</a-tab-pane>
<!-- 短信登录 Tab (占位) -->
<a-tab-pane key="sms" tab="短信登录">
<a-form
:model="loginData"
:rules="loginRules"
layout="vertical"
class="form-container"
>
<a-form-item label="" name="username">
<a-input
v-model:value="loginData.username"
placeholder="请输入手机号"
size="large"
>
<template #prefix>
<MobileOutlined />
</template>
</a-input>
</a-form-item>
<a-form-item label="" name="captcha">
<div class="captcha-row">
<a-input
v-model="loginData.code"
placeholder="请输入验证码"
size="large"
class="captcha-input"
>
<template #prefix>
<LockOutlined />
</template>
<template #suffix>
<a-button
type="text"
size="small"
@click="sendSms"
:disabled="smsButtonDisabled"
:loading="loading"
>
{{
smsCountdown > 0
? `${smsCountdown}秒后重新获取`
: "获取验证码"
}}
</a-button>
</template>
</a-input>
</div>
</a-form-item>
<a-button
type="primary"
size="large"
block
htmlType="submit"
:loading="loading"
>
<span>登录</span>
</a-button>
</a-form>
</a-tab-pane>
</a-tabs>
2026-03-25 10:02:19 +08:00
</div>
</div>
</div>
</div>
</template>
<script setup lang="ts">
2026-04-21 14:42:10 +08:00
import { setPath } from '@/utils/auth';
2026-03-27 14:50:35 +08:00
import { onMounted, reactive, ref, toRefs, watch, nextTick } from "vue";
import loginImg from "@/assets/images/logo.png";
import { UserOutlined, LockOutlined, MobileOutlined } from "@ant-design/icons-vue";
import { getCaptcha } from "@/api/auth";
import { message } from "ant-design-vue";
2026-03-25 10:02:19 +08:00
// 组件依赖
2026-03-27 14:50:35 +08:00
import router from "@/router";
import Cookies from "js-cookie";
2026-03-25 10:02:19 +08:00
// API依赖
2026-03-27 14:50:35 +08:00
import { useRoute } from "vue-router";
import { LoginData } from "@/api/auth/types";
2026-03-25 10:02:19 +08:00
//密码加密
2026-03-27 14:50:35 +08:00
import { encrypt, decrypt } from "@/utils/rsaEncrypt";
2026-03-25 10:02:19 +08:00
// 状态管理依赖
2026-03-27 14:50:35 +08:00
import { useUserStore } from "@/store/modules/user";
2026-03-25 10:02:19 +08:00
// 国际化
2026-03-27 14:50:35 +08:00
import { useI18n } from "vue-i18n";
2026-03-25 10:02:19 +08:00
const { t } = useI18n();
const userStore = useUserStore();
const route = useRoute();
// 图片验证码
2026-03-27 14:50:35 +08:00
const codeUrl = ref("");
2026-03-25 10:02:19 +08:00
2026-03-27 14:50:35 +08:00
const smsCountdown = ref(0); // 短信验证码倒计时
const smsButtonDisabled = ref(false); // 短信按钮禁用状态
2026-03-25 10:02:19 +08:00
// 记住密码
let remember = ref(false);
2026-03-27 14:50:35 +08:00
// 忘记密码表单数据
const forgotPasswordForm = ref({
phone: "",
captcha: "",
newPassword: "",
});
// 忘记密码
const showForgotPassword = ref(false);
// 登录方式
const activeTab = ref("account");
2026-03-25 10:02:19 +08:00
const state = reactive({
2026-03-27 14:50:35 +08:00
redirect: "",
2026-03-25 10:02:19 +08:00
loginData: {
2026-03-27 14:50:35 +08:00
uuid: "",
username: "admin",
password: "123456",
code: "",
2026-03-25 10:02:19 +08:00
} as LoginData,
loginRules: {
2026-03-27 14:50:35 +08:00
username: [{ required: true, trigger: "blur", message: t("login.rulesUsername") }],
password: [{ required: true, trigger: "blur", message: t("login.rulesPassword") }],
code: [{ required: true, trigger: "blur", message: "请输入验证码" }],
2026-03-25 10:02:19 +08:00
},
loginImg: loginImg[0],
loading: false,
2026-03-27 14:50:35 +08:00
passwordType: "password",
2026-03-25 10:02:19 +08:00
// 大写提示禁用
capslockTooltipDisabled: true,
otherQuery: {},
clientHeight: document.documentElement.clientHeight,
showDialog: false,
2026-03-27 14:50:35 +08:00
cookiePass: "",
});
// 忘记密码表单校验规则
const forgotPasswordRules = ref({
phone: [
{ required: true, message: "手机号不能为空", trigger: "blur" },
{ pattern: /^1[3-9]\d{9}$/, message: "请输入正确的手机号", trigger: "blur" },
],
captcha: [{ required: true, message: "验证码不能为空", trigger: "blur" }],
newPassword: [
{ required: true, message: "新密码不能为空", trigger: "blur" },
{ min: 6, message: "密码长度不能少于6位", trigger: "blur" },
],
2026-03-25 10:02:19 +08:00
});
2026-03-27 14:50:35 +08:00
const { loginData, loginRules, loading, passwordType, capslockTooltipDisabled } = toRefs(
state
);
2026-03-25 10:02:19 +08:00
function checkCapslock(e: any) {
const { key } = e;
2026-03-27 14:50:35 +08:00
state.capslockTooltipDisabled = key && key.length === 1 && key >= "A" && key <= "Z";
2026-03-25 10:02:19 +08:00
}
function showPwd() {
2026-03-27 14:50:35 +08:00
if (passwordType.value === "password") {
passwordType.value = "";
2026-03-25 10:02:19 +08:00
} else {
2026-03-27 14:50:35 +08:00
passwordType.value = "password";
2026-03-25 10:02:19 +08:00
}
nextTick(() => {
passwordRef.value.focus();
});
}
/**
* 登录
*/
2026-03-27 14:50:35 +08:00
function onFinish() {
state.loading = true;
const user = {
username: state.loginData.username,
password: state.loginData.password,
// rememberMe: state.loginData.rememberMe,
code: state.loginData.code,
uuid: state.loginData.uuid,
};
if (user.password !== state.cookiePass) {
user.password = encrypt(user.password);
}
console.log(user);
userStore
.login(user)
.then(() => {
Cookies.set("username", user.username);
router.push({ path: "/" });
2026-04-21 14:42:10 +08:00
setPath('/login')
2026-03-27 14:50:35 +08:00
state.loading = false;
})
.catch(() => {
getCode();
state.loading = false;
});
2026-03-25 10:02:19 +08:00
}
watch(
route,
() => {
const query = route.query;
if (query) {
state.redirect = query.redirect as string;
state.otherQuery = getOtherQuery(query);
}
},
{
2026-03-27 14:50:35 +08:00
immediate: true,
2026-03-25 10:02:19 +08:00
}
);
function getOtherQuery(query: any) {
return Object.keys(query).reduce((acc: any, cur: any) => {
2026-03-27 14:50:35 +08:00
if (cur !== "redirect") {
2026-03-25 10:02:19 +08:00
acc[cur] = query[cur];
}
return acc;
}, {});
}
function getCookie() {
2026-03-27 14:50:35 +08:00
const username = Cookies.get("username");
let password = Cookies.get("password");
const rememberMe = Cookies.get("rememberMe");
rememberMe == "true" ? (remember.value = Boolean(rememberMe)) : false;
2026-03-25 10:02:19 +08:00
// 保存cookie里面的加密后的密码
2026-03-27 14:50:35 +08:00
state.cookiePass = password === undefined ? "" : password;
2026-03-25 10:02:19 +08:00
password = password === undefined ? state.loginData.password : password;
state.loginData = {
username: username === undefined ? state.loginData.username : username,
password: decrypt(password),
2026-03-27 14:50:35 +08:00
code: "",
uuid: "",
2026-03-25 10:02:19 +08:00
};
2026-03-27 14:50:35 +08:00
remember.value = rememberMe === undefined ? false : Boolean(rememberMe);
2026-03-25 10:02:19 +08:00
}
function getCode() {
2026-03-27 14:50:35 +08:00
getCaptcha().then((result: any) => {
codeUrl.value = result.data.img;
2026-03-25 10:02:19 +08:00
state.loginData.uuid = result.data.uuid;
2026-03-27 14:50:35 +08:00
});
2026-03-25 10:02:19 +08:00
}
2026-03-27 14:50:35 +08:00
// 开始倒计时
const startCountdown = () => {
smsCountdown.value = 60;
const timer = setInterval(() => {
smsCountdown.value--;
if (smsCountdown.value <= 0) {
clearInterval(timer);
smsButtonDisabled.value = false;
}
}, 1000);
};
// 显示忘记密码页面
const showForgotPasswordPage = () => {
showForgotPassword.value = true;
};
// 返回登录页面
const backToLogin = () => {
showForgotPassword.value = false;
// 重置忘记密码表单
forgotPasswordForm.value = {
phone: "",
captcha: "",
newPassword: "",
};
};
// 发送短信验证码
const sendSms = async () => {
// 检查手机号是否为空
if (!loginData.value.username) {
message.error("请输入手机号");
return;
}
// 检查手机号格式
const phoneRegex = /^1[3-9]\d{9}$/;
if (!phoneRegex.test(loginData.value.username)) {
message.error("请输入正确的手机号");
return;
}
// 如果正在倒计时,不允许重复发送
if (smsCountdown.value > 0) {
return;
}
loading.value = true;
smsButtonDisabled.value = true;
try {
// 模拟发送短信验证码接口
// await axios.post('/sms/send', { phone: loginForm.value.username })
// 模拟发送成功
message.success("验证码发送成功");
// 开始倒计时
startCountdown();
} catch (error) {
console.error("发送验证码失败", error);
message.error("验证码发送失败,请重试");
smsButtonDisabled.value = false;
} finally {
loading.value = false;
}
};
// 发送忘记密码短信验证码
const sendForgotPasswordSms = async () => {
// 检查手机号是否为空
if (!forgotPasswordForm.value.phone) {
message.error("请输入手机号");
return;
}
// 检查手机号格式
const phoneRegex = /^1[3-9]\d{9}$/;
if (!phoneRegex.test(forgotPasswordForm.value.phone)) {
message.error("请输入正确的手机号");
return;
}
// 如果正在倒计时,不允许重复发送
if (smsCountdown.value > 0) {
return;
}
loading.value = true;
smsButtonDisabled.value = true;
try {
// 模拟发送短信验证码接口
// await axios.post('/sms/forgot-password', { phone: forgotPasswordForm.value.phone })
// 模拟发送成功
message.success("验证码发送成功");
// 开始倒计时
startCountdown();
} catch (error) {
console.error("发送验证码失败", error);
message.error("验证码发送失败,请重试");
smsButtonDisabled.value = false;
} finally {
loading.value = false;
}
};
// 处理密码重置
const handleResetPassword = async () => {
loading.value = true;
try {
// 2. 模拟接口请求 (实际替换为真实重置密码接口)
// await axios.post('/user/reset-password', {
// phone: forgotPasswordForm.value.phone,
// captcha: forgotPasswordForm.value.captcha,
// newPassword: forgotPasswordForm.value.newPassword
// })
// 3. 模拟重置成功
message.success("密码重置成功");
// 4. 返回登录页面
backToLogin();
} catch (error) {
console.error("密码重置失败", error);
message.error("密码重置失败,请重试");
} finally {
loading.value = false;
}
};
2026-03-25 10:02:19 +08:00
onMounted(() => {
getCookie();
getCode();
});
</script>
2026-03-27 14:50:35 +08:00
<style scoped lang="scss">
2026-03-25 10:02:19 +08:00
.login-container {
2026-03-27 14:50:35 +08:00
margin: 0px auto;
position: relative;
width: 100%;
height: 100%;
min-width: 1500px;
background-color: rgb(255, 255, 255);
h1 {
2026-03-25 10:02:19 +08:00
position: relative;
2026-03-27 14:50:35 +08:00
padding: 66px 0px 2px 80px;
margin-bottom: 0px;
height: 155px;
2026-03-25 10:02:19 +08:00
2026-03-27 14:50:35 +08:00
div {
position: relative;
line-height: 71px;
2026-03-25 10:02:19 +08:00
}
2026-03-27 14:50:35 +08:00
.logo-img {
2026-03-25 10:02:19 +08:00
position: absolute;
2026-03-27 14:50:35 +08:00
left: 0px;
top: 10px;
width: 55px;
2026-03-25 10:02:19 +08:00
}
2026-03-27 14:50:35 +08:00
.system-title {
margin-left: 71px;
font-weight: 700;
font-size: 35px;
color: rgb(47, 107, 152);
2026-03-25 10:02:19 +08:00
}
}
2026-03-27 14:50:35 +08:00
:deep(.ant-tabs-tab) {
padding: 8px 0;
2026-03-25 10:02:19 +08:00
}
2026-03-27 14:50:35 +08:00
:deep(.ant-input-prefix) {
display: flex;
width: 26px;
2026-03-25 10:02:19 +08:00
2026-03-27 14:50:35 +08:00
svg {
width: 18px;
height: 18px;
margin-right: 4px;
2026-03-25 10:02:19 +08:00
}
}
2026-03-27 14:50:35 +08:00
.login-wrapper {
position: relative;
width: 100%;
height: 72%;
min-height: 600px;
background: url("@/assets/images/login-bg.jpg");
background-repeat: no-repeat;
background-size: 100% 100%;
2026-03-25 10:02:19 +08:00
}
2026-03-27 14:50:35 +08:00
// 左侧背景区域
.left-section {
.slogan {
position: absolute;
top: 28%;
left: 18%;
width: 440px;
height: 112px;
color: rgb(255, 255, 255);
font-size: 40px;
}
2026-03-25 10:02:19 +08:00
}
2026-03-27 14:50:35 +08:00
// 右侧登录卡片区域
.right-section {
2026-03-25 10:02:19 +08:00
position: absolute;
2026-03-27 14:50:35 +08:00
left: 74%;
top: 20%;
width: 20%;
max-height: 402px;
max-width: 400px;
min-height: 362px;
border-radius: 3px;
padding: 15px 24px 24px;
background-color: rgb(255, 255, 255);
.ant-tabs-nav {
height: 38px;
}
2026-03-25 10:02:19 +08:00
}
2026-03-27 14:50:35 +08:00
/* 验证码布局 */
.captcha-row {
display: flex;
.ant-row {
display: flex;
align-items: start;
}
/* gap: 12px;
align-items: center; */
2026-03-25 10:02:19 +08:00
2026-03-27 14:50:35 +08:00
.captcha-img {
width: 100%;
height: 40px;
line-height: 40px;
background-repeat: no-repeat;
background-size: contain;
background-position: center center;
}
a {
line-height: 50px;
float: right;
color: #1890ff;
text-decoration: none;
background-color: transparent;
outline: none;
2026-03-25 10:02:19 +08:00
cursor: pointer;
2026-03-27 14:50:35 +08:00
transition: color 0.3s;
white-space: nowrap;
2026-03-25 10:02:19 +08:00
}
}
2026-03-27 14:50:35 +08:00
// 忘记密码样式
.forgot-link {
float: right;
color: #1677ff;
font-size: 14px;
2026-03-25 10:02:19 +08:00
}
2026-03-27 14:50:35 +08:00
// 短信登录占位提示
.sms-login-tip {
padding: 40px 20px;
text-align: center;
color: #999;
2026-03-25 10:02:19 +08:00
}
2026-03-27 14:50:35 +08:00
// 忘记密码页面样式
.forgot-password-container {
.forgot-password-header {
display: flex;
align-items: center;
margin-bottom: 24px;
.back-button {
padding: 0;
margin-right: 12px;
color: #1677ff;
font-size: 14px;
&:hover {
color: #4096ff;
}
}
.forgot-password-title {
margin: 0;
font-size: 18px;
font-weight: 600;
color: #333;
}
}
2026-03-25 10:02:19 +08:00
}
}
</style>