DocSpace-client/web/ASC.Web.Components/src/components/button/index.js

285 lines
7.3 KiB
JavaScript
Raw Normal View History

import React from 'react';
import styled, { css } from 'styled-components';
import PropTypes from 'prop-types';
import Loader from '../loader';
import isEqual from "lodash/isEqual";
2019-07-01 07:01:57 +00:00
const activeCss = css`
background-color: ${props => (props.primary ? '#1F97CA' : '#ECEEF1')};
color: ${props => (props.primary ? '#ffffff' : '#333333')};
${props =>
!props.primary &&
css`
2019-07-01 07:01:57 +00:00
border: 1px solid #2DA7DB;
box-sizing: border-box;
`}
`;
2019-07-01 07:01:57 +00:00
const hoverCss = css`
background-color: ${props => (props.primary ? '#3DB8EC' : '#FFFFFF')};
color: ${props => (props.primary ? '#ffffff' : '#333333')};
${props =>
!props.primary &&
css`
border: 1px solid #2DA7DB;
box-sizing: border-box;
`}
`;
// eslint-disable-next-line no-unused-vars
const ButtonWrapper = ({primary, scale, size, isHovered, isClicked, isDisabled, isLoading, label, innerRef, ...props}) => <button ref={innerRef} type="button" {...props}></button>;
2019-07-01 07:01:57 +00:00
ButtonWrapper.propTypes = {
label: PropTypes.string,
primary: PropTypes.bool,
2020-07-16 09:10:50 +00:00
size: PropTypes.oneOf(['base', 'medium', 'big', 'large']),
scale: PropTypes.bool,
icon: PropTypes.node,
tabIndex: PropTypes.number,
isHovered: PropTypes.bool,
isClicked: PropTypes.bool,
isDisabled: PropTypes.bool,
isLoading: PropTypes.bool,
onClick: PropTypes.func,
};
2019-07-01 07:01:57 +00:00
const StyledButton = styled(ButtonWrapper).attrs((props) => ({
disabled: props.isDisabled || props.isLoading ? 'disabled' : '',
tabIndex: props.tabIndex
}))`
2020-07-15 14:34:01 +00:00
min-width: ${props =>
props.size === 'base' ? '60px' : '100px'
};
height: ${props =>
2020-07-15 14:17:37 +00:00
(props.size === 'large' && '44px') ||
2019-07-01 07:01:57 +00:00
(props.size === 'big' && '36px') ||
(props.size === 'medium' && '32px') ||
2020-07-15 14:17:37 +00:00
(props.size === 'base' && '24px')
};
2019-07-01 07:01:57 +00:00
line-height: ${props =>
2020-07-15 14:58:33 +00:00
(props.size === 'large' && '20px') ||
(props.size === 'big' && '18px') ||
(props.size === 'medium' && '15px') ||
2019-07-01 07:01:57 +00:00
(props.size === 'base' && '16px')
};
font-size: ${props =>
2020-07-15 14:17:37 +00:00
(props.size === 'large' && '16px') ||
2019-07-01 07:01:57 +00:00
(props.size === 'big' && '14px') ||
(props.size === 'medium' && '13px') ||
2019-07-01 07:01:57 +00:00
(props.size === 'base' && '12px')
};
2019-07-01 07:01:57 +00:00
color: ${props => (props.primary && '#FFFFFF') || (!props.isDisabled ? '#333333' : '#ECEEF1')};
2019-07-01 07:01:57 +00:00
background-color: ${props =>
(!props.isDisabled || props.isLoading
? (props.primary ? '#2DA7DB' : '#FFFFFF')
: (props.primary ? '#A6DCF2' : '#FFFFFF'))
};
${props => props.scale && `width: 100%;`}
padding: ${props =>
2020-07-15 14:17:37 +00:00
(props.size === 'large' && (props.primary
? (props.icon
2020-07-15 14:58:33 +00:00
? (props.label ? '11px 20px 13px 20px' : '11px 20px 13px 20px')
: (props.label ? '12px 20px 12px 20px' : '12px 20px 12px 20px')
2020-07-15 14:17:37 +00:00
)
: (props.icon
2020-07-15 14:58:33 +00:00
? (props.label ? '11px 20px 13px 20px' : '11px 20px 13px 20px')
: (props.label ? '11px 20px 13px 20px' : '11px 20px 13px 20px')
2020-07-15 14:17:37 +00:00
))
) ||
2019-07-02 08:04:54 +00:00
(props.size === 'big' && (props.primary
? (props.icon
2020-07-15 14:58:33 +00:00
? (props.label ? '8px 16px 9px 16px' : '8px 16px 9px 16px')
2020-07-15 15:49:12 +00:00
: (props.label ? '8px 16px 8px 16px' : '8px 10px 9px 16px')
2019-07-02 08:04:54 +00:00
)
: (props.icon
2020-07-15 14:58:33 +00:00
? (props.label ? '8px 16px 9px 16px' : '8px 16px 9px 16px')
: (props.label ? '8px 16px 9px 16px' : '8px 16px 9px 16px')
2019-07-02 08:04:54 +00:00
))
) ||
(props.size === 'medium' && (props.primary
? (props.icon
2020-07-15 15:49:12 +00:00
? (props.label ? '6px 16px 9px 16px' : '6px 16px 8px 16px')
: (props.label ? '7px 16px 8px 16px' : '6px 16px 8px 16px')
)
: (props.icon
2020-07-15 15:49:12 +00:00
? (props.label ? '6px 16px 9px 16px' : '6px 16px 8px 16px')
: (props.label ? '7px 16px 8px 16px' : '6px 16px 8px 16px')
))
) ||
2019-07-02 08:04:54 +00:00
(props.size === 'base' && (props.primary
? (props.icon
2020-07-15 14:58:33 +00:00
? (props.label ? '3px 12px 5px 12px' : '3px 12px 5px 12px')
: (props.label ? '4px 12px 5px 12px' : '4px 12px 5px 12px')
2019-07-02 08:04:54 +00:00
)
: (props.icon
2020-07-15 14:58:33 +00:00
? (props.label ? '3px 12px 5px 12px' : '3px 12px 5px 12px')
: (props.label ? '3px 12px 5px 12px' : '3px 12px 5px 12px')
2019-07-02 08:04:54 +00:00
))
)
};
cursor: ${props => props.isDisabled || props.isLoading ? 'default !important' : 'pointer'};
2020-07-15 15:49:12 +00:00
font-family: 'Open Sans', normal;
border: none;
margin: 0;
display: inline-block;
2019-07-01 07:01:57 +00:00
font-weight: 600;
text-align: center;
text-decoration: none;
vertical-align: middle;
border-radius: 3px;
-moz-border-radius: 3px;
-webkit-border-radius: 3px;
touch-callout: none;
-o-touch-callout: none;
-moz-touch-callout: none;
-webkit-touch-callout: none;
user-select: none;
-o-user-select: none;
-moz-user-select: none;
-webkit-user-select: none;
stroke: none;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
-webkit-tap-highlight-color: rgba(0, 0, 0, 0);
${props =>
!props.primary &&
css`
2019-07-01 07:01:57 +00:00
border: 1px solid;
box-sizing: border-box;
border-color: ${props => (!props.isDisabled && !props.isLoading) ? '#D0D5DA' : '#ECEEF1'};
`}
2019-07-01 07:01:57 +00:00
${props => (!props.isDisabled && !props.isLoading) && (props.isHovered ? hoverCss : css`
&:hover {
${hoverCss}
}`)
}
2019-07-01 07:01:57 +00:00
${props => (!props.isDisabled && !props.isLoading) && (props.isClicked ? activeCss : css`
&:active {
${activeCss}
}`)
}
&:focus {
outline: none
}
2019-07-02 08:04:54 +00:00
.btnIcon,
.loader {
display: inline-block;
2020-07-16 09:31:18 +00:00
vertical-align: ${props => ( props.size === 'large' || props.size === 'base' )
? 'middle'
: props.size === 'medium'
? 'text-bottom'
: 'text-top'};
2019-07-02 08:04:54 +00:00
}
${props => props.label && css`
.btnIcon,
.loader {
padding-right: 4px;
}
`}
`;
2019-07-02 08:04:54 +00:00
const Icon = ({size, primary, icon}) => (
<div className="btnIcon">
{ icon && React.cloneElement(icon,
{
isfill: true,
2020-07-16 09:10:50 +00:00
size: size === "large"
? "large"
: size === "big"
? "medium"
: "small",
2019-07-02 08:04:54 +00:00
color: primary ? "#FFFFFF" : '#333333'
})}
</div>
);
Icon.propTypes = {
icon: PropTypes.node,
size: PropTypes.string,
primary: PropTypes.bool
2019-07-02 08:04:54 +00:00
};
2019-07-02 08:04:54 +00:00
Icon.defaultProps = {
icon: null
};
const Button = React.forwardRef((props, ref) => {
const { primary, size, isLoading, icon, label} = props;
const iconProps = { primary, size, icon };
return (
<StyledButton innerRef={ref} {...props} >
{(isLoading || icon)
? (isLoading
2020-07-16 09:10:50 +00:00
? <Loader type="oval" size={
size === "large" ? '18px' : size === 'big' ? '16px' : '14px'
} color={primary ? "#FFFFFF" : '#333333'} className="loader" />
: <Icon {...iconProps} />)
: ""
}
{label}
</StyledButton>
);
});
Button.propTypes = {
label: PropTypes.string,
2019-07-01 07:01:57 +00:00
primary: PropTypes.bool,
2020-07-15 14:17:37 +00:00
size: PropTypes.oneOf(['base', 'medium', 'big', 'large']),
scale: PropTypes.bool,
2019-07-02 08:04:54 +00:00
icon: PropTypes.node,
tabIndex: PropTypes.number,
className: PropTypes.string,
id: PropTypes.string,
style: PropTypes.oneOfType([PropTypes.object, PropTypes.array]),
2019-07-01 07:01:57 +00:00
isHovered: PropTypes.bool,
2019-07-01 07:01:57 +00:00
isClicked: PropTypes.bool,
isDisabled: PropTypes.bool,
isLoading: PropTypes.bool,
2019-07-01 07:01:57 +00:00
onClick: PropTypes.func
};
Button.defaultProps = {
2019-07-01 07:01:57 +00:00
label: '',
primary: false,
size: 'base',
scale: false,
2019-07-02 08:04:54 +00:00
icon: null,
2019-07-01 07:01:57 +00:00
tabIndex: -1,
2019-07-01 07:01:57 +00:00
isHovered: false,
isClicked: false,
isDisabled: false,
isLoading: false
};
Button.displayName = "Button";
export default Button;