import { GroupRoles } from 'pl-api'; import React, { useMemo } from 'react'; import { defineMessages, FormattedMessage, useIntl } from 'react-intl'; import { useHistory, useRouteMatch } from 'react-router-dom'; import { blockAccount } from 'soapbox/actions/accounts'; import { directCompose, mentionCompose, quoteCompose, replyCompose } from 'soapbox/actions/compose'; import { editEvent } from 'soapbox/actions/events'; import { toggleBookmark, toggleDislike, toggleFavourite, togglePin, toggleReblog } from 'soapbox/actions/interactions'; import { openModal } from 'soapbox/actions/modals'; import { deleteStatusModal, toggleStatusSensitivityModal } from 'soapbox/actions/moderation'; import { initMuteModal } from 'soapbox/actions/mutes'; import { initReport, ReportableEntities } from 'soapbox/actions/reports'; import { changeSetting } from 'soapbox/actions/settings'; import { deleteStatus, editStatus, toggleMuteStatus, translateStatus, undoStatusTranslation } from 'soapbox/actions/statuses'; import { deleteFromTimelines } from 'soapbox/actions/timelines'; import { useBlockGroupMember, useGroup, useGroupRelationship } from 'soapbox/api/hooks'; import { useDeleteGroupStatus } from 'soapbox/api/hooks/groups/useDeleteGroupStatus'; import DropdownMenu from 'soapbox/components/dropdown-menu'; import StatusActionButton from 'soapbox/components/status-action-button'; import StatusReactionWrapper from 'soapbox/components/status-reaction-wrapper'; import { HStack } from 'soapbox/components/ui'; import { languages } from 'soapbox/features/preferences'; import { useAppDispatch, useAppSelector, useFeatures, useInstance, useOwnAccount, useSettings, useSoapboxConfig } from 'soapbox/hooks'; import { useChats } from 'soapbox/queries/chats'; import toast from 'soapbox/toast'; import copy from 'soapbox/utils/copy'; import { getReactForStatus, reduceEmoji } from 'soapbox/utils/emoji-reacts'; import GroupPopover from './groups/popover/group-popover'; import type { Menu } from 'soapbox/components/dropdown-menu'; import type { UnauthorizedModalAction } from 'soapbox/features/ui/components/modals/unauthorized-modal'; import type { Account, Group } from 'soapbox/normalizers'; import type { SelectedStatus } from 'soapbox/selectors'; const messages = defineMessages({ adminAccount: { id: 'status.admin_account', defaultMessage: 'Moderate @{name}' }, admin_status: { id: 'status.admin_status', defaultMessage: 'Open this post in the moderation interface' }, block: { id: 'account.block', defaultMessage: 'Block @{name}' }, blocked: { id: 'group.group_mod_block.success', defaultMessage: '@{name} is banned' }, blockAndReport: { id: 'confirmations.block.block_and_report', defaultMessage: 'Block and report' }, blockConfirm: { id: 'confirmations.block.confirm', defaultMessage: 'Block' }, bookmark: { id: 'status.bookmark', defaultMessage: 'Bookmark' }, bookmarkSetFolder: { id: 'status.bookmark_folder', defaultMessage: 'Set bookmark folder' }, bookmarkChangeFolder: { id: 'status.bookmark_folder_change', defaultMessage: 'Change bookmark folder' }, cancel_reblog_private: { id: 'status.cancel_reblog_private', defaultMessage: 'Un-repost' }, cannot_reblog: { id: 'status.cannot_reblog', defaultMessage: 'This post cannot be reposted' }, chat: { id: 'status.chat', defaultMessage: 'Chat with @{name}' }, copy: { id: 'status.copy', defaultMessage: 'Copy link to post' }, deactivateUser: { id: 'admin.users.actions.deactivate_user', defaultMessage: 'Deactivate @{name}' }, delete: { id: 'status.delete', defaultMessage: 'Delete' }, deleteConfirm: { id: 'confirmations.delete.confirm', defaultMessage: 'Delete' }, deleteFromGroupMessage: { id: 'confirmations.delete_from_group.message', defaultMessage: 'Are you sure you want to delete @{name}\'s post?' }, deleteHeading: { id: 'confirmations.delete.heading', defaultMessage: 'Delete post' }, deleteMessage: { id: 'confirmations.delete.message', defaultMessage: 'Are you sure you want to delete this post?' }, deleteStatus: { id: 'admin.statuses.actions.delete_status', defaultMessage: 'Delete post' }, deleteUser: { id: 'admin.users.actions.delete_user', defaultMessage: 'Delete @{name}' }, direct: { id: 'status.direct', defaultMessage: 'Direct message @{name}' }, disfavourite: { id: 'status.disfavourite', defaultMessage: 'Disike' }, edit: { id: 'status.edit', defaultMessage: 'Edit' }, embed: { id: 'status.embed', defaultMessage: 'Embed post' }, external: { id: 'status.external', defaultMessage: 'View post on {domain}' }, favourite: { id: 'status.favourite', defaultMessage: 'Like' }, groupBlockConfirm: { id: 'confirmations.block_from_group.confirm', defaultMessage: 'Ban user' }, groupBlockFromGroupHeading: { id: 'confirmations.block_from_group.heading', defaultMessage: 'Ban from group' }, groupBlockFromGroupMessage: { id: 'confirmations.block_from_group.message', defaultMessage: 'Are you sure you want to ban @{name} from the group?' }, groupModDelete: { id: 'status.group_mod_delete', defaultMessage: 'Delete post from group' }, group_remove_account: { id: 'status.remove_account_from_group', defaultMessage: 'Remove account from group' }, group_remove_post: { id: 'status.remove_post_from_group', defaultMessage: 'Remove post from group' }, markStatusNotSensitive: { id: 'admin.statuses.actions.mark_status_not_sensitive', defaultMessage: 'Mark post not sensitive' }, markStatusSensitive: { id: 'admin.statuses.actions.mark_status_sensitive', defaultMessage: 'Mark post sensitive' }, mention: { id: 'status.mention', defaultMessage: 'Mention @{name}' }, more: { id: 'status.more', defaultMessage: 'More' }, mute: { id: 'account.mute', defaultMessage: 'Mute @{name}' }, muteConversation: { id: 'status.mute_conversation', defaultMessage: 'Mute conversation' }, open: { id: 'status.open', defaultMessage: 'Show post details' }, pin: { id: 'status.pin', defaultMessage: 'Pin on profile' }, quotePost: { id: 'status.quote', defaultMessage: 'Quote post' }, reactionCry: { id: 'status.reactions.cry', defaultMessage: 'Sad' }, reactionHeart: { id: 'status.reactions.heart', defaultMessage: 'Love' }, reactionLaughing: { id: 'status.reactions.laughing', defaultMessage: 'Haha' }, reactionLike: { id: 'status.reactions.like', defaultMessage: 'Like' }, reactionOpenMouth: { id: 'status.reactions.open_mouth', defaultMessage: 'Wow' }, reactionWeary: { id: 'status.reactions.weary', defaultMessage: 'Weary' }, reblog: { id: 'status.reblog', defaultMessage: 'Repost' }, reblog_private: { id: 'status.reblog_private', defaultMessage: 'Repost to original audience' }, redraft: { id: 'status.redraft', defaultMessage: 'Delete & re-draft' }, redraftConfirm: { id: 'confirmations.redraft.confirm', defaultMessage: 'Delete & redraft' }, redraftHeading: { id: 'confirmations.redraft.heading', defaultMessage: 'Delete & redraft' }, redraftMessage: { id: 'confirmations.redraft.message', defaultMessage: 'Are you sure you want to delete this post and re-draft it? Favorites and reposts will be lost, and replies to the original post will be orphaned.' }, replies_disabled_group: { id: 'status.disabled_replies.group_membership', defaultMessage: 'Only group members can reply' }, reply: { id: 'status.reply', defaultMessage: 'Reply' }, replyAll: { id: 'status.reply_all', defaultMessage: 'Reply to thread' }, replyConfirm: { id: 'confirmations.reply.confirm', defaultMessage: 'Reply' }, replyMessage: { id: 'confirmations.reply.message', defaultMessage: 'Replying now will overwrite the message you are currently composing. Are you sure you want to proceed?' }, report: { id: 'status.report', defaultMessage: 'Report @{name}' }, share: { id: 'status.share', defaultMessage: 'Share' }, unbookmark: { id: 'status.unbookmark', defaultMessage: 'Remove bookmark' }, unmuteConversation: { id: 'status.unmute_conversation', defaultMessage: 'Unmute conversation' }, unpin: { id: 'status.unpin', defaultMessage: 'Unpin from profile' }, addKnownLanguage: { id: 'status.add_known_language', defaultMessage: 'Do not auto-translate posts in {language}.' }, translate: { id: 'status.translate', defaultMessage: 'Translate' }, hideTranslation: { id: 'status.hide_translation', defaultMessage: 'Hide translation' }, }); interface IStatusActionBar { status: SelectedStatus; rebloggedBy?: Account; withLabels?: boolean; expandable?: boolean; space?: 'sm' | 'md' | 'lg'; statusActionButtonTheme?: 'default' | 'inverse'; fromBookmarks?: boolean; } const StatusActionBar: React.FC = ({ status, withLabels = false, expandable = true, space = 'sm', statusActionButtonTheme = 'default', fromBookmarks = false, rebloggedBy, }) => { const intl = useIntl(); const history = useHistory(); const dispatch = useAppDispatch(); const match = useRouteMatch<{ groupId: string }>('/groups/:groupId'); const { group } = useGroup((status.group as Group)?.id as string); const deleteGroupStatus = useDeleteGroupStatus(group as Group, status.id); const blockGroupMember = useBlockGroupMember(group as Group, status.account); const { getOrCreateChatByAccountId } = useChats(); const me = useAppSelector(state => state.me); const { groupRelationship } = useGroupRelationship(status.group_id || undefined); const features = useFeatures(); const instance = useInstance(); const { autoTranslate, boostModal, deleteModal, knownLanguages } = useSettings(); const soapboxConfig = useSoapboxConfig(); const autoTranslating = useMemo(() => { const { allow_remote: allowRemote, allow_unauthenticated: allowUnauthenticated, source_languages: sourceLanguages, target_languages: targetLanguages, } = instance.pleroma.metadata.translation; const renderTranslate = (me || allowUnauthenticated) && (allowRemote || status.account.local) && ['public', 'unlisted'].includes(status.visibility) && status.contentHtml.length > 0 && status.language !== null && !knownLanguages.includes(status.language); const supportsLanguages = (!sourceLanguages || sourceLanguages.includes(status.language!)) && (!targetLanguages || targetLanguages.includes(intl.locale)); return autoTranslate && features.translations && renderTranslate && supportsLanguages; }, [me, status, autoTranslate]); const { allowedEmoji } = soapboxConfig; const { account } = useOwnAccount(); const isStaff = account ? account.is_admin || account.is_moderator : false; const isAdmin = account ? account.is_admin : false; if (!status) { return null; } const onOpenUnauthorizedModal = (action?: UnauthorizedModalAction) => { dispatch(openModal('UNAUTHORIZED', { action, ap_id: status.url, })); }; const handleReplyClick: React.MouseEventHandler = (e) => { if (me) { dispatch(replyCompose(status, rebloggedBy)); } else { onOpenUnauthorizedModal('REPLY'); } }; const handleShareClick = () => { navigator.share({ text: status.search_index, url: status.uri, }).catch((e) => { if (e.name !== 'AbortError') console.error(e); }); }; const handleFavouriteClick: React.EventHandler = (e) => { if (me) { dispatch(toggleFavourite(status)); } else { onOpenUnauthorizedModal('FAVOURITE'); } }; const handleDislikeClick: React.EventHandler = (e) => { if (me) { dispatch(toggleDislike(status)); } else { onOpenUnauthorizedModal('DISLIKE'); } }; const handleBookmarkClick: React.EventHandler = (e) => { dispatch(toggleBookmark(status)); }; const handleBookmarkFolderClick = () => { dispatch(openModal('SELECT_BOOKMARK_FOLDER', { statusId: status.id, })); }; const handleReblogClick: React.EventHandler = e => { if (me) { const modalReblog = () => dispatch(toggleReblog(status)); if ((e && e.shiftKey) || !boostModal) { modalReblog(); } else { dispatch(openModal('BOOST', { statusId: status.id, onReblog: modalReblog })); } } else { onOpenUnauthorizedModal('REBLOG'); } }; const handleQuoteClick: React.EventHandler = (e) => { if (me) { dispatch(quoteCompose(status)); } else { onOpenUnauthorizedModal('REBLOG'); } }; const doDeleteStatus = (withRedraft = false) => { dispatch((_) => { if (!deleteModal) { dispatch(deleteStatus(status.id, withRedraft)); } else { dispatch(openModal('CONFIRM', { heading: intl.formatMessage(withRedraft ? messages.redraftHeading : messages.deleteHeading), message: intl.formatMessage(withRedraft ? messages.redraftMessage : messages.deleteMessage), confirm: intl.formatMessage(withRedraft ? messages.redraftConfirm : messages.deleteConfirm), onConfirm: () => dispatch(deleteStatus(status.id, withRedraft)), })); } }); }; const handleDeleteClick: React.EventHandler = (e) => { doDeleteStatus(); }; const handleRedraftClick: React.EventHandler = (e) => { doDeleteStatus(true); }; const handleEditClick: React.EventHandler = () => { if (status.event) dispatch(editEvent(status.id)); else dispatch(editStatus(status.id)); }; const handlePinClick: React.EventHandler = (e) => { dispatch(togglePin(status)); }; const handleMentionClick: React.EventHandler = (e) => { dispatch(mentionCompose(status.account)); }; const handleDirectClick: React.EventHandler = (e) => { dispatch(directCompose(status.account)); }; const handleChatClick: React.EventHandler = (e) => { const account = status.account; getOrCreateChatByAccountId(account.id) .then((chat) => history.push(`/chats/${chat.id}`)) .catch(() => {}); }; const handleMuteClick: React.EventHandler = (e) => { dispatch(initMuteModal(status.account)); }; const handleBlockClick: React.EventHandler = (e) => { const account = status.account; dispatch(openModal('CONFIRM', { heading: , message: @{account.acct} }} />, confirm: intl.formatMessage(messages.blockConfirm), onConfirm: () => dispatch(blockAccount(account.id)), secondary: intl.formatMessage(messages.blockAndReport), onSecondary: () => { dispatch(blockAccount(account.id)); dispatch(initReport(ReportableEntities.STATUS, account, { status })); }, })); }; const handleEmbed = () => { dispatch(openModal('EMBED', { url: status.url, onError: (error: any) => toast.showAlertForError(error), })); }; const handleReport: React.EventHandler = (e) => { dispatch(initReport(ReportableEntities.STATUS, status.account, { status })); }; const handleConversationMuteClick: React.EventHandler = (e) => { dispatch(toggleMuteStatus(status)); }; const handleCopy: React.EventHandler = (e) => { const { uri } = status; copy(uri); }; const onModerate: React.MouseEventHandler = (e) => { const account = status.account; dispatch(openModal('ACCOUNT_MODERATION', { accountId: account.id })); }; const handleDeleteStatus: React.EventHandler = (e) => { dispatch(deleteStatusModal(intl, status.id)); }; const handleToggleStatusSensitivity: React.EventHandler = (e) => { dispatch(toggleStatusSensitivityModal(intl, status.id, status.sensitive)); }; const handleDeleteFromGroup: React.EventHandler = () => { const account = status.account; dispatch(openModal('CONFIRM', { heading: intl.formatMessage(messages.deleteHeading), message: intl.formatMessage(messages.deleteFromGroupMessage, { name: {account.username} }), confirm: intl.formatMessage(messages.deleteConfirm), onConfirm: () => { deleteGroupStatus.mutate(status.id, { onSuccess() { dispatch(deleteFromTimelines(status.id)); }, }); }, })); }; const handleBlockFromGroup = () => { dispatch(openModal('CONFIRM', { heading: intl.formatMessage(messages.groupBlockFromGroupHeading), message: intl.formatMessage(messages.groupBlockFromGroupMessage, { name: status.account.username }), confirm: intl.formatMessage(messages.groupBlockConfirm), onConfirm: () => { blockGroupMember([status.account.id], { onSuccess() { toast.success(intl.formatMessage(messages.blocked, { name: account?.acct })); }, }); }, })); }; const handleIgnoreLanguage = () => { dispatch(changeSetting(['autoTranslate'], [...knownLanguages, status.language], { showAlert: true })); }; const handleTranslate = () => { if (status.translation) { dispatch(undoStatusTranslation(status.id)); } else { dispatch(translateStatus(status.id, intl.locale)); } }; const _makeMenu = (publicStatus: boolean) => { const mutingConversation = status.muted; const ownAccount = status.account.id === me; const username = status.account.username; const account = status.account; const menu: Menu = []; if (expandable) { menu.push({ text: intl.formatMessage(messages.open), icon: require('@tabler/icons/outline/arrows-vertical.svg'), to: `/@${status.account.acct}/posts/${status.id}`, }); } if (publicStatus) { menu.push({ text: intl.formatMessage(messages.copy), action: handleCopy, icon: require('@tabler/icons/outline/clipboard-copy.svg'), }); if (features.embeds && account.local) { menu.push({ text: intl.formatMessage(messages.embed), action: handleEmbed, icon: require('@tabler/icons/outline/share.svg'), }); } } if (!me) { return menu; } const isGroupStatus = typeof status.group === 'object'; if (features.bookmarks) { menu.push({ text: intl.formatMessage(status.bookmarked ? messages.unbookmark : messages.bookmark), action: handleBookmarkClick, icon: status.bookmarked ? require('@tabler/icons/outline/bookmark-off.svg') : require('@tabler/icons/outline/bookmark.svg'), }); } if (features.bookmarkFolders && fromBookmarks) { menu.push({ text: intl.formatMessage(status.bookmark_folder ? messages.bookmarkChangeFolder : messages.bookmarkSetFolder), action: handleBookmarkFolderClick, icon: require('@tabler/icons/outline/folders.svg'), }); } if (features.federating && !account.local) { const { hostname: domain } = new URL(status.uri); menu.push({ text: intl.formatMessage(messages.external, { domain }), icon: require('@tabler/icons/outline/external-link.svg'), href: status.uri, target: '_blank', }); } menu.push(null); menu.push({ text: intl.formatMessage(mutingConversation ? messages.unmuteConversation : messages.muteConversation), action: handleConversationMuteClick, icon: mutingConversation ? require('@tabler/icons/outline/bell.svg') : require('@tabler/icons/outline/bell-off.svg'), }); menu.push(null); if (ownAccount) { if (publicStatus) { menu.push({ text: intl.formatMessage(status.pinned ? messages.unpin : messages.pin), action: handlePinClick, icon: status.pinned ? require('@tabler/icons/outline/pinned-off.svg') : require('@tabler/icons/outline/pin.svg'), }); } else { if (status.visibility === 'private' || status.visibility === 'mutuals_only') { menu.push({ text: intl.formatMessage(status.reblogged ? messages.cancel_reblog_private : messages.reblog_private), action: handleReblogClick, icon: require('@tabler/icons/outline/repeat.svg'), }); } } menu.push({ text: intl.formatMessage(messages.delete), action: handleDeleteClick, icon: require('@tabler/icons/outline/trash.svg'), destructive: true, }); if (features.editStatuses) { menu.push({ text: intl.formatMessage(messages.edit), action: handleEditClick, icon: require('@tabler/icons/outline/edit.svg'), }); } else { menu.push({ text: intl.formatMessage(messages.redraft), action: handleRedraftClick, icon: require('@tabler/icons/outline/edit.svg'), destructive: true, }); } } else { menu.push({ text: intl.formatMessage(messages.mention, { name: username }), action: handleMentionClick, icon: require('@tabler/icons/outline/at.svg'), }); if (status.account.accepts_chat_messages === true) { menu.push({ text: intl.formatMessage(messages.chat, { name: username }), action: handleChatClick, icon: require('@tabler/icons/outline/messages.svg'), }); } else if (features.privacyScopes) { menu.push({ text: intl.formatMessage(messages.direct, { name: username }), action: handleDirectClick, icon: require('@tabler/icons/outline/mail.svg'), }); } menu.push(null); menu.push({ text: intl.formatMessage(messages.mute, { name: username }), action: handleMuteClick, icon: require('@tabler/icons/outline/volume-3.svg'), }); menu.push({ text: intl.formatMessage(messages.block, { name: username }), action: handleBlockClick, icon: require('@tabler/icons/outline/ban.svg'), }); menu.push({ text: intl.formatMessage(messages.report, { name: username }), action: handleReport, icon: require('@tabler/icons/outline/flag.svg'), }); } if (autoTranslating) { if (status.translation) { menu.push({ text: intl.formatMessage(messages.hideTranslation), action: handleTranslate, icon: require('@tabler/icons/outline/language.svg'), }); } else { menu.push({ text: intl.formatMessage(messages.translate), action: handleTranslate, icon: require('@tabler/icons/outline/language.svg'), }); } menu.push({ text: intl.formatMessage(messages.addKnownLanguage, { language: languages[status.language as 'en'] || status.language }), action: handleIgnoreLanguage, icon: require('@tabler/icons/outline/flag.svg'), }); } if (isGroupStatus && !!status.group) { const isGroupOwner = groupRelationship?.role === GroupRoles.OWNER; const isGroupAdmin = groupRelationship?.role === GroupRoles.ADMIN; // const isStatusFromOwner = group.owner.id === account.id; const canBanUser = match?.isExact && (isGroupOwner || isGroupAdmin) && !ownAccount; const canDeleteStatus = !ownAccount && (isGroupOwner || isGroupAdmin); if (canBanUser || canDeleteStatus) { menu.push(null); } if (canBanUser) { menu.push({ text: 'Ban from Group', action: handleBlockFromGroup, icon: require('@tabler/icons/outline/ban.svg'), destructive: true, }); } if (canDeleteStatus) { menu.push({ text: intl.formatMessage(messages.groupModDelete), action: handleDeleteFromGroup, icon: require('@tabler/icons/outline/trash.svg'), destructive: true, }); } } if (isStaff) { menu.push(null); menu.push({ text: intl.formatMessage(messages.adminAccount, { name: username }), action: onModerate, icon: require('@tabler/icons/outline/gavel.svg'), }); if (isAdmin) { menu.push({ text: intl.formatMessage(messages.admin_status), href: `/pleroma/admin/#/statuses/${status.id}/`, icon: require('@tabler/icons/outline/pencil.svg'), }); } menu.push({ text: intl.formatMessage(status.sensitive === false ? messages.markStatusSensitive : messages.markStatusNotSensitive), action: handleToggleStatusSensitivity, icon: require('@tabler/icons/outline/alert-triangle.svg'), }); if (!ownAccount) { menu.push({ text: intl.formatMessage(messages.deleteStatus), action: handleDeleteStatus, icon: require('@tabler/icons/outline/trash.svg'), destructive: true, }); } } return menu; }; const publicStatus = ['public', 'unlisted', 'group'].includes(status.visibility); const replyCount = status.replies_count; const reblogCount = status.reblogs_count; const favouriteCount = status.favourites_count; const emojiReactCount = status.emoji_reactions ? reduceEmoji( status.emoji_reactions, favouriteCount, status.favourited, allowedEmoji, ).reduce((acc, cur) => acc + (cur.count || 0), 0) : undefined; const meEmojiReact = getReactForStatus(status, allowedEmoji); const meEmojiName = meEmojiReact?.name as keyof typeof reactMessages | undefined; const reactMessages = { '👍': messages.reactionLike, '❤️': messages.reactionHeart, '😆': messages.reactionLaughing, '😮': messages.reactionOpenMouth, '😢': messages.reactionCry, '😩': messages.reactionWeary, '': messages.favourite, }; const meEmojiTitle = intl.formatMessage(reactMessages[meEmojiName || ''] || messages.favourite); const menu = _makeMenu(publicStatus); let reblogIcon = require('@tabler/icons/outline/repeat.svg'); let replyTitle; let replyDisabled = false; if (status.visibility === 'direct') { reblogIcon = require('@tabler/icons/outline/mail.svg'); } else if (status.visibility === 'private' || status.visibility === 'mutuals_only') { reblogIcon = require('@tabler/icons/outline/lock.svg'); } if ((status.group as Group)?.membership_required && !groupRelationship?.member) { replyDisabled = true; replyTitle = intl.formatMessage(messages.replies_disabled_group); } const reblogMenu = [{ text: intl.formatMessage(status.reblogged ? messages.cancel_reblog_private : messages.reblog), action: handleReblogClick, icon: require('@tabler/icons/outline/repeat.svg'), }, { text: intl.formatMessage(messages.quotePost), action: handleQuoteClick, icon: require('@tabler/icons/outline/quote.svg'), }]; const reblogButton = ( ); if (!status.in_reply_to_id) { replyTitle = intl.formatMessage(messages.reply); } else { replyTitle = intl.formatMessage(messages.replyAll); } const canShare = ('share' in navigator) && (status.visibility === 'public' || status.visibility === 'group'); const spacing: { [key: string]: React.ComponentProps['space']; } = { 'sm': 2, 'md': 8, 'lg': 0, // using justifyContent instead on the HStack }; return ( e.stopPropagation()} alignItems='center' > {(features.quotePosts && me) ? ( {reblogButton} ) : ( reblogButton )} {features.emojiReacts ? ( ) : ( )} {features.statusDislikes && ( )} {canShare && ( )} ); }; export { StatusActionBar as default };