1056 lines
23 KiB
Vue
1056 lines
23 KiB
Vue
<template>
|
|
<div class="login-bg">
|
|
<div class="login-card">
|
|
<div class="login-side">
|
|
<div class="brand">
|
|
<img
|
|
src="@/assets/svgs/logo-w.svg"
|
|
alt="Logo"
|
|
style="color: white; width: 50px"
|
|
/>
|
|
<span class="brand-title">后台管理系统</span>
|
|
</div>
|
|
<div class="illus">
|
|
<svg viewBox="0 0 300 160" style="max-width: 100%" fill="none">
|
|
<ellipse cx="150" cy="140" rx="120" ry="16" fill="#edf4fd" />
|
|
<rect x="57" y="58" width="60" height="40" rx="12" fill="#64b6f7" />
|
|
<rect
|
|
x="125"
|
|
y="46"
|
|
width="110"
|
|
height="64"
|
|
rx="14"
|
|
fill="#389bf7"
|
|
opacity="0.11"
|
|
/>
|
|
<rect
|
|
x="136"
|
|
y="60"
|
|
width="60"
|
|
height="41"
|
|
rx="10"
|
|
fill="#b8e1ff"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
<!-- 版权信息 -->
|
|
<div class="copyright">© 2026 Yunzer 管理系统</div>
|
|
</div>
|
|
<div class="login-panel">
|
|
<h2 class="login-title">欢迎登录</h2>
|
|
<div class="login-desc">请填写您的账号信息</div>
|
|
<div class="form-group icon-input-group">
|
|
<span class="input-icon">
|
|
<i class="fa-solid fa-user"></i>
|
|
</span>
|
|
<input
|
|
v-model="account"
|
|
type="text"
|
|
placeholder="用户名"
|
|
autocomplete="account"
|
|
class="input input-with-icon"
|
|
/>
|
|
</div>
|
|
<div class="form-group icon-input-group">
|
|
<span class="input-icon">
|
|
<i class="fa-solid fa-lock"></i>
|
|
</span>
|
|
<input
|
|
v-model="password"
|
|
:type="passwordVisible ? 'text' : 'password'"
|
|
placeholder="密码"
|
|
autocomplete="current-password"
|
|
class="input input-with-icon"
|
|
/>
|
|
<span
|
|
class="visible-btn"
|
|
@click="passwordVisible = !passwordVisible"
|
|
:title="passwordVisible ? '隐藏密码' : '显示密码'"
|
|
>
|
|
<i v-if="passwordVisible" class="fa-regular fa-eye"></i>
|
|
<i v-else class="fa-solid fa-eye-slash"></i>
|
|
</span>
|
|
</div>
|
|
<div
|
|
v-if="openVerifyEnabled && verifyType === 'captcha'"
|
|
class="form-group code-row"
|
|
>
|
|
<input
|
|
v-model="captchaInput"
|
|
type="text"
|
|
placeholder="请输入4位验证码"
|
|
class="input code-input"
|
|
/>
|
|
<button class="code-btn" type="button" @click="generateCaptcha">
|
|
{{ captchaText }}
|
|
</button>
|
|
</div>
|
|
<div
|
|
v-if="
|
|
openVerifyEnabled &&
|
|
(verifyType === 'sms' || verifyType === 'email')
|
|
"
|
|
class="form-group code-row"
|
|
>
|
|
<input
|
|
v-model="verifyCode"
|
|
type="text"
|
|
placeholder="请输入验证码"
|
|
class="input code-input"
|
|
/>
|
|
<button
|
|
class="code-btn"
|
|
type="button"
|
|
:disabled="codeSending || codeCountdown > 0"
|
|
@click="handleSendCode"
|
|
>
|
|
{{
|
|
codeCountdown > 0
|
|
? `${codeCountdown}s`
|
|
: codeSending
|
|
? "发送中..."
|
|
: "发送验证码"
|
|
}}
|
|
</button>
|
|
</div>
|
|
<!-- 极验验证码容器 -->
|
|
<div
|
|
style="display: none"
|
|
v-if="
|
|
openVerifyEnabled &&
|
|
verifyType === 'geetest' &&
|
|
showCaptchaContainer
|
|
"
|
|
class="geetest-container"
|
|
ref="captchaContainer"
|
|
></div>
|
|
|
|
<div class="remember-me-row">
|
|
<label class="remember-me-label">
|
|
<input
|
|
type="checkbox"
|
|
v-model="rememberMe"
|
|
class="remember-me-checkbox"
|
|
@change="handleRememberMeChange"
|
|
/>
|
|
<span>记住我</span>
|
|
</label>
|
|
<div class="action-links">
|
|
<a class="forget-link" @click.prevent="goForget">忘记密码?</a>
|
|
<span> | </span>
|
|
<a class="forget-link" @click.prevent="clearCache">清除缓存</a>
|
|
</div>
|
|
</div>
|
|
<transition name="fade">
|
|
<div v-if="errorMsg" class="error-msg">{{ errorMsg }}</div>
|
|
</transition>
|
|
<button class="login-btn" @click="handleLogin" :disabled="loading">
|
|
{{ loading ? "登录中..." : "登 录" }}
|
|
</button>
|
|
<div></div>
|
|
</div>
|
|
</div>
|
|
<!-- 背景光效 -->
|
|
<div class="login-light light1"></div>
|
|
<div class="login-light light2"></div>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ref, onMounted, nextTick } from "vue";
|
|
import { useRouter } from "vue-router";
|
|
import { useAuthStore } from "@/stores/auth";
|
|
import {
|
|
login,
|
|
getOpenVerify,
|
|
getGeetest4Infos,
|
|
sendLoginCode,
|
|
} from "@/api/login";
|
|
import { getVerifyInfos } from "@/api/sitesettings.js";
|
|
import "@/assets/js/gt4.js";
|
|
import { ElMessageBox, ElMessage } from "element-plus";
|
|
|
|
const router = useRouter();
|
|
const authStore = useAuthStore();
|
|
|
|
// --- 表单数据 ---
|
|
const account = ref("");
|
|
const password = ref("");
|
|
const passwordVisible = ref(false);
|
|
const rememberMe = ref(false);
|
|
const loading = ref(false);
|
|
const errorMsg = ref("");
|
|
const openVerifyEnabled = ref(true);
|
|
const verifyType = ref("captcha");
|
|
const captchaInput = ref("");
|
|
const captchaText = ref("");
|
|
const verifyCode = ref("");
|
|
const codeSending = ref(false);
|
|
const codeCountdown = ref(0);
|
|
let codeTimer = null;
|
|
|
|
// --- 极验相关变量 ---
|
|
const showCaptchaContainer = ref(false);
|
|
const captchaContainer = ref(null);
|
|
const captchaInstance = ref(null);
|
|
|
|
// --- 加载JS脚本 ---
|
|
const loadScript = (url) => {
|
|
return new Promise((resolve, reject) => {
|
|
const script = document.createElement("script");
|
|
script.src = url;
|
|
script.async = true;
|
|
script.onload = () => resolve();
|
|
script.onerror = () => reject(new Error(`加载脚本失败: ${url}`));
|
|
document.head.appendChild(script);
|
|
});
|
|
};
|
|
|
|
// --- 清理验证码实例 ---
|
|
const cleanCaptchaInstance = () => {
|
|
if (captchaInstance.value) {
|
|
try {
|
|
captchaInstance.value.destroy();
|
|
} catch (e) {
|
|
console.warn("销毁验证码实例失败:", e);
|
|
}
|
|
captchaInstance.value = null;
|
|
showCaptchaContainer.value = false;
|
|
}
|
|
};
|
|
|
|
// --- 执行登录请求 ---
|
|
const performLoginRequest = async () => {
|
|
const res = await login({
|
|
account: account.value,
|
|
password: password.value,
|
|
code: verifyCode.value,
|
|
});
|
|
|
|
if (res && res.code === 200) {
|
|
// 登录成功:处理"记住我"
|
|
if (rememberMe.value) {
|
|
localStorage.setItem("loginAccount", account.value);
|
|
localStorage.setItem("loginPassword", password.value);
|
|
localStorage.setItem("loginRememberMe", "true");
|
|
} else {
|
|
localStorage.removeItem("loginAccount");
|
|
localStorage.removeItem("loginTenantName");
|
|
localStorage.removeItem("loginPassword");
|
|
localStorage.setItem("loginRememberMe", "false");
|
|
}
|
|
|
|
authStore.setLoginInfo(res.data);
|
|
|
|
// 重置 Tabs 状态
|
|
const { useTabsStore } = await import("@/stores");
|
|
const tabsStore = useTabsStore();
|
|
tabsStore.resetTabs();
|
|
|
|
router.push({ path: "/home" });
|
|
ElMessage.success("登录成功!");
|
|
return true;
|
|
} else {
|
|
errorMsg.value = res.msg || "登录失败";
|
|
return false;
|
|
}
|
|
};
|
|
|
|
const generateCaptcha = () => {
|
|
// 系统生成4位随机数字验证码
|
|
captchaText.value = `${Math.floor(1000 + Math.random() * 9000)}`;
|
|
};
|
|
|
|
const startCodeCountdown = () => {
|
|
codeCountdown.value = 60;
|
|
if (codeTimer) clearInterval(codeTimer);
|
|
codeTimer = setInterval(() => {
|
|
codeCountdown.value -= 1;
|
|
if (codeCountdown.value <= 0) {
|
|
clearInterval(codeTimer);
|
|
codeTimer = null;
|
|
}
|
|
}, 1000);
|
|
};
|
|
|
|
const handleSendCode = async () => {
|
|
if (codeSending.value || codeCountdown.value > 0) return;
|
|
if (!account.value.trim()) {
|
|
errorMsg.value = "请先输入用户名";
|
|
return;
|
|
}
|
|
codeSending.value = true;
|
|
try {
|
|
const res = await sendLoginCode({
|
|
account: account.value,
|
|
channel: verifyType.value,
|
|
});
|
|
if (res?.code === 200) {
|
|
ElMessage.success("验证码已发送");
|
|
startCodeCountdown();
|
|
} else {
|
|
errorMsg.value = res?.msg || "发送验证码失败";
|
|
}
|
|
} finally {
|
|
codeSending.value = false;
|
|
}
|
|
};
|
|
|
|
// --- 初始化极验4.0 ---
|
|
const startGeetest4 = async () => {
|
|
showCaptchaContainer.value = true;
|
|
await nextTick();
|
|
|
|
if (!captchaContainer.value) {
|
|
errorMsg.value = "验证码容器未找到";
|
|
loading.value = false;
|
|
// 验证码容器未找到,跳过验证直接登录
|
|
return await performLoginRequest();
|
|
}
|
|
|
|
// 获取极验4.0配置
|
|
const res4 = await getGeetest4Infos();
|
|
if (!res4 || res4.code !== 200 || !res4.data || !res4.data.captcha_id) {
|
|
errorMsg.value = "获取极验配置失败,跳过验证直接登录";
|
|
showCaptchaContainer.value = false;
|
|
// 配置获取失败,跳过验证直接登录
|
|
return await performLoginRequest();
|
|
}
|
|
|
|
const config = res4.data;
|
|
|
|
// @ts-ignore
|
|
if (!window.initGeetest4) {
|
|
errorMsg.value = "极验4.0 SDK 加载失败,跳过验证直接登录";
|
|
loading.value = false;
|
|
showCaptchaContainer.value = false;
|
|
return await performLoginRequest();
|
|
}
|
|
|
|
// @ts-ignore
|
|
window.initGeetest4(
|
|
{
|
|
captchaId: config.captcha_id,
|
|
product: "bind",
|
|
language: "zh-CN",
|
|
container: captchaContainer.value,
|
|
},
|
|
(instance) => {
|
|
captchaInstance.value = instance;
|
|
|
|
// 验证成功回调
|
|
instance.onSuccess(async () => {
|
|
const result = instance.getValidate();
|
|
// 将验证结果添加到登录参数中
|
|
const loginRes = await login({
|
|
account: account.value,
|
|
password: password.value,
|
|
captcha_id: result?.captcha_id || "",
|
|
lot_number: result?.lot_number || "",
|
|
pass_token: result?.pass_token || "",
|
|
gen_time: result?.gen_time || "",
|
|
captcha_output: result?.captcha_output || "",
|
|
});
|
|
|
|
if (loginRes && loginRes.code === 200) {
|
|
if (rememberMe.value) {
|
|
localStorage.setItem("loginAccount", account.value);
|
|
localStorage.setItem("loginPassword", password.value);
|
|
localStorage.setItem("loginRememberMe", "true");
|
|
} else {
|
|
localStorage.removeItem("loginAccount");
|
|
localStorage.removeItem("loginTenantName");
|
|
localStorage.removeItem("loginPassword");
|
|
localStorage.setItem("loginRememberMe", "false");
|
|
}
|
|
|
|
authStore.setLoginInfo(loginRes.data);
|
|
const { useTabsStore } = await import("@/stores");
|
|
const tabsStore = useTabsStore();
|
|
tabsStore.resetTabs();
|
|
router.push({ path: "/home" });
|
|
ElMessage.success("登录成功!");
|
|
} else {
|
|
errorMsg.value = loginRes.msg || "登录失败";
|
|
}
|
|
loading.value = false;
|
|
cleanCaptchaInstance();
|
|
});
|
|
|
|
// 验证失败回调
|
|
instance.onFail(() => {
|
|
errorMsg.value = "验证码验证失败,请重试";
|
|
loading.value = false;
|
|
});
|
|
|
|
// 错误回调 - 网络错误时跳过验证直接登录
|
|
instance.onError((err) => {
|
|
errorMsg.value = "验证码加载失败,跳过验证直接登录";
|
|
loading.value = false;
|
|
cleanCaptchaInstance();
|
|
// 跳过验证直接登录
|
|
performLoginRequest();
|
|
});
|
|
|
|
// 显示验证码
|
|
instance.showCaptcha();
|
|
},
|
|
);
|
|
return true;
|
|
};
|
|
|
|
/**
|
|
* 登录请求
|
|
*/
|
|
const handleLogin = async () => {
|
|
// 清空错误提示
|
|
errorMsg.value = "";
|
|
|
|
// 表单验证
|
|
if (!account.value.trim()) {
|
|
errorMsg.value = "请输入用户名";
|
|
return;
|
|
}
|
|
if (!password.value.trim()) {
|
|
errorMsg.value = "请输入密码";
|
|
return;
|
|
}
|
|
if (loading.value) return;
|
|
loading.value = true;
|
|
|
|
try {
|
|
const verifyRes = await getOpenVerify();
|
|
if (verifyRes?.code === 200 && Array.isArray(verifyRes?.data)) {
|
|
const openItem = verifyRes.data.find((i) => i.label === "openVerify");
|
|
const typeItem = verifyRes.data.find((i) => i.label === "verifyType");
|
|
openVerifyEnabled.value = String(openItem?.value || "0") === "1";
|
|
verifyType.value = typeItem?.value || "captcha";
|
|
}
|
|
|
|
if (
|
|
openVerifyEnabled.value &&
|
|
(verifyType.value === "sms" || verifyType.value === "email")
|
|
) {
|
|
if (!verifyCode.value.trim()) {
|
|
errorMsg.value = "请输入验证码";
|
|
return;
|
|
}
|
|
}
|
|
if (openVerifyEnabled.value && verifyType.value === "captcha") {
|
|
if (!captchaInput.value.trim()) {
|
|
errorMsg.value = "请输入验证码";
|
|
return;
|
|
}
|
|
if (captchaInput.value.trim() !== captchaText.value) {
|
|
errorMsg.value = "验证码错误";
|
|
generateCaptcha();
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (!openVerifyEnabled.value) {
|
|
await performLoginRequest();
|
|
} else if (verifyType.value === "geetest") {
|
|
await startGeetest4();
|
|
} else {
|
|
await performLoginRequest();
|
|
}
|
|
} catch (err) {
|
|
errorMsg.value =
|
|
err?.response?.data?.msg || err?.message || "登录失败,请重试";
|
|
} finally {
|
|
loading.value = false;
|
|
}
|
|
};
|
|
|
|
// 记住我逻辑优化
|
|
const handleRememberMeChange = async () => {
|
|
if (rememberMe.value) {
|
|
try {
|
|
await ElMessageBox.confirm(
|
|
"请确认电脑环境是可信的,登录成功后会自动记住密码。",
|
|
"安全提示",
|
|
{
|
|
confirmButtonText: "确定",
|
|
cancelButtonText: "取消",
|
|
type: "warning",
|
|
closeOnClickModal: false,
|
|
},
|
|
);
|
|
} catch {
|
|
rememberMe.value = false;
|
|
}
|
|
}
|
|
};
|
|
|
|
// 页面加载处理
|
|
onMounted(() => {
|
|
// 从本地存储恢复表单
|
|
const savedRemember = localStorage.getItem("loginRememberMe");
|
|
if (savedRemember === "true") {
|
|
account.value = localStorage.getItem("loginAccount") || "";
|
|
password.value = localStorage.getItem("loginPassword") || "";
|
|
rememberMe.value = true;
|
|
}
|
|
generateCaptcha();
|
|
getVerifyInfos()
|
|
.then((res) => {
|
|
if (res?.code === 200 && res?.data?.use_geetest) {
|
|
verifyType.value = res.data.use_geetest;
|
|
openVerifyEnabled.value =
|
|
Number(res.data.openVerify_enabled ?? 1) === 1;
|
|
} else {
|
|
// 兼容旧接口
|
|
return getOpenVerify().then((verifyRes) => {
|
|
if (verifyRes?.code === 200 && Array.isArray(verifyRes?.data)) {
|
|
const openItem = verifyRes.data.find(
|
|
(i) => i.label === "openVerify",
|
|
);
|
|
if (openItem?.value !== undefined) {
|
|
openVerifyEnabled.value = String(openItem.value) === "1";
|
|
}
|
|
const typeItem = verifyRes.data.find(
|
|
(i) => i.label === "verifyType",
|
|
);
|
|
if (typeItem?.value) verifyType.value = typeItem.value;
|
|
}
|
|
});
|
|
}
|
|
})
|
|
.catch(() => {});
|
|
});
|
|
|
|
const goRegister = () => router.push("/register");
|
|
const goForget = () => router.push("/forget");
|
|
|
|
const clearCache = async () => {
|
|
try {
|
|
await ElMessageBox.confirm("确定要清除本地缓存吗?", "提示", {
|
|
confirmButtonText: "确定",
|
|
cancelButtonText: "取消",
|
|
type: "warning",
|
|
});
|
|
localStorage.clear();
|
|
sessionStorage.clear();
|
|
ElMessage.success("缓存已清除");
|
|
// 刷新页面重新初始化
|
|
window.location.reload();
|
|
} catch {
|
|
// 用户取消,不做任何操作
|
|
}
|
|
};
|
|
</script>
|
|
|
|
<style scoped>
|
|
.login-bg {
|
|
min-height: 100vh;
|
|
width: 100%;
|
|
padding: 24px;
|
|
box-sizing: border-box;
|
|
background: linear-gradient(120deg, #e6f0ff 0%, #f5fcff 55%, #eaf6ff 100%);
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
position: relative;
|
|
overflow: hidden;
|
|
}
|
|
|
|
.login-card {
|
|
display: flex;
|
|
width: min(100%, 920px);
|
|
max-width: 920px;
|
|
background: rgba(255, 255, 255, 0.95);
|
|
border-radius: 22px;
|
|
box-shadow:
|
|
0 8px 36px 0 rgba(73, 150, 255, 0.14),
|
|
0 1.5px 4px 0 rgba(30, 42, 79, 0.05);
|
|
overflow: hidden;
|
|
z-index: 10;
|
|
}
|
|
|
|
.login-side {
|
|
width: 320px;
|
|
background: #52a8ff;
|
|
display: flex;
|
|
flex-direction: column;
|
|
align-items: center;
|
|
padding: 44px 16px 32px 16px;
|
|
box-shadow: 4px 0 32px 0 rgba(189, 231, 255, 0.13) inset;
|
|
position: relative;
|
|
}
|
|
|
|
.brand {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 14px;
|
|
margin-bottom: 42px;
|
|
user-select: none;
|
|
}
|
|
|
|
.brand-title {
|
|
font-size: 25px;
|
|
letter-spacing: 2px;
|
|
font-weight: 700;
|
|
color: #fff;
|
|
}
|
|
|
|
.illus {
|
|
margin-top: 30px;
|
|
user-select: none;
|
|
opacity: 0.95;
|
|
}
|
|
|
|
/* 版权信息样式 */
|
|
.copyright {
|
|
width: 100%;
|
|
text-align: center;
|
|
font-size: 13px;
|
|
color: #fff;
|
|
margin-top: auto;
|
|
margin-bottom: 2px;
|
|
letter-spacing: 0.2px;
|
|
padding-top: 25px;
|
|
user-select: none;
|
|
}
|
|
|
|
.login-panel {
|
|
flex: 1;
|
|
padding: 52px 54px 48px 54px;
|
|
display: flex;
|
|
flex-direction: column;
|
|
justify-content: center;
|
|
background: transparent;
|
|
min-width: 0;
|
|
}
|
|
|
|
.login-title {
|
|
margin: 0 0 8px 0;
|
|
font-size: 28px;
|
|
font-weight: 600;
|
|
color: #2560a9;
|
|
text-align: left;
|
|
letter-spacing: 1px;
|
|
}
|
|
|
|
.login-desc {
|
|
color: #7391c4;
|
|
font-size: 15px;
|
|
margin-bottom: 28px;
|
|
letter-spacing: 0.2px;
|
|
}
|
|
|
|
.mode-switch {
|
|
display: flex;
|
|
gap: 8px;
|
|
margin-bottom: 14px;
|
|
}
|
|
|
|
.mode-btn {
|
|
flex: 1;
|
|
padding: 8px 10px;
|
|
border: 1px solid #d6e6fa;
|
|
border-radius: 7px;
|
|
background: #f7fbfe;
|
|
color: #407ad6;
|
|
cursor: pointer;
|
|
}
|
|
|
|
.mode-btn.active {
|
|
background: #e8f3ff;
|
|
border-color: #7cb8ff;
|
|
color: #1d63c2;
|
|
font-weight: 600;
|
|
}
|
|
|
|
.form-group {
|
|
margin-bottom: 15px;
|
|
position: relative;
|
|
}
|
|
|
|
/* 输入框前置图标样式 */
|
|
.icon-input-group {
|
|
display: flex;
|
|
align-items: center;
|
|
position: relative;
|
|
}
|
|
|
|
.input-with-icon {
|
|
padding-left: 36px !important;
|
|
}
|
|
|
|
.input-icon {
|
|
position: absolute;
|
|
left: 10px;
|
|
top: 50%;
|
|
transform: translateY(-50%);
|
|
display: flex;
|
|
align-items: center;
|
|
z-index: 2;
|
|
width: 20px;
|
|
height: 20px;
|
|
color: #4da1ff;
|
|
opacity: 0.95;
|
|
}
|
|
|
|
.visible-btn {
|
|
position: absolute;
|
|
right: 11px;
|
|
top: 50%;
|
|
transform: translateY(-50%);
|
|
cursor: pointer;
|
|
z-index: 2;
|
|
padding: 2px 2px;
|
|
display: flex;
|
|
align-items: center;
|
|
opacity: 0.82;
|
|
user-select: none;
|
|
}
|
|
|
|
.visible-btn:hover {
|
|
opacity: 1;
|
|
}
|
|
|
|
/* 防止密码输入框可见按钮和输入内容重叠 */
|
|
.icon-input-group .input-with-icon {
|
|
padding-right: 34px;
|
|
}
|
|
|
|
.input {
|
|
width: 100%;
|
|
padding: 12px 14px;
|
|
font-size: 16px;
|
|
border: 1.3px solid #d6e6fa;
|
|
border-radius: 7px;
|
|
box-sizing: border-box;
|
|
transition:
|
|
border 0.2s,
|
|
box-shadow 0.2s;
|
|
background: #f7fbfe;
|
|
margin-bottom: 3px;
|
|
}
|
|
|
|
.input:focus {
|
|
border-color: #4da1ff;
|
|
outline: none;
|
|
box-shadow: 0 0 0 2px #e3f2ffb1;
|
|
}
|
|
|
|
.code-row {
|
|
display: flex;
|
|
gap: 8px;
|
|
align-items: center;
|
|
}
|
|
|
|
.code-input {
|
|
margin-bottom: 0;
|
|
flex: 1;
|
|
}
|
|
|
|
.code-btn {
|
|
width: 120px;
|
|
height: 44px;
|
|
border: 1.3px solid #d6e6fa;
|
|
border-radius: 7px;
|
|
background: #fff;
|
|
color: #3a78ca;
|
|
cursor: pointer;
|
|
}
|
|
|
|
.code-btn:disabled {
|
|
opacity: 0.65;
|
|
cursor: not-allowed;
|
|
}
|
|
|
|
/* 极验验证码容器样式 */
|
|
.geetest-container {
|
|
margin: 10px 0;
|
|
padding: 5px 0;
|
|
width: 100%;
|
|
min-height: 60px;
|
|
border-radius: 7px;
|
|
background: #f7fbfe;
|
|
border: 1.3px solid #d6e6fa;
|
|
}
|
|
|
|
/* 记住我单选框 */
|
|
.remember-me-row {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: space-between;
|
|
margin-bottom: 0;
|
|
margin-top: 3px;
|
|
min-height: 32px;
|
|
font-size: 15px;
|
|
color: #6d8eb8;
|
|
user-select: none;
|
|
}
|
|
|
|
.remember-me-label {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 7px;
|
|
cursor: pointer;
|
|
}
|
|
|
|
.remember-me-checkbox {
|
|
width: 16px;
|
|
height: 16px;
|
|
accent-color: #4da1ff;
|
|
margin-right: 2px;
|
|
}
|
|
|
|
.login-btn {
|
|
width: 100%;
|
|
padding: 13px 0;
|
|
font-size: 17px;
|
|
background: linear-gradient(90deg, #3494e6 0%, #52a8ff 100%);
|
|
color: #fff;
|
|
border: none;
|
|
border-radius: 7px;
|
|
box-shadow: 0 2px 12px 0 rgba(81, 173, 255, 0.13);
|
|
font-weight: 600;
|
|
letter-spacing: 1px;
|
|
margin-top: 15px;
|
|
cursor: pointer;
|
|
transition:
|
|
background 0.2s,
|
|
transform 0.13s;
|
|
}
|
|
|
|
.login-btn:active {
|
|
transform: scale(0.98);
|
|
}
|
|
|
|
.login-btn:disabled {
|
|
background: #b6dafc;
|
|
cursor: not-allowed;
|
|
}
|
|
|
|
.error-msg {
|
|
color: #e4574a;
|
|
background: #fdeceb;
|
|
text-align: center;
|
|
border-radius: 4px;
|
|
padding: 7px 4px;
|
|
margin-bottom: 2px;
|
|
font-size: 14.5px;
|
|
letter-spacing: 0.3px;
|
|
animation: shake 0.28s;
|
|
}
|
|
|
|
@keyframes shake {
|
|
0% {
|
|
transform: translateX(0);
|
|
}
|
|
|
|
20% {
|
|
transform: translateX(-6px);
|
|
}
|
|
|
|
40% {
|
|
transform: translateX(6px);
|
|
}
|
|
|
|
60% {
|
|
transform: translateX(-2px);
|
|
}
|
|
|
|
80% {
|
|
transform: translateX(2px);
|
|
}
|
|
|
|
100% {
|
|
transform: translateX(0);
|
|
}
|
|
}
|
|
|
|
/* 注册、忘记密码链接 */
|
|
.action-links {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: flex-end;
|
|
font-size: 14.1px;
|
|
color: #6592c3;
|
|
margin-bottom: 0;
|
|
min-height: 22px;
|
|
}
|
|
|
|
.action-links a {
|
|
cursor: pointer;
|
|
color: #407ad6;
|
|
text-decoration: none;
|
|
transition: color 0.16s;
|
|
}
|
|
|
|
.action-links a:hover {
|
|
color: #165eec;
|
|
text-decoration: underline;
|
|
}
|
|
|
|
.action-links .divider {
|
|
color: #bbd3ee;
|
|
margin: 0 6px;
|
|
font-size: 13px;
|
|
}
|
|
|
|
.register-link {
|
|
margin-right: 0px;
|
|
}
|
|
|
|
.forget-link {
|
|
margin-left: 0px;
|
|
}
|
|
|
|
/* 渐隐提示 */
|
|
.fade-enter-active,
|
|
.fade-leave-active {
|
|
transition: opacity 0.24s;
|
|
}
|
|
|
|
.fade-enter-from,
|
|
.fade-leave-to {
|
|
opacity: 0;
|
|
}
|
|
|
|
/* 炫彩光斑装饰 */
|
|
.login-light {
|
|
position: absolute;
|
|
border-radius: 50%;
|
|
pointer-events: none;
|
|
filter: blur(45px);
|
|
opacity: 0.4;
|
|
z-index: 1;
|
|
}
|
|
|
|
.light1 {
|
|
width: 340px;
|
|
height: 340px;
|
|
top: -90px;
|
|
left: -60px;
|
|
background: radial-gradient(circle at 60% 50%, #55b7f988 0%, #e1e8fa11 95%);
|
|
}
|
|
|
|
.light2 {
|
|
width: 260px;
|
|
height: 260px;
|
|
right: -60px;
|
|
bottom: -90px;
|
|
background: radial-gradient(circle at 55% 60%, #f3e7ff99 0%, #daf3ff10 100%);
|
|
}
|
|
|
|
@media (max-width: 940px) {
|
|
.login-bg {
|
|
padding: 16px;
|
|
}
|
|
|
|
.login-card {
|
|
width: 100%;
|
|
flex-direction: column;
|
|
}
|
|
|
|
.login-side {
|
|
width: 100%;
|
|
min-width: 0;
|
|
padding: 32px 18px 24px;
|
|
border-radius: 0;
|
|
}
|
|
|
|
.brand {
|
|
margin-bottom: 24px;
|
|
}
|
|
|
|
.brand-title {
|
|
font-size: 22px;
|
|
}
|
|
|
|
.illus {
|
|
margin-top: 8px;
|
|
}
|
|
|
|
.login-panel {
|
|
padding: 30px 22px 34px 22px;
|
|
}
|
|
|
|
.copyright {
|
|
padding-top: 13px;
|
|
}
|
|
}
|
|
|
|
@media (max-width: 640px) {
|
|
.login-bg {
|
|
padding: 12px;
|
|
align-items: stretch;
|
|
}
|
|
|
|
.login-card {
|
|
min-height: calc(100vh - 24px);
|
|
border-radius: 18px;
|
|
}
|
|
|
|
.login-side {
|
|
padding: 24px 16px 20px;
|
|
}
|
|
|
|
.brand {
|
|
gap: 10px;
|
|
margin-bottom: 18px;
|
|
}
|
|
|
|
.brand-title {
|
|
font-size: 18px;
|
|
letter-spacing: 1px;
|
|
}
|
|
|
|
.login-panel {
|
|
padding: 24px 16px 28px;
|
|
}
|
|
|
|
.login-title {
|
|
font-size: 24px;
|
|
}
|
|
|
|
.login-desc {
|
|
font-size: 14px;
|
|
margin-bottom: 22px;
|
|
}
|
|
|
|
.input,
|
|
.code-btn,
|
|
.login-btn {
|
|
min-height: 44px;
|
|
}
|
|
|
|
.remember-me-row {
|
|
flex-direction: column;
|
|
align-items: flex-start;
|
|
gap: 10px;
|
|
}
|
|
|
|
.action-links {
|
|
width: 100%;
|
|
justify-content: flex-start;
|
|
flex-wrap: wrap;
|
|
line-height: 1.6;
|
|
}
|
|
}
|
|
|
|
@media (max-width: 420px) {
|
|
.login-bg {
|
|
padding: 0;
|
|
}
|
|
|
|
.login-card {
|
|
min-height: 100vh;
|
|
border-radius: 0;
|
|
}
|
|
|
|
.login-side {
|
|
padding: 20px 14px 18px;
|
|
}
|
|
|
|
.login-panel {
|
|
padding: 20px 14px 24px;
|
|
}
|
|
}
|
|
</style>
|