The TOC was displaying sections from previously viewed articles when navigating between posts. This happened because the DOM query for headings ran before Next.js finished updating the page content. Changes to components/post-toc.tsx: - Clear items and activeId immediately when pathname changes - Add 50ms delay before querying DOM for new headings - Properly handle IntersectionObserver cleanup with timeout This ensures the TOC always shows the correct headings for the current article, not the previous one. 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
159 lines
4.7 KiB
TypeScript
159 lines
4.7 KiB
TypeScript
'use client';
|
|
|
|
import { useEffect, useRef, useState } from 'react';
|
|
import { usePathname } from 'next/navigation';
|
|
import { FiList } from 'react-icons/fi';
|
|
|
|
interface TocItem {
|
|
id: string;
|
|
text: string;
|
|
depth: number;
|
|
}
|
|
|
|
export function PostToc({ onLinkClick }: { onLinkClick?: () => void }) {
|
|
const [items, setItems] = useState<TocItem[]>([]);
|
|
const [activeId, setActiveId] = useState<string | null>(null);
|
|
const listRef = useRef<HTMLDivElement | null>(null);
|
|
const itemRefs = useRef<Record<string, HTMLDivElement | null>>({});
|
|
const [indicator, setIndicator] = useState({ top: 0, opacity: 0 });
|
|
const pathname = usePathname();
|
|
|
|
useEffect(() => {
|
|
// Clear items immediately when pathname changes
|
|
setItems([]);
|
|
setActiveId(null);
|
|
|
|
let observer: IntersectionObserver | null = null;
|
|
|
|
// Small delay to ensure DOM has been updated with new article content
|
|
const timeoutId = setTimeout(() => {
|
|
const headings = Array.from(
|
|
document.querySelectorAll<HTMLElement>('article h2, article h3')
|
|
);
|
|
const mapped = headings
|
|
.filter((el) => el.id)
|
|
.map((el) => ({
|
|
id: el.id,
|
|
text: el.innerText,
|
|
depth: el.tagName === 'H3' ? 3 : 2
|
|
}));
|
|
setItems(mapped);
|
|
|
|
observer = new IntersectionObserver(
|
|
(entries) => {
|
|
entries.forEach((entry) => {
|
|
if (entry.isIntersecting) {
|
|
const id = (entry.target as HTMLElement).id;
|
|
if (id) {
|
|
setActiveId(id);
|
|
}
|
|
}
|
|
});
|
|
},
|
|
{
|
|
// Trigger when heading is in upper 40% of viewport
|
|
rootMargin: '0px 0px -60% 0px',
|
|
threshold: 0.1
|
|
}
|
|
);
|
|
|
|
headings.forEach((el) => observer?.observe(el));
|
|
}, 50); // 50ms delay to ensure DOM is updated
|
|
|
|
return () => {
|
|
clearTimeout(timeoutId);
|
|
if (observer) {
|
|
observer.disconnect();
|
|
}
|
|
};
|
|
}, [pathname]);
|
|
|
|
useEffect(() => {
|
|
if (!activeId || !listRef.current) {
|
|
setIndicator({ top: 0, opacity: 0 });
|
|
return;
|
|
}
|
|
const activeEl = itemRefs.current[activeId];
|
|
if (!activeEl) return;
|
|
const listTop = listRef.current.getBoundingClientRect().top;
|
|
const { top, height } = activeEl.getBoundingClientRect();
|
|
setIndicator({ top: top - listTop + height / 2, opacity: 1 });
|
|
}, [activeId, items.length]);
|
|
|
|
const handleClick = (id: string) => (event: React.MouseEvent<HTMLAnchorElement>) => {
|
|
event.preventDefault();
|
|
const el = document.getElementById(id);
|
|
if (!el) return;
|
|
|
|
el.scrollIntoView({
|
|
behavior: 'smooth',
|
|
block: 'start'
|
|
});
|
|
|
|
// Temporary highlight
|
|
el.classList.add('toc-target-highlight');
|
|
setTimeout(() => {
|
|
el.classList.remove('toc-target-highlight');
|
|
}, 700);
|
|
|
|
// Update hash without instant jump
|
|
if (history.replaceState) {
|
|
const url = new URL(window.location.href);
|
|
url.hash = id;
|
|
history.replaceState(null, '', url.toString());
|
|
}
|
|
|
|
// Trigger callback if provided (e.g. to close mobile menu)
|
|
if (onLinkClick) {
|
|
onLinkClick();
|
|
}
|
|
};
|
|
|
|
if (items.length === 0) return null;
|
|
|
|
return (
|
|
<nav className="not-prose sticky top-20 text-slate-500 dark:text-slate-400">
|
|
<div className="mb-2 inline-flex items-center gap-2 font-semibold text-slate-700 dark:text-slate-200">
|
|
<FiList className="h-4 w-4 text-slate-400" />
|
|
目錄
|
|
</div>
|
|
<div className="relative pl-4">
|
|
<span className="absolute left-1 top-0 h-full w-px bg-slate-200 dark:bg-slate-800" aria-hidden="true" />
|
|
<span
|
|
className="absolute left-0 h-3 w-3 -translate-y-1/2 rounded-full bg-accent transition-all duration-200 ease-snappy"
|
|
style={{ top: `${indicator.top}px`, opacity: indicator.opacity }}
|
|
aria-hidden="true"
|
|
/>
|
|
<div
|
|
ref={listRef}
|
|
className="space-y-1 text-[0.95rem]"
|
|
role="list"
|
|
>
|
|
{items.map((item) => (
|
|
<div
|
|
key={item.id}
|
|
ref={(el) => {
|
|
itemRefs.current[item.id] = el;
|
|
}}
|
|
role="listitem"
|
|
className={`relative ${item.depth === 3 ? 'pl-3' : 'pl-0'}`}
|
|
>
|
|
<a
|
|
href={`#${item.id}`}
|
|
onClick={handleClick(item.id)}
|
|
className={`line-clamp-2 inline-flex items-center pl-2 hover:text-blue-600 dark:hover:text-blue-400 ${
|
|
item.id === activeId
|
|
? 'text-blue-600 dark:text-blue-400 font-semibold'
|
|
: ''
|
|
}`}
|
|
>
|
|
{item.text}
|
|
</a>
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
</nav>
|
|
);
|
|
}
|