209 lines
6.9 KiB
TypeScript
209 lines
6.9 KiB
TypeScript
/*
|
|
Copyright 2024 New Vector Ltd.
|
|
Copyright 2022 The Matrix.org Foundation C.I.C.
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
|
|
Please see LICENSE files in the repository root for full details.
|
|
*/
|
|
|
|
import classNames from "classnames";
|
|
import React, { FunctionComponent, Key, PropsWithChildren, ReactNode, type JSX } from "react";
|
|
|
|
import { MenuItemRadio } from "../../accessibility/context_menu/MenuItemRadio";
|
|
import { ButtonEvent } from "../views/elements/AccessibleButton";
|
|
import ContextMenu, { aboveLeftOf, ChevronFace, ContextMenuButton, useContextMenu } from "./ContextMenu";
|
|
|
|
export type GenericDropdownMenuOption<T> = {
|
|
key: T;
|
|
label: ReactNode;
|
|
description?: ReactNode;
|
|
adornment?: ReactNode;
|
|
};
|
|
|
|
export type GenericDropdownMenuGroup<T> = GenericDropdownMenuOption<T> & {
|
|
options: GenericDropdownMenuOption<T>[];
|
|
};
|
|
|
|
export type GenericDropdownMenuItem<T> = GenericDropdownMenuGroup<T> | GenericDropdownMenuOption<T>;
|
|
|
|
export function GenericDropdownMenuOption<T extends Key>({
|
|
label,
|
|
description,
|
|
onClick,
|
|
isSelected,
|
|
adornment,
|
|
}: GenericDropdownMenuOption<T> & {
|
|
onClick: (ev: ButtonEvent) => void;
|
|
isSelected: boolean;
|
|
}): JSX.Element {
|
|
return (
|
|
<MenuItemRadio
|
|
active={isSelected}
|
|
className="mx_GenericDropdownMenu_Option mx_GenericDropdownMenu_Option--item"
|
|
onClick={onClick}
|
|
>
|
|
<div className="mx_GenericDropdownMenu_Option--label">
|
|
<span>{label}</span>
|
|
<span>{description}</span>
|
|
</div>
|
|
{adornment}
|
|
</MenuItemRadio>
|
|
);
|
|
}
|
|
|
|
export function GenericDropdownMenuGroup<T extends Key>({
|
|
label,
|
|
description,
|
|
adornment,
|
|
children,
|
|
}: PropsWithChildren<GenericDropdownMenuOption<T>>): JSX.Element {
|
|
return (
|
|
<>
|
|
<div className="mx_GenericDropdownMenu_Option mx_GenericDropdownMenu_Option--header">
|
|
<div className="mx_GenericDropdownMenu_Option--label">
|
|
<span>{label}</span>
|
|
<span>{description}</span>
|
|
</div>
|
|
{adornment}
|
|
</div>
|
|
{children}
|
|
</>
|
|
);
|
|
}
|
|
|
|
function isGenericDropdownMenuGroupArray<T>(
|
|
items: readonly GenericDropdownMenuItem<T>[],
|
|
): items is GenericDropdownMenuGroup<T>[] {
|
|
return isGenericDropdownMenuGroup(items[0]);
|
|
}
|
|
|
|
function isGenericDropdownMenuGroup<T>(item: GenericDropdownMenuItem<T>): item is GenericDropdownMenuGroup<T> {
|
|
return "options" in item;
|
|
}
|
|
|
|
type WithKeyFunction<T> = T extends Key
|
|
? {
|
|
toKey?: (key: T) => Key;
|
|
}
|
|
: {
|
|
toKey: (key: T) => Key;
|
|
};
|
|
|
|
export interface AdditionalOptionsProps {
|
|
menuDisplayed: boolean;
|
|
closeMenu: () => void;
|
|
openMenu: () => void;
|
|
}
|
|
|
|
type IProps<T> = WithKeyFunction<T> & {
|
|
value: T;
|
|
options: readonly GenericDropdownMenuOption<T>[] | readonly GenericDropdownMenuGroup<T>[];
|
|
onChange: (option: T) => void;
|
|
selectedLabel: (option: GenericDropdownMenuItem<T> | null | undefined) => ReactNode;
|
|
onOpen?: (ev: ButtonEvent) => void;
|
|
onClose?: (ev: ButtonEvent) => void;
|
|
className?: string;
|
|
AdditionalOptions?: FunctionComponent<AdditionalOptionsProps>;
|
|
};
|
|
|
|
function calculateKey<T>(value: T, toKey: ((key: T) => Key) | undefined): Key {
|
|
return toKey ? toKey(value) : (value as Key);
|
|
}
|
|
|
|
export function GenericDropdownMenu<T>({
|
|
value,
|
|
onChange,
|
|
options,
|
|
selectedLabel,
|
|
onOpen,
|
|
onClose,
|
|
toKey,
|
|
className,
|
|
AdditionalOptions,
|
|
}: IProps<T>): JSX.Element {
|
|
const [menuDisplayed, button, openMenu, closeMenu] = useContextMenu<HTMLElement>();
|
|
|
|
const valueKey = calculateKey(value, toKey);
|
|
const selected: GenericDropdownMenuItem<T> | undefined = options
|
|
.flatMap((it) => (isGenericDropdownMenuGroup(it) ? [it, ...it.options] : [it]))
|
|
.find((option) => calculateKey(option.key, toKey) === valueKey);
|
|
let contextMenuOptions: JSX.Element;
|
|
if (options && isGenericDropdownMenuGroupArray(options)) {
|
|
contextMenuOptions = (
|
|
<>
|
|
{options.map((group) => (
|
|
<GenericDropdownMenuGroup
|
|
key={calculateKey(group.key, toKey)}
|
|
label={group.label}
|
|
description={group.description}
|
|
adornment={group.adornment}
|
|
>
|
|
{group.options.map((option) => (
|
|
<GenericDropdownMenuOption
|
|
key={calculateKey(option.key, toKey)}
|
|
label={option.label}
|
|
description={option.description}
|
|
onClick={(ev: ButtonEvent) => {
|
|
onChange(option.key);
|
|
closeMenu();
|
|
onClose?.(ev);
|
|
}}
|
|
adornment={option.adornment}
|
|
isSelected={calculateKey(option.key, toKey) === valueKey}
|
|
/>
|
|
))}
|
|
</GenericDropdownMenuGroup>
|
|
))}
|
|
</>
|
|
);
|
|
} else {
|
|
contextMenuOptions = (
|
|
<>
|
|
{options.map((option) => (
|
|
<GenericDropdownMenuOption
|
|
key={calculateKey(option.key, toKey)}
|
|
label={option.label}
|
|
description={option.description}
|
|
onClick={(ev: ButtonEvent) => {
|
|
onChange(option.key);
|
|
closeMenu();
|
|
onClose?.(ev);
|
|
}}
|
|
adornment={option.adornment}
|
|
isSelected={calculateKey(option.key, toKey) === valueKey}
|
|
/>
|
|
))}
|
|
</>
|
|
);
|
|
}
|
|
const contextMenu =
|
|
menuDisplayed && button.current ? (
|
|
<ContextMenu
|
|
onFinished={closeMenu}
|
|
chevronFace={ChevronFace.Top}
|
|
wrapperClassName={classNames("mx_GenericDropdownMenu_wrapper", className)}
|
|
{...aboveLeftOf(button.current.getBoundingClientRect())}
|
|
>
|
|
{contextMenuOptions}
|
|
{AdditionalOptions && (
|
|
<AdditionalOptions menuDisplayed={menuDisplayed} openMenu={openMenu} closeMenu={closeMenu} />
|
|
)}
|
|
</ContextMenu>
|
|
) : null;
|
|
return (
|
|
<>
|
|
<ContextMenuButton
|
|
className="mx_GenericDropdownMenu_button"
|
|
ref={button}
|
|
isExpanded={menuDisplayed}
|
|
onClick={(ev: ButtonEvent) => {
|
|
openMenu();
|
|
onOpen?.(ev);
|
|
}}
|
|
>
|
|
{selectedLabel(selected)}
|
|
</ContextMenuButton>
|
|
{contextMenu}
|
|
</>
|
|
);
|
|
}
|