feat: super admin over
parent
d6d86ed657
commit
758f426b93
14
.umirc.ts
14
.umirc.ts
|
@ -9,7 +9,21 @@ export default defineConfig({
|
||||||
layout: false,
|
layout: false,
|
||||||
dva: {},
|
dva: {},
|
||||||
valtio: {},
|
valtio: {},
|
||||||
|
plugins: [require.resolve('@umijs/plugins/dist/unocss')],
|
||||||
|
unocss: {
|
||||||
|
watch: ['src/**/*.tsx'],
|
||||||
|
},
|
||||||
|
extraPostCSSPlugins: [
|
||||||
|
require('tailwindcss')({
|
||||||
|
config: './tailwind.config.ts',
|
||||||
|
}),
|
||||||
|
],
|
||||||
routes: [
|
routes: [
|
||||||
|
{
|
||||||
|
path: '/login',
|
||||||
|
component: 'Login',
|
||||||
|
layout: false,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: '/',
|
path: '/',
|
||||||
id: 0,
|
id: 0,
|
||||||
|
|
|
@ -14,12 +14,18 @@
|
||||||
"@ant-design/icons": "^5.0.1",
|
"@ant-design/icons": "^5.0.1",
|
||||||
"@ant-design/pro-components": "^2.7.9",
|
"@ant-design/pro-components": "^2.7.9",
|
||||||
"@umijs/max": "^4.2.8",
|
"@umijs/max": "^4.2.8",
|
||||||
|
"@umijs/plugins": "^4.2.9",
|
||||||
|
"@unocss/cli": "^0.60.4",
|
||||||
"antd": "^5.18.0",
|
"antd": "^5.18.0",
|
||||||
"antd-style": "^3.6.2",
|
"antd-style": "^3.6.2",
|
||||||
"axios": "^1.7.2",
|
"axios": "^1.7.2",
|
||||||
"qs": "^6.12.1"
|
"lodash-es": "^4.17.21",
|
||||||
|
"qs": "^6.12.1",
|
||||||
|
"tailwindcss": "^3.4.3",
|
||||||
|
"unocss": "^0.60.4"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@types/lodash": "^4.17.4",
|
||||||
"@types/qs": "^6.9.15",
|
"@types/qs": "^6.9.15",
|
||||||
"@types/react": "^18.0.33",
|
"@types/react": "^18.0.33",
|
||||||
"@types/react-dom": "^18.0.11",
|
"@types/react-dom": "^18.0.11",
|
||||||
|
|
979
pnpm-lock.yaml
979
pnpm-lock.yaml
File diff suppressed because it is too large
Load Diff
28
src/app.tsx
28
src/app.tsx
|
@ -1,3 +1,5 @@
|
||||||
|
import { getSupderInfoApi } from '@/services/system/supderLogin';
|
||||||
|
import { history } from '@umijs/max';
|
||||||
import { routes as appRouters, loopMenuItem, type MenuItem } from './utils/router';
|
import { routes as appRouters, loopMenuItem, type MenuItem } from './utils/router';
|
||||||
|
|
||||||
export function patchClientRoutes({ routes }: any) {
|
export function patchClientRoutes({ routes }: any) {
|
||||||
|
@ -10,19 +12,17 @@ export function render(oldRender: any) {
|
||||||
oldRender();
|
oldRender();
|
||||||
}
|
}
|
||||||
|
|
||||||
// 全局初始化数据配置,用于 Layout 用户信息和权限初始化
|
export async function getInitialState(): Promise<LoginAPI.LoginUserInfo | null> {
|
||||||
// 更多信息见文档:https://umijs.org/docs/api/runtime-config#getinitialstate
|
const { data: userInfo }: { data: LoginAPI.LoginUserInfo } = await getSupderInfoApi();
|
||||||
export async function getInitialState(): Promise<{ name: string }> {
|
if (!userInfo) {
|
||||||
return {
|
history.replace('/login');
|
||||||
name: 'xx',
|
return null;
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// export const layout = () => {
|
if (userInfo && history.location.pathname === '/login') {
|
||||||
// return {
|
history.push('/');
|
||||||
// logo: 'https://img.alicdn.com/tfs/TB1YHEpwUT1gK0jSZFhXXaAtVXa-28-27.svg',
|
}
|
||||||
// menu: {
|
return {
|
||||||
// locale: false,
|
...userInfo,
|
||||||
// },
|
};
|
||||||
// };
|
}
|
||||||
// };
|
|
||||||
|
|
|
@ -0,0 +1,91 @@
|
||||||
|
import { updatePasswordAPI, userLogoutAPI } from '@/services/system/supderLogin';
|
||||||
|
import { history, useModel } from '@umijs/max';
|
||||||
|
import { Alert, Form, Input, Modal, Spin, message } from 'antd';
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
|
export default () => {
|
||||||
|
const { initialState } = useModel('@@initialState');
|
||||||
|
const [loading, setLoading] = useState<boolean>(false);
|
||||||
|
const [formRef] = Form.useForm();
|
||||||
|
const [updatePwd, setUpdatePwd] = useState<boolean>(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const isInitPwd = initialState?.isFirstLogin;
|
||||||
|
console.log(initialState?.isFirstLogin);
|
||||||
|
if (isInitPwd) {
|
||||||
|
setUpdatePwd(true);
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleUpdatePwd = async () => {
|
||||||
|
const formValues = await formRef.validateFields();
|
||||||
|
setLoading(true);
|
||||||
|
const params: LoginAPI.UpdatePassWordType = { oldPwd: '123456', newPwd: formValues.password };
|
||||||
|
const result = await updatePasswordAPI(params);
|
||||||
|
setLoading(false);
|
||||||
|
if (result.code === 200) {
|
||||||
|
message.success('密码修改成功,请重新登录');
|
||||||
|
setTimeout(() => {
|
||||||
|
userLogoutAPI();
|
||||||
|
history.replace('/login');
|
||||||
|
}, 2000);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
title="提示"
|
||||||
|
open={updatePwd}
|
||||||
|
cancelButtonProps={{ style: { display: 'none' } }}
|
||||||
|
okText="确定修改"
|
||||||
|
closeIcon={null}
|
||||||
|
onOk={handleUpdatePwd}
|
||||||
|
>
|
||||||
|
<Spin spinning={loading}>
|
||||||
|
<Alert message="系统监测到您的密码为初始密码,请修改。" type="warning" style={{ marginBottom: '10px' }} />
|
||||||
|
<Form form={formRef}>
|
||||||
|
<Form.Item
|
||||||
|
name="password"
|
||||||
|
label="请输入新密码"
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
message: '请输入要修改的密码!',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
pattern: /^[A-Za-z0-9]{6,15}$/,
|
||||||
|
message: '请输入6~15位数的密码,其中包含大小写字母、数字',
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
hasFeedback
|
||||||
|
>
|
||||||
|
<Input.Password />
|
||||||
|
</Form.Item>
|
||||||
|
|
||||||
|
<Form.Item
|
||||||
|
name="confirm"
|
||||||
|
label="请确认新密码"
|
||||||
|
dependencies={['password']}
|
||||||
|
hasFeedback
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
message: '请确定修改的密码!',
|
||||||
|
},
|
||||||
|
({ getFieldValue }) => ({
|
||||||
|
validator(_, value) {
|
||||||
|
if (!value || getFieldValue('password') === value) {
|
||||||
|
return Promise.resolve();
|
||||||
|
}
|
||||||
|
return Promise.reject(new Error('俩次密码不一样!'));
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<Input.Password />
|
||||||
|
</Form.Item>
|
||||||
|
</Form>
|
||||||
|
</Spin>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
};
|
|
@ -1,4 +1,5 @@
|
||||||
import type { ConnectProps, Reducer } from '@umijs/max';
|
import { userLoginAPI } from '@/services/system/supderLogin';
|
||||||
|
import type { ConnectProps, Effect, Reducer } from '@umijs/max';
|
||||||
|
|
||||||
type UserModelState = {
|
type UserModelState = {
|
||||||
loading: boolean;
|
loading: boolean;
|
||||||
|
@ -11,9 +12,9 @@ type UserModelType = {
|
||||||
namespace: 'user';
|
namespace: 'user';
|
||||||
state: UserModelState;
|
state: UserModelState;
|
||||||
effects: {
|
effects: {
|
||||||
// login: Effect;
|
login: Effect;
|
||||||
// setLogin: Effect;
|
setLogin: Effect;
|
||||||
// setKey: Effect;
|
setKey: Effect;
|
||||||
};
|
};
|
||||||
reducers: {
|
reducers: {
|
||||||
save: Reducer<UserModelState>;
|
save: Reducer<UserModelState>;
|
||||||
|
@ -31,14 +32,24 @@ const UserModel: UserModelType = {
|
||||||
loginLoading: false,
|
loginLoading: false,
|
||||||
},
|
},
|
||||||
effects: {
|
effects: {
|
||||||
// 登录
|
*login({ payload }, { call, put }) {
|
||||||
// *login({ payload }, { call, put }) {},
|
try {
|
||||||
// *setLogin({ payload }, { put }) {
|
yield put({ type: 'save', payload: { loginLoading: true } });
|
||||||
// yield put({ type: 'save', payload: { loading: payload } });
|
const result: LoginAPI.LoginResponse = yield call(userLoginAPI, payload);
|
||||||
// },
|
localStorage.setItem('Authorization', result.data.token);
|
||||||
// *setKey({ payload }, { put }) {
|
yield put({ type: 'save', payload: { loginLoading: true, isLogin: true } });
|
||||||
// yield put({ type: 'save', payload: { ...payload } });
|
} catch {
|
||||||
// },
|
yield put({ type: 'save', payload: { loginLoading: false, isLogin: false } });
|
||||||
|
} finally {
|
||||||
|
yield put({ type: 'save', payload: { loginLoading: false } });
|
||||||
|
}
|
||||||
|
},
|
||||||
|
*setLogin({ payload }, { put }) {
|
||||||
|
yield put({ type: 'save', payload: { loading: payload } });
|
||||||
|
},
|
||||||
|
*setKey({ payload }, { put }) {
|
||||||
|
yield put({ type: 'save', payload: { ...payload } });
|
||||||
|
},
|
||||||
},
|
},
|
||||||
reducers: {
|
reducers: {
|
||||||
save(state, action) {
|
save(state, action) {
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import Guide from '@/components/Guide';
|
import Guide from '@/components/Guide';
|
||||||
|
import UpdatePwd from '@/layouts/UpdatePwd';
|
||||||
import { trim } from '@/utils/format';
|
import { trim } from '@/utils/format';
|
||||||
import { PageContainer } from '@ant-design/pro-components';
|
import { PageContainer } from '@ant-design/pro-components';
|
||||||
import { useModel } from '@umijs/max';
|
import { useModel } from '@umijs/max';
|
||||||
|
@ -8,6 +9,7 @@ const HomePage: React.FC = () => {
|
||||||
const { name } = useModel('global');
|
const { name } = useModel('global');
|
||||||
return (
|
return (
|
||||||
<PageContainer ghost>
|
<PageContainer ghost>
|
||||||
|
<UpdatePwd />
|
||||||
<div className={styles.container}>
|
<div className={styles.container}>
|
||||||
<Guide name={trim(name)} />
|
<Guide name={trim(name)} />
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -0,0 +1,49 @@
|
||||||
|
.container {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
height: 100vh;
|
||||||
|
overflow: auto;
|
||||||
|
background: #f0f2f5;
|
||||||
|
}
|
||||||
|
|
||||||
|
.lang {
|
||||||
|
width: 100%;
|
||||||
|
height: 40px;
|
||||||
|
line-height: 44px;
|
||||||
|
text-align: right;
|
||||||
|
|
||||||
|
:global(.ant-dropdown-trigger) {
|
||||||
|
margin-right: 24px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.content {
|
||||||
|
flex: 1;
|
||||||
|
padding: 32px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 768px) {
|
||||||
|
.container {
|
||||||
|
background-image: url('https://gw.alipayobjects.com/zos/rmsportal/TVYTbAXWheQpRcWDaDMu.svg');
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-position: center 110px;
|
||||||
|
background-size: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.content {
|
||||||
|
padding: 32px 0 24px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon {
|
||||||
|
margin-left: 8px;
|
||||||
|
color: rgba(0, 0, 0, 20%);
|
||||||
|
font-size: 24px;
|
||||||
|
vertical-align: middle;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: color 0.3s;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
color: #1890ff;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,91 @@
|
||||||
|
import type { UserConnectedProps } from '@/models/user';
|
||||||
|
import { LockOutlined, UserOutlined } from '@ant-design/icons';
|
||||||
|
import { LoginForm, ProFormInstance, ProFormText } from '@ant-design/pro-components';
|
||||||
|
import { connect } from '@umijs/max';
|
||||||
|
import { Button } from 'antd';
|
||||||
|
import { FC, Fragment, useEffect, useRef } from 'react';
|
||||||
|
import styles from './index.less';
|
||||||
|
|
||||||
|
const Login: FC<UserConnectedProps> = (props) => {
|
||||||
|
const { user, dispatch } = props;
|
||||||
|
|
||||||
|
const { loginLoading, isLogin } = user;
|
||||||
|
const formRef = useRef<ProFormInstance>();
|
||||||
|
|
||||||
|
const handleLoginSubmit = async (values: LoginAPI.LoginParams) => {
|
||||||
|
dispatch?.({
|
||||||
|
type: 'user/login',
|
||||||
|
payload: values,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
console.log(isLogin);
|
||||||
|
if (isLogin) {
|
||||||
|
window.location.href = '/home';
|
||||||
|
}
|
||||||
|
}, [isLogin]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<div className={styles.container}>
|
||||||
|
<div className={styles.content}>
|
||||||
|
<LoginForm
|
||||||
|
formRef={formRef}
|
||||||
|
title="Adseed 管理系统"
|
||||||
|
subTitle=" "
|
||||||
|
initialValues={{
|
||||||
|
autoLogin: true,
|
||||||
|
}}
|
||||||
|
submitter={{
|
||||||
|
render: () => {
|
||||||
|
return (
|
||||||
|
<Button type="primary" htmlType="submit" className="w-full" loading={loginLoading}>
|
||||||
|
登录
|
||||||
|
</Button>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
onFinish={async (values) => {
|
||||||
|
await handleLoginSubmit(values);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Fragment>
|
||||||
|
<ProFormText
|
||||||
|
name="userName"
|
||||||
|
fieldProps={{
|
||||||
|
size: 'large',
|
||||||
|
prefix: <UserOutlined />,
|
||||||
|
}}
|
||||||
|
placeholder="账号"
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
message: '请输入用户名!',
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
<ProFormText.Password
|
||||||
|
name="password"
|
||||||
|
fieldProps={{
|
||||||
|
size: 'large',
|
||||||
|
prefix: <LockOutlined />,
|
||||||
|
}}
|
||||||
|
placeholder="密码"
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
message: '请输入密码!',
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
</Fragment>
|
||||||
|
</LoginForm>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const UserConnect = ({ user }: { user: UserConnectedProps['user'] }) => ({ user });
|
||||||
|
export default connect(UserConnect)(Login);
|
|
@ -1,4 +1,5 @@
|
||||||
import { addSupderAdminAPI, editSupderAdminAPI } from '@/services/system/supderAdmin';
|
import { addSupderAdminAPI, editSupderAdminAPI } from '@/services/system/supderAdmin';
|
||||||
|
import { Reg } from '@/utils/reg';
|
||||||
import { ActionType, ProFormRadio, ProFormText } from '@ant-design/pro-components';
|
import { ActionType, ProFormRadio, ProFormText } from '@ant-design/pro-components';
|
||||||
import { App, Form, Modal } from 'antd';
|
import { App, Form, Modal } from 'antd';
|
||||||
import { useEffect, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
|
@ -65,10 +66,45 @@ const AddSupderAdmin = (props: PropTypes) => {
|
||||||
required
|
required
|
||||||
label="管理员账户"
|
label="管理员账户"
|
||||||
placeholder="请输入管理员账户"
|
placeholder="请输入管理员账户"
|
||||||
rules={[{ required: true, message: '请输入管理员账户' }]}
|
rules={[
|
||||||
|
{ required: true, message: '请输入管理员账户' },
|
||||||
|
{ pattern: Reg.SuperAdminAccount, message: '验证失败(4~15位的非中文账户)' },
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
<ProFormText
|
||||||
|
initialValue={''}
|
||||||
|
width="md"
|
||||||
|
name="email"
|
||||||
|
label="邮箱"
|
||||||
|
placeholder="请输入邮箱"
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
validator: (rule, value) => {
|
||||||
|
if (value && !Reg.Email.test(value)) {
|
||||||
|
return Promise.reject(new Error('邮箱格式不正确'));
|
||||||
|
}
|
||||||
|
return Promise.resolve();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
<ProFormText
|
||||||
|
initialValue={''}
|
||||||
|
width="md"
|
||||||
|
name="phone"
|
||||||
|
label="手机号"
|
||||||
|
placeholder="请输入手机号"
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
validator: (rule, value) => {
|
||||||
|
if (value && !Reg.Phone.test(value)) {
|
||||||
|
return Promise.reject(new Error('手机号格式不正确'));
|
||||||
|
}
|
||||||
|
return Promise.resolve();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]}
|
||||||
/>
|
/>
|
||||||
<ProFormText initialValue={''} width="md" name="email" label="邮箱" placeholder="请输入邮箱" />
|
|
||||||
<ProFormText initialValue={''} width="md" name="phone" label="手机号" placeholder="请输入手机号" />
|
|
||||||
<ProFormRadio.Group
|
<ProFormRadio.Group
|
||||||
width="md"
|
width="md"
|
||||||
required
|
required
|
||||||
|
|
|
@ -42,6 +42,20 @@ const Page = () => {
|
||||||
return record.role;
|
return record.role;
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
title: '状态',
|
||||||
|
dataIndex: 'status',
|
||||||
|
align: 'center',
|
||||||
|
hideInSearch: true,
|
||||||
|
renderText: (_, record: SuperAdmin.SuperAdminItem) => {
|
||||||
|
if (record.status === 1) {
|
||||||
|
return '正常';
|
||||||
|
} else if (record.status === 2) {
|
||||||
|
return '禁用';
|
||||||
|
}
|
||||||
|
return record.status;
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
title: '创建时间',
|
title: '创建时间',
|
||||||
dataIndex: 'createTime',
|
dataIndex: 'createTime',
|
||||||
|
@ -141,6 +155,7 @@ const Page = () => {
|
||||||
<AddSupderAdmin
|
<AddSupderAdmin
|
||||||
visible={createModalVisible}
|
visible={createModalVisible}
|
||||||
onCancel={() => {
|
onCancel={() => {
|
||||||
|
setEditRow(undefined);
|
||||||
setCreateModalVisible(false);
|
setCreateModalVisible(false);
|
||||||
}}
|
}}
|
||||||
tableRef={tableRef}
|
tableRef={tableRef}
|
||||||
|
|
|
@ -0,0 +1,18 @@
|
||||||
|
import request from '@/utils/request';
|
||||||
|
|
||||||
|
//登录
|
||||||
|
export const userLoginAPI = (data: LoginAPI.LoginParams): Promise<LoginAPI.LoginResponse> => {
|
||||||
|
return request.post('/system/account/login', data);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getSupderInfoApi = (): Promise<API.ResponstBody<LoginAPI.LoginUserInfo>> => {
|
||||||
|
return request.post('/system/account/userInfo');
|
||||||
|
};
|
||||||
|
|
||||||
|
export const updatePasswordAPI = (data: LoginAPI.UpdatePassWordType): Promise<API.ResponstBody> => {
|
||||||
|
return request.post('/system/account/updatePwd', data);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const userLogoutAPI = (): Promise<API.ResponstBody> => {
|
||||||
|
return request.post('/system/account/logout');
|
||||||
|
};
|
|
@ -0,0 +1,29 @@
|
||||||
|
declare namespace LoginAPI {
|
||||||
|
type LoginParams = {
|
||||||
|
userName?: string;
|
||||||
|
password?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
type LoginUserInfo = {
|
||||||
|
id?: string;
|
||||||
|
userName?: string;
|
||||||
|
email?: string;
|
||||||
|
phone?: string;
|
||||||
|
status?: number;
|
||||||
|
createTime?: string;
|
||||||
|
updateTime?: string;
|
||||||
|
isFirstLogin?: boolean;
|
||||||
|
role?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
type UpdatePassWordType = {
|
||||||
|
oldPwd: string;
|
||||||
|
newPwd: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
type LoginRespone = {
|
||||||
|
token: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
type LoginResponse = API.ResponstBody<LoginRespone>;
|
||||||
|
}
|
|
@ -0,0 +1,7 @@
|
||||||
|
export const Reg = {
|
||||||
|
Email:
|
||||||
|
// eslint-disable-next-line no-useless-escape
|
||||||
|
/^(([^<>()\[\]\\.,;:\s@"]+(\.[^<>()\[\]\\.,;:\s@"]+)*)|(".+"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/,
|
||||||
|
Phone: /^1\d{10}$/,
|
||||||
|
SuperAdminAccount: /^[A-Za-z][-_!@#$%^&*.a-zA-Z0-9]{4,15}$/,
|
||||||
|
};
|
|
@ -1,4 +1,4 @@
|
||||||
import { notification } from 'antd';
|
import { message as antMessage, notification } from 'antd';
|
||||||
import axios, { AxiosRequestHeaders } from 'axios';
|
import axios, { AxiosRequestHeaders } from 'axios';
|
||||||
|
|
||||||
const instance = axios.create({ baseURL: 'http://127.0.0.1:3008/backend/' });
|
const instance = axios.create({ baseURL: 'http://127.0.0.1:3008/backend/' });
|
||||||
|
@ -17,9 +17,10 @@ instance.interceptors.request.use(
|
||||||
instance.interceptors.response.use(
|
instance.interceptors.response.use(
|
||||||
(response) => {
|
(response) => {
|
||||||
const { data } = response;
|
const { data } = response;
|
||||||
const { code } = data;
|
const { code, message } = data;
|
||||||
|
|
||||||
if (code === 401 || code === 403) {
|
if (code === 401) {
|
||||||
|
antMessage.error(message);
|
||||||
localStorage.removeItem('Authorization');
|
localStorage.removeItem('Authorization');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,20 @@
|
||||||
|
module.exports = {
|
||||||
|
mode: 'jit',
|
||||||
|
purge: ['./public/**/*.html', './src/**/*.{js,jsx,ts,tsx}'],
|
||||||
|
darkMode: false, // or 'media' or 'class'
|
||||||
|
theme: {
|
||||||
|
backgroundColor: (theme) => ({
|
||||||
|
...theme('colors'),
|
||||||
|
dark70: 'rgba(0,0,0,.7)',
|
||||||
|
}),
|
||||||
|
extend: {
|
||||||
|
colors: {
|
||||||
|
primary: '#1677ff',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
variants: {
|
||||||
|
extend: {},
|
||||||
|
},
|
||||||
|
plugins: [],
|
||||||
|
};
|
|
@ -3,3 +3,12 @@ import '@umijs/max/typings';
|
||||||
declare global {
|
declare global {
|
||||||
declare type EditRow<T> = T | undefined;
|
declare type EditRow<T> = T | undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
declare module '*.less';
|
||||||
|
declare module '*.svg';
|
||||||
|
declare module '*.png';
|
||||||
|
declare module '*.jpg';
|
||||||
|
declare module '*.jpeg';
|
||||||
|
declare module '*.gif';
|
||||||
|
declare module '*.bmp';
|
||||||
|
declare module '*.tiff';
|
||||||
|
|
|
@ -0,0 +1,10 @@
|
||||||
|
import { defineConfig, presetAttributify, presetUno } from 'unocss';
|
||||||
|
|
||||||
|
export function createConfig({ strict = true, dev = true } = {}) {
|
||||||
|
return defineConfig({
|
||||||
|
envMode: dev ? 'dev' : 'build',
|
||||||
|
presets: [presetAttributify({ strict }), presetUno()],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export default createConfig();
|
Loading…
Reference in New Issue