新增内容: 1. 添加 AGENTS.md 和 CLAUDE.md AI 助手配置文件 2. 添加安全修复说明文档 (doc/security-fixes.md) 3. 新增单元测试用例: - fund-common: TenantContextHolderTest, UserContextHolderTest, PageResultTest, ResultTest - fund-sys: AuthServiceImplTest, RoleServiceImplTest, TenantServiceImplTest 修改内容: 1. 数据库初始化脚本更新 (fund_sys_init.sql) 2. 前端依赖更新 (package.json) 3. 登录和密码管理功能优化: - 管理后台和移动端登录页面 - 密码修改功能 4. 租户上下文处理优化 (TenantLineHandlerImpl) 5. 网关过滤器增强: - TenantGatewayFilter 租户过滤 - TokenAuthFilter 认证过滤 6. Controller 层代码优化 7. DTO 和 Service 层代码改进 技术改进: - 密码加密方式从 BCrypt 改为 MD5(前后端一致) - 登录验证流程优化,支持多租户 - 增加日志输出便于调试 - 代码规范性和可维护性提升
149 lines
3.3 KiB
Vue
149 lines
3.3 KiB
Vue
<template>
|
|
<div class="login-container">
|
|
<div class="login-card">
|
|
<div class="login-header">
|
|
<img src="/vite.svg" alt="Logo" class="logo" />
|
|
<h1>资金服务平台</h1>
|
|
<p>FundPlatform</p>
|
|
</div>
|
|
<el-form ref="formRef" :model="form" :rules="rules" class="login-form">
|
|
<el-form-item prop="username">
|
|
<el-input
|
|
v-model="form.username"
|
|
placeholder="请输入用户名"
|
|
prefix-icon="User"
|
|
size="large"
|
|
/>
|
|
</el-form-item>
|
|
<el-form-item prop="password">
|
|
<el-input
|
|
v-model="form.password"
|
|
type="password"
|
|
placeholder="请输入密码"
|
|
prefix-icon="Lock"
|
|
size="large"
|
|
show-password
|
|
@keyup.enter="handleLogin"
|
|
/>
|
|
</el-form-item>
|
|
<el-form-item>
|
|
<el-checkbox v-model="rememberMe">记住我</el-checkbox>
|
|
</el-form-item>
|
|
<el-form-item>
|
|
<el-button
|
|
type="primary"
|
|
size="large"
|
|
:loading="loading"
|
|
class="login-btn"
|
|
@click="handleLogin"
|
|
>
|
|
登录
|
|
</el-button>
|
|
</el-form-item>
|
|
</el-form>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ref, reactive } from 'vue'
|
|
import { useRouter, useRoute } from 'vue-router'
|
|
import { ElMessage } from 'element-plus'
|
|
import type { FormInstance, FormRules } from 'element-plus'
|
|
import { useUserStore } from '@/stores/user'
|
|
import md5 from 'js-md5'
|
|
|
|
const router = useRouter()
|
|
const route = useRoute()
|
|
const userStore = useUserStore()
|
|
|
|
const formRef = ref<FormInstance>()
|
|
const loading = ref(false)
|
|
const rememberMe = ref(false)
|
|
|
|
const form = reactive({
|
|
username: '',
|
|
password: ''
|
|
})
|
|
|
|
const rules: FormRules = {
|
|
username: [{ required: true, message: '请输入用户名', trigger: 'blur' }],
|
|
password: [{ required: true, message: '请输入密码', trigger: 'blur' }]
|
|
}
|
|
|
|
const handleLogin = async () => {
|
|
if (!formRef.value) return
|
|
|
|
try {
|
|
await formRef.value.validate()
|
|
loading.value = true
|
|
|
|
// 密码先进行 MD5 加密
|
|
const encryptedPassword = md5(form.password)
|
|
|
|
await userStore.loginAction(form.username, encryptedPassword)
|
|
|
|
// 获取用户信息
|
|
await userStore.fetchUserInfo()
|
|
|
|
ElMessage.success('登录成功')
|
|
|
|
// 跳转到目标页面或首页
|
|
const redirect = (route.query.redirect as string) || '/dashboard'
|
|
router.push(redirect)
|
|
} catch (error: any) {
|
|
ElMessage.error(error.message || '登录失败')
|
|
} finally {
|
|
loading.value = false
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<style scoped>
|
|
.login-container {
|
|
min-height: 100vh;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
|
|
}
|
|
|
|
.login-card {
|
|
width: 400px;
|
|
padding: 40px;
|
|
background: #fff;
|
|
border-radius: 8px;
|
|
box-shadow: 0 10px 40px rgba(0, 0, 0, 0.2);
|
|
}
|
|
|
|
.login-header {
|
|
text-align: center;
|
|
margin-bottom: 30px;
|
|
}
|
|
|
|
.logo {
|
|
width: 60px;
|
|
height: 60px;
|
|
margin-bottom: 15px;
|
|
}
|
|
|
|
.login-header h1 {
|
|
font-size: 24px;
|
|
color: #333;
|
|
margin: 0 0 8px;
|
|
}
|
|
|
|
.login-header p {
|
|
color: #999;
|
|
margin: 0;
|
|
}
|
|
|
|
.login-form {
|
|
width: 100%;
|
|
}
|
|
|
|
.login-btn {
|
|
width: 100%;
|
|
}
|
|
</style>
|