@ -3,12 +3,14 @@ import ImmutablePropTypes from 'react-immutable-proptypes';
import PropTypes from 'prop-types' ;
import { defineMessages , injectIntl , FormattedMessage } from 'react-intl' ;
import ImmutablePureComponent from 'react-immutable-pure-component' ;
import Avatar from 'flavours/glitch/components/avatar' ;
import IconButton from 'flavours/glitch/components/icon_button' ;
import { autoPlayGif , me } from 'flavours/glitch/util/initial_state' ;
import { autoPlayGif , me , isStaff } from 'flavours/glitch/util/initial_state' ;
import classNames from 'classnames' ;
import Icon from 'flavours/glitch/components/icon' ;
import Avatar from 'flavours/glitch/components/avatar' ;
import Button from 'flavours/glitch/components/button' ;
import { shortNumberFormat } from 'flavours/glitch/util/numbers' ;
import { NavLink } from 'react-router-dom' ;
import DropdownMenuContainer from 'flavours/glitch/containers/dropdown_menu_container' ;
const messages = defineMessages ( {
unfollow : { id : 'account.unfollow' , defaultMessage : 'Unfollow' } ,
@ -16,7 +18,34 @@ const messages = defineMessages({
requested : { id : 'account.requested' , defaultMessage : 'Awaiting approval. Click to cancel follow request' } ,
unblock : { id : 'account.unblock' , defaultMessage : 'Unblock @{name}' } ,
edit _profile : { id : 'account.edit_profile' , defaultMessage : 'Edit profile' } ,
link _verified _on : { id : 'account.link_verified_on' , defaultMessage : 'Ownership of this link was checked on {date}' } ,
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 @{name}' } ,
direct : { id : 'account.direct' , defaultMessage : 'Direct message @{name}' } ,
edit _profile : { id : 'account.edit_profile' , defaultMessage : 'Edit profile' } ,
unblock : { id : 'account.unblock' , defaultMessage : 'Unblock @{name}' } ,
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' } ,
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 boosts from @{name}' } ,
showReblogs : { id : 'account.show_reblogs' , defaultMessage : 'Show boosts from @{name}' } ,
pins : { id : 'navigation_bar.pins' , defaultMessage : 'Pinned toots' } ,
preferences : { id : 'navigation_bar.preferences' , defaultMessage : 'Preferences' } ,
follow _requests : { id : 'navigation_bar.follow_requests' , defaultMessage : 'Follow requests' } ,
favourites : { id : 'navigation_bar.favourites' , defaultMessage : 'Favourites' } ,
lists : { id : 'navigation_bar.lists' , defaultMessage : 'Lists' } ,
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' } ,
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}' } ,
} ) ;
const dateFormatOptions = {
@ -28,14 +57,15 @@ const dateFormatOptions = {
minute : '2-digit' ,
} ;
@ injectIntl
export default class Header extends ImmutablePureComponent {
export default @ injectIntl
class Header extends ImmutablePureComponent {
static propTypes = {
account : ImmutablePropTypes . map ,
onFollow : PropTypes . func . isRequired ,
onBlock : PropTypes . func . isRequired ,
intl : PropTypes . object . isRequired ,
domain : PropTypes . string . isRequired ,
} ;
openEditProfile = ( ) => {
@ -43,106 +73,176 @@ export default class Header extends ImmutablePureComponent {
}
render ( ) {
const { account , intl } = this . props ;
const { account , intl , domain } = this . props ;
if ( ! account ) {
return null ;
}
let displayName = account . get ( 'display_name_html' ) ;
let fields = account . get ( 'fields' ) ;
let badge = account . get ( 'bot' ) ? ( < div className = 'roles' > < div className = 'account-role bot' > < FormattedMessage id = 'account.badges.bot' defaultMessage = 'Bot' / > < / d i v > < / d i v > ) : n u l l ;
let info = '' ;
let mutingInfo = '' ;
let info = [ ] ;
let actionBtn = '' ;
let lockedIcon = '' ;
let menu = [ ] ;
if ( me !== account . get ( 'id' ) && account . getIn ( [ 'relationship' , 'followed_by' ] ) ) {
info = < span className = 'account--follows-info '> < FormattedMessage id = 'account.follows_you' defaultMessage = 'Follows you' / > < / s p a n > ;
info . push ( < span className = 'relationship-tag '> < FormattedMessage id = 'account.follows_you' defaultMessage = 'Follows you' / > < / s p a n > ) ;
}
else if ( me !== account . get ( 'id' ) && account . getIn ( [ 'relationship' , 'blocking' ] ) ) {
info = < span className = 'account--follows-info '> < FormattedMessage id = 'account.blocked' defaultMessage = 'Blocked' / > < / s p a n > ;
info . push ( < span className = 'relationship-tag '> < FormattedMessage id = 'account.blocked' defaultMessage = 'Blocked' / > < / s p a n > ) ;
}
if ( me !== account . get ( 'id' ) && account . getIn ( [ 'relationship' , 'muting' ] ) ) {
mutingInfo = < span className = 'account--muting-info '> < FormattedMessage id = 'account.muted' defaultMessage = 'Muted' / > < / s p a n > ;
info. push ( < span className = 'relationship-tag '> < FormattedMessage id = 'account.muted' defaultMessage = 'Muted' / > < / s p a n > ) ;
} else if ( me !== account . get ( 'id' ) && account . getIn ( [ 'relationship' , 'domain_blocking' ] ) ) {
mutingInfo = < span className = 'account--muting-info '> < FormattedMessage id = 'account.domain_blocked' defaultMessage = 'Domain hidden' / > < / s p a n > ;
info. push ( < span className = 'relationship-tag '> < FormattedMessage id = 'account.domain_blocked' defaultMessage = 'Domain hidden' / > < / s p a n > ) ;
}
if ( me !== account . get ( 'id' ) ) {
if ( ! account . get ( 'relationship' ) ) { // Wait until the relationship is loaded
actionBtn = '' ;
} else if ( account . getIn ( [ 'relationship' , 'requested' ] ) ) {
actionBtn = (
< div className = 'account--action-button' >
< IconButton size = { 26 } active icon = 'hourglass' title = { intl . formatMessage ( messages . requested ) } onClick = { this . props . onFollow } / >
< / d i v >
) ;
actionBtn = < Button className = 'logo-button' text = { intl . formatMessage ( messages . requested ) } onClick = { this . props . onFollow } / > ;
} else if ( ! account . getIn ( [ 'relationship' , 'blocking' ] ) ) {
actionBtn = (
< div className = 'account--action-button' >
< IconButton size = { 26 } icon = { account . getIn ( [ 'relationship' , 'following' ] ) ? 'user-times' : 'user-plus' } active = { account . getIn ( [ 'relationship' , 'following' ] ) } title = { intl . formatMessage ( account . getIn ( [ 'relationship' , 'following' ] ) ? messages . unfollow : messages . follow ) } onClick = { this . props . onFollow } / >
< / d i v >
) ;
actionBtn = < Button className = { classNames ( 'logo-button' , { 'button--destructive' : account . getIn ( [ 'relationship' , 'following' ] ) } ) } text = { intl . formatMessage ( account . getIn ( [ 'relationship' , 'following' ] ) ? messages . unfollow : messages . follow ) } onClick = { this . props . onFollow } / > ;
} else if ( account . getIn ( [ 'relationship' , 'blocking' ] ) ) {
actionBtn = (
< div className = 'account--action-button' >
< IconButton size = { 26 } icon = 'unlock' title = { intl . formatMessage ( messages . unblock , { name : account . get ( 'username' ) } ) } onClick = { this . props . onBlock } / >
< / d i v >
) ;
actionBtn = < Button className = 'logo-button' text = { intl . formatMessage ( messages . unblock , { name : account . get ( 'username' ) } ) } onClick = { this . props . onBlock } / > ;
}
} else {
actionBtn = (
< div className = 'account--action-button' >
< IconButton size = { 26 } icon = 'pencil' title = { intl . formatMessage ( messages . edit _profile ) } onClick = { this . openEditProfile } / >
< / d i v >
) ;
actionBtn = < Button className = 'logo-button' text = { intl . formatMessage ( messages . edit _profile ) } onClick = { this . openEditProfile } / > ;
}
if ( account . get ( 'moved' ) && ! account . getIn ( [ 'relationship' , 'following' ] ) ) {
actionBtn = '' ;
}
if ( account . get ( 'locked' ) ) {
lockedIcon = < Icon icon = 'lock' title = { intl . formatMessage ( messages . account _locked ) } / > ;
}
if ( account . get ( 'id' ) !== me ) {
menu . push ( { text : intl . formatMessage ( messages . mention , { name : account . get ( 'username' ) } ) , action : this . props . onMention } ) ;
menu . push ( { text : intl . formatMessage ( messages . direct , { name : account . get ( 'username' ) } ) , action : this . props . onDirect } ) ;
menu . push ( null ) ;
}
if ( 'share' in navigator ) {
menu . push ( { text : intl . formatMessage ( messages . share , { name : account . get ( 'username' ) } ) , action : this . handleShare } ) ;
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' } ) ;
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' } ) ;
} else {
if ( account . getIn ( [ 'relationship' , 'following' ] ) ) {
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 } ) ;
}
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 {
menu . push ( { text : intl . formatMessage ( messages . mute , { name : account . get ( 'username' ) } ) , action : this . props . onMute } ) ;
}
if ( account . getIn ( [ 'relationship' , 'blocking' ] ) ) {
menu . push ( { text : intl . formatMessage ( messages . unblock , { name : account . get ( 'username' ) } ) , action : this . props . onBlock } ) ;
} else {
menu . push ( { text : intl . formatMessage ( messages . block , { name : account . get ( 'username' ) } ) , action : this . props . onBlock } ) ;
}
menu . push ( { text : intl . formatMessage ( messages . report , { name : account . get ( 'username' ) } ) , action : this . props . onReport } ) ;
}
if ( account . get ( 'acct' ) !== account . get ( 'username' ) ) {
const domain = account . get ( 'acct' ) . split ( '@' ) [ 1 ] ;
menu . push ( null ) ;
if ( account . getIn ( [ 'relationship' , 'domain_blocking' ] ) ) {
menu . push ( { text : intl . formatMessage ( messages . unblockDomain , { domain } ) , action : this . props . onUnblockDomain } ) ;
} else {
menu . push ( { text : intl . formatMessage ( messages . blockDomain , { domain } ) , action : this . props . onBlockDomain } ) ;
}
}
if ( account . get ( 'id' ) !== me && isStaff ) {
menu . push ( null ) ;
menu . push ( { text : intl . formatMessage ( messages . admin _account , { name : account . get ( 'username' ) } ) , href : ` /admin/accounts/ ${ account . get ( 'id' ) } ` } ) ;
}
const content = { _ _html : account . get ( 'note_emojified' ) } ;
const displayNameHtml = { _ _html : account . get ( 'display_name_html' ) } ;
const fields = account . get ( 'fields' ) ;
const badge = account . get ( 'bot' ) ? ( < div className = 'account-role bot' > < FormattedMessage id = 'account.badges.bot' defaultMessage = 'Bot' / > < / d i v > ) : n u l l ;
const acct = account . get ( 'acct' ) . indexOf ( '@' ) === - 1 && domain ? ` ${ account . get ( 'acct' ) } @ ${ domain } ` : account . get ( 'acct' ) ;
return (
< div className = 'account__header__wrapper' >
< div className = { classNames ( 'account__header' , { inactive : ! ! account . get ( 'moved' ) } ) } style = { { backgroundImage : ` url( ${ autoPlayGif ? account . get ( 'header' ) : account . get ( 'header_static' ) } ) ` } } >
< div >
< a
href = { account . get ( 'url' ) }
className = 'account__header__avatar'
role = 'presentation'
target = '_blank'
rel = 'noopener'
>
< div className = { classNames ( 'account__header' , { inactive : ! ! account . get ( 'moved' ) } ) } >
< div className = 'account__header__image' >
< div className = 'account__header__info' >
{ info }
< / d i v >
< img src = { autoPlayGif ? account . get ( 'header' ) : account . get ( 'header_static' ) } alt = '' className = 'parallax' / >
< / d i v >
< div className = 'account__header__bar' >
< div className = 'account__header__tabs' >
< a className = 'avatar' href = { account . get ( 'url' ) } >
< Avatar account = { account } size = { 90 } / >
< / a >
< span className = 'account__header__display-name' dangerouslySetInnerHTML = { { _ _html : displayName } } / >
< span className = 'account__header__username' > @ { account . get ( 'acct' ) } { account . get ( 'locked' ) ? < i className = 'fa fa-lock' / > : null } < / s p a n >
< div className = 'spacer' / >
< div className = 'account__header__tabs__buttons' >
< DropdownMenuContainer items = { menu } icon = 'ellipsis-v' size = { 24 } direction = 'right' / >
{ badge }
{ actionBtn }
< / d i v >
< / d i v >
< div className = 'account__header__content' dangerouslySetInnerHTML = { content } / >
< div className = 'account__header__tabs__name' >
< h1 >
< span dangerouslySetInnerHTML = { displayNameHtml } / > { badge }
< small > @ { acct } { lockedIcon } < / s m a l l >
< / h 1 >
< / d i v >
< div className = 'account__header__extra' >
< div className = 'account__header__bio' >
{ fields . size > 0 && (
< div className = 'account__header__fields' >
{ fields . map ( ( pair , i ) => (
< dl key = { i } >
< dt dangerouslySetInnerHTML = { { _ _html : pair . get ( 'name_emojified' ) } } title = { pair . get ( 'name' ) } / >
< dd className = { pair . get ( 'verified_at' ) && 'verified' } title = { pair . get ( 'value_plain' ) } >
{ pair . get ( 'verified_at' ) && < span title = { intl . formatMessage ( messages . link _verified _on , { date : intl . formatDate ( pair . get ( 'verified_at' ) , dateFormatOptions ) } ) } > < i className = 'fa fa-check verified__mark' / > < /span>} <span dangerouslySetInnerHTML={{ __html: pair.get('value_emojified') }} / >
{ pair . get ( 'verified_at' ) && < span title = { intl . formatMessage ( messages . link VerifiedO n, { date : intl . formatDate ( pair . get ( 'verified_at' ) , dateFormatOptions ) } ) } > < Icon id= 'check' className = ' verified__mark' / > < /span>} <span dangerouslySetInnerHTML={{ __html: pair.get('value_emojified') }} / >
< / d d >
< / d l >
) ) }
< / d i v >
) }
{ info }
{ mutingInfo }
{ actionBtn }
{ account . get ( 'note' ) . length > 0 && account . get ( 'note' ) !== '<p></p>' && < div className = 'account__header__content' dangerouslySetInnerHTML = { content } / > }
< / d i v >
< / d i v >
< / d i v >
< / d i v >