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

198 lines
5.3 KiB
JavaScript
Raw Normal View History

import ComboButton from './sub-components/combo-button'
2019-07-18 08:00:54 +00:00
import DropDown from '../drop-down'
import DropDownItem from '../drop-down-item'
import PropTypes from 'prop-types'
import React from 'react'
import isEqual from 'lodash/isEqual';
import styled from 'styled-components';
2019-07-18 08:00:54 +00:00
const StyledComboBox = styled.div`
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')
};
position: relative;
outline: 0;
2019-07-18 08:00:54 +00:00
`;
class ComboBox extends React.Component {
constructor(props) {
super(props);
2019-07-18 08:00:54 +00:00
this.ref = React.createRef();
this.state = {
isOpen: props.opened,
selectedOption: props.selectedOption
};
}
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);
//console.log("shouldComponentUpdate", needUpdate);
2020-01-09 13:27:49 +00:00
return needUpdate;
}
stopAction = (e) => e.preventDefault();
2020-01-09 13:27:49 +00:00
setIsOpen = (isOpen) => this.setState({ isOpen: isOpen });
handleClickOutside = e => {
//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
};
comboBoxClick = (e) => {
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);
})
};
optionClick = (option) => {
this.setState({
isOpen: !this.state.isOpen,
selectedOption: option
2019-07-31 18:42:23 +00:00
});
this.props.onSelect && this.props.onSelect(option);
};
2019-07-18 08:00:54 +00:00
componentDidUpdate(prevProps, prevState) {
if (this.props.opened !== prevProps.opened && this.state.isOpen !== prevState.isOpen) {
this.setIsOpen(this.state.isOpen);
}
if (this.props.selectedOption !== prevProps.selectedOption) {
this.setState({ selectedOption: this.props.selectedOption });
}
}
2019-07-18 08:00:54 +00:00
render() {
//console.log("ComboBox render");
const {
dropDownMaxHeight,
directionX,
directionY,
scaled,
size,
options,
advancedOptions,
isDisabled,
children,
noBorder,
scaledOptions,
displayType,
toggleAction } = this.props;
const { isOpen, selectedOption } = this.state;
const dropDownMaxHeightProp = dropDownMaxHeight
? { maxHeight: dropDownMaxHeight }
: {};
const dropDownManualWidthProp = scaledOptions
? { manualWidth: '100%' }
: {};
const optionsLength = options.length
? options.length
: (displayType !== 'toggle')
? 0
: 1;
const advancedOptionsLength = advancedOptions
? advancedOptions.props.children.length
: 0;
return (
<StyledComboBox
ref={this.ref}
isDisabled={isDisabled}
scaled={scaled}
size={size}
data={selectedOption}
onClick={this.comboBoxClick}
toggleAction={toggleAction}
{...this.props}
>
<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}
/>
{displayType !== 'toggle' &&
<DropDown
directionX={directionX}
directionY={directionY}
manualY='102%'
open={isOpen}
2020-01-09 13:27:49 +00:00
clickOutsideAction={this.handleClickOutside}
{...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>
}
</StyledComboBox>
);
}
}
2019-07-18 08:00:54 +00:00
ComboBox.propTypes = {
advancedOptions: PropTypes.element,
children: PropTypes.any,
className: PropTypes.string,
directionX: PropTypes.oneOf(['left', 'right']),
directionY: PropTypes.oneOf(['bottom', 'top']),
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,
scaled: PropTypes.bool,
scaledOptions: PropTypes.bool,
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 = {
displayType: 'default',
isDisabled: false,
noBorder: false,
scaled: true,
scaledOptions: false,
size: 'base'
2019-07-18 08:00:54 +00:00
}
2019-07-18 08:00:54 +00:00
export default ComboBox;