chuckya/app/javascript/mastodon/features/status/index.jsx

720 lines
22 KiB
React
Raw Normal View History

import Immutable from 'immutable';
import React from 'react';
import { connect } from 'react-redux';
import PropTypes from 'prop-types';
import classNames from 'classnames';
import ImmutablePropTypes from 'react-immutable-proptypes';
import { createSelector } from 'reselect';
import {
fetchStatus,
muteStatus,
unmuteStatus,
deleteStatus,
editStatus,
hideStatus,
revealStatus,
translateStatus,
undoStatusTranslation,
} from '../../actions/statuses';
import MissingIndicator from '../../components/missing_indicator';
import LoadingIndicator from 'mastodon/components/loading_indicator';
import DetailedStatus from './components/detailed_status';
import ActionBar from './components/action_bar';
import Column from '../ui/components/column';
import {
favourite,
unfavourite,
bookmark,
unbookmark,
reblog,
unreblog,
pin,
unpin,
2022-12-01 01:25:36 +09:00
addReaction,
removeReaction,
} from '../../actions/interactions';
import {
replyCompose,
mentionCompose,
directCompose,
} from '../../actions/compose';
import {
unblockAccount,
unmuteAccount,
} from '../../actions/accounts';
import {
blockDomain,
unblockDomain,
} from '../../actions/domain_blocks';
import { initMuteModal } from '../../actions/mutes';
import { initBlockModal } from '../../actions/blocks';
import { initBoostModal } from '../../actions/boosts';
import { initReport } from '../../actions/reports';
import { makeCustomEmojiMap, makeGetStatus, makeGetPictureInPicture } from '../../selectors';
import ScrollContainer from 'mastodon/containers/scroll_container';
import ColumnBackButton from '../../components/column_back_button';
import ColumnHeader from '../../components/column_header';
import StatusContainer from '../../containers/status_container';
2017-04-02 05:11:28 +09:00
import { openModal } from '../../actions/modal';
import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
import ImmutablePureComponent from 'react-immutable-pure-component';
import { HotKeys } from 'react-hotkeys';
import { boostModal, deleteModal } from '../../initial_state';
import { attachFullscreenListener, detachFullscreenListener, isFullscreen } from '../ui/util/fullscreen';
import { textForScreenReader, defaultMediaVisibility } from '../../components/status';
import Icon from 'mastodon/components/icon';
import { Helmet } from 'react-helmet';
const messages = defineMessages({
deleteConfirm: { id: 'confirmations.delete.confirm', defaultMessage: 'Delete' },
deleteMessage: { id: 'confirmations.delete.message', defaultMessage: 'Are you sure you want to delete this status?' },
redraftConfirm: { id: 'confirmations.redraft.confirm', defaultMessage: 'Delete & redraft' },
Weblate translations (2018-08-25) (#8420) * Translated using Weblate (Georgian) Currently translated at 99.8% (674 of 675 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/ka/ * Translated using Weblate (Korean) Currently translated at 100.0% (310 of 310 strings) Translation: Mastodon/React Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/frontend/ko/ * Translated using Weblate (Korean) Currently translated at 96.8% (654 of 675 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/ko/ * Translated using Weblate (Japanese) Currently translated at 100.0% (310 of 310 strings) Translation: Mastodon/React Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/frontend/ja/ * Translated using Weblate (Danish) Currently translated at 100.0% (98 of 98 strings) Translation: Mastodon/Doorkeeper Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/doorkeeper/da/ * Translated using Weblate (Danish) Currently translated at 100.0% (310 of 310 strings) Translation: Mastodon/React Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/frontend/da/ * Translated using Weblate (Danish) Currently translated at 87.2% (589 of 675 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/da/ * Translated using Weblate (Galician) Currently translated at 100.0% (680 of 680 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/gl/ * Translated using Weblate (Czech) Currently translated at 100.0% (310 of 310 strings) Translation: Mastodon/React Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/frontend/cs/ * Translated using Weblate (Czech) Currently translated at 99.4% (676 of 680 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/cs/ * Translated using Weblate (Danish) Currently translated at 88.0% (599 of 680 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/da/ * Translated using Weblate (Arabic) Currently translated at 97.6% (664 of 680 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/ar/ * Translated using Weblate (Japanese) Currently translated at 99.7% (678 of 680 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/ja/ * Translated using Weblate (Occitan) Currently translated at 99.5% (677 of 680 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/oc/ * Translated using Weblate (Arabic) Currently translated at 97.9% (668 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/ar/ * Translated using Weblate (Persian) Currently translated at 99.7% (680 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/fa/ * Translated using Weblate (Dutch) Currently translated at 99.8% (681 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/nl/ * Translated using Weblate (Occitan) Currently translated at 99.8% (681 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/oc/ * Translated using Weblate (Japanese) Currently translated at 0.0% (0 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/ja/ * Translated using Weblate (Japanese) Currently translated at 0.1% (1 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/ja/ * Translated using Weblate (Japanese) Currently translated at 0.1% (676 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/ja/ * Translated using Weblate (Japanese) Currently translated at 99.8% (681 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/ja/ * Translated using Weblate (Galician) Currently translated at 100.0% (682 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/gl/ * Translated using Weblate (Greek) Currently translated at 99.8% (681 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/el/ * Translated using Weblate (Portuguese (Brazil)) Currently translated at 100.0% (310 of 310 strings) Translation: Mastodon/React Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/frontend/pt_BR/ * Translated using Weblate (Portuguese (Brazil)) Currently translated at 98.8% (674 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/pt_BR/ * Translated using Weblate (Danish) Currently translated at 89.2% (609 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/da/ * Translated using Weblate (French) Currently translated at 100.0% (82 of 82 strings) Translation: Mastodon/Preferences Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/simple_form/fr/ * Translated using Weblate (French) Currently translated at 100.0% (310 of 310 strings) Translation: Mastodon/React Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/frontend/fr/ * Translated using Weblate (French) Currently translated at 99.8% (681 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/fr/ * Translated using Weblate (Japanese) Currently translated at 99.8% (681 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/ja/ * Translated using Weblate (Japanese) Currently translated at 99.8% (681 of 682 strings) Translation: Mastodon/Backend Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/backend/ja/ * Translated using Weblate (Korean) Currently translated at 100.0% (310 of 310 strings) Translation: Mastodon/React Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/frontend/ko/ * Translated using Weblate (Occitan) Currently translated at 100.0% (310 of 310 strings) Translation: Mastodon/React Translate-URL: https://weblate.joinmastodon.org/projects/mastodon/frontend/oc/ * yarn manage:translations * i18n-tasks normalize && i18n-tasks remove-unused * revert * Add defaultMessage
2018-08-25 20:27:56 +09:00
redraftMessage: { id: 'confirmations.redraft.message', defaultMessage: 'Are you sure you want to delete this status and re-draft it? Favourites and boosts will be lost, and replies to the original post will be orphaned.' },
revealAll: { id: 'status.show_more_all', defaultMessage: 'Show more for all' },
hideAll: { id: 'status.show_less_all', defaultMessage: 'Show less for all' },
detailedStatus: { id: 'status.detailed_status', defaultMessage: 'Detailed conversation view' },
replyConfirm: { id: 'confirmations.reply.confirm', defaultMessage: 'Reply' },
replyMessage: { id: 'confirmations.reply.message', defaultMessage: 'Replying now will overwrite the message you are currently composing. Are you sure you want to proceed?' },
blockDomainConfirm: { id: 'confirmations.domain_block.confirm', defaultMessage: 'Hide entire domain' },
});
const makeMapStateToProps = () => {
const getStatus = makeGetStatus();
const getPictureInPicture = makeGetPictureInPicture();
const getAncestorsIds = createSelector([
(_, { id }) => id,
state => state.getIn(['contexts', 'inReplyTos']),
], (statusId, inReplyTos) => {
let ancestorsIds = Immutable.List();
ancestorsIds = ancestorsIds.withMutations(mutable => {
let id = statusId;
while (id && !mutable.includes(id)) {
mutable.unshift(id);
id = inReplyTos.get(id);
}
});
return ancestorsIds;
});
const getDescendantsIds = createSelector([
(_, { id }) => id,
state => state.getIn(['contexts', 'replies']),
state => state.get('statuses'),
], (statusId, contextReplies, statuses) => {
let descendantsIds = [];
const ids = [statusId];
while (ids.length > 0) {
let id = ids.pop();
const replies = contextReplies.get(id);
if (statusId !== id) {
descendantsIds.push(id);
}
if (replies) {
replies.reverse().forEach(reply => {
if (!ids.includes(reply) && !descendantsIds.includes(reply) && statusId !== reply) ids.push(reply);
});
}
}
let insertAt = descendantsIds.findIndex((id) => statuses.get(id).get('in_reply_to_account_id') !== statuses.get(id).get('account'));
if (insertAt !== -1) {
descendantsIds.forEach((id, idx) => {
if (idx > insertAt && statuses.get(id).get('in_reply_to_account_id') === statuses.get(id).get('account')) {
descendantsIds.splice(idx, 1);
descendantsIds.splice(insertAt, 0, id);
insertAt += 1;
}
});
}
return Immutable.List(descendantsIds);
});
const mapStateToProps = (state, props) => {
const status = getStatus(state, { id: props.params.statusId });
let ancestorsIds = Immutable.List();
let descendantsIds = Immutable.List();
if (status) {
ancestorsIds = getAncestorsIds(state, { id: status.get('in_reply_to_id') });
descendantsIds = getDescendantsIds(state, { id: status.get('id') });
}
return {
isLoading: state.getIn(['statuses', props.params.statusId, 'isLoading']),
status,
ancestorsIds,
descendantsIds,
askReplyConfirmation: state.getIn(['compose', 'text']).trim().length !== 0,
domain: state.getIn(['meta', 'domain']),
pictureInPicture: getPictureInPicture(state, { id: props.params.statusId }),
2022-12-01 01:25:36 +09:00
emojiMap: makeCustomEmojiMap(state),
};
};
return mapStateToProps;
};
const truncate = (str, num) => {
if (str.length > num) {
return str.slice(0, num) + '…';
} else {
return str;
}
};
const titleFromStatus = status => {
const displayName = status.getIn(['account', 'display_name']);
const username = status.getIn(['account', 'username']);
const prefix = displayName.trim().length === 0 ? username : displayName;
const text = status.get('search_index');
return `${prefix}: "${truncate(text, 30)}"`;
};
2018-09-15 00:59:48 +09:00
class Status extends ImmutablePureComponent {
static contextTypes = {
router: PropTypes.object,
identity: PropTypes.object,
};
static propTypes = {
params: PropTypes.object.isRequired,
dispatch: PropTypes.func.isRequired,
status: ImmutablePropTypes.map,
isLoading: PropTypes.bool,
ancestorsIds: ImmutablePropTypes.list,
descendantsIds: ImmutablePropTypes.list,
intl: PropTypes.object.isRequired,
askReplyConfirmation: PropTypes.bool,
multiColumn: PropTypes.bool,
domain: PropTypes.string.isRequired,
pictureInPicture: ImmutablePropTypes.contains({
inUse: PropTypes.bool,
available: PropTypes.bool,
}),
};
state = {
fullscreen: false,
showMedia: defaultMediaVisibility(this.props.status),
loadedStatusId: undefined,
};
componentWillMount () {
Change IDs to strings rather than numbers in API JSON output (#5019) * Fix JavaScript interface with long IDs Somewhat predictably, the JS interface handled IDs as numbers, which in JS are IEEE double-precision floats. This loses some precision when working with numbers as large as those generated by the new ID scheme, so we instead handle them here as strings. This is relatively simple, and doesn't appear to have caused any problems, but should definitely be tested more thoroughly than the built-in tests. Several days of use appear to support this working properly. BREAKING CHANGE: The major(!) change here is that IDs are now returned as strings by the REST endpoints, rather than as integers. In practice, relatively few changes were required to make the existing JS UI work with this change, but it will likely hit API clients pretty hard: it's an entirely different type to consume. (The one API client I tested, Tusky, handles this with no problems, however.) Twitter ran into this issue when introducing Snowflake IDs, and decided to instead introduce an `id_str` field in JSON responses. I have opted to *not* do that, and instead force all IDs to 64-bit integers represented by strings in one go. (I believe Twitter exacerbated their problem by rolling out the changes three times: once for statuses, once for DMs, and once for user IDs, as well as by leaving an integer ID value in JSON. As they said, "If you’re using the `id` field with JSON in a Javascript-related language, there is a very high likelihood that the integers will be silently munged by Javascript interpreters. In most cases, this will result in behavior such as being unable to load or delete a specific direct message, because the ID you're sending to the API is different than the actual identifier associated with the message." [1]) However, given that this is a significant change for API users, alternatives or a transition time may be appropriate. 1: https://blog.twitter.com/developer/en_us/a/2011/direct-messages-going-snowflake-on-sep-30-2011.html * Additional fixes for stringified IDs in JSON These should be the last two. These were identified using eslint to try to identify any plain casts to JavaScript numbers. (Some such casts are legitimate, but these were not.) Adding the following to .eslintrc.yml will identify casts to numbers: ~~~ no-restricted-syntax: - warn - selector: UnaryExpression[operator='+'] > :not(Literal) message: Avoid the use of unary + - selector: CallExpression[callee.name='Number'] message: Casting with Number() may coerce string IDs to numbers ~~~ The remaining three casts appear legitimate: two casts to array indices, one in a server to turn an environment variable into a number. * Back out RelationshipsController Change This was made to make a test a bit less flakey, but has nothing to do with this branch. * Change internal streaming payloads to stringified IDs as well Per https://github.com/tootsuite/mastodon/pull/5019#issuecomment-330736452 we need these changes to send deleted status IDs as strings, not integers.
2017-09-20 21:53:48 +09:00
this.props.dispatch(fetchStatus(this.props.params.statusId));
}
componentDidMount () {
attachFullscreenListener(this.onFullScreenChange);
}
componentWillReceiveProps (nextProps) {
if (nextProps.params.statusId !== this.props.params.statusId && nextProps.params.statusId) {
this._scrolledIntoView = false;
Change IDs to strings rather than numbers in API JSON output (#5019) * Fix JavaScript interface with long IDs Somewhat predictably, the JS interface handled IDs as numbers, which in JS are IEEE double-precision floats. This loses some precision when working with numbers as large as those generated by the new ID scheme, so we instead handle them here as strings. This is relatively simple, and doesn't appear to have caused any problems, but should definitely be tested more thoroughly than the built-in tests. Several days of use appear to support this working properly. BREAKING CHANGE: The major(!) change here is that IDs are now returned as strings by the REST endpoints, rather than as integers. In practice, relatively few changes were required to make the existing JS UI work with this change, but it will likely hit API clients pretty hard: it's an entirely different type to consume. (The one API client I tested, Tusky, handles this with no problems, however.) Twitter ran into this issue when introducing Snowflake IDs, and decided to instead introduce an `id_str` field in JSON responses. I have opted to *not* do that, and instead force all IDs to 64-bit integers represented by strings in one go. (I believe Twitter exacerbated their problem by rolling out the changes three times: once for statuses, once for DMs, and once for user IDs, as well as by leaving an integer ID value in JSON. As they said, "If you’re using the `id` field with JSON in a Javascript-related language, there is a very high likelihood that the integers will be silently munged by Javascript interpreters. In most cases, this will result in behavior such as being unable to load or delete a specific direct message, because the ID you're sending to the API is different than the actual identifier associated with the message." [1]) However, given that this is a significant change for API users, alternatives or a transition time may be appropriate. 1: https://blog.twitter.com/developer/en_us/a/2011/direct-messages-going-snowflake-on-sep-30-2011.html * Additional fixes for stringified IDs in JSON These should be the last two. These were identified using eslint to try to identify any plain casts to JavaScript numbers. (Some such casts are legitimate, but these were not.) Adding the following to .eslintrc.yml will identify casts to numbers: ~~~ no-restricted-syntax: - warn - selector: UnaryExpression[operator='+'] > :not(Literal) message: Avoid the use of unary + - selector: CallExpression[callee.name='Number'] message: Casting with Number() may coerce string IDs to numbers ~~~ The remaining three casts appear legitimate: two casts to array indices, one in a server to turn an environment variable into a number. * Back out RelationshipsController Change This was made to make a test a bit less flakey, but has nothing to do with this branch. * Change internal streaming payloads to stringified IDs as well Per https://github.com/tootsuite/mastodon/pull/5019#issuecomment-330736452 we need these changes to send deleted status IDs as strings, not integers.
2017-09-20 21:53:48 +09:00
this.props.dispatch(fetchStatus(nextProps.params.statusId));
}
if (nextProps.params.statusId && nextProps.ancestorsIds.size > this.props.ancestorsIds.size) {
this._scrolledIntoView = false;
}
if (nextProps.status && nextProps.status.get('id') !== this.state.loadedStatusId) {
this.setState({ showMedia: defaultMediaVisibility(nextProps.status), loadedStatusId: nextProps.status.get('id') });
}
}
handleToggleMediaVisibility = () => {
this.setState({ showMedia: !this.state.showMedia });
2023-01-30 09:45:35 +09:00
};
handleFavouriteClick = (status) => {
const { dispatch } = this.props;
const { signedIn } = this.context.identity;
if (signedIn) {
if (status.get('favourited')) {
dispatch(unfavourite(status));
} else {
dispatch(favourite(status));
}
} else {
dispatch(openModal('INTERACTION', {
type: 'favourite',
accountId: status.getIn(['account', 'id']),
url: status.get('url'),
}));
}
2023-01-30 09:45:35 +09:00
};
handleReactionAdd = (statusId, name, url) => {
2022-12-01 01:25:36 +09:00
const { dispatch } = this.props;
const { signedIn } = this.context.identity;
if (signedIn) {
dispatch(addReaction(statusId, name, url));
2022-12-01 01:25:36 +09:00
}
}
handleReactionRemove = (statusId, name) => {
this.props.dispatch(removeReaction(statusId, name));
}
Squashed commit of the following: commit 13bca8d5b7344445633ff3c8a162086be8215379 Merge: 1e2782647 ff157b237 Author: Essem <smswessem@gmail.com> Date: Thu Mar 16 11:51:32 2023 -0500 Merge branch 'feat/emoji_reactions' of https://github.com/neatchee/mastodon into feat/emoji_reactions commit ff157b2378a15a786763d7e13ad2de4ff223dc03 Author: neatchee <neatchee@gmail.com> Date: Wed Mar 8 13:27:25 2023 -0800 Remove old .js locale files accidentally restored during rebase commit 52beb88a19626c0b17d0945a43bf8725ceae4b04 Author: Ivan Rodriguez <104603218+IRod22@users.noreply.github.com> Date: Tue Mar 7 23:21:32 2023 -0600 Keep emoji picker within screen bounds Adds the `flip` prop to `<Overlay>`. Fixes #40 commit 7707004cf3d5bacd3ef830ef734dd003b607c2af Author: neatchee <neatchee@gmail.com> Date: Thu Jan 26 11:32:03 2023 -0800 Fix rebase issues commit ce40bbbce1e5e9ab7770f89dd7cffd87d3ceb177 Author: neatchee <neatchee@gmail.com> Date: Thu Jan 26 10:22:15 2023 -0800 Per PR suggestion, split name and domain, and look for emoji ID, for unreact, so remote emoji's can be unreacted commit fc9f34a39fa40b3eda4ccfc0cc8e5c028b9ebe79 Author: fef <owo@fef.moe> Date: Tue Dec 20 17:19:56 2022 +0000 move emoji reaction strings to locales-glitch commit af0f50ca74631e9f2dafde0f0e08417f906c04a0 Author: Jeremy Kescher <jeremy@kescher.at> Date: Sun Dec 18 04:23:42 2022 +0100 Fix status reactions preventing an on_cascade delete commit 2e713c7792438f17ef026f6d79465be25a5fb7db Author: fef <owo@fef.moe> Date: Thu Dec 15 15:27:54 2022 +0000 bypass reaction limit for foreign accounts commit 907e1e490ed9b2373eb18f92af9fda4426274713 Author: fef <owo@fef.moe> Date: Sun Dec 11 13:26:23 2022 +0000 fix 404 when reacting with Keycap Number Sign The Unicode sequence for this emoji starts with an ASCII # character, which the browser's URI parser truncates before sending the request to the backend. commit 9028d3841f7d5d2eb1b06bc210ded055c53fc1d2 Author: fef <owo@fef.moe> Date: Thu Dec 8 09:48:55 2022 +0000 fix status action bar after upstream changes commit 4ce5095d8c187947805ecf21c90ad494440e904a Author: fef <owo@fef.moe> Date: Wed Dec 7 21:52:53 2022 +0100 fix schema after rebase commit 63b9e4392adbb13fe0ce71de4d28419df0f36bb1 Author: fef <owo@fef.moe> Date: Wed Dec 7 12:47:03 2022 +0000 delete reaction notifications when deleting status commit e407cc12b64a664c66717ce06d65342dd8bc609e Author: fef <owo@fef.moe> Date: Wed Dec 7 12:19:36 2022 +0000 support reacting with foreign custom emojis commit b2dbb4dbe4996426a6cdb90e8da27e5096b3e88c Author: fef <owo@fef.moe> Date: Sun Dec 4 12:33:47 2022 +0000 properly disable reactions when not logged in commit ec790f3b1e243e33ccd22685b8e3c017ca7f9273 Author: fef <owo@fef.moe> Date: Sun Dec 4 10:52:02 2022 +0000 serialize custom emoji reactions properly for AP Akkoma and possibly others expect the `tag` field in an EmojiReact activity to be an array, not just a single object, so it's being wrapped into one now. I'm not entirely sure whether this is the idiomatic way of doing it tbh, but it works fine. commit 2637d9b77a6e696face0851328c5bd2e80d1dab2 Author: fef <owo@fef.moe> Date: Sun Dec 4 08:47:24 2022 +0000 also disable reaction buttons in vanilla flavour commit 0b4d5f700b74c0a6c8344321e491082511128f48 Author: fef <owo@fef.moe> Date: Sat Dec 3 16:55:37 2022 +0000 disable reaction button when not signed in commit afd0bb2c05be38477df1a808b32cece5c7e4b416 Author: fef <owo@fef.moe> Date: Sat Dec 3 16:20:29 2022 +0000 fix image for new custom emoji reactions commit b1dcd0eb6ce2f097fab310cdf32cd509426fcfba Author: fef <owo@fef.moe> Date: Sat Dec 3 14:23:55 2022 +0000 run i18n-tasks normalize commit e4e7837bf30082185b2f14e8d60741261f7085f4 Author: fef <owo@fef.moe> Date: Sat Dec 3 11:57:00 2022 +0000 display external custom emoji reactions properly Using an emoji map was completely unnecessary in the first place, because the reaction list from the API response includes URLs for every custom emoji anyway. The reaction list now also contains a boolean field indicating whether it is an external custom emoji, which is required because people should only be able to react with Unicode emojis and local custom ones, not with custom emojis from other servers. commit 6d3e364fa22370ab80f16a311dce15d93976464a Author: fef <owo@fef.moe> Date: Sat Dec 3 10:22:15 2022 +0000 handle incoming custom emoji reactions properly commit 0a47a05905adc3e9d1dc1e7e619891b884b9e512 Author: fef <owo@fef.moe> Date: Sat Dec 3 08:24:23 2022 +0000 support Undo action for EmojiReaction commit b81f4537aced99e427cb30fd54a33c852839e08a Author: fef <owo@fef.moe> Date: Fri Dec 2 17:02:06 2022 +0000 download remote custom emojis from reactions Emoji reactions containing custom emojis from remote instances were assumed to already have been downloaded and stored in the database. This might obviously not be the case. commit ff246bda3731753f58f7e499aae036c331953d0b Author: fef <owo@fef.moe> Date: Fri Dec 2 10:17:59 2022 +0000 fix integer cast bug Gotta love Rails. commit 40a645aab9e8071698051751d88262bb9e10199a Author: fef <owo@fef.moe> Date: Fri Dec 2 09:37:56 2022 +0000 sanitize setting for number of visible reactions This is kind of a hack, but the lack of validation for settings unfortunately makes it necessary. commit 391c6e22f2170f990d30d975c4a5ffb966484c17 Author: Jeremy Kescher <jeremy@kescher.at> Date: Fri Dec 2 08:05:10 2022 +0100 Add reaction limit to instance serializer commit 501ae9e2e1578ef80709ff38910dc46f483c5942 Author: fef <owo@fef.moe> Date: Fri Dec 2 01:52:59 2022 +0000 fix padding on posts without reactions The margins of the elements above and below the main reaction list element overlapped before reactions were added. Adding display: none to empty reaction bars restores this exact look. commit 956edd3ca7cd04a3524addc2c5b6ddc475358734 Author: fef <owo@fef.moe> Date: Fri Dec 2 01:00:08 2022 +0000 rename nop handler to handleNoOp This also adds the comment in action_bar.js to status_action_bar.js, clarifying that a future version could improve this code by modifying EmojiPickerDropdown. commit 2c93f1840f6cc74eef628b370b2c7b8485028cc8 Author: fef <owo@fef.moe> Date: Thu Dec 1 23:30:39 2022 +0100 cleanup JS imports and other minor stuff commit 4a2f91e3de2dfdb53e1a3f09430c6e0eabbf49be Author: fef <owo@fef.moe> Date: Thu Dec 1 04:26:13 2022 +0000 remove unnecessary parameter commit 91d26c871786708f8c2c0188f82318ff2c438127 Author: fef <owo@fef.moe> Date: Thu Dec 1 02:24:08 2022 +0000 change reaction api to match other interactions Status reactions had an API similar to that of announcement reactions, using PUT and DELETE at a single endpoint. I believe that for statuses, it makes more sense to follow the convention of the other interactions and use separate POST endpoints for create and destroy respectively. commit c74e050e3d77920000d7226ae26966d27291e790 Author: fef <owo@fef.moe> Date: Thu Dec 1 01:41:47 2022 +0000 fix reaction deletion bug and clean up controller Turns out the strange error where it would delete the wrong reaction occurred because I forgot to pass the emoji name to the query, which resulted in the database deleting the first reaction it found. Also, this removes the unused set_reaction callback and includes the Authorization module for the status reactions controller. commit e7ed7e37d7995d4bf39ad44c7a738290c81fa9dc Author: fef <owo@fef.moe> Date: Wed Nov 30 19:29:56 2022 +0000 remove outdated comments commit d33f8330c019898f948334453b75fc2802fab9f8 Author: fef <owo@fef.moe> Date: Wed Nov 30 17:09:16 2022 +0000 clean up new imports in vanilla flavour commit 45f803e23f019dbe5246daa3131548acdb99a757 Author: fef <owo@fef.moe> Date: Wed Nov 30 17:25:36 2022 +0100 rebase with upstream commit 8b339e4a2d2632a269dccdba1a60e69b6b38a053 Author: fef <owo@fef.moe> Date: Wed Nov 30 14:59:37 2022 +0000 make number of visible reactions a vanilla setting Reactions will be backported to the vanilla flavour, which requires all related settings to be accessible from the vanilla settings page rather than the glitch specific settings modal. commit 56323209b519a90f9dbf10b1c45a320de4d6aaa5 Author: fef <owo@fef.moe> Date: Wed Nov 30 13:20:20 2022 +0000 make number of displayed reactions a setting This adds an extra item to the local settings for specifying the number of reactions shown in toots. The detailed status view always shows all reactions. commit e96fb97770877749b5c6bca8fe3e6136fffa309a Author: fef <owo@fef.moe> Date: Wed Nov 30 12:01:34 2022 +0000 change default reaction limit to 1 commit 30ae2ad2612bae2abb48c49205bbe71b97435dda Author: fef <owo@fef.moe> Date: Wed Nov 30 09:06:14 2022 +0000 limit number of reactions displayed Too many reactions on a single post quickly get spammy, so they are now sorted by count and only the first MAX_REACTIONS number of different emojis are actually displayed. commit 84fb0f3cc252cabfa374b31f8a34e021a3ad8677 Author: fef <owo@fef.moe> Date: Tue Nov 29 09:07:10 2022 +0000 fix reaction margins and paddings commit 53b685ff1d4589915655da5cafa3a2dd9ee06a51 Author: fef <owo@fef.moe> Date: Tue Nov 29 08:54:35 2022 +0000 cleanup frontend emoji reaction code commit 2b0a474a73a84a3e841ddf44854fa5c2e0681a0f Author: fef <owo@fef.moe> Date: Tue Nov 29 08:15:52 2022 +0000 cleanup backend emoji reaction code commit 3bfd5ceba17c42bee27ff0f10514332caa814332 Author: fef <owo@fef.moe> Date: Tue Nov 29 06:25:43 2022 +0000 fix padding for reaction button commit deabc28e182acc208bffe6fb57430b97c5bc7c6c Author: fef <owo@fef.moe> Date: Tue Nov 29 05:21:53 2022 +0000 handle misskey reactions properly misskey federates emoji reactions as likes. commit 4d47b9929852493b208fdf13dee38e27dde50b8c Author: fef <owo@fef.moe> Date: Tue Nov 29 04:37:44 2022 +0000 move react button to action bar commit 151bcea7d4497086823adc0bca6c674ea78a8d8f Author: fef <owo@fef.moe> Date: Tue Nov 29 04:31:22 2022 +0100 cherry-pick emoji reaction changes commit f150fd0dc8761c2a992875948a2736244e5f7076 Author: fef <owo@fef.moe> Date: Tue Nov 29 00:39:40 2022 +0000 make frontend fetch reaction limit the maximum number of reactions was previously hardcoded to 8. this commit also fixes an incorrect query in StatusReactionValidator where it didn't count per-user reactions but the total amount of different ones. commit 12886aa19ae5be6c7077ff1704b1b4c75c15eae6 Author: fef <owo@fef.moe> Date: Mon Nov 28 23:16:56 2022 +0000 make status reaction count limit configurable commit 25806c568a57e9193b43427df80b60a88143a5eb Author: fef <owo@fef.moe> Date: Mon Nov 28 22:25:12 2022 +0000 remove accidentally created file commit f3784cfef551ad1d9d0df0001b2b480bfc656b14 Author: fef <owo@fef.moe> Date: Mon Nov 28 22:23:13 2022 +0000 federate emoji reactions this is kind of experimental, but it should work in theory. at least i tested it with a remove akkoma instance and it didn't crash. commit d32f3f8c2fb9e8a29e676e54c98d2cac85841728 Author: fef <owo@fef.moe> Date: Fri Nov 25 23:02:40 2022 +0000 show reactions in detailed status view commit 5e9bbb0be253b5b56c6d7b5ee8d843f401995b7c Author: fef <owo@fef.moe> Date: Thu Nov 24 17:30:52 2022 +0000 add frontend for emoji reactions this is still pretty bare bones but hey, it works. commit 85756b572a169857a09d7668ce9f2c856ebefd27 Author: fef <owo@fef.moe> Date: Thu Nov 24 11:50:32 2022 +0000 add backend support for status emoji reactions turns out we can just reuse the code for announcement reactions.
2023-03-17 01:54:49 +09:00
handleReactionAdd = (statusId, name, url) => {
const { dispatch } = this.props;
const { signedIn } = this.context.identity;
if (signedIn) {
dispatch(addReaction(statusId, name, url));
}
}
handleReactionRemove = (statusId, name) => {
this.props.dispatch(removeReaction(statusId, name));
}
handlePin = (status) => {
if (status.get('pinned')) {
this.props.dispatch(unpin(status));
} else {
this.props.dispatch(pin(status));
}
2023-01-30 09:45:35 +09:00
};
handleReplyClick = (status) => {
const { askReplyConfirmation, dispatch, intl } = this.props;
const { signedIn } = this.context.identity;
if (signedIn) {
if (askReplyConfirmation) {
dispatch(openModal('CONFIRM', {
message: intl.formatMessage(messages.replyMessage),
confirm: intl.formatMessage(messages.replyConfirm),
onConfirm: () => dispatch(replyCompose(status, this.context.router.history)),
}));
} else {
dispatch(replyCompose(status, this.context.router.history));
}
} else {
dispatch(openModal('INTERACTION', {
type: 'reply',
accountId: status.getIn(['account', 'id']),
url: status.get('url'),
}));
}
2023-01-30 09:45:35 +09:00
};
handleModalReblog = (status, privacy) => {
this.props.dispatch(reblog(status, privacy));
2023-01-30 09:45:35 +09:00
};
2017-04-11 11:28:52 +09:00
handleReblogClick = (status, e) => {
const { dispatch } = this.props;
const { signedIn } = this.context.identity;
if (signedIn) {
if (status.get('reblogged')) {
dispatch(unreblog(status));
} else {
if ((e && e.shiftKey) || !boostModal) {
this.handleModalReblog(status);
} else {
dispatch(initBoostModal({ status, onReblog: this.handleModalReblog }));
}
}
} else {
dispatch(openModal('INTERACTION', {
type: 'reblog',
accountId: status.getIn(['account', 'id']),
url: status.get('url'),
}));
}
2023-01-30 09:45:35 +09:00
};
handleBookmarkClick = (status) => {
if (status.get('bookmarked')) {
this.props.dispatch(unbookmark(status));
} else {
this.props.dispatch(bookmark(status));
}
2023-01-30 09:45:35 +09:00
};
handleDeleteClick = (status, history, withRedraft = false) => {
const { dispatch, intl } = this.props;
if (!deleteModal) {
dispatch(deleteStatus(status.get('id'), history, withRedraft));
} else {
dispatch(openModal('CONFIRM', {
message: intl.formatMessage(withRedraft ? messages.redraftMessage : messages.deleteMessage),
confirm: intl.formatMessage(withRedraft ? messages.redraftConfirm : messages.deleteConfirm),
onConfirm: () => dispatch(deleteStatus(status.get('id'), history, withRedraft)),
}));
}
2023-01-30 09:45:35 +09:00
};
2016-10-10 05:19:15 +09:00
handleEditClick = (status, history) => {
this.props.dispatch(editStatus(status.get('id'), history));
2023-01-30 09:45:35 +09:00
};
handleDirectClick = (account, router) => {
this.props.dispatch(directCompose(account, router));
2023-01-30 09:45:35 +09:00
};
handleMentionClick = (account, router) => {
this.props.dispatch(mentionCompose(account, router));
2023-01-30 09:45:35 +09:00
};
handleOpenMedia = (media, index) => {
this.props.dispatch(openModal('MEDIA', { statusId: this.props.status.get('id'), media, index }));
2023-01-30 09:45:35 +09:00
};
2016-10-25 01:07:40 +09:00
handleOpenVideo = (media, options) => {
this.props.dispatch(openModal('VIDEO', { statusId: this.props.status.get('id'), media, options }));
2023-01-30 09:45:35 +09:00
};
handleHotkeyOpenMedia = e => {
const { status } = this.props;
e.preventDefault();
if (status.get('media_attachments').size > 0) {
if (status.getIn(['media_attachments', 0, 'type']) === 'video') {
this.handleOpenVideo(status.getIn(['media_attachments', 0]), { startTime: 0 });
} else {
this.handleOpenMedia(status.get('media_attachments'), 0);
}
}
2023-01-30 09:45:35 +09:00
};
handleMuteClick = (account) => {
this.props.dispatch(initMuteModal(account));
2023-01-30 09:45:35 +09:00
};
handleConversationMuteClick = (status) => {
if (status.get('muted')) {
this.props.dispatch(unmuteStatus(status.get('id')));
} else {
this.props.dispatch(muteStatus(status.get('id')));
}
2023-01-30 09:45:35 +09:00
};
handleToggleHidden = (status) => {
if (status.get('hidden')) {
this.props.dispatch(revealStatus(status.get('id')));
} else {
this.props.dispatch(hideStatus(status.get('id')));
}
2023-01-30 09:45:35 +09:00
};
handleToggleAll = () => {
const { status, ancestorsIds, descendantsIds } = this.props;
const statusIds = [status.get('id')].concat(ancestorsIds.toJS(), descendantsIds.toJS());
if (status.get('hidden')) {
this.props.dispatch(revealStatus(statusIds));
} else {
this.props.dispatch(hideStatus(statusIds));
}
2023-01-30 09:45:35 +09:00
};
handleTranslate = status => {
const { dispatch } = this.props;
if (status.get('translation')) {
dispatch(undoStatusTranslation(status.get('id')));
} else {
dispatch(translateStatus(status.get('id')));
}
2023-01-30 09:45:35 +09:00
};
handleBlockClick = (status) => {
const { dispatch } = this.props;
const account = status.get('account');
dispatch(initBlockModal(account));
2023-01-30 09:45:35 +09:00
};
handleReport = (status) => {
this.props.dispatch(initReport(status.get('account'), status));
2023-01-30 09:45:35 +09:00
};
handleEmbed = (status) => {
this.props.dispatch(openModal('EMBED', { url: status.get('url') }));
2023-01-30 09:45:35 +09:00
};
handleUnmuteClick = account => {
this.props.dispatch(unmuteAccount(account.get('id')));
2023-01-30 09:45:35 +09:00
};
handleUnblockClick = account => {
this.props.dispatch(unblockAccount(account.get('id')));
2023-01-30 09:45:35 +09:00
};
handleBlockDomainClick = domain => {
this.props.dispatch(openModal('CONFIRM', {
message: <FormattedMessage id='confirmations.domain_block.message' defaultMessage='Are you really, really sure you want to block the entire {domain}? In most cases a few targeted blocks or mutes are sufficient and preferable. You will not see content from that domain in any public timelines or your notifications. Your followers from that domain will be removed.' values={{ domain: <strong>{domain}</strong> }} />,
confirm: this.props.intl.formatMessage(messages.blockDomainConfirm),
onConfirm: () => this.props.dispatch(blockDomain(domain)),
}));
2023-01-30 09:45:35 +09:00
};
handleUnblockDomainClick = domain => {
this.props.dispatch(unblockDomain(domain));
2023-01-30 09:45:35 +09:00
};
handleHotkeyMoveUp = () => {
this.handleMoveUp(this.props.status.get('id'));
2023-01-30 09:45:35 +09:00
};
handleHotkeyMoveDown = () => {
this.handleMoveDown(this.props.status.get('id'));
2023-01-30 09:45:35 +09:00
};
handleHotkeyReply = e => {
e.preventDefault();
this.handleReplyClick(this.props.status);
2023-01-30 09:45:35 +09:00
};
handleHotkeyFavourite = () => {
this.handleFavouriteClick(this.props.status);
2023-01-30 09:45:35 +09:00
};
handleHotkeyBoost = () => {
this.handleReblogClick(this.props.status);
2023-01-30 09:45:35 +09:00
};
handleHotkeyMention = e => {
e.preventDefault();
this.handleMentionClick(this.props.status.get('account'));
2023-01-30 09:45:35 +09:00
};
handleHotkeyOpenProfile = () => {
this.context.router.history.push(`/@${this.props.status.getIn(['account', 'acct'])}`);
2023-01-30 09:45:35 +09:00
};
handleHotkeyToggleHidden = () => {
this.handleToggleHidden(this.props.status);
2023-01-30 09:45:35 +09:00
};
handleHotkeyToggleSensitive = () => {
this.handleToggleMediaVisibility();
2023-01-30 09:45:35 +09:00
};
handleMoveUp = id => {
const { status, ancestorsIds, descendantsIds } = this.props;
if (id === status.get('id')) {
this._selectChild(ancestorsIds.size - 1, true);
} else {
let index = ancestorsIds.indexOf(id);
if (index === -1) {
index = descendantsIds.indexOf(id);
this._selectChild(ancestorsIds.size + index, true);
} else {
this._selectChild(index - 1, true);
}
}
2023-01-30 09:45:35 +09:00
};
handleMoveDown = id => {
const { status, ancestorsIds, descendantsIds } = this.props;
if (id === status.get('id')) {
this._selectChild(ancestorsIds.size + 1, false);
} else {
let index = ancestorsIds.indexOf(id);
if (index === -1) {
index = descendantsIds.indexOf(id);
this._selectChild(ancestorsIds.size + index + 2, false);
} else {
this._selectChild(index + 1, false);
}
}
2023-01-30 09:45:35 +09:00
};
_selectChild (index, align_top) {
const container = this.node;
const element = container.querySelectorAll('.focusable')[index];
if (element) {
if (align_top && container.scrollTop > element.offsetTop) {
element.scrollIntoView(true);
} else if (!align_top && container.scrollTop + container.clientHeight < element.offsetTop + element.offsetHeight) {
element.scrollIntoView(false);
}
element.focus();
}
}
renderChildren (list) {
return list.map(id => (
<StatusContainer
key={id}
id={id}
onMoveUp={this.handleMoveUp}
onMoveDown={this.handleMoveDown}
contextType='thread'
/>
));
}
setRef = c => {
this.node = c;
2023-01-30 09:45:35 +09:00
};
componentDidUpdate () {
if (this._scrolledIntoView) {
return;
}
const { status, ancestorsIds } = this.props;
if (status && ancestorsIds && ancestorsIds.size > 0) {
const element = this.node.querySelectorAll('.focusable')[ancestorsIds.size - 1];
window.requestAnimationFrame(() => {
element.scrollIntoView(true);
});
this._scrolledIntoView = true;
}
}
componentWillUnmount () {
detachFullscreenListener(this.onFullScreenChange);
}
onFullScreenChange = () => {
this.setState({ fullscreen: isFullscreen() });
2023-01-30 09:45:35 +09:00
};
render () {
let ancestors, descendants;
const { isLoading, status, ancestorsIds, descendantsIds, intl, domain, multiColumn, pictureInPicture } = this.props;
const { fullscreen } = this.state;
if (isLoading) {
return (
<Column>
<LoadingIndicator />
</Column>
);
}
if (status === null) {
2016-10-07 23:00:11 +09:00
return (
<Column>
<ColumnBackButton multiColumn={multiColumn} />
<MissingIndicator />
2016-10-07 23:00:11 +09:00
</Column>
);
}
if (ancestorsIds && ancestorsIds.size > 0) {
ancestors = <div>{this.renderChildren(ancestorsIds)}</div>;
}
if (descendantsIds && descendantsIds.size > 0) {
descendants = <div>{this.renderChildren(descendantsIds)}</div>;
}
const isLocal = status.getIn(['account', 'acct'], '').indexOf('@') === -1;
const isIndexable = !status.getIn(['account', 'noindex']);
const handlers = {
moveUp: this.handleHotkeyMoveUp,
moveDown: this.handleHotkeyMoveDown,
reply: this.handleHotkeyReply,
favourite: this.handleHotkeyFavourite,
boost: this.handleHotkeyBoost,
mention: this.handleHotkeyMention,
openProfile: this.handleHotkeyOpenProfile,
toggleHidden: this.handleHotkeyToggleHidden,
toggleSensitive: this.handleHotkeyToggleSensitive,
openMedia: this.handleHotkeyOpenMedia,
};
return (
<Column bindToDocument={!multiColumn} label={intl.formatMessage(messages.detailedStatus)}>
<ColumnHeader
showBackButton
multiColumn={multiColumn}
extraButton={(
<button type='button' className='column-header__button' title={intl.formatMessage(status.get('hidden') ? messages.revealAll : messages.hideAll)} aria-label={intl.formatMessage(status.get('hidden') ? messages.revealAll : messages.hideAll)} onClick={this.handleToggleAll}><Icon id={status.get('hidden') ? 'eye-slash' : 'eye'} /></button>
)}
/>
2016-09-18 20:03:37 +09:00
<ScrollContainer scrollKey='thread'>
<div className={classNames('scrollable', { fullscreen })} ref={this.setRef}>
{ancestors}
2016-09-18 20:03:37 +09:00
<HotKeys handlers={handlers}>
<div className={classNames('focusable', 'detailed-status__wrapper', `detailed-status__wrapper-${status.get('visibility')}`)} tabIndex={0} aria-label={textForScreenReader(intl, status, false)}>
<DetailedStatus
key={`details-${status.get('id')}`}
status={status}
onOpenVideo={this.handleOpenVideo}
onOpenMedia={this.handleOpenMedia}
2022-12-01 01:25:36 +09:00
onReactionAdd={this.handleReactionAdd}
onReactionRemove={this.handleReactionRemove}
onToggleHidden={this.handleToggleHidden}
onTranslate={this.handleTranslate}
domain={domain}
showMedia={this.state.showMedia}
onToggleMediaVisibility={this.handleToggleMediaVisibility}
pictureInPicture={pictureInPicture}
2022-12-01 01:25:36 +09:00
emojiMap={this.props.emojiMap}
/>
<ActionBar
key={`action-bar-${status.get('id')}`}
status={status}
onReply={this.handleReplyClick}
onFavourite={this.handleFavouriteClick}
2022-12-01 01:25:36 +09:00
onReactionAdd={this.handleReactionAdd}
onReblog={this.handleReblogClick}
onBookmark={this.handleBookmarkClick}
onDelete={this.handleDeleteClick}
onEdit={this.handleEditClick}
onDirect={this.handleDirectClick}
onMention={this.handleMentionClick}
onMute={this.handleMuteClick}
onUnmute={this.handleUnmuteClick}
onMuteConversation={this.handleConversationMuteClick}
onBlock={this.handleBlockClick}
onUnblock={this.handleUnblockClick}
onBlockDomain={this.handleBlockDomainClick}
onUnblockDomain={this.handleUnblockDomainClick}
onReport={this.handleReport}
onPin={this.handlePin}
onEmbed={this.handleEmbed}
/>
</div>
</HotKeys>
{descendants}
</div>
</ScrollContainer>
<Helmet>
<title>{titleFromStatus(status)}</title>
<meta name='robots' content={(isLocal && isIndexable) ? 'all' : 'noindex'} />
</Helmet>
2016-10-07 23:00:11 +09:00
</Column>
);
}
}
export default injectIntl(connect(makeMapStateToProps)(Status));