Improved loading handling
This commit is contained in:
parent
9b35a54ae9
commit
e7327945e3
@ -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>
|
||||||
)
|
)
|
||||||
|
@ -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 = {
|
||||||
|
@ -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 }
|
||||||
|
@ -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='© <a href="https://www.openstreetmap.org/copyright">OpenStreetMap</a> contributors'
|
attribution='© <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>
|
||||||
|
@ -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' : ''}>
|
||||||
|
@ -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} />
|
||||||
))}
|
))}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user