import React from 'react'; import PropTypes from 'prop-types'; import cx from 'classnames'; import TooltipContainer from '/imports/ui/components/common/tooltip/container'; import Styled from './styles'; import BaseButton from './base/component'; import ButtonEmoji from './button-emoji/ButtonEmoji'; const SIZES = [ 'jumbo', 'lg', 'md', 'sm', ]; const COLORS = [ 'default', 'primary', 'danger', 'warning', 'success', 'dark', 'light', 'offline', 'muted', 'secondary', ]; const propTypes = { ...BaseButton.propTypes, /** * Defines the button size style * @type {("lg"|"md"|"sm")} * @defaultValue 'md' */ size: PropTypes.oneOf(SIZES), /** * Defines the button color style * @type {("default"|"primary"|"danger"|"success")} * @defaultValue 'md' */ color: PropTypes.oneOf(COLORS), /** * Defines if the button should be styled as a ghost (outline) * @defaultValue false */ ghost: PropTypes.bool, /** * Defines if the button should be styled as circle * @defaultValue false */ circle: PropTypes.bool, /** * Defines if the button should have `display: block` * @defaultValue false */ block: PropTypes.bool, /** * Defines the button icon * @defaultValue undefined */ icon: PropTypes.string, /** * Defines the button icon is on the right side * @defaultValue false */ iconRight: PropTypes.bool, /** * Defines the button label should be visible * @defaultValue true */ hideLabel: PropTypes.bool, /** * Optional SVG / html object can be passed to the button as an icon * Has to be styled before being sent to the Button * (e.g width, height, position and percentage-based object's coordinates) * @defaultvalue undefined */ customIcon: PropTypes.node, /** * Defines the buttom loading state * @defaultValue false */ loading: PropTypes.bool, }; const defaultProps = { ...BaseButton.defaultProps, size: 'md', color: 'default', ghost: false, circle: false, block: false, iconRight: false, hideLabel: false, tooltipLabel: '', loading: false, }; export default class Button extends BaseButton { _cleanProps(otherProps) { const remainingProps = Object.assign({}, otherProps); delete remainingProps.icon; delete remainingProps.customIcon; delete remainingProps.size; delete remainingProps.color; delete remainingProps.ghost; delete remainingProps.circle; delete remainingProps.block; delete remainingProps.hideLabel; delete remainingProps.tooltipLabel; return remainingProps; } hasButtonEmojiComponent() { const { children } = this.props; if (!children) return false; const buttonEmoji = React.Children.only(children); return (buttonEmoji && buttonEmoji.type && buttonEmoji.type.name) ? (buttonEmoji.type.name === ButtonEmoji.name) : false; } render() { const { circle, hideLabel, label, 'aria-label': ariaLabel, 'aria-expanded': ariaExpanded, tooltipLabel, tooltipdelay, tooltipplacement, } = this.props; const renderFuncName = circle ? 'renderCircle' : 'renderDefault'; if ((hideLabel && !ariaExpanded) || tooltipLabel) { const buttonLabel = label || ariaLabel; return ( {this[renderFuncName]()} ); } return this[renderFuncName](); } renderDefault() { const { className, iconRight, size, color, ghost, circle, block, loading, ...otherProps } = this.props; const remainingProps = this._cleanProps(otherProps); return ( {this.renderIcon()} {this.renderLabel()} ); } renderCircle() { const { className, size, iconRight, children, color, ghost, circle, block, loading, ...otherProps } = this.props; const remainingProps = this._cleanProps(otherProps); return ( {this.renderButtonEmojiSibling()} {!iconRight ? null : this.renderLabel()} {this.renderIcon()} {iconRight ? null : this.renderLabel()} {this.hasButtonEmojiComponent() ? children : null} ); } renderButtonEmojiSibling() { if (!this.hasButtonEmojiComponent()) { return null; } return (); } renderIcon() { const { icon: iconName, customIcon, } = this.props; if (iconName) { return (); } if (customIcon) { return customIcon; } return null; } renderLabel() { const { label, hideLabel } = this.props; if (!label) return null; return ( {label} {!this.hasButtonEmojiComponent() ? this.props.children : null} ); } } Button.propTypes = propTypes; Button.defaultProps = defaultProps;