aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/client
diff options
context:
space:
mode:
authornsfisis <nsfisis@gmail.com>2026-02-04 22:15:54 +0900
committernsfisis <nsfisis@gmail.com>2026-02-04 22:42:12 +0900
commit0d42e351b318302098c585dc5d9730cc741ae8d3 (patch)
treeaed005eab156f48da562ed327b59050e68fc4122 /src/client
parentfc208d960e137dd36590908145a13d8501144b7a (diff)
downloadkioku-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')
-rw-r--r--src/client/pages/DeckCardsPage.tsx53
-rw-r--r--src/client/pages/DeckDetailPage.tsx26
-rw-r--r--src/client/pages/StudyPage.tsx15
3 files changed, 87 insertions, 7 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)}
diff --git a/src/client/pages/DeckDetailPage.tsx b/src/client/pages/DeckDetailPage.tsx
index 97f8378..792bbe2 100644
--- a/src/client/pages/DeckDetailPage.tsx
+++ b/src/client/pages/DeckDetailPage.tsx
@@ -55,14 +55,36 @@ function DeckContent({ deckId }: { deckId: string }) {
<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>
{/* Deck Stats */}
<ErrorBoundary>
- <Suspense fallback={<LoadingSpinner />}>
+ <Suspense
+ fallback={
+ <div className="bg-white rounded-xl border border-border/50 p-6 mb-6">
+ <div className="grid grid-cols-2 gap-6">
+ <div>
+ <div className="h-4 w-20 bg-muted/20 rounded animate-pulse mb-1" />
+ <div className="h-8 w-12 bg-muted/20 rounded animate-pulse" />
+ </div>
+ <div>
+ <div className="h-4 w-20 bg-muted/20 rounded animate-pulse mb-1" />
+ <div className="h-8 w-12 bg-muted/20 rounded animate-pulse" />
+ </div>
+ </div>
+ </div>
+ }
+ >
<DeckStats deckId={deckId} />
</Suspense>
</ErrorBoundary>
diff --git a/src/client/pages/StudyPage.tsx b/src/client/pages/StudyPage.tsx
index 9c79752..423fdd0 100644
--- a/src/client/pages/StudyPage.tsx
+++ b/src/client/pages/StudyPage.tsx
@@ -17,7 +17,6 @@ import { Link, useParams } from "wouter";
import { ApiClientError, apiClient } from "../api";
import { studyDataAtomFamily } from "../atoms";
import { ErrorBoundary } from "../components/ErrorBoundary";
-import { LoadingSpinner } from "../components/LoadingSpinner";
import { renderCard } from "../utils/templateRenderer";
type Rating = 1 | 2 | 3 | 4;
@@ -373,7 +372,19 @@ export function StudyPage() {
{/* Main Content */}
<main className="flex-1 flex flex-col max-w-2xl mx-auto w-full px-4 py-6">
<ErrorBoundary>
- <Suspense fallback={<LoadingSpinner className="flex-1" />}>
+ <Suspense
+ fallback={
+ <div className="flex-1 flex flex-col">
+ <div className="flex items-center justify-between mb-6">
+ <div className="h-7 w-36 bg-muted/20 rounded animate-pulse" />
+ <div className="h-7 w-24 bg-muted/20 rounded-full animate-pulse" />
+ </div>
+ <div className="flex-1 min-h-[280px] bg-white rounded-2xl border border-border/50 shadow-card p-8 flex items-center justify-center">
+ <div className="h-7 w-48 bg-muted/20 rounded animate-pulse" />
+ </div>
+ </div>
+ }
+ >
<StudySession deckId={deckId} />
</Suspense>
</ErrorBoundary>