mirror of
https://github.com/ant-design/ant-design.git
synced 2025-01-12 07:40:51 +08:00
120 lines
3.3 KiB
TypeScript
120 lines
3.3 KiB
TypeScript
import React from 'react';
|
|
import * as AntdIcons from '@ant-design/icons';
|
|
import { App, Badge } from 'antd';
|
|
import { createStyles } from 'antd-style';
|
|
import classNames from 'classnames';
|
|
import CopyToClipboard from 'react-copy-to-clipboard';
|
|
|
|
import type { ThemeType } from './IconSearch';
|
|
|
|
const allIcons: { [key: PropertyKey]: any } = AntdIcons;
|
|
|
|
const useStyle = createStyles(({ token, css }) => {
|
|
const { antCls, iconCls } = token;
|
|
return {
|
|
iconItem: css`
|
|
display: inline-flex;
|
|
flex-direction: column;
|
|
justify-content: center;
|
|
align-items: center;
|
|
margin-inline-start: 0 !important;
|
|
margin-inline-end: 0 !important;
|
|
padding-inline-start: 0 !important;
|
|
padding-inline-end: 0 !important;
|
|
position: relative;
|
|
width: 200px;
|
|
height: 100px;
|
|
overflow: hidden;
|
|
color: #555;
|
|
text-align: center;
|
|
list-style: none;
|
|
background-color: inherit;
|
|
border-radius: ${token.borderRadiusSM}px;
|
|
cursor: pointer;
|
|
transition: all ${token.motionDurationSlow} ease-in-out;
|
|
${token.iconCls} {
|
|
margin: ${token.marginXS}px 0;
|
|
font-size: 36px;
|
|
transition: transform ${token.motionDurationSlow} ease-in-out;
|
|
will-change: transform;
|
|
}
|
|
&:hover {
|
|
color: ${token.colorWhite};
|
|
background-color: ${token.colorPrimary};
|
|
${iconCls} {
|
|
transform: scale(1.3);
|
|
}
|
|
${antCls}-badge {
|
|
color: ${token.colorWhite};
|
|
}
|
|
}
|
|
&.TwoTone:hover {
|
|
background-color: #8ecafe;
|
|
}
|
|
&.copied:hover {
|
|
color: rgba(255, 255, 255, 0.2);
|
|
}
|
|
&::after {
|
|
content: 'Copied!';
|
|
position: absolute;
|
|
top: 0;
|
|
inset-inline-start: 0;
|
|
width: 100%;
|
|
height: 100%;
|
|
line-height: 100px;
|
|
color: ${token.colorTextLightSolid};
|
|
text-align: center;
|
|
background-color: ${token.colorPrimary};
|
|
opacity: 0;
|
|
transition: all ${token.motionDurationSlow} cubic-bezier(0.18, 0.89, 0.32, 1.28);
|
|
}
|
|
&.copied::after {
|
|
opacity: 1;
|
|
}
|
|
`,
|
|
anticonCls: css`
|
|
display: block;
|
|
font-family: 'Lucida Console', Consolas, Monaco, 'Andale Mono', 'Ubuntu Mono', monospace;
|
|
white-space: nowrap;
|
|
text-align: center;
|
|
transform: scale(0.8);
|
|
${antCls}-badge {
|
|
transition: color ${token.motionDurationSlow} ease-in-out;
|
|
}
|
|
`,
|
|
};
|
|
});
|
|
|
|
export interface CopyableIconProps {
|
|
name: string;
|
|
isNew: boolean;
|
|
theme: ThemeType;
|
|
justCopied: string | null;
|
|
onCopied: (type: string, text: string) => void;
|
|
}
|
|
|
|
const CopyableIcon: React.FC<CopyableIconProps> = (props) => {
|
|
const { message } = App.useApp();
|
|
const { name, isNew, justCopied, theme, onCopied } = props;
|
|
const { styles } = useStyle();
|
|
const onCopy = (text: string, result: boolean) => {
|
|
if (result) {
|
|
onCopied(name, text);
|
|
} else {
|
|
message.error('Copy icon name failed, please try again.');
|
|
}
|
|
};
|
|
return (
|
|
<CopyToClipboard text={`<${name} />`} onCopy={onCopy}>
|
|
<li className={classNames(theme, styles.iconItem, { copied: justCopied === name })}>
|
|
{React.createElement(allIcons[name])}
|
|
<span className={styles.anticonCls}>
|
|
<Badge dot={isNew}>{name}</Badge>
|
|
</span>
|
|
</li>
|
|
</CopyToClipboard>
|
|
);
|
|
};
|
|
|
|
export default CopyableIcon;
|