2021-11-07 01:22:08 -08:00
|
|
|
import classNames from 'classnames';
|
2022-01-10 14:17:52 -08:00
|
|
|
import { supportsPassiveEvents } from 'detect-passive-events';
|
|
|
|
import React from 'react';
|
2022-04-02 11:03:12 -07:00
|
|
|
import { spring } from 'react-motion';
|
|
|
|
// @ts-ignore: TODO: upgrade react-overlays. v3.1 and above have TS definitions
|
2022-01-10 14:17:52 -08:00
|
|
|
import Overlay from 'react-overlays/lib/Overlay';
|
2022-04-02 11:03:12 -07:00
|
|
|
import { withRouter, RouteComponentProps } from 'react-router-dom';
|
2022-01-10 14:25:06 -08:00
|
|
|
|
2021-11-04 11:16:04 -07:00
|
|
|
import Icon from 'soapbox/components/icon';
|
2022-04-02 11:03:12 -07:00
|
|
|
import { IconButton } from 'soapbox/components/ui';
|
|
|
|
import Motion from 'soapbox/features/ui/util/optional_motion';
|
2022-01-10 14:25:06 -08:00
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
import type { Status } from 'soapbox/types/entities';
|
2020-03-27 13:59:38 -07:00
|
|
|
|
2020-10-15 07:10:45 -07:00
|
|
|
const listenerOptions = supportsPassiveEvents ? { passive: true } : false;
|
2020-03-27 13:59:38 -07:00
|
|
|
let id = 0;
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
export interface MenuItem {
|
|
|
|
action: React.EventHandler<React.KeyboardEvent | React.MouseEvent>,
|
|
|
|
middleClick?: React.EventHandler<React.MouseEvent>,
|
|
|
|
text: string,
|
|
|
|
href?: string,
|
|
|
|
to?: string,
|
|
|
|
newTab?: boolean,
|
|
|
|
isLogout?: boolean,
|
|
|
|
icon: string,
|
|
|
|
destructive?: boolean,
|
|
|
|
}
|
|
|
|
|
|
|
|
export type Menu = Array<MenuItem | null>;
|
|
|
|
|
|
|
|
interface IDropdownMenu extends RouteComponentProps {
|
|
|
|
items: Menu,
|
|
|
|
onClose: () => void,
|
|
|
|
style?: React.CSSProperties,
|
|
|
|
placement?: DropdownPlacement,
|
|
|
|
arrowOffsetLeft?: string,
|
|
|
|
arrowOffsetTop?: string,
|
|
|
|
openedViaKeyboard: boolean,
|
|
|
|
}
|
2020-03-27 13:59:38 -07:00
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
interface IDropdownMenuState {
|
|
|
|
mounted: boolean,
|
|
|
|
}
|
|
|
|
|
|
|
|
class DropdownMenu extends React.PureComponent<IDropdownMenu, IDropdownMenuState> {
|
|
|
|
|
|
|
|
static defaultProps: Partial<IDropdownMenu> = {
|
2020-03-27 13:59:38 -07:00
|
|
|
style: {},
|
|
|
|
placement: 'bottom',
|
|
|
|
};
|
|
|
|
|
|
|
|
state = {
|
|
|
|
mounted: false,
|
|
|
|
};
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
node: HTMLDivElement | null = null;
|
|
|
|
focusedItem: HTMLAnchorElement | null = null;
|
|
|
|
|
|
|
|
handleDocumentClick = (e: Event) => {
|
|
|
|
if (this.node && !this.node.contains(e.target as Node)) {
|
2020-03-27 13:59:38 -07:00
|
|
|
this.props.onClose();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-14 14:47:35 -07:00
|
|
|
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);
|
2021-08-28 05:52:39 -07:00
|
|
|
if (this.focusedItem && this.props.openedViaKeyboard) {
|
2021-08-28 06:38:51 -07:00
|
|
|
this.focusedItem.focus({ preventScroll: true });
|
2021-08-28 05:52:39 -07:00
|
|
|
}
|
2020-03-27 13:59:38 -07:00
|
|
|
this.setState({ mounted: true });
|
|
|
|
}
|
|
|
|
|
2020-04-14 14:47:35 -07:00
|
|
|
componentWillUnmount() {
|
2022-04-02 11:03:12 -07:00
|
|
|
document.removeEventListener('click', this.handleDocumentClick);
|
|
|
|
document.removeEventListener('keydown', this.handleKeyDown);
|
|
|
|
document.removeEventListener('touchend', this.handleDocumentClick);
|
2020-03-27 13:59:38 -07:00
|
|
|
}
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
setRef: React.RefCallback<HTMLDivElement> = c => {
|
2020-03-27 13:59:38 -07:00
|
|
|
this.node = c;
|
|
|
|
}
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
setFocusRef: React.RefCallback<HTMLAnchorElement> = c => {
|
2020-03-27 13:59:38 -07:00
|
|
|
this.focusedItem = c;
|
|
|
|
}
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
handleKeyDown = (e: KeyboardEvent) => {
|
|
|
|
if (!this.node) return;
|
|
|
|
|
2020-03-27 13:59:38 -07:00
|
|
|
const items = Array.from(this.node.getElementsByTagName('a'));
|
2022-04-02 11:03:12 -07:00
|
|
|
const index = items.indexOf(document.activeElement as any);
|
2021-08-28 06:37:53 -07:00
|
|
|
let element = null;
|
2020-03-27 13:59:38 -07:00
|
|
|
|
|
|
|
switch(e.key) {
|
|
|
|
case 'ArrowDown':
|
2021-08-28 06:37:53 -07:00
|
|
|
element = items[index+1] || items[0];
|
2020-03-27 13:59:38 -07:00
|
|
|
break;
|
|
|
|
case 'ArrowUp':
|
2021-08-28 06:37:53 -07:00
|
|
|
element = items[index-1] || items[items.length-1];
|
2020-03-27 13:59:38 -07:00
|
|
|
break;
|
2021-08-28 05:52:39 -07:00
|
|
|
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;
|
2021-08-28 05:52:39 -07:00
|
|
|
case 'Escape':
|
|
|
|
this.props.onClose();
|
|
|
|
break;
|
2020-03-27 13:59:38 -07:00
|
|
|
}
|
2021-08-28 06:37:53 -07:00
|
|
|
|
|
|
|
if (element) {
|
|
|
|
element.focus();
|
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
|
|
|
}
|
2020-03-27 13:59:38 -07:00
|
|
|
}
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
handleItemKeyPress: React.EventHandler<React.KeyboardEvent> = e => {
|
2021-08-28 05:52:39 -07:00
|
|
|
if (e.key === 'Enter' || e.key === ' ') {
|
2020-03-27 13:59:38 -07:00
|
|
|
this.handleClick(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
handleClick: React.EventHandler<React.MouseEvent | React.KeyboardEvent> = e => {
|
2020-03-27 13:59:38 -07:00
|
|
|
const i = Number(e.currentTarget.getAttribute('data-index'));
|
2022-04-02 11:03:12 -07:00
|
|
|
const item = this.props.items[i];
|
|
|
|
if (!item) return;
|
|
|
|
const { action, to } = item;
|
2020-03-27 13:59:38 -07:00
|
|
|
|
|
|
|
this.props.onClose();
|
|
|
|
|
|
|
|
if (typeof action === 'function') {
|
|
|
|
e.preventDefault();
|
|
|
|
action(e);
|
|
|
|
} else if (to) {
|
|
|
|
e.preventDefault();
|
2022-03-17 18:17:28 -07:00
|
|
|
this.props.history.push(to);
|
2020-03-27 13:59:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
handleMiddleClick: React.EventHandler<React.MouseEvent> = e => {
|
2021-03-29 21:22:54 -07:00
|
|
|
const i = Number(e.currentTarget.getAttribute('data-index'));
|
2022-04-02 11:03:12 -07:00
|
|
|
const item = this.props.items[i];
|
|
|
|
if (!item) return;
|
|
|
|
const { middleClick } = item;
|
2021-03-29 21:22:54 -07:00
|
|
|
|
|
|
|
this.props.onClose();
|
|
|
|
|
|
|
|
if (e.button === 1 && typeof middleClick === 'function') {
|
|
|
|
e.preventDefault();
|
|
|
|
middleClick(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
handleAuxClick: React.EventHandler<React.MouseEvent> = e => {
|
2021-03-29 21:22:54 -07:00
|
|
|
if (e.button === 1) {
|
|
|
|
this.handleMiddleClick(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
renderItem(option: MenuItem | null, i: number): JSX.Element {
|
2020-03-27 13:59:38 -07:00
|
|
|
if (option === null) {
|
|
|
|
return <li key={`sep-${i}`} className='dropdown-menu__separator' />;
|
|
|
|
}
|
|
|
|
|
2021-11-08 08:21:33 -08:00
|
|
|
const { text, href, to, newTab, isLogout, icon, destructive } = option;
|
2020-03-27 13:59:38 -07:00
|
|
|
|
|
|
|
return (
|
2021-11-08 08:21:33 -08:00
|
|
|
<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'
|
2022-04-02 11:03:12 -07:00
|
|
|
tabIndex={0}
|
2020-03-27 13:59:38 -07:00
|
|
|
ref={i === 0 ? this.setFocusRef : null}
|
|
|
|
onClick={this.handleClick}
|
2021-03-29 21:22:54 -07:00
|
|
|
onAuxClick={this.handleAuxClick}
|
2021-08-28 06:05:26 -07:00
|
|
|
onKeyPress={this.handleItemKeyPress}
|
2020-03-27 13:59:38 -07:00
|
|
|
data-index={i}
|
2022-04-02 11:03:12 -07:00
|
|
|
target={newTab ? '_blank' : undefined}
|
|
|
|
data-method={isLogout ? 'delete' : undefined}
|
2020-04-14 11:44:40 -07:00
|
|
|
>
|
2021-11-04 11:16:04 -07:00
|
|
|
{icon && <Icon src={icon} />}
|
2020-03-27 13:59:38 -07:00
|
|
|
{text}
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-04-14 14:47:35 -07:00
|
|
|
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
|
2022-04-02 11:03:12 -07:00
|
|
|
<div className={`dropdown-menu ${placement}`} style={{ ...style, opacity: opacity, transform: mounted ? `scale(${scaleX}, ${scaleY})` : undefined }} ref={this.setRef}>
|
2020-03-27 13:59:38 -07:00
|
|
|
<div className={`dropdown-menu__arrow ${placement}`} style={{ left: arrowOffsetLeft, top: arrowOffsetTop }} />
|
|
|
|
<ul>
|
|
|
|
{items.map((option, i) => this.renderItem(option, i))}
|
|
|
|
</ul>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</Motion>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
const RouterDropdownMenu = withRouter(DropdownMenu);
|
|
|
|
|
|
|
|
export interface IDropdown extends RouteComponentProps {
|
|
|
|
icon?: string,
|
|
|
|
src: string,
|
|
|
|
items: Menu,
|
|
|
|
size?: number,
|
|
|
|
active?: boolean,
|
|
|
|
pressed?: boolean,
|
|
|
|
title: string,
|
|
|
|
disabled?: boolean,
|
|
|
|
status?: Status,
|
|
|
|
isUserTouching?: () => boolean,
|
|
|
|
isModalOpen?: boolean,
|
|
|
|
onOpen?: (
|
|
|
|
id: number,
|
|
|
|
onItemClick: React.EventHandler<React.MouseEvent | React.KeyboardEvent>,
|
|
|
|
dropdownPlacement: DropdownPlacement,
|
|
|
|
keyboard: boolean,
|
|
|
|
) => void,
|
|
|
|
onClose?: (id: number) => void,
|
|
|
|
dropdownPlacement?: string,
|
|
|
|
openDropdownId?: number,
|
|
|
|
openedViaKeyboard?: boolean,
|
|
|
|
text?: string,
|
|
|
|
onShiftClick?: React.EventHandler<React.MouseEvent | React.KeyboardEvent>,
|
|
|
|
}
|
|
|
|
|
|
|
|
interface IDropdownState {
|
|
|
|
id: number,
|
|
|
|
open: boolean,
|
|
|
|
}
|
2020-03-27 13:59:38 -07:00
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
export type DropdownPlacement = 'top' | 'bottom';
|
|
|
|
|
|
|
|
class Dropdown extends React.PureComponent<IDropdown, IDropdownState> {
|
|
|
|
|
|
|
|
static defaultProps: Partial<IDropdown> = {
|
2020-03-27 13:59:38 -07:00
|
|
|
title: 'Menu',
|
|
|
|
};
|
|
|
|
|
|
|
|
state = {
|
|
|
|
id: id++,
|
2022-04-02 11:03:12 -07:00
|
|
|
open: false,
|
2020-03-27 13:59:38 -07:00
|
|
|
};
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
target: HTMLButtonElement | null = null;
|
|
|
|
activeElement: Element | null = null;
|
|
|
|
|
|
|
|
handleClick: React.EventHandler<React.MouseEvent<HTMLButtonElement> | React.KeyboardEvent<HTMLButtonElement>> = e => {
|
2022-01-25 13:59:26 -08:00
|
|
|
const { onOpen, onShiftClick, openDropdownId } = this.props;
|
2022-03-21 11:09:01 -07:00
|
|
|
e.stopPropagation();
|
2022-01-25 13:59:26 -08:00
|
|
|
|
|
|
|
if (onShiftClick && e.shiftKey) {
|
|
|
|
e.preventDefault();
|
|
|
|
onShiftClick(e);
|
|
|
|
} else if (this.state.id === openDropdownId) {
|
2020-03-27 13:59:38 -07:00
|
|
|
this.handleClose();
|
2022-04-02 11:03:12 -07:00
|
|
|
} else if(onOpen) {
|
|
|
|
const { top } = e.currentTarget.getBoundingClientRect();
|
|
|
|
const placement: DropdownPlacement = top * 2 < innerHeight ? 'bottom' : 'top';
|
2020-03-27 13:59:38 -07:00
|
|
|
|
2022-01-25 13:59:26 -08:00
|
|
|
onOpen(this.state.id, this.handleItemClick, placement, e.type !== 'click');
|
2020-03-27 13:59:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
handleClose = () => {
|
2022-04-02 11:03:12 -07:00
|
|
|
if (this.activeElement && this.activeElement === this.target) {
|
|
|
|
(this.activeElement as HTMLButtonElement).focus();
|
2021-08-28 06:05:26 -07:00
|
|
|
this.activeElement = null;
|
|
|
|
}
|
2022-04-02 11:03:12 -07:00
|
|
|
|
|
|
|
if (this.props.onClose) {
|
|
|
|
this.props.onClose(this.state.id);
|
|
|
|
}
|
2020-03-27 13:59:38 -07:00
|
|
|
}
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
handleMouseDown: React.EventHandler<React.MouseEvent | React.KeyboardEvent> = () => {
|
2021-08-28 06:05:26 -07:00
|
|
|
if (!this.state.open) {
|
|
|
|
this.activeElement = document.activeElement;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
handleButtonKeyDown: React.EventHandler<React.KeyboardEvent> = (e) => {
|
2021-08-28 06:05:26 -07:00
|
|
|
switch(e.key) {
|
|
|
|
case ' ':
|
|
|
|
case 'Enter':
|
2022-04-02 11:03:12 -07:00
|
|
|
this.handleMouseDown(e);
|
2021-08-28 06:05:26 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
handleKeyPress: React.EventHandler<React.KeyboardEvent<HTMLButtonElement>> = (e) => {
|
2021-08-28 06:05:26 -07:00
|
|
|
switch(e.key) {
|
|
|
|
case ' ':
|
|
|
|
case 'Enter':
|
|
|
|
this.handleClick(e);
|
|
|
|
e.stopPropagation();
|
|
|
|
e.preventDefault();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
handleItemClick: React.EventHandler<React.MouseEvent> = e => {
|
2020-03-27 13:59:38 -07:00
|
|
|
const i = Number(e.currentTarget.getAttribute('data-index'));
|
2022-04-02 11:03:12 -07:00
|
|
|
const item = this.props.items[i];
|
|
|
|
if (!item) return;
|
|
|
|
|
|
|
|
const { action, to } = item;
|
2020-03-27 13:59:38 -07:00
|
|
|
|
|
|
|
this.handleClose();
|
2022-03-21 11:09:01 -07:00
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
2020-03-27 13:59:38 -07:00
|
|
|
|
|
|
|
if (typeof action === 'function') {
|
2022-03-21 11:09:01 -07:00
|
|
|
action(e);
|
2020-03-27 13:59:38 -07:00
|
|
|
} else if (to) {
|
2022-04-02 11:03:12 -07:00
|
|
|
this.props.history?.push(to);
|
2020-03-27 13:59:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-02 11:03:12 -07:00
|
|
|
setTargetRef: React.RefCallback<HTMLButtonElement> = c => {
|
2020-03-27 13:59:38 -07:00
|
|
|
this.target = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
findTarget = () => {
|
|
|
|
return this.target;
|
|
|
|
}
|
|
|
|
|
|
|
|
componentWillUnmount = () => {
|
|
|
|
if (this.state.id === this.props.openDropdownId) {
|
|
|
|
this.handleClose();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-14 14:47:35 -07:00
|
|
|
render() {
|
2022-04-02 11:03:12 -07:00
|
|
|
const { src, items, title, disabled, dropdownPlacement, openDropdownId, openedViaKeyboard = false, pressed, text } = this.props;
|
2020-03-27 13:59:38 -07:00
|
|
|
const open = this.state.id === openDropdownId;
|
|
|
|
|
|
|
|
return (
|
2022-03-21 11:09:01 -07:00
|
|
|
<>
|
2020-03-27 13:59:38 -07:00
|
|
|
<IconButton
|
2022-03-21 11:09:01 -07:00
|
|
|
disabled={disabled}
|
|
|
|
className={classNames({
|
|
|
|
'text-gray-400 hover:text-gray-600': true,
|
|
|
|
'text-gray-600': open,
|
|
|
|
})}
|
2020-03-27 13:59:38 -07:00
|
|
|
title={title}
|
2022-03-21 11:09:01 -07:00
|
|
|
src={src}
|
2022-04-02 11:03:12 -07:00
|
|
|
aria-pressed={pressed}
|
2022-01-23 13:15:25 -08:00
|
|
|
text={text}
|
2020-03-27 13:59:38 -07:00
|
|
|
onClick={this.handleClick}
|
2021-08-28 06:05:26 -07:00
|
|
|
onMouseDown={this.handleMouseDown}
|
|
|
|
onKeyDown={this.handleButtonKeyDown}
|
|
|
|
onKeyPress={this.handleKeyPress}
|
2022-03-21 11:09:01 -07:00
|
|
|
ref={this.setTargetRef}
|
2020-03-27 13:59:38 -07:00
|
|
|
/>
|
|
|
|
|
|
|
|
<Overlay show={open} placement={dropdownPlacement} target={this.findTarget}>
|
2022-04-02 11:03:12 -07:00
|
|
|
<RouterDropdownMenu items={items} onClose={this.handleClose} openedViaKeyboard={openedViaKeyboard} />
|
2020-03-27 13:59:38 -07:00
|
|
|
</Overlay>
|
2022-03-21 11:09:01 -07:00
|
|
|
</>
|
2020-03-27 13:59:38 -07:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2022-04-02 11:03:12 -07:00
|
|
|
|
|
|
|
export default withRouter(Dropdown);
|