- built in fullwidth styling - letterbox settings toggle - media no longer counts towards height when making toot-collapsing measurements
		
			
				
	
	
		
			369 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			369 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import React from 'react';
 | |
| import ImmutablePropTypes from 'react-immutable-proptypes';
 | |
| import PropTypes from 'prop-types';
 | |
| import Avatar from './avatar';
 | |
| import AvatarOverlay from './avatar_overlay';
 | |
| import DisplayName from './display_name';
 | |
| import MediaGallery from './media_gallery';
 | |
| import VideoPlayer from './video_player';
 | |
| import StatusContent from './status_content';
 | |
| import StatusActionBar from './status_action_bar';
 | |
| import IconButton from './icon_button';
 | |
| import { defineMessages, FormattedMessage } from 'react-intl';
 | |
| import emojify from '../emoji';
 | |
| import escapeTextContentForBrowser from 'escape-html';
 | |
| import ImmutablePureComponent from 'react-immutable-pure-component';
 | |
| import scheduleIdleTask from '../features/ui/util/schedule_idle_task';
 | |
| 
 | |
| const messages = defineMessages({
 | |
|   collapse: { id: 'status.collapse', defaultMessage: 'Collapse' },
 | |
|   uncollapse: { id: 'status.uncollapse', defaultMessage: 'Uncollapse' },
 | |
| });
 | |
| 
 | |
| export default class StatusOrReblog extends ImmutablePureComponent {
 | |
| 
 | |
|   static propTypes = {
 | |
|     status: ImmutablePropTypes.map,
 | |
|     account: ImmutablePropTypes.map,
 | |
|     settings: ImmutablePropTypes.map,
 | |
|     wrapped: PropTypes.bool,
 | |
|     onReply: PropTypes.func,
 | |
|     onFavourite: PropTypes.func,
 | |
|     onReblog: PropTypes.func,
 | |
|     onDelete: PropTypes.func,
 | |
|     onOpenMedia: PropTypes.func,
 | |
|     onOpenVideo: PropTypes.func,
 | |
|     onBlock: PropTypes.func,
 | |
|     me: PropTypes.number,
 | |
|     boostModal: PropTypes.bool,
 | |
|     autoPlayGif: PropTypes.bool,
 | |
|     muted: PropTypes.bool,
 | |
|     collapse: PropTypes.bool,
 | |
|     intersectionObserverWrapper: PropTypes.object,
 | |
|     intl: PropTypes.object.isRequired,
 | |
|   };
 | |
| 
 | |
|   // Avoid checking props that are functions (and whose equality will always
 | |
|   // evaluate to false. See react-immutable-pure-component for usage.
 | |
|   updateOnProps = [
 | |
|     'status',
 | |
|     'account',
 | |
|     'settings',
 | |
|     'wrapped',
 | |
|     'me',
 | |
|     'boostModal',
 | |
|     'autoPlayGif',
 | |
|     'muted',
 | |
|     'collapse',
 | |
|   ]
 | |
| 
 | |
|   render () {
 | |
|     // Exclude intersectionObserverWrapper from `other` variable
 | |
|     // because intersection is managed in here.
 | |
|     const { status, account, ...other } = this.props;
 | |
| 
 | |
|     if (status === null) {
 | |
|       return null;
 | |
|     }
 | |
| 
 | |
|     if (status.get('reblog', null) !== null && typeof status.get('reblog') === 'object') {
 | |
|       let displayName = status.getIn(['account', 'display_name']);
 | |
| 
 | |
|       if (displayName.length === 0) {
 | |
|         displayName = status.getIn(['account', 'username']);
 | |
|       }
 | |
| 
 | |
|       const displayNameHTML = { __html: emojify(escapeTextContentForBrowser(displayName)) };
 | |
| 
 | |
|       return (
 | |
|         <div className='status__wrapper' ref={this.handleRef} data-id={status.get('id')} >
 | |
|           <div className='status__prepend'>
 | |
|             <div className='status__prepend-icon-wrapper'><i className='fa fa-fw fa-retweet status__prepend-icon' /></div>
 | |
|             <FormattedMessage id='status.reblogged_by' defaultMessage='{name} boosted' values={{ name: <a onClick={this.handleAccountClick} data-id={status.getIn(['account', 'id'])} href={status.getIn(['account', 'url'])} className='status__display-name muted'><strong dangerouslySetInnerHTML={displayNameHTML} /></a> }} />
 | |
|           </div>
 | |
| 
 | |
|           <Status {...other} status={status.get('reblog')} account={status.get('account')} wrapped />
 | |
|         </div>
 | |
|       );
 | |
|     } else return <Status {...this.props} />;
 | |
|   }
 | |
| 
 | |
| }
 | |
| 
 | |
| class Status extends ImmutablePureComponent {
 | |
| 
 | |
|   static contextTypes = {
 | |
|     router: PropTypes.object,
 | |
|   };
 | |
| 
 | |
|   static propTypes = {
 | |
|     status: ImmutablePropTypes.map,
 | |
|     account: ImmutablePropTypes.map,
 | |
|     settings: ImmutablePropTypes.map,
 | |
|     wrapped: PropTypes.bool,
 | |
|     onReply: PropTypes.func,
 | |
|     onFavourite: PropTypes.func,
 | |
|     onReblog: PropTypes.func,
 | |
|     onDelete: PropTypes.func,
 | |
|     onOpenMedia: PropTypes.func,
 | |
|     onOpenVideo: PropTypes.func,
 | |
|     onBlock: PropTypes.func,
 | |
|     me: PropTypes.number,
 | |
|     boostModal: PropTypes.bool,
 | |
|     autoPlayGif: PropTypes.bool,
 | |
|     muted: PropTypes.bool,
 | |
|     collapse: PropTypes.bool,
 | |
|     intersectionObserverWrapper: PropTypes.object,
 | |
|     intl: PropTypes.object.isRequired,
 | |
|   };
 | |
| 
 | |
|   state = {
 | |
|     isExpanded: false,
 | |
|     isIntersecting: true, // assume intersecting until told otherwise
 | |
|     isHidden: false, // set to true in requestIdleCallback to trigger un-render
 | |
|     isCollapsed: false,
 | |
|   }
 | |
| 
 | |
|   // Avoid checking props that are functions (and whose equality will always
 | |
|   // evaluate to false. See react-immutable-pure-component for usage.
 | |
|   updateOnProps = [
 | |
|     'status',
 | |
|     'account',
 | |
|     'settings',
 | |
|     'wrapped',
 | |
|     'me',
 | |
|     'boostModal',
 | |
|     'autoPlayGif',
 | |
|     'muted',
 | |
|     'collapse',
 | |
|   ]
 | |
| 
 | |
|   updateOnStates = [
 | |
|     'isExpanded',
 | |
|     'isCollapsed',
 | |
|   ]
 | |
| 
 | |
|   componentWillReceiveProps (nextProps) {
 | |
|     if (!nextProps.settings.getIn(['collapsed', 'enabled'])) this.collapse(false);
 | |
|     else if (nextProps.collapse !== this.props.collapse && nextProps.collapse !== undefined) this.collapse(this.props.collapse);
 | |
|   }
 | |
| 
 | |
|   shouldComponentUpdate (nextProps, nextState) {
 | |
|     if (!nextState.isIntersecting && nextState.isHidden) {
 | |
|       // It's only if we're not intersecting (i.e. offscreen) and isHidden is true
 | |
|       // that either "isIntersecting" or "isHidden" matter, and then they're
 | |
|       // the only things that matter.
 | |
|       return this.state.isIntersecting || !this.state.isHidden;
 | |
|     } else if (nextState.isIntersecting && !this.state.isIntersecting) {
 | |
|       // If we're going from a non-intersecting state to an intersecting state,
 | |
|       // (i.e. offscreen to onscreen), then we definitely need to re-render
 | |
|       return true;
 | |
|     }
 | |
|     // Otherwise, diff based on "updateOnProps" and "updateOnStates"
 | |
|     return super.shouldComponentUpdate(nextProps, nextState);
 | |
|   }
 | |
| 
 | |
|   componentDidUpdate () {
 | |
|     if (this.state.isIntersecting || !this.state.isHidden) this.saveHeight();
 | |
|   }
 | |
| 
 | |
|   componentDidMount () {
 | |
|     const node = this.node;
 | |
| 
 | |
|     const { collapse, settings, status } = this.props;
 | |
| 
 | |
|     if (collapse !== undefined) this.collapse(collapse);
 | |
|     else if (settings.getIn(['collapsed', 'auto', 'all'])) this.collapse();
 | |
|     else if (settings.getIn(['collapsed', 'auto', 'lengthy']) && node.clientHeight > (status.get('media_attachments').size > 0 && !this.props.muted ? 650 : 400)) this.collapse();
 | |
|     else if (settings.getIn(['collapsed', 'auto', 'replies']) && status.get('in_reply_to_id', null) !== null) this.collapse();
 | |
|     else if (settings.getIn(['collapsed', 'auto', 'media']) && !(status.get('spoiler_text').length > 0) && status.get('media_attachments').size > 0) this.collapse();
 | |
| 
 | |
|     if (!this.props.intersectionObserverWrapper) {
 | |
|       // TODO: enable IntersectionObserver optimization for notification statuses.
 | |
|       // These are managed in notifications/index.js rather than status_list.js
 | |
|       return;
 | |
|     }
 | |
|     this.props.intersectionObserverWrapper.observe(
 | |
|       this.props.id,
 | |
|       this.node,
 | |
|       this.handleIntersection
 | |
|     );
 | |
| 
 | |
|     this.componentMounted = true;
 | |
|   }
 | |
| 
 | |
|   componentWillUnmount () {
 | |
|     this.componentMounted = false;
 | |
|   }
 | |
| 
 | |
|   collapse = (collapsedOrNot) => {
 | |
|     if (collapsedOrNot === undefined) collapsedOrNot = true;
 | |
|     if (this.props.settings.getIn(['collapsed', 'enabled'])) this.setState({ isCollapsed: !!collapsedOrNot });
 | |
|   }
 | |
| 
 | |
|   handleIntersection = (entry) => {
 | |
|     // Edge 15 doesn't support isIntersecting, but we can infer it
 | |
|     // https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/12156111/
 | |
|     // https://github.com/WICG/IntersectionObserver/issues/211
 | |
|     const isIntersecting = (typeof entry.isIntersecting === 'boolean') ?
 | |
|     entry.isIntersecting : entry.intersectionRect.height > 0;
 | |
|     this.setState((prevState) => {
 | |
|       if (prevState.isIntersecting && !isIntersecting) {
 | |
|         scheduleIdleTask(this.hideIfNotIntersecting);
 | |
|       }
 | |
|       return {
 | |
|         isIntersecting: isIntersecting,
 | |
|         isHidden: false,
 | |
|       };
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   hideIfNotIntersecting = () => {
 | |
|     if (!this.componentMounted) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // When the browser gets a chance, test if we're still not intersecting,
 | |
|     // and if so, set our isHidden to true to trigger an unrender. The point of
 | |
|     // this is to save DOM nodes and avoid using up too much memory.
 | |
|     // See: https://github.com/tootsuite/mastodon/issues/2900
 | |
|     this.setState((prevState) => ({ isHidden: !prevState.isIntersecting }));
 | |
|   }
 | |
| 
 | |
|   saveHeight = () => {
 | |
|     if (this.node && this.node.children.length !== 0) {
 | |
|       this.height = this.node.getBoundingClientRect().height;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   handleRef = (node) => {
 | |
|     this.node = node;
 | |
|     this.saveHeight();
 | |
|   }
 | |
| 
 | |
|   handleClick = () => {
 | |
|     const { status } = this.props;
 | |
|     const { isCollapsed } = this.state;
 | |
|     if (isCollapsed) this.handleCollapsedClick();
 | |
|     else this.context.router.history.push(`/statuses/${status.getIn(['reblog', 'id'], status.get('id'))}`);
 | |
|   }
 | |
| 
 | |
|   handleAccountClick = (e) => {
 | |
|     if (e.button === 0) {
 | |
|       const id = Number(e.currentTarget.getAttribute('data-id'));
 | |
|       e.preventDefault();
 | |
|       if (this.state.isCollapsed) this.handleCollapsedClick();
 | |
|       else this.context.router.history.push(`/accounts/${id}`);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   handleExpandedToggle = () => {
 | |
|     this.setState({ isExpanded: !this.state.isExpanded, isCollapsed: false });
 | |
|   };
 | |
| 
 | |
|   handleCollapsedClick = () => {
 | |
|     this.collapse(!this.state.isCollapsed);
 | |
|     this.setState({ isExpanded: false });
 | |
|   }
 | |
| 
 | |
|   render () {
 | |
|     let media = null;
 | |
|     let mediaIcon = null;
 | |
|     let statusAvatar;
 | |
| 
 | |
|     // Exclude intersectionObserverWrapper from `other` variable
 | |
|     // because intersection is managed in here.
 | |
|     const { status, account, settings, intersectionObserverWrapper, intl, ...other } = this.props;
 | |
|     const { isExpanded, isIntersecting, isHidden, isCollapsed } = this.state;
 | |
| 
 | |
| 
 | |
|     let background = settings.getIn(['collapsed', 'backgrounds', 'user_backgrounds']) ? status.getIn(['account', 'header']) : null;
 | |
| 
 | |
|     if (status === null) {
 | |
|       return null;
 | |
|     }
 | |
| 
 | |
|     if (!isIntersecting && isHidden) {
 | |
|       return (
 | |
|         <div ref={this.handleRef} data-id={status.get('id')} style={{ height: `${this.height}px`, opacity: 0, overflow: 'hidden' }}>
 | |
|           {status.getIn(['account', 'display_name']) || status.getIn(['account', 'username'])}
 | |
|           {status.get('content')}
 | |
|         </div>
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     if (status.get('media_attachments').size > 0 && !this.props.muted) {
 | |
|       if (status.get('media_attachments').some(item => item.get('type') === 'unknown')) {
 | |
| 
 | |
|       } else if (status.getIn(['media_attachments', 0, 'type']) === 'video') {
 | |
|         media = (
 | |
|           <VideoPlayer
 | |
|             media={status.getIn(['media_attachments', 0])}
 | |
|             sensitive={status.get('sensitive')}
 | |
|             letterbox={settings.getIn(['media', 'letterbox'])}
 | |
|             height={250}
 | |
|             onOpenVideo={this.props.onOpenVideo}
 | |
|           />
 | |
|         );
 | |
|         mediaIcon = 'video-camera';
 | |
|       } else {
 | |
|         media = (
 | |
|           <MediaGallery
 | |
|             media={status.get('media_attachments')}
 | |
|             sensitive={status.get('sensitive')}
 | |
|             letterbox={settings.getIn(['media', 'letterbox'])}
 | |
|             height={250}
 | |
|             onOpenMedia={this.props.onOpenMedia}
 | |
|             autoPlayGif={this.props.autoPlayGif}
 | |
|           />
 | |
|         );
 | |
|         mediaIcon = 'picture-o';
 | |
|       }
 | |
| 
 | |
|       if (!status.get('sensitive') && !(status.get('spoiler_text').length > 0) && settings.getIn(['collapsed', 'backgrounds', 'preview_images'])) background = status.getIn(['media_attachments', 0]).get('preview_url');
 | |
|     }
 | |
| 
 | |
|     if (account === undefined || account === null) {
 | |
|       statusAvatar = <Avatar src={status.getIn(['account', 'avatar'])} staticSrc={status.getIn(['account', 'avatar_static'])} size={48} />;
 | |
|     }else{
 | |
|       statusAvatar = <AvatarOverlay staticSrc={status.getIn(['account', 'avatar_static'])} overlaySrc={account.get('avatar_static')} />;
 | |
|     }
 | |
| 
 | |
|     return (
 | |
|       <div className={`status ${this.props.muted ? 'muted' : ''} status-${status.get('visibility')} ${isCollapsed ? 'status-collapsed' : ''}`} data-id={status.get('id')} ref={this.handleRef} style={{ backgroundImage: background && isCollapsed ? 'url(' + background + ')' : 'none' }}>
 | |
|         <div className='status__info'>
 | |
| 
 | |
|           <div className='status__info__icons'>
 | |
|             {mediaIcon ? <i className={`fa fa-fw fa-${mediaIcon}`} aria-hidden='true' /> : null}
 | |
|             {settings.getIn(['collapsed', 'enabled']) ? <IconButton
 | |
|               className='status__collapse-button'
 | |
|               animate flip
 | |
|               active={isCollapsed}
 | |
|               title={isCollapsed ? intl.formatMessage(messages.uncollapse) : intl.formatMessage(messages.collapse)}
 | |
|               icon='angle-double-up'
 | |
|               onClick={this.handleCollapsedClick}
 | |
|             /> : null}
 | |
|           </div>
 | |
| 
 | |
|           <a onClick={this.handleAccountClick} data-id={status.getIn(['account', 'id'])} href={status.getIn(['account', 'url'])} className='status__display-name'>
 | |
|             <div className='status__avatar'>
 | |
|               {statusAvatar}
 | |
|             </div>
 | |
| 
 | |
|             <DisplayName account={status.get('account')} />
 | |
|           </a>
 | |
| 
 | |
|         </div>
 | |
| 
 | |
|         <StatusContent status={status} mediaIcon={mediaIcon} onClick={this.handleClick} expanded={isExpanded} collapsed={isCollapsed} onExpandedToggle={this.handleExpandedToggle} onHeightUpdate={this.saveHeight}>
 | |
| 
 | |
|           {isCollapsed ? null : media}
 | |
| 
 | |
|         </StatusContent>
 | |
| 
 | |
|         {isCollapsed ? null : <StatusActionBar status={status} account={account} {...other} />}
 | |
|       </div>
 | |
|     );
 | |
|   }
 | |
| 
 | |
| }
 |