diff options
| author | nsfisis <nsfisis@gmail.com> | 2026-02-04 22:15:54 +0900 |
|---|---|---|
| committer | nsfisis <nsfisis@gmail.com> | 2026-02-04 22:42:12 +0900 |
| commit | 0d42e351b318302098c585dc5d9730cc741ae8d3 (patch) | |
| tree | aed005eab156f48da562ed327b59050e68fc4122 /src/client/pages/DeckCardsPage.tsx | |
| parent | fc208d960e137dd36590908145a13d8501144b7a (diff) | |
| download | kioku-0d42e351b318302098c585dc5d9730cc741ae8d3.tar.gz kioku-0d42e351b318302098c585dc5d9730cc741ae8d3.tar.zst kioku-0d42e351b318302098c585dc5d9730cc741ae8d3.zip | |
refactor(ui): replace LoadingSpinner with skeleton fallbacks in Suspense
Reduce layout shift during data loading by using content-shaped
skeleton placeholders instead of generic spinners in DeckDetailPage,
DeckCardsPage, and StudyPage.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
Diffstat (limited to 'src/client/pages/DeckCardsPage.tsx')
| -rw-r--r-- | src/client/pages/DeckCardsPage.tsx | 53 |
1 files changed, 50 insertions, 3 deletions
diff --git a/src/client/pages/DeckCardsPage.tsx b/src/client/pages/DeckCardsPage.tsx index 73a83ae..f4b6c78 100644 --- a/src/client/pages/DeckCardsPage.tsx +++ b/src/client/pages/DeckCardsPage.tsx @@ -30,7 +30,6 @@ import { EditCardModal } from "../components/EditCardModal"; import { EditNoteModal } from "../components/EditNoteModal"; import { ErrorBoundary } from "../components/ErrorBoundary"; import { ImportNotesModal } from "../components/ImportNotesModal"; -import { LoadingSpinner } from "../components/LoadingSpinner"; import { queryClient } from "../queryClient"; /** Combined type for display: note group */ @@ -440,7 +439,14 @@ function CardsContent({ <div className="animate-fade-in"> {/* Deck Header */} <ErrorBoundary> - <Suspense fallback={<LoadingSpinner />}> + <Suspense + fallback={ + <div className="mb-8"> + <div className="h-9 w-48 bg-muted/20 rounded animate-pulse mb-2" /> + <div className="h-5 w-64 bg-muted/20 rounded animate-pulse" /> + </div> + } + > <DeckHeader deckId={deckId} /> </Suspense> </ErrorBoundary> @@ -571,7 +577,48 @@ export function DeckCardsPage() { {/* Main Content */} <main className="max-w-4xl mx-auto px-4 py-8"> <ErrorBoundary> - <Suspense fallback={<LoadingSpinner />}> + <Suspense + fallback={ + <div className="animate-fade-in"> + <div className="mb-8"> + <div className="h-9 w-48 bg-muted/20 rounded animate-pulse mb-2" /> + <div className="h-5 w-64 bg-muted/20 rounded animate-pulse" /> + </div> + <div className="flex items-center justify-between mb-4"> + <div className="h-7 w-32 bg-muted/20 rounded animate-pulse" /> + <div className="flex gap-2"> + <div className="h-10 w-28 bg-muted/20 rounded-lg animate-pulse" /> + <div className="h-10 w-24 bg-muted/20 rounded-lg animate-pulse" /> + </div> + </div> + <div className="h-10 w-full bg-muted/20 rounded-lg animate-pulse mb-6" /> + <div className="space-y-4"> + {[0, 1, 2].map((i) => ( + <div + key={i} + className="bg-white rounded-xl border border-border/50 overflow-hidden" + > + <div className="px-5 py-3 border-b border-border/30 bg-ivory/30"> + <div className="h-4 w-28 bg-muted/20 rounded animate-pulse" /> + </div> + <div className="p-5"> + <div className="grid grid-cols-2 gap-4"> + <div> + <div className="h-3 w-10 bg-muted/20 rounded animate-pulse mb-2" /> + <div className="h-4 w-32 bg-muted/20 rounded animate-pulse" /> + </div> + <div> + <div className="h-3 w-10 bg-muted/20 rounded animate-pulse mb-2" /> + <div className="h-4 w-32 bg-muted/20 rounded animate-pulse" /> + </div> + </div> + </div> + </div> + ))} + </div> + </div> + } + > <CardsContent deckId={deckId} onCreateNote={() => setIsCreateModalOpen(true)} |
