Add lang attribute to media and poll options (#23891)
				
					
				
			This commit is contained in:
		
							parent
							
								
									730bb3e211
								
							
						
					
					
						commit
						d3eefead30
					
				
					 16 changed files with 79 additions and 24 deletions
				
			
		| 
						 | 
					@ -6,6 +6,7 @@ export default class GIFV extends React.PureComponent {
 | 
				
			||||||
  static propTypes = {
 | 
					  static propTypes = {
 | 
				
			||||||
    src: PropTypes.string.isRequired,
 | 
					    src: PropTypes.string.isRequired,
 | 
				
			||||||
    alt: PropTypes.string,
 | 
					    alt: PropTypes.string,
 | 
				
			||||||
 | 
					    lang: PropTypes.string,
 | 
				
			||||||
    width: PropTypes.number,
 | 
					    width: PropTypes.number,
 | 
				
			||||||
    height: PropTypes.number,
 | 
					    height: PropTypes.number,
 | 
				
			||||||
    onClick: PropTypes.func,
 | 
					    onClick: PropTypes.func,
 | 
				
			||||||
| 
						 | 
					@ -35,7 +36,7 @@ export default class GIFV extends React.PureComponent {
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  render () {
 | 
					  render () {
 | 
				
			||||||
    const { src, width, height, alt } = this.props;
 | 
					    const { src, width, height, alt, lang } = this.props;
 | 
				
			||||||
    const { loading } = this.state;
 | 
					    const { loading } = this.state;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return (
 | 
					    return (
 | 
				
			||||||
| 
						 | 
					@ -48,6 +49,7 @@ export default class GIFV extends React.PureComponent {
 | 
				
			||||||
            tabIndex='0'
 | 
					            tabIndex='0'
 | 
				
			||||||
            aria-label={alt}
 | 
					            aria-label={alt}
 | 
				
			||||||
            title={alt}
 | 
					            title={alt}
 | 
				
			||||||
 | 
					            lang={lang}
 | 
				
			||||||
            onClick={this.handleClick}
 | 
					            onClick={this.handleClick}
 | 
				
			||||||
          />
 | 
					          />
 | 
				
			||||||
        )}
 | 
					        )}
 | 
				
			||||||
| 
						 | 
					@ -58,6 +60,7 @@ export default class GIFV extends React.PureComponent {
 | 
				
			||||||
          tabIndex='0'
 | 
					          tabIndex='0'
 | 
				
			||||||
          aria-label={alt}
 | 
					          aria-label={alt}
 | 
				
			||||||
          title={alt}
 | 
					          title={alt}
 | 
				
			||||||
 | 
					          lang={lang}
 | 
				
			||||||
          muted
 | 
					          muted
 | 
				
			||||||
          loop
 | 
					          loop
 | 
				
			||||||
          autoPlay
 | 
					          autoPlay
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -10,6 +10,7 @@ export default class MediaAttachments extends ImmutablePureComponent {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static propTypes = {
 | 
					  static propTypes = {
 | 
				
			||||||
    status: ImmutablePropTypes.map.isRequired,
 | 
					    status: ImmutablePropTypes.map.isRequired,
 | 
				
			||||||
 | 
					    lang: PropTypes.string,
 | 
				
			||||||
    height: PropTypes.number,
 | 
					    height: PropTypes.number,
 | 
				
			||||||
    width: PropTypes.number,
 | 
					    width: PropTypes.number,
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
| 
						 | 
					@ -48,7 +49,7 @@ export default class MediaAttachments extends ImmutablePureComponent {
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  render () {
 | 
					  render () {
 | 
				
			||||||
    const { status, width, height } = this.props;
 | 
					    const { status, lang, width, height } = this.props;
 | 
				
			||||||
    const mediaAttachments = status.get('media_attachments');
 | 
					    const mediaAttachments = status.get('media_attachments');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (mediaAttachments.size === 0) {
 | 
					    if (mediaAttachments.size === 0) {
 | 
				
			||||||
| 
						 | 
					@ -64,6 +65,7 @@ export default class MediaAttachments extends ImmutablePureComponent {
 | 
				
			||||||
            <Component
 | 
					            <Component
 | 
				
			||||||
              src={audio.get('url')}
 | 
					              src={audio.get('url')}
 | 
				
			||||||
              alt={audio.get('description')}
 | 
					              alt={audio.get('description')}
 | 
				
			||||||
 | 
					              lang={lang || status.get('language')}
 | 
				
			||||||
              width={width}
 | 
					              width={width}
 | 
				
			||||||
              height={height}
 | 
					              height={height}
 | 
				
			||||||
              poster={audio.get('preview_url') || status.getIn(['account', 'avatar_static'])}
 | 
					              poster={audio.get('preview_url') || status.getIn(['account', 'avatar_static'])}
 | 
				
			||||||
| 
						 | 
					@ -87,6 +89,7 @@ export default class MediaAttachments extends ImmutablePureComponent {
 | 
				
			||||||
              blurhash={video.get('blurhash')}
 | 
					              blurhash={video.get('blurhash')}
 | 
				
			||||||
              src={video.get('url')}
 | 
					              src={video.get('url')}
 | 
				
			||||||
              alt={video.get('description')}
 | 
					              alt={video.get('description')}
 | 
				
			||||||
 | 
					              lang={lang || status.get('language')}
 | 
				
			||||||
              width={width}
 | 
					              width={width}
 | 
				
			||||||
              height={height}
 | 
					              height={height}
 | 
				
			||||||
              inline
 | 
					              inline
 | 
				
			||||||
| 
						 | 
					@ -102,6 +105,7 @@ export default class MediaAttachments extends ImmutablePureComponent {
 | 
				
			||||||
          {Component => (
 | 
					          {Component => (
 | 
				
			||||||
            <Component
 | 
					            <Component
 | 
				
			||||||
              media={mediaAttachments}
 | 
					              media={mediaAttachments}
 | 
				
			||||||
 | 
					              lang={lang || status.get('language')}
 | 
				
			||||||
              sensitive={status.get('sensitive')}
 | 
					              sensitive={status.get('sensitive')}
 | 
				
			||||||
              defaultWidth={width}
 | 
					              defaultWidth={width}
 | 
				
			||||||
              height={height}
 | 
					              height={height}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -17,6 +17,7 @@ class Item extends React.PureComponent {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static propTypes = {
 | 
					  static propTypes = {
 | 
				
			||||||
    attachment: ImmutablePropTypes.map.isRequired,
 | 
					    attachment: ImmutablePropTypes.map.isRequired,
 | 
				
			||||||
 | 
					    lang: PropTypes.string,
 | 
				
			||||||
    standalone: PropTypes.bool,
 | 
					    standalone: PropTypes.bool,
 | 
				
			||||||
    index: PropTypes.number.isRequired,
 | 
					    index: PropTypes.number.isRequired,
 | 
				
			||||||
    size: PropTypes.number.isRequired,
 | 
					    size: PropTypes.number.isRequired,
 | 
				
			||||||
| 
						 | 
					@ -78,7 +79,7 @@ class Item extends React.PureComponent {
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  render () {
 | 
					  render () {
 | 
				
			||||||
    const { attachment, index, size, standalone, displayWidth, visible } = this.props;
 | 
					    const { attachment, lang, index, size, standalone, displayWidth, visible } = this.props;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    let width  = 50;
 | 
					    let width  = 50;
 | 
				
			||||||
    let height = 100;
 | 
					    let height = 100;
 | 
				
			||||||
| 
						 | 
					@ -134,7 +135,7 @@ class Item extends React.PureComponent {
 | 
				
			||||||
    if (attachment.get('type') === 'unknown') {
 | 
					    if (attachment.get('type') === 'unknown') {
 | 
				
			||||||
      return (
 | 
					      return (
 | 
				
			||||||
        <div className={classNames('media-gallery__item', { standalone })} key={attachment.get('id')} style={{ left: left, top: top, right: right, bottom: bottom, width: `${width}%`, height: `${height}%` }}>
 | 
					        <div className={classNames('media-gallery__item', { standalone })} key={attachment.get('id')} style={{ left: left, top: top, right: right, bottom: bottom, width: `${width}%`, height: `${height}%` }}>
 | 
				
			||||||
          <a className='media-gallery__item-thumbnail' href={attachment.get('remote_url') || attachment.get('url')} style={{ cursor: 'pointer' }} title={attachment.get('description')} target='_blank' rel='noopener noreferrer'>
 | 
					          <a className='media-gallery__item-thumbnail' href={attachment.get('remote_url') || attachment.get('url')} style={{ cursor: 'pointer' }} title={attachment.get('description')} lang={lang} target='_blank' rel='noopener noreferrer'>
 | 
				
			||||||
            <Blurhash
 | 
					            <Blurhash
 | 
				
			||||||
              hash={attachment.get('blurhash')}
 | 
					              hash={attachment.get('blurhash')}
 | 
				
			||||||
              className='media-gallery__preview'
 | 
					              className='media-gallery__preview'
 | 
				
			||||||
| 
						 | 
					@ -174,6 +175,7 @@ class Item extends React.PureComponent {
 | 
				
			||||||
            sizes={sizes}
 | 
					            sizes={sizes}
 | 
				
			||||||
            alt={attachment.get('description')}
 | 
					            alt={attachment.get('description')}
 | 
				
			||||||
            title={attachment.get('description')}
 | 
					            title={attachment.get('description')}
 | 
				
			||||||
 | 
					            lang={lang}
 | 
				
			||||||
            style={{ objectPosition: `${x}% ${y}%` }}
 | 
					            style={{ objectPosition: `${x}% ${y}%` }}
 | 
				
			||||||
            onLoad={this.handleImageLoad}
 | 
					            onLoad={this.handleImageLoad}
 | 
				
			||||||
          />
 | 
					          />
 | 
				
			||||||
| 
						 | 
					@ -188,6 +190,7 @@ class Item extends React.PureComponent {
 | 
				
			||||||
            className='media-gallery__item-gifv-thumbnail'
 | 
					            className='media-gallery__item-gifv-thumbnail'
 | 
				
			||||||
            aria-label={attachment.get('description')}
 | 
					            aria-label={attachment.get('description')}
 | 
				
			||||||
            title={attachment.get('description')}
 | 
					            title={attachment.get('description')}
 | 
				
			||||||
 | 
					            lang={lang}
 | 
				
			||||||
            role='application'
 | 
					            role='application'
 | 
				
			||||||
            src={attachment.get('url')}
 | 
					            src={attachment.get('url')}
 | 
				
			||||||
            onClick={this.handleClick}
 | 
					            onClick={this.handleClick}
 | 
				
			||||||
| 
						 | 
					@ -227,6 +230,7 @@ class MediaGallery extends React.PureComponent {
 | 
				
			||||||
    sensitive: PropTypes.bool,
 | 
					    sensitive: PropTypes.bool,
 | 
				
			||||||
    standalone: PropTypes.bool,
 | 
					    standalone: PropTypes.bool,
 | 
				
			||||||
    media: ImmutablePropTypes.list.isRequired,
 | 
					    media: ImmutablePropTypes.list.isRequired,
 | 
				
			||||||
 | 
					    lang: PropTypes.string,
 | 
				
			||||||
    size: PropTypes.object,
 | 
					    size: PropTypes.object,
 | 
				
			||||||
    height: PropTypes.number.isRequired,
 | 
					    height: PropTypes.number.isRequired,
 | 
				
			||||||
    onOpenMedia: PropTypes.func.isRequired,
 | 
					    onOpenMedia: PropTypes.func.isRequired,
 | 
				
			||||||
| 
						 | 
					@ -310,9 +314,8 @@ class MediaGallery extends React.PureComponent {
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  render () {
 | 
					  render () {
 | 
				
			||||||
    const { media, intl, sensitive, height, defaultWidth, standalone, autoplay } = this.props;
 | 
					    const { media, lang, intl, sensitive, height, defaultWidth, standalone, autoplay } = this.props;
 | 
				
			||||||
    const { visible } = this.state;
 | 
					    const { visible } = this.state;
 | 
				
			||||||
 | 
					 | 
				
			||||||
    const width = this.state.width || defaultWidth;
 | 
					    const width = this.state.width || defaultWidth;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    let children, spoilerButton;
 | 
					    let children, spoilerButton;
 | 
				
			||||||
| 
						 | 
					@ -333,9 +336,9 @@ class MediaGallery extends React.PureComponent {
 | 
				
			||||||
    const uncached = media.every(attachment => attachment.get('type') === 'unknown');
 | 
					    const uncached = media.every(attachment => attachment.get('type') === 'unknown');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (standalone && this.isFullSizeEligible()) {
 | 
					    if (standalone && this.isFullSizeEligible()) {
 | 
				
			||||||
      children = <Item standalone autoplay={autoplay} onClick={this.handleClick} attachment={media.get(0)} displayWidth={width} visible={visible} />;
 | 
					      children = <Item standalone autoplay={autoplay} onClick={this.handleClick} attachment={media.get(0)} lang={lang} displayWidth={width} visible={visible} />;
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
      children = media.take(4).map((attachment, i) => <Item key={attachment.get('id')} autoplay={autoplay} onClick={this.handleClick} attachment={attachment} index={i} size={size} displayWidth={width} visible={visible || uncached} />);
 | 
					      children = media.take(4).map((attachment, i) => <Item key={attachment.get('id')} autoplay={autoplay} onClick={this.handleClick} attachment={attachment} index={i} lang={lang} size={size} displayWidth={width} visible={visible || uncached} />);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (uncached) {
 | 
					    if (uncached) {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -40,6 +40,7 @@ class Poll extends ImmutablePureComponent {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static propTypes = {
 | 
					  static propTypes = {
 | 
				
			||||||
    poll: ImmutablePropTypes.map,
 | 
					    poll: ImmutablePropTypes.map,
 | 
				
			||||||
 | 
					    lang: PropTypes.string,
 | 
				
			||||||
    intl: PropTypes.object.isRequired,
 | 
					    intl: PropTypes.object.isRequired,
 | 
				
			||||||
    disabled: PropTypes.bool,
 | 
					    disabled: PropTypes.bool,
 | 
				
			||||||
    refresh: PropTypes.func,
 | 
					    refresh: PropTypes.func,
 | 
				
			||||||
| 
						 | 
					@ -126,7 +127,7 @@ class Poll extends ImmutablePureComponent {
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  renderOption (option, optionIndex, showResults) {
 | 
					  renderOption (option, optionIndex, showResults) {
 | 
				
			||||||
    const { poll, disabled, intl } = this.props;
 | 
					    const { poll, lang, disabled, intl } = this.props;
 | 
				
			||||||
    const pollVotesCount  = poll.get('voters_count') || poll.get('votes_count');
 | 
					    const pollVotesCount  = poll.get('voters_count') || poll.get('votes_count');
 | 
				
			||||||
    const percent         = pollVotesCount === 0 ? 0 : (option.get('votes_count') / pollVotesCount) * 100;
 | 
					    const percent         = pollVotesCount === 0 ? 0 : (option.get('votes_count') / pollVotesCount) * 100;
 | 
				
			||||||
    const leading         = poll.get('options').filterNot(other => other.get('title') === option.get('title')).every(other => option.get('votes_count') >= other.get('votes_count'));
 | 
					    const leading         = poll.get('options').filterNot(other => other.get('title') === option.get('title')).every(other => option.get('votes_count') >= other.get('votes_count'));
 | 
				
			||||||
| 
						 | 
					@ -159,6 +160,7 @@ class Poll extends ImmutablePureComponent {
 | 
				
			||||||
              onKeyPress={this.handleOptionKeyPress}
 | 
					              onKeyPress={this.handleOptionKeyPress}
 | 
				
			||||||
              aria-checked={active}
 | 
					              aria-checked={active}
 | 
				
			||||||
              aria-label={option.get('title')}
 | 
					              aria-label={option.get('title')}
 | 
				
			||||||
 | 
					              lang={lang}
 | 
				
			||||||
              data-index={optionIndex}
 | 
					              data-index={optionIndex}
 | 
				
			||||||
            />
 | 
					            />
 | 
				
			||||||
          )}
 | 
					          )}
 | 
				
			||||||
| 
						 | 
					@ -175,6 +177,7 @@ class Poll extends ImmutablePureComponent {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
          <span
 | 
					          <span
 | 
				
			||||||
            className='poll__option__text translate'
 | 
					            className='poll__option__text translate'
 | 
				
			||||||
 | 
					            lang={lang}
 | 
				
			||||||
            dangerouslySetInnerHTML={{ __html: titleEmojified }}
 | 
					            dangerouslySetInnerHTML={{ __html: titleEmojified }}
 | 
				
			||||||
          />
 | 
					          />
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -417,6 +417,7 @@ class Status extends ImmutablePureComponent {
 | 
				
			||||||
              <Component
 | 
					              <Component
 | 
				
			||||||
                src={attachment.get('url')}
 | 
					                src={attachment.get('url')}
 | 
				
			||||||
                alt={attachment.get('description')}
 | 
					                alt={attachment.get('description')}
 | 
				
			||||||
 | 
					                lang={status.get('language')}
 | 
				
			||||||
                poster={attachment.get('preview_url') || status.getIn(['account', 'avatar_static'])}
 | 
					                poster={attachment.get('preview_url') || status.getIn(['account', 'avatar_static'])}
 | 
				
			||||||
                backgroundColor={attachment.getIn(['meta', 'colors', 'background'])}
 | 
					                backgroundColor={attachment.getIn(['meta', 'colors', 'background'])}
 | 
				
			||||||
                foregroundColor={attachment.getIn(['meta', 'colors', 'foreground'])}
 | 
					                foregroundColor={attachment.getIn(['meta', 'colors', 'foreground'])}
 | 
				
			||||||
| 
						 | 
					@ -446,6 +447,7 @@ class Status extends ImmutablePureComponent {
 | 
				
			||||||
                blurhash={attachment.get('blurhash')}
 | 
					                blurhash={attachment.get('blurhash')}
 | 
				
			||||||
                src={attachment.get('url')}
 | 
					                src={attachment.get('url')}
 | 
				
			||||||
                alt={attachment.get('description')}
 | 
					                alt={attachment.get('description')}
 | 
				
			||||||
 | 
					                lang={status.get('language')}
 | 
				
			||||||
                width={this.props.cachedMediaWidth}
 | 
					                width={this.props.cachedMediaWidth}
 | 
				
			||||||
                height={110}
 | 
					                height={110}
 | 
				
			||||||
                inline
 | 
					                inline
 | 
				
			||||||
| 
						 | 
					@ -465,6 +467,7 @@ class Status extends ImmutablePureComponent {
 | 
				
			||||||
            {Component => (
 | 
					            {Component => (
 | 
				
			||||||
              <Component
 | 
					              <Component
 | 
				
			||||||
                media={status.get('media_attachments')}
 | 
					                media={status.get('media_attachments')}
 | 
				
			||||||
 | 
					                lang={status.get('language')}
 | 
				
			||||||
                sensitive={status.get('sensitive')}
 | 
					                sensitive={status.get('sensitive')}
 | 
				
			||||||
                height={110}
 | 
					                height={110}
 | 
				
			||||||
                onOpenMedia={this.handleOpenMedia}
 | 
					                onOpenMedia={this.handleOpenMedia}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -242,7 +242,7 @@ class StatusContent extends React.PureComponent {
 | 
				
			||||||
    );
 | 
					    );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const poll = !!status.get('poll') && (
 | 
					    const poll = !!status.get('poll') && (
 | 
				
			||||||
      <PollContainer pollId={status.get('poll')} />
 | 
					      <PollContainer pollId={status.get('poll')} lang={status.get('language')} />
 | 
				
			||||||
    );
 | 
					    );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (status.get('spoiler_text').length > 0) {
 | 
					    if (status.get('spoiler_text').length > 0) {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -76,6 +76,7 @@ export default class MediaItem extends ImmutablePureComponent {
 | 
				
			||||||
          <img
 | 
					          <img
 | 
				
			||||||
            src={attachment.get('preview_url') || attachment.getIn(['account', 'avatar_static'])}
 | 
					            src={attachment.get('preview_url') || attachment.getIn(['account', 'avatar_static'])}
 | 
				
			||||||
            alt={attachment.get('description')}
 | 
					            alt={attachment.get('description')}
 | 
				
			||||||
 | 
					            lang={status.get('language')}
 | 
				
			||||||
            onLoad={this.handleImageLoad}
 | 
					            onLoad={this.handleImageLoad}
 | 
				
			||||||
          />
 | 
					          />
 | 
				
			||||||
        );
 | 
					        );
 | 
				
			||||||
| 
						 | 
					@ -95,6 +96,7 @@ export default class MediaItem extends ImmutablePureComponent {
 | 
				
			||||||
          <img
 | 
					          <img
 | 
				
			||||||
            src={attachment.get('preview_url')}
 | 
					            src={attachment.get('preview_url')}
 | 
				
			||||||
            alt={attachment.get('description')}
 | 
					            alt={attachment.get('description')}
 | 
				
			||||||
 | 
					            lang={status.get('language')}
 | 
				
			||||||
            style={{ objectPosition: `${x}% ${y}%` }}
 | 
					            style={{ objectPosition: `${x}% ${y}%` }}
 | 
				
			||||||
            onLoad={this.handleImageLoad}
 | 
					            onLoad={this.handleImageLoad}
 | 
				
			||||||
          />
 | 
					          />
 | 
				
			||||||
| 
						 | 
					@ -105,6 +107,7 @@ export default class MediaItem extends ImmutablePureComponent {
 | 
				
			||||||
            className='media-gallery__item-gifv-thumbnail'
 | 
					            className='media-gallery__item-gifv-thumbnail'
 | 
				
			||||||
            aria-label={attachment.get('description')}
 | 
					            aria-label={attachment.get('description')}
 | 
				
			||||||
            title={attachment.get('description')}
 | 
					            title={attachment.get('description')}
 | 
				
			||||||
 | 
					            lang={status.get('language')}
 | 
				
			||||||
            role='application'
 | 
					            role='application'
 | 
				
			||||||
            src={attachment.get('url')}
 | 
					            src={attachment.get('url')}
 | 
				
			||||||
            onMouseEnter={this.handleMouseEnter}
 | 
					            onMouseEnter={this.handleMouseEnter}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -28,6 +28,7 @@ class Audio extends React.PureComponent {
 | 
				
			||||||
  static propTypes = {
 | 
					  static propTypes = {
 | 
				
			||||||
    src: PropTypes.string.isRequired,
 | 
					    src: PropTypes.string.isRequired,
 | 
				
			||||||
    alt: PropTypes.string,
 | 
					    alt: PropTypes.string,
 | 
				
			||||||
 | 
					    lang: PropTypes.string,
 | 
				
			||||||
    poster: PropTypes.string,
 | 
					    poster: PropTypes.string,
 | 
				
			||||||
    duration: PropTypes.number,
 | 
					    duration: PropTypes.number,
 | 
				
			||||||
    width: PropTypes.number,
 | 
					    width: PropTypes.number,
 | 
				
			||||||
| 
						 | 
					@ -458,7 +459,7 @@ class Audio extends React.PureComponent {
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  render () {
 | 
					  render () {
 | 
				
			||||||
    const { src, intl, alt, editable, autoPlay, sensitive, blurhash } = this.props;
 | 
					    const { src, intl, alt, lang, editable, autoPlay, sensitive, blurhash } = this.props;
 | 
				
			||||||
    const { paused, muted, volume, currentTime, duration, buffer, dragging, revealed } = this.state;
 | 
					    const { paused, muted, volume, currentTime, duration, buffer, dragging, revealed } = this.state;
 | 
				
			||||||
    const progress = Math.min((currentTime / duration) * 100, 100);
 | 
					    const progress = Math.min((currentTime / duration) * 100, 100);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -503,6 +504,7 @@ class Audio extends React.PureComponent {
 | 
				
			||||||
          onKeyDown={this.handleAudioKeyDown}
 | 
					          onKeyDown={this.handleAudioKeyDown}
 | 
				
			||||||
          title={alt}
 | 
					          title={alt}
 | 
				
			||||||
          aria-label={alt}
 | 
					          aria-label={alt}
 | 
				
			||||||
 | 
					          lang={lang}
 | 
				
			||||||
        />
 | 
					        />
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        <div className={classNames('spoiler-button', { 'spoiler-button--hidden': revealed || editable })}>
 | 
					        <div className={classNames('spoiler-button', { 'spoiler-button--hidden': revealed || editable })}>
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -139,6 +139,7 @@ class DetailedStatus extends ImmutablePureComponent {
 | 
				
			||||||
          <Audio
 | 
					          <Audio
 | 
				
			||||||
            src={attachment.get('url')}
 | 
					            src={attachment.get('url')}
 | 
				
			||||||
            alt={attachment.get('description')}
 | 
					            alt={attachment.get('description')}
 | 
				
			||||||
 | 
					            lang={status.get('language')}
 | 
				
			||||||
            duration={attachment.getIn(['meta', 'original', 'duration'], 0)}
 | 
					            duration={attachment.getIn(['meta', 'original', 'duration'], 0)}
 | 
				
			||||||
            poster={attachment.get('preview_url') || status.getIn(['account', 'avatar_static'])}
 | 
					            poster={attachment.get('preview_url') || status.getIn(['account', 'avatar_static'])}
 | 
				
			||||||
            backgroundColor={attachment.getIn(['meta', 'colors', 'background'])}
 | 
					            backgroundColor={attachment.getIn(['meta', 'colors', 'background'])}
 | 
				
			||||||
| 
						 | 
					@ -161,6 +162,7 @@ class DetailedStatus extends ImmutablePureComponent {
 | 
				
			||||||
            blurhash={attachment.get('blurhash')}
 | 
					            blurhash={attachment.get('blurhash')}
 | 
				
			||||||
            src={attachment.get('url')}
 | 
					            src={attachment.get('url')}
 | 
				
			||||||
            alt={attachment.get('description')}
 | 
					            alt={attachment.get('description')}
 | 
				
			||||||
 | 
					            lang={status.get('language')}
 | 
				
			||||||
            width={300}
 | 
					            width={300}
 | 
				
			||||||
            height={150}
 | 
					            height={150}
 | 
				
			||||||
            inline
 | 
					            inline
 | 
				
			||||||
| 
						 | 
					@ -176,6 +178,7 @@ class DetailedStatus extends ImmutablePureComponent {
 | 
				
			||||||
            standalone
 | 
					            standalone
 | 
				
			||||||
            sensitive={status.get('sensitive')}
 | 
					            sensitive={status.get('sensitive')}
 | 
				
			||||||
            media={status.get('media_attachments')}
 | 
					            media={status.get('media_attachments')}
 | 
				
			||||||
 | 
					            lang={status.get('language')}
 | 
				
			||||||
            height={300}
 | 
					            height={300}
 | 
				
			||||||
            onOpenMedia={this.props.onOpenMedia}
 | 
					            onOpenMedia={this.props.onOpenMedia}
 | 
				
			||||||
            visible={this.props.showMedia}
 | 
					            visible={this.props.showMedia}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -7,15 +7,17 @@ import ImmutablePureComponent from 'react-immutable-pure-component';
 | 
				
			||||||
import Footer from 'mastodon/features/picture_in_picture/components/footer';
 | 
					import Footer from 'mastodon/features/picture_in_picture/components/footer';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const mapStateToProps = (state, { statusId }) => ({
 | 
					const mapStateToProps = (state, { statusId }) => ({
 | 
				
			||||||
 | 
					  language: state.getIn(['statuses', statusId, 'language']),
 | 
				
			||||||
  accountStaticAvatar: state.getIn(['accounts', state.getIn(['statuses', statusId, 'account']), 'avatar_static']),
 | 
					  accountStaticAvatar: state.getIn(['accounts', state.getIn(['statuses', statusId, 'account']), 'avatar_static']),
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default @connect(mapStateToProps)
 | 
					export default @connect(mapStateToProps, null, null, { forwardRef: true })
 | 
				
			||||||
class AudioModal extends ImmutablePureComponent {
 | 
					class AudioModal extends ImmutablePureComponent {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static propTypes = {
 | 
					  static propTypes = {
 | 
				
			||||||
    media: ImmutablePropTypes.map.isRequired,
 | 
					    media: ImmutablePropTypes.map.isRequired,
 | 
				
			||||||
    statusId: PropTypes.string.isRequired,
 | 
					    statusId: PropTypes.string.isRequired,
 | 
				
			||||||
 | 
					    language: PropTypes.string,
 | 
				
			||||||
    accountStaticAvatar: PropTypes.string.isRequired,
 | 
					    accountStaticAvatar: PropTypes.string.isRequired,
 | 
				
			||||||
    options: PropTypes.shape({
 | 
					    options: PropTypes.shape({
 | 
				
			||||||
      autoPlay: PropTypes.bool,
 | 
					      autoPlay: PropTypes.bool,
 | 
				
			||||||
| 
						 | 
					@ -25,7 +27,7 @@ class AudioModal extends ImmutablePureComponent {
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  render () {
 | 
					  render () {
 | 
				
			||||||
    const { media, accountStaticAvatar, statusId, onClose } = this.props;
 | 
					    const { media, language, accountStaticAvatar, statusId, onClose } = this.props;
 | 
				
			||||||
    const options = this.props.options || {};
 | 
					    const options = this.props.options || {};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return (
 | 
					    return (
 | 
				
			||||||
| 
						 | 
					@ -34,6 +36,7 @@ class AudioModal extends ImmutablePureComponent {
 | 
				
			||||||
          <Audio
 | 
					          <Audio
 | 
				
			||||||
            src={media.get('url')}
 | 
					            src={media.get('url')}
 | 
				
			||||||
            alt={media.get('description')}
 | 
					            alt={media.get('description')}
 | 
				
			||||||
 | 
					            lang={language}
 | 
				
			||||||
            duration={media.getIn(['meta', 'original', 'duration'], 0)}
 | 
					            duration={media.getIn(['meta', 'original', 'duration'], 0)}
 | 
				
			||||||
            height={150}
 | 
					            height={150}
 | 
				
			||||||
            poster={media.get('preview_url') || accountStaticAvatar}
 | 
					            poster={media.get('preview_url') || accountStaticAvatar}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -12,6 +12,7 @@ import RelativeTimestamp from 'mastodon/components/relative_timestamp';
 | 
				
			||||||
import MediaAttachments from 'mastodon/components/media_attachments';
 | 
					import MediaAttachments from 'mastodon/components/media_attachments';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const mapStateToProps = (state, { statusId }) => ({
 | 
					const mapStateToProps = (state, { statusId }) => ({
 | 
				
			||||||
 | 
					  language: state.getIn(['statuses', statusId, 'language']),
 | 
				
			||||||
  versions: state.getIn(['history', statusId, 'items']),
 | 
					  versions: state.getIn(['history', statusId, 'items']),
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -30,11 +31,12 @@ class CompareHistoryModal extends React.PureComponent {
 | 
				
			||||||
    onClose: PropTypes.func.isRequired,
 | 
					    onClose: PropTypes.func.isRequired,
 | 
				
			||||||
    index: PropTypes.number.isRequired,
 | 
					    index: PropTypes.number.isRequired,
 | 
				
			||||||
    statusId: PropTypes.string.isRequired,
 | 
					    statusId: PropTypes.string.isRequired,
 | 
				
			||||||
 | 
					    language: PropTypes.string.isRequired,
 | 
				
			||||||
    versions: ImmutablePropTypes.list.isRequired,
 | 
					    versions: ImmutablePropTypes.list.isRequired,
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  render () {
 | 
					  render () {
 | 
				
			||||||
    const { index, versions, onClose } = this.props;
 | 
					    const { index, versions, language, onClose } = this.props;
 | 
				
			||||||
    const currentVersion = versions.get(index);
 | 
					    const currentVersion = versions.get(index);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const emojiMap = currentVersion.get('emojis').reduce((obj, emoji) => {
 | 
					    const emojiMap = currentVersion.get('emojis').reduce((obj, emoji) => {
 | 
				
			||||||
| 
						 | 
					@ -65,12 +67,12 @@ class CompareHistoryModal extends React.PureComponent {
 | 
				
			||||||
          <div className='status__content'>
 | 
					          <div className='status__content'>
 | 
				
			||||||
            {currentVersion.get('spoiler_text').length > 0 && (
 | 
					            {currentVersion.get('spoiler_text').length > 0 && (
 | 
				
			||||||
              <React.Fragment>
 | 
					              <React.Fragment>
 | 
				
			||||||
                <div className='translate' dangerouslySetInnerHTML={spoilerContent} />
 | 
					                <div className='translate' dangerouslySetInnerHTML={spoilerContent} lang={language} />
 | 
				
			||||||
                <hr />
 | 
					                <hr />
 | 
				
			||||||
              </React.Fragment>
 | 
					              </React.Fragment>
 | 
				
			||||||
            )}
 | 
					            )}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            <div className='status__content__text status__content__text--visible translate' dangerouslySetInnerHTML={content} />
 | 
					            <div className='status__content__text status__content__text--visible translate' dangerouslySetInnerHTML={content} lang={language} />
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            {!!currentVersion.get('poll') && (
 | 
					            {!!currentVersion.get('poll') && (
 | 
				
			||||||
              <div className='poll'>
 | 
					              <div className='poll'>
 | 
				
			||||||
| 
						 | 
					@ -82,6 +84,7 @@ class CompareHistoryModal extends React.PureComponent {
 | 
				
			||||||
                      <span
 | 
					                      <span
 | 
				
			||||||
                        className='poll__option__text translate'
 | 
					                        className='poll__option__text translate'
 | 
				
			||||||
                        dangerouslySetInnerHTML={{ __html: emojify(escapeTextContentForBrowser(option.get('title')), emojiMap) }}
 | 
					                        dangerouslySetInnerHTML={{ __html: emojify(escapeTextContentForBrowser(option.get('title')), emojiMap) }}
 | 
				
			||||||
 | 
					                        lang={language}
 | 
				
			||||||
                      />
 | 
					                      />
 | 
				
			||||||
                    </li>
 | 
					                    </li>
 | 
				
			||||||
                  ))}
 | 
					                  ))}
 | 
				
			||||||
| 
						 | 
					@ -89,7 +92,7 @@ class CompareHistoryModal extends React.PureComponent {
 | 
				
			||||||
              </div>
 | 
					              </div>
 | 
				
			||||||
            )}
 | 
					            )}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            <MediaAttachments status={currentVersion} />
 | 
					            <MediaAttachments status={currentVersion} lang={language} />
 | 
				
			||||||
          </div>
 | 
					          </div>
 | 
				
			||||||
        </div>
 | 
					        </div>
 | 
				
			||||||
      </div>
 | 
					      </div>
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -8,6 +8,7 @@ export default class ImageLoader extends PureComponent {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static propTypes = {
 | 
					  static propTypes = {
 | 
				
			||||||
    alt: PropTypes.string,
 | 
					    alt: PropTypes.string,
 | 
				
			||||||
 | 
					    lang: PropTypes.string,
 | 
				
			||||||
    src: PropTypes.string.isRequired,
 | 
					    src: PropTypes.string.isRequired,
 | 
				
			||||||
    previewSrc: PropTypes.string,
 | 
					    previewSrc: PropTypes.string,
 | 
				
			||||||
    width: PropTypes.number,
 | 
					    width: PropTypes.number,
 | 
				
			||||||
| 
						 | 
					@ -18,6 +19,7 @@ export default class ImageLoader extends PureComponent {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static defaultProps = {
 | 
					  static defaultProps = {
 | 
				
			||||||
    alt: '',
 | 
					    alt: '',
 | 
				
			||||||
 | 
					    lang: '',
 | 
				
			||||||
    width: null,
 | 
					    width: null,
 | 
				
			||||||
    height: null,
 | 
					    height: null,
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
| 
						 | 
					@ -129,7 +131,7 @@ export default class ImageLoader extends PureComponent {
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  render () {
 | 
					  render () {
 | 
				
			||||||
    const { alt, src, width, height, onClick } = this.props;
 | 
					    const { alt, lang, src, width, height, onClick } = this.props;
 | 
				
			||||||
    const { loading } = this.state;
 | 
					    const { loading } = this.state;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const className = classNames('image-loader', {
 | 
					    const className = classNames('image-loader', {
 | 
				
			||||||
| 
						 | 
					@ -154,6 +156,7 @@ export default class ImageLoader extends PureComponent {
 | 
				
			||||||
        ) : (
 | 
					        ) : (
 | 
				
			||||||
          <ZoomableImage
 | 
					          <ZoomableImage
 | 
				
			||||||
            alt={alt}
 | 
					            alt={alt}
 | 
				
			||||||
 | 
					            lang={lang}
 | 
				
			||||||
            src={src}
 | 
					            src={src}
 | 
				
			||||||
            onClick={onClick}
 | 
					            onClick={onClick}
 | 
				
			||||||
            width={width}
 | 
					            width={width}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -3,6 +3,7 @@ import ReactSwipeableViews from 'react-swipeable-views';
 | 
				
			||||||
import ImmutablePropTypes from 'react-immutable-proptypes';
 | 
					import ImmutablePropTypes from 'react-immutable-proptypes';
 | 
				
			||||||
import PropTypes from 'prop-types';
 | 
					import PropTypes from 'prop-types';
 | 
				
			||||||
import Video from 'mastodon/features/video';
 | 
					import Video from 'mastodon/features/video';
 | 
				
			||||||
 | 
					import { connect } from 'react-redux';
 | 
				
			||||||
import classNames from 'classnames';
 | 
					import classNames from 'classnames';
 | 
				
			||||||
import { defineMessages, injectIntl } from 'react-intl';
 | 
					import { defineMessages, injectIntl } from 'react-intl';
 | 
				
			||||||
import IconButton from 'mastodon/components/icon_button';
 | 
					import IconButton from 'mastodon/components/icon_button';
 | 
				
			||||||
| 
						 | 
					@ -20,7 +21,12 @@ const messages = defineMessages({
 | 
				
			||||||
  next: { id: 'lightbox.next', defaultMessage: 'Next' },
 | 
					  next: { id: 'lightbox.next', defaultMessage: 'Next' },
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default @injectIntl
 | 
					const mapStateToProps = (state, { statusId }) => ({
 | 
				
			||||||
 | 
					  language: state.getIn(['statuses', statusId, 'language']),
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default @connect(mapStateToProps, null, null, { forwardRef: true })
 | 
				
			||||||
 | 
					@injectIntl
 | 
				
			||||||
class MediaModal extends ImmutablePureComponent {
 | 
					class MediaModal extends ImmutablePureComponent {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static propTypes = {
 | 
					  static propTypes = {
 | 
				
			||||||
| 
						 | 
					@ -129,7 +135,7 @@ class MediaModal extends ImmutablePureComponent {
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  render () {
 | 
					  render () {
 | 
				
			||||||
    const { media, statusId, intl, onClose } = this.props;
 | 
					    const { media, language, statusId, intl, onClose } = this.props;
 | 
				
			||||||
    const { navigationHidden } = this.state;
 | 
					    const { navigationHidden } = this.state;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const index = this.getIndex();
 | 
					    const index = this.getIndex();
 | 
				
			||||||
| 
						 | 
					@ -149,6 +155,7 @@ class MediaModal extends ImmutablePureComponent {
 | 
				
			||||||
            width={width}
 | 
					            width={width}
 | 
				
			||||||
            height={height}
 | 
					            height={height}
 | 
				
			||||||
            alt={image.get('description')}
 | 
					            alt={image.get('description')}
 | 
				
			||||||
 | 
					            lang={language}
 | 
				
			||||||
            key={image.get('url')}
 | 
					            key={image.get('url')}
 | 
				
			||||||
            onClick={this.toggleNavigation}
 | 
					            onClick={this.toggleNavigation}
 | 
				
			||||||
            zoomButtonHidden={this.state.zoomButtonHidden}
 | 
					            zoomButtonHidden={this.state.zoomButtonHidden}
 | 
				
			||||||
| 
						 | 
					@ -171,6 +178,7 @@ class MediaModal extends ImmutablePureComponent {
 | 
				
			||||||
            onCloseVideo={onClose}
 | 
					            onCloseVideo={onClose}
 | 
				
			||||||
            detailed
 | 
					            detailed
 | 
				
			||||||
            alt={image.get('description')}
 | 
					            alt={image.get('description')}
 | 
				
			||||||
 | 
					            lang={language}
 | 
				
			||||||
            key={image.get('url')}
 | 
					            key={image.get('url')}
 | 
				
			||||||
          />
 | 
					          />
 | 
				
			||||||
        );
 | 
					        );
 | 
				
			||||||
| 
						 | 
					@ -182,6 +190,7 @@ class MediaModal extends ImmutablePureComponent {
 | 
				
			||||||
            height={height}
 | 
					            height={height}
 | 
				
			||||||
            key={image.get('preview_url')}
 | 
					            key={image.get('preview_url')}
 | 
				
			||||||
            alt={image.get('description')}
 | 
					            alt={image.get('description')}
 | 
				
			||||||
 | 
					            lang={language}
 | 
				
			||||||
            onClick={this.toggleNavigation}
 | 
					            onClick={this.toggleNavigation}
 | 
				
			||||||
          />
 | 
					          />
 | 
				
			||||||
        );
 | 
					        );
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2,15 +2,22 @@ import React from 'react';
 | 
				
			||||||
import ImmutablePropTypes from 'react-immutable-proptypes';
 | 
					import ImmutablePropTypes from 'react-immutable-proptypes';
 | 
				
			||||||
import PropTypes from 'prop-types';
 | 
					import PropTypes from 'prop-types';
 | 
				
			||||||
import Video from 'mastodon/features/video';
 | 
					import Video from 'mastodon/features/video';
 | 
				
			||||||
 | 
					import { connect } from 'react-redux';
 | 
				
			||||||
import ImmutablePureComponent from 'react-immutable-pure-component';
 | 
					import ImmutablePureComponent from 'react-immutable-pure-component';
 | 
				
			||||||
import Footer from 'mastodon/features/picture_in_picture/components/footer';
 | 
					import Footer from 'mastodon/features/picture_in_picture/components/footer';
 | 
				
			||||||
import { getAverageFromBlurhash } from 'mastodon/blurhash';
 | 
					import { getAverageFromBlurhash } from 'mastodon/blurhash';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default class VideoModal extends ImmutablePureComponent {
 | 
					const mapStateToProps = (state, { statusId }) => ({
 | 
				
			||||||
 | 
					  language: state.getIn(['statuses', statusId, 'language']),
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default @connect(mapStateToProps, null, null, { forwardRef: true })
 | 
				
			||||||
 | 
					class VideoModal extends ImmutablePureComponent {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static propTypes = {
 | 
					  static propTypes = {
 | 
				
			||||||
    media: ImmutablePropTypes.map.isRequired,
 | 
					    media: ImmutablePropTypes.map.isRequired,
 | 
				
			||||||
    statusId: PropTypes.string,
 | 
					    statusId: PropTypes.string,
 | 
				
			||||||
 | 
					    language: PropTypes.string,
 | 
				
			||||||
    options: PropTypes.shape({
 | 
					    options: PropTypes.shape({
 | 
				
			||||||
      startTime: PropTypes.number,
 | 
					      startTime: PropTypes.number,
 | 
				
			||||||
      autoPlay: PropTypes.bool,
 | 
					      autoPlay: PropTypes.bool,
 | 
				
			||||||
| 
						 | 
					@ -31,7 +38,7 @@ export default class VideoModal extends ImmutablePureComponent {
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  render () {
 | 
					  render () {
 | 
				
			||||||
    const { media, statusId, onClose } = this.props;
 | 
					    const { media, statusId, language, onClose } = this.props;
 | 
				
			||||||
    const options = this.props.options || {};
 | 
					    const options = this.props.options || {};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return (
 | 
					    return (
 | 
				
			||||||
| 
						 | 
					@ -49,6 +56,7 @@ export default class VideoModal extends ImmutablePureComponent {
 | 
				
			||||||
            autoFocus
 | 
					            autoFocus
 | 
				
			||||||
            detailed
 | 
					            detailed
 | 
				
			||||||
            alt={media.get('description')}
 | 
					            alt={media.get('description')}
 | 
				
			||||||
 | 
					            lang={language}
 | 
				
			||||||
          />
 | 
					          />
 | 
				
			||||||
        </div>
 | 
					        </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -96,6 +96,7 @@ class ZoomableImage extends React.PureComponent {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static propTypes = {
 | 
					  static propTypes = {
 | 
				
			||||||
    alt: PropTypes.string,
 | 
					    alt: PropTypes.string,
 | 
				
			||||||
 | 
					    lang: PropTypes.string,
 | 
				
			||||||
    src: PropTypes.string.isRequired,
 | 
					    src: PropTypes.string.isRequired,
 | 
				
			||||||
    width: PropTypes.number,
 | 
					    width: PropTypes.number,
 | 
				
			||||||
    height: PropTypes.number,
 | 
					    height: PropTypes.number,
 | 
				
			||||||
| 
						 | 
					@ -106,6 +107,7 @@ class ZoomableImage extends React.PureComponent {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static defaultProps = {
 | 
					  static defaultProps = {
 | 
				
			||||||
    alt: '',
 | 
					    alt: '',
 | 
				
			||||||
 | 
					    lang: '',
 | 
				
			||||||
    width: null,
 | 
					    width: null,
 | 
				
			||||||
    height: null,
 | 
					    height: null,
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
| 
						 | 
					@ -403,7 +405,7 @@ class ZoomableImage extends React.PureComponent {
 | 
				
			||||||
  };
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  render () {
 | 
					  render () {
 | 
				
			||||||
    const { alt, src, width, height, intl } = this.props;
 | 
					    const { alt, lang, src, width, height, intl } = this.props;
 | 
				
			||||||
    const { scale, lockTranslate } = this.state;
 | 
					    const { scale, lockTranslate } = this.state;
 | 
				
			||||||
    const overflow = scale === MIN_SCALE ? 'hidden' : 'scroll';
 | 
					    const overflow = scale === MIN_SCALE ? 'hidden' : 'scroll';
 | 
				
			||||||
    const zoomButtonShouldHide = this.state.navigationHidden || this.props.zoomButtonHidden || this.state.zoomMatrix.rate <= MIN_SCALE ? 'media-modal__zoom-button--hidden' : '';
 | 
					    const zoomButtonShouldHide = this.state.navigationHidden || this.props.zoomButtonHidden || this.state.zoomMatrix.rate <= MIN_SCALE ? 'media-modal__zoom-button--hidden' : '';
 | 
				
			||||||
| 
						 | 
					@ -431,6 +433,7 @@ class ZoomableImage extends React.PureComponent {
 | 
				
			||||||
            ref={this.setImageRef}
 | 
					            ref={this.setImageRef}
 | 
				
			||||||
            alt={alt}
 | 
					            alt={alt}
 | 
				
			||||||
            title={alt}
 | 
					            title={alt}
 | 
				
			||||||
 | 
					            lang={lang}
 | 
				
			||||||
            src={src}
 | 
					            src={src}
 | 
				
			||||||
            width={width}
 | 
					            width={width}
 | 
				
			||||||
            height={height}
 | 
					            height={height}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -102,6 +102,7 @@ class Video extends React.PureComponent {
 | 
				
			||||||
    frameRate: PropTypes.string,
 | 
					    frameRate: PropTypes.string,
 | 
				
			||||||
    src: PropTypes.string.isRequired,
 | 
					    src: PropTypes.string.isRequired,
 | 
				
			||||||
    alt: PropTypes.string,
 | 
					    alt: PropTypes.string,
 | 
				
			||||||
 | 
					    lang: PropTypes.string,
 | 
				
			||||||
    width: PropTypes.number,
 | 
					    width: PropTypes.number,
 | 
				
			||||||
    height: PropTypes.number,
 | 
					    height: PropTypes.number,
 | 
				
			||||||
    sensitive: PropTypes.bool,
 | 
					    sensitive: PropTypes.bool,
 | 
				
			||||||
| 
						 | 
					@ -524,7 +525,7 @@ class Video extends React.PureComponent {
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  render () {
 | 
					  render () {
 | 
				
			||||||
    const { preview, src, inline, onOpenVideo, onCloseVideo, intl, alt, detailed, sensitive, editable, blurhash, autoFocus } = this.props;
 | 
					    const { preview, src, inline, onOpenVideo, onCloseVideo, intl, alt, lang, detailed, sensitive, editable, blurhash, autoFocus } = this.props;
 | 
				
			||||||
    const { containerWidth, currentTime, duration, volume, buffer, dragging, paused, fullscreen, hovered, muted, revealed } = this.state;
 | 
					    const { containerWidth, currentTime, duration, volume, buffer, dragging, paused, fullscreen, hovered, muted, revealed } = this.state;
 | 
				
			||||||
    const progress = Math.min((currentTime / duration) * 100, 100);
 | 
					    const progress = Math.min((currentTime / duration) * 100, 100);
 | 
				
			||||||
    const playerStyle = {};
 | 
					    const playerStyle = {};
 | 
				
			||||||
| 
						 | 
					@ -585,6 +586,7 @@ class Video extends React.PureComponent {
 | 
				
			||||||
          tabIndex='0'
 | 
					          tabIndex='0'
 | 
				
			||||||
          aria-label={alt}
 | 
					          aria-label={alt}
 | 
				
			||||||
          title={alt}
 | 
					          title={alt}
 | 
				
			||||||
 | 
					          lang={lang}
 | 
				
			||||||
          width={width}
 | 
					          width={width}
 | 
				
			||||||
          height={height}
 | 
					          height={height}
 | 
				
			||||||
          volume={volume}
 | 
					          volume={volume}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue