258 lines
7.3 KiB
JavaScript
Raw Normal View History

2020-02-20 19:57:29 -05:00
import { Fragment } from 'react'
2020-02-19 18:57:07 -05:00
import { NavLink } from 'react-router-dom'
import classNames from 'classnames/bind'
2020-02-20 19:57:29 -05:00
import Icon from './icon'
2020-02-19 18:57:07 -05:00
2020-04-23 02:13:29 -04:00
// Bind CSS Modules global variable `_s` to classNames module
2020-02-19 18:57:07 -05:00
const cx = classNames.bind(_s)
2020-04-23 02:13:29 -04:00
// Define colors for enumeration for Button component `color`, `backgroundColor` props
2020-02-19 18:57:07 -05:00
const COLORS = {
primary: 'primary',
secondary: 'secondary',
tertiary: 'tertiary',
2020-02-20 19:57:29 -05:00
white: 'white',
2020-03-25 23:11:32 -04:00
black: 'black',
2020-02-19 18:57:07 -05:00
brand: 'brand',
2020-03-25 23:11:32 -04:00
danger: 'danger',
2020-02-19 18:57:07 -05:00
none: 'none',
}
2020-04-23 02:13:29 -04:00
/**
* Renders a button component
* @param {string} [props.backgroundColor='brand'] - background color of the button
* @param {func|node} [props.buttonRef] - ref to send to button component
* @param {string} [props.className] - add custom className
* @param {string} [props.color='white'] - text color of the button
* @param {string} [props.href] - href to send to on click
* @param {string} [props.icon] - prepend icon id
* @param {string} [props.iconClassName] - add custom className to icon
* @param {string} [props.iconSize] - size of the icon
* @param {bool} [props.isBlock] - if button is width: 100%
* @param {bool} [props.isDisabled] - if the button is disabled
* @param {bool} [props.isNarrow] - if the button is narrow
* @param {bool} [props.isOutline] - if the button is outline design
* @param {bool} [props.noClasses] - if the button has no default classes
* @param {func} [props.onClick] - function to call on button click
* @param {func} [props.onMouseEnter] - function to call on button mouse enter
* @param {func} [props.onMouseLeave] - function to call on button mouse leave
* @param {bool} [props.radiusSmall] - if the button has small radius
2020-04-29 18:32:49 -04:00
* @param {bool} [props.rel] - rel for the button
* @param {bool} [props.target] - target for the button
2020-04-23 02:13:29 -04:00
* @param {bool} [props.text] - if the button is just text (i.e. link)
* @param {bool} [props.title] - `title` attribute for button
* @param {bool} [props.to] - `to` to send to on click
* @param {bool} [props.type] - `type` attribute for button
* @param {bool} [props.underlineOnHover] - if the button has underline on hover
*/
2020-02-19 18:57:07 -05:00
export default class Button extends PureComponent {
static propTypes = {
2020-04-23 02:13:29 -04:00
backgroundColor: PropTypes.string,
buttonRef: PropTypes.oneOfType([
PropTypes.func,
PropTypes.node,
]),
2020-02-19 18:57:07 -05:00
children: PropTypes.node,
className: PropTypes.string,
2020-04-23 02:13:29 -04:00
color: PropTypes.string,
href: PropTypes.string,
2020-02-19 18:57:07 -05:00
icon: PropTypes.string,
2020-02-20 19:57:29 -05:00
iconClassName: PropTypes.string,
2020-04-23 02:13:29 -04:00
iconSize: PropTypes.string,
isBlock: PropTypes.bool,
isDisabled: PropTypes.bool,
isNarrow: PropTypes.bool,
isText: PropTypes.bool,
2020-03-14 13:31:29 -04:00
noClasses: PropTypes.bool,
2020-04-23 02:13:29 -04:00
onClick: PropTypes.func,
onMouseEnter: PropTypes.func,
onMouseLeave: PropTypes.func,
isOutline: PropTypes.bool,
radiusSmall: PropTypes.bool,
2020-04-29 18:32:49 -04:00
rel: PropTypes.string,
target: PropTypes.string,
2020-04-23 02:13:29 -04:00
title: PropTypes.string,
to: PropTypes.string,
type: PropTypes.string,
underlineOnHover: PropTypes.bool,
2020-02-19 18:57:07 -05:00
}
static defaultProps = {
2020-02-20 19:57:29 -05:00
color: COLORS.white,
backgroundColor: COLORS.brand,
2020-02-19 18:57:07 -05:00
}
handleClick = (e) => {
2020-04-23 02:13:29 -04:00
if (!this.props.isDisabled && this.props.onClick) {
2020-02-19 18:57:07 -05:00
this.props.onClick(e)
}
}
handleOnMouseEnter = () => {
if (!this.props.isDisabled && this.props.onMouseEnter) {
this.props.onMouseEnter()
}
}
handleOnMouseLeave = () => {
if (!this.props.isDisabled && this.props.onMouseLeave) {
this.props.onMouseLeave()
}
}
2020-02-19 18:57:07 -05:00
setRef = (c) => {
2020-04-23 02:13:29 -04:00
try {
this.node = c
this.props.buttonRef(c)
} catch (error) {
//
}
2020-02-19 18:57:07 -05:00
}
focus() {
this.node.focus()
}
2020-02-22 18:26:23 -05:00
render() {
2020-02-20 19:57:29 -05:00
const {
2020-04-23 02:13:29 -04:00
backgroundColor,
children,
2020-02-20 19:57:29 -05:00
className,
2020-04-23 02:13:29 -04:00
color,
href,
2020-02-20 19:57:29 -05:00
icon,
iconClassName,
2020-04-23 02:13:29 -04:00
iconSize,
isBlock,
isDisabled,
isNarrow,
isOutline,
isText,
2020-03-14 13:31:29 -04:00
noClasses,
2020-04-23 02:13:29 -04:00
onClick,
onMouseEnter,
onMouseLeave,
radiusSmall,
2020-04-29 18:32:49 -04:00
rel,
target,
2020-04-23 02:13:29 -04:00
title,
to,
type,
underlineOnHover,
2020-02-20 19:57:29 -05:00
} = this.props
2020-04-23 02:13:29 -04:00
// Style the component according to props
2020-03-14 13:31:29 -04:00
const classes = noClasses ? className : cx(className, {
2020-02-19 18:57:07 -05:00
default: 1,
noUnderline: 1,
font: 1,
cursorPointer: 1,
textAlignCenter: 1,
2020-03-06 10:38:22 -05:00
outlineNone: 1,
2020-05-03 01:22:49 -04:00
// outlineOnFocus: !isText,
2020-03-06 23:53:28 -05:00
flexRow: !!children && !!icon,
2020-04-23 02:13:29 -04:00
cursorNotAllowed: isDisabled,
opacity05: isDisabled,
2020-02-19 18:57:07 -05:00
bgPrimary: backgroundColor === COLORS.primary,
bgWhite: backgroundColor === COLORS.white,
2020-04-29 18:32:49 -04:00
bgBlack: backgroundColor === COLORS.black,
bgBrand: backgroundColor === COLORS.brand,
bgTransparent: backgroundColor === COLORS.none,
bgSecondary: backgroundColor === COLORS.tertiary,
bgSubtle: backgroundColor === COLORS.secondary,
bgDanger: backgroundColor === COLORS.danger,
2020-03-25 23:11:32 -04:00
colorPrimary: !!children && color === COLORS.primary,
colorSecondary: !!children && color === COLORS.secondary,
colorTertiary: !!children && color === COLORS.tertiary,
colorWhite: !!children && color === COLORS.white,
colorBrand: !!children && color === COLORS.brand,
2020-02-19 18:57:07 -05:00
2020-04-23 02:13:29 -04:00
borderColorBrand: color === COLORS.brand && isOutline,
border1PX: isOutline,
2020-02-19 18:57:07 -05:00
2020-04-23 02:13:29 -04:00
circle: !isText,
2020-02-22 18:26:23 -05:00
radiusSmall: radiusSmall,
2020-02-19 18:57:07 -05:00
2020-04-23 02:13:29 -04:00
py5: isNarrow,
py10: !isText && !isNarrow,
px15: !isText,
2020-02-19 18:57:07 -05:00
2020-04-23 02:13:29 -04:00
width100PC: isBlock,
2020-02-20 19:57:29 -05:00
underline_onHover: underlineOnHover,
2020-04-29 18:32:49 -04:00
bgSecondaryDark_onHover: backgroundColor === COLORS.tertiary || backgroundColor === COLORS.secondary && !isDisabled,
bgBlackOpaque_onHover: backgroundColor === COLORS.black && !isDisabled,
bgBrandDark_onHover: backgroundColor === COLORS.brand && !isDisabled,
bgDangerDark_onHover: backgroundColor === COLORS.danger && !isDisabled,
2020-02-20 19:57:29 -05:00
2020-04-29 18:32:49 -04:00
bgBrand_onHover: color === COLORS.brand && isOutline && !isDisabled,
2020-04-28 01:33:58 -04:00
colorWhite_onHover: !!children && color === COLORS.brand && isOutline && !isDisabled,
2020-03-25 23:11:32 -04:00
2020-04-29 18:32:49 -04:00
fillSecondary: !!icon && color === COLORS.secondary,
fillWhite: !!icon && color === COLORS.white,
fillBrand: !!icon && color === COLORS.brand,
fillWhite_onHover: !!icon && color === COLORS.brand && isOutline,
2020-02-19 18:57:07 -05:00
})
const tagName = !!href ? 'a' : !!to ? 'NavLink' : 'button'
2020-04-23 02:13:29 -04:00
const theIcon = !!icon ? (
<Icon
id={icon}
size={iconSize}
className={iconClassName}
/>
) : undefined
2020-02-20 19:57:29 -05:00
const theChildren = !!icon ? (
<Fragment>
{theIcon}
{children}
</Fragment>
) : children
2020-04-23 02:13:29 -04:00
const handlers = {
onClick: !!onClick ? this.handleClick : undefined,
onMouseEnter: !!onMouseEnter ? this.handleOnMouseEnter : undefined,
onMouseLeave: !!onMouseLeave ? this.handleOnMouseLeave : undefined,
2020-02-22 18:26:23 -05:00
}
if (tagName === 'NavLink' && !!to) {
return (
2020-04-23 02:13:29 -04:00
<NavLink
title={title}
className={classes}
disabled={isDisabled}
to={to}
{...handlers}
>
2020-02-22 18:26:23 -05:00
{theChildren}
</NavLink>
)
}
2020-05-03 01:22:49 -04:00
const isLogout = href === '/auth/sign_out'
const dataMethod = isLogout ? 'delete' : undefined
2020-04-23 02:13:29 -04:00
const options = {
2020-04-29 18:32:49 -04:00
rel,
target,
2020-04-23 02:13:29 -04:00
title,
type,
disabled: isDisabled,
className: classes,
href: href || undefined,
ref: this.setRef,
2020-05-03 01:22:49 -04:00
'data-method': dataMethod,
2020-04-23 02:13:29 -04:00
...handlers,
}
2020-02-22 18:26:23 -05:00
return React.createElement(tagName, options, theChildren)
2020-02-19 18:57:07 -05:00
}
}