2017-04-21 14:05:35 -04:00
import PropTypes from 'prop-types' ;
2023-05-23 11:15:17 -04:00
2016-11-23 17:34:12 -05:00
import { defineMessages , injectIntl , FormattedMessage } from 'react-intl' ;
2023-05-23 11:15:17 -04:00
2017-11-18 13:39:02 -05:00
import classNames from 'classnames' ;
2023-05-23 11:15:17 -04:00
import { Helmet } from 'react-helmet' ;
2023-10-19 13:44:55 -04:00
import { NavLink , withRouter } from 'react-router-dom' ;
2023-05-23 11:15:17 -04:00
import ImmutablePropTypes from 'react-immutable-proptypes' ;
import ImmutablePureComponent from 'react-immutable-pure-component' ;
2024-01-16 05:27:26 -05:00
import CheckIcon from '@/material-icons/400-24px/check.svg?react' ;
import LockIcon from '@/material-icons/400-24px/lock.svg?react' ;
import MoreHorizIcon from '@/material-icons/400-24px/more_horiz.svg?react' ;
import NotificationsIcon from '@/material-icons/400-24px/notifications.svg?react' ;
import NotificationsActiveIcon from '@/material-icons/400-24px/notifications_active-fill.svg?react' ;
import ShareIcon from '@/material-icons/400-24px/share.svg?react' ;
2023-05-08 21:11:56 -04:00
import { Avatar } from 'mastodon/components/avatar' ;
2023-08-02 11:24:32 -04:00
import { Badge , AutomatedBadge , GroupBadge } from 'mastodon/components/badge' ;
2023-10-23 03:43:00 -04:00
import { Button } from 'mastodon/components/button' ;
2023-11-16 05:23:14 -05:00
import { CopyIconButton } from 'mastodon/components/copy_icon_button' ;
2023-07-08 05:11:22 -04:00
import { FollowersCounter , FollowingCounter , StatusesCounter } from 'mastodon/components/counters' ;
2023-05-23 11:15:17 -04:00
import { Icon } from 'mastodon/components/icon' ;
import { IconButton } from 'mastodon/components/icon_button' ;
2024-03-11 10:33:48 -04:00
import { LoadingIndicator } from 'mastodon/components/loading_indicator' ;
2023-07-08 05:11:58 -04:00
import { ShortNumber } from 'mastodon/components/short_number' ;
2019-03-25 19:36:25 -04:00
import DropdownMenuContainer from 'mastodon/containers/dropdown_menu_container' ;
2024-05-19 13:07:32 -04:00
import { identityContextPropShape , withIdentity } from 'mastodon/identity_context' ;
2024-03-15 09:08:38 -04:00
import { autoPlayGif , me , domain as localDomain } from 'mastodon/initial_state' ;
2023-05-23 11:15:17 -04:00
import { PERMISSION _MANAGE _USERS , PERMISSION _MANAGE _FEDERATION } from 'mastodon/permissions' ;
2023-10-19 13:44:55 -04:00
import { WithRouterPropTypes } from 'mastodon/utils/react_router' ;
2023-05-23 11:15:17 -04:00
2020-06-30 13:19:50 -04:00
import AccountNoteContainer from '../containers/account_note_container' ;
2022-12-15 12:50:11 -05:00
import FollowRequestNoteContainer from '../containers/follow_request_note_container' ;
2016-11-23 17:34:12 -05:00
2024-03-15 09:08:38 -04:00
import { DomainPill } from './domain_pill' ;
2016-11-23 17:34:12 -05:00
const messages = defineMessages ( {
unfollow : { id : 'account.unfollow' , defaultMessage : 'Unfollow' } ,
follow : { id : 'account.follow' , defaultMessage : 'Follow' } ,
2023-12-21 04:44:09 -05:00
followBack : { id : 'account.follow_back' , defaultMessage : 'Follow back' } ,
mutual : { id : 'account.mutual' , defaultMessage : 'Mutual' } ,
2022-10-18 18:12:55 -04:00
cancel _follow _request : { id : 'account.cancel_follow_request' , defaultMessage : 'Withdraw follow request' } ,
2017-09-02 14:44:41 -04:00
requested : { id : 'account.requested' , defaultMessage : 'Awaiting approval. Click to cancel follow request' } ,
2018-03-04 23:09:35 -05:00
unblock : { id : 'account.unblock' , defaultMessage : 'Unblock @{name}' } ,
2018-05-30 12:41:47 -04:00
edit _profile : { id : 'account.edit_profile' , defaultMessage : 'Edit profile' } ,
2018-09-18 10:45:58 -04:00
linkVerifiedOn : { id : 'account.link_verified_on' , defaultMessage : 'Ownership of this link was checked on {date}' } ,
2018-12-01 08:25:15 -05:00
account _locked : { id : 'account.locked_info' , defaultMessage : 'This account privacy status is set to locked. The owner manually reviews who can follow them.' } ,
2019-03-25 19:36:25 -04:00
mention : { id : 'account.mention' , defaultMessage : 'Mention @{name}' } ,
2023-03-30 09:16:20 -04:00
direct : { id : 'account.direct' , defaultMessage : 'Privately mention @{name}' } ,
2019-03-25 19:36:25 -04:00
unmute : { id : 'account.unmute' , defaultMessage : 'Unmute @{name}' } ,
block : { id : 'account.block' , defaultMessage : 'Block @{name}' } ,
mute : { id : 'account.mute' , defaultMessage : 'Mute @{name}' } ,
report : { id : 'account.report' , defaultMessage : 'Report @{name}' } ,
share : { id : 'account.share' , defaultMessage : 'Share @{name}\'s profile' } ,
2023-11-16 05:23:14 -05:00
copy : { id : 'account.copy' , defaultMessage : 'Copy link to profile' } ,
2019-03-25 19:36:25 -04:00
media : { id : 'account.media' , defaultMessage : 'Media' } ,
2020-03-05 17:20:49 -05:00
blockDomain : { id : 'account.block_domain' , defaultMessage : 'Block domain {domain}' } ,
unblockDomain : { id : 'account.unblock_domain' , defaultMessage : 'Unblock domain {domain}' } ,
2019-03-25 19:36:25 -04:00
hideReblogs : { id : 'account.hide_reblogs' , defaultMessage : 'Hide boosts from @{name}' } ,
showReblogs : { id : 'account.show_reblogs' , defaultMessage : 'Show boosts from @{name}' } ,
2020-09-18 11:26:45 -04:00
enableNotifications : { id : 'account.enable_notifications' , defaultMessage : 'Notify me when @{name} posts' } ,
disableNotifications : { id : 'account.disable_notifications' , defaultMessage : 'Stop notifying me when @{name} posts' } ,
2022-04-28 18:24:31 -04:00
pins : { id : 'navigation_bar.pins' , defaultMessage : 'Pinned posts' } ,
2019-03-25 19:36:25 -04:00
preferences : { id : 'navigation_bar.preferences' , defaultMessage : 'Preferences' } ,
follow _requests : { id : 'navigation_bar.follow_requests' , defaultMessage : 'Follow requests' } ,
2023-07-21 13:09:13 -04:00
favourites : { id : 'navigation_bar.favourites' , defaultMessage : 'Favorites' } ,
2019-03-25 19:36:25 -04:00
lists : { id : 'navigation_bar.lists' , defaultMessage : 'Lists' } ,
2023-01-18 10:44:33 -05:00
followed _tags : { id : 'navigation_bar.followed_tags' , defaultMessage : 'Followed hashtags' } ,
2019-03-25 19:36:25 -04:00
blocks : { id : 'navigation_bar.blocks' , defaultMessage : 'Blocked users' } ,
2020-03-09 04:13:21 -04:00
domain _blocks : { id : 'navigation_bar.domain_blocks' , defaultMessage : 'Blocked domains' } ,
2019-03-25 19:36:25 -04:00
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' } ,
add _or _remove _from _list : { id : 'account.add_or_remove_from_list' , defaultMessage : 'Add or Remove from lists' } ,
admin _account : { id : 'status.admin_account' , defaultMessage : 'Open moderation interface for @{name}' } ,
2023-01-05 08:03:46 -05:00
admin _domain : { id : 'status.admin_domain' , defaultMessage : 'Open moderation interface for {domain}' } ,
2022-09-20 17:51:21 -04:00
languages : { id : 'account.languages' , defaultMessage : 'Change subscribed languages' } ,
2022-11-10 02:49:35 -05:00
openOriginalPage : { id : 'account.open_original_page' , defaultMessage : 'Open original page' } ,
2016-11-23 17:34:12 -05:00
} ) ;
2016-09-18 12:18:46 -04:00
2022-09-28 22:39:33 -04:00
const titleFromAccount = account => {
const displayName = account . get ( 'display_name' ) ;
2024-03-15 09:08:38 -04:00
const acct = account . get ( 'acct' ) === account . get ( 'username' ) ? ` ${ account . get ( 'username' ) } @ ${ localDomain } ` : account . get ( 'acct' ) ;
2022-09-28 22:39:33 -04:00
const prefix = displayName . trim ( ) . length === 0 ? account . get ( 'username' ) : displayName ;
return ` ${ prefix } (@ ${ acct } ) ` ;
} ;
2023-12-21 04:44:09 -05:00
const messageForFollowButton = relationship => {
2023-12-21 15:19:18 -05:00
if ( ! relationship ) return messages . follow ;
2023-12-21 04:44:09 -05:00
if ( relationship . get ( 'following' ) && relationship . get ( 'followed_by' ) ) {
return messages . mutual ;
} else if ( ! relationship . get ( 'following' ) && relationship . get ( 'followed_by' ) ) {
return messages . followBack ;
2024-07-01 14:10:22 -04:00
} else if ( relationship . get ( 'following' ) || relationship . get ( 'requested' ) ) {
2023-12-21 04:44:09 -05:00
return messages . unfollow ;
} else {
return messages . follow ;
}
} ;
2018-09-18 10:45:58 -04:00
const dateFormatOptions = {
month : 'short' ,
day : 'numeric' ,
year : 'numeric' ,
hour : '2-digit' ,
minute : '2-digit' ,
} ;
2018-09-14 11:59:48 -04:00
class Header extends ImmutablePureComponent {
2016-09-18 12:18:46 -04:00
2017-05-12 08:44:10 -04:00
static propTypes = {
2024-05-19 13:07:32 -04:00
identity : identityContextPropShape ,
2023-11-03 11:00:03 -04:00
account : ImmutablePropTypes . record ,
2019-03-28 13:01:09 -04:00
identity _props : ImmutablePropTypes . list ,
2017-05-12 08:44:10 -04:00
onFollow : PropTypes . func . isRequired ,
2018-03-04 23:09:35 -05:00
onBlock : PropTypes . func . isRequired ,
2020-08-02 05:20:02 -04:00
onMention : PropTypes . func . isRequired ,
onDirect : PropTypes . func . isRequired ,
onReblogToggle : PropTypes . func . isRequired ,
2020-09-18 11:26:45 -04:00
onNotifyToggle : PropTypes . func . isRequired ,
onReport : PropTypes . func . isRequired ,
2020-08-02 05:20:02 -04:00
onMute : PropTypes . func . isRequired ,
onBlockDomain : PropTypes . func . isRequired ,
onUnblockDomain : PropTypes . func . isRequired ,
onEndorseToggle : PropTypes . func . isRequired ,
onAddToList : PropTypes . func . isRequired ,
onEditAccountNote : PropTypes . func . isRequired ,
2022-09-20 17:51:21 -04:00
onChangeLanguages : PropTypes . func . isRequired ,
2022-10-07 04:14:31 -04:00
onInteractionModal : PropTypes . func . isRequired ,
2022-11-10 02:49:35 -05:00
onOpenAvatar : PropTypes . func . isRequired ,
2023-04-25 00:33:21 -04:00
onOpenURL : PropTypes . func . isRequired ,
2017-05-12 08:44:10 -04:00
intl : PropTypes . object . isRequired ,
2019-03-25 19:36:25 -04:00
domain : PropTypes . string . isRequired ,
2022-05-10 03:44:35 -04:00
hidden : PropTypes . bool ,
2023-10-19 13:44:55 -04:00
... WithRouterPropTypes ,
} ;
2023-04-25 00:33:21 -04:00
setRef = c => {
this . node = c ;
} ;
2018-05-30 12:41:47 -04:00
openEditProfile = ( ) => {
window . open ( '/settings/profile' , '_blank' ) ;
2023-01-29 19:45:35 -05:00
} ;
2018-05-30 12:41:47 -04:00
2019-03-25 19:36:25 -04:00
isStatusesPageActive = ( match , location ) => {
if ( ! match ) {
return false ;
}
return ! location . pathname . match ( /\/(followers|following)\/?$/ ) ;
2023-01-29 19:45:35 -05:00
} ;
2019-03-25 19:36:25 -04:00
2021-01-31 15:25:31 -05:00
handleMouseEnter = ( { currentTarget } ) => {
if ( autoPlayGif ) {
2019-07-21 12:10:40 -04:00
return ;
}
2021-01-31 15:25:31 -05:00
const emojis = currentTarget . querySelectorAll ( '.custom-emoji' ) ;
2019-07-21 12:10:40 -04:00
for ( var i = 0 ; i < emojis . length ; i ++ ) {
let emoji = emojis [ i ] ;
2021-01-31 15:25:31 -05:00
emoji . src = emoji . getAttribute ( 'data-original' ) ;
2019-07-21 12:10:40 -04:00
}
2023-01-29 19:45:35 -05:00
} ;
2019-07-21 12:10:40 -04:00
2021-01-31 15:25:31 -05:00
handleMouseLeave = ( { currentTarget } ) => {
if ( autoPlayGif ) {
return ;
}
2019-07-21 12:10:40 -04:00
2021-01-31 15:25:31 -05:00
const emojis = currentTarget . querySelectorAll ( '.custom-emoji' ) ;
2019-07-21 12:10:40 -04:00
2021-01-31 15:25:31 -05:00
for ( var i = 0 ; i < emojis . length ; i ++ ) {
let emoji = emojis [ i ] ;
emoji . src = emoji . getAttribute ( 'data-static' ) ;
}
2023-01-29 19:45:35 -05:00
} ;
2019-07-21 12:10:40 -04:00
2022-11-10 02:49:35 -05:00
handleAvatarClick = e => {
if ( e . button === 0 && ! ( e . ctrlKey || e . metaKey ) ) {
e . preventDefault ( ) ;
this . props . onOpenAvatar ( ) ;
}
2023-01-29 19:45:35 -05:00
} ;
2022-11-10 02:49:35 -05:00
2022-11-27 14:42:17 -05:00
handleShare = ( ) => {
const { account } = this . props ;
navigator . share ( {
url : account . get ( 'url' ) ,
} ) . catch ( ( e ) => {
if ( e . name !== 'AbortError' ) console . error ( e ) ;
} ) ;
2023-01-29 19:45:35 -05:00
} ;
2022-11-27 14:42:17 -05:00
2023-04-25 00:33:21 -04:00
handleHashtagClick = e => {
2023-10-19 13:44:55 -04:00
const { history } = this . props ;
2023-04-25 00:33:21 -04:00
const value = e . currentTarget . textContent . replace ( /^#/ , '' ) ;
2023-10-19 13:44:55 -04:00
if ( history && e . button === 0 && ! ( e . ctrlKey || e . metaKey ) ) {
2023-04-25 00:33:21 -04:00
e . preventDefault ( ) ;
2023-10-19 13:44:55 -04:00
history . push ( ` /tags/ ${ value } ` ) ;
2023-04-25 00:33:21 -04:00
}
} ;
handleMentionClick = e => {
2023-10-19 13:44:55 -04:00
const { history , onOpenURL } = this . props ;
2023-04-25 00:33:21 -04:00
2023-10-19 13:44:55 -04:00
if ( history && e . button === 0 && ! ( e . ctrlKey || e . metaKey ) ) {
2023-04-25 00:33:21 -04:00
e . preventDefault ( ) ;
const link = e . currentTarget ;
2023-10-19 13:44:55 -04:00
onOpenURL ( link . href , history , ( ) => {
2023-04-25 00:33:21 -04:00
window . location = link . href ;
} ) ;
}
} ;
_attachLinkEvents ( ) {
const node = this . node ;
if ( ! node ) {
return ;
}
const links = node . querySelectorAll ( 'a' ) ;
let link ;
for ( var i = 0 ; i < links . length ; ++ i ) {
link = links [ i ] ;
if ( link . textContent [ 0 ] === '#' || ( link . previousSibling && link . previousSibling . textContent && link . previousSibling . textContent [ link . previousSibling . textContent . length - 1 ] === '#' ) ) {
link . addEventListener ( 'click' , this . handleHashtagClick , false ) ;
} else if ( link . classList . contains ( 'mention' ) ) {
link . addEventListener ( 'click' , this . handleMentionClick , false ) ;
}
}
}
componentDidMount ( ) {
this . _attachLinkEvents ( ) ;
}
componentDidUpdate ( ) {
this . _attachLinkEvents ( ) ;
}
2016-09-18 12:18:46 -04:00
render ( ) {
2024-03-15 09:08:38 -04:00
const { account , hidden , intl } = this . props ;
2024-05-19 13:07:32 -04:00
const { signedIn , permissions } = this . props . identity ;
2016-09-18 12:18:46 -04:00
2017-02-20 18:10:49 -05:00
if ( ! account ) {
return null ;
}
2022-11-10 02:49:35 -05:00
const suspended = account . get ( 'suspended' ) ;
const isRemote = account . get ( 'acct' ) !== account . get ( 'username' ) ;
const remoteDomain = isRemote ? account . get ( 'acct' ) . split ( '@' ) [ 1 ] : null ;
2020-09-09 18:07:19 -04:00
2023-11-16 05:23:14 -05:00
let actionBtn , bellBtn , lockedIcon , shareBtn ;
let info = [ ] ;
let menu = [ ] ;
2016-10-06 16:07:32 -04:00
2023-12-21 04:44:09 -05:00
if ( me !== account . get ( 'id' ) && account . getIn ( [ 'relationship' , 'blocking' ] ) ) {
2019-03-29 19:43:29 -04:00
info . push ( < span key = 'blocked' className = 'relationship-tag' > < FormattedMessage id = 'account.blocked' defaultMessage = 'Blocked' / > < / span > ) ;
2018-03-04 23:09:35 -05:00
}
if ( me !== account . get ( 'id' ) && account . getIn ( [ 'relationship' , 'muting' ] ) ) {
2019-03-29 19:43:29 -04:00
info . push ( < span key = 'muted' className = 'relationship-tag' > < FormattedMessage id = 'account.muted' defaultMessage = 'Muted' / > < / span > ) ;
2018-03-05 10:45:36 -05:00
} else if ( me !== account . get ( 'id' ) && account . getIn ( [ 'relationship' , 'domain_blocking' ] ) ) {
2020-03-09 04:13:21 -04:00
info . push ( < span key = 'domain_blocked' className = 'relationship-tag' > < FormattedMessage id = 'account.domain_blocked' defaultMessage = 'Domain blocked' / > < / span > ) ;
2016-10-09 16:19:15 -04:00
}
2020-12-26 17:50:34 -05:00
if ( account . getIn ( [ 'relationship' , 'requested' ] ) || account . getIn ( [ 'relationship' , 'following' ] ) ) {
2023-10-26 19:37:58 -04:00
bellBtn = < IconButton icon = { account . getIn ( [ 'relationship' , 'notifying' ] ) ? 'bell' : 'bell-o' } iconComponent = { account . getIn ( [ 'relationship' , 'notifying' ] ) ? NotificationsActiveIcon : NotificationsIcon } active = { account . getIn ( [ 'relationship' , 'notifying' ] ) } title = { intl . formatMessage ( account . getIn ( [ 'relationship' , 'notifying' ] ) ? messages . disableNotifications : messages . enableNotifications , { name : account . get ( 'username' ) } ) } onClick = { this . props . onNotifyToggle } / > ;
2020-12-26 17:50:34 -05:00
}
2023-11-16 05:23:14 -05:00
if ( 'share' in navigator ) {
shareBtn = < IconButton className = 'optional' iconComponent = { ShareIcon } title = { intl . formatMessage ( messages . share , { name : account . get ( 'username' ) } ) } onClick = { this . handleShare } / > ;
} else {
shareBtn = < CopyIconButton className = 'optional' title = { intl . formatMessage ( messages . copy ) } value = { account . get ( 'url' ) } / > ;
}
2016-11-23 17:34:12 -05:00
if ( me !== account . get ( 'id' ) ) {
2022-09-28 22:39:33 -04:00
if ( signedIn && ! account . get ( 'relationship' ) ) { // Wait until the relationship is loaded
2024-03-11 10:33:48 -04:00
actionBtn = < Button disabled > < LoadingIndicator / > < / Button > ;
2017-02-05 14:58:09 -05:00
} else if ( ! account . getIn ( [ 'relationship' , 'blocking' ] ) ) {
2024-07-01 14:10:22 -04:00
actionBtn = < Button disabled = { account . getIn ( [ 'relationship' , 'blocked_by' ] ) } className = { classNames ( { 'button--destructive' : ( account . getIn ( [ 'relationship' , 'following' ] ) || account . getIn ( [ 'relationship' , 'requested' ] ) ) } ) } text = { intl . formatMessage ( messageForFollowButton ( account . get ( 'relationship' ) ) ) } onClick = { signedIn ? this . props . onFollow : this . props . onInteractionModal } / > ;
2018-03-04 23:09:35 -05:00
} else if ( account . getIn ( [ 'relationship' , 'blocking' ] ) ) {
2023-07-11 06:26:09 -04:00
actionBtn = < Button text = { intl . formatMessage ( messages . unblock , { name : account . get ( 'username' ) } ) } onClick = { this . props . onBlock } / > ;
2016-12-22 17:03:57 -05:00
}
2018-05-30 12:41:47 -04:00
} else {
2023-07-11 06:26:09 -04:00
actionBtn = < Button text = { intl . formatMessage ( messages . edit _profile ) } onClick = { this . openEditProfile } / > ;
2016-11-23 17:34:12 -05:00
}
2018-01-15 12:42:15 -05:00
if ( account . get ( 'moved' ) && ! account . getIn ( [ 'relationship' , 'following' ] ) ) {
2017-11-18 13:39:02 -05:00
actionBtn = '' ;
}
2016-12-22 18:04:52 -05:00
if ( account . get ( 'locked' ) ) {
2023-10-24 13:45:08 -04:00
lockedIcon = < Icon id = 'lock' icon = { LockIcon } title = { intl . formatMessage ( messages . account _locked ) } / > ;
2016-12-22 18:04:52 -05:00
}
2023-11-09 09:50:25 -05:00
if ( signedIn && account . get ( 'id' ) !== me && ! account . get ( 'suspended' ) ) {
2019-03-25 19:36:25 -04:00
menu . push ( { text : intl . formatMessage ( messages . mention , { name : account . get ( 'username' ) } ) , action : this . props . onMention } ) ;
2022-05-03 03:09:09 -04:00
menu . push ( { text : intl . formatMessage ( messages . direct , { name : account . get ( 'username' ) } ) , action : this . props . onDirect } ) ;
2019-03-25 19:36:25 -04:00
menu . push ( null ) ;
}
2022-11-10 02:49:35 -05:00
if ( isRemote ) {
menu . push ( { text : intl . formatMessage ( messages . openOriginalPage ) , href : account . get ( 'url' ) } ) ;
2019-03-25 19:36:25 -04:00
menu . push ( null ) ;
}
if ( account . get ( 'id' ) === me ) {
menu . push ( { text : intl . formatMessage ( messages . edit _profile ) , href : '/settings/profile' } ) ;
menu . push ( { text : intl . formatMessage ( messages . preferences ) , href : '/settings/preferences' } ) ;
menu . push ( { text : intl . formatMessage ( messages . pins ) , to : '/pinned' } ) ;
menu . push ( null ) ;
menu . push ( { text : intl . formatMessage ( messages . follow _requests ) , to : '/follow_requests' } ) ;
menu . push ( { text : intl . formatMessage ( messages . favourites ) , to : '/favourites' } ) ;
menu . push ( { text : intl . formatMessage ( messages . lists ) , to : '/lists' } ) ;
2023-01-18 10:44:33 -05:00
menu . push ( { text : intl . formatMessage ( messages . followed _tags ) , to : '/followed_tags' } ) ;
2019-03-25 19:36:25 -04:00
menu . push ( null ) ;
menu . push ( { text : intl . formatMessage ( messages . mutes ) , to : '/mutes' } ) ;
menu . push ( { text : intl . formatMessage ( messages . blocks ) , to : '/blocks' } ) ;
menu . push ( { text : intl . formatMessage ( messages . domain _blocks ) , to : '/domain_blocks' } ) ;
2022-09-28 22:39:33 -04:00
} else if ( signedIn ) {
2019-03-25 19:36:25 -04:00
if ( account . getIn ( [ 'relationship' , 'following' ] ) ) {
2020-05-13 15:20:45 -04:00
if ( ! account . getIn ( [ 'relationship' , 'muting' ] ) ) {
if ( account . getIn ( [ 'relationship' , 'showing_reblogs' ] ) ) {
menu . push ( { text : intl . formatMessage ( messages . hideReblogs , { name : account . get ( 'username' ) } ) , action : this . props . onReblogToggle } ) ;
} else {
menu . push ( { text : intl . formatMessage ( messages . showReblogs , { name : account . get ( 'username' ) } ) , action : this . props . onReblogToggle } ) ;
}
2022-09-20 17:51:21 -04:00
menu . push ( { text : intl . formatMessage ( messages . languages ) , action : this . props . onChangeLanguages } ) ;
menu . push ( null ) ;
2019-03-25 19:36:25 -04:00
}
menu . push ( { text : intl . formatMessage ( account . getIn ( [ 'relationship' , 'endorsed' ] ) ? messages . unendorse : messages . endorse ) , action : this . props . onEndorseToggle } ) ;
menu . push ( { text : intl . formatMessage ( messages . add _or _remove _from _list ) , action : this . props . onAddToList } ) ;
menu . push ( null ) ;
}
if ( account . getIn ( [ 'relationship' , 'muting' ] ) ) {
menu . push ( { text : intl . formatMessage ( messages . unmute , { name : account . get ( 'username' ) } ) , action : this . props . onMute } ) ;
} else {
2023-06-06 15:49:49 -04:00
menu . push ( { text : intl . formatMessage ( messages . mute , { name : account . get ( 'username' ) } ) , action : this . props . onMute , dangerous : true } ) ;
2019-03-25 19:36:25 -04:00
}
if ( account . getIn ( [ 'relationship' , 'blocking' ] ) ) {
menu . push ( { text : intl . formatMessage ( messages . unblock , { name : account . get ( 'username' ) } ) , action : this . props . onBlock } ) ;
} else {
2023-06-06 15:49:49 -04:00
menu . push ( { text : intl . formatMessage ( messages . block , { name : account . get ( 'username' ) } ) , action : this . props . onBlock , dangerous : true } ) ;
2019-03-25 19:36:25 -04:00
}
2023-11-09 09:50:25 -05:00
if ( ! account . get ( 'suspended' ) ) {
menu . push ( { text : intl . formatMessage ( messages . report , { name : account . get ( 'username' ) } ) , action : this . props . onReport , dangerous : true } ) ;
}
2019-03-25 19:36:25 -04:00
}
2022-11-10 02:49:35 -05:00
if ( signedIn && isRemote ) {
2019-03-25 19:36:25 -04:00
menu . push ( null ) ;
if ( account . getIn ( [ 'relationship' , 'domain_blocking' ] ) ) {
2022-11-10 02:49:35 -05:00
menu . push ( { text : intl . formatMessage ( messages . unblockDomain , { domain : remoteDomain } ) , action : this . props . onUnblockDomain } ) ;
2019-03-25 19:36:25 -04:00
} else {
2023-06-06 15:49:49 -04:00
menu . push ( { text : intl . formatMessage ( messages . blockDomain , { domain : remoteDomain } ) , action : this . props . onBlockDomain , dangerous : true } ) ;
2019-03-25 19:36:25 -04:00
}
}
2023-01-05 08:03:46 -05:00
if ( ( account . get ( 'id' ) !== me && ( permissions & PERMISSION _MANAGE _USERS ) === PERMISSION _MANAGE _USERS ) || ( isRemote && ( permissions & PERMISSION _MANAGE _FEDERATION ) === PERMISSION _MANAGE _FEDERATION ) ) {
2019-03-25 19:36:25 -04:00
menu . push ( null ) ;
2023-01-05 08:03:46 -05:00
if ( ( permissions & PERMISSION _MANAGE _USERS ) === PERMISSION _MANAGE _USERS ) {
menu . push ( { text : intl . formatMessage ( messages . admin _account , { name : account . get ( 'username' ) } ) , href : ` /admin/accounts/ ${ account . get ( 'id' ) } ` } ) ;
}
2023-01-29 17:44:03 -05:00
if ( isRemote && ( permissions & PERMISSION _MANAGE _FEDERATION ) === PERMISSION _MANAGE _FEDERATION ) {
menu . push ( { text : intl . formatMessage ( messages . admin _domain , { domain : remoteDomain } ) , href : ` /admin/instances/ ${ remoteDomain } ` } ) ;
}
2019-03-25 19:36:25 -04:00
}
2017-08-07 14:32:03 -04:00
const content = { _ _html : account . get ( 'note_emojified' ) } ;
const displayNameHtml = { _ _html : account . get ( 'display_name_html' ) } ;
2018-04-14 06:41:08 -04:00
const fields = account . get ( 'fields' ) ;
2022-10-20 08:35:29 -04:00
const isLocal = account . get ( 'acct' ) . indexOf ( '@' ) === - 1 ;
2024-03-15 09:08:38 -04:00
const username = account . get ( 'acct' ) . split ( '@' ) [ 0 ] ;
const domain = isLocal ? localDomain : account . get ( 'acct' ) . split ( '@' ) [ 1 ] ;
2022-10-20 08:35:29 -04:00
const isIndexable = ! account . get ( 'noindex' ) ;
2016-11-06 19:14:12 -05:00
2023-07-27 10:05:24 -04:00
const badges = [ ] ;
2019-12-04 14:36:33 -05:00
if ( account . get ( 'bot' ) ) {
2023-08-02 11:24:32 -04:00
badges . push ( < AutomatedBadge key = 'bot-badge' / > ) ;
2019-12-04 14:36:33 -05:00
} else if ( account . get ( 'group' ) ) {
2023-08-02 11:24:32 -04:00
badges . push ( < GroupBadge key = 'group-badge' / > ) ;
2019-12-04 14:36:33 -05:00
}
2023-07-27 10:05:24 -04:00
account . get ( 'roles' , [ ] ) . forEach ( ( role ) => {
2024-03-22 07:59:35 -04:00
badges . push ( < Badge key = { ` role-badge- ${ role . get ( 'id' ) } ` } label = { < span > { role . get ( 'name' ) } < / span > } domain = { domain } roleId = { role . get ( 'id' ) } / > ) ;
2023-07-27 10:05:24 -04:00
} ) ;
2016-09-18 12:18:46 -04:00
return (
2021-01-31 15:25:31 -05:00
< div className = { classNames ( 'account__header' , { inactive : ! ! account . get ( 'moved' ) } ) } onMouseEnter = { this . handleMouseEnter } onMouseLeave = { this . handleMouseLeave } >
2022-12-15 12:50:11 -05:00
{ ! ( suspended || hidden || account . get ( 'moved' ) ) && account . getIn ( [ 'relationship' , 'requested_by' ] ) && < FollowRequestNoteContainer account = { account } / > }
2019-03-25 19:36:25 -04:00
< div className = 'account__header__image' >
< div className = 'account__header__info' >
2023-11-09 09:50:25 -05:00
{ info }
2019-03-25 19:36:25 -04:00
< / div >
2016-09-18 12:18:46 -04:00
2022-05-10 03:44:35 -04:00
{ ! ( suspended || hidden ) && < img src = { autoPlayGif ? account . get ( 'header' ) : account . get ( 'header_static' ) } alt = '' className = 'parallax' / > }
2019-03-25 19:36:25 -04:00
< / div >
2018-05-07 03:31:07 -04:00
2019-03-25 19:36:25 -04:00
< div className = 'account__header__bar' >
< div className = 'account__header__tabs' >
2022-11-10 02:49:35 -05:00
< a className = 'avatar' href = { account . get ( 'avatar' ) } rel = 'noopener noreferrer' target = '_blank' onClick = { this . handleAvatarClick } >
2022-05-10 03:44:35 -04:00
< Avatar account = { suspended || hidden ? undefined : account } size = { 90 } / >
2019-03-25 19:36:25 -04:00
< / a >
2018-05-07 03:31:07 -04:00
2023-11-09 09:50:25 -05:00
< div className = 'account__header__tabs__buttons' >
2024-03-11 10:33:48 -04:00
{ ! hidden && bellBtn }
{ ! hidden && shareBtn }
2023-11-09 09:50:25 -05:00
< DropdownMenuContainer disabled = { menu . length === 0 } items = { menu } icon = 'ellipsis-v' iconComponent = { MoreHorizIcon } size = { 24 } direction = 'right' / >
2024-03-11 10:33:48 -04:00
{ ! hidden && actionBtn }
2023-11-09 09:50:25 -05:00
< / div >
2019-03-25 19:36:25 -04:00
< / div >
< div className = 'account__header__tabs__name' >
< h1 >
2023-07-27 10:05:24 -04:00
< span dangerouslySetInnerHTML = { displayNameHtml } / >
2022-12-13 13:43:03 -05:00
< small >
2024-03-15 09:08:38 -04:00
< span > @ { username } < span className = 'invisible' > @ { domain } < / span > < / span >
< DomainPill username = { username } domain = { domain } isSelf = { me === account . get ( 'id' ) } / >
{ lockedIcon }
2022-12-13 13:43:03 -05:00
< / small >
2019-03-25 19:36:25 -04:00
< / h1 >
< / div >
2023-07-27 10:05:24 -04:00
{ badges . length > 0 && (
< div className = 'account__header__badges' >
{ badges }
< / div >
) }
2022-05-10 03:44:35 -04:00
{ ! ( suspended || hidden ) && (
< div className = 'account__header__extra' >
2023-04-25 00:33:21 -04:00
< div className = 'account__header__bio' ref = { this . setRef } >
2022-09-28 22:39:33 -04:00
{ ( account . get ( 'id' ) !== me && signedIn ) && < AccountNoteContainer account = { account } / > }
2020-07-06 19:24:03 -04:00
2022-05-10 03:44:35 -04:00
{ account . get ( 'note' ) . length > 0 && account . get ( 'note' ) !== '<p></p>' && < div className = 'account__header__content translate' dangerouslySetInnerHTML = { content } / > }
2021-05-07 08:33:19 -04:00
2022-10-25 13:02:21 -04:00
< div className = 'account__header__fields' >
< dl >
< dt > < FormattedMessage id = 'account.joined_short' defaultMessage = 'Joined' / > < / dt >
< dd > { intl . formatDate ( account . get ( 'created_at' ) , { year : 'numeric' , month : 'short' , day : '2-digit' } ) } < / dd >
< / dl >
{ fields . map ( ( pair , i ) => (
2022-11-03 21:28:00 -04:00
< dl key = { i } className = { classNames ( { verified : pair . get ( 'verified_at' ) } ) } >
2022-10-25 13:02:21 -04:00
< dt dangerouslySetInnerHTML = { { _ _html : pair . get ( 'name_emojified' ) } } title = { pair . get ( 'name' ) } className = 'translate' / >
2022-11-03 21:28:00 -04:00
< dd className = 'translate' title = { pair . get ( 'value_plain' ) } >
2023-10-24 13:45:08 -04:00
{ pair . get ( 'verified_at' ) && < span title = { intl . formatMessage ( messages . linkVerifiedOn , { date : intl . formatDate ( pair . get ( 'verified_at' ) , dateFormatOptions ) } ) } > < Icon id = 'check' icon = { CheckIcon } className = 'verified__mark' / > < / span > } < span dangerouslySetInnerHTML = { { _ _html : pair . get ( 'value_emojified' ) } } / >
2022-10-25 13:02:21 -04:00
< / dd >
< / dl >
) ) }
< / div >
2022-05-10 03:44:35 -04:00
< / div >
2019-03-25 19:36:25 -04:00
2020-09-09 18:07:19 -04:00
< div className = 'account__header__extra__links' >
2021-09-25 23:46:13 -04:00
< NavLink isActive = { this . isStatusesPageActive } activeClassName = 'active' to = { ` /@ ${ account . get ( 'acct' ) } ` } title = { intl . formatNumber ( account . get ( 'statuses_count' ) ) } >
2020-09-09 18:07:19 -04:00
< ShortNumber
value = { account . get ( 'statuses_count' ) }
2023-07-08 05:11:22 -04:00
renderer = { StatusesCounter }
2020-09-09 18:07:19 -04:00
/ >
< / NavLink >
2021-09-25 23:46:13 -04:00
< NavLink exact activeClassName = 'active' to = { ` /@ ${ account . get ( 'acct' ) } /following ` } title = { intl . formatNumber ( account . get ( 'following_count' ) ) } >
2020-09-09 18:07:19 -04:00
< ShortNumber
value = { account . get ( 'following_count' ) }
2023-07-08 05:11:22 -04:00
renderer = { FollowingCounter }
2020-09-09 18:07:19 -04:00
/ >
< / NavLink >
2021-09-25 23:46:13 -04:00
< NavLink exact activeClassName = 'active' to = { ` /@ ${ account . get ( 'acct' ) } /followers ` } title = { intl . formatNumber ( account . get ( 'followers_count' ) ) } >
2020-09-09 18:07:19 -04:00
< ShortNumber
value = { account . get ( 'followers_count' ) }
2023-07-08 05:11:22 -04:00
renderer = { FollowersCounter }
2020-09-09 18:07:19 -04:00
/ >
< / NavLink >
< / div >
2022-05-10 03:44:35 -04:00
< / div >
) }
2016-09-18 12:18:46 -04:00
< / div >
2022-09-28 22:39:33 -04:00
< Helmet >
2022-10-08 21:55:09 -04:00
< title > { titleFromAccount ( account ) } < / title >
2022-10-20 08:35:29 -04:00
< meta name = 'robots' content = { ( isLocal && isIndexable ) ? 'all' : 'noindex' } / >
2023-07-05 05:25:27 -04:00
< link rel = 'canonical' href = { account . get ( 'url' ) } / >
2022-09-28 22:39:33 -04:00
< / Helmet >
2016-09-18 12:18:46 -04:00
< / div >
) ;
}
2017-04-21 14:05:35 -04:00
}
2023-03-23 22:17:53 -04:00
2024-05-19 13:07:32 -04:00
export default withRouter ( withIdentity ( injectIntl ( Header ) ) ) ;