Moved glitch files to their own location ;)
This commit is contained in:
		@@ -1,199 +0,0 @@
 | 
			
		||||
import React from 'react';
 | 
			
		||||
import ImmutablePropTypes from 'react-immutable-proptypes';
 | 
			
		||||
import PropTypes from 'prop-types';
 | 
			
		||||
import IconButton from './icon_button';
 | 
			
		||||
import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
 | 
			
		||||
import { isIOS } from '../is_mobile';
 | 
			
		||||
 | 
			
		||||
const messages = defineMessages({
 | 
			
		||||
  toggle_visible: { id: 'media_gallery.toggle_visible', defaultMessage: 'Toggle visibility' },
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
class Item extends React.PureComponent {
 | 
			
		||||
 | 
			
		||||
  static propTypes = {
 | 
			
		||||
    attachment: ImmutablePropTypes.map.isRequired,
 | 
			
		||||
    index: PropTypes.number.isRequired,
 | 
			
		||||
    size: PropTypes.number.isRequired,
 | 
			
		||||
    letterbox: PropTypes.bool,
 | 
			
		||||
    onClick: PropTypes.func.isRequired,
 | 
			
		||||
    autoPlayGif: PropTypes.bool.isRequired,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  handleClick = (e) => {
 | 
			
		||||
    const { index, onClick } = this.props;
 | 
			
		||||
 | 
			
		||||
    if (e.button === 0) {
 | 
			
		||||
      e.preventDefault();
 | 
			
		||||
      onClick(index);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    e.stopPropagation();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  render () {
 | 
			
		||||
    const { attachment, index, size, letterbox } = this.props;
 | 
			
		||||
 | 
			
		||||
    let width  = 50;
 | 
			
		||||
    let height = 100;
 | 
			
		||||
    let top    = 'auto';
 | 
			
		||||
    let left   = 'auto';
 | 
			
		||||
    let bottom = 'auto';
 | 
			
		||||
    let right  = 'auto';
 | 
			
		||||
 | 
			
		||||
    if (size === 1) {
 | 
			
		||||
      width = 100;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (size === 4 || (size === 3 && index > 0)) {
 | 
			
		||||
      height = 50;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (size === 2) {
 | 
			
		||||
      if (index === 0) {
 | 
			
		||||
        right = '2px';
 | 
			
		||||
      } else {
 | 
			
		||||
        left = '2px';
 | 
			
		||||
      }
 | 
			
		||||
    } else if (size === 3) {
 | 
			
		||||
      if (index === 0) {
 | 
			
		||||
        right = '2px';
 | 
			
		||||
      } else if (index > 0) {
 | 
			
		||||
        left = '2px';
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (index === 1) {
 | 
			
		||||
        bottom = '2px';
 | 
			
		||||
      } else if (index > 1) {
 | 
			
		||||
        top = '2px';
 | 
			
		||||
      }
 | 
			
		||||
    } else if (size === 4) {
 | 
			
		||||
      if (index === 0 || index === 2) {
 | 
			
		||||
        right = '2px';
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (index === 1 || index === 3) {
 | 
			
		||||
        left = '2px';
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (index < 2) {
 | 
			
		||||
        bottom = '2px';
 | 
			
		||||
      } else {
 | 
			
		||||
        top = '2px';
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    let thumbnail = '';
 | 
			
		||||
 | 
			
		||||
    if (attachment.get('type') === 'image') {
 | 
			
		||||
      const previewUrl = attachment.get('preview_url');
 | 
			
		||||
      const previewWidth = attachment.getIn(['meta', 'small', 'width']);
 | 
			
		||||
 | 
			
		||||
      const originalUrl = attachment.get('url');
 | 
			
		||||
      const originalWidth = attachment.getIn(['meta', 'original', 'width']);
 | 
			
		||||
 | 
			
		||||
      const srcSet = `${originalUrl} ${originalWidth}w, ${previewUrl} ${previewWidth}w`;
 | 
			
		||||
      const sizes = `(min-width: 1025px) ${320 * (width / 100)}px, ${width}vw`;
 | 
			
		||||
 | 
			
		||||
      thumbnail = (
 | 
			
		||||
        <a
 | 
			
		||||
          className='media-gallery__item-thumbnail'
 | 
			
		||||
          href={attachment.get('remote_url') || originalUrl}
 | 
			
		||||
          onClick={this.handleClick}
 | 
			
		||||
          target='_blank'
 | 
			
		||||
        >
 | 
			
		||||
          <img className={letterbox ? 'letterbox' : ''} src={previewUrl} srcSet={srcSet} sizes={sizes} alt='' />
 | 
			
		||||
        </a>
 | 
			
		||||
      );
 | 
			
		||||
    } else if (attachment.get('type') === 'gifv') {
 | 
			
		||||
      const autoPlay = !isIOS() && this.props.autoPlayGif;
 | 
			
		||||
 | 
			
		||||
      thumbnail = (
 | 
			
		||||
        <div className={`media-gallery__gifv ${autoPlay ? 'autoplay' : ''}`}>
 | 
			
		||||
          <video
 | 
			
		||||
            className={`media-gallery__item-gifv-thumbnail${letterbox ? ' letterbox' : ''}`}
 | 
			
		||||
            role='application'
 | 
			
		||||
            src={attachment.get('url')}
 | 
			
		||||
            onClick={this.handleClick}
 | 
			
		||||
            autoPlay={autoPlay}
 | 
			
		||||
            loop
 | 
			
		||||
            muted
 | 
			
		||||
          />
 | 
			
		||||
 | 
			
		||||
          <span className='media-gallery__gifv__label'>GIF</span>
 | 
			
		||||
        </div>
 | 
			
		||||
      );
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
      <div className='media-gallery__item' key={attachment.get('id')} style={{ left: left, top: top, right: right, bottom: bottom, width: `${width}%`, height: `${height}%` }}>
 | 
			
		||||
        {thumbnail}
 | 
			
		||||
      </div>
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@injectIntl
 | 
			
		||||
export default class MediaGallery extends React.PureComponent {
 | 
			
		||||
 | 
			
		||||
  static propTypes = {
 | 
			
		||||
    sensitive: PropTypes.bool,
 | 
			
		||||
    media: ImmutablePropTypes.list.isRequired,
 | 
			
		||||
    letterbox: PropTypes.bool,
 | 
			
		||||
    fullwidth: PropTypes.bool,
 | 
			
		||||
    height: PropTypes.number.isRequired,
 | 
			
		||||
    onOpenMedia: PropTypes.func.isRequired,
 | 
			
		||||
    intl: PropTypes.object.isRequired,
 | 
			
		||||
    autoPlayGif: PropTypes.bool.isRequired,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  state = {
 | 
			
		||||
    visible: !this.props.sensitive,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  handleOpen = () => {
 | 
			
		||||
    this.setState({ visible: !this.state.visible });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleClick = (index) => {
 | 
			
		||||
    this.props.onOpenMedia(this.props.media, index);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  render () {
 | 
			
		||||
    const { media, intl, sensitive, letterbox, fullwidth } = this.props;
 | 
			
		||||
 | 
			
		||||
    let children;
 | 
			
		||||
 | 
			
		||||
    if (!this.state.visible) {
 | 
			
		||||
      let warning;
 | 
			
		||||
 | 
			
		||||
      if (sensitive) {
 | 
			
		||||
        warning = <FormattedMessage id='status.sensitive_warning' defaultMessage='Sensitive content' />;
 | 
			
		||||
      } else {
 | 
			
		||||
        warning = <FormattedMessage id='status.media_hidden' defaultMessage='Media hidden' />;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      children = (
 | 
			
		||||
        <div role='button' tabIndex='0' className='media-spoiler' onClick={this.handleOpen}>
 | 
			
		||||
          <span className='media-spoiler__warning'>{warning}</span>
 | 
			
		||||
          <span className='media-spoiler__trigger'><FormattedMessage id='status.sensitive_toggle' defaultMessage='Click to view' /></span>
 | 
			
		||||
        </div>
 | 
			
		||||
      );
 | 
			
		||||
    } else {
 | 
			
		||||
      const size = media.take(4).size;
 | 
			
		||||
      children = media.take(4).map((attachment, i) => <Item key={attachment.get('id')} onClick={this.handleClick} attachment={attachment} autoPlayGif={this.props.autoPlayGif} index={i} size={size} letterbox={letterbox} />);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
      <div className={`media-gallery ${fullwidth ? 'full-width' : ''}`} style={{ height: `${this.props.height}px` }}>
 | 
			
		||||
        <div className={`spoiler-button ${this.state.visible ? 'spoiler-button--visible' : ''}`}>
 | 
			
		||||
          <IconButton title={intl.formatMessage(messages.toggle_visible)} icon={this.state.visible ? 'eye' : 'eye-slash'} overlay onClick={this.handleOpen} />
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        {children}
 | 
			
		||||
      </div>
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,724 +0,0 @@
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
`<Status>`
 | 
			
		||||
==========
 | 
			
		||||
 | 
			
		||||
Original file by @gargron@mastodon.social et al as part of
 | 
			
		||||
tootsuite/mastodon. *Heavily* rewritten (and documented!) by
 | 
			
		||||
@kibi@glitch.social as a part of glitch-soc/mastodon. The following
 | 
			
		||||
features have been added:
 | 
			
		||||
 | 
			
		||||
 -  Better separating the "guts" of statuses from their wrapper(s)
 | 
			
		||||
 -  Collapsing statuses
 | 
			
		||||
 -  Moving images inside of CWs
 | 
			
		||||
 | 
			
		||||
A number of aspects of this original file have been split off into
 | 
			
		||||
their own components for better maintainance; for these, see:
 | 
			
		||||
 | 
			
		||||
 -  <StatusHeader>
 | 
			
		||||
 -  <StatusPrepend>
 | 
			
		||||
 | 
			
		||||
…And, of course, the other <Status>-related components as well.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
                            /* * * * */
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
Imports:
 | 
			
		||||
--------
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
//  Our standard React imports:
 | 
			
		||||
import React from 'react';
 | 
			
		||||
import PropTypes from 'prop-types';
 | 
			
		||||
import ImmutablePropTypes from 'react-immutable-proptypes';
 | 
			
		||||
 | 
			
		||||
//  `ImmutablePureComponent` gives us `updateOnProps` and
 | 
			
		||||
//  `updateOnStates`:
 | 
			
		||||
import ImmutablePureComponent from 'react-immutable-pure-component';
 | 
			
		||||
 | 
			
		||||
//  These are our various media types:
 | 
			
		||||
import MediaGallery from './media_gallery';
 | 
			
		||||
import VideoPlayer from './video_player';
 | 
			
		||||
 | 
			
		||||
//  These are our core status components:
 | 
			
		||||
import StatusPrepend from './status_prepend';
 | 
			
		||||
import StatusHeader from './status_header';
 | 
			
		||||
import StatusContent from './status_content';
 | 
			
		||||
import StatusActionBar from './status_action_bar';
 | 
			
		||||
 | 
			
		||||
//  This is used to schedule tasks at the browser's convenience:
 | 
			
		||||
import scheduleIdleTask from '../features/ui/util/schedule_idle_task';
 | 
			
		||||
 | 
			
		||||
                            /* * * * */
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
The `<Status>` component:
 | 
			
		||||
-------------------------
 | 
			
		||||
 | 
			
		||||
The `<Status>` component is a container for statuses. It consists of a
 | 
			
		||||
few parts:
 | 
			
		||||
 | 
			
		||||
 -  The `<StatusPrepend>`, which contains tangential information about
 | 
			
		||||
    the status, such as who reblogged it.
 | 
			
		||||
 -  The `<StatusHeader>`, which contains the avatar and username of the
 | 
			
		||||
    status author, as well as a media icon and the "collapse" toggle.
 | 
			
		||||
 -  The `<StatusContent>`, which contains the content of the status.
 | 
			
		||||
 -  The `<StatusActionBar>`, which provides actions to be performed
 | 
			
		||||
    on statuses, like reblogging or sending a reply.
 | 
			
		||||
 | 
			
		||||
###  Context
 | 
			
		||||
 | 
			
		||||
 -  __`router` (`PropTypes.object`) :__
 | 
			
		||||
    We need to get our router from the surrounding React context.
 | 
			
		||||
 | 
			
		||||
###  Props
 | 
			
		||||
 | 
			
		||||
 -  __`id` (`PropTypes.number`) :__
 | 
			
		||||
    The id of the status.
 | 
			
		||||
 | 
			
		||||
 -  __`status` (`ImmutablePropTypes.map`) :__
 | 
			
		||||
    The status object, straight from the store.
 | 
			
		||||
 | 
			
		||||
 -  __`account` (`ImmutablePropTypes.map`) :__
 | 
			
		||||
    Don't be confused by this one! This is **not** the account which
 | 
			
		||||
    posted the status, but the associated account with any further
 | 
			
		||||
    action (eg, a reblog or a favourite).
 | 
			
		||||
 | 
			
		||||
 -  __`settings` (`ImmutablePropTypes.map`) :__
 | 
			
		||||
    These are our local settings, fetched from our store. We need this
 | 
			
		||||
    to determine how best to collapse our statuses, among other things.
 | 
			
		||||
 | 
			
		||||
 -  __`me` (`PropTypes.number`) :__
 | 
			
		||||
    This is the id of the currently-signed-in user.
 | 
			
		||||
 | 
			
		||||
 -  __`onFavourite`, `onReblog`, `onModalReblog`, `onDelete`,
 | 
			
		||||
    `onMention`, `onMute`, `onMuteConversation`, onBlock`, `onReport`,
 | 
			
		||||
    `onOpenMedia`, `onOpenVideo` (`PropTypes.func`) :__
 | 
			
		||||
    These are all functions passed through from the
 | 
			
		||||
    `<StatusContainer>`. We don't deal with them directly here.
 | 
			
		||||
 | 
			
		||||
 -  __`reblogModal`, `deleteModal` (`PropTypes.bool`) :__
 | 
			
		||||
    These tell whether or not the user has modals activated for
 | 
			
		||||
    reblogging and deleting statuses. They are used by the `onReblog`
 | 
			
		||||
    and `onDelete` functions, but we don't deal with them here.
 | 
			
		||||
 | 
			
		||||
 -  __`autoPlayGif` (`PropTypes.bool`) :__
 | 
			
		||||
    This tells the frontend whether or not to autoplay gifs!
 | 
			
		||||
 | 
			
		||||
 -  __`muted` (`PropTypes.bool`) :__
 | 
			
		||||
    This has nothing to do with a user or conversation mute! "Muted" is
 | 
			
		||||
    what Mastodon internally calls the subdued look of statuses in the
 | 
			
		||||
    notifications column. This should be `true` for notifications, and
 | 
			
		||||
    `false` otherwise.
 | 
			
		||||
 | 
			
		||||
 -  __`collapse` (`PropTypes.bool`) :__
 | 
			
		||||
    This prop signals a directive from a higher power to (un)collapse
 | 
			
		||||
    a status. Most of the time it should be `undefined`, in which case
 | 
			
		||||
    we do nothing.
 | 
			
		||||
 | 
			
		||||
 -  __`prepend` (`PropTypes.string`) :__
 | 
			
		||||
    The type of prepend: `'reblogged_by'`, `'reblog'`, or
 | 
			
		||||
    `'favourite'`.
 | 
			
		||||
 | 
			
		||||
 -  __`withDismiss` (`PropTypes.bool`) :__
 | 
			
		||||
    Whether or not the status can be dismissed. Used for notifications.
 | 
			
		||||
 | 
			
		||||
 -  __`intersectionObserverWrapper` (`PropTypes.object`) :__
 | 
			
		||||
    This holds our intersection observer. In Mastodon parlance,
 | 
			
		||||
    an "intersection" is just when the status is viewable onscreen.
 | 
			
		||||
 | 
			
		||||
###  State
 | 
			
		||||
 | 
			
		||||
 -  __`isExpanded` :__
 | 
			
		||||
    Should be either `true`, `false`, or `null`. The meanings of
 | 
			
		||||
    these values are as follows:
 | 
			
		||||
 | 
			
		||||
     -  __`true` :__ The status contains a CW and the CW is expanded.
 | 
			
		||||
     -  __`false` :__ The status is collapsed.
 | 
			
		||||
     -  __`null` :__ The status is not collapsed or expanded.
 | 
			
		||||
 | 
			
		||||
 -  __`isIntersecting` :__
 | 
			
		||||
    This boolean tells us whether or not the status is currently
 | 
			
		||||
    onscreen.
 | 
			
		||||
 | 
			
		||||
 -  __`isHidden` :__
 | 
			
		||||
    This boolean tells us if the status has been unrendered to save
 | 
			
		||||
    CPUs.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
export default class Status extends ImmutablePureComponent {
 | 
			
		||||
 | 
			
		||||
  static contextTypes = {
 | 
			
		||||
    router                      : PropTypes.object,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  static propTypes = {
 | 
			
		||||
    id                          : PropTypes.number,
 | 
			
		||||
    status                      : ImmutablePropTypes.map,
 | 
			
		||||
    account                     : ImmutablePropTypes.map,
 | 
			
		||||
    settings                    : ImmutablePropTypes.map,
 | 
			
		||||
    me                          : PropTypes.number,
 | 
			
		||||
    onFavourite                 : PropTypes.func,
 | 
			
		||||
    onReblog                    : PropTypes.func,
 | 
			
		||||
    onModalReblog               : PropTypes.func,
 | 
			
		||||
    onDelete                    : PropTypes.func,
 | 
			
		||||
    onMention                   : PropTypes.func,
 | 
			
		||||
    onMute                      : PropTypes.func,
 | 
			
		||||
    onMuteConversation          : PropTypes.func,
 | 
			
		||||
    onBlock                     : PropTypes.func,
 | 
			
		||||
    onReport                    : PropTypes.func,
 | 
			
		||||
    onOpenMedia                 : PropTypes.func,
 | 
			
		||||
    onOpenVideo                 : PropTypes.func,
 | 
			
		||||
    reblogModal                 : PropTypes.bool,
 | 
			
		||||
    deleteModal                 : PropTypes.bool,
 | 
			
		||||
    autoPlayGif                 : PropTypes.bool,
 | 
			
		||||
    muted                       : PropTypes.bool,
 | 
			
		||||
    collapse                    : PropTypes.bool,
 | 
			
		||||
    prepend                     : PropTypes.string,
 | 
			
		||||
    withDismiss                 : PropTypes.bool,
 | 
			
		||||
    intersectionObserverWrapper : PropTypes.object,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  state = {
 | 
			
		||||
    isExpanded                  : null,
 | 
			
		||||
    isIntersecting              : true,
 | 
			
		||||
    isHidden                    : false,
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
###  Implementation
 | 
			
		||||
 | 
			
		||||
####  `updateOnProps` and `updateOnStates`.
 | 
			
		||||
 | 
			
		||||
`updateOnProps` and `updateOnStates` tell the component when to update.
 | 
			
		||||
We specify them explicitly because some of our props are dynamically=
 | 
			
		||||
generated functions, which would otherwise always trigger an update.
 | 
			
		||||
Of course, this means that if we add an important prop, we will need
 | 
			
		||||
to remember to specify it here.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  updateOnProps = [
 | 
			
		||||
    'status',
 | 
			
		||||
    'account',
 | 
			
		||||
    'settings',
 | 
			
		||||
    'prepend',
 | 
			
		||||
    'me',
 | 
			
		||||
    'boostModal',
 | 
			
		||||
    'autoPlayGif',
 | 
			
		||||
    'muted',
 | 
			
		||||
    'collapse',
 | 
			
		||||
  ]
 | 
			
		||||
 | 
			
		||||
  updateOnStates = [
 | 
			
		||||
    'isExpanded',
 | 
			
		||||
  ]
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `componentWillReceiveProps()`.
 | 
			
		||||
 | 
			
		||||
If our settings have changed to disable collapsed statuses, then we
 | 
			
		||||
need to make sure that we uncollapse every one. We do that by watching
 | 
			
		||||
for changes to `settings.collapsed.enabled` in
 | 
			
		||||
`componentWillReceiveProps()`.
 | 
			
		||||
 | 
			
		||||
We also need to watch for changes on the `collapse` prop---if this
 | 
			
		||||
changes to anything other than `undefined`, then we need to collapse or
 | 
			
		||||
uncollapse our status accordingly.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  componentWillReceiveProps (nextProps) {
 | 
			
		||||
    if (!nextProps.settings.getIn(['collapsed', 'enabled'])) {
 | 
			
		||||
      this.setExpansion(true);
 | 
			
		||||
    } else if (
 | 
			
		||||
      nextProps.collapse !== this.props.collapse &&
 | 
			
		||||
      nextProps.collapse !== undefined
 | 
			
		||||
    ) this.setExpansion(nextProps.collapse ? false : null);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `componentDidMount()`.
 | 
			
		||||
 | 
			
		||||
When mounting, we just check to see if our status should be collapsed,
 | 
			
		||||
and collapse it if so. We don't need to worry about whether collapsing
 | 
			
		||||
is enabled here, because `setExpansion()` already takes that into
 | 
			
		||||
account.
 | 
			
		||||
 | 
			
		||||
The cases where a status should be collapsed are:
 | 
			
		||||
 | 
			
		||||
 -  The `collapse` prop has been set to `true`
 | 
			
		||||
 -  The user has decided in local settings to collapse all statuses.
 | 
			
		||||
 -  The user has decided to collapse all notifications ('muted'
 | 
			
		||||
    statuses).
 | 
			
		||||
 -  The user has decided to collapse long statuses and the status is
 | 
			
		||||
    over 400px (without media, or 650px with).
 | 
			
		||||
 -  The status is a reply and the user has decided to collapse all
 | 
			
		||||
    replies.
 | 
			
		||||
 -  The status contains media and the user has decided to collapse all
 | 
			
		||||
    statuses with media.
 | 
			
		||||
 | 
			
		||||
We also start up our intersection observer to monitor our statuses.
 | 
			
		||||
`componentMounted` lets us know that everything has been set up
 | 
			
		||||
properly and our intersection observer is good to go.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  componentDidMount () {
 | 
			
		||||
    const { node, handleIntersection } = this;
 | 
			
		||||
    const {
 | 
			
		||||
      status,
 | 
			
		||||
      settings,
 | 
			
		||||
      collapse,
 | 
			
		||||
      muted,
 | 
			
		||||
      id,
 | 
			
		||||
      intersectionObserverWrapper,
 | 
			
		||||
    } = this.props;
 | 
			
		||||
    const autoCollapseSettings = settings.getIn(['collapsed', 'auto']);
 | 
			
		||||
 | 
			
		||||
    if (
 | 
			
		||||
      collapse ||
 | 
			
		||||
      autoCollapseSettings.get('all') || (
 | 
			
		||||
        autoCollapseSettings.get('notifications') && muted
 | 
			
		||||
      ) || (
 | 
			
		||||
        autoCollapseSettings.get('lengthy') &&
 | 
			
		||||
        node.clientHeight > (
 | 
			
		||||
          status.get('media_attachments').size && !muted ? 650 : 400
 | 
			
		||||
        )
 | 
			
		||||
      ) || (
 | 
			
		||||
        autoCollapseSettings.get('replies') &&
 | 
			
		||||
        status.get('in_reply_to_id', null) !== null
 | 
			
		||||
      ) || (
 | 
			
		||||
        autoCollapseSettings.get('media') &&
 | 
			
		||||
        !(status.get('spoiler_text').length) &&
 | 
			
		||||
        status.get('media_attachments').size
 | 
			
		||||
      )
 | 
			
		||||
    ) this.setExpansion(false);
 | 
			
		||||
 | 
			
		||||
    if (!intersectionObserverWrapper) return;
 | 
			
		||||
    else intersectionObserverWrapper.observe(
 | 
			
		||||
      id,
 | 
			
		||||
      node,
 | 
			
		||||
      handleIntersection
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
    this.componentMounted = true;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `shouldComponentUpdate()`.
 | 
			
		||||
 | 
			
		||||
If the status is about to be both offscreen (not intersecting) and
 | 
			
		||||
hidden, then we only need to update it if it's not that way currently.
 | 
			
		||||
If the status is moving from offscreen to onscreen, then we *have* to
 | 
			
		||||
re-render, so that we can unhide the element if necessary.
 | 
			
		||||
 | 
			
		||||
If neither of these cases are true, we can leave it up to our
 | 
			
		||||
`updateOnProps` and `updateOnStates` arrays.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  shouldComponentUpdate (nextProps, nextState) {
 | 
			
		||||
    switch (true) {
 | 
			
		||||
    case !nextState.isIntersecting && nextState.isHidden:
 | 
			
		||||
      return this.state.isIntersecting || !this.state.isHidden;
 | 
			
		||||
    case nextState.isIntersecting && !this.state.isIntersecting:
 | 
			
		||||
      return true;
 | 
			
		||||
    default:
 | 
			
		||||
      return super.shouldComponentUpdate(nextProps, nextState);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `componentDidUpdate()`.
 | 
			
		||||
 | 
			
		||||
If our component is being rendered for any reason and an update has
 | 
			
		||||
triggered, this will save its height.
 | 
			
		||||
 | 
			
		||||
This is, frankly, a bit overkill, as the only instance when we
 | 
			
		||||
actually *need* to update the height right now should be when the
 | 
			
		||||
value of `isExpanded` has changed. But it makes for more readable
 | 
			
		||||
code and prevents bugs in the future where the height isn't set
 | 
			
		||||
properly after some change.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  componentDidUpdate () {
 | 
			
		||||
    if (
 | 
			
		||||
      this.state.isIntersecting || !this.state.isHidden
 | 
			
		||||
    ) this.saveHeight();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `componentWillUnmount()`.
 | 
			
		||||
 | 
			
		||||
If our component is about to unmount, then we'd better unset
 | 
			
		||||
`this.componentMounted`.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  componentWillUnmount () {
 | 
			
		||||
    this.componentMounted = false;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `handleIntersection()`.
 | 
			
		||||
 | 
			
		||||
`handleIntersection()` either hides the status (if it is offscreen) or
 | 
			
		||||
unhides it (if it is onscreen). It's called by
 | 
			
		||||
`intersectionObserverWrapper.observe()`.
 | 
			
		||||
 | 
			
		||||
If our status isn't intersecting, we schedule an idle task (using the
 | 
			
		||||
aptly-named `scheduleIdleTask()`) to hide the status at the next
 | 
			
		||||
available opportunity.
 | 
			
		||||
 | 
			
		||||
tootsuite/mastodon left us with the following enlightening comment
 | 
			
		||||
regarding this function:
 | 
			
		||||
 | 
			
		||||
>   Edge 15 doesn't support isIntersecting, but we can infer it
 | 
			
		||||
 | 
			
		||||
It then implements a polyfill (intersectionRect.height > 0) which isn't
 | 
			
		||||
actually sufficient. The short answer is, this behaviour isn't really
 | 
			
		||||
supported on Edge but we can get kinda close.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  handleIntersection = (entry) => {
 | 
			
		||||
    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()`.
 | 
			
		||||
 | 
			
		||||
This function will hide the status if we're still not intersecting.
 | 
			
		||||
Hiding the status means that it will just render an empty div instead
 | 
			
		||||
of actual content, which saves RAMS and CPUs or some such.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  hideIfNotIntersecting = () => {
 | 
			
		||||
    if (!this.componentMounted) return;
 | 
			
		||||
    this.setState(
 | 
			
		||||
      (prevState) => ({ isHidden: !prevState.isIntersecting })
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `saveHeight()`.
 | 
			
		||||
 | 
			
		||||
`saveHeight()` saves the height of our status so that when whe hide it
 | 
			
		||||
we preserve its dimensions. We only want to store our height, though,
 | 
			
		||||
if our status has content (otherwise, it would imply that it is
 | 
			
		||||
already hidden).
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  saveHeight = () => {
 | 
			
		||||
    if (this.node && this.node.children.length) {
 | 
			
		||||
      this.height = this.node.getBoundingClientRect().height;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `setExpansion()`.
 | 
			
		||||
 | 
			
		||||
`setExpansion()` sets the value of `isExpanded` in our state. It takes
 | 
			
		||||
one argument, `value`, which gives the desired value for `isExpanded`.
 | 
			
		||||
The default for this argument is `null`.
 | 
			
		||||
 | 
			
		||||
`setExpansion()` automatically checks for us whether toot collapsing
 | 
			
		||||
is enabled, so we don't have to.
 | 
			
		||||
 | 
			
		||||
We use a `switch` statement to simplify our code.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  setExpansion = (value) => {
 | 
			
		||||
    switch (true) {
 | 
			
		||||
    case value === undefined || value === null:
 | 
			
		||||
      this.setState({ isExpanded: null });
 | 
			
		||||
      break;
 | 
			
		||||
    case !value && this.props.settings.getIn(['collapsed', 'enabled']):
 | 
			
		||||
      this.setState({ isExpanded: false });
 | 
			
		||||
      break;
 | 
			
		||||
    case !!value:
 | 
			
		||||
      this.setState({ isExpanded: true });
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `handleRef()`.
 | 
			
		||||
 | 
			
		||||
`handleRef()` just saves a reference to our status node to `this.node`.
 | 
			
		||||
It also saves our height, in case the height of our node has changed.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  handleRef = (node) => {
 | 
			
		||||
    this.node = node;
 | 
			
		||||
    this.saveHeight();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `parseClick()`.
 | 
			
		||||
 | 
			
		||||
`parseClick()` takes a click event and responds appropriately.
 | 
			
		||||
If our status is collapsed, then clicking on it should uncollapse it.
 | 
			
		||||
If `Shift` is held, then clicking on it should collapse it.
 | 
			
		||||
Otherwise, we open the url handed to us in `destination`, if
 | 
			
		||||
applicable.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  parseClick = (e, destination) => {
 | 
			
		||||
    const { router } = this.context;
 | 
			
		||||
    const { status } = this.props;
 | 
			
		||||
    const { isExpanded } = this.state;
 | 
			
		||||
    if (destination === undefined) {
 | 
			
		||||
      destination = `/statuses/${
 | 
			
		||||
        status.getIn(['reblog', 'id'], status.get('id'))
 | 
			
		||||
      }`;
 | 
			
		||||
    }
 | 
			
		||||
    if (e.button === 0) {
 | 
			
		||||
      if (isExpanded === false) this.setExpansion(null);
 | 
			
		||||
      else if (e.shiftKey) {
 | 
			
		||||
        this.setExpansion(false);
 | 
			
		||||
        document.getSelection().removeAllRanges();
 | 
			
		||||
      } else router.history.push(destination);
 | 
			
		||||
      e.preventDefault();
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `render()`.
 | 
			
		||||
 | 
			
		||||
`render()` actually puts our element on the screen. The particulars of
 | 
			
		||||
this operation are further explained in the code below.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  render () {
 | 
			
		||||
    const { parseClick, setExpansion, handleRef } = this;
 | 
			
		||||
    const {
 | 
			
		||||
      status,
 | 
			
		||||
      account,
 | 
			
		||||
      settings,
 | 
			
		||||
      collapsed,
 | 
			
		||||
      muted,
 | 
			
		||||
      prepend,
 | 
			
		||||
      intersectionObserverWrapper,
 | 
			
		||||
      onOpenVideo,
 | 
			
		||||
      onOpenMedia,
 | 
			
		||||
      autoPlayGif,
 | 
			
		||||
      ...other
 | 
			
		||||
    } = this.props;
 | 
			
		||||
    const { isExpanded, isIntersecting, isHidden } = this.state;
 | 
			
		||||
    let background = null;
 | 
			
		||||
    let attachments = null;
 | 
			
		||||
    let media = null;
 | 
			
		||||
    let mediaIcon = null;
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
If we don't have a status, then we don't render anything.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
    if (status === null) {
 | 
			
		||||
      return null;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
If our status is offscreen and hidden, then we render an empty <div> in
 | 
			
		||||
its place. We fill it with "content" but note that opacity is set to 0.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
    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 user backgrounds for collapsed statuses are enabled, then we
 | 
			
		||||
initialize our background accordingly. This will only be rendered if
 | 
			
		||||
the status is collapsed.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
    if (
 | 
			
		||||
      settings.getIn(['collapsed', 'backgrounds', 'user_backgrounds'])
 | 
			
		||||
    ) background = status.getIn(['account', 'header']);
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
This handles our media attachments. Note that we don't show media on
 | 
			
		||||
muted (notification) statuses. If the media type is unknown, then we
 | 
			
		||||
simply ignore it.
 | 
			
		||||
 | 
			
		||||
After we have generated our appropriate media element and stored it in
 | 
			
		||||
`media`, we snatch the thumbnail to use as our `background` if media
 | 
			
		||||
backgrounds for collapsed statuses are enabled.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
    attachments = status.get('media_attachments');
 | 
			
		||||
    if (attachments.size && !muted) {
 | 
			
		||||
      if (attachments.some((item) => item.get('type') === 'unknown')) {
 | 
			
		||||
 | 
			
		||||
      } else if (
 | 
			
		||||
        attachments.getIn([0, 'type']) === 'video'
 | 
			
		||||
      ) {
 | 
			
		||||
        media = (  //  Media type is 'video'
 | 
			
		||||
          <VideoPlayer
 | 
			
		||||
            media={attachments.get(0)}
 | 
			
		||||
            sensitive={status.get('sensitive')}
 | 
			
		||||
            letterbox={settings.getIn(['media', 'letterbox'])}
 | 
			
		||||
            fullwidth={settings.getIn(['media', 'fullwidth'])}
 | 
			
		||||
            height={250}
 | 
			
		||||
            onOpenVideo={onOpenVideo}
 | 
			
		||||
          />
 | 
			
		||||
        );
 | 
			
		||||
        mediaIcon = 'video-camera';
 | 
			
		||||
      } else {  //  Media type is 'image' or 'gifv'
 | 
			
		||||
        media = (
 | 
			
		||||
          <MediaGallery
 | 
			
		||||
            media={attachments}
 | 
			
		||||
            sensitive={status.get('sensitive')}
 | 
			
		||||
            letterbox={settings.getIn(['media', 'letterbox'])}
 | 
			
		||||
            fullwidth={settings.getIn(['media', 'fullwidth'])}
 | 
			
		||||
            height={250}
 | 
			
		||||
            onOpenMedia={onOpenMedia}
 | 
			
		||||
            autoPlayGif={autoPlayGif}
 | 
			
		||||
          />
 | 
			
		||||
        );
 | 
			
		||||
        mediaIcon = 'picture-o';
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (
 | 
			
		||||
        !status.get('sensitive') &&
 | 
			
		||||
        !(status.get('spoiler_text').length > 0) &&
 | 
			
		||||
        settings.getIn(['collapsed', 'backgrounds', 'preview_images'])
 | 
			
		||||
      ) background = attachments.getIn([0, 'preview_url']);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
Finally, we can render our status. We just put the pieces together
 | 
			
		||||
from above. We only render the action bar if the status isn't
 | 
			
		||||
collapsed.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
      <article
 | 
			
		||||
        className={
 | 
			
		||||
          `status${
 | 
			
		||||
            muted ? ' muted' : ''
 | 
			
		||||
          } status-${status.get('visibility')}${
 | 
			
		||||
            isExpanded === false ? ' collapsed' : ''
 | 
			
		||||
          }${
 | 
			
		||||
            isExpanded === false && background ? ' has-background' : ''
 | 
			
		||||
          }`
 | 
			
		||||
        }
 | 
			
		||||
        style={{
 | 
			
		||||
          backgroundImage: (
 | 
			
		||||
            isExpanded === false && background ?
 | 
			
		||||
            `url(${background})` :
 | 
			
		||||
            'none'
 | 
			
		||||
          ),
 | 
			
		||||
        }}
 | 
			
		||||
        ref={handleRef}
 | 
			
		||||
      >
 | 
			
		||||
        {prepend && account ? (
 | 
			
		||||
          <StatusPrepend
 | 
			
		||||
            type={prepend}
 | 
			
		||||
            account={account}
 | 
			
		||||
            parseClick={parseClick}
 | 
			
		||||
          />
 | 
			
		||||
        ) : null}
 | 
			
		||||
        <StatusHeader
 | 
			
		||||
          account={status.get('account')}
 | 
			
		||||
          friend={account}
 | 
			
		||||
          mediaIcon={mediaIcon}
 | 
			
		||||
          visibility={status.get('visibility')}
 | 
			
		||||
          collapsible={settings.getIn(['collapsed', 'enabled'])}
 | 
			
		||||
          collapsed={isExpanded === false}
 | 
			
		||||
          parseClick={parseClick}
 | 
			
		||||
          setExpansion={setExpansion}
 | 
			
		||||
        />
 | 
			
		||||
        <StatusContent
 | 
			
		||||
          status={status}
 | 
			
		||||
          media={media}
 | 
			
		||||
          mediaIcon={mediaIcon}
 | 
			
		||||
          expanded={isExpanded}
 | 
			
		||||
          setExpansion={this.setExpansion}
 | 
			
		||||
          onHeightUpdate={this.saveHeight}
 | 
			
		||||
          parseClick={parseClick}
 | 
			
		||||
        />
 | 
			
		||||
        {isExpanded !== false ? (
 | 
			
		||||
          <StatusActionBar
 | 
			
		||||
            {...other}
 | 
			
		||||
            status={status}
 | 
			
		||||
            account={status.get('account')}
 | 
			
		||||
          />
 | 
			
		||||
        ) : null}
 | 
			
		||||
      </article>
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,156 +0,0 @@
 | 
			
		||||
import React from 'react';
 | 
			
		||||
import ImmutablePropTypes from 'react-immutable-proptypes';
 | 
			
		||||
import PropTypes from 'prop-types';
 | 
			
		||||
import IconButton from './icon_button';
 | 
			
		||||
import DropdownMenu from './dropdown_menu';
 | 
			
		||||
import { defineMessages, injectIntl } from 'react-intl';
 | 
			
		||||
import ImmutablePureComponent from 'react-immutable-pure-component';
 | 
			
		||||
import RelativeTimestamp from './relative_timestamp';
 | 
			
		||||
 | 
			
		||||
const messages = defineMessages({
 | 
			
		||||
  delete: { id: 'status.delete', defaultMessage: 'Delete' },
 | 
			
		||||
  mention: { id: 'status.mention', defaultMessage: 'Mention @{name}' },
 | 
			
		||||
  mute: { id: 'account.mute', defaultMessage: 'Mute @{name}' },
 | 
			
		||||
  block: { id: 'account.block', defaultMessage: 'Block @{name}' },
 | 
			
		||||
  reply: { id: 'status.reply', defaultMessage: 'Reply' },
 | 
			
		||||
  replyAll: { id: 'status.replyAll', defaultMessage: 'Reply to thread' },
 | 
			
		||||
  reblog: { id: 'status.reblog', defaultMessage: 'Boost' },
 | 
			
		||||
  cannot_reblog: { id: 'status.cannot_reblog', defaultMessage: 'This post cannot be boosted' },
 | 
			
		||||
  favourite: { id: 'status.favourite', defaultMessage: 'Favourite' },
 | 
			
		||||
  open: { id: 'status.open', defaultMessage: 'Expand this status' },
 | 
			
		||||
  report: { id: 'status.report', defaultMessage: 'Report @{name}' },
 | 
			
		||||
  muteConversation: { id: 'status.mute_conversation', defaultMessage: 'Mute conversation' },
 | 
			
		||||
  unmuteConversation: { id: 'status.unmute_conversation', defaultMessage: 'Unmute conversation' },
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
@injectIntl
 | 
			
		||||
export default class StatusActionBar extends ImmutablePureComponent {
 | 
			
		||||
 | 
			
		||||
  static contextTypes = {
 | 
			
		||||
    router: PropTypes.object,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  static propTypes = {
 | 
			
		||||
    status: ImmutablePropTypes.map.isRequired,
 | 
			
		||||
    onReply: PropTypes.func,
 | 
			
		||||
    onFavourite: PropTypes.func,
 | 
			
		||||
    onReblog: PropTypes.func,
 | 
			
		||||
    onDelete: PropTypes.func,
 | 
			
		||||
    onMention: PropTypes.func,
 | 
			
		||||
    onMute: PropTypes.func,
 | 
			
		||||
    onBlock: PropTypes.func,
 | 
			
		||||
    onReport: PropTypes.func,
 | 
			
		||||
    onMuteConversation: PropTypes.func,
 | 
			
		||||
    me: PropTypes.number.isRequired,
 | 
			
		||||
    withDismiss: PropTypes.bool,
 | 
			
		||||
    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',
 | 
			
		||||
    'me',
 | 
			
		||||
    'withDismiss',
 | 
			
		||||
  ]
 | 
			
		||||
 | 
			
		||||
  handleReplyClick = () => {
 | 
			
		||||
    this.props.onReply(this.props.status, this.context.router.history);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleFavouriteClick = () => {
 | 
			
		||||
    this.props.onFavourite(this.props.status);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleReblogClick = (e) => {
 | 
			
		||||
    this.props.onReblog(this.props.status, e);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleDeleteClick = () => {
 | 
			
		||||
    this.props.onDelete(this.props.status);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleMentionClick = () => {
 | 
			
		||||
    this.props.onMention(this.props.status.get('account'), this.context.router.history);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleMuteClick = () => {
 | 
			
		||||
    this.props.onMute(this.props.status.get('account'));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleBlockClick = () => {
 | 
			
		||||
    this.props.onBlock(this.props.status.get('account'));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleOpen = () => {
 | 
			
		||||
    this.context.router.history.push(`/statuses/${this.props.status.get('id')}`);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleReport = () => {
 | 
			
		||||
    this.props.onReport(this.props.status);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleConversationMuteClick = () => {
 | 
			
		||||
    this.props.onMuteConversation(this.props.status);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  render () {
 | 
			
		||||
    const { status, me, intl, withDismiss } = this.props;
 | 
			
		||||
    const reblogDisabled = status.get('visibility') === 'private' || status.get('visibility') === 'direct';
 | 
			
		||||
    const mutingConversation = status.get('muted');
 | 
			
		||||
 | 
			
		||||
    let menu = [];
 | 
			
		||||
    let reblogIcon = 'retweet';
 | 
			
		||||
    let replyIcon;
 | 
			
		||||
    let replyTitle;
 | 
			
		||||
 | 
			
		||||
    menu.push({ text: intl.formatMessage(messages.open), action: this.handleOpen });
 | 
			
		||||
    menu.push(null);
 | 
			
		||||
 | 
			
		||||
    if (withDismiss) {
 | 
			
		||||
      menu.push({ text: intl.formatMessage(mutingConversation ? messages.unmuteConversation : messages.muteConversation), action: this.handleConversationMuteClick });
 | 
			
		||||
      menu.push(null);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (status.getIn(['account', 'id']) === me) {
 | 
			
		||||
      menu.push({ text: intl.formatMessage(messages.delete), action: this.handleDeleteClick });
 | 
			
		||||
    } else {
 | 
			
		||||
      menu.push({ text: intl.formatMessage(messages.mention, { name: status.getIn(['account', 'username']) }), action: this.handleMentionClick });
 | 
			
		||||
      menu.push(null);
 | 
			
		||||
      menu.push({ text: intl.formatMessage(messages.mute, { name: status.getIn(['account', 'username']) }), action: this.handleMuteClick });
 | 
			
		||||
      menu.push({ text: intl.formatMessage(messages.block, { name: status.getIn(['account', 'username']) }), action: this.handleBlockClick });
 | 
			
		||||
      menu.push({ text: intl.formatMessage(messages.report, { name: status.getIn(['account', 'username']) }), action: this.handleReport });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /*
 | 
			
		||||
    if (status.get('visibility') === 'direct') {
 | 
			
		||||
      reblogIcon = 'envelope';
 | 
			
		||||
    } else if (status.get('visibility') === 'private') {
 | 
			
		||||
      reblogIcon = 'lock';
 | 
			
		||||
    }
 | 
			
		||||
    */
 | 
			
		||||
 | 
			
		||||
    if (status.get('in_reply_to_id', null) === null) {
 | 
			
		||||
      replyIcon = 'reply';
 | 
			
		||||
      replyTitle = intl.formatMessage(messages.reply);
 | 
			
		||||
    } else {
 | 
			
		||||
      replyIcon = 'reply-all';
 | 
			
		||||
      replyTitle = intl.formatMessage(messages.replyAll);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
      <div className='status__action-bar'>
 | 
			
		||||
        <IconButton className='status__action-bar-button' title={replyTitle} icon={replyIcon} onClick={this.handleReplyClick} />
 | 
			
		||||
        <IconButton className='status__action-bar-button' disabled={reblogDisabled} active={status.get('reblogged')} title={reblogDisabled ? intl.formatMessage(messages.cannot_reblog) : intl.formatMessage(messages.reblog)} icon={reblogIcon} onClick={this.handleReblogClick} />
 | 
			
		||||
        <IconButton className='status__action-bar-button star-icon' animate active={status.get('favourited')} title={intl.formatMessage(messages.favourite)} icon='star' onClick={this.handleFavouriteClick} />
 | 
			
		||||
 | 
			
		||||
        <div className='status__action-bar-dropdown'>
 | 
			
		||||
          <DropdownMenu items={menu} icon='ellipsis-h' size={18} direction='right' ariaLabel='More' />
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <a href={status.get('url')} className='status__relative-time' target='_blank' rel='noopener'><RelativeTimestamp timestamp={status.get('created_at')} /></a>
 | 
			
		||||
      </div>
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,236 +0,0 @@
 | 
			
		||||
import React from 'react';
 | 
			
		||||
import ImmutablePropTypes from 'react-immutable-proptypes';
 | 
			
		||||
import escapeTextContentForBrowser from 'escape-html';
 | 
			
		||||
import PropTypes from 'prop-types';
 | 
			
		||||
import emojify from '../emoji';
 | 
			
		||||
import { isRtl } from '../rtl';
 | 
			
		||||
import { FormattedMessage } from 'react-intl';
 | 
			
		||||
import Permalink from './permalink';
 | 
			
		||||
 | 
			
		||||
export default class StatusContent extends React.PureComponent {
 | 
			
		||||
 | 
			
		||||
  static contextTypes = {
 | 
			
		||||
    router: PropTypes.object,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  static propTypes = {
 | 
			
		||||
    status: ImmutablePropTypes.map.isRequired,
 | 
			
		||||
    expanded: PropTypes.oneOf([true, false, null]),
 | 
			
		||||
    setExpansion: PropTypes.func,
 | 
			
		||||
    onHeightUpdate: PropTypes.func,
 | 
			
		||||
    media: PropTypes.element,
 | 
			
		||||
    mediaIcon: PropTypes.string,
 | 
			
		||||
    parseClick: PropTypes.func,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  state = {
 | 
			
		||||
    hidden: true,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  componentDidMount () {
 | 
			
		||||
    const node  = this.node;
 | 
			
		||||
    const links = node.querySelectorAll('a');
 | 
			
		||||
 | 
			
		||||
    for (var i = 0; i < links.length; ++i) {
 | 
			
		||||
      let link    = links[i];
 | 
			
		||||
      let mention = this.props.status.get('mentions').find(item => link.href === item.get('url'));
 | 
			
		||||
 | 
			
		||||
      if (mention) {
 | 
			
		||||
        link.addEventListener('click', this.onMentionClick.bind(this, mention), false);
 | 
			
		||||
        link.setAttribute('title', mention.get('acct'));
 | 
			
		||||
      } else if (link.textContent[0] === '#' || (link.previousSibling && link.previousSibling.textContent && link.previousSibling.textContent[link.previousSibling.textContent.length - 1] === '#')) {
 | 
			
		||||
        link.addEventListener('click', this.onHashtagClick.bind(this, link.text), false);
 | 
			
		||||
      } else {
 | 
			
		||||
        link.addEventListener('click', this.onLinkClick.bind(this), false);
 | 
			
		||||
        link.setAttribute('target', '_blank');
 | 
			
		||||
        link.setAttribute('rel', 'noopener');
 | 
			
		||||
        link.setAttribute('title', link.href);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  componentDidUpdate () {
 | 
			
		||||
    if (this.props.onHeightUpdate) {
 | 
			
		||||
      this.props.onHeightUpdate();
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  onLinkClick = (e) => {
 | 
			
		||||
    if (this.props.expanded === false) {
 | 
			
		||||
      if (this.props.parseClick) this.props.parseClick(e);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  onMentionClick = (mention, e) => {
 | 
			
		||||
    if (this.props.parseClick) {
 | 
			
		||||
      this.props.parseClick(e, `/accounts/${mention.get('id')}`);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  onHashtagClick = (hashtag, e) => {
 | 
			
		||||
    hashtag = hashtag.replace(/^#/, '').toLowerCase();
 | 
			
		||||
 | 
			
		||||
    if (this.props.parseClick) {
 | 
			
		||||
      this.props.parseClick(e, `/timelines/tag/${hashtag}`);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleMouseDown = (e) => {
 | 
			
		||||
    this.startXY = [e.clientX, e.clientY];
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleMouseUp = (e) => {
 | 
			
		||||
    const { parseClick } = this.props;
 | 
			
		||||
 | 
			
		||||
    if (!this.startXY) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const [ startX, startY ] = this.startXY;
 | 
			
		||||
    const [ deltaX, deltaY ] = [Math.abs(e.clientX - startX), Math.abs(e.clientY - startY)];
 | 
			
		||||
 | 
			
		||||
    if (e.target.localName === 'button' || e.target.localName === 'a' || (e.target.parentNode && (e.target.parentNode.localName === 'button' || e.target.parentNode.localName === 'a'))) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (deltaX + deltaY < 5 && e.button === 0 && parseClick) {
 | 
			
		||||
      parseClick(e);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    this.startXY = null;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleSpoilerClick = (e) => {
 | 
			
		||||
    e.preventDefault();
 | 
			
		||||
 | 
			
		||||
    if (this.props.setExpansion) {
 | 
			
		||||
      this.props.setExpansion(this.props.expanded ? null : true);
 | 
			
		||||
    } else {
 | 
			
		||||
      this.setState({ hidden: !this.state.hidden });
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  setRef = (c) => {
 | 
			
		||||
    this.node = c;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  render () {
 | 
			
		||||
    const { status, media, mediaIcon } = this.props;
 | 
			
		||||
 | 
			
		||||
    const hidden = (
 | 
			
		||||
      this.props.setExpansion ?
 | 
			
		||||
      !this.props.expanded :
 | 
			
		||||
      this.state.hidden
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
    const content = { __html: emojify(status.get('content')) };
 | 
			
		||||
    const spoilerContent = {
 | 
			
		||||
      __html: emojify(escapeTextContentForBrowser(
 | 
			
		||||
        status.get('spoiler_text', '')
 | 
			
		||||
      )),
 | 
			
		||||
    };
 | 
			
		||||
    const directionStyle = { direction: 'ltr' };
 | 
			
		||||
 | 
			
		||||
    if (isRtl(status.get('search_index'))) {
 | 
			
		||||
      directionStyle.direction = 'rtl';
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (status.get('spoiler_text').length > 0) {
 | 
			
		||||
      let mentionsPlaceholder = '';
 | 
			
		||||
 | 
			
		||||
      const mentionLinks = status.get('mentions').map(item => (
 | 
			
		||||
        <Permalink
 | 
			
		||||
          to={`/accounts/${item.get('id')}`}
 | 
			
		||||
          href={item.get('url')}
 | 
			
		||||
          key={item.get('id')}
 | 
			
		||||
          className='mention'
 | 
			
		||||
        >
 | 
			
		||||
          @<span>{item.get('username')}</span>
 | 
			
		||||
        </Permalink>
 | 
			
		||||
      )).reduce((aggregate, item) => [...aggregate, item, ' '], []);
 | 
			
		||||
 | 
			
		||||
      const toggleText = hidden ? [
 | 
			
		||||
        <FormattedMessage
 | 
			
		||||
          id='status.show_more'
 | 
			
		||||
          defaultMessage='Show more'
 | 
			
		||||
          key='0'
 | 
			
		||||
        />,
 | 
			
		||||
        mediaIcon ? (
 | 
			
		||||
          <i
 | 
			
		||||
            className={
 | 
			
		||||
              `fa fa-fw fa-${mediaIcon} status__content__spoiler-icon`
 | 
			
		||||
            }
 | 
			
		||||
            aria-hidden='true'
 | 
			
		||||
            key='1'
 | 
			
		||||
          />
 | 
			
		||||
        ) : null,
 | 
			
		||||
      ] : [
 | 
			
		||||
        <FormattedMessage
 | 
			
		||||
          id='status.show_less'
 | 
			
		||||
          defaultMessage='Show less'
 | 
			
		||||
          key='0'
 | 
			
		||||
        />,
 | 
			
		||||
      ];
 | 
			
		||||
 | 
			
		||||
      if (hidden) {
 | 
			
		||||
        mentionsPlaceholder = <div>{mentionLinks}</div>;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      return (
 | 
			
		||||
        <div className='status__content status__content--with-action' ref={this.setRef}>
 | 
			
		||||
          <p
 | 
			
		||||
            style={{ marginBottom: hidden && status.get('mentions').isEmpty() ? '0px' : null }}
 | 
			
		||||
            onMouseDown={this.handleMouseDown}
 | 
			
		||||
            onMouseUp={this.handleMouseUp}
 | 
			
		||||
          >
 | 
			
		||||
            <span dangerouslySetInnerHTML={spoilerContent} />
 | 
			
		||||
            {' '}
 | 
			
		||||
            <button tabIndex='0' className='status__content__spoiler-link' onClick={this.handleSpoilerClick}>
 | 
			
		||||
              {toggleText}
 | 
			
		||||
            </button>
 | 
			
		||||
          </p>
 | 
			
		||||
 | 
			
		||||
          {mentionsPlaceholder}
 | 
			
		||||
 | 
			
		||||
          <div className={`status__content__spoiler ${!hidden ? 'status__content__spoiler--visible' : ''}`}>
 | 
			
		||||
            <div
 | 
			
		||||
              style={directionStyle}
 | 
			
		||||
              onMouseDown={this.handleMouseDown}
 | 
			
		||||
              onMouseUp={this.handleMouseUp}
 | 
			
		||||
              dangerouslySetInnerHTML={content}
 | 
			
		||||
            />
 | 
			
		||||
            {media}
 | 
			
		||||
          </div>
 | 
			
		||||
 | 
			
		||||
        </div>
 | 
			
		||||
      );
 | 
			
		||||
    } else if (this.props.parseClick) {
 | 
			
		||||
      return (
 | 
			
		||||
        <div
 | 
			
		||||
          ref={this.setRef}
 | 
			
		||||
          className='status__content status__content--with-action'
 | 
			
		||||
          style={directionStyle}
 | 
			
		||||
        >
 | 
			
		||||
          <div
 | 
			
		||||
            onMouseDown={this.handleMouseDown}
 | 
			
		||||
            onMouseUp={this.handleMouseUp}
 | 
			
		||||
            dangerouslySetInnerHTML={content}
 | 
			
		||||
          />
 | 
			
		||||
          {media}
 | 
			
		||||
        </div>
 | 
			
		||||
      );
 | 
			
		||||
    } else {
 | 
			
		||||
      return (
 | 
			
		||||
        <div
 | 
			
		||||
          ref={this.setRef}
 | 
			
		||||
          className='status__content'
 | 
			
		||||
          style={directionStyle}
 | 
			
		||||
        >
 | 
			
		||||
          <div dangerouslySetInnerHTML={content} />
 | 
			
		||||
          {media}
 | 
			
		||||
        </div>
 | 
			
		||||
      );
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,249 +0,0 @@
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
`<StatusHeader>`
 | 
			
		||||
================
 | 
			
		||||
 | 
			
		||||
Originally a part of `<Status>`, but extracted into a separate
 | 
			
		||||
component for better documentation and maintainance by
 | 
			
		||||
@kibi@glitch.social as a part of glitch-soc/mastodon.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
                            /* * * * */
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
Imports:
 | 
			
		||||
--------
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
//  Our standard React imports:
 | 
			
		||||
import React from 'react';
 | 
			
		||||
import PropTypes from 'prop-types';
 | 
			
		||||
import ImmutablePropTypes from 'react-immutable-proptypes';
 | 
			
		||||
 | 
			
		||||
//  We will need internationalization in this component:
 | 
			
		||||
import { defineMessages, injectIntl } from 'react-intl';
 | 
			
		||||
 | 
			
		||||
//  The various components used when constructing our header:
 | 
			
		||||
import Avatar from './avatar';
 | 
			
		||||
import AvatarOverlay from './avatar_overlay';
 | 
			
		||||
import DisplayName from './display_name';
 | 
			
		||||
import IconButton from './icon_button';
 | 
			
		||||
 | 
			
		||||
                            /* * * * */
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
Inital setup:
 | 
			
		||||
-------------
 | 
			
		||||
 | 
			
		||||
The `messages` constant is used to define any messages that we need
 | 
			
		||||
from inside props. In our case, these are the `collapse` and
 | 
			
		||||
`uncollapse` messages used with our collapse/uncollapse buttons.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
const messages = defineMessages({
 | 
			
		||||
  collapse: { id: 'status.collapse', defaultMessage: 'Collapse' },
 | 
			
		||||
  uncollapse: { id: 'status.uncollapse', defaultMessage: 'Uncollapse' },
 | 
			
		||||
  public: { id: 'privacy.public.short', defaultMessage: 'Public' },
 | 
			
		||||
  unlisted: { id: 'privacy.unlisted.short', defaultMessage: 'Unlisted' },
 | 
			
		||||
  private: { id: 'privacy.private.short', defaultMessage: 'Followers-only' },
 | 
			
		||||
  direct: { id: 'privacy.direct.short', defaultMessage: 'Direct' },
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
                            /* * * * */
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
The `<StatusHeader>` component:
 | 
			
		||||
-------------------------------
 | 
			
		||||
 | 
			
		||||
The `<StatusHeader>` component wraps together the header information
 | 
			
		||||
(avatar, display name) and upper buttons and icons (collapsing, media
 | 
			
		||||
icons) into a single `<header>` element.
 | 
			
		||||
 | 
			
		||||
###  Props
 | 
			
		||||
 | 
			
		||||
 -  __`account`, `friend` (`ImmutablePropTypes.map`) :__
 | 
			
		||||
    These give the accounts associated with the status. `account` is
 | 
			
		||||
    the author of the post; `friend` will have their avatar appear
 | 
			
		||||
    in the overlay if provided.
 | 
			
		||||
 | 
			
		||||
 -  __`mediaIcon` (`PropTypes.string`) :__
 | 
			
		||||
    If a mediaIcon should be placed in the header, this string
 | 
			
		||||
    specifies it.
 | 
			
		||||
 | 
			
		||||
 -  __`collapsible`, `collapsed` (`PropTypes.bool`) :__
 | 
			
		||||
    These props tell whether a post can be, and is, collapsed.
 | 
			
		||||
 | 
			
		||||
 -  __`parseClick` (`PropTypes.func`) :__
 | 
			
		||||
    This function will be called when the user clicks inside the header
 | 
			
		||||
    information.
 | 
			
		||||
 | 
			
		||||
 -  __`setExpansion` (`PropTypes.func`) :__
 | 
			
		||||
    This function is used to set the expansion state of the post.
 | 
			
		||||
 | 
			
		||||
 -  __`intl` (`PropTypes.object`) :__
 | 
			
		||||
    This is our internationalization object, provided by
 | 
			
		||||
    `injectIntl()`.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
@injectIntl
 | 
			
		||||
export default class StatusHeader extends React.PureComponent {
 | 
			
		||||
 | 
			
		||||
  static propTypes = {
 | 
			
		||||
    account: ImmutablePropTypes.map.isRequired,
 | 
			
		||||
    friend: ImmutablePropTypes.map,
 | 
			
		||||
    mediaIcon: PropTypes.string,
 | 
			
		||||
    collapsible: PropTypes.bool,
 | 
			
		||||
    collapsed: PropTypes.bool,
 | 
			
		||||
    parseClick: PropTypes.func.isRequired,
 | 
			
		||||
    setExpansion: PropTypes.func.isRequired,
 | 
			
		||||
    intl: PropTypes.object.isRequired,
 | 
			
		||||
    visibility: PropTypes.string,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
###  Implementation
 | 
			
		||||
 | 
			
		||||
####  `handleCollapsedClick()`.
 | 
			
		||||
 | 
			
		||||
`handleCollapsedClick()` is just a simple callback for our collapsing
 | 
			
		||||
button. It calls `setExpansion` to set the collapsed state of the
 | 
			
		||||
status.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  handleCollapsedClick = (e) => {
 | 
			
		||||
    const { collapsed, setExpansion } = this.props;
 | 
			
		||||
    if (e.button === 0) {
 | 
			
		||||
      setExpansion(collapsed ? null : false);
 | 
			
		||||
      e.preventDefault();
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `handleAccountClick()`.
 | 
			
		||||
 | 
			
		||||
`handleAccountClick()` handles any clicks on the header info. It calls
 | 
			
		||||
`parseClick()` with our `account` as the anticipatory `destination`.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  handleAccountClick = (e) => {
 | 
			
		||||
    const { account, parseClick } = this.props;
 | 
			
		||||
    parseClick(e, `/accounts/${+account.get('id')}`);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `render()`.
 | 
			
		||||
 | 
			
		||||
`render()` actually puts our element on the screen. `<StatusHeader>`
 | 
			
		||||
has a very straightforward rendering process.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  render () {
 | 
			
		||||
    const {
 | 
			
		||||
      account,
 | 
			
		||||
      friend,
 | 
			
		||||
      mediaIcon,
 | 
			
		||||
      collapsible,
 | 
			
		||||
      collapsed,
 | 
			
		||||
      intl,
 | 
			
		||||
      visibility,
 | 
			
		||||
    } = this.props;
 | 
			
		||||
 | 
			
		||||
    const visibilityClass = {
 | 
			
		||||
      public: 'globe',
 | 
			
		||||
      unlisted: 'unlock-alt',
 | 
			
		||||
      private: 'lock',
 | 
			
		||||
      direct: 'envelope',
 | 
			
		||||
    }[visibility];
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
      <header className='status__info'>
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
We have to include the status icons before the header content because
 | 
			
		||||
it is rendered as a float.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        <div className='status__info__icons'>
 | 
			
		||||
          {mediaIcon ? (
 | 
			
		||||
            <i
 | 
			
		||||
              className={`fa fa-fw fa-${mediaIcon}`}
 | 
			
		||||
              aria-hidden='true'
 | 
			
		||||
            />
 | 
			
		||||
          ) : null}
 | 
			
		||||
          {(
 | 
			
		||||
            <i
 | 
			
		||||
              className={`status__visibility-icon fa fa-fw fa-${visibilityClass}`}
 | 
			
		||||
              title={intl.formatMessage(messages[visibility])}
 | 
			
		||||
              aria-hidden='true'
 | 
			
		||||
            />
 | 
			
		||||
          )}
 | 
			
		||||
          {collapsible ? (
 | 
			
		||||
            <IconButton
 | 
			
		||||
              className='status__collapse-button'
 | 
			
		||||
              animate flip
 | 
			
		||||
              active={collapsed}
 | 
			
		||||
              title={
 | 
			
		||||
                collapsed ?
 | 
			
		||||
                intl.formatMessage(messages.uncollapse) :
 | 
			
		||||
                intl.formatMessage(messages.collapse)
 | 
			
		||||
              }
 | 
			
		||||
              icon='angle-double-up'
 | 
			
		||||
              onClick={this.handleCollapsedClick}
 | 
			
		||||
            />
 | 
			
		||||
          ) : null}
 | 
			
		||||
        </div>
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
This begins our header content. It is all wrapped inside of a link
 | 
			
		||||
which gets handled by `handleAccountClick`. We use an `<AvatarOverlay>`
 | 
			
		||||
if we have a `friend` and a normal `<Avatar>` if we don't.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        <a
 | 
			
		||||
          href={account.get('url')}
 | 
			
		||||
          className='status__display-name'
 | 
			
		||||
          onClick={this.handleAccountClick}
 | 
			
		||||
        >
 | 
			
		||||
          <div className='status__avatar'>{
 | 
			
		||||
            friend ? (
 | 
			
		||||
              <AvatarOverlay
 | 
			
		||||
                staticSrc={account.get('avatar_static')}
 | 
			
		||||
                overlaySrc={friend.get('avatar_static')}
 | 
			
		||||
              />
 | 
			
		||||
            ) : (
 | 
			
		||||
              <Avatar
 | 
			
		||||
                src={account.get('avatar')}
 | 
			
		||||
                staticSrc={account.get('avatar_static')}
 | 
			
		||||
                size={48}
 | 
			
		||||
              />
 | 
			
		||||
            )
 | 
			
		||||
          }</div>
 | 
			
		||||
          <DisplayName account={account} />
 | 
			
		||||
        </a>
 | 
			
		||||
 | 
			
		||||
      </header>
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -2,7 +2,7 @@ import React from 'react';
 | 
			
		||||
import ImmutablePropTypes from 'react-immutable-proptypes';
 | 
			
		||||
import { ScrollContainer } from 'react-router-scroll';
 | 
			
		||||
import PropTypes from 'prop-types';
 | 
			
		||||
import StatusContainer from '../containers/status_container';
 | 
			
		||||
import StatusContainer from '../../glitch/containers/status';
 | 
			
		||||
import LoadMore from './load_more';
 | 
			
		||||
import ImmutablePureComponent from 'react-immutable-pure-component';
 | 
			
		||||
import IntersectionObserverWrapper from '../features/ui/util/intersection_observer_wrapper';
 | 
			
		||||
 
 | 
			
		||||
@@ -1,164 +0,0 @@
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
`<StatusPrepend>`
 | 
			
		||||
=================
 | 
			
		||||
 | 
			
		||||
Originally a part of `<Status>`, but extracted into a separate
 | 
			
		||||
component for better documentation and maintainance by
 | 
			
		||||
@kibi@glitch.social as a part of glitch-soc/mastodon.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
                            /* * * * */
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
Imports:
 | 
			
		||||
--------
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
//  Our standard React imports:
 | 
			
		||||
import React from 'react';
 | 
			
		||||
import PropTypes from 'prop-types';
 | 
			
		||||
import ImmutablePropTypes from 'react-immutable-proptypes';
 | 
			
		||||
 | 
			
		||||
//  This helps us process our text:
 | 
			
		||||
import emojify from '../emoji';
 | 
			
		||||
import escapeTextContentForBrowser from 'escape-html';
 | 
			
		||||
import { FormattedMessage } from 'react-intl';
 | 
			
		||||
 | 
			
		||||
                            /* * * * */
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
The `<StatusPrepend>` component:
 | 
			
		||||
--------------------------------
 | 
			
		||||
 | 
			
		||||
The `<StatusPrepend>` component holds a status's prepend, ie the text
 | 
			
		||||
that says “X reblogged this,” etc. It is represented by an `<aside>`
 | 
			
		||||
element.
 | 
			
		||||
 | 
			
		||||
###  Props
 | 
			
		||||
 | 
			
		||||
 -  __`type` (`PropTypes.string`) :__
 | 
			
		||||
    The type of prepend. One of `'reblogged_by'`, `'reblog'`,
 | 
			
		||||
    `'favourite'`.
 | 
			
		||||
 | 
			
		||||
 -  __`account` (`ImmutablePropTypes.map`) :__
 | 
			
		||||
    The account associated with the prepend.
 | 
			
		||||
 | 
			
		||||
 -  __`parseClick` (`PropTypes.func.isRequired`) :__
 | 
			
		||||
    Our click parsing function.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
export default class StatusPrepend extends React.PureComponent {
 | 
			
		||||
 | 
			
		||||
  static propTypes = {
 | 
			
		||||
    type: PropTypes.string.isRequired,
 | 
			
		||||
    account: ImmutablePropTypes.map.isRequired,
 | 
			
		||||
    parseClick: PropTypes.func.isRequired,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
###  Implementation
 | 
			
		||||
 | 
			
		||||
####  `handleClick()`.
 | 
			
		||||
 | 
			
		||||
This is just a small wrapper for `parseClick()` that gets fired when
 | 
			
		||||
an account link is clicked.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  handleClick = (e) => {
 | 
			
		||||
    const { account, parseClick } = this.props;
 | 
			
		||||
    parseClick(e, `/accounts/${+account.get('id')}`);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `<Message>`.
 | 
			
		||||
 | 
			
		||||
`<Message>` is a quick functional React component which renders the
 | 
			
		||||
actual prepend message based on our provided `type`. First we create a
 | 
			
		||||
`link` for the account's name, and then use `<FormattedMessage>` to
 | 
			
		||||
generate the message.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  Message = () => {
 | 
			
		||||
    const { type, account } = this.props;
 | 
			
		||||
    let link = (
 | 
			
		||||
      <a
 | 
			
		||||
        onClick={this.handleClick}
 | 
			
		||||
        href={account.get('url')}
 | 
			
		||||
        className='status__display-name'
 | 
			
		||||
      >
 | 
			
		||||
        <b
 | 
			
		||||
          dangerouslySetInnerHTML={{
 | 
			
		||||
            __html : emojify(escapeTextContentForBrowser(
 | 
			
		||||
              account.get('display_name') || account.get('username')
 | 
			
		||||
            )),
 | 
			
		||||
          }}
 | 
			
		||||
        />
 | 
			
		||||
      </a>
 | 
			
		||||
    );
 | 
			
		||||
    switch (type) {
 | 
			
		||||
    case 'reblogged_by':
 | 
			
		||||
      return (
 | 
			
		||||
        <FormattedMessage
 | 
			
		||||
          id='status.reblogged_by'
 | 
			
		||||
          defaultMessage='{name} boosted'
 | 
			
		||||
          values={{ name : link }}
 | 
			
		||||
        />
 | 
			
		||||
      );
 | 
			
		||||
    case 'favourite':
 | 
			
		||||
      return (
 | 
			
		||||
        <FormattedMessage
 | 
			
		||||
          id='notification.favourite'
 | 
			
		||||
          defaultMessage='{name} favourited your status'
 | 
			
		||||
          values={{ name : link }}
 | 
			
		||||
        />
 | 
			
		||||
      );
 | 
			
		||||
    case 'reblog':
 | 
			
		||||
      return (
 | 
			
		||||
        <FormattedMessage
 | 
			
		||||
          id='notification.reblog'
 | 
			
		||||
          defaultMessage='{name} boosted your status'
 | 
			
		||||
          values={{ name : link }}
 | 
			
		||||
        />
 | 
			
		||||
      );
 | 
			
		||||
    }
 | 
			
		||||
    return null;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 | 
			
		||||
####  `render()`.
 | 
			
		||||
 | 
			
		||||
Our `render()` is incredibly simple; we just render the icon and then
 | 
			
		||||
the `<Message>` inside of an <aside>.
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
  render () {
 | 
			
		||||
    const { Message } = this;
 | 
			
		||||
    const { type } = this.props;
 | 
			
		||||
 | 
			
		||||
    return !type ? null : (
 | 
			
		||||
      <aside className={type === 'reblogged_by' ? 'status__prepend' : 'notification__message'}>
 | 
			
		||||
        <div className={type === 'reblogged_by' ? 'status__prepend-icon-wrapper' : 'notification__favourite-icon-wrapper'}>
 | 
			
		||||
          <i
 | 
			
		||||
            className={`fa fa-fw fa-${
 | 
			
		||||
              type === 'favourite' ? 'star star-icon' : 'retweet'
 | 
			
		||||
            } status__prepend-icon`}
 | 
			
		||||
          />
 | 
			
		||||
        </div>
 | 
			
		||||
        <Message />
 | 
			
		||||
      </aside>
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,196 +0,0 @@
 | 
			
		||||
import React from 'react';
 | 
			
		||||
import ImmutablePropTypes from 'react-immutable-proptypes';
 | 
			
		||||
import PropTypes from 'prop-types';
 | 
			
		||||
import IconButton from './icon_button';
 | 
			
		||||
import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
 | 
			
		||||
import { isIOS } from '../is_mobile';
 | 
			
		||||
 | 
			
		||||
const messages = defineMessages({
 | 
			
		||||
  toggle_sound: { id: 'video_player.toggle_sound', defaultMessage: 'Toggle sound' },
 | 
			
		||||
  toggle_visible: { id: 'video_player.toggle_visible', defaultMessage: 'Toggle visibility' },
 | 
			
		||||
  expand_video: { id: 'video_player.expand', defaultMessage: 'Expand video' },
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
@injectIntl
 | 
			
		||||
export default class VideoPlayer extends React.PureComponent {
 | 
			
		||||
 | 
			
		||||
  static propTypes = {
 | 
			
		||||
    media: ImmutablePropTypes.map.isRequired,
 | 
			
		||||
    letterbox: PropTypes.bool,
 | 
			
		||||
    fullwidth: PropTypes.bool,
 | 
			
		||||
    height: PropTypes.number,
 | 
			
		||||
    sensitive: PropTypes.bool,
 | 
			
		||||
    intl: PropTypes.object.isRequired,
 | 
			
		||||
    autoplay: PropTypes.bool,
 | 
			
		||||
    onOpenVideo: PropTypes.func.isRequired,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  static defaultProps = {
 | 
			
		||||
    height: 110,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  state = {
 | 
			
		||||
    visible: !this.props.sensitive,
 | 
			
		||||
    preview: true,
 | 
			
		||||
    muted: true,
 | 
			
		||||
    hasAudio: true,
 | 
			
		||||
    videoError: false,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  handleClick = () => {
 | 
			
		||||
    this.setState({ muted: !this.state.muted });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleVideoClick = (e) => {
 | 
			
		||||
    e.stopPropagation();
 | 
			
		||||
 | 
			
		||||
    const node = this.video;
 | 
			
		||||
 | 
			
		||||
    if (node.paused) {
 | 
			
		||||
      node.play();
 | 
			
		||||
    } else {
 | 
			
		||||
      node.pause();
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleOpen = () => {
 | 
			
		||||
    this.setState({ preview: !this.state.preview });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleVisibility = () => {
 | 
			
		||||
    this.setState({
 | 
			
		||||
      visible: !this.state.visible,
 | 
			
		||||
      preview: true,
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleExpand = () => {
 | 
			
		||||
    this.video.pause();
 | 
			
		||||
    this.props.onOpenVideo(this.props.media, this.video.currentTime);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  setRef = (c) => {
 | 
			
		||||
    this.video = c;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleLoadedData = () => {
 | 
			
		||||
    if (('WebkitAppearance' in document.documentElement.style && this.video.audioTracks.length === 0) || this.video.mozHasAudio === false) {
 | 
			
		||||
      this.setState({ hasAudio: false });
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  handleVideoError = () => {
 | 
			
		||||
    this.setState({ videoError: true });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  componentDidMount () {
 | 
			
		||||
    if (!this.video) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    this.video.addEventListener('loadeddata', this.handleLoadedData);
 | 
			
		||||
    this.video.addEventListener('error', this.handleVideoError);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  componentDidUpdate () {
 | 
			
		||||
    if (!this.video) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    this.video.addEventListener('loadeddata', this.handleLoadedData);
 | 
			
		||||
    this.video.addEventListener('error', this.handleVideoError);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  componentWillUnmount () {
 | 
			
		||||
    if (!this.video) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    this.video.removeEventListener('loadeddata', this.handleLoadedData);
 | 
			
		||||
    this.video.removeEventListener('error', this.handleVideoError);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  render () {
 | 
			
		||||
    const { media, intl, letterbox, fullwidth, height, sensitive, autoplay } = this.props;
 | 
			
		||||
 | 
			
		||||
    let spoilerButton = (
 | 
			
		||||
      <div className={`status__video-player-spoiler ${this.state.visible ? 'status__video-player-spoiler--visible' : ''}`}>
 | 
			
		||||
        <IconButton overlay title={intl.formatMessage(messages.toggle_visible)} icon={this.state.visible ? 'eye' : 'eye-slash'} onClick={this.handleVisibility} />
 | 
			
		||||
      </div>
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
    let expandButton = (
 | 
			
		||||
      <div className='status__video-player-expand'>
 | 
			
		||||
        <IconButton overlay title={intl.formatMessage(messages.expand_video)} icon='expand' onClick={this.handleExpand} />
 | 
			
		||||
      </div>
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
    let muteButton = '';
 | 
			
		||||
 | 
			
		||||
    if (this.state.hasAudio) {
 | 
			
		||||
      muteButton = (
 | 
			
		||||
        <div className='status__video-player-mute'>
 | 
			
		||||
          <IconButton overlay title={intl.formatMessage(messages.toggle_sound)} icon={this.state.muted ? 'volume-off' : 'volume-up'} onClick={this.handleClick} />
 | 
			
		||||
        </div>
 | 
			
		||||
      );
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (!this.state.visible) {
 | 
			
		||||
      if (sensitive) {
 | 
			
		||||
        return (
 | 
			
		||||
          <div role='button' tabIndex='0' style={{ height: `${height}px` }} className={`media-spoiler ${fullwidth ? 'full-width' : ''}`} onClick={this.handleVisibility}>
 | 
			
		||||
            {spoilerButton}
 | 
			
		||||
            <span className='media-spoiler__warning'><FormattedMessage id='status.sensitive_warning' defaultMessage='Sensitive content' /></span>
 | 
			
		||||
            <span className='media-spoiler__trigger'><FormattedMessage id='status.sensitive_toggle' defaultMessage='Click to view' /></span>
 | 
			
		||||
          </div>
 | 
			
		||||
        );
 | 
			
		||||
      } else {
 | 
			
		||||
        return (
 | 
			
		||||
          <div role='button' tabIndex='0' style={{ height: `${height}px` }} className={`media-spoiler ${fullwidth ? 'full-width' : ''}`} onClick={this.handleVisibility}>
 | 
			
		||||
            {spoilerButton}
 | 
			
		||||
            <span className='media-spoiler__warning'><FormattedMessage id='status.media_hidden' defaultMessage='Media hidden' /></span>
 | 
			
		||||
            <span className='media-spoiler__trigger'><FormattedMessage id='status.sensitive_toggle' defaultMessage='Click to view' /></span>
 | 
			
		||||
          </div>
 | 
			
		||||
        );
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (this.state.preview && !autoplay) {
 | 
			
		||||
      return (
 | 
			
		||||
        <div role='button' tabIndex='0' className={`media-spoiler-video ${fullwidth ? 'full-width' : ''}`} style={{ height: `${height}px`, backgroundImage: `url(${media.get('preview_url')})` }} onClick={this.handleOpen}>
 | 
			
		||||
          {spoilerButton}
 | 
			
		||||
          <div className='media-spoiler-video-play-icon'><i className='fa fa-play' /></div>
 | 
			
		||||
        </div>
 | 
			
		||||
      );
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (this.state.videoError) {
 | 
			
		||||
      return (
 | 
			
		||||
        <div style={{ height: `${height}px` }} className='video-error-cover' >
 | 
			
		||||
          <span className='media-spoiler__warning'><FormattedMessage id='video_player.video_error' defaultMessage='Video could not be played' /></span>
 | 
			
		||||
        </div>
 | 
			
		||||
      );
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
      <div className={`status__video-player ${fullwidth ? 'full-width' : ''}`} style={{ height: `${height}px` }}>
 | 
			
		||||
        {spoilerButton}
 | 
			
		||||
        {muteButton}
 | 
			
		||||
        {expandButton}
 | 
			
		||||
 | 
			
		||||
        <video
 | 
			
		||||
          className={`status__video-player-video${letterbox ? ' letterbox' : ''}`}
 | 
			
		||||
          role='button'
 | 
			
		||||
          tabIndex='0'
 | 
			
		||||
          ref={this.setRef}
 | 
			
		||||
          src={media.get('url')}
 | 
			
		||||
          autoPlay={!isIOS()}
 | 
			
		||||
          loop
 | 
			
		||||
          muted={this.state.muted}
 | 
			
		||||
          onClick={this.handleVideoClick}
 | 
			
		||||
        />
 | 
			
		||||
      </div>
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
		Reference in New Issue
	
	Block a user