import React, { Component } from 'react'; import PropTypes from 'prop-types'; import { findDOMNode } from 'react-dom'; import cx from 'classnames'; import { defineMessages, injectIntl } from 'react-intl'; import Button from '/imports/ui/components/button/component'; import screenreaderTrap from 'makeup-screenreader-trap'; import { styles } from './styles'; import DropdownTrigger from './trigger/component'; import DropdownContent from './content/component'; const intlMessages = defineMessages({ close: { id: 'app.dropdown.close', description: 'Close button label', }, }); const noop = () => {}; const propTypes = { /** * The dropdown needs a trigger and a content component as children */ children: (props, propName, componentName) => { const children = props[propName]; if (!children || children.length < 2) { return new Error(`Invalid prop \`${propName}\` supplied to` + ` \`${componentName}\`. Validation failed.`); } const trigger = children.find(x => x.type === DropdownTrigger); const content = children.find(x => x.type === DropdownContent); if (!trigger) { return new Error(`Invalid prop \`${propName}\` supplied to` + ` \`${componentName}\`. Missing \`DropdownTrigger\`. Validation failed.`); } if (!content) { return new Error(`Invalid prop \`${propName}\` supplied to` + ` \`${componentName}\`. Missing \`DropdownContent\`. Validation failed.`); } return null; }, isOpen: PropTypes.bool, onHide: PropTypes.func, onShow: PropTypes.func, autoFocus: PropTypes.bool, }; const defaultProps = { children: null, isOpen: false, onShow: noop, onHide: noop, autoFocus: false, }; class Dropdown extends Component { constructor(props) { super(props); this.state = { isOpen: false }; this.handleShow = this.handleShow.bind(this); this.handleHide = this.handleHide.bind(this); this.handleToggle = this.handleToggle.bind(this); this.handleWindowClick = this.handleWindowClick.bind(this); } componentWillUpdate(nextProps, nextState) { return nextState.isOpen ? screenreaderTrap.trap(this.dropdown) : screenreaderTrap.untrap(); } componentDidUpdate(prevProps, prevState) { const { onShow, onHide, } = this.props; if (this.state.isOpen && !prevState.isOpen) { onShow(); } if (!this.state.isOpen && prevState.isOpen) { onHide(); } } handleShow() { this.setState({ isOpen: true }, () => { const { addEventListener } = window; addEventListener('click', this.handleWindowClick, true); }); } handleHide() { this.setState({ isOpen: false }, () => { const { removeEventListener } = window; removeEventListener('click', this.handleWindowClick, true); }); } handleWindowClick() { const triggerElement = findDOMNode(this.trigger); const contentElement = findDOMNode(this.content); const closeDropdown = this.props.isOpen && this.state.isOpen && triggerElement.contains(event.target); const preventHide = this.props.isOpen && contentElement.contains(event.target) || !triggerElement; if (closeDropdown) { return this.props.onHide(); } if (contentElement && preventHide) { return; } this.handleHide(); } handleToggle() { return this.state.isOpen ? this.handleHide() : this.handleShow(); } render() { const { children, className, style, intl, ...otherProps } = this.props; let trigger = children.find(x => x.type === DropdownTrigger); let content = children.find(x => x.type === DropdownContent); trigger = React.cloneElement(trigger, { ref: (ref) => { this.trigger = ref; }, dropdownIsOpen: this.state.isOpen, dropdownToggle: this.handleToggle, dropdownShow: this.handleShow, dropdownHide: this.handleHide, }); content = React.cloneElement(content, { ref: (ref) => { this.content = ref; }, 'aria-expanded': this.state.isOpen, dropdownIsOpen: this.state.isOpen, dropdownToggle: this.handleToggle, dropdownShow: this.handleShow, dropdownHide: this.handleHide, }); return (