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

99 lines
2.3 KiB
TypeScript
Raw Normal View History

import classNames from 'clsx';
2022-03-21 11:09:01 -07:00
import * as React from 'react';
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. */
2022-03-21 11:09:01 -07:00
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 themeClass = useButtonStyles({
theme,
block,
disabled,
size,
});
const renderIcon = () => {
if (!icon) {
return null;
}
2022-04-05 07:42:19 -07:00
return <Icon src={icon} className='mr-2 w-4 h-4' />;
2022-03-21 11:09:01 -07:00
};
const handleClick = React.useCallback((event) => {
if (onClick && !disabled) {
onClick(event);
}
}, [onClick, disabled]);
const renderButton = () => (
<button
className={classNames(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()}
{text || children}
</button>
);
if (to) {
return (
<Link to={to} tabIndex={-1} className='inline-flex'>
{renderButton()}
</Link>
);
}
return renderButton();
});
export default Button;