1
0
Fork 0
mirror of https://github.com/mastodon/mastodon.git synced 2025-01-13 20:04:05 +01:00

Refactor <StatusContent> into TypeScript

This commit is contained in:
Eugen Rochko 2024-12-07 03:26:09 +01:00
parent 72a4da83fd
commit 6be2b59cd3
7 changed files with 378 additions and 282 deletions
app/javascript/mastodon
components
features
direct_timeline/components
report/components
status/components
models

View file

@ -34,7 +34,7 @@ import { DisplayName } from './display_name';
import { getHashtagBarForStatus } from './hashtag_bar';
import { RelativeTimestamp } from './relative_timestamp';
import StatusActionBar from './status_action_bar';
import StatusContent from './status_content';
import { StatusContent } from './status_content';
import { StatusThreadLabel } from './status_thread_label';
import { VisibilityIcon } from './visibility_icon';

View file

@ -1,278 +0,0 @@
import PropTypes from 'prop-types';
import { PureComponent } from 'react';
import { FormattedMessage, injectIntl } from 'react-intl';
import classnames from 'classnames';
import { withRouter } from 'react-router-dom';
import ImmutablePropTypes from 'react-immutable-proptypes';
import { connect } from 'react-redux';
import ChevronRightIcon from '@/material-icons/400-24px/chevron_right.svg?react';
import { Icon } from 'mastodon/components/icon';
import PollContainer from 'mastodon/containers/poll_container';
import { identityContextPropShape, withIdentity } from 'mastodon/identity_context';
import { autoPlayGif, languages as preloadedLanguages } from 'mastodon/initial_state';
const MAX_HEIGHT = 706; // 22px * 32 (+ 2px padding at the top)
/**
*
* @param {any} status
* @returns {string}
*/
export function getStatusContent(status) {
return status.getIn(['translation', 'contentHtml']) || status.get('contentHtml');
}
class TranslateButton extends PureComponent {
static propTypes = {
translation: ImmutablePropTypes.map,
onClick: PropTypes.func,
};
render () {
const { translation, onClick } = this.props;
if (translation) {
const language = preloadedLanguages.find(lang => lang[0] === translation.get('detected_source_language'));
const languageName = language ? language[2] : translation.get('detected_source_language');
const provider = translation.get('provider');
return (
<div className='translate-button'>
<div className='translate-button__meta'>
<FormattedMessage id='status.translated_from_with' defaultMessage='Translated from {lang} using {provider}' values={{ lang: languageName, provider }} />
</div>
<button className='link-button' onClick={onClick}>
<FormattedMessage id='status.show_original' defaultMessage='Show original' />
</button>
</div>
);
}
return (
<button className='status__content__translate-button' onClick={onClick}>
<FormattedMessage id='status.translate' defaultMessage='Translate' />
</button>
);
}
}
const mapStateToProps = state => ({
languages: state.getIn(['server', 'translationLanguages', 'items']),
});
class StatusContent extends PureComponent {
static propTypes = {
identity: identityContextPropShape,
status: ImmutablePropTypes.map.isRequired,
statusContent: PropTypes.string,
onTranslate: PropTypes.func,
onClick: PropTypes.func,
collapsible: PropTypes.bool,
onCollapsedToggle: PropTypes.func,
languages: ImmutablePropTypes.map,
intl: PropTypes.object,
// from react-router
match: PropTypes.object.isRequired,
location: PropTypes.object.isRequired,
history: PropTypes.object.isRequired
};
_updateStatusLinks () {
const node = this.node;
if (!node) {
return;
}
const { status, onCollapsedToggle } = this.props;
const links = node.querySelectorAll('a');
let link, mention;
for (var i = 0; i < links.length; ++i) {
link = links[i];
if (link.classList.contains('status-link')) {
continue;
}
link.classList.add('status-link');
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')}`);
link.setAttribute('href', `/@${mention.get('acct')}`);
link.setAttribute('data-hover-card-account', mention.get('id'));
} 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);
link.setAttribute('href', `/tags/${link.text.replace(/^#/, '')}`);
} else {
link.setAttribute('title', link.href);
link.classList.add('unhandled-link');
}
}
if (status.get('collapsed', null) === null && onCollapsedToggle) {
const { collapsible, onClick } = this.props;
const collapsed =
collapsible
&& onClick
&& node.clientHeight > MAX_HEIGHT
&& status.get('spoiler_text').length === 0;
onCollapsedToggle(collapsed);
}
}
handleMouseEnter = ({ currentTarget }) => {
if (autoPlayGif) {
return;
}
const emojis = currentTarget.querySelectorAll('.custom-emoji');
for (var i = 0; i < emojis.length; i++) {
let emoji = emojis[i];
emoji.src = emoji.getAttribute('data-original');
}
};
handleMouseLeave = ({ currentTarget }) => {
if (autoPlayGif) {
return;
}
const emojis = currentTarget.querySelectorAll('.custom-emoji');
for (var i = 0; i < emojis.length; i++) {
let emoji = emojis[i];
emoji.src = emoji.getAttribute('data-static');
}
};
componentDidMount () {
this._updateStatusLinks();
}
componentDidUpdate () {
this._updateStatusLinks();
}
onMentionClick = (mention, e) => {
if (this.props.history && e.button === 0 && !(e.ctrlKey || e.metaKey)) {
e.preventDefault();
this.props.history.push(`/@${mention.get('acct')}`);
}
};
onHashtagClick = (hashtag, e) => {
hashtag = hashtag.replace(/^#/, '');
if (this.props.history && e.button === 0 && !(e.ctrlKey || e.metaKey)) {
e.preventDefault();
this.props.history.push(`/tags/${hashtag}`);
}
};
handleMouseDown = (e) => {
this.startXY = [e.clientX, e.clientY];
};
handleMouseUp = (e) => {
if (!this.startXY) {
return;
}
const [ startX, startY ] = this.startXY;
const [ deltaX, deltaY ] = [Math.abs(e.clientX - startX), Math.abs(e.clientY - startY)];
let element = e.target;
while (element) {
if (element.localName === 'button' || element.localName === 'a' || element.localName === 'label') {
return;
}
element = element.parentNode;
}
if (deltaX + deltaY < 5 && (e.button === 0 || e.button === 1) && e.detail >= 1 && this.props.onClick) {
this.props.onClick(e);
}
this.startXY = null;
};
handleTranslate = () => {
this.props.onTranslate();
};
setRef = (c) => {
this.node = c;
};
render () {
const { status, intl, statusContent } = this.props;
const renderReadMore = this.props.onClick && status.get('collapsed');
const contentLocale = intl.locale.replace(/[_-].*/, '');
const targetLanguages = this.props.languages?.get(status.get('language') || 'und');
const renderTranslate = this.props.onTranslate && this.props.identity.signedIn && ['public', 'unlisted'].includes(status.get('visibility')) && status.get('search_index').trim().length > 0 && targetLanguages?.includes(contentLocale);
const content = { __html: statusContent ?? getStatusContent(status) };
const language = status.getIn(['translation', 'language']) || status.get('language');
const classNames = classnames('status__content', {
'status__content--with-action': this.props.onClick && this.props.history,
'status__content--collapsed': renderReadMore,
});
const readMoreButton = renderReadMore && (
<button className='status__content__read-more-button' onClick={this.props.onClick} key='read-more'>
<FormattedMessage id='status.read_more' defaultMessage='Read more' /><Icon id='angle-right' icon={ChevronRightIcon} />
</button>
);
const translateButton = renderTranslate && (
<TranslateButton onClick={this.handleTranslate} translation={status.get('translation')} />
);
const poll = !!status.get('poll') && (
<PollContainer pollId={status.get('poll')} status={status} lang={language} />
);
if (this.props.onClick) {
return (
<>
<div className={classNames} ref={this.setRef} tabIndex={0} onMouseDown={this.handleMouseDown} onMouseUp={this.handleMouseUp} key='status-content' onMouseEnter={this.handleMouseEnter} onMouseLeave={this.handleMouseLeave}>
<div className='status__content__text status__content__text--visible translate' lang={language} dangerouslySetInnerHTML={content} />
{poll}
{translateButton}
</div>
{readMoreButton}
</>
);
} else {
return (
<div className={classNames} ref={this.setRef} tabIndex={0} onMouseEnter={this.handleMouseEnter} onMouseLeave={this.handleMouseLeave}>
<div className='status__content__text status__content__text--visible translate' lang={language} dangerouslySetInnerHTML={content} />
{poll}
{translateButton}
</div>
);
}
}
}
export default withRouter(withIdentity(connect(mapStateToProps)(injectIntl(StatusContent))));

View file

@ -0,0 +1,372 @@
import { useCallback, useRef, useLayoutEffect } from 'react';
import { FormattedMessage, useIntl } from 'react-intl';
import classnames from 'classnames';
import { useHistory } from 'react-router-dom';
import type { List as ImmutableList, Map as ImmutableMap } from 'immutable';
import type { History } from 'history';
import ChevronRightIcon from '@/material-icons/400-24px/chevron_right.svg?react';
import { Icon } from 'mastodon/components/icon';
import PollContainer from 'mastodon/containers/poll_container';
import { useIdentity } from 'mastodon/identity_context';
import {
autoPlayGif,
languages as preloadedLanguages,
} from 'mastodon/initial_state';
import type { Status, Translation } from 'mastodon/models/status';
import { useAppSelector } from 'mastodon/store';
const MAX_HEIGHT = 706; // 22px * 32 (+ 2px padding at the top)
export const getStatusContent = (status: Status): string =>
status.getIn(['translation', 'contentHtml']) || status.get('contentHtml');
const TranslateButton: React.FC<{
translation: ImmutableList<Translation>;
onClick: () => void;
}> = ({ translation, onClick }) => {
if (translation) {
const language = preloadedLanguages?.find(
(lang) => lang[0] === translation.get('detected_source_language'),
);
const languageName = language
? language[2]
: translation.get('detected_source_language');
const provider = translation.get('provider');
return (
<div className='translate-button'>
<div className='translate-button__meta'>
<FormattedMessage
id='status.translated_from_with'
defaultMessage='Translated from {lang} using {provider}'
values={{ lang: languageName, provider }}
/>
</div>
<button className='link-button' onClick={onClick}>
<FormattedMessage
id='status.show_original'
defaultMessage='Show original'
/>
</button>
</div>
);
}
return (
<button className='status__content__translate-button' onClick={onClick}>
<FormattedMessage id='status.translate' defaultMessage='Translate' />
</button>
);
};
const handleMentionClick = (
history: History,
mention: string,
e: MouseEvent,
) => {
if (history && e.button === 0 && !(e.ctrlKey || e.metaKey)) {
e.preventDefault();
history.push(`/@${mention}`);
}
};
const handleHashtagClick = (
history: History,
hashtag: string,
e: MouseEvent,
) => {
hashtag = hashtag.replace(/^#/, '');
if (history && e.button === 0 && !(e.ctrlKey || e.metaKey)) {
e.preventDefault();
history.push(`/tags/${hashtag}`);
}
};
type ClickCoordinates = [number, number];
export const StatusContent: React.FC<{
status: Status;
statusContent: string;
onTranslate?: () => void;
onClick?: (arg0?: React.MouseEvent | MouseEvent) => void;
onCollapsedToggle?: (arg0: boolean) => void;
collapsible?: boolean;
}> = ({
status,
statusContent,
onTranslate,
onClick,
collapsible,
onCollapsedToggle,
}) => {
const { signedIn } = useIdentity();
const history = useHistory();
const intl = useIntl();
const languages = useAppSelector(
(state) =>
state.server.getIn(['translationLanguages', 'items']) as ImmutableMap<
string,
ImmutableList<string>
>,
);
const clickCoordinates = useRef<ClickCoordinates | null>(null);
const nodeRef = useRef<HTMLDivElement | null>(null);
const handleMouseEnter = useCallback(
({ currentTarget }: React.MouseEvent) => {
if (autoPlayGif) {
return;
}
const emojis =
currentTarget.querySelectorAll<HTMLImageElement>('.custom-emoji');
for (const emoji of emojis) {
const originalUrl = emoji.getAttribute('data-original');
if (originalUrl) {
emoji.src = originalUrl;
}
}
},
[],
);
const handleMouseLeave = useCallback(
({ currentTarget }: React.MouseEvent) => {
if (autoPlayGif) {
return;
}
const emojis =
currentTarget.querySelectorAll<HTMLImageElement>('.custom-emoji');
for (const emoji of emojis) {
const staticUrl = emoji.getAttribute('data-static');
if (staticUrl) {
emoji.src = staticUrl;
}
}
},
[],
);
const handleMouseDown = useCallback((e: React.MouseEvent) => {
clickCoordinates.current = [e.clientX, e.clientY];
}, []);
const handleMouseUp = useCallback(
(e: React.MouseEvent) => {
if (!clickCoordinates.current) {
return;
}
const [startX, startY] = clickCoordinates.current;
const [deltaX, deltaY] = [
Math.abs(e.clientX - startX),
Math.abs(e.clientY - startY),
];
if (!(e.target instanceof HTMLElement)) {
return;
}
let element: HTMLElement | null = e.target;
while (element) {
if (
element.localName === 'button' ||
element.localName === 'a' ||
element.localName === 'label'
) {
return;
}
if (!(element.parentNode instanceof HTMLElement)) {
break;
}
element = element.parentNode;
}
if (
deltaX + deltaY < 5 &&
(e.button === 0 || e.button === 1) &&
e.detail >= 1 &&
onClick
) {
onClick(e);
}
clickCoordinates.current = null;
},
[onClick],
);
const handleTranslate = useCallback(() => {
onTranslate?.();
}, [onTranslate]);
const mentions = status.get('mentions') as ImmutableList<ImmutableMap<string, string>>;
const spoilerText = status.get('spoiler_text') as string;
const visibility = status.get('visibility') as string;
const searchIndex = status.get('search_index') as string;
const collapsed = status.get('collapsed') as boolean | undefined;
useLayoutEffect(() => {
const node = nodeRef.current;
if (!node) {
return;
}
const links = node.querySelectorAll<HTMLAnchorElement>('a');
for (const link of links) {
if (link.classList.contains('status-link')) {
continue;
}
link.classList.add('status-link');
const mention = mentions.find((item) => link.href === item.get('url'));
if (mention) {
const acct = mention.get('acct')!;
const id = mention.get('id')!;
link.addEventListener(
'click',
handleMentionClick.bind(null, history, acct),
false,
);
link.setAttribute('title', `@${acct}`);
link.setAttribute('href', `/@${acct}`);
link.setAttribute('data-hover-card-account', id);
} else if (
link.textContent?.[0] === '#' ||
(link.previousSibling?.textContent?.endsWith('#'))
) {
link.addEventListener(
'click',
handleHashtagClick.bind(null, history, link.text),
false,
);
link.setAttribute('href', `/tags/${link.text.replace(/^#/, '')}`);
} else {
link.setAttribute('title', link.href);
link.classList.add('unhandled-link');
}
}
if (collapsed && onCollapsedToggle) {
const collapsed =
!!collapsible &&
!!onClick &&
node.clientHeight > MAX_HEIGHT &&
spoilerText.length === 0;
onCollapsedToggle(collapsed);
}
}, [history, mentions, spoilerText, onCollapsedToggle, collapsible, onClick]);
const renderReadMore = onClick && status.get('collapsed');
const contentLocale = intl.locale.replace(/[_-].*/, '');
const originalLanguage = (status.get('language') as string) || 'und';
const targetLanguages = languages.get(originalLanguage);
const renderTranslate =
onTranslate &&
signedIn &&
['public', 'unlisted'].includes(visibility) &&
searchIndex.trim().length > 0 &&
targetLanguages?.includes(contentLocale);
const content = { __html: statusContent ?? getStatusContent(status) };
const language =
(status.getIn(['translation', 'language']) as string) ?? originalLanguage;
const classNames = classnames('status__content', {
'status__content--with-action': onClick && history,
'status__content--collapsed': renderReadMore,
});
const readMoreButton = renderReadMore && (
<button
className='status__content__read-more-button'
onClick={onClick}
key='read-more'
>
<FormattedMessage id='status.read_more' defaultMessage='Read more' />
<Icon id='angle-right' icon={ChevronRightIcon} />
</button>
);
const translateButton = renderTranslate && (
<TranslateButton
onClick={handleTranslate}
translation={status.get('translation')}
/>
);
const poll = !!status.get('poll') && (
<PollContainer
pollId={status.get('poll')}
status={status}
lang={language}
/>
);
if (onClick) {
return (
<>
<div
className={classNames}
ref={nodeRef}
onMouseDown={handleMouseDown}
onMouseUp={handleMouseUp}
key='status-content'
onMouseEnter={handleMouseEnter}
onMouseLeave={handleMouseLeave}
>
<div
className='status__content__text status__content__text--visible translate'
lang={language}
dangerouslySetInnerHTML={content}
/>
{poll}
{translateButton}
</div>
{readMoreButton}
</>
);
} else {
return (
<div
className={classNames}
ref={nodeRef}
onMouseEnter={handleMouseEnter}
onMouseLeave={handleMouseLeave}
>
<div
className='status__content__text status__content__text--visible translate'
lang={language}
dangerouslySetInnerHTML={content}
/>
{poll}
{translateButton}
</div>
);
}
};

View file

@ -23,7 +23,7 @@ import AttachmentList from 'mastodon/components/attachment_list';
import AvatarComposite from 'mastodon/components/avatar_composite';
import { IconButton } from 'mastodon/components/icon_button';
import { RelativeTimestamp } from 'mastodon/components/relative_timestamp';
import StatusContent from 'mastodon/components/status_content';
import { StatusContent } from 'mastodon/components/status_content';
import DropdownMenuContainer from 'mastodon/containers/dropdown_menu_container';
import { autoPlayGif } from 'mastodon/initial_state';
import { makeGetStatus } from 'mastodon/selectors';

View file

@ -7,7 +7,7 @@ import { Avatar } from 'mastodon/components/avatar';
import { DisplayName } from 'mastodon/components/display_name';
import MediaAttachments from 'mastodon/components/media_attachments';
import { RelativeTimestamp } from 'mastodon/components/relative_timestamp';
import StatusContent from 'mastodon/components/status_content';
import { StatusContent } from 'mastodon/components/status_content';
import { VisibilityIcon } from 'mastodon/components/visibility_icon';
import Option from './option';

View file

@ -25,7 +25,7 @@ import { VisibilityIcon } from 'mastodon/components/visibility_icon';
import { Avatar } from '../../../components/avatar';
import { DisplayName } from '../../../components/display_name';
import MediaGallery from '../../../components/media_gallery';
import StatusContent from '../../../components/status_content';
import { StatusContent } from '../../../components/status_content';
import Audio from '../../audio';
import scheduleIdleTask from '../../ui/util/schedule_idle_task';
import Video from '../../video';

View file

@ -12,3 +12,5 @@ type CardShape = Required<ApiPreviewCardJSON>;
export type Card = RecordOf<CardShape>;
export type MediaAttachment = Immutable.Map<string, unknown>;
export type Translation = Immutable.Map<string, unknown>;