245 lines
6.5 KiB
TypeScript
245 lines
6.5 KiB
TypeScript
|
// Copyright 2023 Signal Messenger, LLC
|
||
|
// SPDX-License-Identifier: AGPL-3.0-only
|
||
|
|
||
|
import React, { type RefObject } from 'react';
|
||
|
import { ContextMenu, MenuItem } from 'react-contextmenu';
|
||
|
import ReactDOM from 'react-dom';
|
||
|
import type { LocalizerType } from '../../types/I18N';
|
||
|
|
||
|
export type ContextMenuTriggerType = {
|
||
|
handleContextClick: (
|
||
|
event: React.MouseEvent<HTMLDivElement> | MouseEvent
|
||
|
) => void;
|
||
|
};
|
||
|
|
||
|
type MessageContextProps = {
|
||
|
i18n: LocalizerType;
|
||
|
triggerId: string;
|
||
|
shouldShowAdditional: boolean;
|
||
|
|
||
|
onDownload: (() => void) | undefined;
|
||
|
onEdit: (() => void) | undefined;
|
||
|
onReplyToMessage: (() => void) | undefined;
|
||
|
onReact: (() => void) | undefined;
|
||
|
onRetryMessageSend: (() => void) | undefined;
|
||
|
onRetryDeleteForEveryone: (() => void) | undefined;
|
||
|
onCopy: (() => void) | undefined;
|
||
|
onForward: (() => void) | undefined;
|
||
|
onDeleteMessage: () => void;
|
||
|
onMoreInfo: (() => void) | undefined;
|
||
|
onSelect: (() => void) | undefined;
|
||
|
};
|
||
|
export const MessageContextMenu = ({
|
||
|
i18n,
|
||
|
triggerId,
|
||
|
shouldShowAdditional,
|
||
|
onDownload,
|
||
|
onEdit,
|
||
|
onReplyToMessage,
|
||
|
onReact,
|
||
|
onMoreInfo,
|
||
|
onCopy,
|
||
|
onSelect,
|
||
|
onRetryMessageSend,
|
||
|
onRetryDeleteForEveryone,
|
||
|
onForward,
|
||
|
onDeleteMessage,
|
||
|
}: MessageContextProps): JSX.Element => {
|
||
|
const menu = (
|
||
|
<ContextMenu id={triggerId}>
|
||
|
{shouldShowAdditional && (
|
||
|
<>
|
||
|
{onDownload && (
|
||
|
<MenuItem
|
||
|
attributes={{
|
||
|
className:
|
||
|
'module-message__context--icon module-message__context__download',
|
||
|
}}
|
||
|
onClick={onDownload}
|
||
|
>
|
||
|
{i18n('icu:MessageContextMenu__download')}
|
||
|
</MenuItem>
|
||
|
)}
|
||
|
{onReplyToMessage && (
|
||
|
<MenuItem
|
||
|
attributes={{
|
||
|
className:
|
||
|
'module-message__context--icon module-message__context__reply',
|
||
|
}}
|
||
|
onClick={(event: React.MouseEvent) => {
|
||
|
event.stopPropagation();
|
||
|
event.preventDefault();
|
||
|
|
||
|
onReplyToMessage();
|
||
|
}}
|
||
|
>
|
||
|
{i18n('icu:MessageContextMenu__reply')}
|
||
|
</MenuItem>
|
||
|
)}
|
||
|
{onReact && (
|
||
|
<MenuItem
|
||
|
attributes={{
|
||
|
className:
|
||
|
'module-message__context--icon module-message__context__react',
|
||
|
}}
|
||
|
onClick={(event: React.MouseEvent) => {
|
||
|
event.stopPropagation();
|
||
|
event.preventDefault();
|
||
|
|
||
|
onReact();
|
||
|
}}
|
||
|
>
|
||
|
{i18n('icu:MessageContextMenu__react')}
|
||
|
</MenuItem>
|
||
|
)}
|
||
|
</>
|
||
|
)}
|
||
|
{onForward && (
|
||
|
<MenuItem
|
||
|
attributes={{
|
||
|
className:
|
||
|
'module-message__context--icon module-message__context__forward-message',
|
||
|
}}
|
||
|
onClick={(event: React.MouseEvent) => {
|
||
|
event.stopPropagation();
|
||
|
event.preventDefault();
|
||
|
|
||
|
onForward();
|
||
|
}}
|
||
|
>
|
||
|
{i18n('icu:MessageContextMenu__forward')}
|
||
|
</MenuItem>
|
||
|
)}
|
||
|
{onEdit && (
|
||
|
<MenuItem
|
||
|
attributes={{
|
||
|
className:
|
||
|
'module-message__context--icon module-message__context__edit-message',
|
||
|
}}
|
||
|
onClick={(event: React.MouseEvent) => {
|
||
|
event.stopPropagation();
|
||
|
event.preventDefault();
|
||
|
|
||
|
onEdit();
|
||
|
}}
|
||
|
>
|
||
|
{i18n('icu:edit')}
|
||
|
</MenuItem>
|
||
|
)}
|
||
|
{onSelect && (
|
||
|
<MenuItem
|
||
|
attributes={{
|
||
|
className:
|
||
|
'module-message__context--icon module-message__context__select',
|
||
|
}}
|
||
|
onClick={() => {
|
||
|
onSelect();
|
||
|
}}
|
||
|
>
|
||
|
{i18n('icu:MessageContextMenu__select')}
|
||
|
</MenuItem>
|
||
|
)}
|
||
|
{onCopy && (
|
||
|
<MenuItem
|
||
|
attributes={{
|
||
|
className:
|
||
|
'module-message__context--icon module-message__context__copy-timestamp',
|
||
|
}}
|
||
|
onClick={() => {
|
||
|
onCopy();
|
||
|
}}
|
||
|
>
|
||
|
{i18n('icu:copy')}
|
||
|
</MenuItem>
|
||
|
)}
|
||
|
{onMoreInfo && (
|
||
|
<MenuItem
|
||
|
attributes={{
|
||
|
className:
|
||
|
'module-message__context--icon module-message__context__more-info',
|
||
|
}}
|
||
|
onClick={(event: React.MouseEvent) => {
|
||
|
event.stopPropagation();
|
||
|
event.preventDefault();
|
||
|
|
||
|
onMoreInfo();
|
||
|
}}
|
||
|
>
|
||
|
{i18n('icu:MessageContextMenu__info')}
|
||
|
</MenuItem>
|
||
|
)}
|
||
|
<MenuItem
|
||
|
attributes={{
|
||
|
className:
|
||
|
'module-message__context--icon module-message__context__delete-message',
|
||
|
}}
|
||
|
onClick={(event: React.MouseEvent) => {
|
||
|
event.stopPropagation();
|
||
|
event.preventDefault();
|
||
|
|
||
|
onDeleteMessage();
|
||
|
}}
|
||
|
>
|
||
|
{i18n('icu:MessageContextMenu__deleteMessage')}
|
||
|
</MenuItem>
|
||
|
{onRetryMessageSend && (
|
||
|
<MenuItem
|
||
|
attributes={{
|
||
|
className:
|
||
|
'module-message__context--icon module-message__context__retry-send',
|
||
|
}}
|
||
|
onClick={(event: React.MouseEvent) => {
|
||
|
event.stopPropagation();
|
||
|
event.preventDefault();
|
||
|
|
||
|
onRetryMessageSend();
|
||
|
}}
|
||
|
>
|
||
|
{i18n('icu:retrySend')}
|
||
|
</MenuItem>
|
||
|
)}
|
||
|
{onRetryDeleteForEveryone && (
|
||
|
<MenuItem
|
||
|
attributes={{
|
||
|
className:
|
||
|
'module-message__context--icon module-message__context__delete-message-for-everyone',
|
||
|
}}
|
||
|
onClick={(event: React.MouseEvent) => {
|
||
|
event.stopPropagation();
|
||
|
event.preventDefault();
|
||
|
|
||
|
onRetryDeleteForEveryone();
|
||
|
}}
|
||
|
>
|
||
|
{i18n('icu:retryDeleteForEveryone')}
|
||
|
</MenuItem>
|
||
|
)}
|
||
|
</ContextMenu>
|
||
|
);
|
||
|
|
||
|
return ReactDOM.createPortal(menu, document.body);
|
||
|
};
|
||
|
|
||
|
export function useHandleMessageContextMenu(
|
||
|
menuTriggerRef: RefObject<ContextMenuTriggerType>
|
||
|
): ContextMenuTriggerType['handleContextClick'] {
|
||
|
return React.useCallback(
|
||
|
(event: React.MouseEvent<HTMLDivElement> | MouseEvent): void => {
|
||
|
const selection = window.getSelection();
|
||
|
|
||
|
if (selection && !selection.isCollapsed) {
|
||
|
return;
|
||
|
}
|
||
|
if (event && event.target instanceof HTMLAnchorElement) {
|
||
|
return;
|
||
|
}
|
||
|
if (menuTriggerRef.current) {
|
||
|
menuTriggerRef.current.handleContextClick(
|
||
|
event ?? new MouseEvent('click')
|
||
|
);
|
||
|
}
|
||
|
},
|
||
|
[menuTriggerRef]
|
||
|
);
|
||
|
}
|