import classNames from 'classnames'; import { supportsPassiveEvents } from 'detect-passive-events'; import PropTypes from 'prop-types'; import React from 'react'; import ImmutablePropTypes from 'react-immutable-proptypes'; import spring from 'react-motion/lib/spring'; import Overlay from 'react-overlays/lib/Overlay'; import { withRouter } from 'react-router-dom'; import Icon from 'soapbox/components/icon'; import Motion from '../features/ui/util/optional_motion'; import { IconButton } from './ui'; const listenerOptions = supportsPassiveEvents ? { passive: true } : false; let id = 0; @withRouter class DropdownMenu extends React.PureComponent { static propTypes = { items: PropTypes.array.isRequired, onClose: PropTypes.func.isRequired, style: PropTypes.object, placement: PropTypes.string, arrowOffsetLeft: PropTypes.string, arrowOffsetTop: PropTypes.string, openedViaKeyboard: PropTypes.bool, history: PropTypes.object, }; static defaultProps = { style: {}, placement: 'bottom', }; state = { mounted: false, }; handleDocumentClick = e => { if (this.node && !this.node.contains(e.target)) { this.props.onClose(); } } componentDidMount() { document.addEventListener('click', this.handleDocumentClick, false); document.addEventListener('keydown', this.handleKeyDown, false); document.addEventListener('touchend', this.handleDocumentClick, listenerOptions); if (this.focusedItem && this.props.openedViaKeyboard) { this.focusedItem.focus({ preventScroll: true }); } this.setState({ mounted: true }); } componentWillUnmount() { document.removeEventListener('click', this.handleDocumentClick, false); document.removeEventListener('keydown', this.handleKeyDown, false); document.removeEventListener('touchend', this.handleDocumentClick, listenerOptions); } setRef = c => { this.node = c; } setFocusRef = c => { this.focusedItem = c; } handleKeyDown = e => { const items = Array.from(this.node.getElementsByTagName('a')); const index = items.indexOf(document.activeElement); let element = null; switch(e.key) { case 'ArrowDown': element = items[index+1] || items[0]; break; case 'ArrowUp': element = items[index-1] || items[items.length-1]; break; case 'Tab': if (e.shiftKey) { element = items[index-1] || items[items.length-1]; } else { element = items[index+1] || items[0]; } break; case 'Home': element = items[0]; break; case 'End': element = items[items.length-1]; break; case 'Escape': this.props.onClose(); break; } if (element) { element.focus(); e.preventDefault(); e.stopPropagation(); } } handleItemKeyPress = e => { if (e.key === 'Enter' || e.key === ' ') { this.handleClick(e); } } handleClick = e => { const i = Number(e.currentTarget.getAttribute('data-index')); const { action, to } = this.props.items[i]; this.props.onClose(); if (typeof action === 'function') { e.preventDefault(); action(e); } else if (to) { e.preventDefault(); this.props.history.push(to); } } handleMiddleClick = e => { const i = Number(e.currentTarget.getAttribute('data-index')); const { middleClick } = this.props.items[i]; this.props.onClose(); if (e.button === 1 && typeof middleClick === 'function') { e.preventDefault(); middleClick(e); } } handleAuxClick = e => { if (e.button === 1) { this.handleMiddleClick(e); } } renderItem(option, i) { if (option === null) { return
; } const { text, href, to, newTab, isLogout, icon, destructive } = option; return (