Improved loading handling

This commit is contained in:
Dmitriy Shishkov 2023-07-31 14:55:12 +03:00
parent 9b35a54ae9
commit e7327945e3
Signed by: dm1sh
GPG Key ID: 027994B0AA357688
6 changed files with 85 additions and 35 deletions

View File

@ -1,7 +1,7 @@
import { StoriesPreview } from '.' import { StoriesPreview } from '.'
import { UserCategoriesNames, UserCategory, composeUserCategoriesFilters } from '../assets/userCategories' import { UserCategoriesNames, UserCategory, composeUserCategoriesFilters } from '../assets/userCategories'
import { useAnnouncements } from '../hooks/api' import { useAnnouncements } from '../hooks/api'
import { gotError } from '../hooks/useFetch' import { gotError, gotResponse } from '../hooks/useFetch'
type CategoryPreviewProps = { type CategoryPreviewProps = {
category: UserCategory, category: UserCategory,
@ -15,8 +15,12 @@ function CategoryPreview({ category }: CategoryPreviewProps) {
<h4 className='fw-bold'>{UserCategoriesNames[category]}</h4> <h4 className='fw-bold'>{UserCategoriesNames[category]}</h4>
{gotError(announcements) ? ( {gotError(announcements) ? (
<p className='text-danger'>{announcements.error}</p> <p className='text-danger'>{announcements.error}</p>
) : (announcements.loading ? 'Загрузка...' : ) : (
<StoriesPreview announcements={announcements.data} category={category} /> gotResponse(announcements) ? (
<StoriesPreview announcements={announcements.data} category={category} />
) : (
'Загрузка...'
)
)} )}
</section> </section>
) )

View File

@ -3,7 +3,7 @@ import { CSSProperties } from 'react'
import handStarsIcon from '../assets/handStars.svg' import handStarsIcon from '../assets/handStars.svg'
type PointsProps = { type PointsProps = {
points: number, points: number | string,
} }
const styles = { const styles = {

View File

@ -3,29 +3,44 @@ import { useEffect, useState } from 'react'
import useSend from './useSend' import useSend from './useSend'
type UseFetchShared = { type UseFetchShared = {
loading: boolean,
abort?: () => void, abort?: () => void,
refetch: () => void, refetch: () => void,
} }
type UseFetchSucced<T> = { type UseFetchSucced<T> = {
error: null,
data: T, data: T,
loading: false,
error: null,
} & UseFetchShared
type UseFetchLoading = {
data: undefined,
loading: true,
error: null,
} & UseFetchShared } & UseFetchShared
type UseFetchErrored = { type UseFetchErrored = {
error: string,
data: undefined, data: undefined,
loading: false,
error: string,
} & UseFetchShared } & UseFetchShared
type UseFetchReturn<T> = UseFetchSucced<T> | UseFetchErrored type UseFetchReturn<T> = UseFetchSucced<T> | UseFetchLoading | UseFetchErrored
const gotError = <T>(res: UseFetchReturn<T>): res is UseFetchErrored => ( const gotError = <T>(res: UseFetchReturn<T>): res is UseFetchErrored => (
typeof res.error === 'string' typeof res.error === 'string'
) )
const fallbackError = <T>(res: UseFetchReturn<T>) => ( function fallbackError<T>(res: UseFetchSucced<T> | UseFetchErrored): T | string
gotError(res) ? res.error : res.data function fallbackError<T>(res: UseFetchReturn<T>): T | string | undefined
function fallbackError<T>(res: UseFetchReturn<T>): T | string | undefined {
return (
gotError(res) ? res.error : res.data
)
}
const gotResponse = <T>(res: UseFetchReturn<T>): res is UseFetchSucced<T> | UseFetchErrored => (
!res.loading
) )
function useFetch<R, T extends NonNullable<unknown>>( function useFetch<R, T extends NonNullable<unknown>>(
@ -59,13 +74,28 @@ function useFetch<R, T extends NonNullable<unknown>>(
useEffect(refetch, [doSend]) useEffect(refetch, [doSend])
if (loading === true) {
return {
data: undefined,
loading,
error: null,
refetch,
}
}
if (error !== null) {
return {
data: undefined,
loading,
error,
refetch,
}
}
return { return {
...( data: data!,
error === null ? ({
data: data!, error: null,
}) : ({ data: undefined, error })
),
loading, loading,
error,
refetch, refetch,
} }
} }
@ -74,4 +104,4 @@ export type { UseFetchReturn }
export default useFetch export default useFetch
export { gotError, fallbackError } export { gotError, gotResponse, fallbackError }

View File

@ -8,7 +8,7 @@ import { ClickHandler, LocationMarker, TrashboxMarkers } from '../components'
import { useAddAnnouncement, useTrashboxes } from '../hooks/api' import { useAddAnnouncement, useTrashboxes } from '../hooks/api'
import { categories, categoryNames } from '../assets/category' import { categories, categoryNames } from '../assets/category'
import { stations, lines, lineNames } from '../assets/metro' 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 { useOsmAddresses } from '../hooks/api'
import CardLayout from '../components/CardLayout' import CardLayout from '../components/CardLayout'
@ -89,17 +89,18 @@ function AddPage() {
attribution='&copy; <a href="https://www.openstreetmap.org/copyright">OpenStreetMap</a> contributors' attribution='&copy; <a href="https://www.openstreetmap.org/copyright">OpenStreetMap</a> contributors'
url='https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png' url='https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png'
/> />
<LocationMarker {gotResponse(address) && <LocationMarker
address={fallbackError(address)} address={fallbackError(address)}
position={addressPosition} position={addressPosition}
setPosition={setAddressPosition} setPosition={setAddressPosition}
/> />}
<ClickHandler <ClickHandler
setPosition={setAddressPosition} setPosition={setAddressPosition}
/> />
</MapContainer> </MapContainer>
</div> </div>
<p>Адрес: {fallbackError(address)}</p> <p>Адрес: {gotResponse(address) ? fallbackError(address) : 'Загрузка...'}</p>
</Form.Group> </Form.Group>
<Form.Group className='mb-3' controlId='description'> <Form.Group className='mb-3' controlId='description'>
@ -138,12 +139,8 @@ function AddPage() {
<Form.Group className='mb-3' controlId='trashbox'> <Form.Group className='mb-3' controlId='trashbox'>
<Form.Label>Пункт сбора мусора</Form.Label> <Form.Label>Пункт сбора мусора</Form.Label>
<div className='mb-3'> <div className='mb-3'>
{trashboxes.loading {gotResponse(trashboxes)
? ( ? (
<div style={styles.map}>
<p>Загрузка...</p>
</div>
) : (
gotError(trashboxes) ? ( gotError(trashboxes) ? (
<p <p
style={styles.map} style={styles.map}
@ -167,10 +164,14 @@ function AddPage() {
/> />
</MapContainer> </MapContainer>
) )
) : (
<div style={styles.map}>
<p>Загрузка...</p>
</div>
) )
} }
</div> </div>
{!gotError(trashboxes) && selectedTrashbox.index > -1 ? ( {gotResponse(trashboxes) && !gotError(trashboxes) && selectedTrashbox.index > -1 ? (
<p>Выбран пункт сбора мусора на { <p>Выбран пункт сбора мусора на {
trashboxes.data[selectedTrashbox.index].Address trashboxes.data[selectedTrashbox.index].Address
} с категорией {selectedTrashbox.category}</p> } с категорией {selectedTrashbox.category}</p>

View File

@ -26,16 +26,19 @@ function generateStories(announcements: Announcement[], refresh: () => void): St
} }
function fallbackGenerateStories(announcements: UseFetchReturn<Announcement[]>) { function fallbackGenerateStories(announcements: UseFetchReturn<Announcement[]>) {
if (announcements.loading) if (announcements.loading) {
return fallbackStory() return fallbackStory()
}
if (gotError(announcements)) if (gotError(announcements)) {
return fallbackStory(announcements.error, true) return fallbackStory(announcements.error, true)
}
const stories = generateStories(announcements.data, announcements.refetch) const stories = generateStories(announcements.data, announcements.refetch)
if (stories.length === 0) if (stories.length === 0) {
return fallbackStory('Здесь пока пусто') return fallbackStory('Здесь пока пусто')
}
return stories return stories
} }
@ -43,7 +46,9 @@ function fallbackGenerateStories(announcements: UseFetchReturn<Announcement[]>)
const fallbackStory = (text = '', isError = false): Story[] => [{ const fallbackStory = (text = '', isError = false): Story[] => [{
content: ({ action }) => { content: ({ action }) => {
// eslint-disable-next-line react-hooks/rules-of-hooks // eslint-disable-next-line react-hooks/rules-of-hooks
useEffect(() => { action('pause') }, [action]) useEffect(() => {
action('pause')
}, [action])
return ( return (
<div style={styles.center} className={isError ? 'text-danger' : ''}> <div style={styles.center} className={isError ? 'text-danger' : ''}>

View File

@ -3,7 +3,7 @@ import { Container } from 'react-bootstrap'
import { useUser } from '../hooks/api' import { useUser } from '../hooks/api'
import { userCategories } from '../assets/userCategories' import { userCategories } from '../assets/userCategories'
import { BackHeader, CategoryPreview, Points, SignOut } from '../components' import { BackHeader, CategoryPreview, Points, SignOut } from '../components'
import { gotError } from '../hooks/useFetch' import { gotError, gotResponse } from '../hooks/useFetch'
function UserPage() { function UserPage() {
const user = useUser() const user = useUser()
@ -11,16 +11,26 @@ function UserPage() {
return ( return (
<Container style={{ maxWidth: 'calc(100vh*9/16)' }}> <Container style={{ maxWidth: 'calc(100vh*9/16)' }}>
<BackHeader text={ <BackHeader text={
gotError(user) ? ( gotResponse(user) ? (
user.error gotError(user) ? (
user.error
) : (
`${user.data.name}, с нами с ${new Date(user.data.regDate).toLocaleDateString('ru')}`
)
) : ( ) : (
`${user.data.name}, с нами с ${new Date(user.data.regDate).toLocaleDateString('ru')}` 'Загрузка...'
) )
}> }>
<SignOut /> <SignOut />
</BackHeader> </BackHeader>
<Points points={gotError(user) ? -1 : user.data?.points} /> <Points points={
gotResponse(user) ? (
gotError(user) ? -1 : user.data?.points
) : (
'Загрузка...'
)
} />
{userCategories.map(cat => ( {userCategories.map(cat => (
<CategoryPreview key={cat} category={cat} /> <CategoryPreview key={cat} category={cat} />
))} ))}