'use strict'; import { List as ImmutableList, Map as ImmutableMap } from 'immutable'; import React from 'react'; import { defineMessages, FormattedMessage, useIntl } from 'react-intl'; import { Link, useHistory } from 'react-router-dom'; import { blockAccount, followAccount, pinAccount, removeFromFollowers, unblockAccount, unmuteAccount, unpinAccount } from 'soapbox/actions/accounts'; import { verifyUser, unverifyUser, setDonor, removeDonor, promoteToAdmin, promoteToModerator, demoteToUser, suggestUsers, unsuggestUsers } from 'soapbox/actions/admin'; import { launchChat } from 'soapbox/actions/chats'; import { mentionCompose, directCompose } from 'soapbox/actions/compose'; import { blockDomain, unblockDomain } from 'soapbox/actions/domain_blocks'; import { openModal } from 'soapbox/actions/modals'; import { deactivateUserModal } from 'soapbox/actions/moderation'; import { initMuteModal } from 'soapbox/actions/mutes'; import { initReport } from 'soapbox/actions/reports'; import { setSearchAccount } from 'soapbox/actions/search'; import { getSettings } from 'soapbox/actions/settings'; import snackbar from 'soapbox/actions/snackbar'; import Badge from 'soapbox/components/badge'; import StillImage from 'soapbox/components/still_image'; import { HStack, IconButton, Menu, MenuButton, MenuItem, MenuList, MenuLink, MenuDivider, Avatar } from 'soapbox/components/ui'; import SvgIcon from 'soapbox/components/ui/icon/svg-icon'; import MovedNote from 'soapbox/features/account_timeline/components/moved_note'; import ActionButton from 'soapbox/features/ui/components/action-button'; import SubscriptionButton from 'soapbox/features/ui/components/subscription-button'; import { useAppDispatch, useFeatures, useOwnAccount } from 'soapbox/hooks'; import { Account } from 'soapbox/types/entities'; import { isLocal, isRemote, } from 'soapbox/utils/accounts'; import type { Menu as MenuType } from 'soapbox/components/dropdown_menu'; const messages = defineMessages({ edit_profile: { id: 'account.edit_profile', defaultMessage: 'Edit profile' }, linkVerifiedOn: { id: 'account.link_verified_on', defaultMessage: 'Ownership of this link was checked on {date}' }, account_locked: { id: 'account.locked_info', defaultMessage: 'This account privacy status is set to locked. The owner manually reviews who can follow them.' }, mention: { id: 'account.mention', defaultMessage: 'Mention' }, chat: { id: 'account.chat', defaultMessage: 'Chat with @{name}' }, direct: { id: 'account.direct', defaultMessage: 'Direct message @{name}' }, unmute: { id: 'account.unmute', defaultMessage: 'Unmute @{name}' }, block: { id: 'account.block', defaultMessage: 'Block @{name}' }, unblock: { id: 'account.unblock', defaultMessage: 'Unblock @{name}' }, mute: { id: 'account.mute', defaultMessage: 'Mute @{name}' }, report: { id: 'account.report', defaultMessage: 'Report @{name}' }, share: { id: 'account.share', defaultMessage: 'Share @{name}\'s profile' }, media: { id: 'account.media', defaultMessage: 'Media' }, blockDomain: { id: 'account.block_domain', defaultMessage: 'Hide everything from {domain}' }, unblockDomain: { id: 'account.unblock_domain', defaultMessage: 'Unhide {domain}' }, hideReblogs: { id: 'account.hide_reblogs', defaultMessage: 'Hide reposts from @{name}' }, showReblogs: { id: 'account.show_reblogs', defaultMessage: 'Show reposts from @{name}' }, preferences: { id: 'navigation_bar.preferences', defaultMessage: 'Preferences' }, follow_requests: { id: 'navigation_bar.follow_requests', defaultMessage: 'Follow requests' }, blocks: { id: 'navigation_bar.blocks', defaultMessage: 'Blocked users' }, domain_blocks: { id: 'navigation_bar.domain_blocks', defaultMessage: 'Hidden domains' }, mutes: { id: 'navigation_bar.mutes', defaultMessage: 'Muted users' }, endorse: { id: 'account.endorse', defaultMessage: 'Feature on profile' }, unendorse: { id: 'account.unendorse', defaultMessage: 'Don\'t feature on profile' }, removeFromFollowers: { id: 'account.remove_from_followers', defaultMessage: 'Remove this follower' }, admin_account: { id: 'status.admin_account', defaultMessage: 'Open moderation interface for @{name}' }, add_or_remove_from_list: { id: 'account.add_or_remove_from_list', defaultMessage: 'Add or Remove from lists' }, deactivateUser: { id: 'admin.users.actions.deactivate_user', defaultMessage: 'Deactivate @{name}' }, deleteUser: { id: 'admin.users.actions.delete_user', defaultMessage: 'Delete @{name}' }, verifyUser: { id: 'admin.users.actions.verify_user', defaultMessage: 'Verify @{name}' }, unverifyUser: { id: 'admin.users.actions.unverify_user', defaultMessage: 'Unverify @{name}' }, setDonor: { id: 'admin.users.actions.set_donor', defaultMessage: 'Set @{name} as a donor' }, removeDonor: { id: 'admin.users.actions.remove_donor', defaultMessage: 'Remove @{name} as a donor' }, promoteToAdmin: { id: 'admin.users.actions.promote_to_admin', defaultMessage: 'Promote @{name} to an admin' }, promoteToModerator: { id: 'admin.users.actions.promote_to_moderator', defaultMessage: 'Promote @{name} to a moderator' }, demoteToModerator: { id: 'admin.users.actions.demote_to_moderator', defaultMessage: 'Demote @{name} to a moderator' }, demoteToUser: { id: 'admin.users.actions.demote_to_user', defaultMessage: 'Demote @{name} to a regular user' }, suggestUser: { id: 'admin.users.actions.suggest_user', defaultMessage: 'Suggest @{name}' }, unsuggestUser: { id: 'admin.users.actions.unsuggest_user', defaultMessage: 'Unsuggest @{name}' }, search: { id: 'account.search', defaultMessage: 'Search from @{name}' }, unfollowConfirm: { id: 'confirmations.unfollow.confirm', defaultMessage: 'Unfollow' }, blockConfirm: { id: 'confirmations.block.confirm', defaultMessage: 'Block' }, blockDomainConfirm: { id: 'confirmations.domain_block.confirm', defaultMessage: 'Hide entire domain' }, blockAndReport: { id: 'confirmations.block.block_and_report', defaultMessage: 'Block & Report' }, userVerified: { id: 'admin.users.user_verified_message', defaultMessage: '@{acct} was verified' }, userUnverified: { id: 'admin.users.user_unverified_message', defaultMessage: '@{acct} was unverified' }, setDonorSuccess: { id: 'admin.users.set_donor_message', defaultMessage: '@{acct} was set as a donor' }, removeDonorSuccess: { id: 'admin.users.remove_donor_message', defaultMessage: '@{acct} was removed as a donor' }, promotedToAdmin: { id: 'admin.users.actions.promote_to_admin_message', defaultMessage: '@{acct} was promoted to an admin' }, promotedToModerator: { id: 'admin.users.actions.promote_to_moderator_message', defaultMessage: '@{acct} was promoted to a moderator' }, demotedToModerator: { id: 'admin.users.actions.demote_to_moderator_message', defaultMessage: '@{acct} was demoted to a moderator' }, demotedToUser: { id: 'admin.users.actions.demote_to_user_message', defaultMessage: '@{acct} was demoted to a regular user' }, userSuggested: { id: 'admin.users.user_suggested_message', defaultMessage: '@{acct} was suggested' }, userUnsuggested: { id: 'admin.users.user_unsuggested_message', defaultMessage: '@{acct} was unsuggested' }, removeFromFollowersConfirm: { id: 'confirmations.remove_from_followers.confirm', defaultMessage: 'Remove' }, userEndorsed: { id: 'account.endorse.success', defaultMessage: 'You are now featuring @{acct} on your profile' }, userUnendorsed: { id: 'account.unendorse.success', defaultMessage: 'You are no longer featuring @{acct}' }, }); interface IHeader { account?: Account, } const Header: React.FC = ({ account }) => { const intl = useIntl(); const history = useHistory(); const dispatch = useAppDispatch(); const features = useFeatures(); const ownAccount = useOwnAccount(); if (!account) { return (
); } const onBlock = () => { if (account.relationship?.blocking) { dispatch(unblockAccount(account.id)); } else { dispatch(openModal('CONFIRM', { icon: require('@tabler/icons/ban.svg'), 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(account)); }, })); } }; const onMention = () => { dispatch(mentionCompose(account)); }; const onDirect = () => { dispatch(directCompose(account)); }; const onReblogToggle = () => { if (account.relationship?.showing_reblogs) { dispatch(followAccount(account.id, { reblogs: false })); } else { dispatch(followAccount(account.id, { reblogs: true })); } }; const onEndorseToggle = () => { if (account.relationship?.endorsed) { dispatch(unpinAccount(account.id)) .then(() => dispatch(snackbar.success(intl.formatMessage(messages.userUnendorsed, { acct: account.acct })))) .catch(() => { }); } else { dispatch(pinAccount(account.id)) .then(() => dispatch(snackbar.success(intl.formatMessage(messages.userEndorsed, { acct: account.acct })))) .catch(() => { }); } }; const onReport = () => { dispatch(initReport(account)); }; const onMute = () => { if (account.relationship?.muting) { dispatch(unmuteAccount(account.id)); } else { dispatch(initMuteModal(account)); } }; const onBlockDomain = (domain: string) => { dispatch(openModal('CONFIRM', { icon: require('@tabler/icons/ban.svg'), heading: , message: {domain} }} />, confirm: intl.formatMessage(messages.blockDomainConfirm), onConfirm: () => dispatch(blockDomain(domain)), })); }; const onUnblockDomain = (domain: string) => { dispatch(unblockDomain(domain)); }; const onAddToList = () => { dispatch(openModal('LIST_ADDER', { accountId: account.id, })); }; const onChat = () => { dispatch(launchChat(account.id, history)); }; const onDeactivateUser = () => { dispatch(deactivateUserModal(intl, account.id)); }; const onVerifyUser = () => { const message = intl.formatMessage(messages.userVerified, { acct: account.acct }); dispatch(verifyUser(account.id)) .then(() => dispatch(snackbar.success(message))) .catch(() => { }); }; const onUnverifyUser = () => { const message = intl.formatMessage(messages.userUnverified, { acct: account.acct }); dispatch(unverifyUser(account.id)) .then(() => dispatch(snackbar.success(message))) .catch(() => { }); }; const onSetDonor = () => { const message = intl.formatMessage(messages.setDonorSuccess, { acct: account.acct }); dispatch(setDonor(account.id)) .then(() => dispatch(snackbar.success(message))) .catch(() => { }); }; const onRemoveDonor = () => { const message = intl.formatMessage(messages.removeDonorSuccess, { acct: account.acct }); dispatch(removeDonor(account.id)) .then(() => dispatch(snackbar.success(message))) .catch(() => { }); }; const onPromoteToAdmin = () => { const message = intl.formatMessage(messages.promotedToAdmin, { acct: account.acct }); dispatch(promoteToAdmin(account.id)) .then(() => dispatch(snackbar.success(message))) .catch(() => { }); }; const onPromoteToModerator = () => { const messageType = account.admin ? messages.demotedToModerator : messages.promotedToModerator; const message = intl.formatMessage(messageType, { acct: account.acct }); dispatch(promoteToModerator(account.id)) .then(() => dispatch(snackbar.success(message))) .catch(() => { }); }; const onDemoteToUser = () => { const message = intl.formatMessage(messages.demotedToUser, { acct: account.acct }); dispatch(demoteToUser(account.id)) .then(() => dispatch(snackbar.success(message))) .catch(() => { }); }; const onSuggestUser = () => { const message = intl.formatMessage(messages.userSuggested, { acct: account.acct }); dispatch(suggestUsers([account.id])) .then(() => dispatch(snackbar.success(message))) .catch(() => { }); }; const onUnsuggestUser = () => { const message = intl.formatMessage(messages.userUnsuggested, { acct: account.acct }); dispatch(unsuggestUsers([account.id])) .then(() => dispatch(snackbar.success(message))) .catch(() => { }); }; const onRemoveFromFollowers = () => { dispatch((_, getState) => { const unfollowModal = getSettings(getState()).get('unfollowModal'); if (unfollowModal) { dispatch(openModal('CONFIRM', { message: @{account.acct} }} />, confirm: intl.formatMessage(messages.removeFromFollowersConfirm), onConfirm: () => dispatch(removeFromFollowers(account.id)), })); } else { dispatch(removeFromFollowers(account.id)); } }); }; const onSearch = () => { dispatch(setSearchAccount(account.id)); history.push('/search'); }; const onAvatarClick = () => { const avatar_url = account.avatar; const avatar = ImmutableMap({ type: 'image', preview_url: avatar_url, url: avatar_url, description: '', }); dispatch(openModal('MEDIA', { media: ImmutableList.of(avatar), index: 0 })); }; const handleAvatarClick: React.MouseEventHandler = (e) => { if (e.button === 0 && !(e.ctrlKey || e.metaKey)) { e.preventDefault(); onAvatarClick(); } }; const onHeaderClick = () => { const header_url = account.header; const header = ImmutableMap({ type: 'image', preview_url: header_url, url: header_url, description: '', }); dispatch(openModal('MEDIA', { media: ImmutableList.of(header), index: 0 })); }; const handleHeaderClick: React.MouseEventHandler = (e) => { if (e.button === 0 && !(e.ctrlKey || e.metaKey)) { e.preventDefault(); onHeaderClick(); } }; const handleShare = () => { navigator.share({ text: `@${account.acct}`, url: account.url, }).catch((e) => { if (e.name !== 'AbortError') console.error(e); }); }; const makeMenu = () => { const menu: MenuType = []; if (!account || !ownAccount) { return []; } if ('share' in navigator) { menu.push({ text: intl.formatMessage(messages.share, { name: account.username }), action: handleShare, icon: require('@tabler/icons/upload.svg'), }); menu.push(null); } if (account.id === ownAccount?.id) { menu.push({ text: intl.formatMessage(messages.edit_profile), to: '/settings/profile', icon: require('@tabler/icons/user.svg'), }); menu.push({ text: intl.formatMessage(messages.preferences), to: '/settings', icon: require('@tabler/icons/settings.svg'), }); menu.push(null); menu.push({ text: intl.formatMessage(messages.mutes), to: '/mutes', icon: require('@tabler/icons/circle-x.svg'), }); menu.push({ text: intl.formatMessage(messages.blocks), to: '/blocks', icon: require('@tabler/icons/ban.svg'), }); } else { menu.push({ text: intl.formatMessage(messages.mention, { name: account.username }), action: onMention, icon: require('@tabler/icons/at.svg'), }); if (account.getIn(['pleroma', 'accepts_chat_messages']) === true) { menu.push({ text: intl.formatMessage(messages.chat, { name: account.username }), action: onChat, icon: require('@tabler/icons/messages.svg'), }); } else if (features.privacyScopes) { menu.push({ text: intl.formatMessage(messages.direct, { name: account.username }), action: onDirect, icon: require('@tabler/icons/mail.svg'), }); } if (account.relationship?.following) { if (account.relationship?.showing_reblogs) { menu.push({ text: intl.formatMessage(messages.hideReblogs, { name: account.username }), action: onReblogToggle, icon: require('@tabler/icons/repeat.svg'), }); } else { menu.push({ text: intl.formatMessage(messages.showReblogs, { name: account.username }), action: onReblogToggle, icon: require('@tabler/icons/repeat.svg'), }); } if (features.lists) { menu.push({ text: intl.formatMessage(messages.add_or_remove_from_list), action: onAddToList, icon: require('@tabler/icons/list.svg'), }); } if (features.accountEndorsements) { menu.push({ text: intl.formatMessage(account.relationship?.endorsed ? messages.unendorse : messages.endorse), action: onEndorseToggle, icon: require('@tabler/icons/user-check.svg'), }); } menu.push(null); } else if (features.lists && features.unrestrictedLists) { menu.push({ text: intl.formatMessage(messages.add_or_remove_from_list), action: onAddToList, icon: require('@tabler/icons/list.svg'), }); } if (features.searchFromAccount) { menu.push({ text: intl.formatMessage(messages.search, { name: account.username }), action: onSearch, icon: require('@tabler/icons/search.svg'), }); } if (features.removeFromFollowers && account.relationship?.followed_by) { menu.push({ text: intl.formatMessage(messages.removeFromFollowers), action: onRemoveFromFollowers, icon: require('@tabler/icons/user-x.svg'), }); } if (account.relationship?.muting) { menu.push({ text: intl.formatMessage(messages.unmute, { name: account.username }), action: onMute, icon: require('@tabler/icons/circle-x.svg'), }); } else { menu.push({ text: intl.formatMessage(messages.mute, { name: account.username }), action: onMute, icon: require('@tabler/icons/circle-x.svg'), }); } if (account.relationship?.blocking) { menu.push({ text: intl.formatMessage(messages.unblock, { name: account.username }), action: onBlock, icon: require('@tabler/icons/ban.svg'), }); } else { menu.push({ text: intl.formatMessage(messages.block, { name: account.username }), action: onBlock, icon: require('@tabler/icons/ban.svg'), }); } menu.push({ text: intl.formatMessage(messages.report, { name: account.username }), action: onReport, icon: require('@tabler/icons/flag.svg'), }); } if (isRemote(account)) { const domain = account.fqn.split('@')[1]; menu.push(null); if (account.relationship?.domain_blocking) { menu.push({ text: intl.formatMessage(messages.unblockDomain, { domain }), action: () => onUnblockDomain(domain), icon: require('@tabler/icons/ban.svg'), }); } else { menu.push({ text: intl.formatMessage(messages.blockDomain, { domain }), action: () => onBlockDomain(domain), icon: require('@tabler/icons/ban.svg'), }); } } if (ownAccount?.staff) { menu.push(null); if (ownAccount?.admin) { menu.push({ text: intl.formatMessage(messages.admin_account, { name: account.username }), to: `/pleroma/admin/#/users/${account.id}/`, newTab: true, icon: require('@tabler/icons/gavel.svg'), }); } if (account.id !== ownAccount?.id && isLocal(account) && ownAccount.admin) { if (account.admin) { menu.push({ text: intl.formatMessage(messages.demoteToModerator, { name: account.username }), action: onPromoteToModerator, icon: require('@tabler/icons/arrow-up-circle.svg'), }); menu.push({ text: intl.formatMessage(messages.demoteToUser, { name: account.username }), action: onDemoteToUser, icon: require('@tabler/icons/arrow-down-circle.svg'), }); } else if (account.moderator) { menu.push({ text: intl.formatMessage(messages.promoteToAdmin, { name: account.username }), action: onPromoteToAdmin, icon: require('@tabler/icons/arrow-up-circle.svg'), }); menu.push({ text: intl.formatMessage(messages.demoteToUser, { name: account.username }), action: onDemoteToUser, icon: require('@tabler/icons/arrow-down-circle.svg'), }); } else { menu.push({ text: intl.formatMessage(messages.promoteToAdmin, { name: account.username }), action: onPromoteToAdmin, icon: require('@tabler/icons/arrow-up-circle.svg'), }); menu.push({ text: intl.formatMessage(messages.promoteToModerator, { name: account.username }), action: onPromoteToModerator, icon: require('@tabler/icons/arrow-up-circle.svg'), }); } } if (account.verified) { menu.push({ text: intl.formatMessage(messages.unverifyUser, { name: account.username }), action: onUnverifyUser, icon: require('@tabler/icons/check.svg'), }); } else { menu.push({ text: intl.formatMessage(messages.verifyUser, { name: account.username }), action: onVerifyUser, icon: require('@tabler/icons/check.svg'), }); } if (account.donor) { menu.push({ text: intl.formatMessage(messages.removeDonor, { name: account.username }), action: onRemoveDonor, icon: require('@tabler/icons/coin.svg'), }); } else { menu.push({ text: intl.formatMessage(messages.setDonor, { name: account.username }), action: onSetDonor, icon: require('@tabler/icons/coin.svg'), }); } if (features.suggestionsV2 && ownAccount.admin) { if (account.getIn(['pleroma', 'is_suggested'])) { menu.push({ text: intl.formatMessage(messages.unsuggestUser, { name: account.username }), action: onUnsuggestUser, icon: require('@tabler/icons/user-x.svg'), }); } else { menu.push({ text: intl.formatMessage(messages.suggestUser, { name: account.username }), action: onSuggestUser, icon: require('@tabler/icons/user-check.svg'), }); } } if (account.id !== ownAccount?.id) { menu.push({ text: intl.formatMessage(messages.deactivateUser, { name: account.username }), action: onDeactivateUser, icon: require('@tabler/icons/user-off.svg'), }); menu.push({ text: intl.formatMessage(messages.deleteUser, { name: account.username }), icon: require('@tabler/icons/user-minus.svg'), }); } } return menu; }; const makeInfo = () => { const info: React.ReactNode[] = []; if (!account || !ownAccount) return info; if (ownAccount?.id !== account.id && account.relationship?.followed_by) { info.push( } />, ); } else if (ownAccount?.id !== account.id && account.relationship?.blocking) { info.push( } />, ); } if (ownAccount?.id !== account.id && account.relationship?.muting) { info.push( } />, ); } else if (ownAccount?.id !== account.id && account.relationship?.domain_blocking) { info.push( } />, ); } return info; }; // const renderMessageButton = () => { // if (!ownAccount || !account || account.id === ownAccount?.id) { // return null; // } // const canChat = account.getIn(['pleroma', 'accepts_chat_messages']) === true; // if (canChat) { // return ( // // ); // } else { // return ( // // ); // } // }; const renderShareButton = () => { const canShare = 'share' in navigator; if (!(account && ownAccount?.id && account.id === ownAccount?.id && canShare)) { return null; } return ( ); }; const info = makeInfo(); const menu = makeMenu(); return (
{(account.moved && typeof account.moved === 'object') && ( )}
{account.header && ( )}
{info}
{ownAccount && ( {menu.map((menuItem, idx) => { if (typeof menuItem?.text === 'undefined') { return ; } else { const Comp = (menuItem.action ? MenuItem : MenuLink) as any; const itemProps = menuItem.action ? { onSelect: menuItem.action } : { to: menuItem.to, as: Link, target: menuItem.newTab ? '_blank' : '_self' }; return (
{menuItem.icon && ( )}
{menuItem.text}
); } })}
)} {renderShareButton()} {/* {renderMessageButton()} */}
); }; export default Header;