diff --git a/front/src/hooks/api/useSignIn.ts b/front/src/hooks/api/useSignIn.ts index 7cf7e32..06a5dda 100644 --- a/front/src/hooks/api/useSignIn.ts +++ b/front/src/hooks/api/useSignIn.ts @@ -20,7 +20,7 @@ function useSignIn() { body: formData, }) - if (token !== undefined) { + if (token !== null && token !== undefined) { setToken(token) return true diff --git a/front/src/hooks/useFetch.ts b/front/src/hooks/useFetch.ts index c0e9319..bcac6c5 100644 --- a/front/src/hooks/useFetch.ts +++ b/front/src/hooks/useFetch.ts @@ -20,7 +20,7 @@ type UseFetchLoading = { } & UseFetchShared type UseFetchErrored = { - data: undefined, + data: null, loading: false, error: string, } & UseFetchShared @@ -32,8 +32,7 @@ const gotError = (res: UseFetchReturn): res is UseFetchErrored => ( ) function fallbackError(res: UseFetchSucced | UseFetchErrored): T | string -function fallbackError(res: UseFetchReturn): T | string | undefined -function fallbackError(res: UseFetchReturn): T | string | undefined { +function fallbackError(res: UseFetchReturn): T | string | null | undefined { return ( gotError(res) ? res.error : res.data ) @@ -62,7 +61,6 @@ function useFetch>( needAuth, guardResponse, processResponse, - true, params, ) @@ -70,10 +68,14 @@ function useFetch>( setFetchLoading(true) doSend().then( data => { - if (data !== undefined) { + if (data !== undefined && data !== null) { setData(data) + console.log('Got data', data) + } + + if (data !== undefined) { + setFetchLoading(false) } - setFetchLoading(false) } ).catch( // must never occur err => import.meta.env.DEV && console.error('Failed to do fetch request', err) @@ -93,7 +95,7 @@ function useFetch>( if (error !== null) { return { - data: undefined, + data: null, loading: fetchLoading, error, refetch, diff --git a/front/src/hooks/useSend.ts b/front/src/hooks/useSend.ts index 1742ac5..53eec3a 100644 --- a/front/src/hooks/useSend.ts +++ b/front/src/hooks/useSend.ts @@ -2,7 +2,7 @@ import { useCallback, useEffect, useRef, useState } from 'react' import { useNavigate } from 'react-router-dom' import { getToken } from '../utils/auth' -import { handleHTTPErrors, isAborted } from '../utils' +import { AbortError, handleHTTPErrors, isAborted } from '../utils' function useSend>( url: string, @@ -10,17 +10,19 @@ function useSend>( needAuth: boolean, guardResponse: (data: unknown) => data is R, processResponse: (data: R) => T, - startWithLoading = false, defaultParams?: Omit, ) { - const [loading, setLoading] = useState(startWithLoading) + const [loading, setLoading] = useState(false) const [error, setError] = useState(null) const navigate = useNavigate() const abortControllerRef = useRef() - useEffect(() => () => abortControllerRef.current?.abort(), []) + useEffect(() => () => { + const reason = new AbortError('unmount') + abortControllerRef.current?.abort(reason) + }, []) /** Don't use in useEffect. If you need request result, go with useFetch instead */ const doSend = useCallback(async (urlProps?: Record, params?: Omit) => { @@ -28,7 +30,8 @@ function useSend>( setError(null) if (abortControllerRef.current) { - abortControllerRef.current.abort() + const reason = new AbortError('resent') + abortControllerRef.current.abort(reason) } const abortController = new AbortController() @@ -45,7 +48,7 @@ function useSend>( if (token === null) { navigate('/login') - return undefined + return null } headers.append('Authorization', `Bearer ${token}`) @@ -73,21 +76,33 @@ function useSend>( return processResponse(data) } catch (err) { - if (err instanceof Error && !isAborted(err)) { - if (err instanceof TypeError) { - setError('Ошибка сети') - } else { - setError(err.message) - } + if (err instanceof Error) { + if (isAborted(err)) { + if (err.message !== 'resent') { + setLoading(false) + } - if (import.meta.env.DEV) { - console.error(url, params, err) + if (err.fallback !== undefined) { + return err.fallback + } + + return undefined + } else { + if (err instanceof TypeError) { + setError('Ошибка сети') + } else { + setError(err.message) + } + + if (import.meta.env.DEV) { + console.error(url, params, err) + } + + setLoading(false) } } - setLoading(false) - - return undefined + return null } }, [defaultParams, needAuth, navigate, url, method, guardResponse, processResponse]) diff --git a/front/src/hooks/useSendButtonCaption.ts b/front/src/hooks/useSendButtonCaption.ts index 4ab2f81..55e1fa0 100644 --- a/front/src/hooks/useSendButtonCaption.ts +++ b/front/src/hooks/useSendButtonCaption.ts @@ -11,8 +11,8 @@ function useSendButtonCaption( const [disabled, setDisabled] = useState(false) const [title, setTitle] = useState(initial) - const update = useCallback(>(data: T | undefined) => { - if (data !== undefined) { + const update = useCallback(>(data: T | null | undefined) => { + if (data !== undefined) { // not loading setCaption(result) setTitle('Отправить ещё раз') diff --git a/front/src/utils/index.ts b/front/src/utils/index.ts index e173552..4eda3eb 100644 --- a/front/src/utils/index.ts +++ b/front/src/utils/index.ts @@ -1,7 +1,21 @@ -const isAborted = (err: Error) => ( +const isAborted = (err: Error): err is AbortError => ( err.name === 'AbortError' ) +type AbortErrorMessage = 'resent' | 'unmount' | 'cancel' + +class AbortError extends DOMException { + readonly fallback: T | undefined + message: AbortErrorMessage + + constructor(message: AbortErrorMessage, fallback?: T) { + super(message, 'AbortError') + this.message = message + + this.fallback = fallback + } +} + function handleHTTPErrors(res: Response) { if (!res.ok) { switch (res.status) { @@ -16,4 +30,4 @@ function handleHTTPErrors(res: Response) { } } -export { isAborted, handleHTTPErrors } +export { isAborted, AbortError, handleHTTPErrors }