import { isUserTouching } from 'flavours/glitch/util/is_mobile';
import { assignHandlers } from 'flavours/glitch/util/react_helpers';
-// Handlers.
-const handlers = {
+// The component.
+export default class ComposerOptionsDropdown extends React.PureComponent {
- // Closes the dropdown.
- handleClose () {
- this.setState({ open: false });
- },
+ static propTypes = {
+ active: PropTypes.bool,
+ disabled: PropTypes.bool,
+ icon: PropTypes.string,
+ items: PropTypes.arrayOf(PropTypes.shape({
+ icon: PropTypes.string,
+ meta: PropTypes.node,
+ name: PropTypes.string.isRequired,
+ on: PropTypes.bool,
+ text: PropTypes.node,
+ })).isRequired,
+ onModalOpen: PropTypes.func,
+ onModalClose: PropTypes.func,
+ title: PropTypes.string,
+ value: PropTypes.string,
+ onChange: PropTypes.func,
+ };
+
+ state = {
+ needsModalUpdate: false,
+ open: false,
+ placement: 'bottom',
+ };
- // The enter key toggles the dropdown's open state, and the escape
- // key closes it.
- handleKeyDown ({ key }) {
- const {
- handleClose,
- handleToggle,
- } = this.handlers;
- switch (key) {
+ // Toggles opening and closing the dropdown.
+ handleToggle = ({ target }) => {
+ const { onModalOpen } = this.props;
+ const { open } = this.state;
+
+ if (isUserTouching()) {
+ if (this.state.open) {
+ this.props.onModalClose();
+ } else {
+ const modal = this.handleMakeModal();
+ if (modal && onModalOpen) {
+ onModalOpen(modal);
+ }
+ }
+ } else {
+ const { top } = target.getBoundingClientRect();
+ this.setState({ placement: top * 2 < innerHeight ? 'bottom' : 'top' });
+ this.setState({ open: !this.state.open });
+ }
+ }
+
+ handleKeyDown = (e) => {
+ switch (e.key) {
case 'Enter':
- handleToggle(key);
+ this.handleToggle(key);
break;
case 'Escape':
- handleClose();
+ this.handleClose();
break;
}
- },
+ }
+
+ handleClose = () => {
+ this.setState({ open: false });
+ }
// Creates an action modal object.
- handleMakeModal () {
+ handleMakeModal = () => {
const component = this;
const {
items,
})
),
};
- },
-
- // Toggles opening and closing the dropdown.
- handleToggle ({ target }) {
- const { handleMakeModal } = this.handlers;
- const { onModalOpen } = this.props;
- const { open } = this.state;
-
- // If this is a touch device, we open a modal instead of the
- // dropdown.
- if (isUserTouching()) {
-
- // This gets the modal to open.
- const modal = handleMakeModal();
-
- // If we can, we then open the modal.
- if (modal && onModalOpen) {
- onModalOpen(modal);
- return;
- }
- }
-
- const { top } = target.getBoundingClientRect();
- this.setState({ placement: top * 2 < innerHeight ? 'bottom' : 'top' });
- // Otherwise, we just set our state to open.
- this.setState({ open: !open });
- },
+ }
// If our modal is open and our props update, we need to also update
// the modal.
- handleUpdate () {
- const { handleMakeModal } = this.handlers;
+ handleUpdate = () => {
const { onModalOpen } = this.props;
const { needsModalUpdate } = this.state;
// Gets our modal object.
- const modal = handleMakeModal();
+ const modal = this.handleMakeModal();
// Reopens the modal with the new object.
if (needsModalUpdate && modal && onModalOpen) {
onModalOpen(modal);
}
- },
-};
-
-// The component.
-export default class ComposerOptionsDropdown extends React.PureComponent {
-
- // Constructor.
- constructor (props) {
- super(props);
- assignHandlers(this, handlers);
- this.state = {
- needsModalUpdate: false,
- open: false,
- placement: 'bottom',
- };
}
// Updates our modal as necessary.
componentDidUpdate (prevProps) {
- const { handleUpdate } = this.handlers;
const { items } = this.props;
const { needsModalUpdate } = this.state;
if (needsModalUpdate && items.find(
(item, i) => item.on !== prevProps.items[i].on
)) {
- handleUpdate();
+ this.handleUpdate();
this.setState({ needsModalUpdate: false });
}
}
// Rendering.
render () {
- const {
- handleClose,
- handleKeyDown,
- handleToggle,
- } = this.handlers;
const {
active,
disabled,
return (
<div
className={computedClass}
- onKeyDown={handleKeyDown}
+ onKeyDown={this.handleKeyDown}
>
<IconButton
active={open || active}
disabled={disabled}
icon={icon}
inverted
- onClick={handleToggle}
+ onClick={this.handleToggle}
size={18}
style={{
height: null,
<DropdownMenu
items={items}
onChange={onChange}
- onClose={handleClose}
+ onClose={this.handleClose}
value={value}
/>
</Overlay>
}
}
-
-// Props.
-ComposerOptionsDropdown.propTypes = {
- active: PropTypes.bool,
- disabled: PropTypes.bool,
- icon: PropTypes.string,
- items: PropTypes.arrayOf(PropTypes.shape({
- icon: PropTypes.string,
- meta: PropTypes.node,
- name: PropTypes.string.isRequired,
- on: PropTypes.bool,
- text: PropTypes.node,
- })).isRequired,
- onChange: PropTypes.func,
- onModalClose: PropTypes.func,
- onModalOpen: PropTypes.func,
- title: PropTypes.string,
- value: PropTypes.string,
-};