Add new actions & logic for storing pending replies

This commit is contained in:
diondiondion 2025-10-02 14:55:50 +02:00
parent 4809b38f6e
commit b12ad6a3c9
2 changed files with 94 additions and 29 deletions

View File

@ -9,8 +9,9 @@ import { importFetchedStatuses } from './importer';
export const fetchContext = createDataLoadingThunk( export const fetchContext = createDataLoadingThunk(
'status/context', 'status/context',
({ statusId }: { statusId: string }) => apiGetContext(statusId), ({ statusId }: { statusId: string; prefetchOnly?: boolean }) =>
({ context, refresh }, { dispatch }) => { apiGetContext(statusId),
({ context, refresh }, { dispatch, actionArg: { prefetchOnly = false } }) => {
const statuses = context.ancestors.concat(context.descendants); const statuses = context.ancestors.concat(context.descendants);
dispatch(importFetchedStatuses(statuses)); dispatch(importFetchedStatuses(statuses));
@ -18,6 +19,7 @@ export const fetchContext = createDataLoadingThunk(
return { return {
context, context,
refresh, refresh,
prefetchOnly,
}; };
}, },
); );
@ -26,6 +28,14 @@ export const completeContextRefresh = createAction<{ statusId: string }>(
'status/context/complete', 'status/context/complete',
); );
export const showPendingReplies = createAction<{ statusId: string }>(
'status/context/showPendingReplies',
);
export const clearPendingReplies = createAction<{ statusId: string }>(
'status/context/clearPendingReplies',
);
export const setStatusQuotePolicy = createDataLoadingThunk( export const setStatusQuotePolicy = createDataLoadingThunk(
'status/setQuotePolicy', 'status/setQuotePolicy',
({ statusId, policy }: { statusId: string; policy: ApiQuotePolicy }) => { ({ statusId, policy }: { statusId: string; policy: ApiQuotePolicy }) => {

View File

@ -13,7 +13,12 @@ import type {
import type { Status } from 'mastodon/models/status'; import type { Status } from 'mastodon/models/status';
import { blockAccountSuccess, muteAccountSuccess } from '../actions/accounts'; import { blockAccountSuccess, muteAccountSuccess } from '../actions/accounts';
import { fetchContext, completeContextRefresh } from '../actions/statuses'; import {
fetchContext,
completeContextRefresh,
showPendingReplies,
clearPendingReplies,
} from '../actions/statuses';
import { TIMELINE_UPDATE } from '../actions/timelines'; import { TIMELINE_UPDATE } from '../actions/timelines';
import { compareId } from '../compare_id'; import { compareId } from '../compare_id';
@ -26,27 +31,24 @@ interface TimelineUpdateAction extends UnknownAction {
interface State { interface State {
inReplyTos: Record<string, string>; inReplyTos: Record<string, string>;
replies: Record<string, string[]>; replies: Record<string, string[]>;
pendingReplies: Record<
string,
Pick<ApiStatusJSON, 'id' | 'in_reply_to_id'>[]
>;
refreshing: Record<string, AsyncRefreshHeader>; refreshing: Record<string, AsyncRefreshHeader>;
} }
const initialState: State = { const initialState: State = {
inReplyTos: {}, inReplyTos: {},
replies: {}, replies: {},
pendingReplies: {},
refreshing: {}, refreshing: {},
}; };
const normalizeContext = ( const addReply = (
state: Draft<State>, state: Draft<State>,
id: string, { id, in_reply_to_id }: Pick<ApiStatusJSON, 'id' | 'in_reply_to_id'>,
{ ancestors, descendants }: ApiContextJSON, ) => {
): void => {
const addReply = ({
id,
in_reply_to_id,
}: {
id: string;
in_reply_to_id?: string;
}) => {
if (!in_reply_to_id) { if (!in_reply_to_id) {
return; return;
} }
@ -59,19 +61,54 @@ const normalizeContext = (
} }
}; };
const normalizeContext = (
state: Draft<State>,
id: string,
{ ancestors, descendants }: ApiContextJSON,
): void => {
ancestors.forEach((item) => {
addReply(state, item);
});
// We know in_reply_to_id of statuses but `id` itself. // We know in_reply_to_id of statuses but `id` itself.
// So we assume that the status of the id replies to last ancestors. // So we assume that the status of the id replies to last ancestors.
ancestors.forEach(addReply);
if (ancestors[0]) { if (ancestors[0]) {
addReply({ addReply(state, {
id, id,
in_reply_to_id: ancestors[ancestors.length - 1]?.id, in_reply_to_id: ancestors[ancestors.length - 1]?.id,
}); });
} }
descendants.forEach(addReply); descendants.forEach((item) => {
addReply(state, item);
});
};
const applyPrefetchedReplies = (state: Draft<State>, statusId: string) => {
const pendingReplies = state.pendingReplies[statusId];
if (pendingReplies?.length) {
pendingReplies.forEach((item) => {
addReply(state, item);
});
delete state.pendingReplies[statusId];
}
};
const storePrefetchedReplies = (
state: Draft<State>,
statusId: string,
{ descendants }: ApiContextJSON,
): void => {
descendants.forEach(({ id, in_reply_to_id }) => {
if (!in_reply_to_id) {
return;
}
const isNewReply = !state.replies[in_reply_to_id]?.includes(id);
if (isNewReply) {
const pendingReplies = (state.pendingReplies[statusId] ??= []);
pendingReplies.push({ id, in_reply_to_id });
}
});
}; };
const deleteFromContexts = (state: Draft<State>, ids: string[]): void => { const deleteFromContexts = (state: Draft<State>, ids: string[]): void => {
@ -129,11 +166,29 @@ const updateContext = (state: Draft<State>, status: ApiStatusJSON): void => {
export const contextsReducer = createReducer(initialState, (builder) => { export const contextsReducer = createReducer(initialState, (builder) => {
builder builder
.addCase(fetchContext.fulfilled, (state, action) => { .addCase(fetchContext.fulfilled, (state, action) => {
normalizeContext(state, action.meta.arg.statusId, action.payload.context); if (action.payload.prefetchOnly) {
storePrefetchedReplies(
state,
action.meta.arg.statusId,
action.payload.context,
);
} else {
normalizeContext(
state,
action.meta.arg.statusId,
action.payload.context,
);
if (action.payload.refresh) { if (action.payload.refresh) {
state.refreshing[action.meta.arg.statusId] = action.payload.refresh; state.refreshing[action.meta.arg.statusId] = action.payload.refresh;
} }
}
})
.addCase(showPendingReplies, (state, action) => {
applyPrefetchedReplies(state, action.payload.statusId);
})
.addCase(clearPendingReplies, (state, action) => {
delete state.pendingReplies[action.payload.statusId];
}) })
.addCase(completeContextRefresh, (state, action) => { .addCase(completeContextRefresh, (state, action) => {
delete state.refreshing[action.payload.statusId]; delete state.refreshing[action.payload.statusId];