Files
sundol/sundol-frontend/src/app/knowledge/[id]/page.tsx
joungmin 20210830cf Fix TTS: switch to 1.7B with ref_audio, speakable text on all lines
- Use 1.7B model (0.6B had tensor mismatch with cached prompts)
- Speak endpoint uses ref_audio directly (not cached pkl) as fallback
- Cache voice clone prompts in memory on startup
- Add SpeakableText component: 🔊 icon on each p and li element
- Remove old TTSReader sequential approach
- Add global exception handler to TTS server
- Fix profile localStorage caching
- inference_mode + bf16 optimization

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 12:14:06 +00:00

446 lines
17 KiB
TypeScript

"use client";
import { useEffect, useState } from "react";
import { useParams, useRouter } from "next/navigation";
import AuthGuard from "@/components/auth-guard";
import NavBar from "@/components/nav-bar";
import { useApi } from "@/lib/use-api";
import ReactMarkdown from "react-markdown";
import SpeakableText from "@/components/speakable-text";
interface Category {
ID: string;
NAME: string;
DEPTH: number;
FULL_PATH: string;
}
interface KnowledgeItem {
ID: string;
TYPE: string;
TITLE: string;
SOURCE_URL: string;
RAW_TEXT: string;
STRUCTURED_CONTENT: string | null;
STATUS: string;
CREATED_AT: string;
UPDATED_AT: string;
CATEGORIES: Category[];
}
interface Chunk {
ID: string;
CHUNK_INDEX: number;
CONTENT: string;
TOKEN_COUNT: number;
}
const statusColors: Record<string, string> = {
PENDING: "bg-yellow-500/20 text-yellow-400",
EXTRACTING: "bg-blue-500/20 text-blue-400",
CHUNKING: "bg-purple-500/20 text-purple-400",
CATEGORIZING: "bg-indigo-500/20 text-indigo-400",
STRUCTURING: "bg-orange-500/20 text-orange-400",
EMBEDDING: "bg-cyan-500/20 text-cyan-400",
READY: "bg-green-500/20 text-green-400",
FAILED: "bg-red-500/20 text-red-400",
};
const typeLabels: Record<string, string> = {
YOUTUBE: "YouTube",
WEB: "Web",
TEXT: "Text",
};
function extractYouTubeVideoId(url: string): string | null {
try {
const u = new URL(url);
if (u.hostname === "youtu.be") return u.pathname.slice(1);
if (u.hostname.includes("youtube.com")) return u.searchParams.get("v");
} catch {
// invalid URL
}
return null;
}
export default function KnowledgeDetailPage() {
const { request } = useApi();
const router = useRouter();
const params = useParams();
const id = params.id as string;
const [item, setItem] = useState<KnowledgeItem | null>(null);
const [chunks, setChunks] = useState<Chunk[]>([]);
const [loading, setLoading] = useState(true);
const [error, setError] = useState<string | null>(null);
const [editingTitle, setEditingTitle] = useState(false);
const [titleDraft, setTitleDraft] = useState("");
const [showChunks, setShowChunks] = useState(false);
const [deleting, setDeleting] = useState(false);
const [generating, setGenerating] = useState(false);
const [structuring, setStructuring] = useState(false);
const [showStructured, setShowStructured] = useState(true);
const fetchItem = async () => {
try {
const data = await request<KnowledgeItem>({ method: "GET", url: `/api/knowledge/${id}` });
setItem(data);
} catch (err: unknown) {
const msg = err instanceof Error ? err.message : "Failed to load";
setError(msg);
} finally {
setLoading(false);
}
};
const fetchChunks = async () => {
try {
const data = await request<Chunk[]>({ method: "GET", url: `/api/knowledge/${id}/chunks` });
setChunks(data);
} catch (err) {
console.error("Failed to load chunks:", err);
}
};
useEffect(() => {
fetchItem();
}, [id]);
// Poll while processing
useEffect(() => {
if (!item) return;
const processing = ["PENDING", "EXTRACTING", "STRUCTURING", "CHUNKING", "CATEGORIZING", "EMBEDDING"].includes(item.STATUS);
if (!processing) return;
const interval = setInterval(fetchItem, 3000);
return () => clearInterval(interval);
}, [item?.STATUS]);
const handleSaveTitle = async () => {
if (!titleDraft.trim()) return;
try {
const updated = await request<KnowledgeItem>({
method: "PATCH",
url: `/api/knowledge/${id}`,
data: { title: titleDraft.trim() },
});
setItem(updated);
setEditingTitle(false);
} catch (err) {
console.error("Failed to update title:", err);
}
};
const handleDelete = async () => {
if (!confirm("정말 삭제하시겠습니까?")) return;
setDeleting(true);
try {
await request({ method: "DELETE", url: `/api/knowledge/${id}` });
router.push("/knowledge");
} catch (err) {
console.error("Failed to delete:", err);
setDeleting(false);
}
};
const handleToggleChunks = () => {
if (!showChunks && chunks.length === 0) {
fetchChunks();
}
setShowChunks(!showChunks);
};
if (loading) {
return (
<AuthGuard>
<NavBar />
<main className="max-w-4xl mx-auto px-4 py-8">
<p className="text-[var(--color-text-muted)]">Loading...</p>
</main>
</AuthGuard>
);
}
if (error || !item) {
return (
<AuthGuard>
<NavBar />
<main className="max-w-4xl mx-auto px-4 py-8">
<p className="text-red-400">{error || "Item not found"}</p>
<button
onClick={() => router.push("/knowledge")}
className="mt-4 text-sm text-[var(--color-primary)] hover:underline"
>
Back to Knowledge
</button>
</main>
</AuthGuard>
);
}
const videoId = item.TYPE === "YOUTUBE" && item.SOURCE_URL ? extractYouTubeVideoId(item.SOURCE_URL) : null;
return (
<AuthGuard>
<NavBar />
<main className="max-w-4xl mx-auto px-4 py-8">
{/* Back link */}
<button
onClick={() => router.push("/knowledge")}
className="text-sm text-[var(--color-text-muted)] hover:text-[var(--color-primary)] mb-4 inline-block"
>
Back to Knowledge
</button>
{/* Header */}
<div className="bg-[var(--color-bg-card)] rounded-xl p-6 border border-[var(--color-border)] mb-6">
<div className="flex items-center gap-3 mb-3">
<span className="text-xs px-2 py-0.5 rounded bg-[var(--color-bg-hover)] text-[var(--color-text-muted)]">
{typeLabels[item.TYPE] || item.TYPE}
</span>
<span className={`text-xs px-2 py-0.5 rounded ${statusColors[item.STATUS] || ""}`}>
{item.STATUS}
</span>
</div>
{/* Title (editable) */}
{editingTitle ? (
<div className="flex gap-2 mb-3">
<input
type="text"
value={titleDraft}
onChange={(e) => setTitleDraft(e.target.value)}
onKeyDown={(e) => e.key === "Enter" && handleSaveTitle()}
className="flex-1 px-3 py-1 rounded-lg bg-[var(--color-bg-hover)] border border-[var(--color-border)] focus:border-[var(--color-primary)] focus:outline-none text-lg font-bold"
autoFocus
/>
<button
onClick={handleSaveTitle}
className="px-3 py-1 text-sm bg-[var(--color-primary)] rounded-lg"
>
</button>
<button
onClick={() => setEditingTitle(false)}
className="px-3 py-1 text-sm bg-[var(--color-bg-hover)] border border-[var(--color-border)] rounded-lg"
>
</button>
</div>
) : (
<h1
className="text-xl font-bold mb-3 cursor-pointer hover:text-[var(--color-primary)] transition-colors"
onClick={() => {
setTitleDraft(item.TITLE || "");
setEditingTitle(true);
}}
title="클릭하여 제목 수정"
>
{item.TITLE || "Untitled"}
</h1>
)}
{/* Source URL */}
{item.SOURCE_URL && (
<p className="text-sm text-[var(--color-text-muted)] mb-3 break-all">
<a href={item.SOURCE_URL} target="_blank" rel="noopener noreferrer" className="hover:text-[var(--color-primary)]">
{item.SOURCE_URL}
</a>
</p>
)}
{/* Meta */}
<div className="flex gap-4 text-xs text-[var(--color-text-muted)]">
<span>: {new Date(item.CREATED_AT).toLocaleString("ko-KR")}</span>
<span>: {new Date(item.UPDATED_AT).toLocaleString("ko-KR")}</span>
</div>
</div>
{/* YouTube Embed */}
{videoId && (
<div className="rounded-xl overflow-hidden border border-[var(--color-border)] mb-6">
<div className="relative w-full" style={{ paddingBottom: "56.25%" }}>
<iframe
className="absolute inset-0 w-full h-full"
src={`https://www.youtube.com/embed/${videoId}`}
allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture"
allowFullScreen
/>
</div>
</div>
)}
{/* Processing indicator */}
{/* Categories */}
{item.CATEGORIES && item.CATEGORIES.length > 0 && (
<div className="flex flex-wrap gap-2 mb-6">
{item.CATEGORIES.map((cat) => (
<span
key={cat.ID}
className="text-xs px-2.5 py-1 rounded-full bg-[var(--color-primary)]/15 text-[var(--color-primary)] border border-[var(--color-primary)]/30"
>
{cat.FULL_PATH}
</span>
))}
</div>
)}
{["PENDING", "EXTRACTING", "STRUCTURING", "CHUNKING", "CATEGORIZING", "EMBEDDING"].includes(item.STATUS) && (
<div className="bg-blue-500/10 border border-blue-500/30 rounded-xl p-4 mb-6 flex items-center gap-3">
<div className="w-4 h-4 border-2 border-blue-400 border-t-transparent rounded-full animate-spin" />
<span className="text-sm text-blue-400">
{item.STATUS === "PENDING" && "파이프라인 대기 중..."}
{item.STATUS === "EXTRACTING" && "텍스트 추출 중..."}
{item.STATUS === "STRUCTURING" && "내용 정리 중..."}
{item.STATUS === "CHUNKING" && "청킹 처리 중..."}
{item.STATUS === "CATEGORIZING" && "카테고리 분류 중..."}
{item.STATUS === "EMBEDDING" && "벡터 임베딩 중..."}
</span>
</div>
)}
{/* Structured Content */}
{item.STRUCTURED_CONTENT && (
<div className="mb-6">
<button
onClick={() => setShowStructured(!showStructured)}
className="text-sm text-[var(--color-primary)] hover:underline mb-3"
>
{showStructured ? "▼ 정리된 내용 숨기기" : "▶ 정리된 내용 보기"}
</button>
{showStructured && (
<div className="bg-[var(--color-bg-card)] rounded-xl p-6 border border-[var(--color-border)] max-w-none">
<div className="structured-content text-sm leading-relaxed">
<ReactMarkdown
components={{
h1: ({children}) => <h1 className="text-xl font-bold mt-6 mb-3">{children}</h1>,
h2: ({children}) => <h2 className="text-lg font-bold mt-5 mb-2">{children}</h2>,
h3: ({children}) => <h3 className="text-base font-bold mt-4 mb-2">{children}</h3>,
p: ({children, node}) => {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const txt = node?.children?.map((c: any) => c.type === 'text' ? c.value : '').join('') || '';
return <p className="mb-3"><SpeakableText text={txt}>{children}</SpeakableText></p>;
},
ul: ({children}) => <ul className="list-disc ml-5 mb-3 space-y-1">{children}</ul>,
ol: ({children}) => <ol className="list-decimal ml-5 mb-3 space-y-1">{children}</ol>,
li: ({children, node}) => {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const txt = node?.children?.map((c: any) => c.type === 'text' ? c.value : '').join('') || '';
return <li className="leading-relaxed"><SpeakableText text={txt}>{children}</SpeakableText></li>;
},
strong: ({children}) => <strong className="font-bold">{children}</strong>,
blockquote: ({children}) => <blockquote className="border-l-2 border-[var(--color-primary)] pl-4 my-3 italic text-[var(--color-text-muted)]">{children}</blockquote>,
code: ({children}) => <code className="bg-[var(--color-bg-hover)] px-1.5 py-0.5 rounded text-xs">{children}</code>,
}}
>
{item.STRUCTURED_CONTENT}
</ReactMarkdown>
</div>
</div>
)}
</div>
)}
{/* Chunks toggle */}
{item.STATUS === "READY" && (
<div className="mb-6">
<button
onClick={handleToggleChunks}
className="text-sm text-[var(--color-primary)] hover:underline"
>
{showChunks ? "▼ 청크 숨기기" : "▶ 청크 보기"}
</button>
{showChunks && (
<div className="mt-3 space-y-3">
{chunks.length === 0 ? (
<p className="text-sm text-[var(--color-text-muted)]">Loading chunks...</p>
) : (
chunks.map((chunk) => (
<div
key={chunk.ID}
className="bg-[var(--color-bg-card)] rounded-lg p-4 border border-[var(--color-border)]"
>
<div className="flex justify-between items-center mb-2">
<span className="text-xs text-[var(--color-text-muted)]">
Chunk #{chunk.CHUNK_INDEX}
</span>
<span className="text-xs text-[var(--color-text-muted)]">
~{chunk.TOKEN_COUNT} tokens
</span>
</div>
<p className="text-sm whitespace-pre-wrap">{chunk.CONTENT}</p>
</div>
))
)}
</div>
)}
</div>
)}
{/* Actions */}
<div className="pt-4 border-t border-[var(--color-border)] flex items-center gap-4">
{item.STATUS === "READY" && (
<>
<button
onClick={async () => {
setStructuring(true);
setShowStructured(true);
try {
// 비동기 요청: 서버가 STRUCTURING 상태로 변경 후 백그라운드 처리
// 폴링이 자동으로 진행 상태를 갱신함
await request({
method: "POST",
url: `/api/knowledge/${id}/structure`,
data: {},
});
// 완료 후 최종 상태 갱신
await fetchItem();
} catch (err) {
console.error("Failed to structure:", err);
alert("내용 정리에 실패했습니다.");
await fetchItem();
} finally {
setStructuring(false);
}
}}
disabled={structuring}
className="text-sm text-[var(--color-primary)] hover:underline disabled:opacity-40"
>
{structuring ? "정리 중..." : item.STRUCTURED_CONTENT ? "다시 정리하기" : "내용 정리하기"}
</button>
<button
onClick={async () => {
setGenerating(true);
try {
const result = await request<{ generated: number }>({
method: "POST",
url: `/api/study-cards/generate/${id}`,
});
alert(`${result.generated}개의 스터디 카드가 생성되었습니다.`);
} catch (err) {
console.error("Failed to generate cards:", err);
alert("카드 생성에 실패했습니다.");
} finally {
setGenerating(false);
}
}}
disabled={generating}
className="text-sm text-[var(--color-primary)] hover:underline disabled:opacity-40"
>
{generating ? "카드 생성 중..." : "스터디 카드 생성"}
</button>
</>
)}
<button
onClick={handleDelete}
disabled={deleting}
className="text-sm text-red-400 hover:text-red-300 disabled:opacity-40"
>
{deleting ? "삭제 중..." : "삭제"}
</button>
</div>
</main>
</AuthGuard>
);
}