You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
263 lines
8.0 KiB
263 lines
8.0 KiB
import {
|
|
AlipayCircleOutlined,
|
|
LockOutlined,
|
|
MailOutlined,
|
|
MobileOutlined,
|
|
TaobaoCircleOutlined,
|
|
UserOutlined,
|
|
WeiboCircleOutlined,
|
|
} from '@ant-design/icons';
|
|
import { Alert, Space, message, Tabs } from 'antd';
|
|
import React, { useState } from 'react';
|
|
import ProForm, { ProFormCaptcha, ProFormCheckbox, ProFormText } from '@ant-design/pro-form';
|
|
import { useIntl, connect, FormattedMessage } from 'umi';
|
|
import { getFakeCaptcha } from '@/services/login';
|
|
import type { Dispatch } from 'umi';
|
|
import type { StateType } from '@/models/login';
|
|
import type { LoginParamsType } from '@/services/login';
|
|
import type { ConnectState } from '@/models/connect';
|
|
|
|
import styles from './index.less';
|
|
|
|
export type LoginProps = {
|
|
dispatch: Dispatch;
|
|
userLogin: StateType;
|
|
submitting?: boolean;
|
|
};
|
|
|
|
const LoginMessage: React.FC<{
|
|
content: string;
|
|
}> = ({ content }) => (
|
|
<Alert
|
|
style={{
|
|
marginBottom: 24,
|
|
}}
|
|
message={content}
|
|
type="error"
|
|
showIcon
|
|
/>
|
|
);
|
|
|
|
const Login: React.FC<LoginProps> = (props) => {
|
|
const { userLogin = {}, submitting } = props;
|
|
const { status, type: loginType } = userLogin;
|
|
const [type, setType] = useState<string>('account');
|
|
const intl = useIntl();
|
|
|
|
const handleSubmit = (values: LoginParamsType) => {
|
|
const { dispatch } = props;
|
|
dispatch({
|
|
type: 'login/login',
|
|
payload: { ...values, type },
|
|
});
|
|
};
|
|
return (
|
|
<div className={styles.main}>
|
|
<ProForm
|
|
initialValues={{
|
|
autoLogin: true,
|
|
}}
|
|
submitter={{
|
|
render: (_, dom) => dom.pop(),
|
|
submitButtonProps: {
|
|
loading: submitting,
|
|
size: 'large',
|
|
style: {
|
|
width: '100%',
|
|
},
|
|
},
|
|
}}
|
|
onFinish={(values) => {
|
|
handleSubmit(values as LoginParamsType);
|
|
return Promise.resolve();
|
|
}}
|
|
>
|
|
<Tabs activeKey={type} onChange={setType}>
|
|
<Tabs.TabPane
|
|
key="account"
|
|
tab={intl.formatMessage({
|
|
id: 'pages.login.accountLogin.tab',
|
|
defaultMessage: 'Account password login',
|
|
})}
|
|
/>
|
|
<Tabs.TabPane
|
|
key="mobile"
|
|
tab={intl.formatMessage({
|
|
id: 'pages.login.phoneLogin.tab',
|
|
defaultMessage: 'Mobile phone number login',
|
|
})}
|
|
/>
|
|
</Tabs>
|
|
|
|
{status === 'error' && loginType === 'account' && !submitting && (
|
|
<LoginMessage
|
|
content={intl.formatMessage({
|
|
id: 'pages.login.accountLogin.errorMessage',
|
|
defaultMessage: 'Incorrect account or password(admin/ant.design)',
|
|
})}
|
|
/>
|
|
)}
|
|
{type === 'account' && (
|
|
<>
|
|
<ProFormText
|
|
name="userName"
|
|
fieldProps={{
|
|
size: 'large',
|
|
prefix: <UserOutlined className={styles.prefixIcon} />,
|
|
}}
|
|
placeholder={intl.formatMessage({
|
|
id: 'pages.login.username.placeholder',
|
|
defaultMessage: 'Username: admin or user',
|
|
})}
|
|
rules={[
|
|
{
|
|
required: true,
|
|
message: (
|
|
<FormattedMessage
|
|
id="pages.login.username.required"
|
|
defaultMessage="Please enter user name!"
|
|
/>
|
|
),
|
|
},
|
|
]}
|
|
/>
|
|
<ProFormText.Password
|
|
name="password"
|
|
fieldProps={{
|
|
size: 'large',
|
|
prefix: <LockOutlined className={styles.prefixIcon} />,
|
|
}}
|
|
placeholder={intl.formatMessage({
|
|
id: 'pages.login.password.placeholder',
|
|
defaultMessage: 'Password: ant.design',
|
|
})}
|
|
rules={[
|
|
{
|
|
required: true,
|
|
message: (
|
|
<FormattedMessage
|
|
id="pages.login.password.required"
|
|
defaultMessage="Please enter password!"
|
|
/>
|
|
),
|
|
},
|
|
]}
|
|
/>
|
|
</>
|
|
)}
|
|
|
|
{status === 'error' && loginType === 'mobile' && !submitting && (
|
|
<LoginMessage content="Verification code error" />
|
|
)}
|
|
{type === 'mobile' && (
|
|
<>
|
|
<ProFormText
|
|
fieldProps={{
|
|
size: 'large',
|
|
prefix: <MobileOutlined className={styles.prefixIcon} />,
|
|
}}
|
|
name="mobile"
|
|
placeholder={intl.formatMessage({
|
|
id: 'pages.login.phoneNumber.placeholder',
|
|
defaultMessage: 'Phone number',
|
|
})}
|
|
rules={[
|
|
{
|
|
required: true,
|
|
message: (
|
|
<FormattedMessage
|
|
id="pages.login.phoneNumber.required"
|
|
defaultMessage="Please enter phone number!"
|
|
/>
|
|
),
|
|
},
|
|
{
|
|
pattern: /^1\d{10}$/,
|
|
message: (
|
|
<FormattedMessage
|
|
id="pages.login.phoneNumber.invalid"
|
|
defaultMessage="Malformed phone number!"
|
|
/>
|
|
),
|
|
},
|
|
]}
|
|
/>
|
|
<ProFormCaptcha
|
|
fieldProps={{
|
|
size: 'large',
|
|
prefix: <MailOutlined className={styles.prefixIcon} />,
|
|
}}
|
|
captchaProps={{
|
|
size: 'large',
|
|
}}
|
|
placeholder={intl.formatMessage({
|
|
id: 'pages.login.captcha.placeholder',
|
|
defaultMessage: 'Please enter verification code',
|
|
})}
|
|
captchaTextRender={(timing, count) => {
|
|
if (timing) {
|
|
return `${count} ${intl.formatMessage({
|
|
id: 'pages.getCaptchaSecondText',
|
|
defaultMessage: 'Get verification code',
|
|
})}`;
|
|
}
|
|
return intl.formatMessage({
|
|
id: 'pages.login.phoneLogin.getVerificationCode',
|
|
defaultMessage: 'Get verification code',
|
|
});
|
|
}}
|
|
name="captcha"
|
|
rules={[
|
|
{
|
|
required: true,
|
|
message: (
|
|
<FormattedMessage
|
|
id="pages.login.captcha.required"
|
|
defaultMessage="Please enter verification code!"
|
|
/>
|
|
),
|
|
},
|
|
]}
|
|
onGetCaptcha={async (mobile) => {
|
|
const result = await getFakeCaptcha(mobile);
|
|
if (result === false) {
|
|
return;
|
|
}
|
|
message.success(
|
|
'Get the verification code successfully! The verification code is: 1234',
|
|
);
|
|
}}
|
|
/>
|
|
</>
|
|
)}
|
|
<div
|
|
style={{
|
|
marginBottom: 24,
|
|
}}
|
|
>
|
|
<ProFormCheckbox noStyle name="autoLogin">
|
|
<FormattedMessage id="pages.login.rememberMe" defaultMessage="Auto login" />
|
|
</ProFormCheckbox>
|
|
<a
|
|
style={{
|
|
float: 'right',
|
|
}}
|
|
>
|
|
<FormattedMessage id="pages.login.forgotPassword" defaultMessage="Forget password" />
|
|
</a>
|
|
</div>
|
|
</ProForm>
|
|
<Space className={styles.other}>
|
|
<FormattedMessage id="pages.login.loginWith" defaultMessage="Other login methods" />
|
|
<AlipayCircleOutlined className={styles.icon} />
|
|
<TaobaoCircleOutlined className={styles.icon} />
|
|
<WeiboCircleOutlined className={styles.icon} />
|
|
</Space>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default connect(({ login, loading }: ConnectState) => ({
|
|
userLogin: login,
|
|
submitting: loading.effects['login/login'],
|
|
}))(Login);
|
|
|