You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
276 lines
5.8 KiB
276 lines
5.8 KiB
<script setup lang="ts">
|
|
import { ref } from 'vue'
|
|
import { useRouter } from 'vue-router'
|
|
import { useAuthStore } from '@/stores/auth'
|
|
import Particles from '@/components/Particles/Particles.vue'
|
|
import GlareHover from '@/components/GlareHover/GlareHover.vue'
|
|
import BlurText from '@/components/BlurText/BlurText.vue'
|
|
|
|
const router = useRouter()
|
|
const authStore = useAuthStore()
|
|
|
|
const username = ref('')
|
|
const password = ref('')
|
|
const errorMsg = ref('')
|
|
const isLoading = ref(false)
|
|
|
|
const handleAnimationComplete = () => {
|
|
console.log('All animations complete!');
|
|
};
|
|
|
|
const handleLogin = async () => {
|
|
errorMsg.value = ''
|
|
|
|
if (!username.value || !password.value) {
|
|
errorMsg.value = '请输入用户名和密码'
|
|
return
|
|
}
|
|
|
|
isLoading.value = true
|
|
|
|
try {
|
|
const success = await authStore.login({
|
|
name: username.value,
|
|
password: password.value
|
|
})
|
|
|
|
if (success) {
|
|
router.push('/character')
|
|
} else {
|
|
errorMsg.value = '用户名或密码错误'
|
|
}
|
|
} catch (error: any) {
|
|
errorMsg.value = error?.message || '登录失败,请稍后重试'
|
|
} finally {
|
|
isLoading.value = false
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<template>
|
|
<div class="login-page">
|
|
<Particles :particle-count="150" :particle-colors="['#ffffff', '#cccccc', '#aaaaaa']" class="particles-bg" />
|
|
|
|
<div class="login-container">
|
|
<div class="login-card">
|
|
<div class="login-header">
|
|
<BlurText text="Build Own God" animate-by="words" direction="top" :delay="200" class="blur-title"
|
|
class-name="text-2xl font-semibold text-center" @animation-complete="handleAnimationComplete" />
|
|
<p class="subtitle">从零开始·创造神明</p>
|
|
</div>
|
|
|
|
<form class="login-form" @submit.prevent="handleLogin">
|
|
<div class="form-group">
|
|
<label class="form-label">用户名</label>
|
|
<input v-model="username" type="text" class="form-input" placeholder="请输入用户名" autocomplete="username" />
|
|
</div>
|
|
|
|
<div class="form-group">
|
|
<label class="form-label">密码</label>
|
|
<input v-model="password" type="password" class="form-input" placeholder="请输入密码"
|
|
autocomplete="current-password" />
|
|
</div>
|
|
|
|
<div v-if="errorMsg" class="error-message">
|
|
{{ errorMsg }}
|
|
</div>
|
|
|
|
<GlareHover width="100%" height="50px" background="transparent" border-radius="12px"
|
|
border-color="rgba(255,255,255,0.15)" glare-color="#ffffff" :glare-opacity="0.15" class="login-button"
|
|
@click="handleLogin">
|
|
<span v-if="isLoading" class="loading-text">登录中...</span>
|
|
<span v-else class="button-text">登 录</span>
|
|
</GlareHover>
|
|
</form>
|
|
|
|
<div class="login-footer">
|
|
<span class="footer-text">还没有账号?</span>
|
|
<router-link to="/register" class="footer-link">
|
|
立即注册
|
|
</router-link>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<style scoped>
|
|
.login-page {
|
|
min-height: 100vh;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
background: #000000;
|
|
position: relative;
|
|
overflow: hidden;
|
|
}
|
|
|
|
.particles-bg {
|
|
position: fixed !important;
|
|
top: 0;
|
|
left: 0;
|
|
width: 100% !important;
|
|
height: 100% !important;
|
|
}
|
|
|
|
.login-container {
|
|
position: relative;
|
|
z-index: 10;
|
|
width: 100%;
|
|
max-width: 380px;
|
|
padding: 20px;
|
|
}
|
|
|
|
.login-card {
|
|
background: rgba(255, 255, 255, 0.03);
|
|
backdrop-filter: blur(20px);
|
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
|
border-radius: 20px;
|
|
padding: 36px;
|
|
box-shadow: 0 25px 50px -12px rgba(0, 0, 0, 0.8);
|
|
}
|
|
|
|
.login-header {
|
|
text-align: center;
|
|
margin-bottom: 32px;
|
|
}
|
|
|
|
.blur-title {
|
|
justify-content: center;
|
|
/* 水平居中 flex 项目 */
|
|
text-align: center;
|
|
/* 文字居中 */
|
|
color: white;
|
|
width: 100%;
|
|
/* 确保占满宽度 */
|
|
}
|
|
|
|
.title-glow {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
cursor: default;
|
|
}
|
|
|
|
.title-text {
|
|
font-size: 1.75rem;
|
|
font-weight: 300;
|
|
letter-spacing: 0.2em;
|
|
color: #ffffff;
|
|
text-shadow: 0 0 20px rgba(255, 255, 255, 0.3);
|
|
}
|
|
|
|
.subtitle {
|
|
color: #666666;
|
|
font-size: 0.875rem;
|
|
letter-spacing: 0.1em;
|
|
margin-top: 8px;
|
|
}
|
|
|
|
.login-form {
|
|
display: flex;
|
|
flex-direction: column;
|
|
gap: 20px;
|
|
}
|
|
|
|
.form-group {
|
|
display: flex;
|
|
flex-direction: column;
|
|
gap: 8px;
|
|
}
|
|
|
|
.form-label {
|
|
color: #888888;
|
|
font-size: 0.75rem;
|
|
font-weight: 400;
|
|
letter-spacing: 0.1em;
|
|
text-transform: uppercase;
|
|
}
|
|
|
|
.form-input {
|
|
width: 100%;
|
|
padding: 14px 16px;
|
|
background: rgba(255, 255, 255, 0.03);
|
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
|
border-radius: 10px;
|
|
color: #ffffff;
|
|
font-size: 0.95rem;
|
|
transition: all 0.3s ease;
|
|
outline: none;
|
|
}
|
|
|
|
.form-input::placeholder {
|
|
color: #444444;
|
|
}
|
|
|
|
.form-input:focus {
|
|
border-color: rgba(255, 255, 255, 0.2);
|
|
background: rgba(255, 255, 255, 0.05);
|
|
box-shadow: 0 0 20px rgba(255, 255, 255, 0.05);
|
|
}
|
|
|
|
.error-message {
|
|
color: #888888;
|
|
font-size: 0.8rem;
|
|
text-align: center;
|
|
padding: 10px;
|
|
background: rgba(255, 255, 255, 0.03);
|
|
border-radius: 8px;
|
|
}
|
|
|
|
.login-button {
|
|
margin-top: 8px;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
cursor: pointer;
|
|
}
|
|
|
|
.button-text,
|
|
.loading-text {
|
|
color: #ffffff;
|
|
font-size: 0.9rem;
|
|
font-weight: 400;
|
|
letter-spacing: 0.2em;
|
|
}
|
|
|
|
.login-footer {
|
|
margin-top: 28px;
|
|
text-align: center;
|
|
display: flex;
|
|
justify-content: center;
|
|
gap: 8px;
|
|
}
|
|
|
|
.footer-text {
|
|
color: #444444;
|
|
font-size: 0.8rem;
|
|
}
|
|
|
|
.footer-link {
|
|
color: #888888;
|
|
font-size: 0.8rem;
|
|
text-decoration: none;
|
|
transition: color 0.3s ease;
|
|
}
|
|
|
|
.footer-link:hover {
|
|
color: #ffffff;
|
|
}
|
|
|
|
/* Mobile responsive */
|
|
@media (max-width: 480px) {
|
|
.login-card {
|
|
padding: 28px 20px;
|
|
}
|
|
|
|
.title {
|
|
font-size: 1.5rem;
|
|
}
|
|
|
|
.form-input {
|
|
padding: 12px 14px;
|
|
font-size: 16px;
|
|
}
|
|
}
|
|
</style>
|
|
|