mirror of
https://github.com/cheeaun/phanpy.git
synced 2024-12-21 20:14:30 +03:00
218 lines
6.2 KiB
JavaScript
218 lines
6.2 KiB
JavaScript
import { Link } from 'preact-router/match';
|
|
import { useEffect, useRef, useState } from 'preact/hooks';
|
|
import { InView } from 'react-intersection-observer';
|
|
import { useSnapshot } from 'valtio';
|
|
|
|
import Icon from '../components/icon';
|
|
import Loader from '../components/loader';
|
|
import Status from '../components/status';
|
|
import states from '../utils/states';
|
|
import store from '../utils/store';
|
|
|
|
const LIMIT = 20;
|
|
|
|
export default ({ hidden }) => {
|
|
const snapStates = useSnapshot(states);
|
|
const [uiState, setUIState] = useState('default');
|
|
const [showMore, setShowMore] = useState(false);
|
|
|
|
const statusIterator = useRef(
|
|
masto.timelines.getHomeIterable({
|
|
limit: LIMIT,
|
|
}),
|
|
).current;
|
|
async function fetchStatuses(firstLoad) {
|
|
const allStatuses = await statusIterator.next(
|
|
firstLoad
|
|
? {
|
|
limit: LIMIT,
|
|
}
|
|
: undefined,
|
|
);
|
|
if (allStatuses.value <= 0) {
|
|
return { done: true };
|
|
}
|
|
const homeValues = allStatuses.value.map((status) => {
|
|
states.statuses.set(status.id, status);
|
|
if (status.reblog) {
|
|
states.statuses.set(status.reblog.id, status.reblog);
|
|
}
|
|
return {
|
|
id: status.id,
|
|
reblog: status.reblog?.id,
|
|
reply: !!status.inReplyToAccountId,
|
|
};
|
|
});
|
|
if (firstLoad) {
|
|
states.home = homeValues;
|
|
} else {
|
|
states.home.push(...homeValues);
|
|
}
|
|
states.homeLastFetchTime = Date.now();
|
|
return allStatuses;
|
|
}
|
|
|
|
const loadStatuses = (firstLoad) => {
|
|
setUIState('loading');
|
|
(async () => {
|
|
try {
|
|
const { done } = await fetchStatuses(firstLoad);
|
|
setShowMore(!done);
|
|
setUIState('default');
|
|
} catch (e) {
|
|
console.warn(e);
|
|
setUIState('error');
|
|
}
|
|
})();
|
|
};
|
|
|
|
useEffect(() => {
|
|
loadStatuses(true);
|
|
}, []);
|
|
|
|
useEffect(() => {
|
|
const handleVisibilityChange = () => {
|
|
if (document.hidden) {
|
|
const timestamp = Date.now();
|
|
store.session.set('lastHidden', timestamp);
|
|
} else {
|
|
const timestamp = Date.now();
|
|
const lastHidden = store.session.get('lastHidden');
|
|
const diff = timestamp - lastHidden;
|
|
const diffMins = Math.round(diff / 1000 / 60);
|
|
if (diffMins > 1) {
|
|
console.log('visible', { lastHidden, diffMins });
|
|
setTimeout(() => {
|
|
loadStatuses(true);
|
|
states.homeNew = [];
|
|
}, 100);
|
|
}
|
|
}
|
|
};
|
|
document.addEventListener('visibilitychange', handleVisibilityChange);
|
|
return () => {
|
|
document.removeEventListener('visibilitychange', handleVisibilityChange);
|
|
};
|
|
}, []);
|
|
|
|
const scrollableRef = useRef();
|
|
|
|
return (
|
|
<div class="deck-container" hidden={hidden} ref={scrollableRef}>
|
|
<div class="timeline-deck deck">
|
|
<header
|
|
onClick={() => {
|
|
scrollableRef.current?.scrollTo({ top: 0, behavior: 'smooth' });
|
|
}}
|
|
>
|
|
<div class="header-side">
|
|
<button
|
|
type="button"
|
|
class="plain"
|
|
onClick={(e) => {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
states.showSettings = true;
|
|
}}
|
|
>
|
|
<Icon icon="gear" size="l" alt="Settings" />
|
|
</button>
|
|
</div>
|
|
<h1>Home</h1>
|
|
<div class="header-side">
|
|
<Loader hidden={uiState !== 'loading'} />{' '}
|
|
<a
|
|
href="#/notifications"
|
|
class={`button plain ${
|
|
snapStates.notificationsNew.length > 0 ? 'has-badge' : ''
|
|
}`}
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
}}
|
|
>
|
|
<Icon icon="notification" size="l" alt="Notifications" />
|
|
</a>
|
|
</div>
|
|
{snapStates.homeNew.length > 0 && (
|
|
<button
|
|
class="updates-button"
|
|
type="button"
|
|
onClick={() => {
|
|
const uniqueHomeNew = snapStates.homeNew.filter(
|
|
(status) => !states.home.some((s) => s.id === status.id),
|
|
);
|
|
states.home.unshift(...uniqueHomeNew);
|
|
loadStatuses(true);
|
|
states.homeNew = [];
|
|
|
|
scrollableRef.current?.scrollTo({
|
|
top: 0,
|
|
behavior: 'smooth',
|
|
});
|
|
}}
|
|
>
|
|
<Icon icon="arrow-up" /> New posts
|
|
</button>
|
|
)}
|
|
</header>
|
|
{snapStates.home.length ? (
|
|
<>
|
|
<ul class="timeline">
|
|
{snapStates.home.map(({ id: statusID, reblog }) => {
|
|
const actualStatusID = reblog || statusID;
|
|
return (
|
|
<li key={statusID}>
|
|
<Link
|
|
activeClassName="active"
|
|
class="status-link"
|
|
href={`#/s/${actualStatusID}`}
|
|
>
|
|
<Status statusID={statusID} />
|
|
</Link>
|
|
</li>
|
|
);
|
|
})}
|
|
{showMore && (
|
|
<>
|
|
<InView
|
|
as="li"
|
|
style={{
|
|
height: '20vh',
|
|
}}
|
|
onChange={(inView) => {
|
|
if (inView) loadStatuses();
|
|
}}
|
|
>
|
|
<Status skeleton />
|
|
</InView>
|
|
<li
|
|
style={{
|
|
height: '25vh',
|
|
}}
|
|
>
|
|
<Status skeleton />
|
|
</li>
|
|
</>
|
|
)}
|
|
</ul>
|
|
</>
|
|
) : (
|
|
<>
|
|
{uiState === 'loading' && (
|
|
<ul class="timeline">
|
|
{Array.from({ length: 5 }).map((_, i) => (
|
|
<li key={i}>
|
|
<Status skeleton />
|
|
</li>
|
|
))}
|
|
</ul>
|
|
)}
|
|
{uiState === 'error' && (
|
|
<p class="ui-state">Error loading statuses</p>
|
|
)}
|
|
</>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|