2023-01-28 13:52:18 +03:00
|
|
|
// EXPERIMENTAL: This is a work in progress and may not work as expected.
|
2023-02-05 19:17:19 +03:00
|
|
|
import { useRef } from 'preact/hooks';
|
2023-02-17 20:40:37 +03:00
|
|
|
import { useNavigate, useParams } from 'react-router-dom';
|
2023-01-28 13:52:18 +03:00
|
|
|
|
2023-02-17 20:40:37 +03:00
|
|
|
import Icon from '../components/icon';
|
2023-01-28 13:52:18 +03:00
|
|
|
import Timeline from '../components/timeline';
|
2023-02-05 19:17:19 +03:00
|
|
|
import { api } from '../utils/api';
|
2023-02-03 16:08:08 +03:00
|
|
|
import useTitle from '../utils/useTitle';
|
2023-01-28 13:52:18 +03:00
|
|
|
|
|
|
|
const LIMIT = 20;
|
|
|
|
|
2023-02-18 15:48:24 +03:00
|
|
|
function Public({ local, ...props }) {
|
2023-02-06 15:17:07 +03:00
|
|
|
const isLocal = !!local;
|
2023-02-06 11:35:03 +03:00
|
|
|
const params = useParams();
|
2023-02-18 15:48:24 +03:00
|
|
|
const { masto, instance } = api({
|
|
|
|
instance: props?.instance || params.instance,
|
|
|
|
});
|
2023-02-11 13:19:23 +03:00
|
|
|
const title = `${isLocal ? 'Local' : 'Federated'} timeline (${instance})`;
|
|
|
|
useTitle(title, isLocal ? `/:instance?/p/l` : `/:instance?/p`);
|
2023-02-17 20:40:37 +03:00
|
|
|
const navigate = useNavigate();
|
2023-02-11 11:28:03 +03:00
|
|
|
const latestItem = useRef();
|
2023-02-05 19:17:19 +03:00
|
|
|
|
|
|
|
const publicIterator = useRef();
|
2023-01-28 13:52:18 +03:00
|
|
|
async function fetchPublic(firstLoad) {
|
2023-02-05 19:17:19 +03:00
|
|
|
if (firstLoad || !publicIterator.current) {
|
|
|
|
publicIterator.current = masto.v1.timelines.listPublic({
|
|
|
|
limit: LIMIT,
|
|
|
|
local: isLocal,
|
|
|
|
});
|
2023-01-28 13:52:18 +03:00
|
|
|
}
|
2023-02-11 11:28:03 +03:00
|
|
|
const results = await publicIterator.current.next();
|
|
|
|
const { value } = results;
|
|
|
|
if (value?.length) {
|
|
|
|
if (firstLoad) {
|
|
|
|
latestItem.current = value[0].id;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
|
|
|
async function checkForUpdates() {
|
|
|
|
try {
|
|
|
|
const results = await masto.v1.timelines
|
|
|
|
.listPublic({
|
|
|
|
limit: 1,
|
|
|
|
local: isLocal,
|
|
|
|
since_id: latestItem.current,
|
|
|
|
})
|
|
|
|
.next();
|
|
|
|
const { value } = results;
|
|
|
|
if (value?.length) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
} catch (e) {
|
|
|
|
return false;
|
|
|
|
}
|
2023-01-28 13:52:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Timeline
|
|
|
|
key={instance + isLocal}
|
2023-02-03 16:08:08 +03:00
|
|
|
title={title}
|
2023-02-05 19:17:19 +03:00
|
|
|
titleComponent={
|
|
|
|
<h1 class="header-account">
|
2023-02-11 13:19:23 +03:00
|
|
|
<b>{isLocal ? 'Local timeline' : 'Federated timeline'}</b>
|
|
|
|
<div>{instance}</div>
|
2023-02-05 19:17:19 +03:00
|
|
|
</h1>
|
|
|
|
}
|
2023-01-28 13:52:18 +03:00
|
|
|
id="public"
|
2023-02-05 19:17:19 +03:00
|
|
|
instance={instance}
|
2023-01-28 13:52:18 +03:00
|
|
|
emptyText="No one has posted anything yet."
|
|
|
|
errorText="Unable to load posts"
|
|
|
|
fetchItems={fetchPublic}
|
2023-02-11 11:28:03 +03:00
|
|
|
checkForUpdates={checkForUpdates}
|
2023-02-11 13:21:39 +03:00
|
|
|
headerStart={<></>}
|
2023-02-17 20:40:37 +03:00
|
|
|
headerEnd={
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
class="plain"
|
|
|
|
onClick={() => {
|
2023-02-22 04:46:50 +03:00
|
|
|
let newInstance = prompt(
|
2023-02-17 20:40:37 +03:00
|
|
|
'Enter a new instance e.g. "mastodon.social"',
|
|
|
|
);
|
|
|
|
if (!/\./.test(newInstance)) {
|
2023-02-18 18:29:17 +03:00
|
|
|
if (newInstance) alert('Invalid instance');
|
2023-02-17 20:40:37 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (newInstance) {
|
2023-02-22 04:46:50 +03:00
|
|
|
newInstance = newInstance.toLowerCase().trim();
|
2023-02-17 20:40:37 +03:00
|
|
|
navigate(isLocal ? `/${newInstance}/p/l` : `/${newInstance}/p`);
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<Icon icon="transfer" alt="Switch instance" />
|
|
|
|
</button>
|
|
|
|
}
|
2023-01-28 13:52:18 +03:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export default Public;
|