2023-02-06 10:01:03 -08:00
|
|
|
import clsx from 'clsx';
|
2022-11-16 14:40:56 -08:00
|
|
|
import { supportsPassiveEvents } from 'detect-passive-events';
|
|
|
|
import React, { useCallback, useEffect, useRef } from 'react';
|
2023-02-28 09:04:24 -08:00
|
|
|
import { FormattedMessage, defineMessages, useIntl } from 'react-intl';
|
|
|
|
|
|
|
|
import { Text } from 'soapbox/components/ui';
|
2022-11-16 14:40:56 -08:00
|
|
|
|
|
|
|
const messages = defineMessages({
|
|
|
|
emoji: { id: 'icon_button.label', defaultMessage: 'Select icon' },
|
|
|
|
});
|
|
|
|
|
|
|
|
const listenerOptions = supportsPassiveEvents ? { passive: true } : false;
|
|
|
|
|
|
|
|
interface IIconPickerMenu {
|
2023-02-28 09:04:24 -08:00
|
|
|
icons: Record<string, Array<string>>
|
2023-02-15 13:26:27 -08:00
|
|
|
onClose: () => void
|
2023-02-28 09:04:24 -08:00
|
|
|
onPick: (icon: string) => void
|
2023-02-15 13:26:27 -08:00
|
|
|
style?: React.CSSProperties
|
2022-11-16 14:40:56 -08:00
|
|
|
}
|
|
|
|
|
2023-02-28 09:04:24 -08:00
|
|
|
const IconPickerMenu: React.FC<IIconPickerMenu> = ({ icons, onClose, onPick, style }) => {
|
2022-11-16 14:40:56 -08:00
|
|
|
const intl = useIntl();
|
|
|
|
|
|
|
|
const node = useRef<HTMLDivElement | null>(null);
|
|
|
|
|
|
|
|
const handleDocumentClick = useCallback((e: MouseEvent | TouchEvent) => {
|
|
|
|
if (node.current && !node.current.contains(e.target as Node)) {
|
|
|
|
onClose();
|
|
|
|
}
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
document.addEventListener('click', handleDocumentClick, false);
|
|
|
|
document.addEventListener('touchend', handleDocumentClick, listenerOptions);
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
document.removeEventListener('click', handleDocumentClick, false);
|
|
|
|
document.removeEventListener('touchend', handleDocumentClick, listenerOptions as any);
|
|
|
|
|
|
|
|
};
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
const setRef = (c: HTMLDivElement) => {
|
|
|
|
node.current = c;
|
|
|
|
|
|
|
|
if (!c) return;
|
|
|
|
|
|
|
|
// Nice and dirty hack to display the icons
|
|
|
|
c.querySelectorAll('button.emoji-mart-emoji > img').forEach(elem => {
|
|
|
|
const newIcon = document.createElement('span');
|
|
|
|
newIcon.innerHTML = `<i class="fa fa-${(elem.parentNode as any).getAttribute('title')} fa-hack"></i>`;
|
|
|
|
(elem.parentNode as any).replaceChild(newIcon, elem);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2023-02-28 09:04:24 -08:00
|
|
|
const handleClick = (icon: string) => {
|
2022-11-16 14:40:56 -08:00
|
|
|
onClose();
|
2023-02-28 09:04:24 -08:00
|
|
|
onPick(icon);
|
2022-11-16 14:40:56 -08:00
|
|
|
};
|
|
|
|
|
2023-02-28 09:04:24 -08:00
|
|
|
const renderIcon = (icon: string) => {
|
|
|
|
const name = icon.replace('fa fa-', '');
|
|
|
|
|
|
|
|
return (
|
|
|
|
<li key={icon} className='col-span-1 inline-block'>
|
|
|
|
<button
|
|
|
|
className='flex items-center justify-center rounded-full p-1.5 hover:bg-gray-50 dark:hover:bg-primary-800'
|
|
|
|
aria-label={name}
|
|
|
|
title={name}
|
|
|
|
onClick={() => handleClick(name)}
|
|
|
|
>
|
|
|
|
<i className={clsx(icon, 'h-[1.375rem] w-[1.375rem] text-lg leading-[1.15]')} />
|
|
|
|
</button>
|
|
|
|
</li>
|
|
|
|
);
|
2022-11-16 14:40:56 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
const title = intl.formatMessage(messages.emoji);
|
|
|
|
|
|
|
|
return (
|
2023-02-28 09:04:24 -08:00
|
|
|
<div
|
|
|
|
className={clsx('absolute z-[101] -my-0.5')}
|
|
|
|
style={{ transform: 'translateX(calc(-1 * env(safe-area-inset-right)))', ...style }}
|
|
|
|
ref={setRef}
|
|
|
|
>
|
|
|
|
<div className='h-[270px] overflow-x-hidden overflow-y-scroll rounded bg-white p-1.5 text-gray-900 dark:bg-primary-900 dark:text-gray-100' aria-label={title}>
|
|
|
|
<Text className='px-1.5 py-1'><FormattedMessage id='icon_button.icons' defaultMessage='Icons' /></Text>
|
|
|
|
<ul className='grid grid-cols-8'>
|
|
|
|
{Object.values(icons).flat().map(icon => renderIcon(icon))}
|
|
|
|
</ul>
|
|
|
|
</div>
|
2022-11-16 14:40:56 -08:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default IconPickerMenu;
|