From e7327945e3a05684a05c53060314be81781468ba Mon Sep 17 00:00:00 2001 From: dm1sh Date: Mon, 31 Jul 2023 14:55:12 +0300 Subject: [PATCH] Improved loading handling --- front/src/components/CategoryPreview.tsx | 10 +++-- front/src/components/Points.tsx | 2 +- front/src/hooks/useFetch.ts | 54 ++++++++++++++++++------ front/src/pages/AddPage.tsx | 21 ++++----- front/src/pages/HomePage.tsx | 13 ++++-- front/src/pages/UserPage.tsx | 20 ++++++--- 6 files changed, 85 insertions(+), 35 deletions(-) diff --git a/front/src/components/CategoryPreview.tsx b/front/src/components/CategoryPreview.tsx index d72cb94..d711b61 100644 --- a/front/src/components/CategoryPreview.tsx +++ b/front/src/components/CategoryPreview.tsx @@ -1,7 +1,7 @@ import { StoriesPreview } from '.' import { UserCategoriesNames, UserCategory, composeUserCategoriesFilters } from '../assets/userCategories' import { useAnnouncements } from '../hooks/api' -import { gotError } from '../hooks/useFetch' +import { gotError, gotResponse } from '../hooks/useFetch' type CategoryPreviewProps = { category: UserCategory, @@ -15,8 +15,12 @@ function CategoryPreview({ category }: CategoryPreviewProps) {

{UserCategoriesNames[category]}

{gotError(announcements) ? (

{announcements.error}

- ) : (announcements.loading ? 'Загрузка...' : - + ) : ( + gotResponse(announcements) ? ( + + ) : ( + 'Загрузка...' + ) )} ) diff --git a/front/src/components/Points.tsx b/front/src/components/Points.tsx index 760524a..deebc85 100644 --- a/front/src/components/Points.tsx +++ b/front/src/components/Points.tsx @@ -3,7 +3,7 @@ import { CSSProperties } from 'react' import handStarsIcon from '../assets/handStars.svg' type PointsProps = { - points: number, + points: number | string, } const styles = { diff --git a/front/src/hooks/useFetch.ts b/front/src/hooks/useFetch.ts index f38f497..82a5106 100644 --- a/front/src/hooks/useFetch.ts +++ b/front/src/hooks/useFetch.ts @@ -3,29 +3,44 @@ import { useEffect, useState } from 'react' import useSend from './useSend' type UseFetchShared = { - loading: boolean, abort?: () => void, refetch: () => void, } type UseFetchSucced = { - error: null, data: T, + loading: false, + error: null, +} & UseFetchShared + +type UseFetchLoading = { + data: undefined, + loading: true, + error: null, } & UseFetchShared type UseFetchErrored = { - error: string, data: undefined, + loading: false, + error: string, } & UseFetchShared -type UseFetchReturn = UseFetchSucced | UseFetchErrored +type UseFetchReturn = UseFetchSucced | UseFetchLoading | UseFetchErrored const gotError = (res: UseFetchReturn): res is UseFetchErrored => ( typeof res.error === 'string' ) -const fallbackError = (res: UseFetchReturn) => ( - gotError(res) ? res.error : res.data +function fallbackError(res: UseFetchSucced | UseFetchErrored): T | string +function fallbackError(res: UseFetchReturn): T | string | undefined +function fallbackError(res: UseFetchReturn): T | string | undefined { + return ( + gotError(res) ? res.error : res.data + ) +} + +const gotResponse = (res: UseFetchReturn): res is UseFetchSucced | UseFetchErrored => ( + !res.loading ) function useFetch>( @@ -59,13 +74,28 @@ function useFetch>( useEffect(refetch, [doSend]) + if (loading === true) { + return { + data: undefined, + loading, + error: null, + refetch, + } + } + + if (error !== null) { + return { + data: undefined, + loading, + error, + refetch, + } + } + return { - ...( - error === null ? ({ - data: data!, error: null, - }) : ({ data: undefined, error }) - ), + data: data!, loading, + error, refetch, } } @@ -74,4 +104,4 @@ export type { UseFetchReturn } export default useFetch -export { gotError, fallbackError } +export { gotError, gotResponse, fallbackError } diff --git a/front/src/pages/AddPage.tsx b/front/src/pages/AddPage.tsx index 28159c2..72c82a5 100644 --- a/front/src/pages/AddPage.tsx +++ b/front/src/pages/AddPage.tsx @@ -8,7 +8,7 @@ import { ClickHandler, LocationMarker, TrashboxMarkers } from '../components' import { useAddAnnouncement, useTrashboxes } from '../hooks/api' import { categories, categoryNames } from '../assets/category' import { stations, lines, lineNames } from '../assets/metro' -import { fallbackError, gotError } from '../hooks/useFetch' +import { fallbackError, gotError, gotResponse } from '../hooks/useFetch' import { useOsmAddresses } from '../hooks/api' import CardLayout from '../components/CardLayout' @@ -89,17 +89,18 @@ function AddPage() { attribution='© OpenStreetMap contributors' url='https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png' /> - + />} + -

Адрес: {fallbackError(address)}

+

Адрес: {gotResponse(address) ? fallbackError(address) : 'Загрузка...'}

@@ -138,12 +139,8 @@ function AddPage() { Пункт сбора мусора
- {trashboxes.loading + {gotResponse(trashboxes) ? ( -
-

Загрузка...

-
- ) : ( gotError(trashboxes) ? (

) + ) : ( +

+

Загрузка...

+
) }
- {!gotError(trashboxes) && selectedTrashbox.index > -1 ? ( + {gotResponse(trashboxes) && !gotError(trashboxes) && selectedTrashbox.index > -1 ? (

Выбран пункт сбора мусора на { trashboxes.data[selectedTrashbox.index].Address } с категорией {selectedTrashbox.category}

diff --git a/front/src/pages/HomePage.tsx b/front/src/pages/HomePage.tsx index 9526a8c..ed09a45 100644 --- a/front/src/pages/HomePage.tsx +++ b/front/src/pages/HomePage.tsx @@ -26,16 +26,19 @@ function generateStories(announcements: Announcement[], refresh: () => void): St } function fallbackGenerateStories(announcements: UseFetchReturn) { - if (announcements.loading) + if (announcements.loading) { return fallbackStory() + } - if (gotError(announcements)) + if (gotError(announcements)) { return fallbackStory(announcements.error, true) + } const stories = generateStories(announcements.data, announcements.refetch) - if (stories.length === 0) + if (stories.length === 0) { return fallbackStory('Здесь пока пусто') + } return stories } @@ -43,7 +46,9 @@ function fallbackGenerateStories(announcements: UseFetchReturn) const fallbackStory = (text = '', isError = false): Story[] => [{ content: ({ action }) => { // eslint-disable-next-line react-hooks/rules-of-hooks - useEffect(() => { action('pause') }, [action]) + useEffect(() => { + action('pause') + }, [action]) return (
diff --git a/front/src/pages/UserPage.tsx b/front/src/pages/UserPage.tsx index 7ba9603..f3c4007 100644 --- a/front/src/pages/UserPage.tsx +++ b/front/src/pages/UserPage.tsx @@ -3,7 +3,7 @@ import { Container } from 'react-bootstrap' import { useUser } from '../hooks/api' import { userCategories } from '../assets/userCategories' import { BackHeader, CategoryPreview, Points, SignOut } from '../components' -import { gotError } from '../hooks/useFetch' +import { gotError, gotResponse } from '../hooks/useFetch' function UserPage() { const user = useUser() @@ -11,16 +11,26 @@ function UserPage() { return ( - + {userCategories.map(cat => ( ))}