Luxx/dashboard/src/views/AuthView.vue

117 lines
5.1 KiB
Vue

<template>
<div class="auth-page">
<div class="auth-card">
<div class="tabs">
<button @click="mode = 'login'" :class="{ active: mode === 'login' }">登录</button>
<button @click="mode = 'register'" :class="{ active: mode === 'register' }">注册</button>
</div>
<form @submit.prevent="handleSubmit">
<div class="form-group">
<label>用户名</label>
<input v-model="form.username" type="text" placeholder="请输入用户名" required />
</div>
<div v-if="mode === 'register'" class="form-group">
<label>邮箱</label>
<input v-model="form.email" type="email" placeholder="请输入邮箱" required />
</div>
<div class="form-group">
<label>密码</label>
<input v-model="form.password" type="password" placeholder="请输入密码" required />
</div>
<div v-if="mode === 'register'" class="form-group">
<label>确认密码</label>
<input v-model="confirmPassword" type="password" placeholder="请再次输入密码" required />
</div>
<div v-if="error" class="error-msg">{{ error }}</div>
<button type="submit" :disabled="loading" class="btn-submit">
<span v-if="loading" class="spinner"></span>
<span v-else>{{ mode === 'login' ? '登录' : '注册' }}</span>
</button>
</form>
<p class="switch-mode">
{{ mode === 'login' ? '没有账户?' : '已有账户?' }}
<button @click="mode = mode === 'login' ? 'register' : 'login'">
{{ mode === 'login' ? '立即注册' : '立即登录' }}
</button>
</p>
</div>
</div>
</template>
<script setup>
import { ref, reactive } from 'vue'
import { useRouter } from 'vue-router'
import { authAPI } from '../services/api.js'
import { useAuth } from '../composables/useAuth.js'
const router = useRouter()
const { login } = useAuth()
const mode = ref('login')
const loading = ref(false)
const error = ref('')
const confirmPassword = ref('')
const form = reactive({ username: '', email: '', password: '' })
const handleSubmit = async () => {
error.value = ''
loading.value = true
try {
if (mode.value === 'register' && form.password !== confirmPassword.value) {
throw new Error('两次密码不一致')
}
const api = mode.value === 'login' ? authAPI.login : authAPI.register
const response = await api(mode.value === 'login'
? { username: form.username, password: form.password }
: form
)
if (response.success) {
if (mode.value === 'login' && response.data?.access_token) {
login(response.data.access_token, response.data.user)
router.push('/')
} else {
mode.value = 'login'
form.username = form.password = form.email = ''
confirmPassword.value = ''
}
} else {
throw new Error(response.message || '操作失败')
}
} catch (err) {
error.value = err.message || '操作失败,请重试'
} finally {
loading.value = false
}
}
</script>
<style scoped>
.auth-page { min-height: calc(100vh - 70px); display: flex; align-items: center; justify-content: center; padding: 2rem; }
.auth-card { width: 100%; max-width: 400px; }
.tabs { display: flex; margin-bottom: 2rem; background: var(--code-bg); border-radius: 12px; padding: 0.25rem; }
.tabs button { flex: 1; padding: 0.75rem; background: transparent; border: none; border-radius: 8px; font-size: 1rem; font-weight: 500; color: var(--text); cursor: pointer; transition: all 0.2s; }
.tabs button.active { background: var(--bg); color: var(--text-h); box-shadow: 0 2px 8px var(--shadow); }
.form-group { margin-bottom: 1.25rem; }
.form-group label { display: block; margin-bottom: 0.5rem; font-weight: 500; color: var(--text-h); }
.form-group input { width: 100%; padding: 0.875rem 1rem; border: 1px solid var(--border); border-radius: 10px; font-size: 1rem; background: var(--bg); color: var(--text); box-sizing: border-box; }
.form-group input:focus { outline: none; border-color: var(--accent); box-shadow: 0 0 0 3px var(--accent-bg); }
.error-msg { padding: 0.75rem 1rem; background: #fef2f2; border: 1px solid #fecaca; border-radius: 8px; color: #dc2626; font-size: 0.9rem; margin-bottom: 1rem; }
.btn-submit { width: 100%; padding: 1rem; background: var(--accent); color: white; border: none; border-radius: 10px; font-size: 1.1rem; font-weight: 600; cursor: pointer; transition: all 0.2s; display: flex; align-items: center; justify-content: center; }
.btn-submit:hover:not(:disabled) { background: var(--accent-border); }
.btn-submit:disabled { opacity: 0.7; cursor: not-allowed; }
.spinner { width: 20px; height: 20px; border: 2px solid rgba(255,255,255,0.3); border-top-color: white; border-radius: 50%; animation: spin 1s linear infinite; }
@keyframes spin { to { transform: rotate(360deg); } }
.switch-mode { text-align: center; color: var(--text); margin-top: 1.5rem; }
.switch-mode button { background: none; border: none; color: var(--accent); cursor: pointer; font-weight: 500; text-decoration: underline; }
</style>