From f6f5a8af7e6c5359e13d7c75172adb0bc6cac555 Mon Sep 17 00:00:00 2001
From: hyb <kk_huangyangbo@163.com>
Date: Fri, 23 Jan 2026 01:53:11 +0000
Subject: [PATCH] feat: 添加用户登录功能、添加注册功能、注册审批功能,参考:注册审批功能测试指南.md、驱动代码模块增加插入其他项目代码功能,驱动代码模块增加插入其他项目代码功能,优化测试报告模块前端布局和美化 - 实现超级用户管理系统用户,可进行增删改查,其他用户只可查看本人的数据,不可进行增删改 - 将Django后台管理中的用户配置相关功能迁移到系统中进行适配、可配置禁用、是否管理员、是否超级用户 - 增加用户管理的分组字段、可直接进行分组操作,实现用户管理即可通过分组配置实现各项目的访问权限 - 实现登录页面注册 - 实现管理员首页点击注册用户审批可实现通过不通过 - 实现未审核用户的登录提示与审核未通过的用户登录提示 - 实现一键插入其他本人可访问项目的脚本代码
---
测试组/Test_platform/Interface_automation/frontend/src/pages/login/Login.vue | 822 ++++++++++++++++++++++++++++++++++++++++++++++++++++------
1 files changed, 733 insertions(+), 89 deletions(-)
diff --git "a/\346\265\213\350\257\225\347\273\204/Test_platform/Interface_automation/frontend/src/pages/login/Login.vue" "b/\346\265\213\350\257\225\347\273\204/Test_platform/Interface_automation/frontend/src/pages/login/Login.vue"
index 61d620c..49e1f77 100644
--- "a/\346\265\213\350\257\225\347\273\204/Test_platform/Interface_automation/frontend/src/pages/login/Login.vue"
+++ "b/\346\265\213\350\257\225\347\273\204/Test_platform/Interface_automation/frontend/src/pages/login/Login.vue"
@@ -1,64 +1,223 @@
<template>
<div class="login-container">
-<!-- <div class="login-background">-->
-<!-- <div class="gradient-overlay"></div>-->
-<!-- </div>-->
+ <div class="background-animation">
+ <div class="floating-shapes">
+ <div class="shape shape-1"></div>
+ <div class="shape shape-2"></div>
+ <div class="shape shape-3"></div>
+ <div class="shape shape-4"></div>
+ </div>
+ <div class="network-lines"></div>
+ <div class="gradient-overlay"></div>
+ </div>
- <el-card class="login-card">
+ <div class="login-content">
<div class="brand-section">
- <img
- src="~@/assets/images/img.png"
- class="brand-logo"
- alt="系统logo"
- />
- <h1 class="system-name">接口自动化平台</h1>
+ <div class="brand-logo-container">
+ <div class="logo-icon">
+ <svg viewBox="0 0 100 100" class="api-icon">
+ <path d="M20,30 L50,10 L80,30 L80,70 L50,90 L20,70 Z" fill="rgba(0,212,255,0.1)" stroke="rgba(0,212,255,0.6)" stroke-width="2"/>
+ <path d="M30,40 L50,25 L70,40 L70,60 L50,75 L30,60 Z" fill="rgba(148,0,211,0.1)" stroke="rgba(148,0,211,0.6)" stroke-width="1.5"/>
+ <path d="M40,50 L50,40 L60,50 L60,55 L50,60 L40,55 Z" fill="rgba(255,255,255,0.2)" stroke="rgba(255,255,255,0.8)" stroke-width="1"/>
+ <line x1="35" y1="45" x2="45" y2="52" stroke="#00d4ff" stroke-width="1.5" stroke-dasharray="2,2"/>
+ <line x1="55" y1="45" x2="65" y2="52" stroke="#9400d3" stroke-width="1.5" stroke-dasharray="2,2"/>
+ <circle cx="50" cy="50" r="4" fill="#00d4ff">
+ <animate attributeName="r" values="4;6;4" dur="2s" repeatCount="indefinite"/>
+ <animate attributeName="opacity" values="1;0.7;1" dur="2s" repeatCount="indefinite"/>
+ </circle>
+ <circle cx="35" cy="35" r="2" fill="#00d4ff">
+ <animate attributeName="opacity" values="0.5;1;0.5" dur="1.5s" repeatCount="indefinite"/>
+ </circle>
+ <circle cx="65" cy="35" r="2" fill="#9400d3">
+ <animate attributeName="opacity" values="0.5;1;0.5" dur="1.5s" repeatCount="indefinite" begin="0.5s"/>
+ </circle>
+ <circle cx="35" cy="65" r="2" fill="#9400d3">
+ <animate attributeName="opacity" values="0.5;1;0.5" dur="1.5s" repeatCount="indefinite" begin="1s"/>
+ </circle>
+ <circle cx="65" cy="65" r="2" fill="#00d4ff">
+ <animate attributeName="opacity" values="0.5;1;0.5" dur="1.5s" repeatCount="indefinite" begin="1.5s"/>
+ </circle>
+ </svg>
+ </div>
+ <h1 class="system-name">APITest Pro</h1>
+ <p class="system-desc">智能接口自动化测试平台</p>
+ </div>
+
+ <div class="feature-list">
+ <div class="feature-item">
+ <span class="feature-icon">🚀</span>
+ <span>高效自动化测试</span>
+ </div>
+ <div class="feature-item">
+ <span class="feature-icon">🔗</span>
+ <span>智能接口管理</span>
+ </div>
+ <div class="feature-item">
+ <span class="feature-icon">📊</span>
+ <span>实时数据监控</span>
+ </div>
+ </div>
</div>
- <el-form
- ref="loginForm"
- @submit.native.prevent="submitForm"
- class="login-form"
- >
- <el-form-item>
- <el-input
- v-model="loginForm.username"
- placeholder="账号"
- prefix-icon="el-icon-user-solid"
- class="custom-input"
- :class="{ 'input-error': usernameInvalid }"
- @blur="validateUserName"
- />
- <transition name="el-zoom-in-top">
- <div v-if="usernameInvalid" class="error-msg">{{ usernameInvalid }}</div>
- </transition>
- </el-form-item>
+ <div class="login-form-section">
+ <div class="form-container">
+ <div class="form-header">
+ <h2>{{ isLoginMode ? '欢迎登录' : '用户注册' }}</h2>
+ <p>{{ isLoginMode ? '请使用您的账号密码登录系统' : '请填写以下信息完成注册' }}</p>
+ </div>
- <el-form-item>
- <el-input
- v-model="loginForm.password"
- type="password"
- placeholder="密码"
- prefix-icon="el-icon-lock"
- show-password
- class="custom-input"
- :class="{ 'input-error': passwordInvalid }"
- @blur="validatePassword"
- />
- <transition name="el-zoom-in-top">
- <div v-if="passwordInvalid" class="error-msg">{{ passwordInvalid }}</div>
- </transition>
- </el-form-item>
+ <el-form
+ ref="loginForm"
+ @submit.native.prevent="submitForm"
+ class="login-form"
+ >
+ <div class="input-group" v-if="!isLoginMode">
+ <label class="input-label">用户名</label>
+ <el-input
+ v-model="registerForm.username"
+ placeholder="请输入用户名"
+ prefix-icon="el-icon-user"
+ class="modern-input"
+ :class="{ 'input-error': registerErrors.username }"
+ @blur="validateRegisterField('username')"
+ />
+ <transition name="slide-fade">
+ <div v-if="registerErrors.username" class="error-msg">{{ registerErrors.username }}</div>
+ </transition>
+ </div>
- <el-button
- type="primary"
- class="login-button"
- :loading="isLoading"
- @click="submitForm"
- >
- {{ isLoading ? '登录中...' : '立即登录' }}
- </el-button>
- </el-form>
- </el-card>
+ <div class="input-group" v-if="!isLoginMode">
+ <label class="input-label">姓名</label>
+ <el-input
+ v-model="registerForm.name"
+ placeholder="请输入姓名"
+ prefix-icon="el-icon-edit"
+ class="modern-input"
+ :class="{ 'input-error': registerErrors.name }"
+ @blur="validateRegisterField('name')"
+ />
+ <transition name="slide-fade">
+ <div v-if="registerErrors.name" class="error-msg">{{ registerErrors.name }}</div>
+ </transition>
+ </div>
+
+ <div class="input-group" v-if="!isLoginMode">
+ <label class="input-label">手机号</label>
+ <el-input
+ v-model="registerForm.phone"
+ placeholder="请输入手机号"
+ prefix-icon="el-icon-phone"
+ class="modern-input"
+ :class="{ 'input-error': registerErrors.phone }"
+ @blur="validateRegisterField('phone')"
+ />
+ <transition name="slide-fade">
+ <div v-if="registerErrors.phone" class="error-msg">{{ registerErrors.phone }}</div>
+ </transition>
+ </div>
+
+ <div class="input-group" v-if="!isLoginMode">
+ <label class="input-label">所属分组(可选)</label>
+ <el-select
+ v-model="registerForm.groups"
+ multiple
+ placeholder="请选择分组"
+ class="modern-input"
+ style="width: 100%"
+ >
+ <el-option
+ v-for="group in groupList"
+ :key="group.id"
+ :label="group.name"
+ :value="group.id"
+ ></el-option>
+ </el-select>
+ </div>
+
+ <div class="input-group" v-if="!isLoginMode">
+ <label class="input-label">密码</label>
+ <el-input
+ v-model="registerForm.password"
+ type="password"
+ placeholder="请输入密码"
+ prefix-icon="el-icon-lock"
+ show-password
+ class="modern-input"
+ :class="{ 'input-error': registerErrors.password }"
+ @blur="validateRegisterField('password')"
+ />
+ <transition name="slide-fade">
+ <div v-if="registerErrors.password" class="error-msg">{{ registerErrors.password }}</div>
+ </transition>
+ </div>
+
+ <div class="input-group" v-if="!isLoginMode">
+ <label class="input-label">确认密码</label>
+ <el-input
+ v-model="registerForm.confirm_password"
+ type="password"
+ placeholder="请再次输入密码"
+ prefix-icon="el-icon-lock"
+ show-password
+ class="modern-input"
+ :class="{ 'input-error': registerErrors.confirm_password }"
+ @blur="validateRegisterField('confirm_password')"
+ />
+ <transition name="slide-fade">
+ <div v-if="registerErrors.confirm_password" class="error-msg">{{ registerErrors.confirm_password }}</div>
+ </transition>
+ </div>
+
+ <div class="input-group" v-if="isLoginMode">
+ <label class="input-label">用户名</label>
+ <el-input
+ v-model="loginForm.username"
+ placeholder="请输入用户名"
+ prefix-icon="el-icon-user"
+ class="modern-input"
+ :class="{ 'input-error': usernameInvalid }"
+ @blur="validateUserName"
+ />
+ <transition name="slide-fade">
+ <div v-if="usernameInvalid" class="error-msg">{{ usernameInvalid }}</div>
+ </transition>
+ </div>
+
+ <div class="input-group" v-if="isLoginMode">
+ <label class="input-label">密码</label>
+ <el-input
+ v-model="loginForm.password"
+ type="password"
+ placeholder="请输入密码"
+ prefix-icon="el-icon-lock"
+ show-password
+ class="modern-input"
+ :class="{ 'input-error': passwordInvalid }"
+ @blur="validatePassword"
+ />
+ <transition name="slide-fade">
+ <div v-if="passwordInvalid" class="error-msg">{{ passwordInvalid }}</div>
+ </transition>
+ </div>
+
+ <el-button
+ type="primary"
+ class="login-button"
+ :loading="isLoading"
+ @click="submitForm"
+ >
+ {{ isLoading ? (isLoginMode ? '登录中...' : '注册中...') : (isLoginMode ? '登录系统' : '注册账号') }}
+ </el-button>
+ </el-form>
+
+ <div class="form-footer">
+ <p v-if="isLoginMode">还没有账号?<a href="#" class="register-link" @click.prevent="switchMode">立即注册</a></p>
+ <p v-else>已有账号?<a href="#" class="register-link" @click.prevent="switchMode">返回登录</a></p>
+ <p class="copyright">© 2025 APITest Pro 智能接口自动化平台</p>
+ </div>
+ </div>
+ </div>
+ </div>
</div>
</template>
@@ -69,16 +228,63 @@
data() {
return {
isLoading: false,
+ isLoginMode: true,
loginForm: {
username: "",
password: ""
},
+ registerForm: {
+ username: "",
+ name: "",
+ phone: "",
+ password: "",
+ confirm_password: "",
+ groups: []
+ },
usernameInvalid: "",
- passwordInvalid: ""
+ passwordInvalid: "",
+ registerErrors: {
+ username: "",
+ name: "",
+ phone: "",
+ password: "",
+ confirm_password: ""
+ },
+ groupList: []
};
},
methods: {
+ switchMode() {
+ this.isLoginMode = !this.isLoginMode;
+ this.resetForms();
+ if (!this.isLoginMode) {
+ this.getGroupList();
+ }
+ },
+ resetForms() {
+ this.loginForm = {
+ username: "",
+ password: ""
+ };
+ this.registerForm = {
+ username: "",
+ name: "",
+ phone: "",
+ password: "",
+ confirm_password: "",
+ groups: []
+ };
+ this.usernameInvalid = "";
+ this.passwordInvalid = "";
+ this.registerErrors = {
+ username: "",
+ name: "",
+ phone: "",
+ password: "",
+ confirm_password: ""
+ };
+ },
validateUserName() {
if (this.loginForm.username.replace(/(^\s*)/g, "") === "") {
this.usernameInvalid = "用户名不能为空";
@@ -93,18 +299,83 @@
}
return true;
},
+ validateRegisterField(field) {
+ const form = this.registerForm;
+ let isValid = true;
+
+ switch (field) {
+ case 'username':
+ if (!form.username) {
+ this.registerErrors.username = "用户名不能为空";
+ isValid = false;
+ } else {
+ this.registerErrors.username = "";
+ }
+ break;
+ case 'name':
+ if (!form.name) {
+ this.registerErrors.name = "姓名不能为空";
+ isValid = false;
+ } else {
+ this.registerErrors.name = "";
+ }
+ break;
+ case 'phone':
+ if (!form.phone) {
+ this.registerErrors.phone = "手机号不能为空";
+ isValid = false;
+ } else if (!/^1[3-9]\d{9}$/.test(form.phone)) {
+ this.registerErrors.phone = "手机号格式不正确";
+ isValid = false;
+ } else {
+ this.registerErrors.phone = "";
+ }
+ break;
+ case 'password':
+ if (!form.password) {
+ this.registerErrors.password = "密码不能为空";
+ isValid = false;
+ } else if (form.password.length < 6) {
+ this.registerErrors.password = "密码长度不能少于6位";
+ isValid = false;
+ } else {
+ this.registerErrors.password = "";
+ }
+ break;
+ case 'confirm_password':
+ if (!form.confirm_password) {
+ this.registerErrors.confirm_password = "请再次输入密码";
+ isValid = false;
+ } else if (form.password !== form.confirm_password) {
+ this.registerErrors.confirm_password = "两次输入的密码不一致";
+ isValid = false;
+ } else {
+ this.registerErrors.confirm_password = "";
+ }
+ break;
+ }
+ return isValid;
+ },
+ validateRegisterForm() {
+ let isValid = true;
+ const fields = ['username', 'name', 'phone', 'password', 'confirm_password'];
+ fields.forEach(field => {
+ if (!this.validateRegisterField(field)) {
+ isValid = false;
+ }
+ });
+ return isValid;
+ },
handleLoginSuccess(resp) {
if (resp.success) {
- // 显示登录信息提示
this.showLoginNotification(resp);
-
- // 原有路由跳转和存储逻辑
this.$router.push({ name: "ProjectList" });
this.$store.commit("isLogin", resp.token);
this.$store.commit("setUser", resp.user);
this.$store.commit("setName", resp.name);
this.$store.commit("setId", resp.id);
this.$store.commit("setIsSuperuser", resp.is_superuser);
+ this.$store.commit("setIsStaff", resp.is_staff);
this.$store.commit("setRouterName", "ProjectList");
this.$store.commit("setShowHosts", resp.show_hosts);
@@ -113,6 +384,7 @@
this.setLocalValue("name", resp.name);
this.setLocalValue("id", resp.id);
this.setLocalValue("is_superuser", resp.is_superuser);
+ this.setLocalValue("is_staff", resp.is_staff);
this.setLocalValue("routerName", "ProjectList");
this.setLocalValue("show_hosts", resp.show_hosts);
} else {
@@ -123,10 +395,10 @@
});
}
},
-
+
showLoginNotification(resp) {
const currentTime = new Date().toLocaleString();
-
+
this.$notify({
title: '登录成功',
message: `登录时间: ${currentTime}`,
@@ -136,14 +408,68 @@
});
},
submitForm() {
- if (this.validateUserName() && this.validatePassword()) {
- this.isLoading = true;
- this.$api.login(this.loginForm).then(resp => {
- this.handleLoginSuccess(resp);
- this.isLoading = false;
- });
+ if (this.isLoginMode) {
+ if (this.validateUserName() && this.validatePassword()) {
+ this.isLoading = true;
+ this.$api.login(this.loginForm).then(resp => {
+ this.handleLoginSuccess(resp);
+ this.isLoading = false;
+ }).catch(error => {
+ this.isLoading = false;
+ if (error.response && error.response.data && error.response.data.msg) {
+ this.$message.error({
+ message: error.response.data.msg,
+ duration: 2000,
+ center: true
+ });
+ }
+ });
+ }
+ } else {
+ if (this.validateRegisterForm()) {
+ this.isLoading = true;
+ this.$api.register(this.registerForm).then(resp => {
+ this.$message.success(resp.msg);
+ this.isLoading = false;
+ this.switchMode();
+ }).catch(error => {
+ this.isLoading = false;
+ if (error.response && error.response.data && error.response.data.msg) {
+ this.$message.error({
+ message: error.response.data.msg,
+ duration: 2000,
+ center: true
+ });
+ } else if (error.response && error.response.data) {
+ const errors = error.response.data;
+ let errorMsg = '';
+ for (let key in errors) {
+ if (Array.isArray(errors[key])) {
+ errorMsg += errors[key].join('、') + ';';
+ } else if (typeof errors[key] === 'string') {
+ errorMsg += errors[key] + ';';
+ }
+ }
+ this.$message.error({
+ message: errorMsg || '注册失败,请检查输入信息',
+ duration: 3000,
+ center: true
+ });
+ }
+ });
+ }
}
+ },
+ getGroupList() {
+ this.$api.getGroupList().then(resp => {
+ this.groupList = resp;
+ }).catch(() => {
+ this.groupList = [];
+ });
}
+ },
+ mounted() {
+ this.getGroupList();
}
};
</script>
@@ -151,55 +477,373 @@
<style scoped>
.login-container {
min-height: 100vh;
- display: flex;
- justify-content: center;
- align-items: center;
- background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
+ background: linear-gradient(135deg, #0c0e27 0%, #1a1f3d 50%, #2d1b69 100%);
position: relative;
overflow: hidden;
+ font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif;
}
-.login-background {
+/* 动态背景动画 */
+.background-animation {
position: absolute;
width: 100%;
height: 100%;
- background: rgba(255, 255, 255, 0.1);
- opacity: 0.1;
+ top: 0;
+ left: 0;
+}
+
+.floating-shapes {
+ position: absolute;
+ width: 100%;
+ height: 100%;
+}
+
+.shape {
+ position: absolute;
+ border-radius: 50%;
+ background: linear-gradient(45deg, rgba(0, 212, 255, 0.1), rgba(148, 0, 211, 0.1));
+ animation: float 20s infinite linear;
+}
+
+.shape-1 {
+ width: 100px;
+ height: 100px;
+ top: 10%;
+ left: 10%;
+ animation-delay: 0s;
+}
+
+.shape-2 {
+ width: 150px;
+ height: 150px;
+ top: 60%;
+ right: 10%;
+ animation-delay: -5s;
+}
+
+.shape-3 {
+ width: 80px;
+ height: 80px;
+ bottom: 20%;
+ left: 20%;
+ animation-delay: -10s;
+}
+
+.shape-4 {
+ width: 120px;
+ height: 120px;
+ top: 30%;
+ right: 30%;
+ animation-delay: -15s;
+}
+
+.network-lines {
+ position: absolute;
+ width: 100%;
+ height: 100%;
+ background-image:
+ linear-gradient(90deg, transparent 24px, rgba(0, 212, 255, 0.03) 25px, rgba(0, 212, 255, 0.03) 26px, transparent 27px, transparent 74px, rgba(148, 0, 211, 0.03) 75px, rgba(148, 0, 211, 0.03) 76px, transparent 77px),
+ linear-gradient(0deg, transparent 24px, rgba(0, 212, 255, 0.03) 25px, rgba(0, 212, 255, 0.03) 26px, transparent 27px, transparent 74px, rgba(148, 0, 211, 0.03) 75px, rgba(148, 0, 211, 0.03) 76px, transparent 77px);
+ background-size: 100px 100px;
+ animation: gridMove 40s linear infinite;
}
.gradient-overlay {
position: absolute;
width: 100%;
height: 100%;
- background: linear-gradient(135deg, rgba(102, 126, 234, 0.2) 0%, rgba(118, 75, 162, 0.2) 100%);
+ background: radial-gradient(circle at 20% 80%, rgba(0, 212, 255, 0.1) 0%, transparent 50%),
+ radial-gradient(circle at 80% 20%, rgba(148, 0, 211, 0.1) 0%, transparent 50%);
}
-.login-card {
- width: 420px;
- border-radius: 12px;
- box-shadow: 0 12px 24px rgba(0, 0, 0, 0.15);
- border: none;
- z-index: 1;
- background: rgba(255, 255, 255, 0.95);
+/* 主内容区域 */
+.login-content {
+ display: flex;
+ min-height: 100vh;
+ max-width: 1400px;
+ margin: 0 auto;
+ position: relative;
+ z-index: 2;
}
+/* 左侧品牌区域 */
.brand-section {
+ flex: 1;
+ display: flex;
+ flex-direction: column;
+ justify-content: center;
+ padding: 0 4rem;
+ color: white;
+}
+
+.brand-logo-container {
+ margin-bottom: 3rem;
+}
+
+.logo-icon {
+ width: 120px;
+ height: 120px;
+ margin-bottom: 1.5rem;
+ filter: drop-shadow(0 0 20px rgba(0, 212, 255, 0.5));
+}
+
+.api-icon {
+ width: 100%;
+ height: 100%;
+ animation: pulse 3s ease-in-out infinite;
+}
+
+.system-name {
+ font-size: 3rem;
+ font-weight: 700;
+ margin-bottom: 0.5rem;
+ background: linear-gradient(135deg, #00d4ff 0%, #9400d3 100%);
+ -webkit-background-clip: text;
+ -webkit-text-fill-color: transparent;
+ background-clip: text;
+}
+
+.system-desc {
+ font-size: 1.2rem;
+ opacity: 0.8;
+ margin: 0;
+}
+
+.feature-list {
+ margin-top: 2rem;
+}
+
+.feature-item {
+ display: flex;
+ align-items: center;
+ margin-bottom: 1rem;
+ font-size: 1.1rem;
+ opacity: 0.9;
+ transition: opacity 0.3s ease;
+}
+
+.feature-item:hover {
+ opacity: 1;
+}
+
+.feature-icon {
+ font-size: 1.5rem;
+ margin-right: 0.8rem;
+ filter: drop-shadow(0 0 10px rgba(0, 212, 255, 0.5));
+}
+
+/* 右侧表单区域 */
+.login-form-section {
+ flex: 0 0 500px;
+ display: flex;
+ align-items: center;
+ justify-content: center;
+ padding: 2rem;
+}
+
+.form-container {
+ background: rgba(255, 255, 255, 0.95);
+ backdrop-filter: blur(20px);
+ border-radius: 20px;
+ padding: 3rem 2.5rem;
+ width: 100%;
+ max-width: 400px;
+ box-shadow: 0 25px 50px rgba(0, 0, 0, 0.3);
+ border: 1px solid rgba(255, 255, 255, 0.2);
+}
+
+.form-header {
text-align: center;
- margin-bottom: 2rem;
+ margin-bottom: 2.5rem;
}
-.brand-logo {
- width: 120px; /* 尺寸放大50% */
- height: auto;
- margin-bottom: 1.5rem; /* 增加下边距保持间距 */
- transition: transform 0.3s ease; /* 添加悬停动效 */
+.form-header h2 {
+ font-size: 2rem;
+ font-weight: 600;
+ margin-bottom: 0.5rem;
+ color: #1a1f3d;
}
-/* 可选悬停效果 */
-.brand-logo:hover {
- transform: scale(1.05);
+.form-header p {
+ color: #666;
+ margin: 0;
}
+.input-group {
+ margin-bottom: 1.5rem;
+}
+
+.input-label {
+ display: block;
+ margin-bottom: 0.5rem;
+ font-weight: 500;
+ color: #333;
+ font-size: 0.9rem;
+}
+
+.modern-input {
+ width: 100%;
+}
+
+.modern-input >>> .el-input__inner {
+ height: 48px;
+ border-radius: 12px;
+ border: 2px solid #e1e5e9;
+ font-size: 1rem;
+ transition: all 0.3s ease;
+ background: #f8f9fa;
+}
+
+.modern-input >>> .el-input__inner:focus {
+ border-color: #00d4ff;
+ box-shadow: 0 0 0 3px rgba(0, 212, 255, 0.1);
+ background: white;
+}
+
+.modern-input.input-error >>> .el-input__inner {
+ border-color: #ff4757;
+}
+
+.login-button {
+ width: 100%;
+ height: 50px;
+ border-radius: 12px;
+ font-size: 1.1rem;
+ font-weight: 600;
+ background: linear-gradient(135deg, #00d4ff 0%, #9400d3 100%);
+ border: none;
+ transition: all 0.3s ease;
+ margin-top: 0.5rem;
+}
+
+.login-button:hover {
+ transform: translateY(-2px);
+ box-shadow: 0 10px 25px rgba(0, 212, 255, 0.4);
+}
+
+.login-button:active {
+ transform: translateY(0);
+}
+
+.error-msg {
+ color: #ff4757;
+ font-size: 0.85rem;
+ margin-top: 0.3rem;
+ animation: slideIn 0.3s ease;
+}
+
+.form-footer {
+ margin-top: 2rem;
+ text-align: center;
+}
+
+.form-footer p {
+ margin: 0.5rem 0;
+ color: #666;
+ font-size: 0.9rem;
+}
+
+.register-link {
+ color: #00d4ff;
+ text-decoration: none;
+ font-weight: 500;
+}
+
+.register-link:hover {
+ text-decoration: underline;
+}
+
+.copyright {
+ opacity: 0.6;
+ font-size: 0.8rem;
+}
+
+/* 动画效果 */
+@keyframes float {
+ 0%, 100% {
+ transform: translateY(0px) rotate(0deg);
+ }
+ 33% {
+ transform: translateY(-20px) rotate(120deg);
+ }
+ 66% {
+ transform: translateY(10px) rotate(240deg);
+ }
+}
+
+@keyframes gridMove {
+ 0% {
+ transform: translate(0, 0);
+ }
+ 100% {
+ transform: translate(100px, 100px);
+ }
+}
+
+@keyframes pulse {
+ 0%, 100% {
+ transform: scale(1);
+ opacity: 1;
+ }
+ 50% {
+ transform: scale(1.05);
+ opacity: 0.8;
+ }
+}
+
+@keyframes slideIn {
+ from {
+ opacity: 0;
+ transform: translateY(-10px);
+ }
+ to {
+ opacity: 1;
+ transform: translateY(0);
+ }
+}
+
+.slide-fade-enter-active {
+ transition: all 0.3s ease;
+}
+
+.slide-fade-leave-active {
+ transition: all 0.3s cubic-bezier(1, 0.5, 0.8, 1);
+}
+
+.slide-fade-enter, .slide-fade-leave-to {
+ transform: translateY(-10px);
+ opacity: 0;
+}
+
+/* 响应式设计 */
+@media (max-width: 1024px) {
+ .login-content {
+ flex-direction: column;
+ }
+
+ .brand-section {
+ padding: 2rem;
+ text-align: center;
+ }
+
+ .login-form-section {
+ flex: none;
+ width: 100%;
+ }
+}
+
+@media (max-width: 768px) {
+ .form-container {
+ margin: 1rem;
+ padding: 2rem 1.5rem;
+ }
+
+ .system-name {
+ font-size: 2.5rem;
+ }
+}
+
+
.system-name {
font-size: 2rem; /* 同步放大系统名称字号 */
margin-top: 0.5rem; /* 增加与LOGO的间距 */
--
Gitblit v1.9.1