2017-04-22 03:05:35 +09:00
import PropTypes from 'prop-types' ;
2023-05-23 17:15:17 +02:00
2023-10-24 19:45:08 +02:00
import { FormattedDate , FormattedMessage } from 'react-intl' ;
2023-05-23 17:15:17 +02:00
import classNames from 'classnames' ;
2023-10-19 19:44:55 +02:00
import { Link , withRouter } from 'react-router-dom' ;
2023-05-23 17:15:17 +02:00
2016-09-25 14:20:29 +02:00
import ImmutablePropTypes from 'react-immutable-proptypes' ;
2023-05-23 17:15:17 +02:00
import ImmutablePureComponent from 'react-immutable-pure-component' ;
2024-01-16 11:27:26 +01:00
import AlternateEmailIcon from '@/material-icons/400-24px/alternate_email.svg?react' ;
2023-05-23 17:15:17 +02:00
import { AnimatedNumber } from 'mastodon/components/animated_number' ;
2024-08-22 19:12:35 +02:00
import { ContentWarning } from 'mastodon/components/content_warning' ;
2023-05-23 17:15:17 +02:00
import EditedTimestamp from 'mastodon/components/edited_timestamp' ;
2023-08-21 19:39:01 +02:00
import { getHashtagBarForStatus } from 'mastodon/components/hashtag_bar' ;
2023-05-23 17:15:17 +02:00
import { Icon } from 'mastodon/components/icon' ;
import PictureInPicturePlaceholder from 'mastodon/components/picture_in_picture_placeholder' ;
2023-10-24 19:45:08 +02:00
import { VisibilityIcon } from 'mastodon/components/visibility_icon' ;
2023-10-19 19:44:55 +02:00
import { WithRouterPropTypes } from 'mastodon/utils/react_router' ;
2023-05-23 17:15:17 +02:00
2023-05-09 03:11:56 +02:00
import { Avatar } from '../../../components/avatar' ;
2023-05-10 06:08:54 +09:00
import { DisplayName } from '../../../components/display_name' ;
2016-11-16 17:20:52 +01:00
import MediaGallery from '../../../components/media_gallery' ;
2023-05-23 17:15:17 +02:00
import StatusContent from '../../../components/status_content' ;
2019-08-23 22:38:02 +02:00
import Audio from '../../audio' ;
2019-01-16 19:47:46 +01:00
import scheduleIdleTask from '../../ui/util/schedule_idle_task' ;
2023-05-23 17:15:17 +02:00
import Video from '../../video' ;
import Card from './card' ;
2016-09-25 14:20:29 +02:00
2020-06-26 05:43:59 +09:00
class DetailedStatus extends ImmutablePureComponent {
2016-09-25 14:20:29 +02:00
2017-05-12 21:44:10 +09:00
static propTypes = {
2019-03-12 17:34:00 +01:00
status : ImmutablePropTypes . map ,
2017-05-12 21:44:10 +09:00
onOpenMedia : PropTypes . func . isRequired ,
onOpenVideo : PropTypes . func . isRequired ,
2018-03-11 09:52:59 +01:00
onToggleHidden : PropTypes . func . isRequired ,
2022-09-23 23:00:12 +02:00
onTranslate : PropTypes . func . isRequired ,
2019-01-16 19:47:46 +01:00
measureHeight : PropTypes . bool ,
onHeightChange : PropTypes . func ,
domain : PropTypes . string . isRequired ,
2019-01-17 14:06:08 +01:00
compact : PropTypes . bool ,
2019-05-25 23:20:51 +02:00
showMedia : PropTypes . bool ,
2020-12-07 19:36:36 +01:00
pictureInPicture : ImmutablePropTypes . contains ( {
inUse : PropTypes . bool ,
available : PropTypes . bool ,
} ) ,
2019-05-25 23:20:51 +02:00
onToggleMediaVisibility : PropTypes . func ,
2023-10-19 19:44:55 +02:00
... WithRouterPropTypes ,
2019-01-16 19:47:46 +01:00
} ;
state = {
height : null ,
2017-05-12 21:44:10 +09:00
} ;
handleAccountClick = ( e ) => {
2023-10-19 19:44:55 +02:00
if ( e . button === 0 && ! ( e . ctrlKey || e . metaKey ) && this . props . history ) {
2016-09-25 14:20:29 +02:00
e . preventDefault ( ) ;
2023-10-27 22:03:21 +08:00
this . props . history . push ( ` /@ ${ this . props . status . getIn ( [ 'account' , 'acct' ] ) } ` ) ;
2016-09-25 14:20:29 +02:00
}
e . stopPropagation ( ) ;
2023-01-29 19:45:35 -05:00
} ;
2016-09-25 14:20:29 +02:00
2020-12-07 04:29:37 +01:00
handleOpenVideo = ( options ) => {
this . props . onOpenVideo ( this . props . status . getIn ( [ 'media_attachments' , 0 ] ) , options ) ;
2023-01-29 19:45:35 -05:00
} ;
2017-09-14 03:39:10 +02:00
2018-03-11 09:52:59 +01:00
handleExpandedToggle = ( ) => {
this . props . onToggleHidden ( this . props . status ) ;
2023-01-29 19:45:35 -05:00
} ;
2018-03-11 09:52:59 +01:00
2019-01-16 19:47:46 +01:00
_measureHeight ( heightJustChanged ) {
if ( this . props . measureHeight && this . node ) {
2019-01-18 20:58:11 +01:00
scheduleIdleTask ( ( ) => this . node && this . setState ( { height : Math . ceil ( this . node . scrollHeight ) + 1 } ) ) ;
2019-01-16 19:47:46 +01:00
if ( this . props . onHeightChange && heightJustChanged ) {
this . props . onHeightChange ( ) ;
}
}
}
setRef = c => {
this . node = c ;
this . _measureHeight ( ) ;
2023-01-29 19:45:35 -05:00
} ;
2019-01-16 19:47:46 +01:00
componentDidUpdate ( prevProps , prevState ) {
this . _measureHeight ( prevState . height !== this . state . height ) ;
}
handleModalLink = e => {
e . preventDefault ( ) ;
let href ;
if ( e . target . nodeName !== 'A' ) {
href = e . target . parentNode . href ;
} else {
href = e . target . href ;
}
window . open ( href , 'mastodon-intent' , 'width=445,height=600,resizable=no,menubar=no,status=no,scrollbars=yes' ) ;
2023-01-29 19:45:35 -05:00
} ;
2019-01-16 19:47:46 +01:00
2022-09-23 23:00:12 +02:00
handleTranslate = ( ) => {
const { onTranslate , status } = this . props ;
onTranslate ( status ) ;
2023-01-29 19:45:35 -05:00
} ;
2022-09-23 23:00:12 +02:00
2023-07-24 13:46:55 +02:00
_properStatus ( ) {
const { status } = this . props ;
if ( status . get ( 'reblog' , null ) !== null && typeof status . get ( 'reblog' ) === 'object' ) {
return status . get ( 'reblog' ) ;
} else {
return status ;
}
}
getAttachmentAspectRatio ( ) {
const attachments = this . _properStatus ( ) . get ( 'media_attachments' ) ;
if ( attachments . getIn ( [ 0 , 'type' ] ) === 'video' ) {
return ` ${ attachments . getIn ( [ 0 , 'meta' , 'original' , 'width' ] ) } / ${ attachments . getIn ( [ 0 , 'meta' , 'original' , 'height' ] ) } ` ;
} else if ( attachments . getIn ( [ 0 , 'type' ] ) === 'audio' ) {
return '16 / 9' ;
} else {
2023-10-09 13:38:29 +02:00
return ( attachments . size === 1 && attachments . getIn ( [ 0 , 'meta' , 'small' , 'aspect' ] ) ) ? attachments . getIn ( [ 0 , 'meta' , 'small' , 'aspect' ] ) : '3 / 2' ;
2023-07-24 13:46:55 +02:00
}
}
2016-09-25 14:20:29 +02:00
render ( ) {
2023-07-24 13:46:55 +02:00
const status = this . _properStatus ( ) ;
2019-01-16 19:47:46 +01:00
const outerStyle = { boxSizing : 'border-box' } ;
2023-10-24 19:45:08 +02:00
const { compact , pictureInPicture } = this . props ;
2019-01-16 19:47:46 +01:00
if ( ! status ) {
return null ;
}
2017-01-15 14:01:33 +01:00
let media = '' ;
let applicationLink = '' ;
2017-10-16 20:10:12 +08:00
let reblogLink = '' ;
2019-01-16 19:47:46 +01:00
let favouriteLink = '' ;
if ( this . props . measureHeight ) {
outerStyle . height = ` ${ this . state . height } px ` ;
}
2016-09-25 14:58:07 +02:00
2023-06-01 00:10:21 +02:00
const language = status . getIn ( [ 'translation' , 'language' ] ) || status . get ( 'language' ) ;
2020-12-07 19:36:36 +01:00
if ( pictureInPicture . get ( 'inUse' ) ) {
2023-07-24 13:46:55 +02:00
media = < PictureInPicturePlaceholder aspectRatio = { this . getAttachmentAspectRatio ( ) } / > ;
2020-09-28 13:29:43 +02:00
} else if ( status . get ( 'media_attachments' ) . size > 0 ) {
2019-08-23 22:38:02 +02:00
if ( status . getIn ( [ 'media_attachments' , 0 , 'type' ] ) === 'audio' ) {
const attachment = status . getIn ( [ 'media_attachments' , 0 ] ) ;
2023-06-01 00:10:21 +02:00
const description = attachment . getIn ( [ 'translation' , 'description' ] ) || attachment . get ( 'description' ) ;
2019-08-23 22:38:02 +02:00
media = (
< Audio
src = { attachment . get ( 'url' ) }
2023-06-01 00:10:21 +02:00
alt = { description }
lang = { language }
2019-08-23 22:38:02 +02:00
duration = { attachment . getIn ( [ 'meta' , 'original' , 'duration' ] , 0 ) }
2020-06-29 13:56:55 +02:00
poster = { attachment . get ( 'preview_url' ) || status . getIn ( [ 'account' , 'avatar_static' ] ) }
2020-07-05 18:28:25 +02:00
backgroundColor = { attachment . getIn ( [ 'meta' , 'colors' , 'background' ] ) }
foregroundColor = { attachment . getIn ( [ 'meta' , 'colors' , 'foreground' ] ) }
accentColor = { attachment . getIn ( [ 'meta' , 'colors' , 'accent' ] ) }
2022-08-13 15:39:05 +02:00
sensitive = { status . get ( 'sensitive' ) }
visible = { this . props . showMedia }
blurhash = { attachment . get ( 'blurhash' ) }
2020-06-23 12:20:14 +02:00
height = { 150 }
2022-08-13 15:39:05 +02:00
onToggleVisibility = { this . props . onToggleMediaVisibility }
2019-08-23 22:38:02 +02:00
/ >
) ;
} else if ( status . getIn ( [ 'media_attachments' , 0 , 'type' ] ) === 'video' ) {
2019-06-19 23:42:38 +02:00
const attachment = status . getIn ( [ 'media_attachments' , 0 ] ) ;
2023-06-01 00:10:21 +02:00
const description = attachment . getIn ( [ 'translation' , 'description' ] ) || attachment . get ( 'description' ) ;
2017-09-14 03:39:10 +02:00
media = (
< Video
2019-06-19 23:42:38 +02:00
preview = { attachment . get ( 'preview_url' ) }
2020-11-21 23:19:04 +01:00
frameRate = { attachment . getIn ( [ 'meta' , 'original' , 'frame_rate' ] ) }
2023-07-24 13:46:55 +02:00
aspectRatio = { ` ${ attachment . getIn ( [ 'meta' , 'original' , 'width' ] ) } / ${ attachment . getIn ( [ 'meta' , 'original' , 'height' ] ) } ` }
2019-06-19 23:42:38 +02:00
blurhash = { attachment . get ( 'blurhash' ) }
src = { attachment . get ( 'url' ) }
2023-06-01 00:10:21 +02:00
alt = { description }
lang = { language }
2017-09-14 03:39:10 +02:00
width = { 300 }
height = { 150 }
onOpenVideo = { this . handleOpenVideo }
sensitive = { status . get ( 'sensitive' ) }
2019-05-25 23:20:51 +02:00
visible = { this . props . showMedia }
onToggleVisibility = { this . props . onToggleMediaVisibility }
2017-09-14 03:39:10 +02:00
/ >
) ;
2016-09-25 14:58:07 +02:00
} else {
2017-09-24 05:58:30 +02:00
media = (
< MediaGallery
standalone
sensitive = { status . get ( 'sensitive' ) }
media = { status . get ( 'media_attachments' ) }
2023-06-01 00:10:21 +02:00
lang = { language }
2017-09-24 05:58:30 +02:00
height = { 300 }
onOpenMedia = { this . props . onOpenMedia }
2019-05-25 23:20:51 +02:00
visible = { this . props . showMedia }
onToggleVisibility = { this . props . onToggleMediaVisibility }
2017-09-24 05:58:30 +02:00
/ >
) ;
2016-09-25 14:58:07 +02:00
}
2017-04-19 08:28:00 -04:00
} else if ( status . get ( 'spoiler_text' ) . length === 0 ) {
2020-06-06 17:41:56 +02:00
media = < Card sensitive = { status . get ( 'sensitive' ) } onOpenMedia = { this . props . onOpenMedia } card = { status . get ( 'card' , null ) } / > ;
2016-09-25 14:58:07 +02:00
}
2016-09-25 14:20:29 +02:00
2017-01-15 14:01:33 +01:00
if ( status . get ( 'application' ) ) {
2024-03-14 10:18:24 +01:00
applicationLink = < > · < a className = 'detailed-status__application' href = { status . getIn ( [ 'application' , 'website' ] ) } target = '_blank' rel = 'noopener noreferrer' > { status . getIn ( [ 'application' , 'name' ] ) } < / a > < / > ;
2017-01-15 14:01:33 +01:00
}
2024-03-14 10:18:24 +01:00
const visibilityLink = < > · < VisibilityIcon visibility = { status . get ( 'visibility' ) } / > < / > ;
2017-10-16 20:10:12 +08:00
2020-03-10 18:39:47 +01:00
if ( [ 'private' , 'direct' ] . includes ( status . get ( 'visibility' ) ) ) {
2020-06-26 05:43:59 +09:00
reblogLink = '' ;
2023-10-19 19:44:55 +02:00
} else if ( this . props . history ) {
2019-01-16 19:47:46 +01:00
reblogLink = (
2024-03-14 10:18:24 +01:00
< Link to = { ` /@ ${ status . getIn ( [ 'account' , 'acct' ] ) } / ${ status . get ( 'id' ) } /reblogs ` } className = 'detailed-status__link' >
< span className = 'detailed-status__reblogs' >
< AnimatedNumber value = { status . get ( 'reblogs_count' ) } / >
< / span >
< FormattedMessage id = 'status.reblogs' defaultMessage = '{count, plural, one {boost} other {boosts}}' values = { { count : status . get ( 'reblogs_count' ) } } / >
< / Link >
2019-01-16 19:47:46 +01:00
) ;
} else {
reblogLink = (
2024-03-14 10:18:24 +01:00
< a href = { ` /interact/ ${ status . get ( 'id' ) } ?type=reblog ` } className = 'detailed-status__link' onClick = { this . handleModalLink } >
< span className = 'detailed-status__reblogs' >
< AnimatedNumber value = { status . get ( 'reblogs_count' ) } / >
< / span >
< FormattedMessage id = 'status.reblogs' defaultMessage = '{count, plural, one {boost} other {boosts}}' values = { { count : status . get ( 'reblogs_count' ) } } / >
< / a >
2019-01-16 19:47:46 +01:00
) ;
}
2023-10-19 19:44:55 +02:00
if ( this . props . history ) {
2019-01-16 19:47:46 +01:00
favouriteLink = (
2021-09-26 05:46:13 +02:00
< Link to = { ` /@ ${ status . getIn ( [ 'account' , 'acct' ] ) } / ${ status . get ( 'id' ) } /favourites ` } className = 'detailed-status__link' >
2019-01-16 19:47:46 +01:00
< span className = 'detailed-status__favorites' >
2020-01-25 05:23:05 +01:00
< AnimatedNumber value = { status . get ( 'favourites_count' ) } / >
2019-01-16 19:47:46 +01:00
< / span >
2024-03-14 10:18:24 +01:00
< FormattedMessage id = 'status.favourites' defaultMessage = '{count, plural, one {favorite} other {favorites}}' values = { { count : status . get ( 'favourites_count' ) } } / >
2019-01-16 19:47:46 +01:00
< / Link >
) ;
2017-10-16 20:10:12 +08:00
} else {
2019-01-16 19:47:46 +01:00
favouriteLink = (
< a href = { ` /interact/ ${ status . get ( 'id' ) } ?type=favourite ` } className = 'detailed-status__link' onClick = { this . handleModalLink } >
< span className = 'detailed-status__favorites' >
2020-01-25 05:23:05 +01:00
< AnimatedNumber value = { status . get ( 'favourites_count' ) } / >
2019-01-16 19:47:46 +01:00
< / span >
2024-03-14 10:18:24 +01:00
< FormattedMessage id = 'status.favourites' defaultMessage = '{count, plural, one {favorite} other {favorites}}' values = { { count : status . get ( 'favourites_count' ) } } / >
2019-01-16 19:47:46 +01:00
< / a >
) ;
2017-10-16 20:10:12 +08:00
}
2023-08-21 19:39:01 +02:00
const { statusContentProps , hashtagBar } = getHashtagBarForStatus ( status ) ;
2023-09-19 12:25:39 +02:00
const expanded = ! status . get ( 'hidden' ) || status . get ( 'spoiler_text' ) . length === 0 ;
2023-08-21 19:39:01 +02:00
2016-09-25 14:20:29 +02:00
return (
2019-01-17 14:06:08 +01:00
< div style = { outerStyle } >
2023-03-30 15:16:20 +02:00
< div ref = { this . setRef } className = { classNames ( 'detailed-status' , { compact } ) } >
{ status . get ( 'visibility' ) === 'direct' && (
< div className = 'status__prepend' >
2023-10-24 19:45:08 +02:00
< div className = 'status__prepend-icon-wrapper' > < Icon id = 'at' icon = { AlternateEmailIcon } className = 'status__prepend-icon' / > < / div >
2023-03-30 15:16:20 +02:00
< FormattedMessage id = 'status.direct_indicator' defaultMessage = 'Private mention' / >
< / div >
) }
2024-06-26 21:33:38 +02:00
< a href = { ` /@ ${ status . getIn ( [ 'account' , 'acct' ] ) } ` } data - hover - card - account = { status . getIn ( [ 'account' , 'id' ] ) } onClick = { this . handleAccountClick } className = 'detailed-status__display-name' >
2022-10-25 19:02:21 +02:00
< div className = 'detailed-status__display-avatar' > < Avatar account = { status . get ( 'account' ) } size = { 46 } / > < / div >
2019-01-17 14:06:08 +01:00
< DisplayName account = { status . get ( 'account' ) } localDomain = { this . props . domain } / >
< / a >
2024-08-22 19:12:35 +02:00
{ status . get ( 'spoiler_text' ) . length > 0 && < ContentWarning text = { status . getIn ( [ 'translation' , 'spoilerHtml' ] ) || status . get ( 'spoilerHtml' ) } expanded = { expanded } onClick = { this . handleExpandedToggle } / > }
2019-01-17 14:06:08 +01:00
2024-08-22 19:12:35 +02:00
{ expanded && (
< >
< StatusContent
status = { status }
onTranslate = { this . handleTranslate }
{ ... statusContentProps }
/ >
2019-01-17 14:06:08 +01:00
2024-08-22 19:12:35 +02:00
{ media }
{ hashtagBar }
< / >
) }
2023-08-14 23:42:30 +02:00
2019-01-17 14:06:08 +01:00
< div className = 'detailed-status__meta' >
2024-03-14 10:18:24 +01:00
< div className = 'detailed-status__meta__line' >
< a className = 'detailed-status__datetime' href = { ` /@ ${ status . getIn ( [ 'account' , 'acct' ] ) } / ${ status . get ( 'id' ) } ` } target = '_blank' rel = 'noopener noreferrer' >
< FormattedDate value = { new Date ( status . get ( 'created_at' ) ) } year = 'numeric' month = 'short' day = '2-digit' hour = '2-digit' minute = '2-digit' / >
< / a >
{ visibilityLink }
{ applicationLink }
< / div >
{ status . get ( 'edited_at' ) && < div className = 'detailed-status__meta__line' > < EditedTimestamp statusId = { status . get ( 'id' ) } timestamp = { status . get ( 'edited_at' ) } / > < / div > }
< div className = 'detailed-status__meta__line' >
{ reblogLink }
2024-03-16 02:11:59 +01:00
{ reblogLink && < > · < / > }
2024-03-14 10:18:24 +01:00
{ favouriteLink }
< / div >
2019-01-17 14:06:08 +01:00
< / div >
2016-09-25 14:20:29 +02:00
< / div >
< / div >
) ;
}
2017-04-22 03:05:35 +09:00
}
2023-03-24 11:17:53 +09:00
2023-10-24 19:45:08 +02:00
export default withRouter ( DetailedStatus ) ;