2017-04-21 18:05:35 +00:00
import PropTypes from 'prop-types' ;
2023-05-23 15:15:17 +00:00
2020-06-25 20:43:59 +00:00
import { injectIntl , defineMessages , FormattedMessage } from 'react-intl' ;
2023-05-23 15:15:17 +00:00
import classNames from 'classnames' ;
import ImmutablePropTypes from 'react-immutable-proptypes' ;
2017-05-03 00:04:16 +00:00
import ImmutablePureComponent from 'react-immutable-pure-component' ;
2023-05-23 15:15:17 +00:00
2017-10-05 23:07:59 +00:00
import { HotKeys } from 'react-hotkeys' ;
2023-05-23 15:15:17 +00:00
2024-01-16 10:27:26 +00:00
import AlternateEmailIcon from '@/material-icons/400-24px/alternate_email.svg?react' ;
import PushPinIcon from '@/material-icons/400-24px/push_pin.svg?react' ;
import RepeatIcon from '@/material-icons/400-24px/repeat.svg?react' ;
import ReplyIcon from '@/material-icons/400-24px/reply.svg?react' ;
2024-08-22 17:12:35 +00:00
import { ContentWarning } from 'mastodon/components/content_warning' ;
import { FilterWarning } from 'mastodon/components/filter_warning' ;
2023-05-09 01:11:56 +00:00
import { Icon } from 'mastodon/components/icon' ;
2020-09-28 11:29:43 +00:00
import PictureInPicturePlaceholder from 'mastodon/components/picture_in_picture_placeholder' ;
2023-10-23 09:39:53 +00:00
import { withOptionalRouter , WithOptionalRouterPropTypes } from 'mastodon/utils/react_router' ;
2017-07-07 22:06:02 +00:00
2023-05-23 15:15:17 +00:00
import Card from '../features/status/components/card' ;
2017-07-07 22:06:02 +00:00
// We use the component (and not the container) since we do not want
// to use the progress bar to show download progress
import Bundle from '../features/ui/components/bundle' ;
2023-05-23 15:15:17 +00:00
import { MediaGallery , Video , Audio } from '../features/ui/util/async-components' ;
2024-03-13 16:47:48 +00:00
import { SensitiveMediaContext } from '../features/ui/util/sensitive_media_context' ;
2023-05-23 15:15:17 +00:00
import { displayMedia } from '../initial_state' ;
import { Avatar } from './avatar' ;
import { AvatarOverlay } from './avatar_overlay' ;
import { DisplayName } from './display_name' ;
2023-08-21 17:39:01 +00:00
import { getHashtagBarForStatus } from './hashtag_bar' ;
2023-05-23 15:15:17 +00:00
import { RelativeTimestamp } from './relative_timestamp' ;
import StatusActionBar from './status_action_bar' ;
import StatusContent from './status_content' ;
2023-10-24 17:45:08 +00:00
import { VisibilityIcon } from './visibility_icon' ;
2016-08-24 15:56:44 +00:00
2023-05-31 22:10:21 +00:00
const domParser = new DOMParser ( ) ;
2018-08-26 15:53:26 +00:00
export const textForScreenReader = ( intl , status , rebloggedByText = false ) => {
2018-08-23 18:56:57 +00:00
const displayName = status . getIn ( [ 'account' , 'display_name' ] ) ;
2023-05-31 22:10:21 +00:00
const spoilerText = status . getIn ( [ 'translation' , 'spoiler_text' ] ) || status . get ( 'spoiler_text' ) ;
const contentHtml = status . getIn ( [ 'translation' , 'contentHtml' ] ) || status . get ( 'contentHtml' ) ;
const contentText = domParser . parseFromString ( contentHtml , 'text/html' ) . documentElement . textContent ;
2018-08-23 18:56:57 +00:00
const values = [
displayName . length === 0 ? status . getIn ( [ 'account' , 'acct' ] ) . split ( '@' ) [ 0 ] : displayName ,
2023-05-31 22:10:21 +00:00
spoilerText && status . get ( 'hidden' ) ? spoilerText : contentText ,
2018-08-23 18:56:57 +00:00
intl . formatDate ( status . get ( 'created_at' ) , { hour : '2-digit' , minute : '2-digit' , month : 'short' , day : 'numeric' } ) ,
status . getIn ( [ 'account' , 'acct' ] ) ,
] ;
if ( rebloggedByText ) {
values . push ( rebloggedByText ) ;
}
return values . join ( ', ' ) ;
} ;
2019-05-26 11:48:16 +00:00
export const defaultMediaVisibility = ( status ) => {
if ( ! status ) {
return undefined ;
}
if ( status . get ( 'reblog' , null ) !== null && typeof status . get ( 'reblog' ) === 'object' ) {
status = status . get ( 'reblog' ) ;
}
return ( displayMedia !== 'hide_all' && ! status . get ( 'sensitive' ) || displayMedia === 'show_all' ) ;
} ;
2020-06-25 20:43:59 +00:00
const messages = defineMessages ( {
public _short : { id : 'privacy.public.short' , defaultMessage : 'Public' } ,
2024-01-25 15:41:31 +00:00
unlisted _short : { id : 'privacy.unlisted.short' , defaultMessage : 'Quiet public' } ,
private _short : { id : 'privacy.private.short' , defaultMessage : 'Followers' } ,
direct _short : { id : 'privacy.direct.short' , defaultMessage : 'Specific people' } ,
2022-01-19 21:37:27 +00:00
edited : { id : 'status.edited' , defaultMessage : 'Edited {date}' } ,
2020-06-25 20:43:59 +00:00
} ) ;
2018-09-14 15:59:48 +00:00
class Status extends ImmutablePureComponent {
2017-04-21 18:05:35 +00:00
2024-03-13 16:47:48 +00:00
static contextType = SensitiveMediaContext ;
2017-05-12 12:44:10 +00:00
static propTypes = {
status : ImmutablePropTypes . map ,
2023-11-03 15:00:03 +00:00
account : ImmutablePropTypes . record ,
2023-04-24 06:07:03 +00:00
previousId : PropTypes . string ,
nextInReplyToId : PropTypes . string ,
rootId : PropTypes . string ,
2018-10-20 00:23:58 +00:00
onClick : PropTypes . func ,
2017-05-12 12:44:10 +00:00
onReply : PropTypes . func ,
onFavourite : PropTypes . func ,
onReblog : PropTypes . func ,
onDelete : PropTypes . func ,
2018-04-09 15:09:11 +00:00
onDirect : PropTypes . func ,
onMention : PropTypes . func ,
2017-08-24 23:41:18 +00:00
onPin : PropTypes . func ,
2017-05-12 12:44:10 +00:00
onOpenMedia : PropTypes . func ,
onOpenVideo : PropTypes . func ,
onBlock : PropTypes . func ,
2022-08-25 02:27:47 +00:00
onAddFilter : PropTypes . func ,
2017-09-01 19:30:13 +00:00
onEmbed : PropTypes . func ,
2017-08-07 18:32:03 +00:00
onHeightChange : PropTypes . func ,
2018-03-11 08:52:59 +00:00
onToggleHidden : PropTypes . func ,
Summary: fix slowness due to layout thrashing when reloading a large … (#12661)
* Summary: fix slowness due to layout thrashing when reloading a large set of status updates
in order to limit the maximum size of a status in a list view (e.g. the home timeline), so as to avoid having to scroll all the way through an abnormally large status update (see https://github.com/tootsuite/mastodon/pull/8205), the following steps are taken:
•the element containing the status is rendered in the browser
•its height is calculated, to determine if it exceeds the maximum height threshold.
Unfortunately for performance, these steps are carried out in the componentDidMount(/Update) method, which also performs style modifications on the element. The combination of height request and style modification during javascript evaluation in the browser leads to layout-thrashing, where the elements are repeatedly re-laid-out (see https://developers.google.com/web/fundamentals/performance/rendering/avoid-large-complex-layouts-and-layout-thrashing & https://developer.mozilla.org/en-US/docs/Mozilla/Firefox/Performance_best_practices_for_Firefox_fe_engineers).
The solution implemented here is to memoize the collapsed state in Redux the first time the status is seen (e.g. when fetched as part of a small batch, to populate the home timeline) , so that on subsequent re-renders, the value can be queried, rather than recalculated. This strategy is derived from https://github.com/tootsuite/mastodon/pull/4439 & https://github.com/tootsuite/mastodon/pull/4909, and should resolve https://github.com/tootsuite/mastodon/issues/12455.
Andrew Lin (https://github.com/onethreeseven) is thanked for his assistance in root cause analysis and solution brainstorming
* remove getSnapshotBeforeUpdate from status
* remove componentWillUnmount from status
* persist last-intersected status update and restore when ScrollableList is restored
e.g. when navigating from home-timeline to a status conversational thread and <Back again
* cache currently-viewing status id to avoid calling redux with identical value
* refactor collapse toggle to pass explicit boolean
2019-12-29 04:39:48 +00:00
onToggleCollapsed : PropTypes . func ,
2022-09-23 21:00:12 +00:00
onTranslate : PropTypes . func ,
2022-10-07 08:14:31 +00:00
onInteractionModal : PropTypes . func ,
2017-05-20 15:31:47 +00:00
muted : PropTypes . bool ,
2017-08-28 20:23:44 +00:00
hidden : PropTypes . bool ,
2018-10-20 00:23:58 +00:00
unread : PropTypes . bool ,
2017-10-05 23:07:59 +00:00
onMoveUp : PropTypes . func ,
onMoveDown : PropTypes . func ,
2018-11-08 20:08:57 +00:00
showThread : PropTypes . bool ,
2019-02-11 12:19:59 +00:00
getScrollPosition : PropTypes . func ,
updateScrollBottom : PropTypes . func ,
cacheMediaWidth : PropTypes . func ,
cachedMediaWidth : PropTypes . number ,
2020-07-09 13:09:19 +00:00
scrollKey : PropTypes . string ,
2024-07-18 14:36:09 +00:00
skipPrepend : PropTypes . bool ,
avatarSize : PropTypes . number ,
2020-09-28 11:29:43 +00:00
deployPictureInPicture : PropTypes . func ,
2024-07-23 06:20:17 +00:00
unfocusable : PropTypes . bool ,
2020-12-07 18:36:36 +00:00
pictureInPicture : ImmutablePropTypes . contains ( {
2020-11-16 04:16:39 +00:00
inUse : PropTypes . bool ,
available : PropTypes . bool ,
} ) ,
2023-10-23 09:39:53 +00:00
... WithOptionalRouterPropTypes ,
2017-05-12 12:44:10 +00:00
} ;
2017-05-26 12:05:52 +00:00
// Avoid checking props that are functions (and whose equality will always
// evaluate to false. See react-immutable-pure-component for usage.
updateOnProps = [
'status' ,
'account' ,
'muted' ,
2017-08-28 20:23:44 +00:00
'hidden' ,
2020-09-26 18:57:07 +00:00
'unread' ,
2020-11-16 04:16:39 +00:00
'pictureInPicture' ,
2019-01-16 18:47:46 +00:00
] ;
2017-05-26 12:05:52 +00:00
2019-05-25 21:20:51 +00:00
state = {
2024-03-13 16:47:48 +00:00
showMedia : defaultMediaVisibility ( this . props . status ) && ! ( this . context ? . hideMediaByDefault ) ,
2024-08-22 17:12:35 +00:00
showDespiteFilter : undefined ,
2019-05-25 21:20:51 +00:00
} ;
2024-03-13 16:47:48 +00:00
componentDidUpdate ( prevProps ) {
// This will potentially cause a wasteful redraw, but in most cases `Status` components are used
// with a `key` directly depending on their `id`, preventing re-use of the component across
// different IDs.
// But just in case this does change, reset the state on status change.
if ( this . props . status ? . get ( 'id' ) !== prevProps . status ? . get ( 'id' ) ) {
this . setState ( {
showMedia : defaultMediaVisibility ( this . props . status ) && ! ( this . context ? . hideMediaByDefault ) ,
2024-08-22 17:12:35 +00:00
showDespiteFilter : undefined ,
2024-03-13 16:47:48 +00:00
} ) ;
2019-05-26 11:48:16 +00:00
}
}
2019-05-25 21:20:51 +00:00
handleToggleMediaVisibility = ( ) => {
this . setState ( { showMedia : ! this . state . showMedia } ) ;
2023-01-30 00:45:35 +00:00
} ;
2019-05-25 21:20:51 +00:00
2021-09-26 03:46:13 +00:00
handleClick = e => {
if ( e && ( e . button !== 0 || e . ctrlKey || e . metaKey ) ) {
2018-10-20 00:23:58 +00:00
return ;
}
2021-09-26 03:46:13 +00:00
if ( e ) {
e . preventDefault ( ) ;
2017-07-11 13:27:59 +00:00
}
2021-09-26 03:46:13 +00:00
this . handleHotkeyOpen ( ) ;
2023-01-30 00:45:35 +00:00
} ;
2016-09-15 22:21:51 +00:00
2021-11-26 21:04:09 +00:00
handlePrependAccountClick = e => {
this . handleAccountClick ( e , false ) ;
2023-01-30 00:45:35 +00:00
} ;
2021-11-26 21:04:09 +00:00
handleAccountClick = ( e , proper = true ) => {
2021-09-26 03:46:13 +00:00
if ( e && ( e . button !== 0 || e . ctrlKey || e . metaKey ) ) {
2019-06-10 17:27:10 +00:00
return ;
}
2021-09-26 03:46:13 +00:00
if ( e ) {
2016-09-15 22:21:51 +00:00
e . preventDefault ( ) ;
2023-02-20 07:11:23 +00:00
e . stopPropagation ( ) ;
2016-09-15 22:21:51 +00:00
}
2021-09-26 03:46:13 +00:00
2021-11-26 21:04:09 +00:00
this . _openProfile ( proper ) ;
2023-01-30 00:45:35 +00:00
} ;
2016-09-13 00:24:40 +00:00
2017-06-13 18:46:21 +00:00
handleExpandedToggle = ( ) => {
2018-03-11 08:52:59 +00:00
this . props . onToggleHidden ( this . _properStatus ( ) ) ;
2023-01-30 00:45:35 +00:00
} ;
Summary: fix slowness due to layout thrashing when reloading a large … (#12661)
* Summary: fix slowness due to layout thrashing when reloading a large set of status updates
in order to limit the maximum size of a status in a list view (e.g. the home timeline), so as to avoid having to scroll all the way through an abnormally large status update (see https://github.com/tootsuite/mastodon/pull/8205), the following steps are taken:
•the element containing the status is rendered in the browser
•its height is calculated, to determine if it exceeds the maximum height threshold.
Unfortunately for performance, these steps are carried out in the componentDidMount(/Update) method, which also performs style modifications on the element. The combination of height request and style modification during javascript evaluation in the browser leads to layout-thrashing, where the elements are repeatedly re-laid-out (see https://developers.google.com/web/fundamentals/performance/rendering/avoid-large-complex-layouts-and-layout-thrashing & https://developer.mozilla.org/en-US/docs/Mozilla/Firefox/Performance_best_practices_for_Firefox_fe_engineers).
The solution implemented here is to memoize the collapsed state in Redux the first time the status is seen (e.g. when fetched as part of a small batch, to populate the home timeline) , so that on subsequent re-renders, the value can be queried, rather than recalculated. This strategy is derived from https://github.com/tootsuite/mastodon/pull/4439 & https://github.com/tootsuite/mastodon/pull/4909, and should resolve https://github.com/tootsuite/mastodon/issues/12455.
Andrew Lin (https://github.com/onethreeseven) is thanked for his assistance in root cause analysis and solution brainstorming
* remove getSnapshotBeforeUpdate from status
* remove componentWillUnmount from status
* persist last-intersected status update and restore when ScrollableList is restored
e.g. when navigating from home-timeline to a status conversational thread and <Back again
* cache currently-viewing status id to avoid calling redux with identical value
* refactor collapse toggle to pass explicit boolean
2019-12-29 04:39:48 +00:00
handleCollapsedToggle = isCollapsed => {
this . props . onToggleCollapsed ( this . _properStatus ( ) , isCollapsed ) ;
2023-01-30 00:45:35 +00:00
} ;
2017-06-13 18:46:21 +00:00
2022-09-23 21:00:12 +00:00
handleTranslate = ( ) => {
this . props . onTranslate ( this . _properStatus ( ) ) ;
2023-01-30 00:45:35 +00:00
} ;
2022-09-23 21:00:12 +00:00
2023-07-24 11:46:55 +00:00
getAttachmentAspectRatio ( ) {
const attachments = this . _properStatus ( ) . get ( 'media_attachments' ) ;
2017-07-07 22:06:02 +00:00
2023-07-24 11:46:55 +00:00
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 11:38:29 +00:00
return ( attachments . size === 1 && attachments . getIn ( [ 0 , 'meta' , 'small' , 'aspect' ] ) ) ? attachments . getIn ( [ 0 , 'meta' , 'small' , 'aspect' ] ) : '3 / 2' ;
2023-07-24 11:46:55 +00:00
}
2019-08-23 20:38:02 +00:00
}
2023-07-24 11:46:55 +00:00
renderLoadingMediaGallery = ( ) => {
return (
< div className = 'media-gallery' style = { { aspectRatio : this . getAttachmentAspectRatio ( ) } } / >
) ;
} ;
renderLoadingVideoPlayer = ( ) => {
return (
< div className = 'video-player' style = { { aspectRatio : this . getAttachmentAspectRatio ( ) } } / >
) ;
} ;
renderLoadingAudioPlayer = ( ) => {
return (
< div className = 'audio-player' style = { { aspectRatio : this . getAttachmentAspectRatio ( ) } } / >
) ;
} ;
2017-07-07 22:06:02 +00:00
2020-12-07 03:29:37 +00:00
handleOpenVideo = ( options ) => {
const status = this . _properStatus ( ) ;
2023-05-31 22:10:21 +00:00
const lang = status . getIn ( [ 'translation' , 'language' ] ) || status . get ( 'language' ) ;
this . props . onOpenVideo ( status . get ( 'id' ) , status . getIn ( [ 'media_attachments' , 0 ] ) , lang , options ) ;
2023-01-30 00:45:35 +00:00
} ;
2020-11-27 02:24:11 +00:00
handleOpenMedia = ( media , index ) => {
2023-05-11 10:41:55 +00:00
const status = this . _properStatus ( ) ;
2023-05-31 22:10:21 +00:00
const lang = status . getIn ( [ 'translation' , 'language' ] ) || status . get ( 'language' ) ;
this . props . onOpenMedia ( status . get ( 'id' ) , media , index , lang ) ;
2023-01-30 00:45:35 +00:00
} ;
2017-10-05 23:07:59 +00:00
2019-11-29 16:02:36 +00:00
handleHotkeyOpenMedia = e => {
2019-12-04 23:50:51 +00:00
const { onOpenMedia , onOpenVideo } = this . props ;
2020-12-08 23:24:13 +00:00
const status = this . _properStatus ( ) ;
2019-11-29 16:02:36 +00:00
e . preventDefault ( ) ;
if ( status . get ( 'media_attachments' ) . size > 0 ) {
2023-05-31 22:10:21 +00:00
const lang = status . getIn ( [ 'translation' , 'language' ] ) || status . get ( 'language' ) ;
2020-11-27 02:24:11 +00:00
if ( status . getIn ( [ 'media_attachments' , 0 , 'type' ] ) === 'video' ) {
2023-05-11 10:41:55 +00:00
onOpenVideo ( status . get ( 'id' ) , status . getIn ( [ 'media_attachments' , 0 ] ) , lang , { startTime : 0 } ) ;
2019-11-29 16:02:36 +00:00
} else {
2023-05-11 10:41:55 +00:00
onOpenMedia ( status . get ( 'id' ) , status . get ( 'media_attachments' ) , 0 , lang ) ;
2019-11-29 16:02:36 +00:00
}
}
2023-01-30 00:45:35 +00:00
} ;
2019-11-29 16:02:36 +00:00
2020-09-28 11:29:43 +00:00
handleDeployPictureInPicture = ( type , mediaProps ) => {
const { deployPictureInPicture } = this . props ;
const status = this . _properStatus ( ) ;
deployPictureInPicture ( status , type , mediaProps ) ;
2023-01-30 00:45:35 +00:00
} ;
2020-09-28 11:29:43 +00:00
2017-10-05 23:07:59 +00:00
handleHotkeyReply = e => {
e . preventDefault ( ) ;
2024-07-19 15:26:44 +00:00
this . props . onReply ( this . _properStatus ( ) ) ;
2023-01-30 00:45:35 +00:00
} ;
2017-10-05 23:07:59 +00:00
handleHotkeyFavourite = ( ) => {
this . props . onFavourite ( this . _properStatus ( ) ) ;
2023-01-30 00:45:35 +00:00
} ;
2017-10-05 23:07:59 +00:00
handleHotkeyBoost = e => {
this . props . onReblog ( this . _properStatus ( ) , e ) ;
2023-01-30 00:45:35 +00:00
} ;
2017-10-05 23:07:59 +00:00
handleHotkeyMention = e => {
e . preventDefault ( ) ;
2024-07-19 15:26:44 +00:00
this . props . onMention ( this . _properStatus ( ) . get ( 'account' ) ) ;
2023-01-30 00:45:35 +00:00
} ;
2017-10-05 23:07:59 +00:00
handleHotkeyOpen = ( ) => {
2021-09-26 03:46:13 +00:00
if ( this . props . onClick ) {
this . props . onClick ( ) ;
return ;
}
2023-10-19 17:44:55 +00:00
const { history } = this . props ;
2021-09-26 03:46:13 +00:00
const status = this . _properStatus ( ) ;
2023-10-19 17:44:55 +00:00
if ( ! history ) {
2021-09-26 03:46:13 +00:00
return ;
}
2023-10-19 17:44:55 +00:00
history . push ( ` /@ ${ status . getIn ( [ 'account' , 'acct' ] ) } / ${ status . get ( 'id' ) } ` ) ;
2023-01-30 00:45:35 +00:00
} ;
2017-10-05 23:07:59 +00:00
handleHotkeyOpenProfile = ( ) => {
2021-11-26 21:04:09 +00:00
this . _openProfile ( ) ;
2023-01-30 00:45:35 +00:00
} ;
2021-11-26 21:04:09 +00:00
_openProfile = ( proper = true ) => {
2023-10-19 17:44:55 +00:00
const { history } = this . props ;
2021-11-26 21:04:09 +00:00
const status = proper ? this . _properStatus ( ) : this . props . status ;
2021-09-26 03:46:13 +00:00
2023-10-19 17:44:55 +00:00
if ( ! history ) {
2021-09-26 03:46:13 +00:00
return ;
}
2023-10-19 17:44:55 +00:00
history . push ( ` /@ ${ status . getIn ( [ 'account' , 'acct' ] ) } ` ) ;
2023-01-30 00:45:35 +00:00
} ;
2017-10-05 23:07:59 +00:00
2018-04-20 16:14:21 +00:00
handleHotkeyMoveUp = e => {
this . props . onMoveUp ( this . props . status . get ( 'id' ) , e . target . getAttribute ( 'data-featured' ) ) ;
2023-01-30 00:45:35 +00:00
} ;
2017-10-05 23:07:59 +00:00
2018-04-20 16:14:21 +00:00
handleHotkeyMoveDown = e => {
this . props . onMoveDown ( this . props . status . get ( 'id' ) , e . target . getAttribute ( 'data-featured' ) ) ;
2023-01-30 00:45:35 +00:00
} ;
2017-10-05 23:07:59 +00:00
2018-04-18 01:33:59 +00:00
handleHotkeyToggleHidden = ( ) => {
2024-08-22 17:12:35 +00:00
const { onToggleHidden } = this . props ;
const status = this . _properStatus ( ) ;
if ( status . get ( 'matched_filters' ) ) {
const expandedBecauseOfCW = ! status . get ( 'hidden' ) || status . get ( 'spoiler_text' ) . length === 0 ;
const expandedBecauseOfFilter = this . state . showDespiteFilter ;
if ( expandedBecauseOfFilter && ! expandedBecauseOfCW ) {
onToggleHidden ( status ) ;
} else if ( expandedBecauseOfFilter && expandedBecauseOfCW ) {
onToggleHidden ( status ) ;
this . handleFilterToggle ( ) ;
} else {
this . handleFilterToggle ( ) ;
}
} else {
onToggleHidden ( status ) ;
}
2023-01-30 00:45:35 +00:00
} ;
2018-04-18 01:33:59 +00:00
2019-05-25 21:20:51 +00:00
handleHotkeyToggleSensitive = ( ) => {
this . handleToggleMediaVisibility ( ) ;
2023-01-30 00:45:35 +00:00
} ;
2019-05-25 21:20:51 +00:00
2024-08-22 17:12:35 +00:00
handleFilterToggle = ( ) => {
this . setState ( state => ( { ... state , showDespiteFilter : ! state . showDespiteFilter } ) ) ;
2023-01-30 00:45:35 +00:00
} ;
Revamp post filtering system (#18058)
* Add model for custom filter keywords
* Use CustomFilterKeyword internally
Does not change the API
* Fix /filters/edit and /filters/new
* Add migration tests
* Remove whole_word column from custom_filters (covered by custom_filter_keywords)
* Redesign /filters
Instead of a list, present a card that displays more information and handles
multiple keywords per filter.
* Redesign /filters/new and /filters/edit to add and remove keywords
This adds a new gem dependency: cocoon, as well as a npm dependency:
cocoon-js-vanilla. Those are used to easily populate and remove form fields
from the user interface when manipulating multiple keyword filters at once.
* Add /api/v2/filters to edit filter with multiple keywords
Entities:
- `Filter`: `id`, `title`, `filter_action` (either `hide` or `warn`), `context`
`keywords`
- `FilterKeyword`: `id`, `keyword`, `whole_word`
API endpoits:
- `GET /api/v2/filters` to list filters (including keywords)
- `POST /api/v2/filters` to create a new filter
`keywords_attributes` can also be passed to create keywords in one request
- `GET /api/v2/filters/:id` to read a particular filter
- `PUT /api/v2/filters/:id` to update a new filter
`keywords_attributes` can also be passed to edit, delete or add keywords in
one request
- `DELETE /api/v2/filters/:id` to delete a particular filter
- `GET /api/v2/filters/:id/keywords` to list keywords for a filter
- `POST /api/v2/filters/:filter_id/keywords/:id` to add a new keyword to a
filter
- `GET /api/v2/filter_keywords/:id` to read a particular keyword
- `PUT /api/v2/filter_keywords/:id` to edit a particular keyword
- `DELETE /api/v2/filter_keywords/:id` to delete a particular keyword
* Change from `irreversible` boolean to `action` enum
* Remove irrelevent `irreversible_must_be_within_context` check
* Fix /filters/new and /filters/edit with update for filter_action
* Fix Rubocop/Codeclimate complaining about task names
* Refactor FeedManager#phrase_filtered?
This moves regexp building and filter caching to the `CustomFilter` class.
This does not change the functional behavior yet, but this changes how the
cache is built, doing per-custom_filter regexps so that filters can be matched
independently, while still offering caching.
* Perform server-side filtering and output result in REST API
* Fix numerous filters_changed events being sent when editing multiple keywords at once
* Add some tests
* Use the new API in the WebUI
- use client-side logic for filters we have fetched rules for.
This is so that filter changes can be retroactively applied without
reloading the UI.
- use server-side logic for filters we haven't fetched rules for yet
(e.g. network error, or initial timeline loading)
* Minor optimizations and refactoring
* Perform server-side filtering on the streaming server
* Change the wording of filter action labels
* Fix issues pointed out by linter
* Change design of “Show anyway” link in accordence to review comments
* Drop “irreversible” filtering behavior
* Move /api/v2/filter_keywords to /api/v1/filters/keywords
* Rename `filter_results` attribute to `filtered`
* Rename REST::LegacyFilterSerializer to REST::V1::FilterSerializer
* Fix systemChannelId value in streaming server
* Simplify code by removing client-side filtering code
The simplifcation comes at a cost though: filters aren't retroactively
applied anymore.
2022-06-28 07:42:13 +00:00
2017-10-05 23:07:59 +00:00
_properStatus ( ) {
const { status } = this . props ;
if ( status . get ( 'reblog' , null ) !== null && typeof status . get ( 'reblog' ) === 'object' ) {
return status . get ( 'reblog' ) ;
} else {
return status ;
}
2017-09-14 01:39:10 +00:00
}
2019-02-11 12:19:59 +00:00
handleRef = c => {
this . node = c ;
2023-01-30 00:45:35 +00:00
} ;
2019-02-11 12:19:59 +00:00
2016-08-24 19:08:00 +00:00
render ( ) {
2024-07-23 06:20:17 +00:00
const { intl , hidden , featured , unfocusable , unread , showThread , scrollKey , pictureInPicture , previousId , nextInReplyToId , rootId , skipPrepend , avatarSize = 46 } = this . props ;
2016-09-05 18:38:31 +00:00
2017-10-05 23:07:59 +00:00
let { status , account , ... other } = this . props ;
2016-10-25 09:13:16 +00:00
if ( status === null ) {
2017-05-26 12:07:48 +00:00
return null ;
2017-05-24 15:55:00 +00:00
}
2019-08-19 17:00:33 +00:00
const handlers = this . props . muted ? { } : {
reply : this . handleHotkeyReply ,
favourite : this . handleHotkeyFavourite ,
boost : this . handleHotkeyBoost ,
mention : this . handleHotkeyMention ,
open : this . handleHotkeyOpen ,
openProfile : this . handleHotkeyOpenProfile ,
moveUp : this . handleHotkeyMoveUp ,
moveDown : this . handleHotkeyMoveDown ,
toggleHidden : this . handleHotkeyToggleHidden ,
toggleSensitive : this . handleHotkeyToggleSensitive ,
2019-11-29 16:02:36 +00:00
openMedia : this . handleHotkeyOpenMedia ,
2019-08-19 17:00:33 +00:00
} ;
2023-04-24 06:07:03 +00:00
let media , statusAvatar , prepend , rebloggedByText ;
2017-08-28 20:23:44 +00:00
if ( hidden ) {
2017-05-24 15:55:00 +00:00
return (
2024-07-23 06:20:17 +00:00
< HotKeys handlers = { handlers } tabIndex = { unfocusable ? null : - 1 } >
< div ref = { this . handleRef } className = { classNames ( 'status__wrapper' , { focusable : ! this . props . muted } ) } tabIndex = { unfocusable ? null : 0 } >
2021-07-23 00:53:17 +00:00
< span > { status . getIn ( [ 'account' , 'display_name' ] ) || status . getIn ( [ 'account' , 'username' ] ) } < / span >
< span > { status . get ( 'content' ) } < / span >
2019-08-19 17:00:33 +00:00
< / div >
< / HotKeys >
2017-05-24 15:55:00 +00:00
) ;
2016-10-25 09:13:16 +00:00
}
2023-04-24 06:07:03 +00:00
const connectUp = previousId && previousId === status . get ( 'in_reply_to_id' ) ;
const connectToRoot = rootId && rootId === status . get ( 'in_reply_to_id' ) ;
const connectReply = nextInReplyToId && nextInReplyToId === status . get ( 'id' ) ;
2022-06-30 07:51:55 +00:00
const matchedFilters = status . get ( 'matched_filters' ) ;
2023-04-24 06:07:03 +00:00
2018-03-04 08:19:11 +00:00
if ( featured ) {
prepend = (
< div className = 'status__prepend' >
2023-10-24 17:45:08 +00:00
< div className = 'status__prepend-icon-wrapper' > < Icon id = 'thumb-tack' icon = { PushPinIcon } className = 'status__prepend-icon' / > < / div >
2022-04-28 22:24:31 +00:00
< FormattedMessage id = 'status.pinned' defaultMessage = 'Pinned post' / >
2018-03-04 08:19:11 +00:00
< / div >
) ;
} else if ( status . get ( 'reblog' , null ) !== null && typeof status . get ( 'reblog' ) === 'object' ) {
2017-08-07 18:32:03 +00:00
const display _name _html = { _ _html : status . getIn ( [ 'account' , 'display_name_html' ] ) } ;
2016-11-18 23:28:42 +00:00
2017-10-05 23:07:59 +00:00
prepend = (
< div className = 'status__prepend' >
2023-10-24 17:45:08 +00:00
< div className = 'status__prepend-icon-wrapper' > < Icon id = 'retweet' icon = { RepeatIcon } className = 'status__prepend-icon' / > < / div >
2024-06-26 19:33:38 +00:00
< FormattedMessage id = 'status.reblogged_by' defaultMessage = '{name} boosted' values = { { name : < a onClick = { this . handlePrependAccountClick } data - id = { status . getIn ( [ 'account' , 'id' ] ) } data - hover - card - account = { status . getIn ( [ 'account' , 'id' ] ) } href = { ` /@ ${ status . getIn ( [ 'account' , 'acct' ] ) } ` } className = 'status__display-name muted' > < bdi > < strong dangerouslySetInnerHTML = { display _name _html } / > < / bdi > < / a > } } / >
2017-08-28 20:23:44 +00:00
< / div >
2016-09-01 12:12:11 +00:00
) ;
2017-10-05 23:07:59 +00:00
2018-08-23 18:56:57 +00:00
rebloggedByText = intl . formatMessage ( { id : 'status.reblogged_by' , defaultMessage : '{name} boosted' } , { name : status . getIn ( [ 'account' , 'acct' ] ) } ) ;
2017-10-05 23:07:59 +00:00
account = status . get ( 'account' ) ;
status = status . get ( 'reblog' ) ;
2023-03-30 13:16:20 +00:00
} else if ( status . get ( 'visibility' ) === 'direct' ) {
prepend = (
< div className = 'status__prepend' >
2023-10-24 17:45:08 +00:00
< div className = 'status__prepend-icon-wrapper' > < Icon id = 'at' icon = { AlternateEmailIcon } className = 'status__prepend-icon' / > < / div >
2023-03-30 13:16:20 +00:00
< FormattedMessage id = 'status.direct_indicator' defaultMessage = 'Private mention' / >
< / div >
) ;
2022-10-25 17:02:21 +00:00
} else if ( showThread && status . get ( 'in_reply_to_id' ) && status . get ( 'in_reply_to_account_id' ) === status . getIn ( [ 'account' , 'id' ] ) ) {
const display _name _html = { _ _html : status . getIn ( [ 'account' , 'display_name_html' ] ) } ;
prepend = (
< div className = 'status__prepend' >
2023-10-24 17:45:08 +00:00
< div className = 'status__prepend-icon-wrapper' > < Icon id = 'reply' icon = { ReplyIcon } className = 'status__prepend-icon' / > < / div >
2024-06-26 19:33:38 +00:00
< FormattedMessage id = 'status.replied_to' defaultMessage = 'Replied to {name}' values = { { name : < a onClick = { this . handlePrependAccountClick } data - id = { status . getIn ( [ 'account' , 'id' ] ) } data - hover - card - account = { status . getIn ( [ 'account' , 'id' ] ) } href = { ` /@ ${ status . getIn ( [ 'account' , 'acct' ] ) } ` } className = 'status__display-name muted' > < bdi > < strong dangerouslySetInnerHTML = { display _name _html } / > < / bdi > < / a > } } / >
2022-10-25 17:02:21 +00:00
< / div >
) ;
2016-09-01 12:12:11 +00:00
}
2016-08-24 15:56:44 +00:00
2020-12-07 18:36:36 +00:00
if ( pictureInPicture . get ( 'inUse' ) ) {
2023-07-24 11:46:55 +00:00
media = < PictureInPicturePlaceholder aspectRatio = { this . getAttachmentAspectRatio ( ) } / > ;
2020-09-28 11:29:43 +00:00
} else if ( status . get ( 'media_attachments' ) . size > 0 ) {
2023-05-31 22:10:21 +00:00
const language = status . getIn ( [ 'translation' , 'language' ] ) || status . get ( 'language' ) ;
2023-07-24 11:46:55 +00:00
if ( status . getIn ( [ 'media_attachments' , 0 , 'type' ] ) === 'audio' ) {
2019-08-23 20:38:02 +00:00
const attachment = status . getIn ( [ 'media_attachments' , 0 ] ) ;
2023-05-31 22:10:21 +00:00
const description = attachment . getIn ( [ 'translation' , 'description' ] ) || attachment . get ( 'description' ) ;
2019-08-23 20:38:02 +00:00
media = (
< Bundle fetchComponent = { Audio } loading = { this . renderLoadingAudioPlayer } >
{ Component => (
< Component
src = { attachment . get ( 'url' ) }
2023-05-31 22:10:21 +00:00
alt = { description }
lang = { language }
2020-06-29 11:56:55 +00:00
poster = { attachment . get ( 'preview_url' ) || status . getIn ( [ 'account' , 'avatar_static' ] ) }
2020-07-05 16:28:25 +00:00
backgroundColor = { attachment . getIn ( [ 'meta' , 'colors' , 'background' ] ) }
foregroundColor = { attachment . getIn ( [ 'meta' , 'colors' , 'foreground' ] ) }
accentColor = { attachment . getIn ( [ 'meta' , 'colors' , 'accent' ] ) }
2019-08-23 20:38:02 +00:00
duration = { attachment . getIn ( [ 'meta' , 'original' , 'duration' ] , 0 ) }
2020-06-21 00:27:19 +00:00
width = { this . props . cachedMediaWidth }
2020-06-23 10:20:14 +00:00
height = { 110 }
2020-06-21 00:27:19 +00:00
cacheWidth = { this . props . cacheMediaWidth }
2020-12-07 18:36:36 +00:00
deployPictureInPicture = { pictureInPicture . get ( 'available' ) ? this . handleDeployPictureInPicture : undefined }
2022-08-13 13:39:05 +00:00
sensitive = { status . get ( 'sensitive' ) }
blurhash = { attachment . get ( 'blurhash' ) }
visible = { this . state . showMedia }
onToggleVisibility = { this . handleToggleMediaVisibility }
2019-08-23 20:38:02 +00:00
/ >
) }
< / Bundle >
) ;
} else if ( status . getIn ( [ 'media_attachments' , 0 , 'type' ] ) === 'video' ) {
2019-06-19 21:42:38 +00:00
const attachment = status . getIn ( [ 'media_attachments' , 0 ] ) ;
2023-05-31 22:10:21 +00:00
const description = attachment . getIn ( [ 'translation' , 'description' ] ) || attachment . get ( 'description' ) ;
2017-09-14 01:39:10 +00:00
2017-07-07 22:06:02 +00:00
media = (
2017-09-14 01:39:10 +00:00
< Bundle fetchComponent = { Video } loading = { this . renderLoadingVideoPlayer } >
2018-01-17 15:57:15 +00:00
{ Component => (
< Component
2019-06-19 21:42:38 +00:00
preview = { attachment . get ( 'preview_url' ) }
2020-11-21 22:19:04 +00:00
frameRate = { attachment . getIn ( [ 'meta' , 'original' , 'frame_rate' ] ) }
2023-07-24 11:46:55 +00:00
aspectRatio = { ` ${ attachment . getIn ( [ 'meta' , 'original' , 'width' ] ) } / ${ attachment . getIn ( [ 'meta' , 'original' , 'height' ] ) } ` }
2019-06-19 21:42:38 +00:00
blurhash = { attachment . get ( 'blurhash' ) }
src = { attachment . get ( 'url' ) }
2023-05-31 22:10:21 +00:00
alt = { description }
lang = { language }
2018-01-17 15:57:15 +00:00
sensitive = { status . get ( 'sensitive' ) }
onOpenVideo = { this . handleOpenVideo }
2020-12-07 18:36:36 +00:00
deployPictureInPicture = { pictureInPicture . get ( 'available' ) ? this . handleDeployPictureInPicture : undefined }
2019-05-25 21:20:51 +00:00
visible = { this . state . showMedia }
onToggleVisibility = { this . handleToggleMediaVisibility }
2018-01-17 15:57:15 +00:00
/ >
) }
2017-07-07 22:06:02 +00:00
< / Bundle >
) ;
2016-09-17 16:05:02 +00:00
} else {
2017-07-07 22:06:02 +00:00
media = (
2018-05-08 11:33:09 +00:00
< Bundle fetchComponent = { MediaGallery } loading = { this . renderLoadingMediaGallery } >
2019-02-11 12:19:59 +00:00
{ Component => (
< Component
media = { status . get ( 'media_attachments' ) }
2023-05-31 22:10:21 +00:00
lang = { language }
2019-02-11 12:19:59 +00:00
sensitive = { status . get ( 'sensitive' ) }
height = { 110 }
2020-11-27 02:24:11 +00:00
onOpenMedia = { this . handleOpenMedia }
2019-02-11 12:19:59 +00:00
cacheWidth = { this . props . cacheMediaWidth }
defaultWidth = { this . props . cachedMediaWidth }
2019-05-25 21:20:51 +00:00
visible = { this . state . showMedia }
onToggleVisibility = { this . handleToggleMediaVisibility }
2019-02-11 12:19:59 +00:00
/ >
) }
2017-07-07 22:06:02 +00:00
< / Bundle >
) ;
2016-09-17 16:05:02 +00:00
}
2023-07-24 11:46:55 +00:00
} else if ( status . get ( 'spoiler_text' ) . length === 0 && status . get ( 'card' ) ) {
2018-10-28 05:35:03 +00:00
media = (
< Card
2020-12-08 11:07:54 +00:00
onOpenMedia = { this . handleOpenMedia }
2018-10-28 05:35:03 +00:00
card = { status . get ( 'card' ) }
compact
2020-06-06 15:41:56 +00:00
sensitive = { status . get ( 'sensitive' ) }
2018-10-28 05:35:03 +00:00
/ >
) ;
2016-09-05 18:38:31 +00:00
}
2022-05-30 15:10:13 +00:00
if ( account === undefined || account === null ) {
2024-07-18 14:36:09 +00:00
statusAvatar = < Avatar account = { status . get ( 'account' ) } size = { avatarSize } / > ;
2018-10-20 00:23:58 +00:00
} else {
2017-08-07 17:44:55 +00:00
statusAvatar = < AvatarOverlay account = { status . get ( 'account' ) } friend = { account } / > ;
2017-05-03 09:43:37 +00:00
}
2023-08-21 17:39:01 +00:00
const { statusContentProps , hashtagBar } = getHashtagBarForStatus ( status ) ;
2024-08-22 17:12:35 +00:00
const expanded = ( ! matchedFilters || this . state . showDespiteFilter ) && ( ! status . get ( 'hidden' ) || status . get ( 'spoiler_text' ) . length === 0 ) ;
2023-08-21 17:39:01 +00:00
2016-08-24 15:56:44 +00:00
return (
2024-07-23 06:20:17 +00:00
< HotKeys handlers = { handlers } tabIndex = { unfocusable ? null : - 1 } >
< div className = { classNames ( 'status__wrapper' , ` status__wrapper- ${ status . get ( 'visibility' ) } ` , { 'status__wrapper-reply' : ! ! status . get ( 'in_reply_to_id' ) , unread , focusable : ! this . props . muted } ) } tabIndex = { this . props . muted || unfocusable ? null : 0 } data - featured = { featured ? 'true' : null } aria - label = { textForScreenReader ( intl , status , rebloggedByText ) } ref = { this . handleRef } data - nosnippet = { status . getIn ( [ 'account' , 'noindex' ] , true ) || undefined } >
2024-07-18 14:36:09 +00:00
{ ! skipPrepend && prepend }
2016-08-24 19:08:00 +00:00
2023-04-24 06:07:03 +00:00
< div className = { classNames ( 'status' , ` status- ${ status . get ( 'visibility' ) } ` , { 'status-reply' : ! ! status . get ( 'in_reply_to_id' ) , 'status--in-thread' : ! ! rootId , 'status--first-in-thread' : previousId && ( ! connectUp || connectToRoot ) , muted : this . props . muted } ) } data - id = { status . get ( 'id' ) } >
{ ( connectReply || connectUp || connectToRoot ) && < div className = { classNames ( 'status__line' , { 'status__line--full' : connectReply , 'status__line--first' : ! status . get ( 'in_reply_to_id' ) && ! connectToRoot } ) } / > }
2023-04-23 20:24:53 +00:00
{ /* eslint-disable-next-line jsx-a11y/no-static-element-interactions */ }
2023-02-20 07:11:23 +00:00
< div onClick = { this . handleClick } className = 'status__info' >
< a href = { ` /@ ${ status . getIn ( [ 'account' , 'acct' ] ) } / ${ status . get ( 'id' ) } ` } className = 'status__relative-time' target = '_blank' rel = 'noopener noreferrer' >
2023-10-24 17:45:08 +00:00
< span className = 'status__visibility-icon' > < VisibilityIcon visibility = { status . get ( 'visibility' ) } / > < / span >
2024-02-29 09:40:13 +00:00
< RelativeTimestamp timestamp = { status . get ( 'created_at' ) } / > { status . get ( 'edited_at' ) && < abbr title = { intl . formatMessage ( messages . edited , { date : intl . formatDate ( status . get ( 'edited_at' ) , { year : 'numeric' , month : 'short' , day : '2-digit' , hour : '2-digit' , minute : '2-digit' } ) } ) } > * < / abbr > }
2020-10-27 02:00:47 +00:00
< / a >
2016-08-24 19:08:00 +00:00
2024-07-11 19:42:58 +00:00
< a onClick = { this . handleAccountClick } href = { ` /@ ${ status . getIn ( [ 'account' , 'acct' ] ) } ` } title = { status . getIn ( [ 'account' , 'acct' ] ) } data - hover - card - account = { status . getIn ( [ 'account' , 'id' ] ) } className = 'status__display-name' target = '_blank' rel = 'noopener noreferrer' >
2017-10-05 23:07:59 +00:00
< div className = 'status__avatar' >
{ statusAvatar }
< / div >
2016-08-31 20:58:10 +00:00
2022-05-30 15:10:13 +00:00
< DisplayName account = { status . get ( 'account' ) } / >
2017-10-05 23:07:59 +00:00
< / a >
< / div >
2024-08-22 17:12:35 +00:00
{ matchedFilters && < FilterWarning title = { matchedFilters . join ( ', ' ) } expanded = { this . state . showDespiteFilter } onClick = { this . handleFilterToggle } / > }
2016-08-24 19:08:00 +00:00
2024-08-22 17:12:35 +00:00
{ ( status . get ( 'spoiler_text' ) . length > 0 && ( ! matchedFilters || this . state . showDespiteFilter ) ) && < ContentWarning text = { status . getIn ( [ 'translation' , 'spoilerHtml' ] ) || status . get ( 'spoilerHtml' ) } expanded = { expanded } onClick = { this . handleExpandedToggle } / > }
2016-09-05 18:38:31 +00:00
2024-08-22 17:12:35 +00:00
{ expanded && (
< >
< StatusContent
status = { status }
onClick = { this . handleClick }
onTranslate = { this . handleTranslate }
collapsible
onCollapsedToggle = { this . handleCollapsedToggle }
{ ... statusContentProps }
/ >
{ media }
{ hashtagBar }
< / >
) }
2023-08-14 21:42:30 +00:00
2024-08-22 17:12:35 +00:00
< StatusActionBar scrollKey = { scrollKey } status = { status } account = { account } { ...other } / >
2017-10-05 23:07:59 +00:00
< / div >
< / div >
< / HotKeys >
2016-08-24 15:56:44 +00:00
) ;
}
2016-08-31 14:15:12 +00:00
2017-04-21 18:05:35 +00:00
}
2023-03-24 02:17:53 +00:00
2023-10-19 17:44:55 +00:00
export default withOptionalRouter ( injectIntl ( Status ) ) ;