2019-12-11 12:22:07 +00:00
|
|
|
import ComboButton from './sub-components/combo-button'
|
2019-07-18 08:00:54 +00:00
|
|
|
import DropDown from '../drop-down'
|
2019-12-11 12:22:07 +00:00
|
|
|
import DropDownItem from '../drop-down-item'
|
|
|
|
import PropTypes from 'prop-types'
|
|
|
|
import React from 'react'
|
2019-09-12 12:21:58 +00:00
|
|
|
import isEqual from 'lodash/isEqual';
|
2019-12-11 12:22:07 +00:00
|
|
|
import styled from 'styled-components';
|
2019-07-18 08:00:54 +00:00
|
|
|
|
|
|
|
const StyledComboBox = styled.div`
|
2019-08-12 08:19:52 +00:00
|
|
|
width: ${props =>
|
|
|
|
(props.scaled && '100%') ||
|
|
|
|
(props.size === 'base' && '173px') ||
|
|
|
|
(props.size === 'middle' && '300px') ||
|
|
|
|
(props.size === 'big' && '350px') ||
|
|
|
|
(props.size === 'huge' && '500px') ||
|
|
|
|
(props.size === 'content' && 'fit-content')
|
|
|
|
};
|
2019-08-08 11:14:25 +00:00
|
|
|
|
2019-08-12 08:19:52 +00:00
|
|
|
position: relative;
|
2019-12-16 12:46:42 +00:00
|
|
|
outline: 0;
|
2019-07-18 08:00:54 +00:00
|
|
|
`;
|
|
|
|
|
2019-09-12 12:21:58 +00:00
|
|
|
class ComboBox extends React.Component {
|
2019-07-30 13:49:33 +00:00
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
2019-07-18 08:00:54 +00:00
|
|
|
|
2019-07-30 13:49:33 +00:00
|
|
|
this.ref = React.createRef();
|
2019-07-26 09:02:08 +00:00
|
|
|
|
2019-07-30 13:49:33 +00:00
|
|
|
this.state = {
|
|
|
|
isOpen: props.opened,
|
2019-08-14 12:53:05 +00:00
|
|
|
selectedOption: props.selectedOption
|
2019-07-30 13:49:33 +00:00
|
|
|
};
|
|
|
|
}
|
2019-07-18 08:00:54 +00:00
|
|
|
|
2020-01-09 13:27:49 +00:00
|
|
|
shouldComponentUpdate(nextProps, nextState) {
|
|
|
|
const needUpdate = !isEqual(this.props, nextProps) || !isEqual(this.state, nextState);
|
|
|
|
|
2020-01-10 09:29:44 +00:00
|
|
|
//console.log("shouldComponentUpdate", needUpdate);
|
2020-01-09 13:27:49 +00:00
|
|
|
|
|
|
|
return needUpdate;
|
2019-09-19 11:59:31 +00:00
|
|
|
}
|
2019-08-05 12:45:12 +00:00
|
|
|
|
2019-07-30 13:49:33 +00:00
|
|
|
stopAction = (e) => e.preventDefault();
|
2019-08-05 12:45:12 +00:00
|
|
|
|
2020-01-09 13:27:49 +00:00
|
|
|
setIsOpen = (isOpen) => this.setState({ isOpen: isOpen });
|
|
|
|
|
|
|
|
handleClickOutside = e => {
|
2020-01-10 09:29:44 +00:00
|
|
|
//console.log(`ComboBox handleClickOutside`, e);
|
2020-01-10 10:47:06 +00:00
|
|
|
this.setState({ isOpen: !this.state.isOpen }, () => {
|
|
|
|
this.props.toggleAction && this.props.toggleAction(e, this.state.isOpen);
|
|
|
|
})
|
2020-01-09 13:27:49 +00:00
|
|
|
};
|
2019-08-05 12:45:12 +00:00
|
|
|
|
|
|
|
comboBoxClick = (e) => {
|
2019-09-19 10:44:53 +00:00
|
|
|
if (this.props.isDisabled || e && e.target.closest('.optionalBlock')) return;
|
2020-01-10 10:47:06 +00:00
|
|
|
this.setState({ isOpen: !this.state.isOpen }, () => {
|
|
|
|
this.props.toggleAction && this.props.toggleAction(e, this.state.isOpen);
|
|
|
|
})
|
2019-07-30 13:49:33 +00:00
|
|
|
};
|
2019-08-05 12:45:12 +00:00
|
|
|
|
2019-07-30 13:49:33 +00:00
|
|
|
optionClick = (option) => {
|
2019-08-16 06:54:23 +00:00
|
|
|
this.setState({
|
2019-08-14 12:53:05 +00:00
|
|
|
isOpen: !this.state.isOpen,
|
|
|
|
selectedOption: option
|
2019-07-31 18:42:23 +00:00
|
|
|
});
|
2019-09-18 11:45:29 +00:00
|
|
|
|
2019-07-30 13:49:33 +00:00
|
|
|
this.props.onSelect && this.props.onSelect(option);
|
|
|
|
};
|
2019-07-18 08:00:54 +00:00
|
|
|
|
2020-01-10 09:29:44 +00:00
|
|
|
componentDidUpdate(prevProps, prevState) {
|
|
|
|
if (this.props.opened !== prevProps.opened && this.state.isOpen !== prevState.isOpen) {
|
|
|
|
this.setIsOpen(this.state.isOpen);
|
2019-08-02 07:35:54 +00:00
|
|
|
}
|
2019-08-05 15:12:25 +00:00
|
|
|
|
2019-08-06 07:39:33 +00:00
|
|
|
if (this.props.selectedOption !== prevProps.selectedOption) {
|
2019-08-16 06:54:23 +00:00
|
|
|
this.setState({ selectedOption: this.props.selectedOption });
|
2019-08-05 15:12:25 +00:00
|
|
|
}
|
2019-09-09 13:32:04 +00:00
|
|
|
}
|
2019-07-18 08:00:54 +00:00
|
|
|
|
2019-07-30 13:49:33 +00:00
|
|
|
render() {
|
2019-08-28 12:53:38 +00:00
|
|
|
//console.log("ComboBox render");
|
2019-09-02 11:15:24 +00:00
|
|
|
const {
|
2019-09-18 11:45:29 +00:00
|
|
|
dropDownMaxHeight,
|
|
|
|
directionX,
|
|
|
|
directionY,
|
|
|
|
scaled,
|
|
|
|
size,
|
|
|
|
options,
|
|
|
|
advancedOptions,
|
|
|
|
isDisabled,
|
|
|
|
children,
|
2019-09-18 13:24:14 +00:00
|
|
|
noBorder,
|
2019-12-11 12:22:07 +00:00
|
|
|
scaledOptions,
|
|
|
|
displayType,
|
|
|
|
toggleAction } = this.props;
|
2019-08-14 12:53:05 +00:00
|
|
|
const { isOpen, selectedOption } = this.state;
|
2019-08-12 08:19:52 +00:00
|
|
|
|
2019-09-18 11:45:29 +00:00
|
|
|
const dropDownMaxHeightProp = dropDownMaxHeight
|
|
|
|
? { maxHeight: dropDownMaxHeight }
|
|
|
|
: {};
|
2019-09-18 13:24:14 +00:00
|
|
|
const dropDownManualWidthProp = scaledOptions
|
2019-09-18 11:45:29 +00:00
|
|
|
? { manualWidth: '100%' }
|
|
|
|
: {};
|
|
|
|
|
|
|
|
const optionsLength = options.length
|
|
|
|
? options.length
|
2019-12-11 12:22:07 +00:00
|
|
|
: (displayType !== 'toggle')
|
|
|
|
? 0
|
|
|
|
: 1;
|
2019-09-18 11:45:29 +00:00
|
|
|
|
|
|
|
const advancedOptionsLength = advancedOptions
|
|
|
|
? advancedOptions.props.children.length
|
|
|
|
: 0;
|
|
|
|
|
2019-07-30 13:49:33 +00:00
|
|
|
return (
|
2019-09-18 11:45:29 +00:00
|
|
|
<StyledComboBox
|
|
|
|
ref={this.ref}
|
|
|
|
isDisabled={isDisabled}
|
|
|
|
scaled={scaled}
|
|
|
|
size={size}
|
2019-08-14 12:53:05 +00:00
|
|
|
data={selectedOption}
|
2019-08-08 11:14:25 +00:00
|
|
|
onClick={this.comboBoxClick}
|
2019-12-11 12:22:07 +00:00
|
|
|
toggleAction={toggleAction}
|
2019-09-18 11:45:29 +00:00
|
|
|
{...this.props}
|
2019-08-08 11:14:25 +00:00
|
|
|
>
|
2019-09-18 11:45:29 +00:00
|
|
|
<ComboButton
|
|
|
|
noBorder={noBorder}
|
|
|
|
isDisabled={isDisabled}
|
|
|
|
selectedOption={selectedOption}
|
|
|
|
withOptions={optionsLength > 0}
|
|
|
|
optionsLength={optionsLength}
|
|
|
|
withAdvancedOptions={advancedOptionsLength > 0}
|
|
|
|
innerContainer={children}
|
|
|
|
innerContainerClassName='optionalBlock'
|
|
|
|
isOpen={isOpen}
|
|
|
|
size={size}
|
|
|
|
scaled={scaled}
|
|
|
|
/>
|
2019-12-11 12:22:07 +00:00
|
|
|
{displayType !== 'toggle' &&
|
|
|
|
<DropDown
|
|
|
|
directionX={directionX}
|
|
|
|
directionY={directionY}
|
|
|
|
manualY='102%'
|
2019-12-17 14:27:06 +00:00
|
|
|
open={isOpen}
|
2020-01-09 13:27:49 +00:00
|
|
|
clickOutsideAction={this.handleClickOutside}
|
2019-12-11 12:22:07 +00:00
|
|
|
{...dropDownMaxHeightProp}
|
|
|
|
{...dropDownManualWidthProp}
|
|
|
|
>
|
|
|
|
{advancedOptions
|
|
|
|
? advancedOptions
|
|
|
|
: options.map((option) =>
|
|
|
|
<DropDownItem {...option}
|
|
|
|
key={option.key}
|
|
|
|
disabled={
|
|
|
|
option.disabled
|
|
|
|
|| (option.label === selectedOption.label)}
|
|
|
|
onClick={this.optionClick.bind(this, option)}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</DropDown>
|
|
|
|
}
|
2019-07-30 13:49:33 +00:00
|
|
|
</StyledComboBox>
|
|
|
|
);
|
|
|
|
}
|
2019-09-09 13:32:04 +00:00
|
|
|
}
|
2019-07-18 08:00:54 +00:00
|
|
|
|
|
|
|
ComboBox.propTypes = {
|
2019-08-30 13:44:04 +00:00
|
|
|
advancedOptions: PropTypes.element,
|
2019-09-09 13:32:04 +00:00
|
|
|
children: PropTypes.any,
|
2019-12-11 12:22:07 +00:00
|
|
|
className: PropTypes.string,
|
2019-09-09 13:32:04 +00:00
|
|
|
directionX: PropTypes.oneOf(['left', 'right']),
|
|
|
|
directionY: PropTypes.oneOf(['bottom', 'top']),
|
2019-12-11 12:22:07 +00:00
|
|
|
displayType: PropTypes.oneOf(['default', 'toggle']),
|
|
|
|
dropDownMaxHeight: PropTypes.number,
|
|
|
|
id: PropTypes.string,
|
|
|
|
isDisabled: PropTypes.bool,
|
|
|
|
noBorder: PropTypes.bool,
|
|
|
|
onSelect: PropTypes.func,
|
|
|
|
opened: PropTypes.bool,
|
|
|
|
options: PropTypes.array.isRequired,
|
2019-09-18 13:24:14 +00:00
|
|
|
scaled: PropTypes.bool,
|
2019-11-27 11:47:21 +00:00
|
|
|
scaledOptions: PropTypes.bool,
|
2019-12-11 12:22:07 +00:00
|
|
|
selectedOption: PropTypes.object.isRequired,
|
|
|
|
size: PropTypes.oneOf(['base', 'middle', 'big', 'huge', 'content']),
|
|
|
|
style: PropTypes.oneOfType([PropTypes.object, PropTypes.array]),
|
|
|
|
toggleAction: PropTypes.func
|
2019-07-18 08:00:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ComboBox.defaultProps = {
|
2019-12-11 12:22:07 +00:00
|
|
|
displayType: 'default',
|
2019-07-30 13:49:33 +00:00
|
|
|
isDisabled: false,
|
2019-12-11 12:22:07 +00:00
|
|
|
noBorder: false,
|
2019-09-18 13:24:14 +00:00
|
|
|
scaled: true,
|
2019-12-11 12:22:07 +00:00
|
|
|
scaledOptions: false,
|
|
|
|
size: 'base'
|
2019-07-18 08:00:54 +00:00
|
|
|
}
|
2019-07-26 09:02:08 +00:00
|
|
|
|
2019-07-18 08:00:54 +00:00
|
|
|
export default ComboBox;
|