diff --git a/src/actions/importer/index.ts b/src/actions/importer/index.ts index 5afb880c0..ce4e45ba7 100644 --- a/src/actions/importer/index.ts +++ b/src/actions/importer/index.ts @@ -151,47 +151,50 @@ const isBroken = (status: APIEntity) => { } }; -const importFetchedStatuses = (statuses: APIEntity[]) => - (dispatch: AppDispatch, getState: () => RootState) => { - const accounts: APIEntity[] = []; - const normalStatuses: APIEntity[] = []; - const polls: APIEntity[] = []; +const importFetchedStatuses = (statuses: APIEntity[]) => (dispatch: AppDispatch) => { + const accounts: APIEntity[] = []; + const normalStatuses: APIEntity[] = []; + const polls: APIEntity[] = []; - function processStatus(status: APIEntity) { - // Skip broken statuses - if (isBroken(status)) return; + function processStatus(status: APIEntity) { + // Skip broken statuses + if (isBroken(status)) return; - normalStatuses.push(status); - accounts.push(status.account); + normalStatuses.push(status); - if (status.reblog?.id) { - processStatus(status.reblog); - } - - // Fedibird quotes - if (status.quote?.id) { - processStatus(status.quote); - } - - if (status.pleroma?.quote?.id) { - processStatus(status.pleroma.quote); - } - - if (status.poll?.id) { - polls.push(status.poll); - } - - if (status.group?.id) { - dispatch(importFetchedGroup(status.group)); - } + accounts.push(status.account); + if (status.accounts) { + accounts.push(...status.accounts); } - statuses.forEach(processStatus); + if (status.reblog?.id) { + processStatus(status.reblog); + } - dispatch(importPolls(polls)); - dispatch(importFetchedAccounts(accounts)); - dispatch(importStatuses(normalStatuses)); - }; + // Fedibird quotes + if (status.quote?.id) { + processStatus(status.quote); + } + + if (status.pleroma?.quote?.id) { + processStatus(status.pleroma.quote); + } + + if (status.poll?.id) { + polls.push(status.poll); + } + + if (status.group?.id) { + dispatch(importFetchedGroup(status.group)); + } + } + + statuses.forEach(processStatus); + + dispatch(importPolls(polls)); + dispatch(importFetchedAccounts(accounts)); + dispatch(importStatuses(normalStatuses)); +}; const importFetchedPoll = (poll: APIEntity) => (dispatch: AppDispatch) => { diff --git a/src/actions/notifications.ts b/src/actions/notifications.ts index c6ce8043a..4cbc73c74 100644 --- a/src/actions/notifications.ts +++ b/src/actions/notifications.ts @@ -46,7 +46,7 @@ const NOTIFICATIONS_MARK_READ_FAIL = 'NOTIFICATIONS_MARK_READ_FAIL'; const MAX_QUEUED_NOTIFICATIONS = 40; defineMessages({ - mention: { id: 'notification.mention', defaultMessage: '{name} mentioned you' }, + mention: { id: 'notification.mentioned', defaultMessage: '{name} mentioned you' }, group: { id: 'notifications.group', defaultMessage: '{count, plural, one {# notification} other {# notifications}}' }, }); @@ -175,6 +175,47 @@ const excludeTypesFromFilter = (filter: string) => { const noOp = () => new Promise(f => f(undefined)); +const STATUS_NOTIFICATION_TYPES = [ + 'favourite', + 'group_favourite', + 'mention', + 'reblog', + 'group_reblog', + 'status', + 'poll', + 'update', + 'pleroma:emoji_reaction', + 'pleroma:event_reminder', + 'pleroma:participation_accepted', + 'pleroma:participation_request', +]; + +const deduplicateNotifications = (notifications: any[]) => { + const deduplicatedNotifications: any[] = []; + + for (const notification of notifications) { + if (STATUS_NOTIFICATION_TYPES.includes(notification.type)) { + const existingNotification = deduplicatedNotifications + .find(deduplicatedNotification => deduplicatedNotification.type === notification.type && deduplicatedNotification.status?.id === notification.status?.id); + + if (existingNotification) { + if (existingNotification?.accounts) { + existingNotification.accounts.push(notification.account); + } else { + existingNotification.accounts = [existingNotification.account, notification.account]; + } + existingNotification.id += ':' + notification.id; + } else { + deduplicatedNotifications.push(notification); + } + } else { + deduplicatedNotifications.push(notification); + } + } + + return deduplicatedNotifications; +}; + const expandNotifications = ({ maxId }: Record = {}, done: () => any = noOp) => (dispatch: AppDispatch, getState: () => RootState) => { if (!isLoggedIn(getState)) return dispatch(noOp); @@ -240,7 +281,9 @@ const expandNotifications = ({ maxId }: Record = {}, done: () => an const statusesFromGroups = (Object.values(entries.statuses) as Status[]).filter((status) => !!status.group); dispatch(fetchGroupRelationships(statusesFromGroups.map((status: any) => status.group?.id))); - dispatch(expandNotificationsSuccess(response.data, next ? next.uri : null, isLoadingMore)); + const deduplicatedNotifications = deduplicateNotifications(response.data); + + dispatch(expandNotificationsSuccess(deduplicatedNotifications, next ? next.uri : null, isLoadingMore)); fetchRelatedRelationships(dispatch, response.data); done(); }).catch(error => { diff --git a/src/actions/timelines.ts b/src/actions/timelines.ts index 08d514e57..1370519d6 100644 --- a/src/actions/timelines.ts +++ b/src/actions/timelines.ts @@ -147,6 +147,27 @@ const parseTags = (tags: Record = {}, mode: 'any' | 'all' | 'none }); }; +const deduplicateStatuses = (statuses: any[]) => { + const deduplicatedStatuses: any[] = []; + + for (const status of statuses) { + const reblogged = status.reblog && deduplicatedStatuses.find((deduplicatedStatuses) => deduplicatedStatuses.reblog?.id === status.reblog.id); + + if (reblogged) { + if (reblogged.accounts) { + reblogged.accounts.push(status.account); + } else { + reblogged.accounts = [reblogged.account, status.account]; + } + reblogged.id += ':' + status.id; + } else { + deduplicatedStatuses.push(status); + } + } + + return deduplicatedStatuses; +}; + const expandTimeline = (timelineId: string, path: string, params: Record = {}, intl?: IntlShape, done = noOp) => (dispatch: AppDispatch, getState: () => RootState) => { const timeline = getState().timelines.get(timelineId) || {} as Record; @@ -175,12 +196,15 @@ const expandTimeline = (timelineId: string, path: string, params: Record { dispatch(importFetchedStatuses(response.data)); + const statuses = deduplicateStatuses(response.data); + dispatch(importFetchedStatuses(statuses.filter(status => status.accounts))); + const statusesFromGroups = (response.data as Status[]).filter((status) => !!status.group); dispatch(fetchGroupRelationships(statusesFromGroups.map((status: any) => status.group?.id))); dispatch(expandTimelineSuccess( timelineId, - response.data, + statuses, getNextLink(response), getPrevLink(response), response.status === 206, diff --git a/src/components/status.tsx b/src/components/status.tsx index 54f5b2e88..9d4cebc9c 100644 --- a/src/components/status.tsx +++ b/src/components/status.tsx @@ -1,6 +1,7 @@ import clsx from 'clsx'; +import { List as ImmutableList } from 'immutable'; import React, { useEffect, useRef, useState } from 'react'; -import { useIntl, FormattedMessage, defineMessages } from 'react-intl'; +import { useIntl, FormattedMessage, defineMessages, FormattedList } from 'react-intl'; import { Link, useHistory } from 'react-router-dom'; import { mentionCompose, replyCompose } from 'soapbox/actions/compose'; @@ -240,6 +241,31 @@ const Status: React.FC = (props) => { /> ); } else if (isReblog) { + const accounts = status.accounts || ImmutableList([status.account]); + + const renderedAccounts = accounts.slice(0, 2).map(account => !!account && ( + + + + + + )).toArray().filter(Boolean); + + if (accounts.size > 2) { + renderedAccounts.push( + , + ); + } + return ( = (props) => { id='status.reblogged_by' defaultMessage='{name} reposted' values={{ - name: ( - - - - - - ), + name: , + count: accounts.size, }} /> } diff --git a/src/features/compose/components/compose-form.tsx b/src/features/compose/components/compose-form.tsx index 7937a85fa..a5c2a2a5a 100644 --- a/src/features/compose/components/compose-form.tsx +++ b/src/features/compose/components/compose-form.tsx @@ -102,13 +102,13 @@ const ComposeForm = ({ id, shouldCondense, autoFocus, clickab const spoilerTextRef = useRef(null); const editorRef = useRef(null); - const characterCountProgress = length(text) / maxTootChars; - const { isDraggedOver } = useDraggedFiles(formRef); const text = editorRef.current?.getEditorState().read(() => $getRoot().getTextContent()) ?? ''; const fulltext = [spoilerText, countableText(text)].join(''); + const characterCountProgress = length(text) / maxTootChars; + const isEmpty = !(fulltext.trim() || anyMedia); const condensed = shouldCondense && !isDraggedOver && !composeFocused && isEmpty && !isUploading; const shouldAutoFocus = autoFocus && !showSearch; diff --git a/src/features/notifications/components/notification.tsx b/src/features/notifications/components/notification.tsx index 5f81d69bb..1c2057645 100644 --- a/src/features/notifications/components/notification.tsx +++ b/src/features/notifications/components/notification.tsx @@ -1,5 +1,6 @@ +import { List as ImmutableList } from 'immutable'; import React, { useCallback } from 'react'; -import { defineMessages, useIntl, FormattedMessage, IntlShape, MessageDescriptor, defineMessage } from 'react-intl'; +import { defineMessages, useIntl, FormattedMessage, IntlShape, MessageDescriptor, FormattedList } from 'react-intl'; import { Link, useHistory } from 'react-router-dom'; import { mentionCompose } from 'soapbox/actions/compose'; @@ -17,7 +18,11 @@ import { makeGetNotification } from 'soapbox/selectors'; import { NotificationType, validType } from 'soapbox/utils/notification'; import type { ScrollPosition } from 'soapbox/components/status'; -import type { Account as AccountEntity, Status as StatusEntity, Notification as NotificationEntity } from 'soapbox/types/entities'; +import type { + Account as AccountEntity, + Status as StatusEntity, + Notification as NotificationEntity, +} from 'soapbox/types/entities'; const notificationForScreenReader = (intl: IntlShape, message: string, timestamp: Date) => { const output = [message]; @@ -58,11 +63,6 @@ const icons: Record = { 'pleroma:participation_accepted': require('@tabler/icons/calendar-event.svg'), }; -const nameMessage = defineMessage({ - id: 'notification.name', - defaultMessage: '{link}{others}', -}); - const messages: Record = defineMessages({ follow: { id: 'notification.follow', @@ -138,26 +138,30 @@ const buildMessage = ( intl: IntlShape, type: NotificationType, account: AccountEntity, - totalCount: number | null, + accounts: ImmutableList | null, targetName: string, instanceTitle: string, ): React.ReactNode => { - const link = buildLink(account); - const name = intl.formatMessage(nameMessage, { - link, - others: totalCount && totalCount > 0 ? ( + + if (!accounts) accounts = accounts || ImmutableList([account]); + + const renderedAccounts = accounts.slice(0, 2).map(account => buildLink(account)).toArray().filter(Boolean); + + if (accounts.size > 2) { + renderedAccounts.push( - ) : '', - }); + id='notification.more' + defaultMessage='{count, plural, one {# other} other {# others}}' + values={{ count: accounts.size - renderedAccounts.length }} + />, + ); + } return intl.formatMessage(messages[type], { - name, + name: , targetName, instance: instanceTitle, + count: accounts.size, }); }; @@ -187,7 +191,7 @@ const Notification: React.FC = (props) => { const instance = useInstance(); const type = notification.type; - const { account, status } = notification; + const { account, accounts, status } = notification; const getHandlers = () => ({ reply: handleMention, @@ -356,7 +360,9 @@ const Notification: React.FC = (props) => { const targetName = notification.target && typeof notification.target === 'object' ? notification.target.acct : ''; - const message: React.ReactNode = validType(type) && account && typeof account === 'object' ? buildMessage(intl, type, account, notification.total_count, targetName, instance.title) : null; + const message: React.ReactNode = validType(type) && account && typeof account === 'object' + ? buildMessage(intl, type, account, accounts as ImmutableList, targetName, instance.title) + : null; const ariaLabel = validType(type) ? ( notificationForScreenReader( diff --git a/src/locales/pl.json b/src/locales/pl.json index c3b471b86..08af496cf 100644 --- a/src/locales/pl.json +++ b/src/locales/pl.json @@ -1062,25 +1062,25 @@ "new_group_panel.action": "Utwórz grupę", "new_group_panel.subtitle": "Nie możesz znaleźć tego, czego szukasz? Utwórz własną prywatną lub publiczną grupę.", "new_group_panel.title": "Utwórz grupę", - "notification.favourite": "{name} dodał(a) Twój wpis do ulubionych", - "notification.follow": "{name} zaczął(-ęła) Cię obserwować", - "notification.follow_request": "{name} poprosił(a) Cię o możliwość obserwacji", - "notification.group_favourite": "{name} polubił(a) Twój graficzny wpis", - "notification.group_reblog": "{name} udostępnił(a) Twój grupowy wpis", - "notification.mention": "{name} wspomniał(a) o tobie", + "notification.favourite": "{name} {count, plural, one {dodał(a)} other {dodali}} Twój wpis do ulubionych", + "notification.follow": "{name} {count, plural, one {zaczął(-ęła)} other {zaczęli}} Cię obserwować", + "notification.follow_request": "{name} {count, plural, one {poprosił(a)} other {poprosili}} Cię o możliwość obserwacji", + "notification.group_favourite": "{name} {count, plural, one {polubił(a)} other {polubili}} Twój grupowy wpis", + "notification.group_reblog": "{name} {count, plural, one {udostępnił(a)} other {udostępnili}} Twój grupowy wpis", "notification.mentioned": "{name} wspomniał(a) o tobie", - "notification.move": "{name} przeniósł(-osła) się na {targetName}", + "notification.more": "{count, plural, one {# inny użytkownik} other {# innych użytkowników}}", + "notification.move": "{name} {count, plural, one {przeniosł(a)} other {przenieśli}} się na {targetName}", "notification.name": "{link}{others}", "notification.others": "+ {count} więcej", - "notification.pleroma:chat_mention": "{name} wysłał(a) Ci wiadomośść", - "notification.pleroma:emoji_reaction": "{name} zareagował(a) na Twój wpis", + "notification.pleroma:chat_mention": "{name} {count, plural, one {wysłał(a)} other {wysłali}} Ci wiadomośść", + "notification.pleroma:emoji_reaction": "{name} {count, plural, one {zareagował(a)} other {zareagowali}} na Twój wpis", "notification.pleroma:event_reminder": "Wydarzenie w którym bierzesz udział wkrótce się zaczyna", "notification.pleroma:participation_accepted": "Twoje zgłoszenie udziału do wydarzenia zostało przyjęte", "notification.pleroma:participation_request": "{name} cce wziąć udział w Twoim wydarzeniu", "notification.poll": "Głosowanie w którym brałeś(-aś) udział zakończyła się", "notification.reblog": "{name} podbił(a) Twój wpis", - "notification.status": "{name} właśnie opublikował(a) wpis", - "notification.update": "{name} zedytował(a) wpis, który podbiłeś(-aś)", + "notification.status": "{name} właśnie {count, plural, one {opublikował(a)} other {opublikowali}} wpis", + "notification.update": "{name} {count, plural, one {zedytował(a)} other {zedytowali}} wpis, który podbiłeś(-aś)", "notification.user_approved": "Witamy w {instance}!", "notifications.filter.all": "Wszystkie", "notifications.filter.boosts": "Podbicia", @@ -1423,7 +1423,7 @@ "status.read_more": "Czytaj dalej", "status.reblog": "Podbij", "status.reblog_private": "Podbij dla odbiorców oryginalnego wpisu", - "status.reblogged_by": "{name} podbił(a)", + "status.reblogged_by": "{name} {count, plural, one {podbił(a)} other {podbili}}", "status.reblogged_by_with_group": "{name} udostępnił(a) z {group}", "status.reblogs.empty": "Nikt nie podbił jeszcze tego wpisu. Gdy ktoś to zrobi, pojawi się tutaj.", "status.redraft": "Usuń i przeredaguj", diff --git a/src/normalizers/notification.ts b/src/normalizers/notification.ts index 45eb93fb3..ab76c976f 100644 --- a/src/normalizers/notification.ts +++ b/src/normalizers/notification.ts @@ -4,6 +4,7 @@ * @see {@link https://docs.joinmastodon.org/entities/notification/} */ import { + List as ImmutableList, Map as ImmutableMap, Record as ImmutableRecord, fromJS, @@ -14,6 +15,7 @@ import type { Account, Status, EmbeddedEntity } from 'soapbox/types/entities'; // https://docs.joinmastodon.org/entities/notification/ export const NotificationRecord = ImmutableRecord({ account: null as EmbeddedEntity, + accounts: null as ImmutableList> | null, chat_message: null as ImmutableMap | string | null, // pleroma:chat_mention created_at: new Date(), emoji: null as string | null, // pleroma:emoji_reaction diff --git a/src/normalizers/status.ts b/src/normalizers/status.ts index d444f91e5..fce97c782 100644 --- a/src/normalizers/status.ts +++ b/src/normalizers/status.ts @@ -44,6 +44,7 @@ interface Tombstone { // https://docs.joinmastodon.org/entities/status/ export const StatusRecord = ImmutableRecord({ account: null as unknown as Account, + accounts: null as ImmutableList | null, application: null as ImmutableMap | null, approval_status: 'approved' as StatusApprovalStatus, bookmarked: false, @@ -274,6 +275,17 @@ const parseAccount = (status: ImmutableMap) => { } }; +const parseAccounts = (status: ImmutableMap) => { + try { + if (status.get('accounts')) { + const accounts = status.get('accounts').map((account: ImmutableMap) => accountSchema.parse(maybeFromJS(account))); + return status.set('accounts', accounts); + } + } catch (_e) { + return status.set('accounts', null); + } +}; + const parseGroup = (status: ImmutableMap) => { try { const group = groupSchema.parse(status.get('group').toJS()); @@ -303,6 +315,7 @@ export const normalizeStatus = (status: Record) => { normalizeDislikes(status); normalizeTombstone(status); parseAccount(status); + parseAccounts(status); parseGroup(status); }), ); diff --git a/src/reducers/notifications.ts b/src/reducers/notifications.ts index f2147b2cf..5c7ea011f 100644 --- a/src/reducers/notifications.ts +++ b/src/reducers/notifications.ts @@ -71,6 +71,7 @@ const comparator = (a: NotificationRecord, b: NotificationRecord) => { const minifyNotification = (notification: NotificationRecord) => { return notification.mergeWith((o, n) => n || o, { account: notification.getIn(['account', 'id']) as string, + accounts: notification.accounts?.map((account: any) => account.get('id')), target: notification.getIn(['target', 'id']) as string, status: notification.getIn(['status', 'id']) as string, }); diff --git a/src/schemas/status.ts b/src/schemas/status.ts index cf66e4e3d..885a61746 100644 --- a/src/schemas/status.ts +++ b/src/schemas/status.ts @@ -26,6 +26,7 @@ const statusPleromaSchema = z.object({ const baseStatusSchema = z.object({ account: accountSchema, + accounts: z.array(accountSchema), application: z.object({ name: z.string(), website: z.string().url().nullable().catch(null), diff --git a/src/selectors/index.ts b/src/selectors/index.ts index 03487579b..6dfe7562e 100644 --- a/src/selectors/index.ts +++ b/src/selectors/index.ts @@ -28,6 +28,10 @@ export function selectAccount(state: RootState, accountId: string) { return state.entities[Entities.ACCOUNTS]?.store[accountId] as AccountSchema | undefined; } +export function selectAccounts(state: RootState, accountIds: ImmutableList) { + return accountIds.map(accountId => state.entities[Entities.ACCOUNTS]?.store[accountId] as AccountSchema | undefined); +} + export function selectOwnAccount(state: RootState) { if (state.me) { return selectAccount(state, state.me); @@ -162,7 +166,8 @@ export const makeGetNotification = () => { (state: RootState, notification: Notification) => selectAccount(state, normalizeId(notification.account)), (state: RootState, notification: Notification) => selectAccount(state, normalizeId(notification.target)), (state: RootState, notification: Notification) => state.statuses.get(normalizeId(notification.status)), - ], (notification, account, target, status) => { + (state: RootState, notification: Notification) => notification.accounts ? selectAccounts(state, notification.accounts?.map(normalizeId)) : null, + ], (notification, account, target, status, accounts) => { return notification.merge({ // @ts-ignore account: account || null, @@ -170,6 +175,8 @@ export const makeGetNotification = () => { target: target || null, // @ts-ignore status: status || null, + // @ts-ignore + accounts, }); }); };