feat: 更新登录和注册页面,添加表单验证和样式
This commit is contained in:
@@ -1,22 +1,621 @@
|
||||
<script lang='ts' setup>
|
||||
import { ref } from "vue";
|
||||
import { useRouter } from "vue-router";
|
||||
|
||||
const router = useRouter();
|
||||
|
||||
const formData = ref({
|
||||
phoneNumber: "",
|
||||
password: "",
|
||||
confirmPassword: "",
|
||||
realName: "",
|
||||
idCard: "",
|
||||
inviteCode: "",
|
||||
});
|
||||
const agreed = ref(false);
|
||||
const showPassword = ref(false);
|
||||
const showConfirmPassword = ref(false);
|
||||
|
||||
function handleSignup() {
|
||||
// 验证手机号
|
||||
if (!formData.value.phoneNumber) {
|
||||
// TODO: 显示提示信息
|
||||
return;
|
||||
}
|
||||
|
||||
// 验证密码
|
||||
if (!formData.value.password) {
|
||||
// TODO: 提示输入密码
|
||||
return;
|
||||
}
|
||||
|
||||
// 验证确认密码
|
||||
if (formData.value.password !== formData.value.confirmPassword) {
|
||||
// TODO: 提示密码不一致
|
||||
return;
|
||||
}
|
||||
|
||||
// 验证姓名
|
||||
if (!formData.value.realName) {
|
||||
// TODO: 提示输入姓名
|
||||
return;
|
||||
}
|
||||
|
||||
// 验证身份证
|
||||
if (!formData.value.idCard) {
|
||||
// TODO: 提示输入身份证
|
||||
return;
|
||||
}
|
||||
|
||||
if (!agreed.value) {
|
||||
// TODO: 提示需要同意条款
|
||||
return;
|
||||
}
|
||||
|
||||
// TODO: 实现注册逻辑
|
||||
console.log("注册", formData.value);
|
||||
}
|
||||
|
||||
function handleLogin() {
|
||||
router.push("/auth/login");
|
||||
}
|
||||
|
||||
function goToTerms(type: "service" | "privacy") {
|
||||
router.push(`/auth/term?type=${type}`);
|
||||
}
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<ion-page>
|
||||
<ion-header>
|
||||
<ion-toolbar class="ion-toolbar">
|
||||
<back-button slot="start" />
|
||||
<ion-title>登录</ion-title>
|
||||
</ion-toolbar>
|
||||
</ion-header>
|
||||
<ion-content :fullscreen="true" class="signup-page">
|
||||
<!-- 渐变背景 -->
|
||||
<div class="gradient-bg" />
|
||||
|
||||
<ion-content :fullscreen="true" class="ion-padding">
|
||||
<div class="max-w-md mx-auto flex flex-col gap-6">
|
||||
<!-- 登录表单内容省略 -->
|
||||
<!-- 装饰元素 - 顶部飞鸟 -->
|
||||
<div class="decoration-top">
|
||||
<div class="bird bird-1" />
|
||||
<div class="bird bird-2" />
|
||||
<div class="bird bird-3" />
|
||||
</div>
|
||||
|
||||
<!-- 主要内容区域 -->
|
||||
<div class="content-wrapper">
|
||||
<!-- 顶部Logo和标题 -->
|
||||
<div class="header-section">
|
||||
<div class="emblem">
|
||||
<div class="emblem-circle">
|
||||
<div class="emblem-inner">
|
||||
<div class="star">
|
||||
★
|
||||
</div>
|
||||
<div class="emblem-text">
|
||||
中华人民共和国
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<h1 class="welcome-title">
|
||||
创建账号
|
||||
</h1>
|
||||
<h2 class="welcome-subtitle">
|
||||
加入深化改革平台
|
||||
</h2>
|
||||
</div>
|
||||
|
||||
<!-- 注册表单 -->
|
||||
<div class="form-section">
|
||||
<!-- 手机号输入 -->
|
||||
<ion-item lines="none" class="input-item">
|
||||
<ion-input
|
||||
v-model="formData.phoneNumber"
|
||||
type="tel"
|
||||
placeholder="请输入手机号"
|
||||
class="custom-input"
|
||||
:maxlength="11"
|
||||
/>
|
||||
</ion-item>
|
||||
|
||||
<!-- 密码输入 -->
|
||||
<ion-item lines="none" class="input-item">
|
||||
<ion-input
|
||||
v-model="formData.password"
|
||||
:type="showPassword ? 'text' : 'password'"
|
||||
placeholder="请输入密码"
|
||||
class="custom-input"
|
||||
/>
|
||||
<ion-icon
|
||||
slot="end"
|
||||
:icon="showPassword ? 'eye-outline' : 'eye-off-outline'"
|
||||
class="password-toggle"
|
||||
@click="showPassword = !showPassword"
|
||||
/>
|
||||
</ion-item>
|
||||
|
||||
<!-- 确认密码输入 -->
|
||||
<ion-item lines="none" class="input-item">
|
||||
<ion-input
|
||||
v-model="formData.confirmPassword"
|
||||
:type="showConfirmPassword ? 'text' : 'password'"
|
||||
placeholder="请确认密码"
|
||||
class="custom-input"
|
||||
/>
|
||||
<ion-icon
|
||||
slot="end"
|
||||
:icon="showConfirmPassword ? 'eye-outline' : 'eye-off-outline'"
|
||||
class="password-toggle"
|
||||
@click="showConfirmPassword = !showConfirmPassword"
|
||||
/>
|
||||
</ion-item>
|
||||
|
||||
<!-- 姓名输入 -->
|
||||
<ion-item lines="none" class="input-item">
|
||||
<ion-input
|
||||
v-model="formData.realName"
|
||||
type="text"
|
||||
placeholder="请输入真实姓名"
|
||||
class="custom-input"
|
||||
/>
|
||||
</ion-item>
|
||||
|
||||
<!-- 身份证输入 -->
|
||||
<ion-item lines="none" class="input-item">
|
||||
<ion-input
|
||||
v-model="formData.idCard"
|
||||
type="text"
|
||||
placeholder="请输入身份证号码"
|
||||
class="custom-input"
|
||||
:maxlength="18"
|
||||
/>
|
||||
</ion-item>
|
||||
|
||||
<!-- 邀请码输入 -->
|
||||
<ion-item lines="none" class="input-item">
|
||||
<ion-input
|
||||
v-model="formData.inviteCode"
|
||||
type="text"
|
||||
placeholder="请输入邀请码(选填)"
|
||||
class="custom-input"
|
||||
/>
|
||||
</ion-item>
|
||||
|
||||
<!-- 协议同意 -->
|
||||
<div class="agreement-section">
|
||||
<ion-checkbox v-model="agreed" class="agreement-checkbox" />
|
||||
<span class="agreement-text">
|
||||
我已阅读并同意
|
||||
<a class="link-service" @click="goToTerms('service')">《服务条款》</a>
|
||||
和
|
||||
<a class="link-privacy" @click="goToTerms('privacy')">《隐私政策》</a>
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<!-- 注册按钮 -->
|
||||
<ion-button
|
||||
expand="block"
|
||||
class="signup-button"
|
||||
@click="handleSignup"
|
||||
>
|
||||
注册
|
||||
</ion-button>
|
||||
|
||||
<!-- 登录按钮 -->
|
||||
<ion-button
|
||||
expand="block"
|
||||
fill="outline"
|
||||
class="login-button"
|
||||
@click="handleLogin"
|
||||
>
|
||||
已有账号?立即登录
|
||||
</ion-button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 底部装饰 - 天安门和飞鸟 -->
|
||||
<div class="decoration-bottom">
|
||||
<div class="building-wrapper">
|
||||
<div class="building" />
|
||||
<div class="birds-bottom">
|
||||
<div class="bird-small bird-b-1" />
|
||||
<div class="bird-small bird-b-2" />
|
||||
<div class="bird-small bird-b-3" />
|
||||
<div class="bird-small bird-b-4" />
|
||||
</div>
|
||||
</div>
|
||||
<div class="wave-red" />
|
||||
</div>
|
||||
</ion-content>
|
||||
</ion-page>
|
||||
</template>
|
||||
|
||||
<style lang='css' scoped></style>
|
||||
<style lang='css' scoped>
|
||||
.signup-page {
|
||||
--background: transparent;
|
||||
position: relative;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
/* 渐变背景 */
|
||||
.gradient-bg {
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
background: linear-gradient(
|
||||
180deg,
|
||||
#c41e3a 0%,
|
||||
#e8756d 15%,
|
||||
#f5d5c8 35%,
|
||||
#fef5f1 50%,
|
||||
#fef5f1 65%,
|
||||
#f5d5c8 85%,
|
||||
#e8756d 100%
|
||||
);
|
||||
z-index: 0;
|
||||
}
|
||||
|
||||
/* 顶部装饰飞鸟 */
|
||||
.decoration-top {
|
||||
position: absolute;
|
||||
top: 60px;
|
||||
left: 20px;
|
||||
z-index: 1;
|
||||
}
|
||||
|
||||
.bird {
|
||||
width: 16px;
|
||||
height: 16px;
|
||||
background: linear-gradient(135deg, #c41e3a 0%, #8b1a2e 100%);
|
||||
position: absolute;
|
||||
clip-path: polygon(50% 0%, 0% 40%, 20% 50%, 0% 60%, 50% 100%, 100% 60%, 80% 50%, 100% 40%);
|
||||
opacity: 0.8;
|
||||
animation: float 3s ease-in-out infinite;
|
||||
}
|
||||
|
||||
.bird-1 {
|
||||
left: 0;
|
||||
top: 0;
|
||||
animation-delay: 0s;
|
||||
}
|
||||
|
||||
.bird-2 {
|
||||
left: 25px;
|
||||
top: 10px;
|
||||
width: 14px;
|
||||
height: 14px;
|
||||
animation-delay: 0.5s;
|
||||
}
|
||||
|
||||
.bird-3 {
|
||||
left: 15px;
|
||||
top: 30px;
|
||||
width: 12px;
|
||||
height: 12px;
|
||||
animation-delay: 1s;
|
||||
}
|
||||
|
||||
@keyframes float {
|
||||
0%,
|
||||
100% {
|
||||
transform: translateY(0) rotate(0deg);
|
||||
}
|
||||
50% {
|
||||
transform: translateY(-10px) rotate(5deg);
|
||||
}
|
||||
}
|
||||
|
||||
/* 内容包装器 */
|
||||
.content-wrapper {
|
||||
position: relative;
|
||||
z-index: 2;
|
||||
height: 100%;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
padding: 30px 24px 20px;
|
||||
overflow-y: auto;
|
||||
}
|
||||
|
||||
/* 头部区域 */
|
||||
.header-section {
|
||||
text-align: center;
|
||||
margin-bottom: 30px;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
/* 国徽 */
|
||||
.emblem {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
margin-bottom: 16px;
|
||||
}
|
||||
|
||||
.emblem-circle {
|
||||
width: 60px;
|
||||
height: 60px;
|
||||
background: linear-gradient(135deg, #ffd700 0%, #ffed4e 100%);
|
||||
border-radius: 50%;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
box-shadow: 0 4px 12px rgba(196, 30, 58, 0.3);
|
||||
border: 3px solid #c41e3a;
|
||||
}
|
||||
|
||||
.emblem-inner {
|
||||
width: 48px;
|
||||
height: 48px;
|
||||
background: linear-gradient(135deg, #c41e3a 0%, #8b1a2e 100%);
|
||||
border-radius: 50%;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.star {
|
||||
font-size: 20px;
|
||||
color: #ffd700;
|
||||
font-weight: bold;
|
||||
margin-bottom: -3px;
|
||||
}
|
||||
|
||||
.emblem-text {
|
||||
font-size: 5px;
|
||||
color: #ffd700;
|
||||
font-weight: bold;
|
||||
letter-spacing: 0.5px;
|
||||
text-align: center;
|
||||
line-height: 1.2;
|
||||
width: 36px;
|
||||
}
|
||||
|
||||
/* 欢迎文字 */
|
||||
.welcome-title {
|
||||
font-size: 28px;
|
||||
font-weight: bold;
|
||||
color: #1a1a1a;
|
||||
margin: 0 0 6px 0;
|
||||
letter-spacing: 2px;
|
||||
}
|
||||
|
||||
.welcome-subtitle {
|
||||
font-size: 18px;
|
||||
font-weight: 600;
|
||||
color: #2c3e50;
|
||||
margin: 0;
|
||||
letter-spacing: 1px;
|
||||
}
|
||||
|
||||
/* 表单区域 */
|
||||
.form-section {
|
||||
flex: 1;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 12px;
|
||||
max-width: 400px;
|
||||
width: 100%;
|
||||
margin: 0 auto;
|
||||
padding-bottom: 200px;
|
||||
}
|
||||
|
||||
/* 输入框 */
|
||||
.input-item {
|
||||
--background: rgba(255, 255, 255, 0.95);
|
||||
--border-radius: 12px;
|
||||
--padding-start: 16px;
|
||||
--padding-end: 16px;
|
||||
--inner-padding-end: 0;
|
||||
border-radius: 12px;
|
||||
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08);
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.custom-input {
|
||||
--padding-start: 0;
|
||||
--padding-end: 0;
|
||||
font-size: 16px;
|
||||
color: #2c3e50;
|
||||
}
|
||||
|
||||
.password-toggle {
|
||||
color: #8b949e;
|
||||
font-size: 22px;
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
/* 协议同意 */
|
||||
.agreement-section {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
padding: 8px 4px 4px;
|
||||
}
|
||||
|
||||
.agreement-checkbox {
|
||||
--size: 20px;
|
||||
--border-radius: 4px;
|
||||
--background-checked: #c41e3a;
|
||||
--border-color-checked: #c41e3a;
|
||||
margin: 0;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.agreement-text {
|
||||
font-size: 13px;
|
||||
color: #666;
|
||||
line-height: 1.4;
|
||||
}
|
||||
|
||||
.link-service,
|
||||
.link-privacy {
|
||||
color: #c41e3a;
|
||||
text-decoration: none;
|
||||
cursor: pointer;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.link-service:hover,
|
||||
.link-privacy:hover {
|
||||
text-decoration: underline;
|
||||
}
|
||||
|
||||
/* 注册按钮 */
|
||||
.signup-button {
|
||||
--background: linear-gradient(135deg, #c41e3a 0%, #8b1a2e 100%);
|
||||
--background-hover: linear-gradient(135deg, #a01830 0%, #6b1424 100%);
|
||||
--border-radius: 25px;
|
||||
--box-shadow: 0 4px 12px rgba(196, 30, 58, 0.3);
|
||||
height: 50px;
|
||||
font-size: 18px;
|
||||
font-weight: 600;
|
||||
letter-spacing: 2px;
|
||||
margin-top: 8px;
|
||||
text-transform: none;
|
||||
}
|
||||
|
||||
/* 登录按钮 */
|
||||
.login-button {
|
||||
--border-color: #8b1a2e;
|
||||
--border-width: 2px;
|
||||
--border-radius: 25px;
|
||||
--color: #8b1a2e;
|
||||
--background: transparent;
|
||||
--background-hover: rgba(196, 30, 58, 0.05);
|
||||
height: 50px;
|
||||
font-size: 16px;
|
||||
font-weight: 600;
|
||||
letter-spacing: 1px;
|
||||
text-transform: none;
|
||||
}
|
||||
|
||||
/* 底部装饰 */
|
||||
.decoration-bottom {
|
||||
position: fixed;
|
||||
bottom: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
z-index: 1;
|
||||
pointer-events: none;
|
||||
}
|
||||
|
||||
.building-wrapper {
|
||||
position: relative;
|
||||
height: 180px;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.building {
|
||||
position: absolute;
|
||||
bottom: 40px;
|
||||
left: 0;
|
||||
right: 0;
|
||||
height: 140px;
|
||||
background: linear-gradient(
|
||||
to top,
|
||||
rgba(196, 30, 58, 0.9) 0%,
|
||||
rgba(196, 30, 58, 0.8) 20%,
|
||||
rgba(232, 117, 109, 0.7) 40%,
|
||||
rgba(232, 117, 109, 0.5) 60%,
|
||||
rgba(232, 117, 109, 0.3) 80%,
|
||||
transparent 100%
|
||||
);
|
||||
clip-path: polygon(
|
||||
0% 100%,
|
||||
0% 60%,
|
||||
10% 55%,
|
||||
15% 50%,
|
||||
15% 45%,
|
||||
18% 45%,
|
||||
18% 40%,
|
||||
22% 40%,
|
||||
22% 35%,
|
||||
28% 35%,
|
||||
28% 30%,
|
||||
35% 30%,
|
||||
35% 25%,
|
||||
42% 25%,
|
||||
45% 22%,
|
||||
50% 20%,
|
||||
55% 22%,
|
||||
58% 25%,
|
||||
65% 25%,
|
||||
65% 30%,
|
||||
72% 30%,
|
||||
72% 35%,
|
||||
78% 35%,
|
||||
78% 40%,
|
||||
82% 40%,
|
||||
82% 45%,
|
||||
85% 45%,
|
||||
85% 50%,
|
||||
90% 55%,
|
||||
100% 60%,
|
||||
100% 100%
|
||||
);
|
||||
}
|
||||
|
||||
.birds-bottom {
|
||||
position: absolute;
|
||||
bottom: 80px;
|
||||
right: 20px;
|
||||
}
|
||||
|
||||
.bird-small {
|
||||
width: 12px;
|
||||
height: 12px;
|
||||
background: linear-gradient(135deg, #c41e3a 0%, #8b1a2e 100%);
|
||||
position: absolute;
|
||||
clip-path: polygon(50% 0%, 0% 40%, 20% 50%, 0% 60%, 50% 100%, 100% 60%, 80% 50%, 100% 40%);
|
||||
opacity: 0.9;
|
||||
animation: float-bottom 2.5s ease-in-out infinite;
|
||||
}
|
||||
|
||||
.bird-b-1 {
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
animation-delay: 0s;
|
||||
}
|
||||
|
||||
.bird-b-2 {
|
||||
right: 20px;
|
||||
bottom: 15px;
|
||||
width: 14px;
|
||||
height: 14px;
|
||||
animation-delay: 0.3s;
|
||||
}
|
||||
|
||||
.bird-b-3 {
|
||||
right: 45px;
|
||||
bottom: 8px;
|
||||
animation-delay: 0.6s;
|
||||
}
|
||||
|
||||
.bird-b-4 {
|
||||
right: 70px;
|
||||
bottom: 20px;
|
||||
width: 10px;
|
||||
height: 10px;
|
||||
animation-delay: 0.9s;
|
||||
}
|
||||
|
||||
@keyframes float-bottom {
|
||||
0%,
|
||||
100% {
|
||||
transform: translate(0, 0) rotate(0deg);
|
||||
}
|
||||
50% {
|
||||
transform: translate(-5px, -8px) rotate(-5deg);
|
||||
}
|
||||
}
|
||||
|
||||
/* 红色波浪 */
|
||||
.wave-red {
|
||||
position: absolute;
|
||||
bottom: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
height: 40px;
|
||||
background: linear-gradient(180deg, transparent 0%, #c41e3a 100%);
|
||||
}
|
||||
</style>
|
||||
|
||||
Reference in New Issue
Block a user