bigbuffet-rw/app/soapbox/components/dropdown_menu.js

343 lines
8.8 KiB
JavaScript
Raw Normal View History

import classNames from 'classnames';
import { supportsPassiveEvents } from 'detect-passive-events';
2020-03-27 13:59:38 -07:00
import PropTypes from 'prop-types';
import React from 'react';
2020-03-27 13:59:38 -07:00
import ImmutablePropTypes from 'react-immutable-proptypes';
import spring from 'react-motion/lib/spring';
import Overlay from 'react-overlays/lib/Overlay';
import Icon from 'soapbox/components/icon';
2022-01-10 14:01:24 -08:00
import Motion from '../features/ui/util/optional_motion';
2022-01-10 14:01:24 -08:00
import IconButton from './icon_button';
2020-03-27 13:59:38 -07:00
const listenerOptions = supportsPassiveEvents ? { passive: true } : false;
2020-03-27 13:59:38 -07:00
let id = 0;
class DropdownMenu extends React.PureComponent {
static contextTypes = {
router: PropTypes.object,
};
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,
};
static defaultProps = {
style: {},
placement: 'bottom',
};
state = {
mounted: false,
};
handleDocumentClick = e => {
if (this.node && !this.node.contains(e.target)) {
this.props.onClose();
}
}
componentDidMount() {
2020-03-27 13:59:38 -07:00
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 });
}
2020-03-27 13:59:38 -07:00
this.setState({ mounted: true });
}
componentWillUnmount() {
2020-03-27 13:59:38 -07:00
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;
2020-03-27 13:59:38 -07:00
switch(e.key) {
case 'ArrowDown':
element = items[index+1] || items[0];
2020-03-27 13:59:38 -07:00
break;
case 'ArrowUp':
element = items[index-1] || items[items.length-1];
2020-03-27 13:59:38 -07:00
break;
case 'Tab':
if (e.shiftKey) {
element = items[index-1] || items[items.length-1];
} else {
element = items[index+1] || items[0];
}
break;
2020-03-27 13:59:38 -07:00
case 'Home':
element = items[0];
break;
case 'End':
element = items[items.length-1];
break;
case 'Escape':
this.props.onClose();
break;
2020-03-27 13:59:38 -07:00
}
if (element) {
element.focus();
e.preventDefault();
e.stopPropagation();
}
2020-03-27 13:59:38 -07:00
}
handleItemKeyPress = e => {
if (e.key === 'Enter' || e.key === ' ') {
2020-03-27 13:59:38 -07:00
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.context.router.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) {
2020-03-27 13:59:38 -07:00
if (option === null) {
return <li key={`sep-${i}`} className='dropdown-menu__separator' />;
}
const { text, href, to, newTab, isLogout, icon, destructive } = option;
2020-03-27 13:59:38 -07:00
return (
<li className={classNames('dropdown-menu__item', { destructive })} key={`${text}-${i}`}>
2020-03-27 13:59:38 -07:00
<a
2021-03-29 21:26:50 -07:00
href={href || to || '#'}
2020-03-27 13:59:38 -07:00
role='button'
tabIndex='0'
ref={i === 0 ? this.setFocusRef : null}
onClick={this.handleClick}
onAuxClick={this.handleAuxClick}
onKeyPress={this.handleItemKeyPress}
2020-03-27 13:59:38 -07:00
data-index={i}
target={newTab ? '_blank' : null}
2020-04-14 11:44:40 -07:00
data-method={isLogout ? 'delete' : null}
>
{icon && <Icon src={icon} />}
2020-03-27 13:59:38 -07:00
{text}
</a>
</li>
);
}
render() {
2020-03-27 13:59:38 -07:00
const { items, style, placement, arrowOffsetLeft, arrowOffsetTop } = this.props;
const { mounted } = this.state;
return (
<Motion defaultStyle={{ opacity: 0, scaleX: 1, scaleY: 1 }} style={{ opacity: spring(1, { damping: 35, stiffness: 400 }), scaleX: spring(1, { damping: 35, stiffness: 400 }), scaleY: spring(1, { damping: 35, stiffness: 400 }) }}>
{({ opacity, scaleX, scaleY }) => (
// It should not be transformed when mounting because the resulting
// size will be used to determine the coordinate of the menu by
// react-overlays
<div className={`dropdown-menu ${placement}`} style={{ ...style, opacity: opacity, transform: mounted ? `scale(${scaleX}, ${scaleY})` : null }} ref={this.setRef}>
<div className={`dropdown-menu__arrow ${placement}`} style={{ left: arrowOffsetLeft, top: arrowOffsetTop }} />
<ul>
{items.map((option, i) => this.renderItem(option, i))}
</ul>
</div>
)}
</Motion>
);
}
}
export default class Dropdown extends React.PureComponent {
static contextTypes = {
router: PropTypes.object,
};
static propTypes = {
2021-09-21 11:52:45 -07:00
icon: PropTypes.string,
src: PropTypes.string,
2020-03-27 13:59:38 -07:00
items: PropTypes.array.isRequired,
2021-09-21 14:43:44 -07:00
size: PropTypes.number,
active: PropTypes.bool,
pressed: PropTypes.bool,
2020-03-27 13:59:38 -07:00
title: PropTypes.string,
disabled: PropTypes.bool,
status: ImmutablePropTypes.map,
isUserTouching: PropTypes.func,
isModalOpen: PropTypes.bool.isRequired,
onOpen: PropTypes.func.isRequired,
onClose: PropTypes.func.isRequired,
dropdownPlacement: PropTypes.string,
openDropdownId: PropTypes.number,
openedViaKeyboard: PropTypes.bool,
text: PropTypes.string,
onShiftClick: PropTypes.func,
2020-03-27 13:59:38 -07:00
};
static defaultProps = {
title: 'Menu',
};
state = {
id: id++,
};
handleClick = e => {
const { onOpen, onShiftClick, openDropdownId } = this.props;
if (onShiftClick && e.shiftKey) {
e.preventDefault();
onShiftClick(e);
} else if (this.state.id === openDropdownId) {
2020-03-27 13:59:38 -07:00
this.handleClose();
} else {
const { top } = e.target.getBoundingClientRect();
2020-03-27 13:59:38 -07:00
const placement = top * 2 < innerHeight ? 'bottom' : 'top';
onOpen(this.state.id, this.handleItemClick, placement, e.type !== 'click');
2020-03-27 13:59:38 -07:00
}
}
handleClose = () => {
if (this.activeElement) {
this.activeElement.focus();
this.activeElement = null;
}
2020-03-27 13:59:38 -07:00
this.props.onClose(this.state.id);
}
handleMouseDown = () => {
if (!this.state.open) {
this.activeElement = document.activeElement;
}
}
handleButtonKeyDown = (e) => {
switch(e.key) {
case ' ':
case 'Enter':
this.handleMouseDown();
break;
}
}
handleKeyPress = (e) => {
switch(e.key) {
case ' ':
case 'Enter':
this.handleClick(e);
e.stopPropagation();
e.preventDefault();
break;
}
}
2020-03-27 13:59:38 -07:00
handleItemClick = e => {
const i = Number(e.currentTarget.getAttribute('data-index'));
const { action, to } = this.props.items[i];
this.handleClose();
if (typeof action === 'function') {
e.preventDefault();
action();
} else if (to) {
e.preventDefault();
this.context.router.history.push(to);
}
}
setTargetRef = c => {
this.target = c;
}
findTarget = () => {
return this.target;
}
componentWillUnmount = () => {
if (this.state.id === this.props.openDropdownId) {
this.handleClose();
}
}
render() {
const { icon, src, items, size, title, disabled, dropdownPlacement, openDropdownId, openedViaKeyboard, active, pressed, text } = this.props;
2020-03-27 13:59:38 -07:00
const open = this.state.id === openDropdownId;
return (
<div>
2020-03-27 13:59:38 -07:00
<IconButton
icon={icon}
2021-09-21 11:52:45 -07:00
src={src}
2020-03-27 13:59:38 -07:00
title={title}
active={open || active}
pressed={pressed}
2020-03-27 13:59:38 -07:00
disabled={disabled}
size={size}
text={text}
2020-03-27 13:59:38 -07:00
ref={this.setTargetRef}
onClick={this.handleClick}
onMouseDown={this.handleMouseDown}
onKeyDown={this.handleButtonKeyDown}
onKeyPress={this.handleKeyPress}
2020-03-27 13:59:38 -07:00
/>
<Overlay show={open} placement={dropdownPlacement} target={this.findTarget}>
<DropdownMenu items={items} onClose={this.handleClose} openedViaKeyboard={openedViaKeyboard} />
</Overlay>
</div>
);
}
}