From 5fcdad57e93ea532664ea0ec0b0fc5702a853a41 Mon Sep 17 00:00:00 2001 From: solo5star Date: Fri, 6 Oct 2023 21:20:05 +0900 Subject: [PATCH 01/17] =?UTF-8?q?feat:=20=EC=97=90=EB=9F=AC=20=EC=A0=95?= =?UTF-8?q?=EC=9D=98=20(`AppError`,=20`NetworkError`)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: jeongwusi Co-authored-by: geuntaek1013 --- client/src/client.ts | 22 ++++++++++++++-------- client/src/errors/AppError.ts | 3 +++ client/src/errors/NetworkError.ts | 9 +++++++++ 3 files changed, 26 insertions(+), 8 deletions(-) create mode 100644 client/src/errors/AppError.ts create mode 100644 client/src/errors/NetworkError.ts diff --git a/client/src/client.ts b/client/src/client.ts index f0823e4a..d9d66894 100644 --- a/client/src/client.ts +++ b/client/src/client.ts @@ -1,10 +1,16 @@ -import type { AuthProvider, AuthUrl, Cafe, CafeMapLocation, CafeMenu, LikedCafe, MapBounds, Rank, SearchedCafe, User } from './types'; - -export class ClientNetworkError extends Error { - constructor() { - super('인터넷에 연결할 수 없습니다'); - } -} +import NetworkError from './errors/NetworkError'; +import type { + AuthProvider, + AuthUrl, + Cafe, + CafeMapLocation, + CafeMenu, + LikedCafe, + MapBounds, + Rank, + SearchedCafe, + User, +} from './types'; class Client { isAccessTokenRefreshing = false; @@ -52,7 +58,7 @@ class Client { } return response; } catch (error) { - throw new ClientNetworkError(); + throw new NetworkError(); } } diff --git a/client/src/errors/AppError.ts b/client/src/errors/AppError.ts new file mode 100644 index 00000000..eeab6070 --- /dev/null +++ b/client/src/errors/AppError.ts @@ -0,0 +1,3 @@ +class AppError extends Error {} + +export default AppError; diff --git a/client/src/errors/NetworkError.ts b/client/src/errors/NetworkError.ts new file mode 100644 index 00000000..2882a561 --- /dev/null +++ b/client/src/errors/NetworkError.ts @@ -0,0 +1,9 @@ +import AppError from './AppError'; + +class NetworkError extends AppError { + constructor() { + super('인터넷 연결에 문제가 생겼어요'); + } +} + +export default NetworkError; From fa83f9784e278cd7de96e579d209584c568500c2 Mon Sep 17 00:00:00 2001 From: solo5star Date: Fri, 6 Oct 2023 21:20:46 +0900 Subject: [PATCH 02/17] =?UTF-8?q?feat:=20ErrorPage=EC=97=90=EC=84=9C=20App?= =?UTF-8?q?Error=20=ED=98=B9=EC=9D=80=20Error=20=EC=97=AC=EB=B6=80?= =?UTF-8?q?=EC=97=90=20=EB=94=B0=EB=9D=BC=20=EB=A9=94=EC=84=B8=EC=A7=80=20?= =?UTF-8?q?=ED=91=9C=EC=8B=9C=20=EB=B6=84=EA=B8=B0=20=EC=B2=98=EB=A6=AC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: jeongwusi Co-authored-by: geuntaek1013 --- client/src/pages/ErrorPage.tsx | 20 +++++++++++++------- 1 file changed, 13 insertions(+), 7 deletions(-) diff --git a/client/src/pages/ErrorPage.tsx b/client/src/pages/ErrorPage.tsx index 77df7ee6..cae250dd 100644 --- a/client/src/pages/ErrorPage.tsx +++ b/client/src/pages/ErrorPage.tsx @@ -1,31 +1,36 @@ import { MdOutlineErrorOutline } from 'react-icons/md'; -import { Link } from 'react-router-dom'; +import { Link, useRouteError } from 'react-router-dom'; import { keyframes, styled } from 'styled-components'; import Button from '../components/Button'; +import AppError from '../errors/AppError'; const ErrorPage = () => { + const error = useRouteError(); + const message = error instanceof Error ? error.message : '알 수 없는 에러'; + const isExpectedError = error instanceof AppError; + return ( - + - 다시 확인해주세요 - 요청하신 내용을 찾을 수 없어요 + {isExpectedError ? '다시 확인해주세요' : '예기치 못한 에러가 발생하였습니다'} + {isExpectedError && {message}} - + ); }; export default ErrorPage; -const Contaienr = styled.section` +const Container = styled.section` display: flex; flex-direction: column; align-items: center; - height: 100vh; + height: 100%; padding-top: 150px; `; @@ -49,6 +54,7 @@ const MainSentence = styled.h1` const Sentence = styled.span` font-size: ${({ theme }) => theme.fontSize.sm}; color: ${({ theme }) => theme.color.text.secondary}; + text-align: center; `; const bounce = keyframes` From 73bcefcbdf7c01e421a5a1df6a6fd3d5c5c6749d Mon Sep 17 00:00:00 2001 From: solo5star Date: Fri, 6 Oct 2023 21:21:30 +0900 Subject: [PATCH 03/17] =?UTF-8?q?feat:=20NotFoundPage=EC=97=90=EC=84=9C=20?= =?UTF-8?q?AppError=EB=A5=BC=20throw=ED=95=98=EB=8F=84=EB=A1=9D=20?= =?UTF-8?q?=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: jeongwusi Co-authored-by: geuntaek1013 --- client/src/pages/NotFoundPage.tsx | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/client/src/pages/NotFoundPage.tsx b/client/src/pages/NotFoundPage.tsx index 08d7c579..f592342a 100644 --- a/client/src/pages/NotFoundPage.tsx +++ b/client/src/pages/NotFoundPage.tsx @@ -1,5 +1,7 @@ +import AppError from '../errors/AppError'; + const NotFoundPage = () => { - return
NotFound
; + throw new AppError('경로를 찾을 수 없어요. 홈 화면으로 이동하거나 경로를 다시 한 번 확인해주시겠어요?'); }; export default NotFoundPage; From d5629b4a36afc6ef8286b1a53c59564f3cb04868 Mon Sep 17 00:00:00 2001 From: solo5star Date: Fri, 6 Oct 2023 21:22:03 +0900 Subject: [PATCH 04/17] =?UTF-8?q?fix:=20=EC=97=B0=EC=86=8D=EC=9C=BC?= =?UTF-8?q?=EB=A1=9C=20Toast=EB=A5=BC=20=EB=9D=84=EC=9A=B8=20=EB=95=8C=20?= =?UTF-8?q?=EC=9E=98=20=EB=9D=84=EC=9B=8C=EC=A7=80=EC=A7=80=20=EC=95=8A?= =?UTF-8?q?=EB=8A=94=20=EC=98=A4=EB=A5=98=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: jeongwusi Co-authored-by: geuntaek1013 --- client/src/context/ToastContext.tsx | 47 +++++++++++------------------ 1 file changed, 18 insertions(+), 29 deletions(-) diff --git a/client/src/context/ToastContext.tsx b/client/src/context/ToastContext.tsx index e86d873a..b2b38f5f 100644 --- a/client/src/context/ToastContext.tsx +++ b/client/src/context/ToastContext.tsx @@ -1,5 +1,5 @@ import type { PropsWithChildren } from 'react'; -import { createContext, useContext, useEffect, useState } from 'react'; +import { createContext, useContext, useState } from 'react'; import styled, { css, keyframes } from 'styled-components'; type ToastVariant = 'default' | 'success' | 'warning' | 'error'; @@ -14,34 +14,24 @@ type ToastProviderPros = PropsWithChildren; export const ToastProvider = (props: ToastProviderPros) => { const { children } = props; + const [toastId, setToastId] = useState(0); const [toastState, setToastState] = useState<{ variant: ToastVariant; message: string | null }>({ variant: 'default', message: null, }); - useEffect(() => { - if (toastState.variant && toastState.message !== null) { - const timer = setTimeout(() => { - setToastState({ variant: 'default', message: null }); - }, 5000); - - return () => clearTimeout(timer); - } - }, [toastState]); - const showToast = (variant: ToastVariant, message: string) => { setToastState({ variant, message }); + setToastId((toastId) => toastId + 1); }; - const contextValue = { - showToast, - }; + const contextValue = { showToast }; return ( {children} {toastState.message !== null && ( - + {toastState.message} )} @@ -59,6 +49,18 @@ export const useToast = () => { return context.showToast; }; +const ToastAnimation = keyframes` + from { + transform: translate(-50%) translateY(100px); + opacity: 0; + } + + to { + transform: translate(-50%) translateY(0px); + opacity: 1; + } +`; + const Container = styled.div` position: absolute; bottom: 50px; @@ -69,18 +71,8 @@ const Container = styled.div` flex-direction: column; white-space: nowrap; -`; -const ToastAnimation = keyframes` - from { - transform: translateY(100px); - opacity: 0; - } - - to { - transform: translateY(0px); - opacity: 1; - } + animation: 0.3s ${ToastAnimation}, 0.3s 3s reverse forwards ${ToastAnimation}; `; const ToastColorVariants = { @@ -100,12 +92,9 @@ const ToastColorVariants = { const ToastMessage = styled.div<{ $variant: ToastVariant }>` padding: ${({ theme }) => theme.space['3.5']}; - color: ${({ theme }) => theme.color.white}; - border-radius: 20px; box-shadow: rgba(60, 64, 67, 0.3) 0px 1px 2px, rgba(60, 64, 67, 0.15) 0px 1px 3px 1px; - animation: 0.3s ${ToastAnimation}, 0.3s 3s reverse forwards ${ToastAnimation}; ${({ $variant }) => ToastColorVariants[$variant]} `; From ab622241bee7b668dc9a92031aadad8cf72361f2 Mon Sep 17 00:00:00 2001 From: solo5star Date: Fri, 6 Oct 2023 21:23:34 +0900 Subject: [PATCH 05/17] =?UTF-8?q?feat:=20ErrorBoundary=EC=99=80=20QueryErr?= =?UTF-8?q?orBoundary=20=EA=B5=AC=ED=98=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: jeongwusi Co-authored-by: geuntaek1013 --- client/src/components/ErrorBoundary.tsx | 48 ++++++++++++++++++++ client/src/components/QueryErrorBoundary.tsx | 39 ++++++++++++++++ 2 files changed, 87 insertions(+) create mode 100644 client/src/components/ErrorBoundary.tsx create mode 100644 client/src/components/QueryErrorBoundary.tsx diff --git a/client/src/components/ErrorBoundary.tsx b/client/src/components/ErrorBoundary.tsx new file mode 100644 index 00000000..da8cffdc --- /dev/null +++ b/client/src/components/ErrorBoundary.tsx @@ -0,0 +1,48 @@ +import type { ErrorInfo, PropsWithChildren, ReactNode } from 'react'; +import React from 'react'; + +export type ErrorBoundaryProps = PropsWithChildren<{ + fallbackRender?: ({ resetErrorBoundary }: { resetErrorBoundary: () => void }) => ReactNode; + onReset?: () => void; +}>; + +type ErrorBoundaryState = { + error: Error | null; +}; + +class ErrorBoundary extends React.Component { + constructor(props: ErrorBoundaryProps) { + super(props); + this.state = { + error: null, + }; + } + + static getDerivedStateFromError(error: Error): ErrorBoundaryState { + return { error }; + } + + componentDidCatch(error: unknown, errorInfo: ErrorInfo) { + if (error instanceof Error) { + return; + } + throw error; + } + + render() { + const { fallbackRender } = this.props; + const { error } = this.state; + if (error) { + return fallbackRender?.({ resetErrorBoundary: () => this.resetErrorBoundary() }); + } + return this.props.children; + } + + resetErrorBoundary() { + this.setState({ error: null }); + const { onReset } = this.props; + onReset?.(); + } +} + +export default ErrorBoundary; diff --git a/client/src/components/QueryErrorBoundary.tsx b/client/src/components/QueryErrorBoundary.tsx new file mode 100644 index 00000000..655fc8fd --- /dev/null +++ b/client/src/components/QueryErrorBoundary.tsx @@ -0,0 +1,39 @@ +import { QueryErrorResetBoundary, onlineManager } from '@tanstack/react-query'; +import type { PropsWithChildren } from 'react'; +import { useEffect, useState } from 'react'; +import type { ErrorBoundaryProps } from './ErrorBoundary'; +import ErrorBoundary from './ErrorBoundary'; + +type QueryErrorBoundaryProps = PropsWithChildren; + +const QueryErrorBoundary = (props: QueryErrorBoundaryProps) => { + const { children, onReset, ...restProps } = props; + const [id, setId] = useState(0); + + useEffect(() => { + return onlineManager.subscribe(() => { + if (onlineManager.isOnline()) { + setId((id) => id + 1); + } + }); + }, []); + + return ( + + {({ reset }) => ( + { + reset(); + onReset?.(); + }} + {...restProps} + > + {children} + + )} + + ); +}; + +export default QueryErrorBoundary; From 637fea60f4892060923a543c49e73d72b534a59b Mon Sep 17 00:00:00 2001 From: solo5star Date: Fri, 6 Oct 2023 21:24:53 +0900 Subject: [PATCH 06/17] =?UTF-8?q?feat:=20useSilentLink=20=EA=B5=AC?= =?UTF-8?q?=ED=98=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: jeongwusi Co-authored-by: geuntaek1013 --- client/src/hooks/useSilentLink.ts | 15 +++++++++++++++ client/src/pages/Root.tsx | 3 +++ 2 files changed, 18 insertions(+) create mode 100644 client/src/hooks/useSilentLink.ts diff --git a/client/src/hooks/useSilentLink.ts b/client/src/hooks/useSilentLink.ts new file mode 100644 index 00000000..f9812c1b --- /dev/null +++ b/client/src/hooks/useSilentLink.ts @@ -0,0 +1,15 @@ +import { onlineManager } from '@tanstack/react-query'; +import { useEffect } from 'react'; +import { useToast } from '../context/ToastContext'; + +const useSilentLink = () => { + const setToast = useToast(); + useEffect(() => { + return onlineManager.subscribe(() => { + const isOnline = onlineManager.isOnline(); + setToast(isOnline ? 'success' : 'warning', isOnline ? '온라인이네요 굿' : '오프라인 헉헉'); + }); + }, [setToast]); +}; + +export default useSilentLink; diff --git a/client/src/pages/Root.tsx b/client/src/pages/Root.tsx index 5d317e17..03e07ce0 100644 --- a/client/src/pages/Root.tsx +++ b/client/src/pages/Root.tsx @@ -2,8 +2,11 @@ import { Suspense } from 'react'; import { Outlet } from 'react-router-dom'; import { styled } from 'styled-components'; import Navbar from '../components/Navbar'; +import useSilentLink from '../hooks/useSilentLink'; const Root = () => { + useSilentLink(); + return ( <> From e592dee67c69e68f12fe841a1b60ec54297bab2b Mon Sep 17 00:00:00 2001 From: solo5star Date: Fri, 6 Oct 2023 21:29:36 +0900 Subject: [PATCH 07/17] =?UTF-8?q?feat:=20ErrorRetryPrompt=20=EA=B5=AC?= =?UTF-8?q?=ED=98=84=20=EB=B0=8F=20=EC=A0=81=EC=9A=A9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: jeongwusi Co-authored-by: geuntaek1013 --- client/src/components/CafeMenuBottomSheet.tsx | 123 +++++++++++------- client/src/components/ErrorRetryPrompt.tsx | 59 +++++++++ 2 files changed, 133 insertions(+), 49 deletions(-) create mode 100644 client/src/components/ErrorRetryPrompt.tsx diff --git a/client/src/components/CafeMenuBottomSheet.tsx b/client/src/components/CafeMenuBottomSheet.tsx index 54b94a9e..d6aa93fa 100644 --- a/client/src/components/CafeMenuBottomSheet.tsx +++ b/client/src/components/CafeMenuBottomSheet.tsx @@ -1,4 +1,4 @@ -import { useEffect, useState } from 'react'; +import { Suspense, useEffect, useState } from 'react'; import { createPortal } from 'react-dom'; import { BsX } from 'react-icons/bs'; import { styled } from 'styled-components'; @@ -8,19 +8,77 @@ import type { Theme } from '../styles/theme'; import type { Cafe } from '../types'; import Resource from '../utils/Resource'; import CafeMenuList from './CafeMenuList'; +import ErrorRetryPrompt from './ErrorRetryPrompt'; import ImageModal from './ImageModal'; +import QueryErrorBoundary from './QueryErrorBoundary'; -type CafeMenuBottomSheetProps = { +type CafeMenuBottomSheetContentProps = { cafe: Cafe; - onClose: () => void; }; -const CafeMenuBottomSheet = (props: CafeMenuBottomSheetProps) => { - const { cafe, onClose } = props; +const CafeMenuBottomSheetContent = (props: CafeMenuBottomSheetContentProps) => { + const { cafe } = props; const { data: { menus, menuBoards }, } = useCafeMenus(cafe.id); const [isImageModalOpen, setIsImageModalOpen] = useState(false); + + const recommendedMenus = menus.filter((menuItem) => menuItem.isRecommended); + const otherMenus = menus.filter((menuItem) => !menuItem.isRecommended); + + return ( + <> + {menuBoards.length > 0 && ( + <> + setIsImageModalOpen(true)}> + 메뉴판 이미지로 보기 ({menuBoards.length}) + + + + )} + + {recommendedMenus.length > 0 && ( + <> + 대표 메뉴 + + + + )} + + {otherMenus.length > 0 && ( + <> + 메뉴 + + + + )} + + {menus.length === 0 && ( + <> + + 등록된 메뉴가 없습니다 + + )} + + {isImageModalOpen && + createPortal( + menuBoard.imageUrl)} + onClose={() => setIsImageModalOpen(false)} + />, + document.bodyRoot, + )} + + ); +}; + +type CafeMenuBottomSheetProps = { + cafe: Cafe; + onClose: () => void; +}; + +const CafeMenuBottomSheet = (props: CafeMenuBottomSheetProps) => { + const { cafe, onClose } = props; const scrollSnapGuardHandlers = useScrollSnapGuard(); useEffect(() => { @@ -32,10 +90,6 @@ const CafeMenuBottomSheet = (props: CafeMenuBottomSheetProps) => { const handlePreventClickPropagation: React.MouseEventHandler = (event) => { event.stopPropagation(); }; - - const recommendedMenus = menus.filter((menuItem) => menuItem.isRecommended); - const otherMenus = menus.filter((menuItem) => !menuItem.isRecommended); - return ( <> {createPortal( @@ -44,49 +98,20 @@ const CafeMenuBottomSheet = (props: CafeMenuBottomSheetProps) => { - {menuBoards.length > 0 && ( - <> - setIsImageModalOpen(true)}> - 메뉴판 이미지로 보기 ({menuBoards.length}) - - - - )} - - {recommendedMenus.length > 0 && ( - <> - 대표 메뉴 - - - - )} - - {otherMenus.length > 0 && ( - <> - 메뉴 - - - - )} - - {menus.length === 0 && ( - <> - - 등록된 메뉴가 없습니다 - - )} + ( + <> + + + )} + > + + + + , document.bodyRoot, )} - - {isImageModalOpen && - createPortal( - menuBoard.imageUrl)} - onClose={() => setIsImageModalOpen(false)} - />, - document.bodyRoot, - )} ); }; diff --git a/client/src/components/ErrorRetryPrompt.tsx b/client/src/components/ErrorRetryPrompt.tsx new file mode 100644 index 00000000..82393055 --- /dev/null +++ b/client/src/components/ErrorRetryPrompt.tsx @@ -0,0 +1,59 @@ +import styled from 'styled-components'; +import Button from './Button'; + +type ErrorRetryPromptProps = { + resetErrorBoundary: () => void; +}; + +const ErrorRetryPrompt = (Props: ErrorRetryPromptProps) => { + const { resetErrorBoundary } = Props; + + return ( + + + 일시적인 장애가 발생했습니다 + 요청사항을 처리하는데 실패했습니다. + + + + + + ); +}; + +export default ErrorRetryPrompt; + +const Container = styled.section` + display: flex; + flex-direction: column; + align-items: center; + + width: 100%; + height: 100%; + padding-top: 230px; +`; + +const SentenceContainer = styled.article` + display: flex; + flex-direction: column; + gap: ${({ theme }) => theme.space[5]}; + align-items: center; +`; + +const ButtonContainer = styled.div` + width: 133px; +`; + +const MainSentence = styled.h1` + font-size: ${({ theme }) => theme.fontSize['2xl']}; + font-weight: bold; + color: ${({ theme }) => theme.color.text.secondary}; +`; + +const Sentence = styled.span` + font-size: ${({ theme }) => theme.fontSize.sm}; + color: ${({ theme }) => theme.color.text.secondary}; + text-align: center; +`; From ce26068e718a601fd034fa9008bcb3865edc3b77 Mon Sep 17 00:00:00 2001 From: solo5star Date: Fri, 6 Oct 2023 21:30:35 +0900 Subject: [PATCH 08/17] =?UTF-8?q?feat:=20=EC=97=90=EB=9F=AC=20=EC=B2=98?= =?UTF-8?q?=EB=A6=AC=20router=EC=97=90=20=EC=A0=81=EC=9A=A9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: jeongwusi Co-authored-by: geuntaek1013 --- client/src/router.tsx | 64 +++++++++++++++++++++++++------------------ 1 file changed, 37 insertions(+), 27 deletions(-) diff --git a/client/src/router.tsx b/client/src/router.tsx index d03ff658..726ddf44 100644 --- a/client/src/router.tsx +++ b/client/src/router.tsx @@ -1,5 +1,6 @@ import React, { Suspense } from 'react'; import { createBrowserRouter } from 'react-router-dom'; +import ErrorPage from './pages/ErrorPage'; import Root from './pages/Root'; const SearchPage = React.lazy(() => import('./pages/SearchPage')); @@ -18,35 +19,44 @@ const router = createBrowserRouter([ { path: '/', element: , - errorElement: , - children: [ - { index: true, element: }, - { path: 'my-profile', element: }, - { path: 'cafes/:cafeId', element: }, - { path: 'rank', element: }, - { path: 'my-profile/cafes/:cafeId', element: }, - { path: 'search', element: }, - { path: 'map', element: }, - ], - }, - { - path: '/auth/:provider', - element: ( - }> - - - ), - }, - { - path: '/test', children: [ { - path: 'auth/:provider', - element: ( - - - - ), + path: '*', + errorElement: , + children: [ + { index: true, element: }, + { path: 'cafes/:cafeId', element: }, + { path: 'rank', element: }, + { path: 'my-profile', element: }, + { path: 'my-profile/cafes/:cafeId', element: }, + { path: 'search', element: }, + { path: 'map', element: }, + { + path: 'auth/:provider', + element: ( + }> + + + ), + }, + { + path: 'test', + children: [ + { + path: 'auth/:provider', + element: ( + + + + ), + }, + ], + }, + { + path: '*', + element: , + }, + ], }, ], }, From 8f39e61978797997cbb319b4b4f1b87d95d2facc Mon Sep 17 00:00:00 2001 From: solo5star Date: Sun, 8 Oct 2023 17:22:11 +0900 Subject: [PATCH 09/17] =?UTF-8?q?feat:=20CafeDetailBottomSheet=EC=97=90=20?= =?UTF-8?q?QueryErrorBoundary=20=EC=A0=81=EC=9A=A9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- client/src/components/CafeDetailBottomSheet.tsx | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/client/src/components/CafeDetailBottomSheet.tsx b/client/src/components/CafeDetailBottomSheet.tsx index 11a19f36..b9ddda19 100644 --- a/client/src/components/CafeDetailBottomSheet.tsx +++ b/client/src/components/CafeDetailBottomSheet.tsx @@ -7,6 +7,7 @@ import type { Theme } from '../styles/theme'; import type { Cafe } from '../types'; import CafeMenuMiniList from './CafeMenuMiniList'; import OpeningHoursDetail from './OpeningHoursDetail'; +import QueryErrorBoundary from './QueryErrorBoundary'; type CafeDetailBottomSheetProps = { cafe: Cafe; @@ -34,9 +35,11 @@ const CafeDetailBottomSheet = (props: CafeDetailBottomSheetProps) => { {cafe.name} - - - + + + + + From 34534f824b42997f165022a8500c0313396f2905 Mon Sep 17 00:00:00 2001 From: solo5star Date: Sun, 8 Oct 2023 17:22:40 +0900 Subject: [PATCH 10/17] =?UTF-8?q?style:=20=EC=BB=B4=ED=8F=AC=EB=84=8C?= =?UTF-8?q?=ED=8A=B8=20=EC=BD=94=EB=93=9C=20=EC=88=9C=EC=84=9C=20=EB=B3=80?= =?UTF-8?q?=EA=B2=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- client/src/components/CafeMenuBottomSheet.tsx | 87 +++++++++---------- 1 file changed, 40 insertions(+), 47 deletions(-) diff --git a/client/src/components/CafeMenuBottomSheet.tsx b/client/src/components/CafeMenuBottomSheet.tsx index d6aa93fa..ce5d13a9 100644 --- a/client/src/components/CafeMenuBottomSheet.tsx +++ b/client/src/components/CafeMenuBottomSheet.tsx @@ -8,10 +8,49 @@ import type { Theme } from '../styles/theme'; import type { Cafe } from '../types'; import Resource from '../utils/Resource'; import CafeMenuList from './CafeMenuList'; -import ErrorRetryPrompt from './ErrorRetryPrompt'; import ImageModal from './ImageModal'; import QueryErrorBoundary from './QueryErrorBoundary'; +type CafeMenuBottomSheetProps = { + cafe: Cafe; + onClose: () => void; +}; + +const CafeMenuBottomSheet = (props: CafeMenuBottomSheetProps) => { + const { cafe, onClose } = props; + const scrollSnapGuardHandlers = useScrollSnapGuard(); + + useEffect(() => { + document.addEventListener('click', onClose); + + return () => document.removeEventListener('click', onClose); + }, [onClose]); + + const handlePreventClickPropagation: React.MouseEventHandler = (event) => { + event.stopPropagation(); + }; + return ( + <> + {createPortal( + + + + + + + + + + + , + document.bodyRoot, + )} + + ); +}; + +export default CafeMenuBottomSheet; + type CafeMenuBottomSheetContentProps = { cafe: Cafe; }; @@ -72,52 +111,6 @@ const CafeMenuBottomSheetContent = (props: CafeMenuBottomSheetContentProps) => { ); }; -type CafeMenuBottomSheetProps = { - cafe: Cafe; - onClose: () => void; -}; - -const CafeMenuBottomSheet = (props: CafeMenuBottomSheetProps) => { - const { cafe, onClose } = props; - const scrollSnapGuardHandlers = useScrollSnapGuard(); - - useEffect(() => { - document.addEventListener('click', onClose); - - return () => document.removeEventListener('click', onClose); - }, [onClose]); - - const handlePreventClickPropagation: React.MouseEventHandler = (event) => { - event.stopPropagation(); - }; - return ( - <> - {createPortal( - - - - - - ( - <> - - - )} - > - - - - - , - document.bodyRoot, - )} - - ); -}; - -export default CafeMenuBottomSheet; - const Container = styled.div` position: absolute; bottom: 0; From 8f8bd53a2b378fc3ed8938806d1c909be8bf9e6a Mon Sep 17 00:00:00 2001 From: solo5star Date: Sun, 8 Oct 2023 17:35:51 +0900 Subject: [PATCH 11/17] =?UTF-8?q?feat:=20ErrorBoundary,=20QueryErrorBounda?= =?UTF-8?q?ry=20=EB=AA=87=EB=AA=87=20props=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * caught, FallbackComponent 추가 * fallbackRender에서 error도 주입 --- client/src/components/ErrorBoundary.tsx | 60 ++++++++++++++++---- client/src/components/QueryErrorBoundary.tsx | 24 +++++--- 2 files changed, 66 insertions(+), 18 deletions(-) diff --git a/client/src/components/ErrorBoundary.tsx b/client/src/components/ErrorBoundary.tsx index da8cffdc..15347efb 100644 --- a/client/src/components/ErrorBoundary.tsx +++ b/client/src/components/ErrorBoundary.tsx @@ -1,39 +1,77 @@ -import type { ErrorInfo, PropsWithChildren, ReactNode } from 'react'; +import type { ComponentType, ErrorInfo, PropsWithChildren, ReactNode } from 'react'; import React from 'react'; -export type ErrorBoundaryProps = PropsWithChildren<{ - fallbackRender?: ({ resetErrorBoundary }: { resetErrorBoundary: () => void }) => ReactNode; +type FallbackProps = { + error: TError; + resetErrorBoundary: () => void; +}; + +export type ErrorBoundaryBaseProps = PropsWithChildren<{ onReset?: () => void; + caught?: (error: unknown) => error is TError; }>; -type ErrorBoundaryState = { - error: Error | null; +export type ErrorBoundaryWithRenderProps = ErrorBoundaryBaseProps & { + fallbackRender: ({ error, resetErrorBoundary }: FallbackProps) => ReactNode; + FallbackComponent?: never; +}; + +export type ErrorBoundaryWithComponentProps = ErrorBoundaryBaseProps & { + fallbackRender?: never; + FallbackComponent: ComponentType>; +}; + +export type ErrorBoundaryWithNothingProps = ErrorBoundaryBaseProps & { + fallbackRender?: never; + FallbackComponent?: never; +}; + +export type ErrorBoundaryProps = + | ErrorBoundaryWithRenderProps + | ErrorBoundaryWithComponentProps + | ErrorBoundaryWithNothingProps; + +type ErrorBoundaryState = { + error: TError | null; }; -class ErrorBoundary extends React.Component { - constructor(props: ErrorBoundaryProps) { +/** + * 자식 컴포넌트를 렌더링하는 중 에러가 발생하였을 때(throw error) + * 대체 컴포넌트(fallback)을 렌더합니다. + */ +class ErrorBoundary extends React.Component, ErrorBoundaryState> { + constructor(props: ErrorBoundaryProps) { super(props); this.state = { error: null, }; } - static getDerivedStateFromError(error: Error): ErrorBoundaryState { + static getDerivedStateFromError(error: any): ErrorBoundaryState { return { error }; } componentDidCatch(error: unknown, errorInfo: ErrorInfo) { - if (error instanceof Error) { + const defaultCaught = (error: unknown): error is Error => error instanceof Error; + const { caught = defaultCaught } = this.props; + if (caught(error)) { return; } throw error; } render() { - const { fallbackRender } = this.props; const { error } = this.state; if (error) { - return fallbackRender?.({ resetErrorBoundary: () => this.resetErrorBoundary() }); + if ('fallbackRender' in this.props && this.props.fallbackRender) { + const { fallbackRender } = this.props; + return fallbackRender({ error, resetErrorBoundary: () => this.resetErrorBoundary() }); + } + if ('FallbackComponent' in this.props && this.props.FallbackComponent) { + const { FallbackComponent } = this.props; + return this.resetErrorBoundary()} />; + } + return null; } return this.props.children; } diff --git a/client/src/components/QueryErrorBoundary.tsx b/client/src/components/QueryErrorBoundary.tsx index 655fc8fd..36cadc54 100644 --- a/client/src/components/QueryErrorBoundary.tsx +++ b/client/src/components/QueryErrorBoundary.tsx @@ -1,20 +1,28 @@ import { QueryErrorResetBoundary, onlineManager } from '@tanstack/react-query'; -import type { PropsWithChildren } from 'react'; import { useEffect, useState } from 'react'; import type { ErrorBoundaryProps } from './ErrorBoundary'; import ErrorBoundary from './ErrorBoundary'; +import ErrorRetryPrompt from './ErrorRetryPrompt'; -type QueryErrorBoundaryProps = PropsWithChildren; +type QueryErrorBoundaryProps = ErrorBoundaryProps; -const QueryErrorBoundary = (props: QueryErrorBoundaryProps) => { +/** + * react-query에서 `useQuery` 혹은 `useMutation` 등의 쿼리 함수를 + * `suspense: true`로 사용 중 발생하는 에러를 위한 + * ErrorBoundary 컴포넌트입니다. + * + * ErrorBoundary에서 다음 부가 기능이 추가되어있습니다 + * * network 상태가 온라인으로 전환되었을 때 자동으로 재시도(error reset) + * * ErrorBoundary reset 시, query cache도 reset + * * 아무런 fallback이 주어지지 않았을 시 {@link ErrorRetryPrompt} 를 기본으로 사용 + */ +const QueryErrorBoundary = (props: QueryErrorBoundaryProps) => { const { children, onReset, ...restProps } = props; const [id, setId] = useState(0); useEffect(() => { return onlineManager.subscribe(() => { - if (onlineManager.isOnline()) { - setId((id) => id + 1); - } + if (onlineManager.isOnline()) setId((id) => id + 1); }); }, []); @@ -27,7 +35,9 @@ const QueryErrorBoundary = (props: QueryErrorBoundaryProps) => { reset(); onReset?.(); }} - {...restProps} + {...('FallbackComponent' in restProps || 'fallbackRender' in restProps + ? restProps + : { FallbackComponent: ErrorRetryPrompt, ...restProps })} > {children} From 9871b03b60ef079689d2151dc5b8aab2d026d791 Mon Sep 17 00:00:00 2001 From: solo5star Date: Sun, 8 Oct 2023 17:36:12 +0900 Subject: [PATCH 12/17] =?UTF-8?q?fix:=20ErrorRetryPrompt=EA=B0=80=20?= =?UTF-8?q?=EB=8C=80=EB=B6=80=EB=B6=84=EC=9D=98=20box=EC=97=90=EC=84=9C=20?= =?UTF-8?q?=EC=9E=98=20=ED=91=9C=EC=8B=9C=EB=90=98=EB=8F=84=EB=A1=9D=20?= =?UTF-8?q?=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- client/src/components/ErrorRetryPrompt.tsx | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/client/src/components/ErrorRetryPrompt.tsx b/client/src/components/ErrorRetryPrompt.tsx index 82393055..e846fe19 100644 --- a/client/src/components/ErrorRetryPrompt.tsx +++ b/client/src/components/ErrorRetryPrompt.tsx @@ -29,10 +29,11 @@ const Container = styled.section` display: flex; flex-direction: column; align-items: center; + justify-content: center; width: 100%; height: 100%; - padding-top: 230px; + padding: ${({ theme }) => theme.space['5']}; `; const SentenceContainer = styled.article` From 98acc1234f7c400249508dfc999a8b0b7684592c Mon Sep 17 00:00:00 2001 From: solo5star Date: Sun, 8 Oct 2023 17:38:38 +0900 Subject: [PATCH 13/17] =?UTF-8?q?feat:=20=EB=A9=94=EC=84=B8=EC=A7=80=20?= =?UTF-8?q?=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- client/src/hooks/useSilentLink.ts | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/client/src/hooks/useSilentLink.ts b/client/src/hooks/useSilentLink.ts index f9812c1b..13fc799d 100644 --- a/client/src/hooks/useSilentLink.ts +++ b/client/src/hooks/useSilentLink.ts @@ -2,12 +2,18 @@ import { onlineManager } from '@tanstack/react-query'; import { useEffect } from 'react'; import { useToast } from '../context/ToastContext'; +/** + * 온라인 혹은 오프라인 상태를 감시하고 변화가 발생할 시 토스트로 알림을 띄워주는 훅입니다. + */ const useSilentLink = () => { const setToast = useToast(); useEffect(() => { return onlineManager.subscribe(() => { const isOnline = onlineManager.isOnline(); - setToast(isOnline ? 'success' : 'warning', isOnline ? '온라인이네요 굿' : '오프라인 헉헉'); + setToast( + isOnline ? 'success' : 'warning', + isOnline ? '인터넷이 다시 연결되었습니다' : '인터넷 연결이 끊어졌습니다. 확인해주세요', + ); }); }, [setToast]); }; From 13558c714ea7a2606add3a8c149a09313058660c Mon Sep 17 00:00:00 2001 From: solo5star Date: Sun, 8 Oct 2023 18:37:12 +0900 Subject: [PATCH 14/17] =?UTF-8?q?docs:=20AppError=20=ED=81=B4=EB=9E=98?= =?UTF-8?q?=EC=8A=A4=EC=97=90=20JSDoc=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- client/src/errors/AppError.ts | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/client/src/errors/AppError.ts b/client/src/errors/AppError.ts index eeab6070..1fc2a59f 100644 --- a/client/src/errors/AppError.ts +++ b/client/src/errors/AppError.ts @@ -1,3 +1,13 @@ +/** + * 앱에서 의도적으로 발생시킨 에러이며, 복구 가능한 에러이다 + * + * 사용자가 조치할 수 있는 수준의 에러인 경우 `AppError` 혹은 그 자식 클래스여야 한다 + * + * @example + * if (!user.isAdmin) { + * throw new AppError('이 자원에 접근할 수 없습니다. 권한이 부족합니다.'); + * } + */ class AppError extends Error {} export default AppError; From bea286a1c23c8ba4583b4a5a86d48c67ff80f0d0 Mon Sep 17 00:00:00 2001 From: solo5star Date: Sun, 8 Oct 2023 18:37:41 +0900 Subject: [PATCH 15/17] =?UTF-8?q?docs:=20NetworkError=EC=97=90=20JSDoc=20?= =?UTF-8?q?=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- client/src/errors/NetworkError.ts | 3 +++ 1 file changed, 3 insertions(+) diff --git a/client/src/errors/NetworkError.ts b/client/src/errors/NetworkError.ts index 2882a561..b3d94c1b 100644 --- a/client/src/errors/NetworkError.ts +++ b/client/src/errors/NetworkError.ts @@ -1,5 +1,8 @@ import AppError from './AppError'; +/** + * 네트워크 오류에 해당되는 에러 클래스이다 + */ class NetworkError extends AppError { constructor() { super('인터넷 연결에 문제가 생겼어요'); From 15f1074bc8f17beba61de503c64472d1c1b8c32c Mon Sep 17 00:00:00 2001 From: solo5star Date: Sun, 8 Oct 2023 18:38:34 +0900 Subject: [PATCH 16/17] =?UTF-8?q?feat:=20APIError=20=EC=B6=94=EA=B0=80=20?= =?UTF-8?q?=EB=B0=8F=20=EC=A0=81=EC=9A=A9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- client/src/client.ts | 67 ++++++++++++++++++++--------------- client/src/errors/APIError.ts | 44 +++++++++++++++++++++++ client/src/types/index.ts | 7 +++- 3 files changed, 88 insertions(+), 30 deletions(-) create mode 100644 client/src/errors/APIError.ts diff --git a/client/src/client.ts b/client/src/client.ts index d9d66894..83c1b6c0 100644 --- a/client/src/client.ts +++ b/client/src/client.ts @@ -1,3 +1,4 @@ +import APIError from './errors/APIError'; import NetworkError from './errors/NetworkError'; import type { AuthProvider, @@ -26,40 +27,48 @@ class Client { } async fetch(input: RequestInfo | URL, init?: RequestInit): Promise { + const fetchFn: () => Promise = () => + fetch(`/api${input}`, { + ...init, + headers: { + ...init?.headers, + ...(this.accessToken ? { Authorization: `Bearer ${this.accessToken}` } : {}), + }, + }); + let response: Response; + try { - const fetchFn = () => - fetch(`/api${input}`, { - ...init, - headers: { - ...init?.headers, - ...(this.accessToken ? { Authorization: `Bearer ${this.accessToken}` } : {}), - }, - }); - - let response = await fetchFn(); - if (!response.ok) { - if (response.status === 401 && !this.isAccessTokenRefreshing) { - // this.refreshAccessToken() 을 호출하기 전, - // this.isAccessTokenRefreshing을 true로 설정해줘야 잠재적인 recursion loop를 방지할 수 있습니다. - this.isAccessTokenRefreshing = true; - const accessToken = await this.refreshAccessToken(); - this.accessToken = accessToken; - this.accessTokenRefreshListener?.(accessToken); - this.isAccessTokenRefreshing = false; - - response = await fetchFn(); - } + response = await fetchFn(); + } catch (error) { + throw new NetworkError(); + } - if (!response.ok) { - // access token 재발급이 불가하기 때문에 access token을 삭제한다. - this.accessTokenRefreshListener?.(null); - throw response; + if (!response.ok) { + if (response.status === 401 && !this.isAccessTokenRefreshing) { + // this.refreshAccessToken() 을 호출하기 전, + // this.isAccessTokenRefreshing을 true로 설정해줘야 잠재적인 recursion loop를 방지할 수 있습니다. + this.isAccessTokenRefreshing = true; + const accessToken = await this.refreshAccessToken(); + this.accessToken = accessToken; + this.accessTokenRefreshListener?.(accessToken); + this.isAccessTokenRefreshing = false; + + response = await fetchFn(); + } + + if (!response.ok) { + // access token 재발급이 불가하기 때문에 access token을 삭제한다. + this.accessTokenRefreshListener?.(null); + let body: unknown; + try { + body = await response.json(); + } catch (error) { + body = await response.text(); } + throw new APIError(response, body); } - return response; - } catch (error) { - throw new NetworkError(); } + return response; } async fetchJson(input: RequestInfo | URL, init?: RequestInit): Promise { diff --git a/client/src/errors/APIError.ts b/client/src/errors/APIError.ts new file mode 100644 index 00000000..84a87bc5 --- /dev/null +++ b/client/src/errors/APIError.ts @@ -0,0 +1,44 @@ +import type { ErrorResponseBody } from '../types'; +import AppError from './AppError'; + +/** + * API 호출의 응답이 400, 500 등의 error response일 경우 + * 이를 wrapping하는 에러 클래스 + */ +class APIError extends AppError { + readonly response: Response; + + constructor(response: Response, body: unknown) { + super(APIError.getErrorMessageFromBody(body)); + this.response = response; + } + + /** + * 서버 측에서 지정해 준 에러 코드 혹은 메세지가 포함되어 있는지 검사합니다 + */ + static isErrorResponseBody(body: unknown): body is ErrorResponseBody { + return !!body && typeof body === 'object' && 'code' in body && 'message' in body; + } + + /** + * 응답 body에서 적절한 에러 메세지를 추출하여 반환합니다 + * + * @param body 에러가 포함될 수 있는 모든 객체 + * @returns 에러 메세지 + */ + static getErrorMessageFromBody(body: unknown): string { + if (APIError.isErrorResponseBody(body)) { + return body.message; + } + if (!!body && typeof body === 'object') { + if ('message' in body) { + return String(body.message); + } + + return JSON.stringify(body); + } + return String(body); + } +} + +export default APIError; diff --git a/client/src/types/index.ts b/client/src/types/index.ts index 221122c9..afcedd5a 100644 --- a/client/src/types/index.ts +++ b/client/src/types/index.ts @@ -91,7 +91,7 @@ export type SearchedCafe = { address: Cafe['address']; image: string; likeCount: Cafe['likeCount']; -} +}; export type CafeMapLocation = { id: number; @@ -107,3 +107,8 @@ export type MapBounds = { longitudeDelta: number; latitudeDelta: number; }; + +export type ErrorResponseBody = { + code: string; + message: string; +}; From c5bbb90581d6322901d9493812c6317f4a1b3950 Mon Sep 17 00:00:00 2001 From: solo5star Date: Sun, 8 Oct 2023 18:39:05 +0900 Subject: [PATCH 17/17] feat: query retry off --- client/src/App.tsx | 1 + 1 file changed, 1 insertion(+) diff --git a/client/src/App.tsx b/client/src/App.tsx index 15334d5b..edff20a7 100644 --- a/client/src/App.tsx +++ b/client/src/App.tsx @@ -13,6 +13,7 @@ const queryClient = new QueryClient({ defaultOptions: { queries: { suspense: true, + retry: false, }, }, });