2017-05-03 02:04:16 +02:00
import React from 'react' ;
2016-09-18 18:18:46 +02:00
import ImmutablePropTypes from 'react-immutable-proptypes' ;
2017-04-22 03:05:35 +09:00
import PropTypes from 'prop-types' ;
2016-11-23 23:34:12 +01:00
import { defineMessages , injectIntl , FormattedMessage } from 'react-intl' ;
import IconButton from '../../../components/icon_button' ;
2017-10-16 00:36:15 -07:00
import Motion from '../../ui/util/optional_motion' ;
2017-05-20 14:58:13 +02:00
import spring from 'react-motion/lib/spring' ;
2017-05-03 02:04:16 +02:00
import ImmutablePureComponent from 'react-immutable-pure-component' ;
2017-10-30 19:27:48 -07:00
import { autoPlayGif , me } from '../../../initial_state' ;
2017-11-18 19:39:02 +01:00
import classNames from 'classnames' ;
2019-02-01 00:14:05 +01:00
import Icon from 'mastodon/components/icon' ;
2016-11-23 23:34:12 +01:00
const messages = defineMessages ( {
unfollow : { id : 'account.unfollow' , defaultMessage : 'Unfollow' } ,
follow : { id : 'account.follow' , defaultMessage : 'Follow' } ,
2017-09-02 20:44:41 +02:00
requested : { id : 'account.requested' , defaultMessage : 'Awaiting approval. Click to cancel follow request' } ,
2018-03-05 05:09:35 +01:00
unblock : { id : 'account.unblock' , defaultMessage : 'Unblock @{name}' } ,
2018-05-30 18:41:47 +02:00
edit _profile : { id : 'account.edit_profile' , defaultMessage : 'Edit profile' } ,
2018-09-18 16:45:58 +02:00
linkVerifiedOn : { id : 'account.link_verified_on' , defaultMessage : 'Ownership of this link was checked on {date}' } ,
2018-12-01 14:25:15 +01:00
account _locked : { id : 'account.locked_info' , defaultMessage : 'This account privacy status is set to locked. The owner manually reviews who can follow them.' } ,
2016-11-23 23:34:12 +01:00
} ) ;
2016-09-18 18:18:46 +02:00
2018-09-18 16:45:58 +02:00
const dateFormatOptions = {
month : 'short' ,
day : 'numeric' ,
year : 'numeric' ,
hour12 : false ,
hour : '2-digit' ,
minute : '2-digit' ,
} ;
2017-05-03 02:04:16 +02:00
class Avatar extends ImmutablePureComponent {
2017-03-31 14:23:44 +02:00
2017-05-12 21:44:10 +09:00
static propTypes = {
account : ImmutablePropTypes . map . isRequired ,
} ;
2017-04-22 21:26:55 -05:00
2017-05-12 21:44:10 +09:00
state = {
2017-05-21 00:31:47 +09:00
isHovered : false ,
2017-05-12 21:44:10 +09:00
} ;
2017-04-22 21:26:55 -05:00
2017-05-12 21:44:10 +09:00
handleMouseOver = ( ) => {
2017-03-31 14:23:44 +02:00
if ( this . state . isHovered ) return ;
this . setState ( { isHovered : true } ) ;
2017-04-22 03:05:35 +09:00
}
2017-03-31 14:23:44 +02:00
2017-05-12 21:44:10 +09:00
handleMouseOut = ( ) => {
2017-03-31 14:23:44 +02:00
if ( ! this . state . isHovered ) return ;
this . setState ( { isHovered : false } ) ;
2017-04-22 03:05:35 +09:00
}
2017-03-31 14:23:44 +02:00
render ( ) {
2017-10-27 08:04:44 -07:00
const { account } = this . props ;
2017-03-31 14:23:44 +02:00
const { isHovered } = this . state ;
return (
< Motion defaultStyle = { { radius : 90 } } style = { { radius : spring ( isHovered ? 30 : 90 , { stiffness : 180 , damping : 12 } ) } } >
2018-01-18 00:57:15 +09:00
{ ( { radius } ) => (
2017-07-28 00:54:48 +02:00
< a
2017-04-15 05:27:27 -06:00
href = { account . get ( 'url' ) }
className = 'account__header__avatar'
2017-07-28 00:54:48 +02:00
role = 'presentation'
2017-04-15 05:27:27 -06:00
target = '_blank'
rel = 'noopener'
2017-04-22 21:26:55 -05:00
style = { { borderRadius : ` ${ radius } px ` , backgroundImage : ` url( ${ autoPlayGif || isHovered ? account . get ( 'avatar' ) : account . get ( 'avatar_static' ) } ) ` } }
2017-04-15 05:27:27 -06:00
onMouseOver = { this . handleMouseOver }
onMouseOut = { this . handleMouseOut }
onFocus = { this . handleMouseOver }
2017-04-18 01:57:50 +02:00
onBlur = { this . handleMouseOut }
2017-07-28 00:54:48 +02:00
>
< span style = { { display : 'none' } } > { account . get ( 'acct' ) } < / s p a n >
< / a >
2018-01-18 00:57:15 +09:00
) }
2017-03-31 14:23:44 +02:00
< / M o t i o n >
) ;
}
2017-04-22 03:05:35 +09:00
}
2017-03-31 14:23:44 +02:00
2018-09-15 00:59:48 +09:00
export default @ injectIntl
class Header extends ImmutablePureComponent {
2016-09-18 18:18:46 +02:00
2017-05-12 21:44:10 +09:00
static propTypes = {
account : ImmutablePropTypes . map ,
onFollow : PropTypes . func . isRequired ,
2018-03-05 05:09:35 +01:00
onBlock : PropTypes . func . isRequired ,
2017-05-12 21:44:10 +09:00
intl : PropTypes . object . isRequired ,
} ;
2018-05-30 18:41:47 +02:00
openEditProfile = ( ) => {
window . open ( '/settings/profile' , '_blank' ) ;
}
2016-09-18 18:18:46 +02:00
render ( ) {
2017-10-30 19:27:48 -07:00
const { account , intl } = this . props ;
2016-09-18 18:18:46 +02:00
2017-02-21 00:10:49 +01:00
if ( ! account ) {
return null ;
}
2016-11-15 18:38:57 +01:00
let info = '' ;
2018-03-05 05:09:35 +01:00
let mutingInfo = '' ;
2016-11-23 23:34:12 +01:00
let actionBtn = '' ;
2016-12-23 00:04:52 +01:00
let lockedIcon = '' ;
2016-10-06 22:07:32 +02:00
2016-10-09 22:19:15 +02:00
if ( me !== account . get ( 'id' ) && account . getIn ( [ 'relationship' , 'followed_by' ] ) ) {
2017-05-21 00:31:47 +09:00
info = < span className = 'account--follows-info' > < FormattedMessage id = 'account.follows_you' defaultMessage = 'Follows you' / > < / s p a n > ;
2018-03-05 05:09:35 +01:00
} 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 > ;
}
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 > ;
2018-03-05 16:45:36 +01:00
} 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 > ;
2016-10-09 22:19:15 +02:00
}
2016-11-23 23:34:12 +01:00
if ( me !== account . get ( 'id' ) ) {
2018-08-25 22:46:59 +02:00
if ( ! account . get ( 'relationship' ) ) { // Wait until the relationship is loaded
actionBtn = '' ;
} else if ( account . getIn ( [ 'relationship' , 'requested' ] ) ) {
2016-12-22 23:03:57 +01:00
actionBtn = (
2017-05-19 18:42:54 +09:00
< div className = 'account--action-button' >
2017-09-02 20:44:41 +02:00
< IconButton size = { 26 } active icon = 'hourglass' title = { intl . formatMessage ( messages . requested ) } onClick = { this . props . onFollow } / >
2016-12-22 23:03:57 +01:00
< / d i v >
) ;
2017-02-05 20:58:09 +01:00
} else if ( ! account . getIn ( [ 'relationship' , 'blocking' ] ) ) {
2016-12-22 23:03:57 +01:00
actionBtn = (
2017-05-19 18:42:54 +09:00
< div className = 'account--action-button' >
2016-12-22 23:03:57 +01:00
< 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 >
) ;
2018-03-05 05:09:35 +01:00
} else if ( account . getIn ( [ 'relationship' , 'blocking' ] ) ) {
actionBtn = (
< div className = 'account--action-button' >
2019-01-31 07:45:15 -05:00
< IconButton size = { 26 } icon = 'unlock' title = { intl . formatMessage ( messages . unblock , { name : account . get ( 'username' ) } ) } onClick = { this . props . onBlock } / >
2018-03-05 05:09:35 +01:00
< / d i v >
) ;
2016-12-22 23:03:57 +01:00
}
2018-05-30 18:41:47 +02:00
} else {
actionBtn = (
< div className = 'account--action-button' >
2018-09-28 02:11:14 +02:00
< IconButton size = { 26 } icon = 'pencil' title = { intl . formatMessage ( messages . edit _profile ) } onClick = { this . openEditProfile } / >
2018-05-30 18:41:47 +02:00
< / d i v >
) ;
2016-11-23 23:34:12 +01:00
}
2018-01-15 18:42:15 +01:00
if ( account . get ( 'moved' ) && ! account . getIn ( [ 'relationship' , 'following' ] ) ) {
2017-11-18 19:39:02 +01:00
actionBtn = '' ;
}
2016-12-23 00:04:52 +01:00
if ( account . get ( 'locked' ) ) {
2019-02-01 00:14:05 +01:00
lockedIcon = < Icon id = 'lock' title = { intl . formatMessage ( messages . account _locked ) } / > ;
2016-12-23 00:04:52 +01:00
}
2017-08-07 20:32:03 +02:00
const content = { _ _html : account . get ( 'note_emojified' ) } ;
const displayNameHtml = { _ _html : account . get ( 'display_name_html' ) } ;
2018-04-14 12:41:08 +02:00
const fields = account . get ( 'fields' ) ;
2018-05-07 09:31:07 +02:00
const 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 ;
2016-11-07 01:14:12 +01:00
2016-09-18 18:18:46 +02:00
return (
2018-12-14 20:34:18 +01:00
< div className = { classNames ( 'account__header' , { inactive : ! ! account . get ( 'moved' ) } ) } style = { { backgroundImage : ` url( ${ autoPlayGif ? account . get ( 'header' ) : account . get ( 'header_static' ) } ) ` } } >
2017-05-19 18:42:54 +09:00
< div >
2017-10-27 08:04:44 -07:00
< Avatar account = { account } / >
2016-09-18 18:18:46 +02:00
2017-08-07 20:32:03 +02:00
< span className = 'account__header__display-name' dangerouslySetInnerHTML = { displayNameHtml } / >
2017-05-19 18:42:54 +09:00
< span className = 'account__header__username' > @ { account . get ( 'acct' ) } { lockedIcon } < / s p a n >
2018-05-07 09:31:07 +02:00
{ badge }
2017-05-19 18:42:54 +09:00
< div className = 'account__header__content' dangerouslySetInnerHTML = { content } / >
2016-10-09 22:19:15 +02:00
2018-04-14 12:41:08 +02:00
{ fields . size > 0 && (
2018-05-05 00:55:09 +02:00
< div className = 'account__header__fields' >
{ fields . map ( ( pair , i ) => (
< dl key = { i } >
< dt dangerouslySetInnerHTML = { { _ _html : pair . get ( 'name_emojified' ) } } title = { pair . get ( 'name' ) } / >
2018-09-18 16:45:58 +02:00
< dd className = { pair . get ( 'verified_at' ) && 'verified' } title = { pair . get ( 'value_plain' ) } >
2019-02-01 00:14:05 +01:00
{ pair . get ( 'verified_at' ) && < span title = { intl . formatMessage ( messages . linkVerifiedOn , { date : intl . formatDate ( pair . get ( 'verified_at' ) , dateFormatOptions ) } ) } > < Icon id = 'check' className = 'verified__mark' / > < /span>} <span dangerouslySetInnerHTML={{ __html: pair.get('value_emojified') }} / >
2018-09-18 16:45:58 +02:00
< / d d >
2018-05-05 00:55:09 +02:00
< / d l >
) ) }
< / d i v >
2018-04-14 12:41:08 +02:00
) }
2016-10-09 22:19:15 +02:00
{ info }
2018-03-05 05:09:35 +01:00
{ mutingInfo }
2016-11-23 23:34:12 +01:00
{ actionBtn }
2016-09-18 18:18:46 +02:00
< / d i v >
< / d i v >
) ;
}
2017-04-22 03:05:35 +09:00
}