signal-desktop/ts/components/stickers/StickerButton.tsx

386 lines
12 KiB
TypeScript
Raw Normal View History

2023-01-03 19:55:46 +00:00
// Copyright 2019 Signal Messenger, LLC
2020-10-30 20:34:04 +00:00
// SPDX-License-Identifier: AGPL-3.0-only
import * as React from 'react';
import classNames from 'classnames';
import { get, noop } from 'lodash';
import { Manager, Popper, Reference } from 'react-popper';
import { createPortal } from 'react-dom';
2021-12-01 02:14:25 +00:00
import type { StickerPackType, StickerType } from '../../state/ducks/stickers';
import type { LocalizerType } from '../../types/Util';
2021-12-01 02:14:25 +00:00
import type { Theme } from '../../util/theme';
import { StickerPicker } from './StickerPicker';
import { countStickers } from './lib';
import { offsetDistanceModifier } from '../../util/popperUtil';
2021-12-01 02:14:25 +00:00
import { themeClassName } from '../../util/theme';
import { handleOutsideClick } from '../../util/handleOutsideClick';
import * as KeyboardLayout from '../../services/keyboardLayout';
2022-06-15 17:53:08 +00:00
import { useRefMerger } from '../../hooks/useRefMerger';
2023-04-20 17:03:43 +00:00
import { UserText } from '../UserText';
export type OwnProps = {
2021-12-01 02:14:25 +00:00
readonly className?: string;
readonly i18n: LocalizerType;
readonly receivedPacks: ReadonlyArray<StickerPackType>;
readonly installedPacks: ReadonlyArray<StickerPackType>;
readonly blessedPacks: ReadonlyArray<StickerPackType>;
readonly knownPacks: ReadonlyArray<StickerPackType>;
readonly installedPack?: StickerPackType | null;
readonly recentStickers: ReadonlyArray<StickerType>;
readonly onOpenStateChanged?: (isOpen: boolean) => void;
readonly clearInstalledStickerPack: () => unknown;
2021-12-01 02:14:25 +00:00
readonly onClickAddPack?: () => unknown;
readonly onPickSticker: (
packId: string,
stickerId: number,
url: string
) => unknown;
2023-03-01 19:00:50 +00:00
readonly onPickTimeSticker?: (style: 'analog' | 'digital') => unknown;
readonly showIntroduction?: boolean;
readonly clearShowIntroduction: () => unknown;
readonly showPickerHint: boolean;
readonly clearShowPickerHint: () => unknown;
2019-08-06 19:18:37 +00:00
readonly position?: 'top-end' | 'top-start';
2021-12-01 02:14:25 +00:00
readonly theme?: Theme;
};
export type Props = OwnProps;
2022-11-18 00:45:19 +00:00
export const StickerButton = React.memo(function StickerButtonInner({
className,
i18n,
clearInstalledStickerPack,
onClickAddPack,
onPickSticker,
2023-03-01 19:00:50 +00:00
onPickTimeSticker,
2022-11-18 00:45:19 +00:00
recentStickers,
onOpenStateChanged,
receivedPacks,
installedPack,
installedPacks,
blessedPacks,
knownPacks,
showIntroduction,
clearShowIntroduction,
showPickerHint,
clearShowPickerHint,
position = 'top-end',
theme,
}: Props) {
const [open, internalSetOpen] = React.useState(false);
2022-11-18 00:45:19 +00:00
const setOpen = React.useCallback(
(value: boolean) => {
internalSetOpen(value);
if (onOpenStateChanged) {
onOpenStateChanged(value);
2022-06-15 17:53:08 +00:00
}
2022-11-18 00:45:19 +00:00
},
[internalSetOpen, onOpenStateChanged]
);
2022-11-18 00:45:19 +00:00
const [popperRoot, setPopperRoot] = React.useState<HTMLElement | null>(null);
const buttonRef = React.useRef<HTMLButtonElement | null>(null);
const refMerger = useRefMerger();
2022-11-18 00:45:19 +00:00
const handleClickButton = React.useCallback(() => {
// Clear tooltip state
clearInstalledStickerPack();
clearShowIntroduction();
2019-05-29 22:11:41 +00:00
2022-11-18 00:45:19 +00:00
// Handle button click
if (installedPacks.length === 0) {
2021-12-01 02:14:25 +00:00
onClickAddPack?.();
2022-11-18 00:45:19 +00:00
} else if (popperRoot) {
setOpen(false);
} else {
setOpen(true);
}
}, [
clearInstalledStickerPack,
clearShowIntroduction,
installedPacks,
onClickAddPack,
popperRoot,
setOpen,
]);
2022-11-18 00:45:19 +00:00
const handlePickSticker = React.useCallback(
(packId: string, stickerId: number, url: string) => {
setOpen(false);
onPickSticker(packId, stickerId, url);
},
[setOpen, onPickSticker]
);
2023-03-01 19:00:50 +00:00
const handlePickTimeSticker = React.useCallback(
(style: 'analog' | 'digital') => {
setOpen(false);
onPickTimeSticker?.(style);
},
[setOpen, onPickTimeSticker]
);
2022-11-18 00:45:19 +00:00
const handleClose = React.useCallback(() => {
setOpen(false);
}, [setOpen]);
2022-11-18 00:45:19 +00:00
const handleClickAddPack = React.useCallback(() => {
setOpen(false);
if (showPickerHint) {
clearShowPickerHint();
}
onClickAddPack?.();
}, [onClickAddPack, showPickerHint, setOpen, clearShowPickerHint]);
2022-11-18 00:45:19 +00:00
const handleClearIntroduction = React.useCallback(() => {
clearInstalledStickerPack();
clearShowIntroduction();
}, [clearInstalledStickerPack, clearShowIntroduction]);
2022-11-18 00:45:19 +00:00
// Create popper root and handle outside clicks
React.useEffect(() => {
if (open) {
const root = document.createElement('div');
setPopperRoot(root);
document.body.appendChild(root);
2022-11-18 00:45:19 +00:00
return () => {
document.body.removeChild(root);
setPopperRoot(null);
};
}
2020-01-08 17:44:54 +00:00
2022-11-18 00:45:19 +00:00
return noop;
}, [open, setOpen, setPopperRoot]);
2020-01-08 17:44:54 +00:00
2022-11-18 00:45:19 +00:00
React.useEffect(() => {
if (!open) {
return noop;
}
2022-11-18 00:45:19 +00:00
return handleOutsideClick(
target => {
const targetElement = target as HTMLElement;
const targetClassName = targetElement
? targetElement.className || ''
: '';
2022-11-18 00:45:19 +00:00
// We need to special-case sticker picker header buttons, because they can
// disappear after being clicked, which breaks the .contains() check below.
const isMissingButtonClass =
!targetClassName ||
targetClassName.indexOf('module-sticker-picker__header__button') < 0;
2020-01-08 17:44:54 +00:00
2022-11-18 00:45:19 +00:00
if (!isMissingButtonClass) {
return false;
2020-01-08 17:44:54 +00:00
}
2022-11-18 00:45:19 +00:00
setOpen(false);
return true;
},
{
containerElements: [popperRoot, buttonRef],
name: 'StickerButton',
}
);
}, [open, popperRoot, setOpen]);
2019-11-07 21:36:16 +00:00
2022-11-18 00:45:19 +00:00
// Install keyboard shortcut to open sticker picker
React.useEffect(() => {
const handleKeydown = (event: KeyboardEvent) => {
const { ctrlKey, metaKey, shiftKey } = event;
const commandKey = get(window, 'platform') === 'darwin' && metaKey;
const controlKey = get(window, 'platform') !== 'darwin' && ctrlKey;
const commandOrCtrl = commandKey || controlKey;
const key = KeyboardLayout.lookup(event);
2019-11-07 21:36:16 +00:00
2022-11-18 00:45:19 +00:00
// We don't want to open up if the conversation has any panels open
const panels = document.querySelectorAll('.conversation .panel');
if (panels && panels.length > 1) {
return;
}
2019-11-07 21:36:16 +00:00
if (commandOrCtrl && shiftKey && (key === 'g' || key === 'G')) {
2022-11-18 00:45:19 +00:00
event.stopPropagation();
event.preventDefault();
2022-11-18 00:45:19 +00:00
setOpen(!open);
2020-01-08 17:44:54 +00:00
}
2022-11-18 00:45:19 +00:00
};
document.addEventListener('keydown', handleKeydown);
2020-01-08 17:44:54 +00:00
2022-11-18 00:45:19 +00:00
return () => {
document.removeEventListener('keydown', handleKeydown);
};
}, [open, setOpen]);
// Clear the installed pack after one minute
React.useEffect(() => {
if (installedPack) {
const timerId = setTimeout(clearInstalledStickerPack, 10 * 1000);
2022-11-18 00:45:19 +00:00
return () => {
clearTimeout(timerId);
};
}
2022-11-18 00:45:19 +00:00
return noop;
}, [installedPack, clearInstalledStickerPack]);
if (
countStickers({
knownPacks,
blessedPacks,
installedPacks,
receivedPacks,
}) === 0
) {
return null;
}
return (
<Manager>
<Reference>
{({ ref }) => (
<button
type="button"
ref={refMerger(buttonRef, ref)}
onClick={handleClickButton}
className={classNames(
{
'module-sticker-button__button': true,
'module-sticker-button__button--active': open,
},
className
)}
2023-03-30 00:03:25 +00:00
aria-label={i18n('icu:stickers--StickerPicker--Open')}
2022-11-18 00:45:19 +00:00
/>
)}
</Reference>
{!open && !showIntroduction && installedPack ? (
<Popper
placement={position}
key={installedPack.id}
modifiers={[offsetDistanceModifier(6)]}
>
{({ ref, style, placement, arrowProps }) => (
<div className={theme ? themeClassName(theme) : undefined}>
<button
type="button"
ref={ref}
style={style}
className="module-sticker-button__tooltip"
onClick={clearInstalledStickerPack}
>
{installedPack.cover ? (
2021-12-01 02:14:25 +00:00
<img
2022-11-18 00:45:19 +00:00
className="module-sticker-button__tooltip__image"
src={installedPack.cover.url}
alt={installedPack.title}
2021-12-01 02:14:25 +00:00
/>
2022-11-18 00:45:19 +00:00
) : (
<div className="module-sticker-button__tooltip__image-placeholder" />
)}
<span className="module-sticker-button__tooltip__text">
<span className="module-sticker-button__tooltip__text__title">
2023-04-20 17:03:43 +00:00
<UserText text={installedPack.title} />
2022-11-18 00:45:19 +00:00
</span>{' '}
installed
</span>
<div
ref={arrowProps.ref}
style={arrowProps.style}
className={classNames(
'module-sticker-button__tooltip__triangle',
`module-sticker-button__tooltip__triangle--${placement}`
)}
/>
</button>
</div>
)}
</Popper>
) : null}
{!open && showIntroduction ? (
<Popper placement={position} modifiers={[offsetDistanceModifier(6)]}>
{({ ref, style, placement, arrowProps }) => (
<div className={theme ? themeClassName(theme) : undefined}>
<button
type="button"
ref={ref}
style={style}
className={classNames(
'module-sticker-button__tooltip',
'module-sticker-button__tooltip--introduction'
)}
onClick={handleClearIntroduction}
>
<img
className="module-sticker-button__tooltip--introduction__image"
srcSet="images/sticker_splash@1x.png 1x, images/sticker_splash@2x.png 2x"
2023-03-30 00:03:25 +00:00
alt={i18n(
'icu:stickers--StickerManager--Introduction--Image'
)}
2022-11-18 00:45:19 +00:00
/>
<div className="module-sticker-button__tooltip--introduction__meta">
<div className="module-sticker-button__tooltip--introduction__meta__title">
2023-03-30 00:03:25 +00:00
{i18n('icu:stickers--StickerManager--Introduction--Title')}
</div>
2022-11-18 00:45:19 +00:00
<div className="module-sticker-button__tooltip--introduction__meta__subtitle">
2023-03-30 00:03:25 +00:00
{i18n('icu:stickers--StickerManager--Introduction--Body')}
</div>
2022-11-18 00:45:19 +00:00
</div>
<div className="module-sticker-button__tooltip--introduction__close">
<button
type="button"
className="module-sticker-button__tooltip--introduction__close__button"
onClick={handleClearIntroduction}
2023-03-30 00:03:25 +00:00
aria-label={i18n('icu:close')}
/>
2022-11-18 00:45:19 +00:00
</div>
<div
ref={arrowProps.ref}
style={arrowProps.style}
className={classNames(
'module-sticker-button__tooltip__triangle',
'module-sticker-button__tooltip__triangle--introduction',
`module-sticker-button__tooltip__triangle--${placement}`
)}
/>
</button>
</div>
)}
</Popper>
) : null}
{open && popperRoot
? createPortal(
<Popper placement={position}>
{({ ref, style }) => (
<div className={theme ? themeClassName(theme) : undefined}>
<StickerPicker
ref={ref}
i18n={i18n}
style={style}
packs={installedPacks}
onClose={handleClose}
onClickAddPack={
onClickAddPack ? handleClickAddPack : undefined
}
onPickSticker={handlePickSticker}
2023-03-01 19:00:50 +00:00
onPickTimeSticker={
onPickTimeSticker ? handlePickTimeSticker : undefined
}
2022-11-18 00:45:19 +00:00
recentStickers={recentStickers}
showPickerHint={showPickerHint}
/>
</div>
)}
</Popper>,
popperRoot
)
: null}
</Manager>
);
});