104 lines
2.8 KiB
TypeScript
104 lines
2.8 KiB
TypeScript
import classNames from 'clsx';
|
|
import React from 'react';
|
|
|
|
import { Text, Icon, Emoji } from 'soapbox/components/ui';
|
|
import { shortNumberFormat } from 'soapbox/utils/numbers';
|
|
|
|
const COLORS = {
|
|
accent: 'accent',
|
|
success: 'success',
|
|
};
|
|
|
|
type Color = keyof typeof COLORS;
|
|
|
|
interface IStatusActionCounter {
|
|
count: number,
|
|
}
|
|
|
|
/** Action button numerical counter, eg "5" likes. */
|
|
const StatusActionCounter: React.FC<IStatusActionCounter> = ({ count = 0 }): JSX.Element => {
|
|
return (
|
|
<Text size='xs' weight='semibold' theme='inherit'>
|
|
{shortNumberFormat(count)}
|
|
</Text>
|
|
);
|
|
};
|
|
|
|
interface IStatusActionButton extends React.ButtonHTMLAttributes<HTMLButtonElement> {
|
|
iconClassName?: string,
|
|
icon: string,
|
|
count?: number,
|
|
active?: boolean,
|
|
color?: Color,
|
|
filled?: boolean,
|
|
emoji?: string,
|
|
text?: React.ReactNode,
|
|
}
|
|
|
|
const StatusActionButton = React.forwardRef<HTMLButtonElement, IStatusActionButton>((props, ref): JSX.Element => {
|
|
const { icon, className, iconClassName, active, color, filled = false, count = 0, emoji, text, ...filteredProps } = props;
|
|
|
|
const renderIcon = () => {
|
|
if (emoji) {
|
|
return (
|
|
<span className='flex h-6 w-6 items-center justify-center'>
|
|
<Emoji className='h-full w-full p-0.5' emoji={emoji} />
|
|
</span>
|
|
);
|
|
} else {
|
|
return (
|
|
<Icon
|
|
src={icon}
|
|
className={classNames(
|
|
{
|
|
'fill-accent-300 hover:fill-accent-300': active && filled && color === COLORS.accent,
|
|
},
|
|
iconClassName,
|
|
)}
|
|
/>
|
|
);
|
|
}
|
|
};
|
|
|
|
const renderText = () => {
|
|
if (text) {
|
|
return (
|
|
<Text tag='span' theme='inherit' size='sm'>
|
|
{text}
|
|
</Text>
|
|
);
|
|
} else if (count) {
|
|
return (
|
|
<StatusActionCounter count={count} />
|
|
);
|
|
}
|
|
};
|
|
|
|
return (
|
|
<button
|
|
ref={ref}
|
|
type='button'
|
|
className={classNames(
|
|
'flex items-center p-1 rounded-full rtl:space-x-reverse',
|
|
'text-gray-600 hover:text-gray-600 dark:hover:text-white',
|
|
'bg-white dark:bg-transparent',
|
|
'focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-primary-500 dark:ring-offset-0',
|
|
{
|
|
'text-black dark:text-white': active && emoji,
|
|
'hover:text-gray-600 dark:hover:text-white': !filteredProps.disabled,
|
|
'text-accent-300 hover:text-accent-300 dark:hover:text-accent-300': active && !emoji && color === COLORS.accent,
|
|
'text-success-600 hover:text-success-600 dark:hover:text-success-600': active && !emoji && color === COLORS.success,
|
|
'space-x-1': !text,
|
|
'space-x-2': text,
|
|
},
|
|
className,
|
|
)}
|
|
{...filteredProps}
|
|
>
|
|
{renderIcon()}
|
|
{renderText()}
|
|
</button>
|
|
);
|
|
});
|
|
|
|
export default StatusActionButton;
|