import { useEffect, useState, useCallback } from 'react'; import { useIntl, defineMessages } from 'react-intl'; import { fetchContext, completeContextRefresh, showPendingReplies, clearPendingReplies, } from 'flavours/glitch/actions/statuses'; import type { AsyncRefreshHeader } from 'flavours/glitch/api'; import { apiGetAsyncRefresh } from 'flavours/glitch/api/async_refreshes'; import { Alert } from 'flavours/glitch/components/alert'; import { ExitAnimationWrapper } from 'flavours/glitch/components/exit_animation_wrapper'; import { LoadingIndicator } from 'flavours/glitch/components/loading_indicator'; import { useAppSelector, useAppDispatch } from 'flavours/glitch/store'; const AnimatedAlert: React.FC< React.ComponentPropsWithoutRef & { withEntryDelay?: boolean } > = ({ isActive = false, withEntryDelay, ...props }) => ( {(delayedIsActive) => } ); const messages = defineMessages({ moreFound: { id: 'status.context.more_replies_found', defaultMessage: 'More replies found', }, show: { id: 'status.context.show', defaultMessage: 'Show', }, loadingInitial: { id: 'status.context.loading', defaultMessage: 'Loading', }, success: { id: 'status.context.loading_success', defaultMessage: 'New replies loaded', }, error: { id: 'status.context.loading_error', defaultMessage: "Couldn't load new replies", }, retry: { id: 'status.context.retry', defaultMessage: 'Retry', }, }); type LoadingState = 'idle' | 'more-available' | 'loading' | 'success' | 'error'; export const RefreshController: React.FC<{ statusId: string; }> = ({ statusId }) => { const dispatch = useAppDispatch(); const intl = useIntl(); const refreshHeader = useAppSelector( (state) => state.contexts.refreshing[statusId], ); const hasPendingReplies = useAppSelector( (state) => !!state.contexts.pendingReplies[statusId]?.length, ); const [partialLoadingState, setLoadingState] = useState( refreshHeader ? 'loading' : 'idle', ); const loadingState = hasPendingReplies ? 'more-available' : partialLoadingState; const [wasDismissed, setWasDismissed] = useState(false); const dismissPrompt = useCallback(() => { setWasDismissed(true); setLoadingState('idle'); dispatch(clearPendingReplies({ statusId })); }, [dispatch, statusId]); useEffect(() => { let timeoutId: ReturnType; const scheduleRefresh = ( refresh: AsyncRefreshHeader, iteration: number, ) => { timeoutId = setTimeout(() => { void apiGetAsyncRefresh(refresh.id).then((result) => { // At three scheduled refreshes, we consider the job // long-running and attempt to fetch any new replies so far const isLongRunning = iteration === 3; const { status, result_count } = result.async_refresh; // If the refresh status is not finished and not long-running, // we just schedule another refresh and exit if (status === 'running' && !isLongRunning) { scheduleRefresh(refresh, iteration + 1); return; } // If refresh status is finished, clear `refreshHeader` // (we don't want to do this if it's just a long-running job) if (status === 'finished') { dispatch(completeContextRefresh({ statusId })); } // Exit if there's nothing to fetch if (result_count === 0) { if (status === 'finished') { setLoadingState('idle'); } else { scheduleRefresh(refresh, iteration + 1); } 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`. If the fetch has // resulted in new pending replies, the `hasPendingReplies` // flag will switch the loading state to 'more-available' if (status === 'finished') { setLoadingState('idle'); } else { // Keep background fetch going if `isLongRunning` is true scheduleRefresh(refresh, iteration + 1); } }) .catch(() => { // Show an error if the fetch failed setLoadingState('error'); }); }); }, refresh.retry * 1000); }; // Initialise a refresh if (refreshHeader && !wasDismissed) { scheduleRefresh(refreshHeader, 1); setLoadingState('loading'); } return () => { clearTimeout(timeoutId); }; }, [dispatch, statusId, refreshHeader, wasDismissed]); useEffect(() => { // Hide success message after a short delay if (loadingState === 'success') { const timeoutId = setTimeout(() => { setLoadingState('idle'); }, 2500); return () => { clearTimeout(timeoutId); }; } return () => ''; }, [loadingState]); useEffect(() => { // Clear pending replies on unmount return () => { dispatch(clearPendingReplies({ statusId })); }; }, [dispatch, statusId]); const handleClick = useCallback(() => { dispatch(showPendingReplies({ statusId })); setLoadingState('success'); }, [dispatch, statusId]); if (loadingState === 'loading') { return (
); } return (
); };