- // Package imports.
- import classNames from 'classnames';
- import PropTypes from 'prop-types';
- import React from 'react';
- import Overlay from 'react-overlays/lib/Overlay';
-
- // Components.
- import IconButton from 'flavours/glitch/components/icon_button';
- import ComposerOptionsDropdownContent from './content';
-
- // Utils.
- import { isUserTouching } from 'flavours/glitch/util/is_mobile';
- import { assignHandlers } from 'flavours/glitch/util/react_helpers';
-
- // Handlers.
- const handlers = {
-
- // Closes the dropdown.
- handleClose () {
- this.setState({ open: false });
- },
-
- // The enter key toggles the dropdown's open state, and the escape
- // key closes it.
- handleKeyDown ({ key }) {
- const {
- handleClose,
- handleToggle,
- } = this.handlers;
- switch (key) {
- case 'Enter':
- handleToggle(key);
- break;
- case 'Escape':
- handleClose();
- break;
- }
- },
-
- // Creates an action modal object.
- handleMakeModal () {
- const component = this;
- const {
- items,
- onChange,
- onModalOpen,
- onModalClose,
- value,
- } = this.props;
-
- // Required props.
- if (!(onChange && onModalOpen && onModalClose && items)) {
- return null;
- }
-
- // The object.
- return {
- actions: items.map(
- ({
- name,
- ...rest
- }) => ({
- ...rest,
- active: value && name === value,
- name,
- onClick (e) {
- e.preventDefault(); // Prevents focus from changing
- onModalClose();
- onChange(name);
- },
- onPassiveClick (e) {
- e.preventDefault(); // Prevents focus from changing
- onChange(name);
- component.setState({ needsModalUpdate: true });
- },
- })
- ),
- };
- },
-
- // 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;
- const { onModalOpen } = this.props;
- const { needsModalUpdate } = this.state;
-
- // Gets our modal object.
- const modal = 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.setState({ needsModalUpdate: false });
- }
- }
-
- // Rendering.
- render () {
- const {
- handleClose,
- handleKeyDown,
- handleToggle,
- } = this.handlers;
- const {
- active,
- disabled,
- title,
- icon,
- items,
- onChange,
- value,
- } = this.props;
- const { open, placement } = this.state;
- const computedClass = classNames('composer--options--dropdown', {
- active,
- open,
- top: placement === 'top',
- });
-
- // The result.
- return (
- <div
- className={computedClass}
- onKeyDown={handleKeyDown}
- >
- <IconButton
- active={open || active}
- className='value'
- disabled={disabled}
- icon={icon}
- onClick={handleToggle}
- size={18}
- style={{
- height: null,
- lineHeight: '27px',
- }}
- title={title}
- />
- <Overlay
- containerPadding={20}
- placement={placement}
- show={open}
- target={this}
- >
- <ComposerOptionsDropdownContent
- items={items}
- onChange={onChange}
- onClose={handleClose}
- value={value}
- />
- </Overlay>
- </div>
- );
- }
-
- }
-
- // 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,
- };
|