nas/app/javascript/mastodon/reducers/contexts.ts
2025-04-28 13:38:40 +00:00

155 lines
4.1 KiB
TypeScript

/* eslint-disable @typescript-eslint/no-dynamic-delete */
import { createReducer } from '@reduxjs/toolkit';
import type { Draft, UnknownAction } from '@reduxjs/toolkit';
import type { List as ImmutableList } from 'immutable';
import { timelineDelete } from 'mastodon/actions/timelines_typed';
import type { ApiRelationshipJSON } from 'mastodon/api_types/relationships';
import type {
ApiStatusJSON,
ApiContextJSON,
} from 'mastodon/api_types/statuses';
import type { Status } from 'mastodon/models/status';
import { blockAccountSuccess, muteAccountSuccess } from '../actions/accounts';
import { fetchContext } from '../actions/statuses';
import { TIMELINE_UPDATE } from '../actions/timelines';
import { compareId } from '../compare_id';
interface TimelineUpdateAction extends UnknownAction {
timeline: string;
status: ApiStatusJSON;
usePendingItems: boolean;
}
interface State {
inReplyTos: Record<string, string>;
replies: Record<string, string[]>;
}
const initialState: State = {
inReplyTos: {},
replies: {},
};
const normalizeContext = (
state: Draft<State>,
id: string,
{ ancestors, descendants }: ApiContextJSON,
): void => {
const addReply = ({
id,
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.
// So we assume that the status of the id replies to last ancestors.
ancestors.forEach(addReply);
if (ancestors[0]) {
addReply({
id,
in_reply_to_id: ancestors[ancestors.length - 1]?.id,
});
}
descendants.forEach(addReply);
};
const deleteFromContexts = (state: Draft<State>, ids: string[]): void => {
ids.forEach((id) => {
const inReplyToIdOfId = state.inReplyTos[id];
const repliesOfId = state.replies[id];
if (inReplyToIdOfId) {
const siblings = state.replies[inReplyToIdOfId];
if (siblings) {
state.replies[inReplyToIdOfId] = siblings.filter(
(sibling) => sibling !== id,
);
}
}
if (repliesOfId) {
repliesOfId.forEach((reply) => {
delete state.inReplyTos[reply];
});
}
delete state.inReplyTos[id];
delete state.replies[id];
});
};
const filterContexts = (
state: Draft<State>,
relationship: ApiRelationshipJSON,
statuses: ImmutableList<Status>,
): void => {
const ownedStatusIds = statuses
.filter((status) => (status.get('account') as string) === relationship.id)
.map((status) => status.get('id') as string);
deleteFromContexts(state, ownedStatusIds.toArray());
};
const updateContext = (state: Draft<State>, status: ApiStatusJSON): void => {
if (!status.in_reply_to_id) {
return;
}
const siblings = (state.replies[status.in_reply_to_id] ??= []);
state.inReplyTos[status.id] = status.in_reply_to_id;
if (!siblings.includes(status.id)) {
siblings.push(status.id);
}
};
export const contextsReducer = createReducer(initialState, (builder) => {
builder
.addCase(fetchContext.fulfilled, (state, action) => {
normalizeContext(state, action.meta.arg.statusId, action.payload.context);
})
.addCase(blockAccountSuccess, (state, action) => {
filterContexts(
state,
action.payload.relationship,
action.payload.statuses as ImmutableList<Status>,
);
})
.addCase(muteAccountSuccess, (state, action) => {
filterContexts(
state,
action.payload.relationship,
action.payload.statuses as ImmutableList<Status>,
);
})
.addCase(timelineDelete, (state, action) => {
deleteFromContexts(state, [action.payload.statusId]);
})
.addMatcher(
(action: UnknownAction): action is TimelineUpdateAction =>
action.type === TIMELINE_UPDATE,
(state, action) => {
updateContext(state, action.status);
},
);
});