合并代码
This commit is contained in:
341
src/views/AgentLogin.vue
Normal file
341
src/views/AgentLogin.vue
Normal file
@@ -0,0 +1,341 @@
|
||||
<template>
|
||||
<div class="agent-login-container">
|
||||
<div class="login-card">
|
||||
<div class="login-header">
|
||||
<h2>区域代理登录</h2>
|
||||
<p>浙江省区域代理管理系统</p>
|
||||
</div>
|
||||
|
||||
<el-form
|
||||
ref="loginFormRef"
|
||||
:model="loginForm"
|
||||
:rules="loginRules"
|
||||
class="login-form"
|
||||
@submit.prevent="handleLogin"
|
||||
>
|
||||
<el-form-item prop="phone">
|
||||
<el-input
|
||||
v-model="loginForm.phone"
|
||||
placeholder="请输入手机号"
|
||||
prefix-icon="Phone"
|
||||
size="large"
|
||||
/>
|
||||
</el-form-item>
|
||||
|
||||
<el-form-item prop="password">
|
||||
<el-input
|
||||
v-model="loginForm.password"
|
||||
type="password"
|
||||
placeholder="请输入密码"
|
||||
prefix-icon="Lock"
|
||||
size="large"
|
||||
show-password
|
||||
/>
|
||||
</el-form-item>
|
||||
|
||||
<el-form-item>
|
||||
<el-button
|
||||
type="primary"
|
||||
size="large"
|
||||
class="login-btn"
|
||||
:loading="loading"
|
||||
@click="handleLogin"
|
||||
>
|
||||
登录
|
||||
</el-button>
|
||||
</el-form-item>
|
||||
</el-form>
|
||||
|
||||
<div class="login-footer">
|
||||
<el-link type="primary" @click="showApplyDialog = true">
|
||||
申请成为区域代理
|
||||
</el-link>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 申请代理对话框 -->
|
||||
<el-dialog
|
||||
v-model="showApplyDialog"
|
||||
title="申请成为区域代理"
|
||||
width="500px"
|
||||
:close-on-click-modal="false"
|
||||
>
|
||||
<el-alert
|
||||
title="重要提示"
|
||||
type="info"
|
||||
:closable="false"
|
||||
style="margin-bottom: 20px"
|
||||
>
|
||||
<template #default>
|
||||
<p>• 每个区域只能有一个代理账号</p>
|
||||
<p>• 每个用户只能申请一个区域的代理</p>
|
||||
<p>• 申请提交后需要等待管理员审核</p>
|
||||
</template>
|
||||
</el-alert>
|
||||
|
||||
<el-form
|
||||
ref="applyFormRef"
|
||||
:model="applyForm"
|
||||
:rules="applyRules"
|
||||
label-width="100px"
|
||||
>
|
||||
<el-form-item label="选择区域" prop="region_id">
|
||||
<el-select
|
||||
v-model="applyForm.region_id"
|
||||
placeholder="请选择代理区域"
|
||||
style="width: 100%"
|
||||
filterable
|
||||
>
|
||||
<el-option-group
|
||||
v-for="city in groupedRegions"
|
||||
:key="city.name"
|
||||
:label="city.name"
|
||||
>
|
||||
<el-option
|
||||
v-for="region in city.districts"
|
||||
:key="region.id"
|
||||
:label="region.district_name"
|
||||
:value="region.id"
|
||||
/>
|
||||
</el-option-group>
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
|
||||
<el-form-item label="真实姓名" prop="real_name">
|
||||
<el-input v-model="applyForm.real_name" placeholder="请输入真实姓名" />
|
||||
</el-form-item>
|
||||
|
||||
<el-form-item label="手机号" prop="phone">
|
||||
<el-input v-model="applyForm.phone" placeholder="请输入手机号" />
|
||||
</el-form-item>
|
||||
|
||||
<el-form-item label="身份证号" prop="id_card">
|
||||
<el-input v-model="applyForm.id_card" placeholder="请输入身份证号" />
|
||||
</el-form-item>
|
||||
|
||||
<el-form-item label="联系地址" prop="contact_address">
|
||||
<el-input
|
||||
v-model="applyForm.contact_address"
|
||||
type="textarea"
|
||||
placeholder="请输入联系地址"
|
||||
:rows="3"
|
||||
/>
|
||||
</el-form-item>
|
||||
</el-form>
|
||||
|
||||
<template #footer>
|
||||
<el-button @click="showApplyDialog = false">取消</el-button>
|
||||
<el-button type="primary" :loading="applyLoading" @click="handleApply">
|
||||
提交申请
|
||||
</el-button>
|
||||
</template>
|
||||
</el-dialog>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, reactive, computed, onMounted } from 'vue'
|
||||
import { useRouter } from 'vue-router'
|
||||
import { ElMessage } from 'element-plus'
|
||||
import api from '@/utils/api'
|
||||
|
||||
const router = useRouter()
|
||||
|
||||
// 响应式数据
|
||||
const loading = ref(false)
|
||||
const applyLoading = ref(false)
|
||||
const showApplyDialog = ref(false)
|
||||
const regions = ref([])
|
||||
const loginFormRef = ref()
|
||||
const applyFormRef = ref()
|
||||
|
||||
// 登录表单
|
||||
const loginForm = reactive({
|
||||
phone: '',
|
||||
password: ''
|
||||
})
|
||||
|
||||
// 申请表单
|
||||
const applyForm = reactive({
|
||||
region_id: '',
|
||||
real_name: '',
|
||||
phone: '',
|
||||
id_card: '',
|
||||
contact_address: ''
|
||||
})
|
||||
|
||||
// 表单验证规则
|
||||
const loginRules = {
|
||||
phone: [
|
||||
{ required: true, message: '请输入手机号', trigger: 'blur' },
|
||||
{ pattern: /^1[3-9]\d{9}$/, message: '手机号格式不正确', trigger: 'blur' }
|
||||
],
|
||||
password: [
|
||||
{ required: true, message: '请输入密码', trigger: 'blur' }
|
||||
]
|
||||
}
|
||||
|
||||
const applyRules = {
|
||||
region_id: [
|
||||
{ required: true, message: '请选择代理区域', trigger: 'change' }
|
||||
],
|
||||
real_name: [
|
||||
{ required: true, message: '请输入真实姓名', trigger: 'blur' }
|
||||
],
|
||||
phone: [
|
||||
{ required: true, message: '请输入手机号', trigger: 'blur' },
|
||||
{ pattern: /^1[3-9]\d{9}$/, message: '手机号格式不正确', trigger: 'blur' }
|
||||
],
|
||||
id_card: [
|
||||
{ required: true, message: '请输入身份证号', trigger: 'blur' },
|
||||
{ pattern: /^[1-9]\d{5}(18|19|20)\d{2}((0[1-9])|(1[0-2]))(([0-2][1-9])|10|20|30|31)\d{3}[0-9Xx]$/, message: '身份证号格式不正确', trigger: 'blur' }
|
||||
]
|
||||
}
|
||||
|
||||
// 计算属性
|
||||
const groupedRegions = computed(() => {
|
||||
const grouped = {}
|
||||
regions.value.forEach(region => {
|
||||
if (!grouped[region.city_name]) {
|
||||
grouped[region.city_name] = {
|
||||
name: region.city_name,
|
||||
districts: []
|
||||
}
|
||||
}
|
||||
grouped[region.city_name].districts.push(region)
|
||||
})
|
||||
return Object.values(grouped)
|
||||
})
|
||||
|
||||
// 方法
|
||||
const loadRegions = async () => {
|
||||
try {
|
||||
const { data } = await api.get('/agents/regions')
|
||||
regions.value = data.data
|
||||
} catch (error) {
|
||||
ElMessage.error('加载区域列表失败')
|
||||
}
|
||||
}
|
||||
|
||||
const handleLogin = async () => {
|
||||
if (!loginFormRef.value) return
|
||||
|
||||
try {
|
||||
await loginFormRef.value.validate()
|
||||
loading.value = true
|
||||
|
||||
const { data } = await api.post('/agents/login', loginForm)
|
||||
|
||||
if (data.success) {
|
||||
// 保存代理信息到localStorage
|
||||
localStorage.setItem('agentInfo', JSON.stringify(data.data))
|
||||
ElMessage.success('登录成功')
|
||||
router.push('/agent/dashboard')
|
||||
}
|
||||
} catch (error) {
|
||||
if (error.response?.data?.message) {
|
||||
ElMessage.error(error.response.data.message)
|
||||
} else {
|
||||
ElMessage.error('登录失败')
|
||||
}
|
||||
} finally {
|
||||
loading.value = false
|
||||
}
|
||||
}
|
||||
|
||||
const handleApply = async () => {
|
||||
if (!applyFormRef.value) return
|
||||
|
||||
try {
|
||||
await applyFormRef.value.validate()
|
||||
applyLoading.value = true
|
||||
|
||||
const { data } = await api.post('/agents/apply', applyForm)
|
||||
|
||||
if (data.success) {
|
||||
ElMessage.success(data.message)
|
||||
showApplyDialog.value = false
|
||||
// 重置表单
|
||||
Object.keys(applyForm).forEach(key => {
|
||||
applyForm[key] = ''
|
||||
})
|
||||
}
|
||||
} catch (error) {
|
||||
if (error.response?.data?.message) {
|
||||
ElMessage.error(error.response.data.message)
|
||||
} else {
|
||||
ElMessage.error('申请失败')
|
||||
}
|
||||
} finally {
|
||||
applyLoading.value = false
|
||||
}
|
||||
}
|
||||
|
||||
// 生命周期
|
||||
onMounted(() => {
|
||||
loadRegions()
|
||||
})
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.agent-login-container {
|
||||
min-height: 100vh;
|
||||
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
padding: 20px;
|
||||
}
|
||||
|
||||
.login-card {
|
||||
background: white;
|
||||
border-radius: 12px;
|
||||
box-shadow: 0 8px 32px rgba(0, 0, 0, 0.1);
|
||||
padding: 40px;
|
||||
width: 100%;
|
||||
max-width: 400px;
|
||||
}
|
||||
|
||||
.login-header {
|
||||
text-align: center;
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.login-header h2 {
|
||||
color: #333;
|
||||
margin-bottom: 8px;
|
||||
font-size: 24px;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.login-header p {
|
||||
color: #666;
|
||||
font-size: 14px;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.login-form {
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.login-btn {
|
||||
width: 100%;
|
||||
height: 44px;
|
||||
font-size: 16px;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.login-footer {
|
||||
text-align: center;
|
||||
padding-top: 20px;
|
||||
border-top: 1px solid #eee;
|
||||
}
|
||||
|
||||
:deep(.el-input__wrapper) {
|
||||
border-radius: 8px;
|
||||
}
|
||||
|
||||
:deep(.el-button) {
|
||||
border-radius: 8px;
|
||||
}
|
||||
</style>
|
||||
Reference in New Issue
Block a user