Files
wecom_it_smart_desk/frontend-agent/src/views/Login.vue
T

171 lines
4.9 KiB
Vue
Raw Normal View History

<!-- =============================================================================
// 企微IT智能服务台 — 坐席登录页面
// =============================================================================
// 说明:坐席登录页面,简单的用户名+姓名表单
// 登录成功后跳转到工作台页面
// 第一步不做密码验证,仅输入用户ID和姓名即可登录
// ============================================================================= -->
<template>
<div class="login-container">
<div class="login-card">
<!-- 标题区域 -->
<div class="login-title">
<h1>🛠 IT智能服务台</h1>
<p>坐席工作台 · 登录</p>
</div>
<!-- 登录表单 -->
<el-form
ref="loginFormRef"
:model="loginForm"
:rules="loginRules"
label-position="top"
@submit.prevent="handleLogin"
>
<!-- 企微用户ID -->
<el-form-item label="企微用户ID" prop="userId">
<el-input
v-model="loginForm.userId"
placeholder="请输入企微用户ID"
prefix-icon="User"
size="large"
clearable
/>
</el-form-item>
<!-- 坐席姓名 -->
<el-form-item label="姓名" prop="name">
<el-input
v-model="loginForm.name"
placeholder="请输入您的姓名"
prefix-icon="UserFilled"
size="large"
clearable
/>
</el-form-item>
<!-- OTP 动态码admin 角色需要 -->
<el-form-item v-if="requireOtp" label="OTP动态码" prop="otpCode">
<el-input
v-model="loginForm.otpCode"
placeholder="请输入Google Authenticator中的6位动态码"
prefix-icon="Lock"
size="large"
maxlength="6"
clearable
@keyup.enter="handleLogin"
/>
</el-form-item>
<!-- 登录按钮 -->
<el-form-item>
<el-button
type="primary"
size="large"
:loading="agentStore.logging"
style="width: 100%"
@click="handleLogin"
>
{{ agentStore.logging ? '登录中...' : '登 录' }}
</el-button>
</el-form-item>
</el-form>
<!-- 提示信息 -->
<div class="login-hint">
使用企微账号登录姓名将自动获取
</div>
</div>
</div>
</template>
<script setup lang="ts">
// ============================================================================
// 导入
// ============================================================================
import { ref, reactive } from 'vue'
import { ElMessage } from 'element-plus'
import type { FormInstance, FormRules } from 'element-plus'
import { useAgentStore } from '@/stores/agent'
// ============================================================================
// 状态
// ============================================================================
/** 坐席 Store */
const agentStore = useAgentStore()
/** 表单引用 */
const loginFormRef = ref<FormInstance>()
/** 登录表单数据 */
const loginForm = reactive({
/** 企微用户ID */
userId: '',
/** 坐席姓名 */
name: '',
/** OTP 动态码 */
otpCode: '',
})
/** 是否需要 OTP 验证 */
const requireOtp = ref(false)
/** 表单校验规则 */
const loginRules = reactive<FormRules>({
userId: [
{ required: true, message: '请输入企微用户ID', trigger: 'blur' },
{ min: 1, max: 64, message: '用户ID长度为1-64个字符', trigger: 'blur' },
],
name: [
{ required: true, message: '请输入姓名', trigger: 'blur' },
{ min: 1, max: 128, message: '姓名长度为1-128个字符', trigger: 'blur' },
],
})
// ============================================================================
// 方法
// ============================================================================
/**
* 处理登录
* 1. 校验表单
* 2. 调用登录 API
* 3. 如果返回 require_otp,显示 OTP 输入框
* 4. 用户输入 OTP 后再次登录
* 5. 成功后自动跳转
*/
async function handleLogin(): Promise<void> {
// 表单校验
if (!loginFormRef.value) return
const valid = await loginFormRef.value.validate().catch(() => false)
if (!valid) return
try {
const data = await agentStore.login(loginForm.userId, loginForm.name, loginForm.otpCode || undefined)
// 检查是否需要 OTP 验证
if (data && 'require_otp' in data && data.require_otp) {
requireOtp.value = true
ElMessage.warning('请输入OTP动态码')
return
}
ElMessage.success('登录成功')
} catch (error: any) {
// 错误信息已在 Axios 拦截器中显示
console.error('登录失败:', error)
}
}
</script>
<style scoped>
.login-hint {
text-align: center;
color: var(--text-tertiary);
font-size: 12px;
margin-top: 16px;
}
</style>