mirror of
https://github.com/mastodon/mastodon.git
synced 2025-10-05 16:42:47 +00:00
Add new actions & logic for storing pending replies
This commit is contained in:
parent
4809b38f6e
commit
b12ad6a3c9
|
@ -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 }) => {
|
||||||
|
|
|
@ -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,52 +31,84 @@ 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 addReply = (
|
||||||
|
state: Draft<State>,
|
||||||
|
{ id, in_reply_to_id }: Pick<ApiStatusJSON, 'id' | 'in_reply_to_id'>,
|
||||||
|
) => {
|
||||||
|
if (!in_reply_to_id) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!state.inReplyTos[id]) {
|
||||||
|
const siblings = (state.replies[in_reply_to_id] ??= []);
|
||||||
|
const index = siblings.findIndex((sibling) => compareId(sibling, id) < 0);
|
||||||
|
siblings.splice(index + 1, 0, id);
|
||||||
|
state.inReplyTos[id] = in_reply_to_id;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const normalizeContext = (
|
const normalizeContext = (
|
||||||
state: Draft<State>,
|
state: Draft<State>,
|
||||||
id: string,
|
id: string,
|
||||||
{ ancestors, descendants }: ApiContextJSON,
|
{ ancestors, descendants }: ApiContextJSON,
|
||||||
): void => {
|
): void => {
|
||||||
const addReply = ({
|
ancestors.forEach((item) => {
|
||||||
id,
|
addReply(state, item);
|
||||||
in_reply_to_id,
|
});
|
||||||
}: {
|
|
||||||
id: string;
|
|
||||||
in_reply_to_id?: string;
|
|
||||||
}) => {
|
|
||||||
if (!in_reply_to_id) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!state.inReplyTos[id]) {
|
|
||||||
const siblings = (state.replies[in_reply_to_id] ??= []);
|
|
||||||
const index = siblings.findIndex((sibling) => compareId(sibling, id) < 0);
|
|
||||||
siblings.splice(index + 1, 0, id);
|
|
||||||
state.inReplyTos[id] = in_reply_to_id;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// 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,12 +166,30 @@ 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];
|
||||||
})
|
})
|
||||||
|
|
Loading…
Reference in New Issue
Block a user