2022-12-05 14:15:26 +08:00
|
|
|
import { ReloadOutlined } from '@ant-design/icons';
|
2023-02-22 18:18:26 +08:00
|
|
|
import classNames from 'classnames';
|
|
|
|
import { QRCodeCanvas } from 'qrcode.react';
|
|
|
|
import React, { useContext, useMemo } from 'react';
|
|
|
|
import Button from '../button';
|
2022-12-05 14:15:26 +08:00
|
|
|
import type { ConfigConsumerProps } from '../config-provider';
|
2023-02-22 18:18:26 +08:00
|
|
|
import { ConfigContext } from '../config-provider';
|
|
|
|
import useLocale from '../locale/useLocale';
|
2022-12-05 14:15:26 +08:00
|
|
|
import Spin from '../spin';
|
|
|
|
import theme from '../theme';
|
2023-02-22 18:18:26 +08:00
|
|
|
import warning from '../_util/warning';
|
|
|
|
import type { QRCodeProps, QRPropsCanvas } from './interface';
|
|
|
|
import useStyle from './style/index';
|
2022-12-05 14:15:26 +08:00
|
|
|
|
|
|
|
const { useToken } = theme;
|
|
|
|
|
|
|
|
const QRCode: React.FC<QRCodeProps> = (props) => {
|
|
|
|
const {
|
|
|
|
value,
|
|
|
|
icon = '',
|
|
|
|
size = 160,
|
|
|
|
iconSize = 40,
|
|
|
|
color = '#000',
|
|
|
|
errorLevel = 'M',
|
|
|
|
status = 'active',
|
|
|
|
bordered = true,
|
|
|
|
onRefresh,
|
|
|
|
style,
|
|
|
|
className,
|
2023-01-20 11:03:50 +08:00
|
|
|
rootClassName,
|
2022-12-05 14:15:26 +08:00
|
|
|
prefixCls: customizePrefixCls,
|
|
|
|
} = props;
|
|
|
|
const { getPrefixCls } = useContext<ConfigConsumerProps>(ConfigContext);
|
|
|
|
const prefixCls = getPrefixCls('qrcode', customizePrefixCls);
|
|
|
|
const [wrapSSR, hashId] = useStyle(prefixCls);
|
|
|
|
const { token } = useToken();
|
|
|
|
const qrCodeProps = useMemo<QRPropsCanvas>(() => {
|
|
|
|
const imageSettings: QRCodeProps['imageSettings'] = {
|
|
|
|
src: icon,
|
|
|
|
x: undefined,
|
|
|
|
y: undefined,
|
|
|
|
height: iconSize,
|
|
|
|
width: iconSize,
|
|
|
|
excavate: true,
|
|
|
|
};
|
|
|
|
return {
|
|
|
|
value,
|
|
|
|
size: size - (token.paddingSM + token.lineWidth) * 2,
|
|
|
|
level: errorLevel,
|
|
|
|
bgColor: 'transparent',
|
|
|
|
fgColor: color,
|
|
|
|
imageSettings: icon ? imageSettings : undefined,
|
|
|
|
};
|
|
|
|
}, [errorLevel, color, icon, iconSize, size, value]);
|
|
|
|
|
2023-02-22 18:18:26 +08:00
|
|
|
const locale = useLocale('QRCode');
|
|
|
|
|
2022-12-05 14:15:26 +08:00
|
|
|
if (!value) {
|
|
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
|
|
warning(false, 'QRCode', 'need to receive `value` props');
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2022-12-07 23:32:04 +08:00
|
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
|
|
warning(
|
|
|
|
!(icon && errorLevel === 'L'),
|
|
|
|
'QRCode',
|
|
|
|
'ErrorLevel `L` is not recommended to be used with `icon`, for scanning result would be affected by low level.',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-01-20 11:03:50 +08:00
|
|
|
const cls = classNames(prefixCls, className, rootClassName, hashId, {
|
2022-12-05 14:15:26 +08:00
|
|
|
[`${prefixCls}-borderless`]: !bordered,
|
|
|
|
});
|
|
|
|
|
|
|
|
return wrapSSR(
|
2023-02-22 18:18:26 +08:00
|
|
|
<div style={{ ...style, width: size, height: size }} className={cls}>
|
|
|
|
{status !== 'active' && (
|
|
|
|
<div className={`${prefixCls}-mask`}>
|
|
|
|
{status === 'loading' && <Spin />}
|
|
|
|
{status === 'expired' && (
|
|
|
|
<>
|
|
|
|
<p className={`${prefixCls}-expired`}>{locale?.expired}</p>
|
|
|
|
{typeof onRefresh === 'function' && (
|
|
|
|
<Button type="link" icon={<ReloadOutlined />} onClick={onRefresh}>
|
|
|
|
{locale?.refresh}
|
|
|
|
</Button>
|
2022-12-05 14:15:26 +08:00
|
|
|
)}
|
2023-02-22 18:18:26 +08:00
|
|
|
</>
|
2022-12-05 14:15:26 +08:00
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
)}
|
2023-02-22 18:18:26 +08:00
|
|
|
<QRCodeCanvas {...qrCodeProps} />
|
|
|
|
</div>,
|
2022-12-05 14:15:26 +08:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2023-01-08 21:30:41 +08:00
|
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
|
|
QRCode.displayName = 'QRCode';
|
|
|
|
}
|
|
|
|
|
2022-12-05 14:15:26 +08:00
|
|
|
export default QRCode;
|