bigbuffet-rw/app/soapbox/components/ui/button/button.tsx

107 lines
2.4 KiB
TypeScript
Raw Normal View History

2023-02-06 10:01:03 -08:00
import clsx from 'clsx';
import React from 'react';
2022-03-21 11:09:01 -07:00
import { Link } from 'react-router-dom';
2022-04-05 07:42:19 -07:00
import Icon from '../icon/icon';
2022-03-21 11:09:01 -07:00
import { useButtonStyles } from './useButtonStyles';
import type { ButtonSizes, ButtonThemes } from './useButtonStyles';
interface IButton {
/** Whether this button expands the width of its container. */
2022-03-21 11:09:01 -07:00
block?: boolean,
/** Elements inside the <button> */
2022-03-21 11:09:01 -07:00
children?: React.ReactNode,
/** Extra class names for the button. */
className?: string,
/** Prevent the button from being clicked. */
2022-03-21 11:09:01 -07:00
disabled?: boolean,
/** URL to an SVG icon to render inside the button. */
2022-03-21 11:09:01 -07:00
icon?: string,
/** Action when the button is clicked. */
onClick?: (event: React.MouseEvent<HTMLButtonElement>) => void,
/** A predefined button size. */
2022-03-21 11:09:01 -07:00
size?: ButtonSizes,
/** Text inside the button. Takes precedence over `children`. */
2022-03-21 11:09:01 -07:00
text?: React.ReactNode,
/** Makes the button into a navlink, if provided. */
2022-03-21 11:09:01 -07:00
to?: string,
/** Styles the button visually with a predefined theme. */
2022-03-21 11:09:01 -07:00
theme?: ButtonThemes,
/** Whether this button should submit a form by default. */
2022-03-21 11:09:01 -07:00
type?: 'button' | 'submit',
}
/** Customizable button element with various themes. */
2022-03-21 11:09:01 -07:00
const Button = React.forwardRef<HTMLButtonElement, IButton>((props, ref): JSX.Element => {
const {
block = false,
children,
disabled = false,
icon,
onClick,
size = 'md',
text,
theme = 'secondary',
2022-03-21 11:09:01 -07:00
to,
type = 'button',
className,
2022-03-21 11:09:01 -07:00
} = props;
const body = text || children;
2022-03-21 11:09:01 -07:00
const themeClass = useButtonStyles({
theme,
block,
disabled,
size,
});
const renderIcon = () => {
if (!icon) {
return null;
}
2023-02-01 14:13:42 -08:00
return <Icon src={icon} className='h-4 w-4' />;
2022-03-21 11:09:01 -07:00
};
2023-01-10 15:03:15 -08:00
const handleClick: React.MouseEventHandler<HTMLButtonElement> = React.useCallback((event) => {
2022-03-21 11:09:01 -07:00
if (onClick && !disabled) {
onClick(event);
}
}, [onClick, disabled]);
const renderButton = () => (
<button
2023-02-06 10:01:03 -08:00
className={clsx('space-x-2 rtl:space-x-reverse', themeClass, className)}
2022-03-21 11:09:01 -07:00
disabled={disabled}
onClick={handleClick}
ref={ref}
type={type}
data-testid='button'
2022-03-21 11:09:01 -07:00
>
{renderIcon()}
{body && (
<span>{body}</span>
)}
2022-03-21 11:09:01 -07:00
</button>
);
if (to) {
return (
<Link to={to} tabIndex={-1} className='inline-flex'>
{renderButton()}
</Link>
);
}
return renderButton();
});
2023-02-02 21:00:51 -08:00
export {
Button as default,
Button,
};