feat(login): 添加验证码功能并优化登录页面

- 在登录页面添加验证码输入和验证功能
- 实现验证码图片的获取和更新
- 优化登录逻辑,处理登录失败的情况
- 调整表单字段和布局以支持验证码
main
Tuzki 4 months ago
parent e01ba9d839
commit 7e65a15c86
  1. 4
      web/client/api/system/index.ts
  2. 7
      web/new-components/common/blurredCard/style.css
  3. 58
      web/pages/login.tsx

@ -137,3 +137,7 @@ export const getUserMenu = () => {
export const getMenuDetail = (id: number) => {
return GET<any, any>(`/api/menu/menu_detail/` + id);
};
//验证码
export const getVerifyCode = () => {
return GET<any, any>(`/api/captcha`);
};

@ -26,4 +26,9 @@
position: relative;
overflow: hidden;
}
.m-dropdown-menu {
z-index: 1050; /* 小于 Modal 的 z-index */
}
.ant-modal-wrap {
z-index: 1051;
}

@ -1,27 +1,33 @@
// login.tsx 最简测试版本
import { doLogin, getUserMenu } from '@/client/api';
import { doLogin, getUserMenu, getVerifyCode } from '@/client/api';
import { STORAGE_USERINFO_KEY, STORAGE_USERINFO_VALID_TIME_KEY } from '@/utils/constants/index';
import { useRequest } from 'ahooks';
import { Button, Form, Image, Input, Spin, notification } from 'antd';
import { useRouter } from 'next/router';
import { FC } from 'react';
import { FC, useState, useEffect } from 'react';
const imgSrc =
'';
'';
const LoginPage: FC = () => {
const router = useRouter();
const [form] = Form.useForm(); // ✅ 新增 form 实例
const [verifyCodeImage, setVerifyCodeImage] = useState<string>(imgSrc);
const [catcha_id, setCaptcha_id] = useState<string>('');
const { run: doLoginFn, loading } = useRequest(
async (formData: { username: string; password: string }) => {
return await doLogin(formData); // 直接传递表单对象
async (formData: { username: string; password: string; catcha_id: string; verificationCode: string }) => {
formData.catcha_id = catcha_id;
return await doLogin(formData);
},
{
manual: true,
onSuccess: async (res, params) => {
const [formData] = params;
if (!res.data.success) {
notification.error({
message: '登录失败,请检查账号密码',
message: res.data.err_msg ||'登录失败,请检查账号密码',
});
return;
}
@ -36,7 +42,6 @@ const LoginPage: FC = () => {
localStorage.setItem(STORAGE_USERINFO_KEY, JSON.stringify(user));
localStorage.setItem(STORAGE_USERINFO_VALID_TIME_KEY, Date.now().toString());
// 获取用户菜单
const menuRes = await getUserMenu();
if (!menuRes.data.success) {
notification.error({
@ -47,10 +52,42 @@ const LoginPage: FC = () => {
localStorage.setItem('menu', JSON.stringify(menuRes.data.data));
router.push('/playground/appGround');
},
onError: (error, params) => {
notification.error({
message: error.response.data.err_msg || '登录失败,请稍后再试',
});
},
debounceWait: 500,
}
);
const { run: fetchVerifyCode, loading: verifying } = useRequest(
async () => {
const res = await getVerifyCode();
if (res.data && res.data.success) {
setVerifyCodeImage(res.data.data.captcha_image);
const newCaptchaId = res.data.data.captcha_id;
setCaptcha_id(newCaptchaId); // ✅ 新增 captcha_id 变量
form.setFieldsValue({ captcha_id: newCaptchaId }); // ✅ 更新表单字段
console.log('captcha_id:', newCaptchaId); // ✅ 打印最新值
} else {
notification.error({
message: '获取验证码失败',
description: res.data.err_msg || '请稍后再试',
});
}
},
{
manual: true,
debounceWait: 500,
}
);
useEffect(() => {
fetchVerifyCode();
}, []);
return (
<div className='relative h-screen w-screen overflow-hidden'>
{/* 视频背景 */}
@ -69,14 +106,17 @@ const LoginPage: FC = () => {
<Input placeholder='用户名' />
</Form.Item>
<Form.Item name='phone' rules={[{ required: true, message: '请输入验证码' }]}>
<Form.Item name='verificationCode' rules={[{ required: true, message: '请输入验证码' }]}>
<div className='flex items-center justify-between'>
<Input placeholder="请输入验证码">
</Input>
<img src={imgSrc} className='h-[30px] mx-[40px]' />
<img onClick={fetchVerifyCode} src={verifyCodeImage} className='h-[30px] mx-[40px]' />
</div>
</Form.Item>
<Form.Item name="catcha_id" noStyle>
<Input type="hidden" />
</Form.Item>
<Form.Item name='password' rules={[{ required: true, message: '请输入密码' }]}>
<Input.Password placeholder='密码' />
</Form.Item>

Loading…
Cancel
Save