import React, {PureComponent} from 'react'; import PropTypes from 'prop-types'; import ReactDOM from 'react-dom'; import autobind from 'autobind-decorator'; import classnames from 'classnames'; import DropdownButton from './dropdown-button'; import DropdownItem from './dropdown-item'; import DropdownDivider from './dropdown-divider'; @autobind class Dropdown extends PureComponent { constructor (props) { super(props); this.state = { open: false, dropUp: false, // Filter Stuff filter: '', filterVisible: false, filterItems: null, filterActiveIndex: 0, // Use this to force new menu every time dropdown opens uniquenessKey: 0 }; } _setRef (n) { this._node = n; } _handleCheckFilterSubmit (e) { if (e.key === 'Enter') { // Listen for the Enter key and "click" on the active list item const selector = `li[data-filter-index="${this.state.filterActiveIndex}"] button`; const button = this._dropdownList.querySelector(selector); button && button.click(); } } _handleChangeFilter (e) { const newFilter = e.target.value; // Nothing to do if the filter didn't change if (newFilter === this.state.filter) { return; } // Filter the list items that are filterable (have data-filter-index property) const filterItems = []; for (const listItem of this._dropdownList.querySelectorAll('li')) { if (!listItem.hasAttribute('data-filter-index')) { continue; } // We're matching against lowercase things with no special characters const listItemTextWithoutSpaces = listItem.textContent.toLowerCase().replace(/[^\w_]*/g, ''); const filterWithoutSpaces = newFilter.toLowerCase().replace(/[^\w_]*/g, ''); if (!newFilter || listItemTextWithoutSpaces.includes(filterWithoutSpaces)) { const filterIndex = listItem.getAttribute('data-filter-index'); filterItems.push(parseInt(filterIndex, 10)); } } this.setState({ filter: newFilter, filterItems: newFilter ? filterItems : null, filterActiveIndex: filterItems[0] || -1, filterVisible: this.state.filterVisible ? true : newFilter.length > 0 }); } _handleDropdownNavigation (e) { const {key, shiftKey} = e; // Handle tab and arrows to move up and down dropdown entries const {filterItems, filterActiveIndex} = this.state; if (['Tab', 'ArrowDown', 'ArrowUp'].includes(key)) { e.preventDefault(); let items = filterItems || []; if (!filterItems) { for (const li of this._dropdownList.querySelectorAll('li')) { if (li.hasAttribute('data-filter-index')) { const filterIndex = li.getAttribute('data-filter-index'); items.push(parseInt(filterIndex, 10)); } } } const i = items.indexOf(filterActiveIndex); if (key === 'ArrowUp' || (key === 'Tab' && shiftKey)) { const nextI = i > 0 ? items[i - 1] : items[items.length - 1]; this.setState({filterActiveIndex: nextI}); } else { this.setState({filterActiveIndex: items[i + 1] || items[0]}); } } this._filter.focus(); } _handleBodyKeyDown (e) { if (!this.state.open) { return; } // Catch all key presses (like global app hotkeys) if we're open e.stopPropagation(); this._handleDropdownNavigation(e); if (e.key === 'Escape') { e.preventDefault(); this.hide(); } } _checkSizeAndPosition () { if (!this.state.open || !this._dropdownList) { return; } // Make the dropdown scroll if it drops off screen. const dropdownRect = this._node.getBoundingClientRect(); const bodyRect = document.body.getBoundingClientRect(); // Should it drop up? const bodyHeight = bodyRect.height; const dropdownTop = dropdownRect.top; const dropUp = dropdownTop > bodyHeight - 200; // Reset all the things so we can start fresh this._dropdownList.style.left = 'initial'; this._dropdownList.style.right = 'initial'; this._dropdownList.style.top = 'initial'; this._dropdownList.style.bottom = 'initial'; this._dropdownList.style.minWidth = 'initial'; this._dropdownList.style.maxWidth = 'initial'; // Make dropdown keep it's shape when filtering const ul = this._dropdownList.querySelector('ul'); const ulRect = ul.getBoundingClientRect(); if (!ul.hasAttribute('data-fixed-shape')) { ul.style.minHeight = `${ulRect.height}px`; ul.style.minWidth = `${ulRect.width}px`; ul.style.width = '100%'; ul.setAttribute('data-fixed-shape', 'on'); } const {right, wide} = this.props; if (right || wide) { const {right} = dropdownRect; const {beside} = this.props; const offset = beside ? dropdownRect.width - dropdownRect.height : 0; this._dropdownList.style.right = `${bodyRect.width - right + offset}px`; this._dropdownList.style.maxWidth = `${right + offset}px`; this._dropdownList.style.minWidth = `${Math.min(right, 200)}px`; } if (!right || wide) { const {left} = dropdownRect; const {beside} = this.props; const offset = beside ? dropdownRect.width - dropdownRect.height : 0; this._dropdownList.style.left = `${left + offset}px`; this._dropdownList.style.maxWidth = `${bodyRect.width - left - 5 - offset}px`; this._dropdownList.style.minWidth = `${Math.min(bodyRect.width - left, 200)}px`; } if (dropUp) { const {top} = dropdownRect; this._dropdownList.style.bottom = `${bodyRect.height - top}px`; this._dropdownList.style.maxHeight = `${top - 5}px`; } else { const {bottom} = dropdownRect; this._dropdownList.style.top = `${bottom}px`; this._dropdownList.style.maxHeight = `${bodyRect.height - bottom - 5}px`; } } _handleClick () { this.toggle(); } _handleMouseDown (e) { // Intercept mouse down so that clicks don't trigger things like drag and drop. e.preventDefault(); } _addDropdownListRef (n) { this._dropdownList = n; } _addFilterRef (n) { this._filter = n; // Automatically focus the filter element when mounted so we can start typing if (this._filter) { this._filter.focus(); } } _addDropdownMenuRef (n) { this._dropdownMenu = n; } _getFlattenedChildren (children) { let newChildren = []; for (const child of children) { if (!child) { // Ignore null components continue; } if (Array.isArray(child)) { newChildren = [...newChildren, ...this._getFlattenedChildren(child)]; } else { newChildren.push(child); } } return newChildren; } componentDidUpdate () { this._checkSizeAndPosition(); } _getContainer () { let container = document.querySelector('#dropdowns-container'); if (!container) { container = document.createElement('div'); container.id = 'dropdowns-container'; document.body.appendChild(container); } return container; } componentDidMount () { document.body.addEventListener('keydown', this._handleBodyKeyDown); // Move the element to the body so we can position absolutely if (this._dropdownMenu) { const el = ReactDOM.findDOMNode(this._dropdownMenu); this._getContainer().appendChild(el); } } componentWillUnmount () { document.body.removeEventListener('keydown', this._handleBodyKeyDown); // Remove the element from the body if (this._dropdownMenu) { const el = ReactDOM.findDOMNode(this._dropdownMenu); this._getContainer().removeChild(el); } } hide () { // Focus the dropdown button after hiding if (this._node) { const button = this._node.querySelector('button'); button && button.focus(); } this.setState({open: false}); this.props.onHide && this.props.onHide(); } show () { const bodyHeight = document.body.getBoundingClientRect().height; const dropdownTop = this._node.getBoundingClientRect().top; const dropUp = dropdownTop > bodyHeight - 200; this.setState({ open: true, dropUp, filterVisible: false, filter: '', filterItems: null, filterActiveIndex: -1, uniquenessKey: this.state.uniquenessKey + 1 }); this.props.onOpen && this.props.onOpen(); } toggle () { if (this.state.open) { this.hide(); } else { this.show(); } } render () { const { right, outline, wide, className, style, children } = this.props; const { dropUp, open, uniquenessKey, filterVisible, filterActiveIndex, filterItems, filter } = this.state; const classes = classnames('dropdown', className, { 'dropdown--wide': wide, 'dropdown--open': open }); const menuClasses = classnames({ 'dropdown__menu': true, 'theme--dropdown__menu': true, 'dropdown__menu--open': open, 'dropdown__menu--outlined': outline, 'dropdown__menu--up': dropUp, 'dropdown__menu--right': right }); const dropdownButtons = []; const dropdownItems = []; const listedChildren = Array.isArray(children) ? children : [children]; const allChildren = this._getFlattenedChildren(listedChildren); const visibleChildren = allChildren.filter((child, i) => { if (child.type.name !== DropdownItem.name) { return true; } // It's visible if its index is in the filterItems return !filterItems || filterItems.includes(i); }); for (let i = 0; i < allChildren.length; i++) { const child = allChildren[i]; if (child.type.name === DropdownButton.name) { dropdownButtons.push(child); } else if (child.type.name === DropdownItem.name) { const active = i === filterActiveIndex; const hide = !visibleChildren.includes(child); dropdownItems.push(