2023-02-07 19:31:46 +03:00
|
|
|
import { useEffect, useRef } from 'preact/hooks';
|
2023-02-03 16:08:08 +03:00
|
|
|
import { useSnapshot } from 'valtio';
|
|
|
|
|
|
|
|
import Timeline from '../components/timeline';
|
2023-02-05 19:17:19 +03:00
|
|
|
import { api } from '../utils/api';
|
2023-03-21 19:09:36 +03:00
|
|
|
import { filteredItems } from '../utils/filters';
|
2023-02-05 20:10:49 +03:00
|
|
|
import states from '../utils/states';
|
2023-02-07 19:31:46 +03:00
|
|
|
import { getStatus, saveStatus } from '../utils/states';
|
2023-03-26 18:18:36 +03:00
|
|
|
import { dedupeBoosts } from '../utils/timeline-utils';
|
2023-02-03 16:08:08 +03:00
|
|
|
import useTitle from '../utils/useTitle';
|
|
|
|
|
|
|
|
const LIMIT = 20;
|
|
|
|
|
2023-02-15 05:49:36 +03:00
|
|
|
function Following({ title, path, id, ...props }) {
|
2023-02-18 16:37:34 +03:00
|
|
|
useTitle(title || 'Following', path || '/following');
|
2023-02-06 18:50:00 +03:00
|
|
|
const { masto, instance } = api();
|
2023-02-03 16:08:08 +03:00
|
|
|
const snapStates = useSnapshot(states);
|
|
|
|
const homeIterator = useRef();
|
2023-02-07 19:31:46 +03:00
|
|
|
const latestItem = useRef();
|
|
|
|
|
2023-04-14 10:30:04 +03:00
|
|
|
console.debug('RENDER Following', title, id);
|
|
|
|
|
2023-02-03 16:08:08 +03:00
|
|
|
async function fetchHome(firstLoad) {
|
|
|
|
if (firstLoad || !homeIterator.current) {
|
|
|
|
homeIterator.current = masto.v1.timelines.listHome({ limit: LIMIT });
|
|
|
|
}
|
2023-02-06 18:50:00 +03:00
|
|
|
const results = await homeIterator.current.next();
|
2023-03-21 19:09:36 +03:00
|
|
|
let { value } = results;
|
2023-02-06 18:50:00 +03:00
|
|
|
if (value?.length) {
|
2023-02-07 19:31:46 +03:00
|
|
|
if (firstLoad) {
|
|
|
|
latestItem.current = value[0].id;
|
2023-03-21 19:09:36 +03:00
|
|
|
console.log('First load', latestItem.current);
|
2023-02-07 19:31:46 +03:00
|
|
|
}
|
|
|
|
|
2023-03-21 19:09:36 +03:00
|
|
|
value = filteredItems(value, 'home');
|
2023-02-06 18:50:00 +03:00
|
|
|
value.forEach((item) => {
|
|
|
|
saveStatus(item, instance);
|
|
|
|
});
|
2023-03-26 19:47:29 +03:00
|
|
|
value = dedupeBoosts(value, instance);
|
2023-02-10 17:23:19 +03:00
|
|
|
|
|
|
|
// ENFORCE sort by datetime (Latest first)
|
|
|
|
value.sort((a, b) => {
|
|
|
|
const aDate = new Date(a.createdAt);
|
|
|
|
const bDate = new Date(b.createdAt);
|
|
|
|
return bDate - aDate;
|
|
|
|
});
|
2023-02-06 18:50:00 +03:00
|
|
|
}
|
2023-05-14 16:13:36 +03:00
|
|
|
return {
|
|
|
|
...results,
|
|
|
|
value,
|
|
|
|
};
|
2023-02-03 16:08:08 +03:00
|
|
|
}
|
|
|
|
|
2023-02-07 19:31:46 +03:00
|
|
|
async function checkForUpdates() {
|
|
|
|
try {
|
|
|
|
const results = await masto.v1.timelines
|
|
|
|
.listHome({
|
|
|
|
limit: 5,
|
|
|
|
since_id: latestItem.current,
|
|
|
|
})
|
|
|
|
.next();
|
2023-03-21 19:09:36 +03:00
|
|
|
let { value } = results;
|
2023-02-09 17:27:49 +03:00
|
|
|
console.log('checkForUpdates', latestItem.current, value);
|
2023-03-25 15:18:53 +03:00
|
|
|
if (value?.length) {
|
|
|
|
latestItem.current = value[0].id;
|
2023-03-26 18:18:36 +03:00
|
|
|
value = dedupeBoosts(value, instance);
|
2023-03-25 15:18:53 +03:00
|
|
|
value = filteredItems(value, 'home');
|
|
|
|
if (value.some((item) => !item.reblog)) {
|
|
|
|
return true;
|
|
|
|
}
|
2023-02-07 19:31:46 +03:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
} catch (e) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const ws = useRef();
|
2023-02-08 14:11:33 +03:00
|
|
|
const streamUser = async () => {
|
|
|
|
console.log('🎏 Start streaming user', ws.current);
|
2023-02-07 19:31:46 +03:00
|
|
|
if (
|
|
|
|
ws.current &&
|
|
|
|
(ws.current.readyState === WebSocket.CONNECTING ||
|
|
|
|
ws.current.readyState === WebSocket.OPEN)
|
|
|
|
) {
|
|
|
|
console.log('🎏 Streaming user already open');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const stream = await masto.v1.stream.streamUser();
|
|
|
|
ws.current = stream.ws;
|
2023-02-08 14:11:33 +03:00
|
|
|
ws.current.__id = Math.random();
|
|
|
|
console.log('🎏 Streaming user', ws.current);
|
2023-02-07 19:31:46 +03:00
|
|
|
|
|
|
|
stream.on('status.update', (status) => {
|
|
|
|
console.log(`🔄 Status ${status.id} updated`);
|
|
|
|
saveStatus(status, instance);
|
|
|
|
});
|
|
|
|
|
|
|
|
stream.on('delete', (statusID) => {
|
|
|
|
console.log(`❌ Status ${statusID} deleted`);
|
|
|
|
// delete states.statuses[statusID];
|
|
|
|
const s = getStatus(statusID, instance);
|
|
|
|
if (s) s._deleted = true;
|
|
|
|
});
|
|
|
|
|
2023-02-08 14:11:33 +03:00
|
|
|
stream.ws.onclose = () => {
|
|
|
|
console.log('🎏 Streaming user closed');
|
|
|
|
};
|
|
|
|
|
2023-02-07 19:31:46 +03:00
|
|
|
return stream;
|
2023-02-08 14:11:33 +03:00
|
|
|
};
|
2023-02-07 19:31:46 +03:00
|
|
|
useEffect(() => {
|
2023-02-08 14:11:33 +03:00
|
|
|
let stream;
|
|
|
|
(async () => {
|
|
|
|
stream = await streamUser();
|
|
|
|
})();
|
2023-02-07 19:31:46 +03:00
|
|
|
return () => {
|
2023-02-08 14:11:33 +03:00
|
|
|
if (stream) {
|
|
|
|
stream.ws.close();
|
2023-02-07 19:31:46 +03:00
|
|
|
ws.current = null;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}, []);
|
|
|
|
|
2023-02-03 16:08:08 +03:00
|
|
|
return (
|
|
|
|
<Timeline
|
2023-02-09 17:27:49 +03:00
|
|
|
title={title || 'Following'}
|
|
|
|
id={id || 'following'}
|
2023-02-03 16:08:08 +03:00
|
|
|
emptyText="Nothing to see here."
|
|
|
|
errorText="Unable to load posts."
|
2023-02-18 19:05:46 +03:00
|
|
|
instance={instance}
|
2023-02-03 16:08:08 +03:00
|
|
|
fetchItems={fetchHome}
|
2023-02-07 19:31:46 +03:00
|
|
|
checkForUpdates={checkForUpdates}
|
2023-02-06 18:50:00 +03:00
|
|
|
useItemID
|
2023-02-03 16:08:08 +03:00
|
|
|
boostsCarousel={snapStates.settings.boostsCarousel}
|
2023-02-15 05:49:36 +03:00
|
|
|
{...props}
|
2023-03-21 19:09:36 +03:00
|
|
|
allowFilters
|
2023-02-03 16:08:08 +03:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export default Following;
|