Refactor refresh controller to handle pending replies

This commit is contained in:
diondiondion 2025-10-02 14:56:01 +02:00
parent b12ad6a3c9
commit 381116366a

View File

@ -5,6 +5,8 @@ import { useIntl, defineMessages } from 'react-intl';
import { import {
fetchContext, fetchContext,
completeContextRefresh, completeContextRefresh,
showPendingReplies,
clearPendingReplies,
} from 'mastodon/actions/statuses'; } from 'mastodon/actions/statuses';
import type { AsyncRefreshHeader } from 'mastodon/api'; import type { AsyncRefreshHeader } from 'mastodon/api';
import { apiGetAsyncRefresh } from 'mastodon/api/async_refreshes'; import { apiGetAsyncRefresh } from 'mastodon/api/async_refreshes';
@ -34,10 +36,6 @@ const messages = defineMessages({
id: 'status.context.loading', id: 'status.context.loading',
defaultMessage: 'Loading', defaultMessage: 'Loading',
}, },
loadingMore: {
id: 'status.context.loading_more',
defaultMessage: 'Loading more replies',
},
success: { success: {
id: 'status.context.loading_success', id: 'status.context.loading_success',
defaultMessage: 'All replies loaded', defaultMessage: 'All replies loaded',
@ -52,36 +50,33 @@ const messages = defineMessages({
}, },
}); });
type LoadingState = type LoadingState = 'idle' | 'more-available' | 'loading' | 'success' | 'error';
| 'idle'
| 'more-available'
| 'loading-initial'
| 'loading-more'
| 'success'
| 'error';
export const RefreshController: React.FC<{ export const RefreshController: React.FC<{
statusId: string; statusId: string;
}> = ({ statusId }) => { }> = ({ statusId }) => {
const refresh = useAppSelector(
(state) => state.contexts.refreshing[statusId],
);
const currentReplyCount = useAppSelector(
(state) => state.contexts.replies[statusId]?.length ?? 0,
);
const autoRefresh = !currentReplyCount;
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const intl = useIntl(); const intl = useIntl();
const [loadingState, setLoadingState] = useState<LoadingState>( const refreshHeader = useAppSelector(
refresh && autoRefresh ? 'loading-initial' : 'idle', (state) => state.contexts.refreshing[statusId],
); );
const hasPendingReplies = useAppSelector(
(state) => !!state.contexts.pendingReplies[statusId]?.length,
);
const [partialLoadingState, setLoadingState] = useState<LoadingState>(
refreshHeader ? 'loading' : 'idle',
);
const loadingState = hasPendingReplies
? 'more-available'
: partialLoadingState;
const [wasDismissed, setWasDismissed] = useState(false); const [wasDismissed, setWasDismissed] = useState(false);
const dismissPrompt = useCallback(() => { const dismissPrompt = useCallback(() => {
setWasDismissed(true); setWasDismissed(true);
setLoadingState('idle'); setLoadingState('idle');
}, []); dispatch(clearPendingReplies({ statusId }));
}, [dispatch, statusId]);
useEffect(() => { useEffect(() => {
let timeoutId: ReturnType<typeof setTimeout>; let timeoutId: ReturnType<typeof setTimeout>;
@ -89,36 +84,51 @@ export const RefreshController: React.FC<{
const scheduleRefresh = (refresh: AsyncRefreshHeader) => { const scheduleRefresh = (refresh: AsyncRefreshHeader) => {
timeoutId = setTimeout(() => { timeoutId = setTimeout(() => {
void apiGetAsyncRefresh(refresh.id).then((result) => { void apiGetAsyncRefresh(refresh.id).then((result) => {
if (result.async_refresh.status === 'finished') { // If the refresh status is not finished,
dispatch(completeContextRefresh({ statusId })); // schedule another refresh and exit
if (result.async_refresh.status !== 'finished') {
if (result.async_refresh.result_count > 0) {
if (autoRefresh) {
void dispatch(fetchContext({ statusId })).then(() => {
setLoadingState('idle');
});
} else {
setLoadingState('more-available');
}
} else {
setLoadingState('idle');
}
} else {
scheduleRefresh(refresh); scheduleRefresh(refresh);
return;
} }
// Refresh status is finished. The action below will clear `refreshHeader`
dispatch(completeContextRefresh({ statusId }));
// Exit if there's nothing to fetch
if (result.async_refresh.result_count === 0) {
setLoadingState('idle');
return;
}
// A positive result count means there _might_ be new replies,
// so we fetch the context in the background to check if there
// are any new replies.
// If so, they will populate `contexts.pendingReplies[statusId]`
void dispatch(fetchContext({ statusId, prefetchOnly: true }))
.then(() => {
// Reset loading state to `idle` but if the fetch
// has resulted in new pending replies, the `hasPendingReplies`
// flag will switch the loading state to 'more-available'
setLoadingState('idle');
})
.catch(() => {
// Show an error if the fetch failed
setLoadingState('error');
});
}); });
}, refresh.retry * 1000); }, refresh.retry * 1000);
}; };
if (refresh && !wasDismissed) { // Initialise a refresh
scheduleRefresh(refresh); if (refreshHeader && !wasDismissed) {
setLoadingState('loading-initial'); scheduleRefresh(refreshHeader);
setLoadingState('loading');
} }
return () => { return () => {
clearTimeout(timeoutId); clearTimeout(timeoutId);
}; };
}, [dispatch, statusId, refresh, autoRefresh, wasDismissed]); }, [dispatch, statusId, refreshHeader, wasDismissed]);
useEffect(() => { useEffect(() => {
// Hide success message after a short delay // Hide success message after a short delay
@ -134,20 +144,19 @@ export const RefreshController: React.FC<{
return () => ''; return () => '';
}, [loadingState]); }, [loadingState]);
const handleClick = useCallback(() => { useEffect(() => {
setLoadingState('loading-more'); // Clear pending replies on unmount
return () => {
dispatch(fetchContext({ statusId })) dispatch(clearPendingReplies({ statusId }));
.then(() => { };
setLoadingState('success');
return '';
})
.catch(() => {
setLoadingState('error');
});
}, [dispatch, statusId]); }, [dispatch, statusId]);
if (loadingState === 'loading-initial') { const handleClick = useCallback(() => {
dispatch(showPendingReplies({ statusId }));
setLoadingState('success');
}, [dispatch, statusId]);
if (loadingState === 'loading') {
return ( return (
<div <div
className='load-more load-gap' className='load-more load-gap'
@ -170,13 +179,6 @@ export const RefreshController: React.FC<{
onDismiss={dismissPrompt} onDismiss={dismissPrompt}
animateFrom='below' animateFrom='below'
/> />
<AnimatedAlert
isLoading
withEntryDelay
isActive={loadingState === 'loading-more'}
message={intl.formatMessage(messages.loadingMore)}
animateFrom='below'
/>
<AnimatedAlert <AnimatedAlert
withEntryDelay withEntryDelay
isActive={loadingState === 'error'} isActive={loadingState === 'error'}