feat: 完成情绪博物馆项目重构和功能增强 - 新增日记评论和帖子功能 - 重构前端架构,优化用户体验 - 完善WebSocket通信机制 - 更新项目文档和部署配置

This commit is contained in:
2025-07-27 10:05:59 +08:00
parent 6903ac1c0d
commit cc886cd4d5
126 changed files with 21179 additions and 15734 deletions
+378 -270
View File
@@ -1,328 +1,436 @@
<template>
<div class="login-page">
<div class="login-container">
<div class="login-card">
<!-- Logo和标题 -->
<div class="login-header">
<router-link to="/" class="logo">
<span class="logo-text">开心APP</span>
</router-link>
<h1 class="login-title">欢迎回来</h1>
<p class="login-subtitle">登录您的账户继续与开开的对话</p>
<div class="container mx-auto px-4 py-8">
<div class="max-w-md mx-auto card">
<div class="text-center mb-8">
<h1 class="text-3xl font-bold text-gray-900 mb-2">登录</h1>
<p class="text-gray-600">欢迎回到情绪博物馆</p>
</div>
<!-- 登录表单 -->
<a-form
<el-form
ref="loginFormRef"
:model="loginForm"
:rules="loginRules"
@finish="handleLogin"
@finishFailed="handleLoginFailed"
layout="vertical"
class="login-form"
label-width="80px"
@submit.prevent="handleLogin"
>
<a-form-item label="账号" name="account">
<a-input
v-model:value="loginForm.account"
placeholder="请输入手机号或邮箱"
size="large"
:prefix="h(UserOutlined)"
<el-form-item label="账号" prop="account">
<el-input
v-model="loginForm.account"
placeholder="请输入账号/邮箱/手机号"
:prefix-icon="User"
clearable
@keyup.enter="handleLogin"
/>
</a-form-item>
</el-form-item>
<a-form-item label="密码" name="password">
<a-input-password
v-model:value="loginForm.password"
<el-form-item label="密码" prop="password">
<el-input
v-model="loginForm.password"
type="password"
placeholder="请输入密码"
size="large"
:prefix="h(LockOutlined)"
:prefix-icon="Lock"
show-password
clearable
@keyup.enter="handleLogin"
/>
</a-form-item>
</el-form-item>
<!-- 验证码 -->
<a-form-item label="验证码" name="captcha">
<div class="captcha-container">
<a-input
v-model:value="loginForm.captcha"
<el-form-item label="验证码" prop="captcha">
<div class="flex gap-2">
<el-input
v-model="loginForm.captcha"
placeholder="请输入验证码"
size="large"
style="flex: 1"
:prefix-icon="Key"
clearable
class="flex-1"
@keyup.enter="handleLogin"
/>
<div class="captcha-image" @click="refreshCaptcha">
<div class="captcha-container" @click="refreshCaptcha">
<img
v-if="captchaImage"
:src="captchaImage"
alt="验证码"
style="width: 100%; height: 100%; cursor: pointer;"
class="captcha-image"
/>
<div v-else class="captcha-loading">
<a-spin size="small" />
<el-icon class="is-loading"><Loading /></el-icon>
</div>
</div>
</div>
<div class="captcha-tip">点击图片刷新验证码</div>
</a-form-item>
</el-form-item>
<a-form-item>
<div class="login-options">
<a-checkbox v-model:checked="loginForm.remember">记住我</a-checkbox>
<a href="#" class="forgot-password">忘记密码</a>
<el-form-item>
<div class="flex justify-between items-center mb-4">
<el-checkbox v-model="loginForm.rememberMe">
记住我
</el-checkbox>
<router-link to="/forgot-password" class="text-primary-600 hover:underline text-sm">
忘记密码
</router-link>
</div>
</a-form-item>
</el-form-item>
<a-form-item>
<a-button
<el-form-item>
<el-button
type="primary"
html-type="submit"
size="large"
:loading="loginLoading"
class="login-button"
block
class="w-full"
:loading="loading"
@click="handleLogin"
>
登录
</a-button>
</a-form-item>
</a-form>
{{ loading ? '登录中...' : '登录' }}
</el-button>
</el-form-item>
<!-- 注册链接 -->
<div class="register-link">
还没有账户
<router-link to="/register" class="register-btn">立即注册</router-link>
</el-form>
<div class="divider">
<span></span>
</div>
<div class="social-login">
<el-button class="social-btn wechat" @click="handleSocialLogin('wechat')">
<el-icon><ChatDotRound /></el-icon>
微信登录
</el-button>
<el-button class="social-btn qq" @click="handleSocialLogin('qq')">
<el-icon><User /></el-icon>
QQ登录
</el-button>
</div>
<div class="text-center mt-6">
<router-link to="/register" class="text-primary-600 hover:underline">
还没有账号立即注册
</router-link>
</div>
</div>
</div>
</div>
</template>
<script setup lang="ts">
import { ref, reactive, onMounted, nextTick, h } from 'vue'
import { useRouter } from 'vue-router'
import { message } from 'ant-design-vue'
import { UserOutlined, LockOutlined } from '@ant-design/icons-vue'
import { authService } from '@/services/auth'
import { useUserStore } from '@/stores/user'
import type { LoginRequest } from '@/types/auth'
import { ref, reactive, onMounted } from 'vue'
import { useRouter, useRoute } from 'vue-router'
import { ElMessage, ElForm } from 'element-plus'
import {
User,
Lock,
Key,
Loading,
ChatDotRound
} from '@element-plus/icons-vue'
import { useAuthStore } from '@/stores/auth'
import AuthService from '@/services/auth'
import { envConfig } from '@/config/env'
import type { LoginRequest } from '@/types/auth'
import type { FormInstance, FormRules } from 'element-plus'
const router = useRouter()
const userStore = useUserStore()
const router = useRouter()
const route = useRoute()
const authStore = useAuthStore()
// 表单数据
const loginForm = reactive<LoginRequest>({
account: '',
password: '',
captcha: '',
remember: false
})
// 表单引用
const loginFormRef = ref<FormInstance>()
// 表单验证规则
const loginRules = {
account: [
{ required: true, message: '请输入账号', trigger: 'blur' },
{ min: 3, message: '账号长度不能少于3位', trigger: 'blur' }
],
password: [
{ required: true, message: '请输入密码', trigger: 'blur' },
{ min: 6, message: '密码长度不能少于6位', trigger: 'blur' }
],
captcha: [
{ required: true, message: '请输入验证码', trigger: 'blur' },
{ len: 4, message: '验证码长度为4位', trigger: 'blur' }
]
// 加载状态
const loading = ref(false)
// 验证码相关
const captchaImage = ref('')
const captchaKey = ref('')
// 登录表单数据
const loginForm = reactive<LoginRequest>({
account: '',
password: '',
captcha: '',
captchaKey: '',
rememberMe: false
})
// 表单验证规则
const loginRules: FormRules = {
account: [
{ required: true, message: '请输入账号', trigger: 'blur' }
],
password: [
{ required: true, message: '请输入密码', trigger: 'blur' },
{ min: 6, max: 20, message: '密码长度必须在6-20位之间', trigger: 'blur' }
],
captcha: [
{ required: true, message: '请输入验证码', trigger: 'blur' },
{ min: 4, max: 6, message: '验证码长度不正确', trigger: 'blur' }
]
}
/**
* 获取验证码
*/
const getCaptcha = async () => {
try {
const response = await AuthService.getCaptcha()
// 后端返回的数据已经包含了 data:image/png;base64, 前缀,直接使用
captchaImage.value = response.captchaImage
captchaKey.value = response.captchaKey
loginForm.captchaKey = response.captchaKey
} catch (error) {
console.error('获取验证码失败:', error)
ElMessage.error('获取验证码失败')
}
}
// 状态
const loginLoading = ref(false)
const captchaImage = ref('')
const captchaKey = ref('')
/**
* 刷新验证码
*/
const refreshCaptcha = () => {
loginForm.captcha = ''
getCaptcha()
}
// 获取验证码
const getCaptcha = async () => {
try {
const response = await authService.getCaptcha()
console.log('验证码响应:', response)
captchaImage.value = response.captchaImage // 修正字段
captchaKey.value = response.captchaKey // 修正字段
console.log('验证码图片URL:', captchaImage.value.substring(0, 50) + '...')
} catch (error) {
console.error('获取验证码失败:', error)
message.error('获取验证码失败')
}
}
/**
* 处理登录
*/
const handleLogin = async () => {
if (!loginFormRef.value) return
// 刷新验证码
const refreshCaptcha = () => {
getCaptcha()
}
try {
console.log('开始登录流程...')
console.log('登录表单数据:', loginForm)
// 登录处理
const handleLogin = async (values: LoginRequest) => {
loginLoading.value = true
try {
const loginData = {
...values,
captchaKey: captchaKey.value
}
// const data = await userStore.loginWithAuth(loginData)
await userStore.loginWithAuth(loginData)
message.success('登录成功')
await nextTick()
const redirect = router.currentRoute.value.query.redirect as string
const targetPath = redirect || '/'
setTimeout(() => {
try {
router.replace(targetPath).then(() => {
console.log('路由跳转完成')
}).catch((_error) => {
window.location.href = targetPath
})
} catch (error) {
window.location.href = targetPath
}
}, 100)
} catch (error: any) {
message.error(error.message || '登录失败,请稍后重试')
// 表单验证
await loginFormRef.value.validate()
console.log('表单验证通过')
loading.value = true
// 调用登录接口
console.log('调用登录接口...')
const success = await authStore.login(loginForm)
console.log('登录结果:', success)
if (success) {
// 登录成功,确保认证状态已正确设置
console.log('登录成功,当前认证状态:', {
isLoggedIn: authStore.isLoggedIn,
hasToken: !!authStore.accessToken,
hasUserInfo: !!authStore.userInfo
})
// 跳转到目标页面或首页
const redirect = route.query.redirect as string || '/'
console.log('登录成功,跳转到:', redirect)
// 使用路由跳转而不是window.location.href,避免base路径问题
await router.push(redirect)
} else {
// 登录失败,刷新验证码
console.log('登录失败,刷新验证码')
refreshCaptcha()
} finally {
loginLoading.value = false
}
} catch (error) {
console.error('登录过程中发生错误:', error)
ElMessage.error('登录失败,请检查网络连接或稍后重试')
// 刷新验证码
refreshCaptcha()
} finally {
loading.value = false
}
}
// 登录失败处理
const handleLoginFailed = (errorInfo: any) => {
console.log('Login failed:', errorInfo)
/**
* 处理第三方登录
*/
const handleSocialLogin = (platform: 'wechat' | 'qq') => {
ElMessage.info(`${platform === 'wechat' ? '微信' : 'QQ'}登录功能开发中...`)
// TODO: 实现第三方登录逻辑
}
// 组件挂载时获取验证码
onMounted(() => {
getCaptcha()
// 如果已经登录,直接跳转
if (authStore.isLoggedIn) {
const redirect = route.query.redirect as string || '/'
router.push(redirect)
}
// 初始化
onMounted(() => {
getCaptcha()
})
})
</script>
<style scoped>
.login-page {
min-height: 100vh;
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
background-attachment: fixed;
display: flex;
align-items: center;
justify-content: center;
}
<style lang="scss" scoped>
@use "@/assets/styles/variables.scss" as *;
.login-page {
min-height: 100vh;
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
display: flex;
align-items: center;
justify-content: center;
padding: 20px;
}
.container {
background: rgba(255, 255, 255, 0.95);
backdrop-filter: blur(10px);
border-radius: 20px;
box-shadow: 0 20px 40px rgba(0, 0, 0, 0.1);
padding: 2rem;
width: 100%;
max-width: 450px;
}
.login-container {
width: 100%;
max-width: 400px;
}
.card {
background: transparent;
border: none;
box-shadow: none;
}
.login-card {
background: white;
.captcha-container {
width: 120px;
height: 40px;
border: 1px solid #dcdfe6;
border-radius: 4px;
cursor: pointer;
display: flex;
align-items: center;
justify-content: center;
background: #f5f7fa;
transition: border-color 0.3s;
}
.captcha-container:hover {
border-color: #409eff;
}
.captcha-image {
width: 100%;
height: 100%;
object-fit: cover;
border-radius: 3px;
}
.captcha-loading {
color: #909399;
font-size: 14px;
}
.divider {
position: relative;
text-align: center;
margin: 1.5rem 0;
}
.divider::before {
content: '';
position: absolute;
top: 50%;
left: 0;
right: 0;
height: 1px;
background: #e4e7ed;
}
.divider span {
background: rgba(255, 255, 255, 0.95);
padding: 0 1rem;
color: #909399;
font-size: 14px;
}
.social-login {
display: flex;
gap: 0.5rem;
}
.social-btn {
flex: 1;
border-radius: 8px;
font-weight: 500;
transition: all 0.3s;
}
.social-btn.wechat {
background: #07c160;
border-color: #07c160;
color: white;
}
.social-btn.wechat:hover {
background: #06ad56;
border-color: #06ad56;
transform: translateY(-1px);
}
.social-btn.qq {
background: #12b7f5;
border-color: #12b7f5;
color: white;
}
.social-btn.qq:hover {
background: #0ea5e9;
border-color: #0ea5e9;
transform: translateY(-1px);
}
:deep(.el-form-item__label) {
font-weight: 500;
color: #303133;
}
:deep(.el-input__wrapper) {
border-radius: 8px;
box-shadow: 0 0 0 1px #dcdfe6 inset;
transition: all 0.3s;
}
:deep(.el-input__wrapper:hover) {
box-shadow: 0 0 0 1px #c0c4cc inset;
}
:deep(.el-input__wrapper.is-focus) {
box-shadow: 0 0 0 1px #409eff inset;
}
:deep(.el-button--primary) {
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
border: none;
border-radius: 8px;
font-weight: 500;
padding: 12px 20px;
transition: all 0.3s;
}
:deep(.el-button--primary:hover) {
background: linear-gradient(135deg, #5a6fd8 0%, #6a4190 100%);
transform: translateY(-1px);
box-shadow: 0 4px 12px rgba(102, 126, 234, 0.4);
}
.text-primary-600 {
color: #667eea;
text-decoration: none;
font-weight: 500;
transition: color 0.3s;
}
.text-primary-600:hover {
color: #5a6fd8;
text-decoration: underline;
}
/* 响应式设计 */
@media (max-width: 640px) {
.container {
margin: 1rem;
padding: 1.5rem;
border-radius: 16px;
padding: 40px;
box-shadow: 0 20px 40px rgba(0, 0, 0, 0.1);
}
.login-header {
text-align: center;
margin-bottom: 32px;
.logo {
display: inline-block;
text-decoration: none;
color: #4A90E2;
font-size: 24px;
font-weight: bold;
margin-bottom: 16px;
}
.login-title {
font-size: 28px;
font-weight: bold;
color: #333;
margin-bottom: 8px;
}
.login-subtitle {
color: #888;
font-size: 14px;
margin: 0;
}
.captcha-container {
width: 100px;
height: 36px;
}
.login-form {
.captcha-container {
display: flex;
gap: 12px;
align-items: center;
}
.captcha-image {
width: 100px;
height: 40px;
border: 1px solid #d9d9d9;
border-radius: 6px;
display: flex;
align-items: center;
justify-content: center;
background: #f5f5f5;
cursor: pointer;
transition: border-color 0.3s;
&:hover {
border-color: #4A90E2;
}
}
.captcha-loading {
display: flex;
align-items: center;
justify-content: center;
}
.captcha-tip {
font-size: 12px;
color: #888;
margin-top: 4px;
}
.login-options {
display: flex;
justify-content: space-between;
align-items: center;
.forgot-password {
color: #4A90E2;
text-decoration: none;
font-size: 14px;
&:hover {
text-decoration: underline;
}
}
}
.login-button {
background: linear-gradient(135deg, #4A90E2 0%, #5BA0F2 100%);
border: none;
border-radius: 8px;
font-weight: 600;
height: 48px;
font-size: 16px;
}
.social-login {
flex-direction: column;
}
.register-link {
text-align: center;
margin-top: 24px;
color: #888;
font-size: 14px;
.register-btn {
color: #4A90E2;
text-decoration: none;
font-weight: 500;
&:hover {
text-decoration: underline;
}
}
}
</style>
}
</style>