phanpy/src/utils/states.js

83 lines
2.3 KiB
JavaScript
Raw Normal View History

2022-12-10 12:14:48 +03:00
import { proxy } from 'valtio';
2023-01-09 14:11:34 +03:00
const states = proxy({
2022-12-10 12:14:48 +03:00
history: [],
statuses: {},
2023-01-10 14:59:02 +03:00
statusThreadNumber: {},
2022-12-10 12:14:48 +03:00
home: [],
homeNew: [],
homeLastFetchTime: null,
notifications: [],
notificationsNew: [],
notificationsLastFetchTime: null,
accounts: {},
2022-12-10 12:14:48 +03:00
reloadStatusPage: 0,
spoilers: {},
scrollPositions: {},
2022-12-10 12:14:48 +03:00
// Modals
showCompose: false,
showSettings: false,
showAccount: false,
composeCharacterCount: 0,
2022-12-10 12:14:48 +03:00
});
2023-01-09 14:11:34 +03:00
export default states;
export function saveStatus(status, opts) {
2023-01-10 14:59:02 +03:00
const { override, skipThreading } = Object.assign(
{ override: true, skipThreading: false },
opts,
);
2023-01-09 14:11:34 +03:00
if (!status) return;
if (!override && states.statuses[status.id]) return;
states.statuses[status.id] = status;
if (status.reblog) {
states.statuses[status.reblog.id] = status.reblog;
}
2023-01-10 14:59:02 +03:00
// THREAD TRAVERSER
if (!skipThreading) {
requestAnimationFrame(() => {
threadifyStatus(status);
2023-01-10 17:58:54 +03:00
if (status.reblog) {
threadifyStatus(status.reblog);
}
2023-01-10 14:59:02 +03:00
});
}
}
function threadifyStatus(status) {
// Return all statuses in the thread, via inReplyToId, if inReplyToAccountId === account.id
let fetchIndex = 0;
async function traverse(status, index = 0) {
const { inReplyToId, inReplyToAccountId } = status;
if (!inReplyToId || inReplyToAccountId !== status.account.id) {
return [status];
}
if (inReplyToId && inReplyToAccountId !== status.account.id) {
throw 'Not a thread';
// Possibly thread of replies by multiple people?
}
let prevStatus = states.statuses[inReplyToId];
if (!prevStatus) {
if (fetchIndex++ > 3) throw 'Too many fetches for thread'; // Some people revive old threads
await new Promise((r) => setTimeout(r, 500 * fetchIndex)); // Be nice to rate limits
prevStatus = await masto.v1.statuses.fetch(inReplyToId);
saveStatus(prevStatus, { skipThreading: true });
}
// Prepend so that first status in thread will be index 0
return [...(await traverse(prevStatus, ++index)), status];
}
return traverse(status)
.then((statuses) => {
if (statuses.length > 1) {
console.debug('THREAD', statuses);
statuses.forEach((status, index) => {
states.statusThreadNumber[status.id] = index + 1;
});
}
})
.catch((e) => {
console.error(e, status);
});
2023-01-09 14:11:34 +03:00
}