优化玩家服务中心登录页

This commit is contained in:
Stev_Wang
2026-01-05 16:59:39 +08:00
parent 468d24c3bb
commit 7b8e6c7f7e
5 changed files with 882 additions and 64 deletions

View File

@@ -2,6 +2,7 @@ import { createApp } from 'vue'
import { createPinia } from 'pinia'
import router from './router'
import App from './App.vue'
import 'animate.css'
const app = createApp(App)

View File

@@ -1,5 +1,5 @@
<template>
<div class="login-container">
<div class="view-account">
<!-- 维护公告遮罩 -->
<div v-if="!playerServiceEnabled" class="maintenance-overlay">
<n-card title="系统维护公告" style="width: 500px; text-align: center;">
@@ -13,45 +13,141 @@
</n-card>
</div>
<!-- 登录表单 -->
<n-card v-else title="玩家登录" style="width: 400px;">
<n-form ref="formRef" :model="formValue" :rules="rules" size="large">
<n-form-item path="username" label="用户名">
<n-input v-model:value="formValue.username" placeholder="请输入用户名" />
</n-form-item>
<n-form-item path="password" label="密码">
<n-input
v-model:value="formValue.password"
type="password"
show-password-on="click"
placeholder="请输入密码"
/>
</n-form-item>
<n-form-item v-if="captchaEnabled" path="captchaCode" label="验证码">
<n-space style="width: 100%">
<div class="view-account-header"></div>
<div class="view-account-background">
<div class="line line-1"></div>
<div class="line line-2"></div>
<div class="line line-3"></div>
<div class="square square-1"></div>
<div class="square square-2"></div>
<div class="triangle"></div>
<div class="wave wave-1"></div>
<div class="wave wave-2"></div>
<div class="wave wave-3"></div>
</div>
<div class="view-account-container animate__animated animate__fadeInDown">
<div class="view-account-top">
<div class="view-account-top-logo">
<img src="/assets/account-logo-D9-eMHsk.png" alt="梦幻西游" />
</div>
<div class="view-account-top-desc">梦幻西游玩家服务中心</div>
</div>
<div class="view-account-form">
<h2 class="view-account-title">游戏账号登录</h2>
<div class="login-welcome">欢迎回来请登录您的游戏账号</div>
<n-form
ref="formRef"
label-placement="left"
size="large"
:model="formValue"
:rules="rules"
class="login-form"
>
<n-form-item path="username" class="username-item">
<n-input
v-model:value="formValue.username"
placeholder="请输入游戏账号"
class="login-input"
>
<template #prefix>
<n-icon size="18" color="#808695">
<PersonOutline />
</n-icon>
</template>
</n-input>
</n-form-item>
<n-form-item path="password" class="password-item">
<n-input
v-model:value="formValue.captchaCode"
placeholder="请输入验证码"
@keyup.enter="handleLogin"
/>
<div class="captcha-image" @click="handleRefreshCaptcha" v-html="captchaSvg"></div>
</n-space>
</n-form-item>
<n-form-item>
<n-button type="primary" block @click="handleLogin" :loading="loading">
登录
</n-button>
</n-form-item>
</n-form>
</n-card>
v-model:value="formValue.password"
type="password"
showPasswordOn="click"
placeholder="请输入密码"
class="login-input"
>
<template #prefix>
<n-icon size="18" color="#808695">
<LockClosedOutline />
</n-icon>
</template>
</n-input>
</n-form-item>
<n-form-item v-if="captchaEnabled" path="captchaCode" class="captcha-item">
<n-space style="width: 100%">
<n-input
v-model:value="formValue.captchaCode"
placeholder="请输入验证码"
class="login-input"
>
<template #prefix>
<n-icon size="18" color="#808695">
<ShieldOutline />
</n-icon>
</template>
</n-input>
<div class="captcha-image" @click="handleRefreshCaptcha" v-html="captchaSvg"></div>
</n-space>
</n-form-item>
<n-form-item class="default-color remember-forgot">
<div class="flex-between-wrapper">
<div class="left">
<n-checkbox v-model:checked="autoLogin">自动登录</n-checkbox>
</div>
<div class="right">
<a href="javascript:" class="forgot-link">忘记密码</a>
</div>
</div>
</n-form-item>
<n-form-item>
<n-button
type="primary"
@click="handleLogin"
size="large"
:loading="loading"
block
class="login-button"
>
登录
</n-button>
</n-form-item>
<n-form-item class="default-color other-item">
<div class="flex view-account-other">
<div class="flex-initial other-text">
<span>其它登录方式</span>
</div>
<!-- <div class="social-login">
<a href="javascript:" class="social-icon">
<n-icon size="24" color="#909399">
<LogoGithub />
</n-icon>
</a>
<a href="javascript:" class="social-icon">
<n-icon size="24" color="#909399">
<LogoFacebook />
</n-icon>
</a>
<a href="javascript:" class="social-icon">
<n-icon size="24" color="#909399">
<LogoWechat />
</n-icon>
</a>
</div> -->
<div class="flex-initial" style="margin-left: auto">
<a href="javascript:" class="register-link">注册账号</a>
</div>
</div>
</n-form-item>
</n-form>
</div>
</div>
</div>
</template>
<script setup lang="ts">
import { ref, onMounted } from 'vue'
import { useRouter } from 'vue-router'
import { NCard, NForm, NFormItem, NInput, NButton, NSpace, NIcon, useMessage } from 'naive-ui'
import { NCard, NForm, NFormItem, NInput, NButton, NSpace, NIcon, NCheckbox, useMessage } from 'naive-ui'
import { RiAlertLine } from '@remixicon/vue'
import { PersonOutline, LockClosedOutline, ShieldOutline } from '@vicons/ionicons5'
import { usePlayerStore } from '@/stores/player'
import { generateCaptcha, checkCaptchaEnabled, checkPlayerServiceStatus } from '@/api/player'
@@ -66,6 +162,7 @@ const captchaSvg = ref('')
const captchaId = ref('')
const playerServiceEnabled = ref(true)
const playerServiceCloseMsg = ref('玩家服务中心系统维护中')
const autoLogin = ref(true)
const formValue = ref({
username: '',
@@ -110,7 +207,7 @@ const handleLogin = async () => {
)
if (success) {
message.success('登录成功')
message.success('登录成功,即将进入系统')
router.push('/player/dashboard')
} else {
message.error('登录失败,请检查用户名和密码')
@@ -177,49 +274,532 @@ const handleRefreshStatus = async () => {
onMounted(() => {
checkCaptchaStatus()
checkServiceStatus()
setTimeout(() => {
const usernameInput = document.querySelector('input[placeholder="请输入用户名"]')
if (usernameInput) {
(usernameInput as HTMLElement).focus()
}
}, 500)
})
</script>
<style scoped>
.login-container {
<style lang="less" scoped>
.view-account {
display: flex;
justify-content: center;
align-items: center;
flex-direction: column;
height: 100vh;
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
overflow: auto;
background-color: #f0f2f5;
background: linear-gradient(140deg, #e8f1fa, #c2d9ec, #a1c3e0, #80aed3);
position: relative;
&::before {
content: '';
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-image: url('');
opacity: 0.6;
z-index: 0;
}
&::after {
content: '';
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-image: url('');
opacity: 0.8;
z-index: 0;
}
&-container {
padding: 32px 40px 20px;
max-width: 580px;
min-width: 460px;
margin: 0 auto;
background-color: #fff;
border-radius: 12px;
box-shadow: 0 15px 35px rgba(0, 0, 0, 0.1);
margin-top: 10vh;
position: relative;
backdrop-filter: blur(10px);
background: rgba(255, 255, 255, 0.95);
border: 1px solid rgba(255, 255, 255, 0.18);
transition: all 0.3s ease;
z-index: 1;
&:hover {
box-shadow: 0 20px 40px rgba(0, 0, 0, 0.15);
transform: translateY(-5px);
}
}
&-title {
text-align: center;
font-size: 22px;
font-weight: 500;
color: #333;
margin-bottom: 8px;
position: relative;
&::after {
content: '';
position: absolute;
bottom: -10px;
left: 50%;
transform: translateX(-50%);
width: 40px;
height: 2px;
background: linear-gradient(to right, #2d8cf0, #0081ff);
border-radius: 2px;
}
}
.login-welcome {
text-align: center;
font-size: 14px;
color: #606266;
margin-bottom: 30px;
margin-top: 20px;
}
&-top {
padding: 10px 0;
text-align: center;
&-logo {
margin-bottom: 8px;
display: flex;
justify-content: center;
img {
height: 60px;
}
}
&-desc {
font-size: 14px;
color: #606266;
}
}
&-other {
width: 100%;
display: flex;
align-items: center;
}
.default-color {
color: #515a6e;
:deep(.ant-checkbox-wrapper) {
color: #515a6e;
}
}
.login-button {
margin-top: 10px;
height: 42px;
font-size: 16px;
border-radius: 4px;
transition: all 0.3s;
position: relative;
overflow: hidden;
&:hover {
transform: translateY(-1px);
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.09);
}
&::after {
content: '';
position: absolute;
top: 50%;
left: 50%;
width: 5px;
height: 5px;
background: rgba(255, 255, 255, 0.5);
opacity: 0;
border-radius: 100%;
transform: scale(1, 1) translate(-50%);
transform-origin: 50% 50%;
}
&:focus:not(:active)::after {
animation: ripple 1s ease-out;
}
@keyframes ripple {
0% {
transform: scale(0, 0);
opacity: 0.5;
}
20% {
transform: scale(25, 25);
opacity: 0.3;
}
100% {
opacity: 0;
transform: scale(40, 40);
}
}
}
.remember-forgot {
margin-bottom: 5px;
.flex-between-wrapper {
display: flex;
justify-content: space-between;
align-items: center;
width: 100%;
}
.right {
text-align: right;
}
}
.forgot-link {
color: #606266;
transition: all 0.2s;
&:hover {
color: #2d8cf0;
}
}
.social-login {
display: flex;
margin-left: 16px;
}
.social-icon {
display: flex;
justify-content: center;
align-items: center;
width: 36px;
height: 36px;
border-radius: 50%;
margin-right: 12px;
transition: all 0.3s;
background-color: rgba(144, 147, 153, 0.1);
&:hover {
background-color: rgba(45, 140, 240, 0.2);
transform: scale(1.1);
:deep(svg) {
color: #2d8cf0 !important;
}
}
}
.register-link {
color: #2d8cf0;
transition: all 0.3s;
&:hover {
color: #57a3f3;
text-decoration: underline;
}
}
.login-form {
:deep(.n-form-item-feedback-wrapper) {
min-height: 18px;
}
:deep(.n-input) {
border-radius: 4px;
}
padding: 0;
}
.login-input {
:deep(.n-input__input-el) {
padding-left: 5px;
}
:deep(.n-input-wrapper) {
transition: all 0.3s ease;
}
&:hover {
:deep(.n-input-wrapper) {
box-shadow: 0 0 0 1px rgba(45, 140, 240, 0.2);
}
}
}
.username-item, .password-item, .captcha-item {
margin-bottom: 24px;
}
.captcha-image {
width: 120px;
height: 40px;
cursor: pointer;
border: 1px solid #dcdfe6;
border-radius: 4px;
display: flex;
align-items: center;
justify-content: center;
background-color: #fff;
transition: all 0.3s;
}
.captcha-image:hover {
border-color: #409eff;
box-shadow: 0 0 0 1px rgba(64, 158, 255, 0.2);
}
.captcha-image :deep(svg) {
width: 100%;
height: 100%;
}
.other-text {
padding-left: 5px;
}
.other-item {
margin-bottom: 0;
}
.maintenance-overlay {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.7);
display: flex;
justify-content: center;
align-items: center;
z-index: 1000;
}
}
.maintenance-overlay {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.7);
display: flex;
justify-content: center;
align-items: center;
z-index: 1000;
@media (min-width: 768px) {
.view-account {
background-image: url('../../assets/images/login.svg'),
radial-gradient(circle at 10% 20%, rgba(100, 149, 237, 0.25) 0%, rgba(65, 105, 225, 0.2) 40%, rgba(30, 144, 255, 0.1) 90%);
background-repeat: no-repeat;
background-position: 50%;
background-size: cover;
position: relative;
&::before {
content: '';
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: linear-gradient(135deg, rgba(45, 140, 240, 0.1), rgba(45, 140, 240, 0.05));
backdrop-filter: blur(10px);
z-index: 0;
}
&::after {
content: '';
position: absolute;
width: 100%;
height: 100%;
top: 0;
left: 0;
background-image: url('');
opacity: 0.3;
z-index: 0;
pointer-events: none;
}
&-container {
margin-top: 15vh;
z-index: 1;
position: relative;
}
}
}
.captcha-image {
width: 120px;
height: 40px;
cursor: pointer;
border: 1px solid #dcdfe6;
border-radius: 4px;
display: flex;
align-items: center;
justify-content: center;
background-color: #fff;
@media (max-height: 650px) {
.view-account-container {
margin-top: 5vh;
}
}
.captcha-image:hover {
border-color: #409eff;
}
.captcha-image :deep(svg) {
.view-account-background {
position: absolute;
width: 100%;
height: 100%;
top: 0;
left: 0;
overflow: hidden;
pointer-events: none;
z-index: 0;
.line {
position: absolute;
background: linear-gradient(90deg, rgba(45, 140, 240, 0.2), rgba(0, 129, 255, 0.1));
&-1 {
width: 300px;
height: 2px;
top: 15%;
right: 5%;
transform: rotate(-30deg);
animation: pulse 8s ease-in-out infinite;
}
&-2 {
width: 200px;
height: 2px;
bottom: 20%;
left: 10%;
transform: rotate(45deg);
animation: pulse 6s ease-in-out infinite 1s;
}
&-3 {
width: 150px;
height: 2px;
top: 40%;
left: 5%;
transform: rotate(-15deg);
animation: pulse 7s ease-in-out infinite 2s;
}
}
.square {
position: absolute;
&-1 {
width: 80px;
height: 80px;
top: 10%;
left: 15%;
background: linear-gradient(45deg, rgba(45, 140, 240, 0.15), rgba(0, 129, 255, 0.05));
transform: rotate(30deg);
animation: rotate 15s linear infinite;
}
&-2 {
width: 60px;
height: 60px;
bottom: 15%;
right: 10%;
border: 2px solid rgba(45, 140, 240, 0.1);
background: transparent;
animation: rotate 12s linear infinite reverse;
}
}
.triangle {
position: absolute;
bottom: 30%;
right: 20%;
width: 0;
height: 0;
border-left: 50px solid transparent;
border-right: 50px solid transparent;
border-bottom: 80px solid rgba(45, 140, 240, 0.08);
animation: float 10s ease-in-out infinite;
}
@keyframes pulse {
0% {
opacity: 0.3;
}
50% {
opacity: 0.6;
}
100% {
opacity: 0.3;
}
}
@keyframes rotate {
0% {
transform: rotate(0deg);
}
100% {
transform: rotate(360deg);
}
}
.wave {
position: absolute;
opacity: 0.3;
transform-origin: bottom left;
&-1 {
bottom: 0;
left: 0;
width: 100%;
height: 120px;
background: url('');
background-size: 100% 120px;
animation: wave-left-to-right 15s ease-in-out infinite;
transform: rotate(-2deg);
}
&-2 {
bottom: 0;
left: 0;
width: 100%;
height: 100px;
background: url('');
background-size: 100% 100px;
animation: wave-left-to-right 18s ease-in-out infinite;
animation-delay: -5s;
transform: rotate(-1deg);
}
&-3 {
bottom: 0;
left: 0;
width: 100%;
height: 80px;
background: url('');
background-size: 100% 80px;
animation: wave-left-to-right 20s ease-in-out infinite;
animation-delay: -2s;
}
}
@keyframes wave-left-to-right {
0% {
background-position-x: 0;
background-position-y: 100%;
}
50% {
background-position-x: 720px;
background-position-y: 50%;
}
100% {
background-position-x: 1440px;
background-position-y: 0%;
}
}
@keyframes float {
0% {
transform: translateY(0);
}
50% {
transform: translateY(-15px);
}
100% {
transform: translateY(0);
}
}
}
</style>