bigbuffet-rw/app/soapbox/components/status-action-button.tsx

108 lines
3 KiB
TypeScript
Raw Normal View History

2023-02-06 10:01:03 -08:00
import clsx from 'clsx';
2022-04-02 16:43:34 -07:00
import React from 'react';
import { Text, Icon, Emoji } from 'soapbox/components/ui';
2022-04-02 16:43:34 -07:00
import { shortNumberFormat } from 'soapbox/utils/numbers';
import type { Map as ImmutableMap } from 'immutable';
2022-04-02 16:43:34 -07:00
const COLORS = {
accent: 'accent',
success: 'success',
2022-04-02 16:43:34 -07:00
};
type Color = keyof typeof COLORS;
interface IStatusActionCounter {
count: number
2022-04-02 16:43:34 -07:00
}
/** Action button numerical counter, eg "5" likes. */
2022-04-02 16:43:34 -07:00
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?: ImmutableMap<string, any>
text?: React.ReactNode
theme?: 'default' | 'inverse'
2022-04-02 16:43:34 -07:00
}
const StatusActionButton = React.forwardRef<HTMLButtonElement, IStatusActionButton>((props, ref): JSX.Element => {
const { icon, className, iconClassName, active, color, filled = false, count = 0, emoji, text, theme = 'default', ...filteredProps } = props;
2022-08-09 16:40:33 -07:00
const renderIcon = () => {
if (emoji) {
return (
2023-02-01 14:13:42 -08:00
<span className='flex h-6 w-6 items-center justify-center'>
<Emoji className='h-full w-full p-0.5' emoji={emoji.get('name')} src={emoji.get('url')} />
2022-08-09 16:40:33 -07:00
</span>
);
} else {
return (
<Icon
src={icon}
2023-02-06 10:01:03 -08:00
className={clsx(
2022-08-09 16:40:33 -07:00
{
2023-04-12 13:09:18 -07:00
'fill-accent-300 text-accent-300 hover:fill-accent-300': active && filled && color === COLORS.accent,
2022-08-09 16:40:33 -07:00
},
iconClassName,
)}
/>
);
}
};
const renderText = () => {
if (text) {
return (
<Text tag='span' theme='inherit' size='sm'>
{text}
</Text>
);
} else if (count) {
return (
<StatusActionCounter count={count} />
);
}
};
2022-04-02 16:43:34 -07:00
return (
<button
ref={ref}
type='button'
2023-02-06 10:01:03 -08:00
className={clsx(
2023-02-06 10:06:44 -08:00
'flex items-center rounded-full p-1 rtl:space-x-reverse',
'focus:outline-none focus:ring-2 focus:ring-primary-500 focus:ring-offset-2 dark:ring-offset-0',
2022-04-02 16:43:34 -07:00
{
'text-gray-600 hover:text-gray-600 dark:hover:text-white bg-white dark:bg-transparent': theme === 'default',
'text-white/80 hover:text-white bg-transparent dark:bg-transparent': theme === 'inverse',
'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,
2022-08-09 16:46:16 -07:00
'space-x-2': text,
2022-04-02 16:43:34 -07:00
},
className,
)}
{...filteredProps}
>
2022-08-09 16:40:33 -07:00
{renderIcon()}
{renderText()}
2022-04-02 16:43:34 -07:00
</button>
);
});
export default StatusActionButton;