import React, { useCallback } from 'react'; import { HotKeys } from 'react-hotkeys'; import { defineMessages, useIntl, FormattedMessage, IntlShape, MessageDescriptor, defineMessage } from 'react-intl'; import { Link, useHistory } from 'react-router-dom'; import { mentionCompose } from 'soapbox/actions/compose'; import { reblog, favourite, unreblog, unfavourite } from 'soapbox/actions/interactions'; import { openModal } from 'soapbox/actions/modals'; import { getSettings } from 'soapbox/actions/settings'; import { hideStatus, revealStatus } from 'soapbox/actions/statuses'; import Icon from 'soapbox/components/icon'; import { HStack, Text, Emoji } from 'soapbox/components/ui'; import AccountContainer from 'soapbox/containers/account_container'; import StatusContainer from 'soapbox/containers/status_container'; import { useAppDispatch, useAppSelector } from 'soapbox/hooks'; import { makeGetNotification } from 'soapbox/selectors'; import { NotificationType, validType } from 'soapbox/utils/notification'; import type { ScrollPosition } from 'soapbox/components/status'; import type { Account, Status as StatusEntity, Notification as NotificationEntity } from 'soapbox/types/entities'; const notificationForScreenReader = (intl: IntlShape, message: string, timestamp: Date) => { const output = [message]; output.push(intl.formatDate(timestamp, { hour: '2-digit', minute: '2-digit', month: 'short', day: 'numeric' })); return output.join(', '); }; const buildLink = (account: Account): JSX.Element => ( ); const icons: Record = { follow: require('@tabler/icons/user-plus.svg'), follow_request: require('@tabler/icons/user-plus.svg'), mention: require('@tabler/icons/at.svg'), favourite: require('@tabler/icons/heart.svg'), reblog: require('@tabler/icons/repeat.svg'), status: require('@tabler/icons/bell-ringing.svg'), poll: require('@tabler/icons/chart-bar.svg'), move: require('@tabler/icons/briefcase.svg'), 'pleroma:chat_mention': require('@tabler/icons/messages.svg'), 'pleroma:emoji_reaction': require('@tabler/icons/mood-happy.svg'), user_approved: require('@tabler/icons/user-plus.svg'), update: require('@tabler/icons/pencil.svg'), }; const nameMessage = defineMessage({ id: 'notification.name', defaultMessage: '{link}{others}', }); const messages: Record = defineMessages({ follow: { id: 'notification.follow', defaultMessage: '{name} followed you', }, follow_request: { id: 'notification.follow_request', defaultMessage: '{name} has requested to follow you', }, mention: { id: 'notification.mentioned', defaultMessage: '{name} mentioned you', }, favourite: { id: 'notification.favourite', defaultMessage: '{name} liked your post', }, reblog: { id: 'notification.reblog', defaultMessage: '{name} reposted your post', }, status: { id: 'notification.status', defaultMessage: '{name} just posted', }, poll: { id: 'notification.poll', defaultMessage: 'A poll you have voted in has ended', }, move: { id: 'notification.move', defaultMessage: '{name} moved to {targetName}', }, 'pleroma:chat_mention': { id: 'notification.pleroma:chat_mention', defaultMessage: '{name} sent you a message', }, 'pleroma:emoji_reaction': { id: 'notification.pleroma:emoji_reaction', defaultMessage: '{name} reacted to your post', }, user_approved: { id: 'notification.user_approved', defaultMessage: 'Welcome to {instance}!', }, update: { id: 'notification.update', defaultMessage: '{name} edited a post you interacted with', }, }); const buildMessage = ( intl: IntlShape, type: NotificationType, account: Account, totalCount: number | null, targetName: string, instanceTitle: string, ): React.ReactNode => { const link = buildLink(account); const name = intl.formatMessage(nameMessage, { link, others: totalCount && totalCount > 0 ? ( ) : '', }); return intl.formatMessage(messages[type], { name, targetName, instance: instanceTitle, }); }; interface INotificaton { hidden?: boolean, notification: NotificationEntity, onMoveUp?: (notificationId: string) => void, onMoveDown?: (notificationId: string) => void, onReblog?: (status: StatusEntity, e?: KeyboardEvent) => void, getScrollPosition?: () => ScrollPosition | undefined, updateScrollBottom?: (bottom: number) => void, } const Notification: React.FC = (props) => { const { hidden = false, onMoveUp, onMoveDown } = props; const dispatch = useAppDispatch(); const getNotification = useCallback(makeGetNotification(), []); const notification = useAppSelector((state) => getNotification(state, props.notification)); const history = useHistory(); const intl = useIntl(); const instance = useAppSelector((state) => state.instance); const type = notification.type; const { account, status } = notification; const getHandlers = () => ({ reply: handleMention, favourite: handleHotkeyFavourite, boost: handleHotkeyBoost, mention: handleMention, open: handleOpen, openProfile: handleOpenProfile, moveUp: handleMoveUp, moveDown: handleMoveDown, toggleHidden: handleHotkeyToggleHidden, }); const handleOpen = () => { if (status && typeof status === 'object' && account && typeof account === 'object') { history.push(`/@${account.acct}/posts/${status.id}`); } else { handleOpenProfile(); } }; const handleOpenProfile = () => { if (account && typeof account === 'object') { history.push(`/@${account.acct}`); } }; const handleMention = useCallback((e?: KeyboardEvent) => { e?.preventDefault(); if (account && typeof account === 'object') { dispatch(mentionCompose(account)); } }, [account]); const handleHotkeyFavourite = useCallback((e?: KeyboardEvent) => { if (status && typeof status === 'object') { if (status.favourited) { dispatch(unfavourite(status)); } else { dispatch(favourite(status)); } } }, [status]); const handleHotkeyBoost = useCallback((e?: KeyboardEvent) => { if (status && typeof status === 'object') { dispatch((_, getState) => { const boostModal = getSettings(getState()).get('boostModal'); if (status.reblogged) { dispatch(unreblog(status)); } else { if (e?.shiftKey || !boostModal) { dispatch(reblog(status)); } else { dispatch(openModal('BOOST', { status, onReblog: (status: StatusEntity) => { dispatch(reblog(status)); } })); } } }); } }, [status]); const handleHotkeyToggleHidden = useCallback((e?: KeyboardEvent) => { if (status && typeof status === 'object') { if (status.hidden) { dispatch(revealStatus(status.id)); } else { dispatch(hideStatus(status.id)); } } }, [status]); const handleMoveUp = () => { if (onMoveUp) { onMoveUp(notification.id); } }; const handleMoveDown = () => { if (onMoveDown) { onMoveDown(notification.id); } }; const renderIcon = (): React.ReactNode => { if (type === 'pleroma:emoji_reaction' && notification.emoji) { return ( ); } else if (validType(type)) { return ( ); } else { return null; } }; const renderContent = () => { switch (type as NotificationType) { case 'follow': case 'user_approved': return account && typeof account === 'object' ? (