mirror of
https://github.com/cheeaun/phanpy.git
synced 2025-01-03 16:17:19 +03:00
145 lines
3.9 KiB
React
145 lines
3.9 KiB
React
|
import { useEffect, useRef, useState } from 'preact/hooks';
|
||
|
|
||
|
import Icon from '../components/Icon';
|
||
|
import Link from '../components/link';
|
||
|
import Loader from '../components/Loader';
|
||
|
import Status from '../components/status';
|
||
|
import { saveStatus } from '../utils/states';
|
||
|
import useTitle from '../utils/useTitle';
|
||
|
|
||
|
const LIMIT = 40;
|
||
|
|
||
|
function Bookmarks() {
|
||
|
useTitle('Bookmarks');
|
||
|
const [bookmarks, setBookmarks] = useState([]);
|
||
|
const [uiState, setUIState] = useState('default');
|
||
|
const [showMore, setShowMore] = useState(false);
|
||
|
|
||
|
const bookmarksIterator = useRef(masto.v1.bookmarks.list({ limit: LIMIT }));
|
||
|
async function fetchBookmarks(firstLoad) {
|
||
|
console.log('fetchBookmarks', firstLoad);
|
||
|
if (firstLoad) {
|
||
|
bookmarksIterator.current = masto.v1.bookmarks.list({ limit: LIMIT });
|
||
|
}
|
||
|
const allBookmarks = await bookmarksIterator.current.next();
|
||
|
if (allBookmarks.value?.length) {
|
||
|
const bookmarksValue = allBookmarks.value.map((status) => {
|
||
|
saveStatus(status, {
|
||
|
skipThreading: true,
|
||
|
override: false,
|
||
|
});
|
||
|
return status;
|
||
|
});
|
||
|
if (firstLoad) {
|
||
|
setBookmarks(bookmarksValue);
|
||
|
} else {
|
||
|
setBookmarks([...bookmarks, ...bookmarksValue]);
|
||
|
}
|
||
|
}
|
||
|
return allBookmarks;
|
||
|
}
|
||
|
|
||
|
const loadBookmarks = (firstLoad) => {
|
||
|
setUIState('loading');
|
||
|
(async () => {
|
||
|
try {
|
||
|
console.log('loadBookmarks', firstLoad);
|
||
|
const { done } = await fetchBookmarks(firstLoad);
|
||
|
console.log('loadBookmarks', firstLoad);
|
||
|
setShowMore(!done);
|
||
|
setUIState('default');
|
||
|
} catch (e) {
|
||
|
console.error(e);
|
||
|
setUIState('error');
|
||
|
}
|
||
|
})();
|
||
|
};
|
||
|
|
||
|
useEffect(() => {
|
||
|
loadBookmarks(true);
|
||
|
}, []);
|
||
|
|
||
|
const scrollableRef = useRef(null);
|
||
|
|
||
|
return (
|
||
|
<div
|
||
|
id="bookmarks-page"
|
||
|
class="deck-container"
|
||
|
ref={scrollableRef}
|
||
|
tabIndex="-1"
|
||
|
>
|
||
|
<div class="timeline-deck deck">
|
||
|
<header
|
||
|
onClick={(e) => {
|
||
|
if (e.target === e.currentTarget) {
|
||
|
scrollableRef.current?.scrollTo({
|
||
|
top: 0,
|
||
|
behavior: 'smooth',
|
||
|
});
|
||
|
}
|
||
|
}}
|
||
|
>
|
||
|
<div class="header-side">
|
||
|
<Link to="/" class="button plain">
|
||
|
<Icon icon="home" size="l" />
|
||
|
</Link>
|
||
|
</div>
|
||
|
<h1>Bookmarks</h1>
|
||
|
<div class="header-side"></div>{' '}
|
||
|
</header>
|
||
|
{!!bookmarks.length ? (
|
||
|
<>
|
||
|
<ul class="timeline">
|
||
|
{bookmarks.map((status) => (
|
||
|
<li key={`bookmark-${status.id}`}>
|
||
|
<Link class="status-link" to={`/s/${status.id}`}>
|
||
|
<Status status={status} />
|
||
|
</Link>
|
||
|
</li>
|
||
|
))}
|
||
|
</ul>
|
||
|
|
||
|
{showMore && (
|
||
|
<button
|
||
|
type="button"
|
||
|
class="plain block"
|
||
|
disabled={uiState === 'loading'}
|
||
|
onClick={() => loadBookmarks()}
|
||
|
style={{ marginBlockEnd: '6em' }}
|
||
|
>
|
||
|
{uiState === 'loading' ? <Loader /> : <>Show more…</>}
|
||
|
</button>
|
||
|
)}
|
||
|
</>
|
||
|
) : (
|
||
|
uiState !== 'loading' && (
|
||
|
<p class="ui-state">No bookmarks yet. Go bookmark something!</p>
|
||
|
)
|
||
|
)}
|
||
|
{uiState === 'loading' ? (
|
||
|
<div class="ui-state">
|
||
|
<Loader />
|
||
|
</div>
|
||
|
) : uiState === 'error' ? (
|
||
|
<p class="ui-state">
|
||
|
Unable to load bookmarks.
|
||
|
<br />
|
||
|
<br />
|
||
|
<button
|
||
|
class="button plain"
|
||
|
onClick={() => loadBookmarks(!bookmarks.length)}
|
||
|
>
|
||
|
Try again
|
||
|
</button>
|
||
|
</p>
|
||
|
) : (
|
||
|
bookmarks.length &&
|
||
|
!showMore && <p class="ui-state insignificant">The end.</p>
|
||
|
)}
|
||
|
</div>
|
||
|
</div>
|
||
|
);
|
||
|
}
|
||
|
|
||
|
export default Bookmarks;
|