Normalize poll emoji

This commit is contained in:
Alex Gleason 2022-03-10 19:55:14 -06:00
parent d600a74410
commit 75db92719d
No known key found for this signature in database
GPG key ID: 7211D1F99744FBB7
10 changed files with 322 additions and 60 deletions

View file

@ -0,0 +1,236 @@
{
"account": {
"acct": "alex",
"avatar": "https://media.gleasonator.com/6d64aecb17348b23aaff78db4687b9476cb0da1c07cc6a819c2e6ec7144c18b1.png",
"avatar_static": "https://media.gleasonator.com/6d64aecb17348b23aaff78db4687b9476cb0da1c07cc6a819c2e6ec7144c18b1.png",
"bot": false,
"created_at": "2020-01-08T01:25:43.000Z",
"display_name": "Alex Gleason",
"emojis": [],
"fields": [
{
"name": "Website",
"value": "<a href=\"https://alexgleason.me\" rel=\"ugc\">https://alexgleason.me</a>"
},
{
"name": "Soapbox",
"value": "<a href=\"https://soapbox.pub\" rel=\"ugc\">https://soapbox.pub</a>"
},
{
"name": "Email",
"value": "alex@alexgleason.me"
},
{
"name": "Gender identity",
"value": "Soyboy"
},
{
"name": "Donate (PayPal)",
"value": "<a href=\"https://paypal.me/gleasonator\" rel=\"ugc\">https://paypal.me/gleasonator</a>"
},
{
"name": "$BTC",
"value": "bc1q9cx35adpm73aq2fw40ye6ts8hfxqzjr5unwg0n"
},
{
"name": "$ETH",
"value": "0xAc9aB5Fc04Dc1cB1789Af75b523Bd23C70B2D717"
},
{
"name": "$DOGE",
"value": "D5zVZs6jrRakaPVGiErkQiHt9sayzm6V5D"
},
{
"name": "$XMR",
"value": "45JDCLrjJ4bgVUSbbs2yjy9m5Mf4VLPW8fG7jw9sq5u69rXZZopQogZNeyYkMBnXpkaip4p4QwaaJNhdTotPa9g44DBCzdK"
}
],
"followers_count": 2467,
"following_count": 1581,
"fqn": "alex@gleasonator.com",
"header": "https://media.gleasonator.com/accounts/headers/000/000/001/original/9d0e4dbf1c9dbc8f.png",
"header_static": "https://media.gleasonator.com/accounts/headers/000/000/001/original/9d0e4dbf1c9dbc8f.png",
"id": "9v5bmRalQvjOy0ECcC",
"last_status_at": "2022-03-11T01:33:19",
"locked": false,
"note": "I create Fediverse software that empowers people online.<br/><br/>I&#39;m vegan btw<br/><br/>Note: If you have a question for me, please tag me publicly. This gives the opportunity for others to chime in, and bystanders to learn.",
"pleroma": {
"accepts_chat_messages": true,
"also_known_as": [
"https://mitra.social/users/alex"
],
"ap_id": "https://gleasonator.com/users/alex",
"background_image": null,
"birthday": "1993-07-03",
"favicon": "https://gleasonator.com/favicon.png",
"hide_favorites": true,
"hide_followers": false,
"hide_followers_count": false,
"hide_follows": false,
"hide_follows_count": false,
"is_admin": true,
"is_confirmed": true,
"is_moderator": false,
"is_suggested": true,
"relationship": {},
"skip_thread_containment": false,
"tags": []
},
"source": {
"fields": [
{
"name": "Website",
"value": "https://alexgleason.me"
},
{
"name": "Soapbox",
"value": "https://soapbox.pub"
},
{
"name": "Email",
"value": "alex@alexgleason.me"
},
{
"name": "Gender identity",
"value": "Soyboy"
},
{
"name": "Donate (PayPal)",
"value": "https://paypal.me/gleasonator"
},
{
"name": "$BTC",
"value": "bc1q9cx35adpm73aq2fw40ye6ts8hfxqzjr5unwg0n"
},
{
"name": "$ETH",
"value": "0xAc9aB5Fc04Dc1cB1789Af75b523Bd23C70B2D717"
},
{
"name": "$DOGE",
"value": "D5zVZs6jrRakaPVGiErkQiHt9sayzm6V5D"
},
{
"name": "$XMR",
"value": "45JDCLrjJ4bgVUSbbs2yjy9m5Mf4VLPW8fG7jw9sq5u69rXZZopQogZNeyYkMBnXpkaip4p4QwaaJNhdTotPa9g44DBCzdK"
}
],
"note": "I create Fediverse software that empowers people online.\r\n\r\nI'm vegan btw\r\n\r\nNote: If you have a question for me, please tag me publicly. This gives the opportunity for others to chime in, and bystanders to learn.",
"pleroma": {
"actor_type": "Person",
"discoverable": false
},
"sensitive": false
},
"statuses_count": 23651,
"url": "https://gleasonator.com/users/alex",
"username": "alex"
},
"application": {
"name": "Soapbox FE",
"website": "https://soapbox.pub/"
},
"bookmarked": false,
"card": null,
"content": "<p>Test poll</p>",
"created_at": "2022-03-11T01:33:18.000Z",
"emojis": [
{
"shortcode": "gleason_excited",
"static_url": "https://gleasonator.com/emoji/gleason_emojis/gleason_excited.png",
"url": "https://gleasonator.com/emoji/gleason_emojis/gleason_excited.png",
"visible_in_picker": false
},
{
"shortcode": "soapbox",
"static_url": "https://gleasonator.com/emoji/Gleasonator/soapbox.png",
"url": "https://gleasonator.com/emoji/Gleasonator/soapbox.png",
"visible_in_picker": false
}
],
"favourited": false,
"favourites_count": 1,
"id": "AHHue68kB59xtUv7MO",
"in_reply_to_account_id": null,
"in_reply_to_id": null,
"language": null,
"media_attachments": [],
"mentions": [],
"muted": false,
"pinned": false,
"pleroma": {
"content": {
"text/plain": "Test poll"
},
"conversation_id": "AHHue65YMwbjjbQZO4",
"direct_conversation_id": null,
"emoji_reactions": [],
"expires_at": null,
"in_reply_to_account_acct": null,
"local": true,
"parent_visible": false,
"pinned_at": null,
"quote": null,
"quote_url": null,
"quote_visible": false,
"spoiler_text": {
"text/plain": ""
},
"thread_muted": false
},
"poll": {
"emojis": [
{
"shortcode": "gleason_excited",
"static_url": "https://gleasonator.com/emoji/gleason_emojis/gleason_excited.png",
"url": "https://gleasonator.com/emoji/gleason_emojis/gleason_excited.png",
"visible_in_picker": false
},
{
"shortcode": "soapbox",
"static_url": "https://gleasonator.com/emoji/Gleasonator/soapbox.png",
"url": "https://gleasonator.com/emoji/Gleasonator/soapbox.png",
"visible_in_picker": false
}
],
"expired": false,
"expires_at": "2022-03-12T01:33:18.000Z",
"id": "AHHue67gF2JDqCQGhc",
"multiple": false,
"options": [
{
"title": "Regular emoji 😍 ",
"votes_count": 0
},
{
"title": "Custom emoji :gleason_excited: ",
"votes_count": 1
},
{
"title": "No emoji",
"votes_count": 0
},
{
"title": "🤔 😮 😠 ",
"votes_count": 1
},
{
"title": ":soapbox:",
"votes_count": 1
}
],
"voters_count": 3,
"votes_count": 3
},
"reblog": null,
"reblogged": false,
"reblogs_count": 1,
"replies_count": 1,
"sensitive": false,
"spoiler_text": "",
"tags": [],
"text": null,
"uri": "https://gleasonator.com/objects/46d2ab26-3497-442b-999f-612fe717b0a3",
"url": "https://gleasonator.com/notice/AHHue68kB59xtUv7MO",
"visibility": "public"
}

View file

@ -1,9 +1,6 @@
import { getSettings } from '../settings'; import { getSettings } from '../settings';
import { import { normalizeAccount } from './normalizer';
normalizeAccount,
normalizePoll,
} from './normalizer';
export const ACCOUNT_IMPORT = 'ACCOUNT_IMPORT'; export const ACCOUNT_IMPORT = 'ACCOUNT_IMPORT';
export const ACCOUNTS_IMPORT = 'ACCOUNTS_IMPORT'; export const ACCOUNTS_IMPORT = 'ACCOUNTS_IMPORT';
@ -130,7 +127,7 @@ export function importFetchedStatuses(statuses) {
} }
if (status.poll?.id) { if (status.poll?.id) {
polls.push(normalizePoll(status.poll)); polls.push(status.poll);
} }
} }
@ -144,7 +141,7 @@ export function importFetchedStatuses(statuses) {
export function importFetchedPoll(poll) { export function importFetchedPoll(poll) {
return dispatch => { return dispatch => {
dispatch(importPolls([normalizePoll(poll)])); dispatch(importPolls([poll]));
}; };
} }

View file

@ -41,20 +41,6 @@ export function normalizeAccount(account) {
return account; return account;
} }
export function normalizePoll(poll) {
const normalPoll = { ...poll };
const emojiMap = makeEmojiMap(normalPoll);
normalPoll.options = poll.options.map((option, index) => ({
...option,
voted: Boolean(poll.own_votes?.includes(index)),
title_emojified: emojify(escapeTextContentForBrowser(option.title), emojiMap),
}));
return normalPoll;
}
export function normalizeChat(chat, normalOldChat) { export function normalizeChat(chat, normalOldChat) {
const normalChat = { ...chat }; const normalChat = { ...chat };
const { account, last_message: lastMessage } = chat; const { account, last_message: lastMessage } = chat;

View file

@ -1,5 +1,4 @@
import classNames from 'classnames'; import classNames from 'classnames';
import escapeTextContentForBrowser from 'escape-html';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import React from 'react'; import React from 'react';
import ImmutablePropTypes from 'react-immutable-proptypes'; import ImmutablePropTypes from 'react-immutable-proptypes';
@ -10,7 +9,6 @@ import spring from 'react-motion/lib/spring';
import { openModal } from 'soapbox/actions/modals'; import { openModal } from 'soapbox/actions/modals';
import { vote, fetchPoll } from 'soapbox/actions/polls'; import { vote, fetchPoll } from 'soapbox/actions/polls';
import Icon from 'soapbox/components/icon'; import Icon from 'soapbox/components/icon';
import emojify from 'soapbox/features/emoji/emoji';
import Motion from 'soapbox/features/ui/util/optional_motion'; import Motion from 'soapbox/features/ui/util/optional_motion';
import SoapboxPropTypes from 'soapbox/utils/soapbox_prop_types'; import SoapboxPropTypes from 'soapbox/utils/soapbox_prop_types';
@ -22,11 +20,6 @@ const messages = defineMessages({
votes: { id: 'poll.votes', defaultMessage: '{votes, plural, one {# vote} other {# votes}}' }, votes: { id: 'poll.votes', defaultMessage: '{votes, plural, one {# vote} other {# votes}}' },
}); });
const makeEmojiMap = record => record.get('emojis').reduce((obj, emoji) => {
obj[`:${emoji.get('shortcode')}:`] = emoji.toJS();
return obj;
}, {});
export default @injectIntl export default @injectIntl
class Poll extends ImmutablePureComponent { class Poll extends ImmutablePureComponent {
@ -106,12 +99,7 @@ class Poll extends ImmutablePureComponent {
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'));
const active = !!this.state.selected[`${optionIndex}`]; const active = !!this.state.selected[`${optionIndex}`];
const voted = poll.own_votes?.includes(optionIndex); const voted = poll.own_votes?.includes(optionIndex);
const titleEmojified = option.get('title_emojified');
let titleEmojified = option.get('title_emojified');
if (!titleEmojified) {
const emojiMap = makeEmojiMap(poll);
titleEmojified = emojify(escapeTextContentForBrowser(option.get('title')), emojiMap);
}
return ( return (
<li key={option.get('title')}> <li key={option.get('title')}>

View file

@ -173,4 +173,17 @@ describe('normalizeStatus', () => {
expect(result.poll.voted).toBe(false); expect(result.poll.voted).toBe(false);
expect(result.poll.own_votes).toBe(null); expect(result.poll.own_votes).toBe(null);
}); });
it('normalizes poll with emojis', () => {
const status = fromJS(require('soapbox/__fixtures__/pleroma-status-with-poll-with-emojis.json'));
const result = normalizeStatus(status);
// Emojifies poll options
expect(result.poll.options.get(1).title_emojified)
.toEqual('Custom emoji <img draggable="false" class="emojione" alt=":gleason_excited:" title=":gleason_excited:" src="https://gleasonator.com/emoji/gleason_emojis/gleason_excited.png" /> ');
// Parses emojis as Immutable.Record's
expect(ImmutableRecord.isRecord(result.poll.emojis.get(0))).toBe(true);
expect(result.poll.emojis.get(1).shortcode).toEqual('soapbox');
});
}); });

View file

@ -1,12 +1,14 @@
import escapeTextContentForBrowser from 'escape-html';
import { import {
Map as ImmutableMap, Map as ImmutableMap,
List as ImmutableList, List as ImmutableList,
Record as ImmutableRecord, Record as ImmutableRecord,
} from 'immutable'; } from 'immutable';
import emojify from 'soapbox/features/emoji/emoji';
import { normalizeAccount } from 'soapbox/normalizers/account'; import { normalizeAccount } from 'soapbox/normalizers/account';
import { IStatus } from 'soapbox/types'; import { IStatus } from 'soapbox/types';
import { mergeDefined } from 'soapbox/utils/normalizers'; import { mergeDefined, makeEmojiMap } from 'soapbox/utils/normalizers';
const StatusRecord = ImmutableRecord({ const StatusRecord = ImmutableRecord({
account: ImmutableMap(), account: ImmutableMap(),
@ -47,25 +49,6 @@ const StatusRecord = ImmutableRecord({
spoilerHtml: '', spoilerHtml: '',
}); });
const PollOptionRecord = ImmutableRecord({
title: '',
votes_count: 0,
});
// https://docs.joinmastodon.org/entities/poll/
const PollRecord = ImmutableRecord({
emojis: ImmutableList(),
expired: false,
expires_at: new Date(),
id: '',
multiple: false,
options: ImmutableList(),
voters_count: 0,
votes_count: 0,
own_votes: null,
voted: false,
});
// https://docs.joinmastodon.org/entities/attachment/ // https://docs.joinmastodon.org/entities/attachment/
const AttachmentRecord = ImmutableRecord({ const AttachmentRecord = ImmutableRecord({
blurhash: undefined, blurhash: undefined,
@ -87,6 +70,38 @@ const MentionRecord = ImmutableRecord({
url: '', url: '',
}); });
// https://docs.joinmastodon.org/entities/poll/
const PollRecord = ImmutableRecord({
emojis: ImmutableList(),
expired: false,
expires_at: new Date(),
id: '',
multiple: false,
options: ImmutableList(),
voters_count: 0,
votes_count: 0,
own_votes: null,
voted: false,
});
// Sub-entity of Poll
const PollOptionRecord = ImmutableRecord({
title: '',
votes_count: 0,
// Internal fields
title_emojified: '',
});
// https://docs.joinmastodon.org/entities/emoji/
const EmojiRecord = ImmutableRecord({
category: '',
shortcode: '',
static_url: '',
url: '',
visible_in_picker: true,
});
// Ensure attachments have required fields // Ensure attachments have required fields
// https://docs.joinmastodon.org/entities/attachment/ // https://docs.joinmastodon.org/entities/attachment/
const normalizeAttachment = (attachment: ImmutableMap<string, any>) => { const normalizeAttachment = (attachment: ImmutableMap<string, any>) => {
@ -121,10 +136,28 @@ const normalizeMentions = (status: ImmutableMap<string, any>) => {
}); });
}; };
// Normalize emojis
const normalizeEmojis = (map: ImmutableMap<string, any>) => {
return map.update('emojis', ImmutableList(), emojis => {
return emojis.map(EmojiRecord);
});
};
const normalizePollOption = (option: ImmutableMap<string, any>, emojis: ImmutableList<ImmutableMap<string, string>> = ImmutableList()) => {
const emojiMap = makeEmojiMap(emojis);
const titleEmojified = emojify(escapeTextContentForBrowser(option.get('title')), emojiMap);
return PollOptionRecord(
option.set('title_emojified', titleEmojified),
);
};
// Normalize poll options // Normalize poll options
const normalizePollOptions = (poll: ImmutableMap<string, any>) => { const normalizePollOptions = (poll: ImmutableMap<string, any>) => {
const emojis = poll.get('emojis');
return poll.update('options', (options: ImmutableList<ImmutableMap<string, any>>) => { return poll.update('options', (options: ImmutableList<ImmutableMap<string, any>>) => {
return options.map(PollOptionRecord); return options.map(option => normalizePollOption(option, emojis));
}); });
}; };
@ -146,6 +179,7 @@ const normalizePollVoted = (poll: ImmutableMap<string, any>) => {
const normalizePoll = (poll: ImmutableMap<string, any>) => { const normalizePoll = (poll: ImmutableMap<string, any>) => {
return PollRecord( return PollRecord(
poll.withMutations((poll: ImmutableMap<string, any>) => { poll.withMutations((poll: ImmutableMap<string, any>) => {
normalizeEmojis(poll);
normalizePollOptions(poll); normalizePollOptions(poll);
normalizePollOwnVotes(poll); normalizePollOwnVotes(poll);
normalizePollVoted(poll); normalizePollVoted(poll);
@ -209,6 +243,7 @@ export const normalizeStatus = (status: ImmutableMap<string, any>): IStatus => {
status.withMutations(status => { status.withMutations(status => {
normalizeAttachments(status); normalizeAttachments(status);
normalizeMentions(status); normalizeMentions(status);
normalizeEmojis(status);
normalizeStatusPoll(status); normalizeStatusPoll(status);
fixMentionsOrder(status); fixMentionsOrder(status);
addSelfMention(status); addSelfMention(status);

View file

@ -5,6 +5,7 @@ import emojify from 'soapbox/features/emoji/emoji';
import { normalizeStatus } from 'soapbox/normalizers/status'; import { normalizeStatus } from 'soapbox/normalizers/status';
import { simulateEmojiReact, simulateUnEmojiReact } from 'soapbox/utils/emoji_reacts'; import { simulateEmojiReact, simulateUnEmojiReact } from 'soapbox/utils/emoji_reacts';
import { stripCompatibilityFeatures } from 'soapbox/utils/html'; import { stripCompatibilityFeatures } from 'soapbox/utils/html';
import { makeEmojiMap } from 'soapbox/utils/normalizers';
import { import {
EMOJI_REACT_REQUEST, EMOJI_REACT_REQUEST,
@ -32,11 +33,6 @@ import { TIMELINE_DELETE } from '../actions/timelines';
const domParser = new DOMParser(); const domParser = new DOMParser();
const makeEmojiMap = record => record.get('emojis').reduce((obj, emoji) => {
obj[`:${emoji.get('shortcode')}:`] = emoji.toJS();
return obj;
}, {});
const minifyStatus = status => { const minifyStatus = status => {
return status.mergeWith((o, n) => n || o, { return status.mergeWith((o, n) => n || o, {
account: status.getIn(['account', 'id']), account: status.getIn(['account', 'id']),
@ -59,7 +55,7 @@ export const calculateStatus = (status, oldStatus, expandSpoilers = false) => {
} else { } else {
const spoilerText = status.get('spoiler_text') || ''; const spoilerText = status.get('spoiler_text') || '';
const searchContent = (ImmutableList([spoilerText, status.get('content')]).concat(status.getIn(['poll', 'options'], ImmutableList()).map(option => option.get('title')))).join('\n\n').replace(/<br\s*\/?>/g, '\n').replace(/<\/p><p>/g, '\n\n'); const searchContent = (ImmutableList([spoilerText, status.get('content')]).concat(status.getIn(['poll', 'options'], ImmutableList()).map(option => option.get('title')))).join('\n\n').replace(/<br\s*\/?>/g, '\n').replace(/<\/p><p>/g, '\n\n');
const emojiMap = makeEmojiMap(status); const emojiMap = makeEmojiMap(status.get('emojis'));
return status.merge({ return status.merge({
search_index: domParser.parseFromString(searchContent, 'text/html').documentElement.textContent, search_index: domParser.parseFromString(searchContent, 'text/html').documentElement.textContent,

View file

@ -1,2 +1,7 @@
// Use new value only if old value is undefined // Use new value only if old value is undefined
export const mergeDefined = (oldVal, newVal) => oldVal === undefined ? newVal : oldVal; export const mergeDefined = (oldVal, newVal) => oldVal === undefined ? newVal : oldVal;
export const makeEmojiMap = emojis => emojis.reduce((obj, emoji) => {
obj[`:${emoji.shortcode}:`] = emoji;
return obj;
}, {});

View file

@ -58,6 +58,7 @@
"@sentry/react": "^6.12.0", "@sentry/react": "^6.12.0",
"@sentry/tracing": "^6.12.0", "@sentry/tracing": "^6.12.0",
"@tabler/icons": "^1.53.0", "@tabler/icons": "^1.53.0",
"@types/escape-html": "^1.0.1",
"array-includes": "^3.0.3", "array-includes": "^3.0.3",
"autoprefixer": "^10.0.0", "autoprefixer": "^10.0.0",
"axios": "^0.21.4", "axios": "^0.21.4",

View file

@ -1747,6 +1747,11 @@
dependencies: dependencies:
"@babel/types" "^7.3.0" "@babel/types" "^7.3.0"
"@types/escape-html@^1.0.1":
version "1.0.1"
resolved "https://registry.yarnpkg.com/@types/escape-html/-/escape-html-1.0.1.tgz#b19b4646915f0ae2c306bf984dc0a59c5cfc97ba"
integrity sha512-4mI1FuUUZiuT95fSVqvZxp/ssQK9zsa86S43h9x3zPOSU9BBJ+BfDkXwuaU7BfsD+e7U0/cUUfJFk3iW2M4okA==
"@types/eslint-scope@^3.7.0": "@types/eslint-scope@^3.7.0":
version "3.7.1" version "3.7.1"
resolved "https://registry.yarnpkg.com/@types/eslint-scope/-/eslint-scope-3.7.1.tgz#8dc390a7b4f9dd9f1284629efce982e41612116e" resolved "https://registry.yarnpkg.com/@types/eslint-scope/-/eslint-scope-3.7.1.tgz#8dc390a7b4f9dd9f1284629efce982e41612116e"