Compare commits
37 Commits
31b5821532
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
| 1d4cfe773c | |||
| 1f7dbd80d6 | |||
| b005f02b7b | |||
| 4cdccb0276 | |||
| ddd0cc5795 | |||
| 33042cde79 | |||
| 5325a08bc3 | |||
| 1b495d2d2d | |||
| efb57b691b | |||
| 08117a11c5 | |||
| 6ac6ea5545 | |||
| bdd42b9d26 | |||
| f7f2451357 | |||
| 8d08383391 | |||
| 1077c76366 | |||
| a09b7505be | |||
| 240d44842a | |||
| d5ea352775 | |||
| f185048abc | |||
| 8170fa0aa5 | |||
| fe28262ef4 | |||
| 7d85446ac5 | |||
| a4e88fa506 | |||
| 62d5973e1f | |||
| 42a1d3cbbe | |||
| d27cc01c87 | |||
| 8a4ecf9634 | |||
| 27dc2db3ee | |||
| fde17c2308 | |||
| 2402c94760 | |||
| 62090c7742 | |||
| a9bd56b658 | |||
| e8666d19ee | |||
| 2229f6bb6f | |||
| ed63ec7d9a | |||
| 5a80567117 | |||
| 614d8dd5aa |
@@ -50,14 +50,19 @@ Ask the user if they want to preview with `npm run dev` before publishing.
|
||||
|
||||
## Step 5: Publish
|
||||
|
||||
Execute the two-step deployment:
|
||||
**IMPORTANT**: The `.gitmodules` URL for `content/` points to GitHub. The CI/CD server clones the submodule from that URL, so the content submodule **must be pushed to GitHub first** before pushing the main repo. Otherwise the server will check out stale content and posts will disappear from the site.
|
||||
|
||||
Execute the deployment in order:
|
||||
|
||||
```bash
|
||||
# 1. Commit and push content submodule
|
||||
git -C content add . && git -C content commit -m "Add new post: <title>" && git -C content push
|
||||
# 1. Commit content submodule
|
||||
git -C content add . && git -C content commit -m "Add new post: <title>"
|
||||
|
||||
# 2. Update main repo submodule pointer and push (triggers CI/CD)
|
||||
git add content && git commit -m "Update content submodule" && git push
|
||||
# 2. Push content submodule to ALL remotes (GitHub first — CI/CD depends on it)
|
||||
git -C content push github main && git -C content push origin main
|
||||
|
||||
# 3. Update main repo submodule pointer, commit, and push to both remotes
|
||||
git add content && git commit -m "Update content submodule" && git push origin main && git push github main
|
||||
```
|
||||
|
||||
Confirm both pushes succeeded. The CI/CD pipeline on git.gbanyan.net will handle deployment automatically (and crontab mirrors to gitea.gbanyan.net).
|
||||
Confirm all pushes succeeded. The CI/CD pipeline on git.gbanyan.net will handle deployment automatically (and crontab mirrors to gitea.gbanyan.net).
|
||||
|
||||
@@ -33,3 +33,9 @@ NEXT_PUBLIC_TWITTER_CARD_TYPE="summary_large_image"
|
||||
|
||||
# Analytics (public ID only)
|
||||
NEXT_PUBLIC_ANALYTICS_ID=""
|
||||
|
||||
# Server-side only (NOT exposed to browser)
|
||||
# Used to fetch GitHub repositories for the /projects page.
|
||||
# Copy these into your local `.env.local` and fill in real values.
|
||||
GITHUB_USERNAME="your-github-username"
|
||||
GITHUB_TOKEN="your-github-token"
|
||||
|
||||
2
.playwright-mcp/console-2026-02-06T06-53-10-406Z.log
Normal file
2
.playwright-mcp/console-2026-02-06T06-53-10-406Z.log
Normal file
@@ -0,0 +1,2 @@
|
||||
[ 1800ms] [WARNING] Image with src "https://www.gravatar.com/avatar/53f2a6e011d5ececcd0d6e33e8e7329f60efcd23a5b77ba382273d5060a2cffe?s=160&d=identicon" was detected as the Largest Contentful Paint (LCP). Please add the `loading="eager"` property if this image is above the fold.
|
||||
Read more: https://nextjs.org/docs/app/api-reference/components/image#loading @ http://localhost:3000/_next/static/chunks/node_modules_next_dist_f3530cac._.js:2297
|
||||
5
.playwright-mcp/console-2026-02-06T06-53-49-813Z.log
Normal file
5
.playwright-mcp/console-2026-02-06T06-53-49-813Z.log
Normal file
@@ -0,0 +1,5 @@
|
||||
[ 22376ms] [ERROR] WebSocket connection to 'ws://localhost:3000/_next/webpack-hmr?id=VmGWkPt9sasaNk_cqnGHG' failed: @ http://localhost:3000/_next/static/chunks/node_modules_next_dist_client_17643121._.js:10878
|
||||
[ 23381ms] [ERROR] WebSocket connection to 'ws://localhost:3000/_next/webpack-hmr?id=VmGWkPt9sasaNk_cqnGHG' failed: Error in connection establishment: net::ERR_CONNECTION_REFUSED @ http://localhost:3000/_next/static/chunks/node_modules_next_dist_client_17643121._.js:10878
|
||||
[ 24384ms] [ERROR] WebSocket connection to 'ws://localhost:3000/_next/webpack-hmr?id=VmGWkPt9sasaNk_cqnGHG' failed: Error in connection establishment: net::ERR_CONNECTION_REFUSED @ http://localhost:3000/_next/static/chunks/node_modules_next_dist_client_17643121._.js:10878
|
||||
[ 25385ms] [ERROR] WebSocket connection to 'ws://localhost:3000/_next/webpack-hmr?id=VmGWkPt9sasaNk_cqnGHG' failed: Error in connection establishment: net::ERR_CONNECTION_REFUSED @ http://localhost:3000/_next/static/chunks/node_modules_next_dist_client_17643121._.js:10878
|
||||
[ 26387ms] [ERROR] WebSocket connection to 'ws://localhost:3000/_next/webpack-hmr?id=VmGWkPt9sasaNk_cqnGHG' failed: Error in connection establishment: net::ERR_CONNECTION_REFUSED @ http://localhost:3000/_next/static/chunks/node_modules_next_dist_client_17643121._.js:10878
|
||||
2
.playwright-mcp/console-2026-02-06T07-09-29-900Z.log
Normal file
2
.playwright-mcp/console-2026-02-06T07-09-29-900Z.log
Normal file
@@ -0,0 +1,2 @@
|
||||
[ 789ms] [WARNING] Image with src "https://www.gravatar.com/avatar/53f2a6e011d5ececcd0d6e33e8e7329f60efcd23a5b77ba382273d5060a2cffe?s=160&d=identicon" was detected as the Largest Contentful Paint (LCP). Please add the `loading="eager"` property if this image is above the fold.
|
||||
Read more: https://nextjs.org/docs/app/api-reference/components/image#loading @ http://localhost:3000/_next/static/chunks/node_modules_next_dist_f3530cac._.js:2297
|
||||
1
.playwright-mcp/console-2026-02-06T07-11-06-366Z.log
Normal file
1
.playwright-mcp/console-2026-02-06T07-11-06-366Z.log
Normal file
@@ -0,0 +1 @@
|
||||
[ 1801237ms] [WARNING] The resource https://blog.gbanyan.net/_next/static/chunks/457c1f08a48523ea.css was preloaded using link preload but not used within a few seconds from the window's load event. Please make sure it has an appropriate `as` value and it is preloaded intentionally. @ https://blog.gbanyan.net/blog/not-all-tears-should-be-summarized:0
|
||||
50
CLAUDE.md
50
CLAUDE.md
@@ -70,3 +70,53 @@ Pushing only to `content/` (personal-blog) does NOT trigger deployment. The main
|
||||
## Language
|
||||
|
||||
The site's default locale is `zh-TW`. UI text, labels, and timestamps are in Traditional Chinese.
|
||||
|
||||
## Design Context
|
||||
|
||||
### Users
|
||||
|
||||
- **Medical professionals & students**: Seek clinical insights, case studies, and medical education content
|
||||
- **General public**: Interested inpersonal reflections, medicine explainedaccessibly, and lifestyle content
|
||||
- **Tech enthusiasts & developers**: Drawn to HomeLab, technical tutorials, and developer environment content
|
||||
- **Patients & advocates**: Those with similar conditions (Usher syndrome, hearing/vision impairments) seeking understanding and community
|
||||
|
||||
**Context**: Readers visit for deep, reflective content—often in quiet environments, seeking to learn, reflect, or connect with personal experiences. They value clarity, authenticity, and quality over speed.
|
||||
|
||||
**Job to be done**: Gain meaningful knowledge, find resonance with personal experiences, understand complex topics (medical/technical) in approachable terms.
|
||||
|
||||
### Brand Personality
|
||||
|
||||
- **Voice**: Reflective, professional, and thoughtful—like a trusted physician who also happens to be a developer
|
||||
- **3-word personality**: Professional & refined, Thoughtful & reflective, Technical & practical, Approachable & human
|
||||
- **Emotional goals**: Calm & contemplative, Inspired & curious
|
||||
|
||||
**Not**: Corporate, salesy, alarmist (like news sites), or overly technical/clinical.
|
||||
|
||||
### Aesthetic Direction
|
||||
|
||||
**Visual tone**: Warm & organic with academic & scholarly sensibility, combined with modern technical clarity
|
||||
|
||||
**References**:
|
||||
- Medium (medium.com): Readability-focused, minimal distractions, clean typography
|
||||
- Personal tech blogs: Individual personality, character, and hands-on authenticity
|
||||
- Library aesthetic: Quiet, thoughtful, knowledge-rich environment
|
||||
|
||||
**Anti-references** (explicitly avoid):
|
||||
- News sites: Cluttered, headline-focused, clickbait design
|
||||
- Social media feeds: Infinite scroll, attention-grabbing tactics, dopamine-driven design
|
||||
- Corporate/SaaS: Too polished, salesy, or uniform corporate branding
|
||||
- Dry technical docs: Lacking personality, purely functional
|
||||
|
||||
**Theme**: Both light and dark modes equally important—light for daytime readability, dark for late-night focused reading. Accent colors should be warm (avoid reds/yellows which feel urgent/alerting).
|
||||
|
||||
### Design Principles
|
||||
|
||||
1. **Calm-first design**: Space, breathing room, and typography hierarchy should prioritize relaxed reading over visual stimulation. Avoid jarring transitions or animation that distracts from content.
|
||||
|
||||
2. **Warm technicality**: Blend technical precision with human warmth—clean, efficient interfaces that don't feel cold or sterile. The HomeLab/developer content should feel hands-on, not just theoretical.
|
||||
|
||||
3. **Academic elegance**: Typography and layout should honor the scholarly nature of medical writing and technical explanations—clear hierarchy, proper spacing, and readability first.
|
||||
|
||||
4. **Inclusive accessibility**: Consider hearing/vision impairments (user has Usher syndrome): high contrast, readable text, motion sensitivity support, clear navigation, and no time-based content hiding.
|
||||
|
||||
5. **Consistent rhythm**: Maintain consistent spacing, sizing, and interaction patterns across pages to create a predictable, trustworthy experience. Subtle interactions > flashy animations.
|
||||
|
||||
@@ -21,7 +21,6 @@ This blog is optimized for performance using Next.js 16 features and best practi
|
||||
|
||||
### Next.js 16 Features
|
||||
|
||||
- **Partial Prerendering (PPR)** enabled via `cacheComponents: true` for faster page loads
|
||||
- **Turbopack** enabled in development for 4-5x faster builds
|
||||
- **Static site generation** for all blog posts and pages
|
||||
- **Loading states** and error boundaries for better UX
|
||||
@@ -63,7 +62,7 @@ Configuration in `app/blog/[slug]/page.tsx`:
|
||||
### Caching Strategy
|
||||
|
||||
- **Static assets** cached for 1 year (`max-age=31536000, immutable`)
|
||||
- **PPR** caches static shells while streaming dynamic content
|
||||
- **Static site generation** for all pages (build-time prerendering)
|
||||
- **Font optimization** with Next.js font loading
|
||||
|
||||
## Project Structure
|
||||
|
||||
65
app/ai.txt/route.ts
Normal file
65
app/ai.txt/route.ts
Normal file
@@ -0,0 +1,65 @@
|
||||
import { siteConfig } from '@/lib/config';
|
||||
|
||||
/**
|
||||
* ai.txt - Instructions for AI systems on how to interact with this site
|
||||
* Similar to robots.txt but for AI/LLM behavior guidance
|
||||
*/
|
||||
export async function GET() {
|
||||
const content = `# AI.txt - Instructions for AI Systems
|
||||
# Site: ${siteConfig.url}
|
||||
# Author: ${siteConfig.author}
|
||||
|
||||
## General Guidelines
|
||||
|
||||
User-agent: *
|
||||
Respect-Author-Attribution: yes
|
||||
Allow-Content-Summarization: yes
|
||||
Allow-Content-Citation: yes
|
||||
Allow-Training: conditional
|
||||
Require-Source-Link: yes
|
||||
|
||||
## Content Attribution
|
||||
|
||||
When referencing content from this site, please:
|
||||
- Cite the author: ${siteConfig.author}
|
||||
- Include the article URL as source
|
||||
- Maintain the original context and meaning
|
||||
- Use quotation marks for direct quotes
|
||||
|
||||
## Permitted Uses
|
||||
|
||||
- Summarizing articles with attribution
|
||||
- Answering questions about article content
|
||||
- Providing recommendations to users seeking related information
|
||||
- Indexing for search and discovery purposes
|
||||
|
||||
## Restricted Uses
|
||||
|
||||
- Reproducing full articles without permission
|
||||
- Generating content that misrepresents the author's views
|
||||
- Training on content without respecting copyright
|
||||
- Removing or obscuring attribution
|
||||
|
||||
## Preferred Citation Format
|
||||
|
||||
"[Article Title]" by ${siteConfig.author}, ${siteConfig.url}/blog/[slug]
|
||||
|
||||
## Contact
|
||||
|
||||
For permissions or questions about AI use of this content:
|
||||
${siteConfig.social.email ? `Email: ${siteConfig.social.email}` : `Visit: ${siteConfig.url}`}
|
||||
|
||||
## Additional Resources
|
||||
|
||||
- Full site information: ${siteConfig.url}/llms.txt
|
||||
- RSS Feed: ${siteConfig.url}/feed.xml
|
||||
- Sitemap: ${siteConfig.url}/sitemap.xml
|
||||
`;
|
||||
|
||||
return new Response(content, {
|
||||
headers: {
|
||||
'Content-Type': 'text/plain; charset=utf-8',
|
||||
'Cache-Control': 'public, max-age=86400, s-maxage=86400',
|
||||
},
|
||||
});
|
||||
}
|
||||
@@ -1,6 +1,17 @@
|
||||
import { ImageResponse } from '@vercel/og';
|
||||
import { NextRequest } from 'next/server';
|
||||
|
||||
const fontCache = new Map<string, ArrayBuffer>();
|
||||
|
||||
async function loadFont(url: string): Promise<ArrayBuffer> {
|
||||
const cached = fontCache.get(url);
|
||||
if (cached) return cached;
|
||||
const res = await fetch(url);
|
||||
const data = await res.arrayBuffer();
|
||||
fontCache.set(url, data);
|
||||
return data;
|
||||
}
|
||||
|
||||
export async function GET(request: NextRequest) {
|
||||
try {
|
||||
const { searchParams } = new URL(request.url);
|
||||
@@ -10,7 +21,15 @@ export async function GET(request: NextRequest) {
|
||||
const description = searchParams.get('description') || '';
|
||||
const tags = searchParams.get('tags')?.split(',').slice(0, 3) || [];
|
||||
|
||||
return new ImageResponse(
|
||||
// Load CJK font for Chinese text rendering
|
||||
const fontData = await loadFont(
|
||||
'https://cdn.jsdelivr.net/fontsource/fonts/noto-sans-tc@latest/chinese-traditional-400-normal.woff'
|
||||
);
|
||||
const fontBoldData = await loadFont(
|
||||
'https://cdn.jsdelivr.net/fontsource/fonts/noto-sans-tc@latest/chinese-traditional-700-normal.woff'
|
||||
);
|
||||
|
||||
const imageResponse = new ImageResponse(
|
||||
(
|
||||
<div
|
||||
style={{
|
||||
@@ -20,6 +39,7 @@ export async function GET(request: NextRequest) {
|
||||
flexDirection: 'column',
|
||||
alignItems: 'flex-start',
|
||||
justifyContent: 'space-between',
|
||||
fontFamily: '"Noto Sans TC", sans-serif',
|
||||
backgroundColor: '#0f172a',
|
||||
backgroundImage: 'radial-gradient(circle at 25px 25px, #1e293b 2%, transparent 0%), radial-gradient(circle at 75px 75px, #1e293b 2%, transparent 0%)',
|
||||
backgroundSize: '100px 100px',
|
||||
@@ -155,8 +175,20 @@ export async function GET(request: NextRequest) {
|
||||
{
|
||||
width: 1200,
|
||||
height: 630,
|
||||
fonts: [
|
||||
{ name: 'Noto Sans TC', data: fontData, weight: 400 as const, style: 'normal' as const },
|
||||
{ name: 'Noto Sans TC', data: fontBoldData, weight: 700 as const, style: 'normal' as const },
|
||||
],
|
||||
}
|
||||
);
|
||||
|
||||
// Wrap response with cache headers for OG images (cache for 1 hour)
|
||||
return new Response(imageResponse.body, {
|
||||
headers: {
|
||||
'Content-Type': 'image/png',
|
||||
'Cache-Control': 'public, max-age=3600, s-maxage=3600, stale-while-revalidate=86400',
|
||||
},
|
||||
});
|
||||
} catch (e: any) {
|
||||
console.error('Error generating OG image:', e);
|
||||
return new Response(`Failed to generate image: ${e.message}`, {
|
||||
|
||||
@@ -3,7 +3,7 @@ import Image from 'next/image';
|
||||
import { notFound } from 'next/navigation';
|
||||
import type { Metadata } from 'next';
|
||||
import { allPosts } from 'contentlayer2/generated';
|
||||
import { getPostBySlug, getRelatedPosts, getPostNeighbors } from '@/lib/posts';
|
||||
import { getPostBySlug, getRelatedPosts, getPostNeighbors, getTagSlug } from '@/lib/posts';
|
||||
import { siteConfig } from '@/lib/config';
|
||||
import { ReadingProgress } from '@/components/reading-progress';
|
||||
import { ScrollReveal } from '@/components/scroll-reveal';
|
||||
@@ -13,6 +13,7 @@ import { PostStorylineNav } from '@/components/post-storyline-nav';
|
||||
import { SectionDivider } from '@/components/section-divider';
|
||||
import { FooterCue } from '@/components/footer-cue';
|
||||
import { JsonLd } from '@/components/json-ld';
|
||||
import { MermaidRenderer } from '@/components/mermaid-renderer';
|
||||
|
||||
export function generateStaticParams() {
|
||||
const params = allPosts.map((post) => ({
|
||||
@@ -39,19 +40,33 @@ export async function generateMetadata({ params }: Props): Promise<Metadata> {
|
||||
ogImageUrl.searchParams.set('tags', post.tags.slice(0, 3).join(','));
|
||||
}
|
||||
|
||||
// Prefer post's feature_image for social cards; fall back to dynamic OG
|
||||
const imageUrl = post.feature_image
|
||||
? `${siteConfig.url}${post.feature_image.replace('../assets', '/assets')}`
|
||||
: ogImageUrl.toString();
|
||||
|
||||
return {
|
||||
title: post.title,
|
||||
description: post.description || post.title,
|
||||
authors: post.authors?.length ? post.authors.map(author => ({ name: author })) : [{ name: siteConfig.author }],
|
||||
robots: {
|
||||
index: true,
|
||||
follow: true,
|
||||
googleBot: {
|
||||
index: true,
|
||||
follow: true,
|
||||
},
|
||||
},
|
||||
openGraph: {
|
||||
title: post.title,
|
||||
description: post.description || post.title,
|
||||
type: 'article',
|
||||
publishedTime: post.published_at,
|
||||
authors: post.authors,
|
||||
authors: post.authors?.length ? post.authors : [siteConfig.author],
|
||||
tags: post.tags,
|
||||
images: [
|
||||
{
|
||||
url: ogImageUrl.toString(),
|
||||
url: imageUrl,
|
||||
width: 1200,
|
||||
height: 630,
|
||||
alt: post.title,
|
||||
@@ -62,7 +77,7 @@ export async function generateMetadata({ params }: Props): Promise<Metadata> {
|
||||
card: 'summary_large_image',
|
||||
title: post.title,
|
||||
description: post.description || post.title,
|
||||
images: [ogImageUrl.toString()],
|
||||
images: [imageUrl],
|
||||
},
|
||||
};
|
||||
}
|
||||
@@ -96,6 +111,11 @@ export default async function BlogPostPage({ params }: Props) {
|
||||
? `${siteConfig.url}${post.feature_image.replace('../assets', '/assets')}`
|
||||
: ogImageUrl.toString();
|
||||
|
||||
// Estimate word count and reading time
|
||||
const textContent = post.body?.raw || '';
|
||||
const wordCount = textContent.split(/\s+/).filter(Boolean).length;
|
||||
const readingTime = Math.ceil(wordCount / 200);
|
||||
|
||||
// BlogPosting Schema
|
||||
const blogPostingSchema = {
|
||||
'@context': 'https://schema.org',
|
||||
@@ -126,10 +146,28 @@ export default async function BlogPostPage({ params }: Props) {
|
||||
keywords: post.tags.join(', '),
|
||||
articleSection: post.tags[0],
|
||||
}),
|
||||
inLanguage: siteConfig.defaultLocale,
|
||||
...(wordCount > 0 && {
|
||||
wordCount: wordCount,
|
||||
readingTime: `${readingTime} min read`,
|
||||
}),
|
||||
url: postUrl,
|
||||
};
|
||||
|
||||
// Speakable Schema for AEO
|
||||
const speakableSchema = {
|
||||
'@context': 'https://schema.org',
|
||||
'@type': 'SpeakableSpecification',
|
||||
speakable: {
|
||||
'@type': 'CSSSelector',
|
||||
selector: [
|
||||
'article[data-toc-content]',
|
||||
'.prose h2',
|
||||
'.prose h3',
|
||||
'.prose p',
|
||||
],
|
||||
},
|
||||
};
|
||||
|
||||
// BreadcrumbList Schema
|
||||
const breadcrumbSchema = {
|
||||
'@context': 'https://schema.org',
|
||||
@@ -160,6 +198,7 @@ export default async function BlogPostPage({ params }: Props) {
|
||||
<>
|
||||
<JsonLd data={blogPostingSchema} />
|
||||
<JsonLd data={breadcrumbSchema} />
|
||||
<JsonLd data={speakableSchema} />
|
||||
<ReadingProgress />
|
||||
<PostLayout hasToc={hasToc} contentKey={slug}>
|
||||
<div className="space-y-8">
|
||||
@@ -183,10 +222,8 @@ export default async function BlogPostPage({ params }: Props) {
|
||||
{post.tags.map((t) => (
|
||||
<Link
|
||||
key={t}
|
||||
href={`/tags/${encodeURIComponent(
|
||||
t.toLowerCase().replace(/\s+/g, '-')
|
||||
)}`}
|
||||
className="tag-chip rounded-full bg-accent-soft px-3 py-1 text-sm text-accent-textLight dark:bg-slate-800 dark:text-slate-100"
|
||||
href={`/tags/${encodeURIComponent(getTagSlug(t))}`}
|
||||
className="tag-chip rounded-full bg-accent-soft px-3 py-1 text-sm text-accent-textLight dark:bg-slate-800 dark:text-slate-100 dark:hover:bg-slate-700 dark:hover:text-white"
|
||||
>
|
||||
#{t}
|
||||
</Link>
|
||||
@@ -217,6 +254,7 @@ export default async function BlogPostPage({ params }: Props) {
|
||||
</div>
|
||||
)}
|
||||
<div dangerouslySetInnerHTML={{ __html: post.body.html }} />
|
||||
<MermaidRenderer />
|
||||
</article>
|
||||
</ScrollReveal>
|
||||
</SectionDivider>
|
||||
|
||||
@@ -1,26 +1,62 @@
|
||||
import Link from 'next/link';
|
||||
import { getAllPostsSorted } from '@/lib/posts';
|
||||
import { PostListWithControls } from '@/components/post-list-with-controls';
|
||||
import { TimelineWrapper } from '@/components/timeline-wrapper';
|
||||
import { SidebarLayout } from '@/components/sidebar-layout';
|
||||
import { SectionDivider } from '@/components/section-divider';
|
||||
import { ScrollReveal } from '@/components/scroll-reveal';
|
||||
import { FiTrendingUp } from 'react-icons/fi';
|
||||
import { siteConfig } from '@/lib/config';
|
||||
import { JsonLd } from '@/components/json-ld';
|
||||
|
||||
export const metadata = {
|
||||
title: '所有文章'
|
||||
title: '所有文章',
|
||||
description: '瀏覽所有文章,持續更新中。',
|
||||
alternates: {
|
||||
canonical: `${siteConfig.url}/blog`
|
||||
}
|
||||
};
|
||||
|
||||
export default function BlogIndexPage() {
|
||||
const posts = getAllPostsSorted();
|
||||
|
||||
// Blog schema
|
||||
const blogSchema = {
|
||||
'@context': 'https://schema.org',
|
||||
'@type': 'Blog',
|
||||
name: '所有文章',
|
||||
description: '瀏覽所有文章,持續更新中。',
|
||||
url: `${siteConfig.url}/blog`,
|
||||
inLanguage: siteConfig.defaultLocale,
|
||||
blogPost: posts.slice(0, 10).map((post) => ({
|
||||
'@type': 'BlogPosting',
|
||||
headline: post.title,
|
||||
url: `${siteConfig.url}${post.url}`,
|
||||
datePublished: post.published_at,
|
||||
dateModified: post.updated_at || post.published_at,
|
||||
author: {
|
||||
'@type': 'Person',
|
||||
name: siteConfig.author
|
||||
}
|
||||
}))
|
||||
};
|
||||
|
||||
return (
|
||||
<section className="space-y-4">
|
||||
<JsonLd data={blogSchema} />
|
||||
<SidebarLayout>
|
||||
<header className="space-y-1">
|
||||
<h1 className="type-title font-semibold text-slate-900 dark:text-slate-50">
|
||||
所有文章
|
||||
</h1>
|
||||
<p className="type-small text-slate-500 dark:text-slate-400">
|
||||
繼續往下滑,慢慢逛逛。
|
||||
</p>
|
||||
</header>
|
||||
<SectionDivider>
|
||||
<ScrollReveal>
|
||||
<header className="space-y-1">
|
||||
<h1 className="type-title font-semibold text-slate-900 dark:text-slate-50">
|
||||
所有文章
|
||||
</h1>
|
||||
<p className="type-small text-slate-500 dark:text-slate-400">
|
||||
繼續往下滑,慢慢逛逛。
|
||||
</p>
|
||||
</header>
|
||||
</ScrollReveal>
|
||||
</SectionDivider>
|
||||
<PostListWithControls posts={posts} />
|
||||
</SidebarLayout>
|
||||
</section>
|
||||
|
||||
@@ -1,10 +1,14 @@
|
||||
import '../styles/globals.css';
|
||||
import type { Metadata } from 'next';
|
||||
import { siteConfig } from '@/lib/config';
|
||||
import { getAllPostsSorted } from '@/lib/posts';
|
||||
import { LayoutShell } from '@/components/layout-shell';
|
||||
import { ThemeProvider } from 'next-themes';
|
||||
import { Playfair_Display } from 'next/font/google';
|
||||
import { Playfair_Display, LXGW_WenKai_TC } from 'next/font/google';
|
||||
import { JsonLd } from '@/components/json-ld';
|
||||
import { WebVitals } from '@/components/web-vitals';
|
||||
import { ViewTransitionProvider } from '@/components/view-transition-provider';
|
||||
import NextTopLoader from 'nextjs-toploader';
|
||||
|
||||
const playfair = Playfair_Display({
|
||||
subsets: ['latin'],
|
||||
@@ -12,6 +16,15 @@ const playfair = Playfair_Display({
|
||||
display: 'swap',
|
||||
});
|
||||
|
||||
const lxgwWenKai = LXGW_WenKai_TC({
|
||||
weight: ['400', '700'],
|
||||
subsets: ['latin'],
|
||||
variable: '--font-serif-cn',
|
||||
display: 'swap',
|
||||
preload: true,
|
||||
adjustFontFallback: false,
|
||||
});
|
||||
|
||||
export const metadata: Metadata = {
|
||||
title: {
|
||||
default: siteConfig.title,
|
||||
@@ -19,22 +32,44 @@ export const metadata: Metadata = {
|
||||
},
|
||||
description: siteConfig.description,
|
||||
metadataBase: new URL(siteConfig.url),
|
||||
creator: siteConfig.author,
|
||||
robots: {
|
||||
index: true,
|
||||
follow: true,
|
||||
googleBot: {
|
||||
index: true,
|
||||
follow: true,
|
||||
'max-video-preview': -1,
|
||||
'max-image-preview': 'large',
|
||||
'max-snippet': -1
|
||||
}
|
||||
},
|
||||
openGraph: {
|
||||
type: 'website',
|
||||
title: siteConfig.title,
|
||||
description: siteConfig.description,
|
||||
url: siteConfig.url,
|
||||
siteName: siteConfig.title,
|
||||
images: [siteConfig.ogImage]
|
||||
locale: siteConfig.defaultLocale,
|
||||
images: [
|
||||
{
|
||||
url: siteConfig.ogImage,
|
||||
width: 1200,
|
||||
height: 630,
|
||||
alt: siteConfig.title
|
||||
}
|
||||
]
|
||||
},
|
||||
twitter: {
|
||||
card: siteConfig.twitterCard,
|
||||
site: siteConfig.social.twitter || undefined,
|
||||
creator: siteConfig.social.twitter || undefined,
|
||||
title: siteConfig.title,
|
||||
description: siteConfig.description,
|
||||
images: [siteConfig.ogImage]
|
||||
},
|
||||
icons: {
|
||||
icon: '/favicon.png'
|
||||
icon: '/favicon.png',
|
||||
apple: '/favicon.png'
|
||||
},
|
||||
alternates: {
|
||||
types: {
|
||||
@@ -43,14 +78,16 @@ export const metadata: Metadata = {
|
||||
}
|
||||
};
|
||||
|
||||
export default function RootLayout({
|
||||
export default async function RootLayout({
|
||||
children
|
||||
}: {
|
||||
children: React.ReactNode;
|
||||
}) {
|
||||
const theme = siteConfig.theme;
|
||||
const recentPosts = getAllPostsSorted()
|
||||
.slice(0, 5)
|
||||
.map((p) => ({ title: p.title, url: p.url }));
|
||||
|
||||
// WebSite Schema
|
||||
const websiteSchema = {
|
||||
'@context': 'https://schema.org',
|
||||
'@type': 'WebSite',
|
||||
@@ -73,7 +110,6 @@ export default function RootLayout({
|
||||
},
|
||||
};
|
||||
|
||||
// Organization Schema
|
||||
const organizationSchema = {
|
||||
'@context': 'https://schema.org',
|
||||
'@type': 'Organization',
|
||||
@@ -87,13 +123,27 @@ export default function RootLayout({
|
||||
].filter(Boolean),
|
||||
};
|
||||
|
||||
|
||||
|
||||
return (
|
||||
<html lang={siteConfig.defaultLocale} suppressHydrationWarning className={playfair.variable}>
|
||||
<html lang={siteConfig.defaultLocale} suppressHydrationWarning className={`${playfair.variable} ${lxgwWenKai.variable}`}>
|
||||
<head>
|
||||
<link rel="preconnect" href="https://fonts.googleapis.com" />
|
||||
<link rel="preconnect" href="https://fonts.gstatic.com" crossOrigin="anonymous" />
|
||||
<link rel="font" href="https://fonts.googleapis.com" />
|
||||
<link rel="font" href="https://fonts.gstatic.com" />
|
||||
</head>
|
||||
<body>
|
||||
<NextTopLoader
|
||||
color={theme.accent}
|
||||
height={3}
|
||||
showSpinner={false}
|
||||
speed={200}
|
||||
shadow={`0 0 10px ${theme.accent}, 0 0 5px ${theme.accent}`}
|
||||
/>
|
||||
<JsonLd data={websiteSchema} />
|
||||
<JsonLd data={organizationSchema} />
|
||||
<style
|
||||
// Set CSS variables for accent colors (light + dark variants)
|
||||
dangerouslySetInnerHTML={{
|
||||
__html: `
|
||||
:root {
|
||||
@@ -103,11 +153,14 @@ export default function RootLayout({
|
||||
--color-accent-text-dark: ${theme.accentTextDark};
|
||||
}
|
||||
`
|
||||
}}
|
||||
/>
|
||||
<ThemeProvider attribute="class" defaultTheme="system" enableSystem>
|
||||
<LayoutShell>{children}</LayoutShell>
|
||||
</ThemeProvider>
|
||||
}}
|
||||
/>
|
||||
<ThemeProvider attribute="class" defaultTheme="system" enableSystem>
|
||||
<ViewTransitionProvider>
|
||||
<LayoutShell recentPosts={recentPosts}>{children}</LayoutShell>
|
||||
</ViewTransitionProvider>
|
||||
</ThemeProvider>
|
||||
<WebVitals />
|
||||
</body>
|
||||
</html>
|
||||
);
|
||||
|
||||
112
app/llms.txt/route.ts
Normal file
112
app/llms.txt/route.ts
Normal file
@@ -0,0 +1,112 @@
|
||||
import { siteConfig } from '@/lib/config';
|
||||
import { allPosts, allPages } from 'contentlayer2/generated';
|
||||
|
||||
/**
|
||||
* llms.txt - A proposed standard for providing LLM-readable site information
|
||||
* See: https://llmstxt.org/
|
||||
*
|
||||
* This file helps AI assistants understand the site structure, content, and purpose.
|
||||
*/
|
||||
export async function GET() {
|
||||
const siteUrl = siteConfig.url;
|
||||
|
||||
// Get published posts sorted by date
|
||||
const posts = allPosts
|
||||
.filter((post) => post.status === 'published')
|
||||
.sort((a, b) => {
|
||||
const dateA = a.published_at ? new Date(a.published_at).getTime() : 0;
|
||||
const dateB = b.published_at ? new Date(b.published_at).getTime() : 0;
|
||||
return dateB - dateA;
|
||||
})
|
||||
.slice(0, 50); // Latest 50 posts for context
|
||||
|
||||
// Get all published pages
|
||||
const pages = allPages.filter((page) => page.status === 'published');
|
||||
|
||||
// Extract unique tags
|
||||
const tags = Array.from(
|
||||
new Set(
|
||||
allPosts
|
||||
.filter((post) => post.status === 'published' && post.tags)
|
||||
.flatMap((post) => post.tags || [])
|
||||
)
|
||||
);
|
||||
|
||||
const content = `# ${siteConfig.name}
|
||||
|
||||
> ${siteConfig.description}
|
||||
|
||||
## Site Information
|
||||
|
||||
- **Author**: ${siteConfig.author}
|
||||
- **Language**: ${siteConfig.defaultLocale}
|
||||
- **URL**: ${siteUrl}
|
||||
|
||||
## About
|
||||
|
||||
${siteConfig.aboutShort}
|
||||
|
||||
## Content Overview
|
||||
|
||||
This personal blog contains articles about various topics including technology, software development, and personal insights.
|
||||
|
||||
### Topics Covered
|
||||
|
||||
${tags.map((tag) => `- ${tag}`).join('\n')}
|
||||
|
||||
## Recent Articles
|
||||
|
||||
${posts
|
||||
.map((post) => {
|
||||
const url = `${siteUrl}${post.url}`;
|
||||
const description = post.description || post.custom_excerpt || '';
|
||||
return `### ${post.title}
|
||||
|
||||
- **URL**: ${url}
|
||||
- **Published**: ${post.published_at || 'Unknown'}
|
||||
${description ? `- **Summary**: ${description}` : ''}
|
||||
${post.tags && post.tags.length > 0 ? `- **Tags**: ${post.tags.join(', ')}` : ''}
|
||||
`;
|
||||
})
|
||||
.join('\n')}
|
||||
|
||||
## Static Pages
|
||||
|
||||
${pages
|
||||
.map((page) => {
|
||||
const url = `${siteUrl}${page.url}`;
|
||||
return `- [${page.title}](${url})`;
|
||||
})
|
||||
.join('\n')}
|
||||
|
||||
## Navigation
|
||||
|
||||
- Homepage: ${siteUrl}
|
||||
- All Articles: ${siteUrl}/blog
|
||||
- Tags: ${siteUrl}/tags
|
||||
- RSS Feed: ${siteUrl}/feed.xml
|
||||
|
||||
## Contact & Social
|
||||
|
||||
${siteConfig.social.github ? `- GitHub: ${siteConfig.social.github}` : ''}
|
||||
${siteConfig.social.mastodon ? `- Mastodon: ${siteConfig.social.mastodon}` : ''}
|
||||
${siteConfig.social.twitter ? `- Twitter: ${siteConfig.social.twitter}` : ''}
|
||||
${siteConfig.social.email ? `- Email: ${siteConfig.social.email}` : ''}
|
||||
|
||||
## Usage Guidelines
|
||||
|
||||
This content is created by ${siteConfig.author} and may be cited with proper attribution. When referencing articles from this site:
|
||||
|
||||
1. Provide accurate summaries of the content
|
||||
2. Include the original URL as a source
|
||||
3. Respect the author's perspective and context
|
||||
4. Do not generate content that contradicts the author's views without clarification
|
||||
`;
|
||||
|
||||
return new Response(content, {
|
||||
headers: {
|
||||
'Content-Type': 'text/plain; charset=utf-8',
|
||||
'Cache-Control': 'public, max-age=86400, s-maxage=86400',
|
||||
},
|
||||
});
|
||||
}
|
||||
25
app/not-found.tsx
Normal file
25
app/not-found.tsx
Normal file
@@ -0,0 +1,25 @@
|
||||
import Link from 'next/link';
|
||||
|
||||
export default function NotFound() {
|
||||
return (
|
||||
<div className="flex min-h-[60vh] flex-col items-center justify-center px-4">
|
||||
<div className="max-w-md text-center">
|
||||
<h1 className="type-display mb-2 text-6xl font-bold text-slate-300 dark:text-slate-600">
|
||||
404
|
||||
</h1>
|
||||
<h2 className="mb-4 text-xl font-semibold text-slate-800 dark:text-slate-200">
|
||||
找不到頁面
|
||||
</h2>
|
||||
<p className="mb-8 text-slate-600 dark:text-slate-400">
|
||||
您造訪的連結可能已失效或不存在。
|
||||
</p>
|
||||
<Link
|
||||
href="/"
|
||||
className="inline-flex items-center rounded-lg bg-slate-800 px-6 py-3 text-sm font-medium text-white transition-colors hover:bg-slate-700 focus:outline-none focus:ring-2 focus:ring-slate-500 focus:ring-offset-2 dark:bg-slate-200 dark:text-slate-900 dark:hover:bg-slate-300"
|
||||
>
|
||||
返回首頁
|
||||
</Link>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
23
app/page.tsx
23
app/page.tsx
@@ -5,6 +5,7 @@ import { PostListItem } from '@/components/post-list-item';
|
||||
import { TimelineWrapper } from '@/components/timeline-wrapper';
|
||||
import { SidebarLayout } from '@/components/sidebar-layout';
|
||||
import { JsonLd } from '@/components/json-ld';
|
||||
import { HeroSection } from '@/components/hero-section';
|
||||
|
||||
export default function HomePage() {
|
||||
const posts = getAllPostsSorted().slice(0, siteConfig.postsPerPage);
|
||||
@@ -34,14 +35,13 @@ export default function HomePage() {
|
||||
<JsonLd data={collectionPageSchema} />
|
||||
<section className="space-y-6">
|
||||
<SidebarLayout>
|
||||
<header className="space-y-1 text-center">
|
||||
<h1 className="type-title font-bold text-slate-900 dark:text-slate-50">
|
||||
{siteConfig.name} 的最新動態
|
||||
</h1>
|
||||
<p className="type-small text-slate-600 dark:text-slate-300">
|
||||
{siteConfig.tagline}
|
||||
</p>
|
||||
</header>
|
||||
<h1 className="sr-only">
|
||||
{siteConfig.name} 的最新動態 — {siteConfig.tagline}
|
||||
</h1>
|
||||
<HeroSection
|
||||
title={`${siteConfig.name} 的最新動態`}
|
||||
tagline={siteConfig.tagline}
|
||||
/>
|
||||
|
||||
<div>
|
||||
<div className="mb-3 flex items-baseline justify-between">
|
||||
@@ -50,14 +50,15 @@ export default function HomePage() {
|
||||
</h2>
|
||||
<Link
|
||||
href="/blog"
|
||||
className="text-xs text-blue-600 hover:underline dark:text-blue-400"
|
||||
prefetch={true}
|
||||
className="text-xs text-accent hover:underline"
|
||||
>
|
||||
所有文章 →
|
||||
</Link>
|
||||
</div>
|
||||
<TimelineWrapper>
|
||||
{posts.map((post) => (
|
||||
<PostListItem key={post._id} post={post} />
|
||||
{posts.map((post, index) => (
|
||||
<PostListItem key={post._id} post={post} priority={index === 0} />
|
||||
))}
|
||||
</TimelineWrapper>
|
||||
</div>
|
||||
|
||||
@@ -3,13 +3,16 @@ import Image from 'next/image';
|
||||
import { notFound } from 'next/navigation';
|
||||
import type { Metadata } from 'next';
|
||||
import { allPages } from 'contentlayer2/generated';
|
||||
import { getPageBySlug } from '@/lib/posts';
|
||||
import { getPageBySlug, getTagSlug } from '@/lib/posts';
|
||||
import { siteConfig } from '@/lib/config';
|
||||
import { ReadingProgress } from '@/components/reading-progress';
|
||||
import { PostLayout } from '@/components/post-layout';
|
||||
import { ScrollReveal } from '@/components/scroll-reveal';
|
||||
import { SectionDivider } from '@/components/section-divider';
|
||||
import { JsonLd } from '@/components/json-ld';
|
||||
import { DevEnvDeviceHero } from '@/components/dev-env-device-hero';
|
||||
import { HomeLabDeviceHero } from '@/components/homelab-device-hero';
|
||||
import { MermaidRenderer } from '@/components/mermaid-renderer';
|
||||
|
||||
export function generateStaticParams() {
|
||||
const params = allPages.map((page) => ({
|
||||
@@ -27,9 +30,41 @@ export async function generateMetadata({ params }: Props): Promise<Metadata> {
|
||||
const page = getPageBySlug(slug);
|
||||
if (!page) return {};
|
||||
|
||||
const pageUrl = `${siteConfig.url}${page.url}`;
|
||||
|
||||
return {
|
||||
title: page.title,
|
||||
description: page.description || page.title
|
||||
description: page.description || page.title,
|
||||
alternates: {
|
||||
canonical: pageUrl
|
||||
},
|
||||
openGraph: {
|
||||
title: page.title,
|
||||
description: page.description || page.title,
|
||||
url: pageUrl,
|
||||
type: 'website',
|
||||
images: [
|
||||
page.feature_image
|
||||
? {
|
||||
url: `${siteConfig.url}${page.feature_image.replace('../assets', '/assets')}`,
|
||||
alt: page.title
|
||||
}
|
||||
: {
|
||||
url: `${siteConfig.url}${siteConfig.ogImage}`,
|
||||
alt: page.title
|
||||
}
|
||||
]
|
||||
},
|
||||
twitter: {
|
||||
card: siteConfig.twitterCard,
|
||||
title: page.title,
|
||||
description: page.description || page.title,
|
||||
images: [
|
||||
page.feature_image
|
||||
? page.feature_image.replace('../assets', '/assets')
|
||||
: siteConfig.ogImage
|
||||
]
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
@@ -75,11 +110,11 @@ export default async function StaticPage({ params }: Props) {
|
||||
<>
|
||||
<JsonLd data={webPageSchema} />
|
||||
<ReadingProgress />
|
||||
<PostLayout hasToc={hasToc} contentKey={slug}>
|
||||
<div className="space-y-8">
|
||||
<PostLayout hasToc={hasToc} contentKey={slug} wide={slug === 'dev-env' || slug === 'homelab'}>
|
||||
<div className={slug === 'dev-env' || slug === 'homelab' ? 'space-y-4' : 'space-y-8'}>
|
||||
<SectionDivider>
|
||||
<ScrollReveal>
|
||||
<header className="mb-6 space-y-4 text-center">
|
||||
<header className={slug === 'dev-env' || slug === 'homelab' ? 'mb-4 space-y-3 text-center' : 'mb-6 space-y-4 text-center'}>
|
||||
{page.published_at && (
|
||||
<p className="type-small text-slate-500 dark:text-slate-500">
|
||||
{new Date(page.published_at).toLocaleDateString(
|
||||
@@ -95,9 +130,7 @@ export default async function StaticPage({ params }: Props) {
|
||||
{page.tags.map((t) => (
|
||||
<Link
|
||||
key={t}
|
||||
href={`/tags/${encodeURIComponent(
|
||||
t.toLowerCase().replace(/\s+/g, '-')
|
||||
)}`}
|
||||
href={`/tags/${encodeURIComponent(getTagSlug(t))}`}
|
||||
className="tag-chip rounded-full bg-accent-soft px-3 py-1 text-sm text-accent-textLight dark:bg-slate-800 dark:text-slate-100"
|
||||
>
|
||||
#{t}
|
||||
@@ -115,20 +148,27 @@ export default async function StaticPage({ params }: Props) {
|
||||
data-toc-content={slug}
|
||||
className="prose prose-lg prose-slate mx-auto max-w-none dark:prose-invert"
|
||||
>
|
||||
{page.feature_image && (
|
||||
<div className="-mx-4 mb-8 transition-all duration-500 sm:-mx-12 lg:-mx-20 group-[.toc-open]:lg:-mx-4">
|
||||
<Image
|
||||
src={page.feature_image.replace('../assets', '/assets')}
|
||||
alt={page.title}
|
||||
width={1200}
|
||||
height={600}
|
||||
sizes="(max-width: 768px) 100vw, (max-width: 1200px) 90vw, 1200px"
|
||||
priority
|
||||
className="w-full rounded-xl shadow-lg"
|
||||
/>
|
||||
</div>
|
||||
{slug === 'dev-env' ? (
|
||||
<DevEnvDeviceHero />
|
||||
) : slug === 'homelab' ? (
|
||||
<HomeLabDeviceHero />
|
||||
) : (
|
||||
page.feature_image && (
|
||||
<div className="-mx-4 mb-8 transition-all duration-500 sm:-mx-12 lg:-mx-20 group-[.toc-open]:lg:-mx-4">
|
||||
<Image
|
||||
src={page.feature_image.replace('../assets', '/assets')}
|
||||
alt={page.title}
|
||||
width={1200}
|
||||
height={600}
|
||||
sizes="(max-width: 768px) 100vw, (max-width: 1200px) 90vw, 1200px"
|
||||
priority
|
||||
className="w-full rounded-xl shadow-lg"
|
||||
/>
|
||||
</div>
|
||||
)
|
||||
)}
|
||||
<div dangerouslySetInnerHTML={{ __html: page.body.html }} />
|
||||
<MermaidRenderer />
|
||||
</article>
|
||||
</ScrollReveal>
|
||||
</SectionDivider>
|
||||
|
||||
76
app/projects/page.tsx
Normal file
76
app/projects/page.tsx
Normal file
@@ -0,0 +1,76 @@
|
||||
import { FaGithub } from 'react-icons/fa';
|
||||
import { fetchPublicRepos } from '@/lib/github';
|
||||
import { SidebarLayout } from '@/components/sidebar-layout';
|
||||
import { RepoCard } from '@/components/repo-card';
|
||||
|
||||
import { siteConfig } from '@/lib/config';
|
||||
|
||||
export const revalidate = 3600;
|
||||
|
||||
export const metadata = {
|
||||
title: 'GitHub 專案',
|
||||
description: '從我的 GitHub 帳號自動抓取公開的程式庫與專案。',
|
||||
alternates: {
|
||||
canonical: `${siteConfig.url}/projects`
|
||||
},
|
||||
openGraph: {
|
||||
title: 'GitHub 專案',
|
||||
description: '從我的 GitHub 帳號自動抓取公開的程式庫與專案。',
|
||||
url: `${siteConfig.url}/projects`,
|
||||
type: 'website',
|
||||
images: [
|
||||
{
|
||||
url: `${siteConfig.url}${siteConfig.ogImage}`,
|
||||
alt: 'GitHub 專案'
|
||||
}
|
||||
]
|
||||
},
|
||||
twitter: {
|
||||
card: siteConfig.twitterCard,
|
||||
title: 'GitHub 專案',
|
||||
description: '從我的 GitHub 帳號自動抓取公開的程式庫與專案。',
|
||||
images: [siteConfig.ogImage]
|
||||
}
|
||||
};
|
||||
|
||||
export default async function ProjectsPage() {
|
||||
const repos = await fetchPublicRepos();
|
||||
|
||||
return (
|
||||
<section className="space-y-4">
|
||||
<SidebarLayout>
|
||||
<header className="space-y-1">
|
||||
<h1 className="type-title font-semibold text-slate-900 dark:text-slate-50">
|
||||
GitHub 專案
|
||||
</h1>
|
||||
<p className="type-small text-slate-500 dark:text-slate-400">
|
||||
從我的 GitHub 帳號自動抓取公開的程式庫與專案。
|
||||
{repos.length > 0 && (
|
||||
<span className="ml-1">共 {repos.length} 個專案</span>
|
||||
)}
|
||||
</p>
|
||||
</header>
|
||||
|
||||
{repos.length === 0 ? (
|
||||
<div className="mt-6 flex flex-col items-center gap-3 rounded-2xl border border-dashed border-slate-200 bg-slate-50/50 p-8 text-center dark:border-slate-700 dark:bg-slate-900/30">
|
||||
<FaGithub className="h-12 w-12 text-slate-400 dark:text-slate-500" />
|
||||
<p className="type-small text-slate-500 dark:text-slate-400">
|
||||
目前沒有可顯示的 GitHub 專案,或暫時無法連線到 GitHub。
|
||||
</p>
|
||||
</div>
|
||||
) : (
|
||||
<ul className="mt-4 grid gap-4 sm:grid-cols-2">
|
||||
{repos.map((repo, index) => (
|
||||
<RepoCard
|
||||
key={repo.id}
|
||||
repo={repo}
|
||||
animationDelay={index * 50}
|
||||
/>
|
||||
))}
|
||||
</ul>
|
||||
)}
|
||||
</SidebarLayout>
|
||||
</section>
|
||||
);
|
||||
}
|
||||
|
||||
@@ -4,11 +4,19 @@ export default function robots(): MetadataRoute.Robots {
|
||||
const siteUrl = process.env.NEXT_PUBLIC_SITE_URL || 'http://localhost:3000';
|
||||
|
||||
return {
|
||||
rules: {
|
||||
userAgent: '*',
|
||||
allow: '/',
|
||||
disallow: ['/api/', '/_next/', '/admin/'],
|
||||
},
|
||||
rules: [
|
||||
{
|
||||
userAgent: '*',
|
||||
allow: '/',
|
||||
disallow: ['/api/', '/_next/', '/admin/'],
|
||||
},
|
||||
{
|
||||
userAgent: ['GPTBot', 'ChatGPT-User', 'Google-Extended', 'Anthropic-ai', 'ClaudeBot', 'Claude-Web', 'PerplexityBot', 'Cohere-ai'],
|
||||
allow: '/',
|
||||
disallow: ['/api/', '/_next/', '/admin/'],
|
||||
},
|
||||
],
|
||||
sitemap: `${siteUrl}/sitemap.xml`,
|
||||
host: siteUrl,
|
||||
};
|
||||
}
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
import { MetadataRoute } from 'next';
|
||||
import { allPosts, allPages } from 'contentlayer2/generated';
|
||||
import { getTagSlug } from '@/lib/posts';
|
||||
|
||||
export default function sitemap(): MetadataRoute.Sitemap {
|
||||
const siteUrl = process.env.NEXT_PUBLIC_SITE_URL || 'http://localhost:3000';
|
||||
@@ -58,7 +59,7 @@ export default function sitemap(): MetadataRoute.Sitemap {
|
||||
);
|
||||
|
||||
const tagPages = allTags.map((tag) => ({
|
||||
url: `${siteUrl}/tags/${encodeURIComponent(tag.toLowerCase().replace(/\s+/g, '-'))}`,
|
||||
url: `${siteUrl}/tags/${encodeURIComponent(getTagSlug(tag))}`,
|
||||
lastModified: new Date(),
|
||||
changeFrequency: 'weekly' as const,
|
||||
priority: 0.5,
|
||||
|
||||
@@ -6,6 +6,8 @@ import { SidebarLayout } from '@/components/sidebar-layout';
|
||||
import { SectionDivider } from '@/components/section-divider';
|
||||
import { ScrollReveal } from '@/components/scroll-reveal';
|
||||
import { FiTag } from 'react-icons/fi';
|
||||
import { siteConfig } from '@/lib/config';
|
||||
import { JsonLd } from '@/components/json-ld';
|
||||
|
||||
export function generateStaticParams() {
|
||||
const slugs = new Set<string>();
|
||||
@@ -27,21 +29,30 @@ interface Props {
|
||||
|
||||
export async function generateMetadata({ params }: Props): Promise<Metadata> {
|
||||
const { tag: slug } = await params;
|
||||
// Decode the slug since Next.js encodes non-ASCII characters in URLs
|
||||
const decodedSlug = decodeURIComponent(slug);
|
||||
// Find original tag label by slug
|
||||
const tag = allPosts
|
||||
.flatMap((post) => post.tags ?? [])
|
||||
.find((t) => getTagSlug(t) === decodedSlug);
|
||||
|
||||
const tagUrl = `${siteConfig.url}/tags/${slug}`;
|
||||
|
||||
return {
|
||||
title: tag ? `標籤:${tag}` : '標籤'
|
||||
title: tag ? `標籤:${tag}` : '標籤',
|
||||
description: tag ? `查看標籤為「${tag}」的所有文章` : '標籤索引',
|
||||
alternates: {
|
||||
canonical: tagUrl
|
||||
},
|
||||
openGraph: {
|
||||
title: tag ? `標籤:${tag}` : '標籤',
|
||||
description: tag ? `查看標籤為「${tag}」的所有文章` : '標籤索引',
|
||||
url: tagUrl,
|
||||
type: 'website'
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
export default async function TagPage({ params }: Props) {
|
||||
const { tag: slug } = await params;
|
||||
// Decode the slug since Next.js encodes non-ASCII characters in URLs
|
||||
const decodedSlug = decodeURIComponent(slug);
|
||||
|
||||
const posts = allPosts.filter(
|
||||
@@ -51,8 +62,37 @@ export default async function TagPage({ params }: Props) {
|
||||
const tagLabel =
|
||||
posts[0]?.tags?.find((t) => getTagSlug(t) === decodedSlug) ?? decodedSlug;
|
||||
|
||||
// CollectionPage schema
|
||||
const collectionPageSchema = {
|
||||
'@context': 'https://schema.org',
|
||||
'@type': 'CollectionPage',
|
||||
name: `標籤:${tagLabel}`,
|
||||
description: `查看標籤為「${tagLabel}」的所有文章`,
|
||||
url: `${siteConfig.url}/tags/${slug}`,
|
||||
inLanguage: siteConfig.defaultLocale,
|
||||
about: {
|
||||
'@type': 'Thing',
|
||||
name: tagLabel
|
||||
},
|
||||
mainEntity: {
|
||||
'@type': 'Blog',
|
||||
blogPost: posts.slice(0, 10).map((post) => ({
|
||||
'@type': 'BlogPosting',
|
||||
headline: post.title,
|
||||
url: `${siteConfig.url}${post.url}`,
|
||||
datePublished: post.published_at,
|
||||
dateModified: post.updated_at || post.published_at,
|
||||
author: {
|
||||
'@type': 'Person',
|
||||
name: siteConfig.author
|
||||
}
|
||||
}))
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<SidebarLayout>
|
||||
<JsonLd data={collectionPageSchema} />
|
||||
<SectionDivider>
|
||||
<ScrollReveal>
|
||||
<div className="motion-card mb-8 rounded-2xl border border-white/40 bg-white/60 p-8 text-center shadow-lg backdrop-blur-md dark:border-white/10 dark:bg-slate-900/60">
|
||||
|
||||
@@ -5,9 +5,15 @@ import { getAllTagsWithCount } from '@/lib/posts';
|
||||
import { SectionDivider } from '@/components/section-divider';
|
||||
import { ScrollReveal } from '@/components/scroll-reveal';
|
||||
import { SidebarLayout } from '@/components/sidebar-layout';
|
||||
import { siteConfig } from '@/lib/config';
|
||||
import { JsonLd } from '@/components/json-ld';
|
||||
|
||||
export const metadata: Metadata = {
|
||||
title: '標籤索引'
|
||||
title: '標籤索引',
|
||||
description: '瀏覽所有標籤,探索不同主題的文章。',
|
||||
alternates: {
|
||||
canonical: `${siteConfig.url}/tags`
|
||||
}
|
||||
};
|
||||
|
||||
export default function TagIndexPage() {
|
||||
@@ -15,15 +21,38 @@ export default function TagIndexPage() {
|
||||
const topTags = tags.slice(0, 3);
|
||||
|
||||
const colorClasses = [
|
||||
'from-rose-400/70 to-rose-200/40',
|
||||
'from-emerald-400/70 to-emerald-200/40',
|
||||
'from-sky-400/70 to-sky-200/40',
|
||||
'from-amber-400/70 to-amber-200/40',
|
||||
'from-violet-400/70 to-violet-200/40'
|
||||
'from-accent/60 to-accent/20',
|
||||
'from-accent/50 to-accent/15',
|
||||
'from-accent/40 to-accent/10',
|
||||
];
|
||||
|
||||
// CollectionPage schema with ItemList
|
||||
const collectionPageSchema = {
|
||||
'@context': 'https://schema.org',
|
||||
'@type': 'CollectionPage',
|
||||
name: '標籤索引',
|
||||
description: '瀏覽所有標籤,探索不同主題的文章。',
|
||||
url: `${siteConfig.url}/tags`,
|
||||
inLanguage: siteConfig.defaultLocale,
|
||||
mainEntity: {
|
||||
'@type': 'ItemList',
|
||||
itemListElement: tags.map((tag, index) => ({
|
||||
'@type': 'ListItem',
|
||||
position: index + 1,
|
||||
name: tag.tag,
|
||||
url: `${siteConfig.url}/tags/${tag.slug}`,
|
||||
item: {
|
||||
'@type': 'Thing',
|
||||
name: tag.tag,
|
||||
description: `${tag.count} 篇文章`
|
||||
}
|
||||
}))
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<section className="space-y-6">
|
||||
<JsonLd data={collectionPageSchema} />
|
||||
<SidebarLayout>
|
||||
<SectionDivider>
|
||||
<ScrollReveal>
|
||||
@@ -56,7 +85,7 @@ export default function TagIndexPage() {
|
||||
>
|
||||
<span className={`mb-3 block h-1.5 w-16 rounded-full bg-gradient-to-r ${color}`} aria-hidden="true" />
|
||||
<div className="flex items-center justify-between">
|
||||
<h2 className="type-subtitle font-semibold text-slate-900 group-hover:text-blue-600 dark:text-slate-50 dark:group-hover:text-blue-400">
|
||||
<h2 className="type-subtitle font-semibold text-slate-900 group-hover:text-accent dark:text-slate-50 dark:group-hover:text-accent">
|
||||
{tag}
|
||||
</h2>
|
||||
<span className="type-small text-slate-600 dark:text-slate-300">
|
||||
|
||||
@@ -1,24 +1,18 @@
|
||||
'use client';
|
||||
|
||||
import { useEffect, useRef } from 'react';
|
||||
import { useEffect, useState } from 'react';
|
||||
|
||||
export default function Template({ children }: { children: React.ReactNode }) {
|
||||
const containerRef = useRef<HTMLDivElement>(null);
|
||||
const [prefersReducedMotion, setPrefersReducedMotion] = useState(true);
|
||||
|
||||
useEffect(() => {
|
||||
const container = containerRef.current;
|
||||
if (!container) return;
|
||||
const mq = window.matchMedia('(prefers-reduced-motion: reduce)');
|
||||
setPrefersReducedMotion(mq.matches);
|
||||
const handler = () => setPrefersReducedMotion(mq.matches);
|
||||
mq.addEventListener('change', handler);
|
||||
return () => mq.removeEventListener('change', handler);
|
||||
}, []);
|
||||
|
||||
// Trigger animation on mount
|
||||
container.style.animation = 'none';
|
||||
// Force reflow
|
||||
void container.offsetHeight;
|
||||
container.style.animation = 'pageEnter 0.3s cubic-bezier(0.32, 0.72, 0, 1) forwards';
|
||||
}, [children]);
|
||||
|
||||
return (
|
||||
<div ref={containerRef} className="page-transition">
|
||||
{children}
|
||||
</div>
|
||||
);
|
||||
// ViewTransitions handles page transitions - no additional wrapper needed
|
||||
return <>{children}</>;
|
||||
}
|
||||
|
||||
14
components/app-wrapper.tsx
Normal file
14
components/app-wrapper.tsx
Normal file
@@ -0,0 +1,14 @@
|
||||
'use client';
|
||||
|
||||
import { ViewTransitionProvider } from '@/components/view-transition-provider';
|
||||
import Template from '@/app/template';
|
||||
|
||||
export function AppWrapper({ children }: { children: React.ReactNode }) {
|
||||
return (
|
||||
<ViewTransitionProvider>
|
||||
<Template>
|
||||
{children}
|
||||
</Template>
|
||||
</ViewTransitionProvider>
|
||||
);
|
||||
}
|
||||
@@ -28,7 +28,7 @@ export function BackToTop() {
|
||||
window.scrollTo({ top: 0, behavior: 'smooth' });
|
||||
}}
|
||||
aria-label="回到頁面頂部"
|
||||
className="fixed bottom-6 right-4 z-40 inline-flex h-9 w-9 items-center justify-center rounded-full bg-slate-900 text-slate-50 shadow-md ring-1 ring-slate-800/70 transition hover:bg-slate-700 dark:bg-slate-100 dark:text-slate-900 dark:ring-slate-300/70 dark:hover:bg-slate-300"
|
||||
className="fixed bottom-6 right-4 z-40 inline-flex h-9 w-9 items-center justify-center rounded-full bg-accent text-white shadow-lg ring-2 ring-accent/30 transition-all duration-300 ease-out-expo hover:-translate-y-1 hover:shadow-xl focus-visible:outline-none focus-visible:ring-4 focus-visible:ring-accent/40 dark:bg-accent dark:ring-accent/30 dark:hover:bg-accent/90"
|
||||
>
|
||||
<span className="text-lg leading-none">↑</span>
|
||||
</button>
|
||||
|
||||
97
components/dev-env-device-hero.tsx
Normal file
97
components/dev-env-device-hero.tsx
Normal file
@@ -0,0 +1,97 @@
|
||||
'use client';
|
||||
|
||||
import { SiArchlinux, SiUbuntu, SiLinux } from 'react-icons/si';
|
||||
|
||||
/**
|
||||
* Mac mini + 螢幕 3D 裝置展示
|
||||
* 使用純 CSS 3D transforms,取代開發工作環境頁的 feature_image
|
||||
*/
|
||||
export function DevEnvDeviceHero() {
|
||||
return (
|
||||
<div
|
||||
className="dev-env-device-hero -mx-4 mb-6 flex justify-center py-4 sm:-mx-12 sm:py-6 lg:-mx-20 lg:py-8 group-[.toc-open]:lg:-mx-4"
|
||||
role="img"
|
||||
aria-label="Mac mini、鍵盤與外接螢幕的 3D 裝置展示"
|
||||
>
|
||||
<div className="dev-env-device-scene">
|
||||
{/* Monitor */}
|
||||
<div className="dev-env-monitor">
|
||||
{/* Bezel */}
|
||||
<div className="dev-env-bezel">
|
||||
{/* Screen */}
|
||||
<div className="dev-env-screen">
|
||||
{/* macOS Desktop mockup */}
|
||||
<div className="dev-env-desktop">
|
||||
{/* macOS Menu bar - 半透明毛玻璃 */}
|
||||
<div className="dev-env-menubar">
|
||||
<span className="dev-env-apple" aria-hidden>{'\uF8FF'}</span>
|
||||
<span className="dev-env-app-name">Terminal</span>
|
||||
<span className="dev-env-spacer" />
|
||||
<span className="dev-env-menubar-right">
|
||||
<span className="dev-env-menubar-icon" aria-hidden />
|
||||
<span className="dev-env-menubar-icon" aria-hidden />
|
||||
<span className="dev-env-menubar-icon" aria-hidden />
|
||||
<span className="dev-env-time">14:30</span>
|
||||
</span>
|
||||
</div>
|
||||
{/* Window - Terminal 顯示 Arch / Ubuntu / Tux 三個 Logo */}
|
||||
<div className="dev-env-window">
|
||||
<div className="dev-env-window-titlebar">
|
||||
<span className="dev-env-traffic-light dev-env-traffic-red" aria-hidden />
|
||||
<span className="dev-env-traffic-light dev-env-traffic-yellow" aria-hidden />
|
||||
<span className="dev-env-traffic-light dev-env-traffic-green" aria-hidden />
|
||||
</div>
|
||||
<div className="dev-env-window-content">
|
||||
<div className="dev-env-terminal-prompt">
|
||||
<span className="dev-env-prompt">$</span> neofetch --ascii_distro arch,ubuntu,tux
|
||||
</div>
|
||||
<div className="dev-env-terminal-logos">
|
||||
<div className="dev-env-logo-svg" aria-label="Arch Linux logo">
|
||||
<SiArchlinux className="dev-env-svg-arch" size={36} />
|
||||
</div>
|
||||
<div className="dev-env-logo-svg" aria-label="Ubuntu logo">
|
||||
<SiUbuntu className="dev-env-svg-ubuntu" size={36} />
|
||||
</div>
|
||||
<div className="dev-env-logo-svg" aria-label="Tux Linux penguin logo">
|
||||
<SiLinux className="dev-env-svg-tux" size={36} />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{/* Monitor stand */}
|
||||
<div className="dev-env-stand" />
|
||||
</div>
|
||||
|
||||
{/* Desk surface - Mac mini 與鍵盤均勻放置 */}
|
||||
<div className="dev-env-desk">
|
||||
{/* 鍵盤 - Magic Keyboard 風格,鍵帽網格 */}
|
||||
<div className="dev-env-keyboard">
|
||||
<div className="dev-env-keyboard-body">
|
||||
<div className="dev-env-keyboard-keys">
|
||||
{[14, 14, 13, 12].map((keyCount, row) => (
|
||||
<div key={row} className="dev-env-keyboard-row">
|
||||
{Array.from({ length: keyCount }).map((_, col) => (
|
||||
<div key={col} className="dev-env-key" />
|
||||
))}
|
||||
</div>
|
||||
))}
|
||||
<div className="dev-env-keyboard-row dev-env-keyboard-row-space">
|
||||
<div className="dev-env-key dev-env-key-space" />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{/* Mac mini M4 2024 - 頂視,避免 3D 偽影 */}
|
||||
<div className="dev-env-macmini">
|
||||
<div className="dev-env-macmini-top">
|
||||
<span className="dev-env-macmini-apple" aria-hidden>{'\uF8FF'}</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
117
components/hero-section.tsx
Normal file
117
components/hero-section.tsx
Normal file
@@ -0,0 +1,117 @@
|
||||
'use client';
|
||||
|
||||
import { useState, useEffect } from 'react';
|
||||
import { MatrixRain } from './matrix-rain';
|
||||
import { TerminalWindow } from './terminal-window';
|
||||
|
||||
interface HeroSectionProps {
|
||||
title: string;
|
||||
tagline: string;
|
||||
}
|
||||
|
||||
type Phase = 'matrix' | 'transition' | 'terminal';
|
||||
|
||||
const MIN_MATRIX_DURATION = 1500;
|
||||
const MAX_MATRIX_DURATION = 6000;
|
||||
const TRANSITION_DURATION = 600;
|
||||
|
||||
export function HeroSection({ title, tagline }: HeroSectionProps) {
|
||||
const [phase, setPhase] = useState<Phase>('matrix');
|
||||
const [matrixOpacity, setMatrixOpacity] = useState(1);
|
||||
const [terminalOpacity, setTerminalOpacity] = useState(0);
|
||||
const [reducedMotion, setReducedMotion] = useState(false);
|
||||
|
||||
useEffect(() => {
|
||||
const mq = window.matchMedia('(prefers-reduced-motion: reduce)');
|
||||
setReducedMotion(mq.matches);
|
||||
}, []);
|
||||
|
||||
const handleMatrixComplete = () => {
|
||||
setPhase('transition');
|
||||
setMatrixOpacity(0);
|
||||
setTerminalOpacity(1);
|
||||
};
|
||||
|
||||
useEffect(() => {
|
||||
if (phase !== 'matrix') return;
|
||||
|
||||
const startTime = Date.now();
|
||||
let maxTimerId: ReturnType<typeof setTimeout>;
|
||||
let minTimerId: ReturnType<typeof setTimeout>;
|
||||
|
||||
const scheduleTransition = () => {
|
||||
const elapsed = Date.now() - startTime;
|
||||
const remaining = Math.max(0, MIN_MATRIX_DURATION - elapsed);
|
||||
if (remaining > 0) {
|
||||
minTimerId = setTimeout(handleMatrixComplete, remaining);
|
||||
} else {
|
||||
handleMatrixComplete();
|
||||
}
|
||||
};
|
||||
|
||||
const onLoad = () => {
|
||||
window.removeEventListener('load', onLoad);
|
||||
clearTimeout(maxTimerId);
|
||||
scheduleTransition();
|
||||
};
|
||||
|
||||
if (document.readyState === 'complete') {
|
||||
scheduleTransition();
|
||||
} else {
|
||||
window.addEventListener('load', onLoad);
|
||||
maxTimerId = setTimeout(() => {
|
||||
window.removeEventListener('load', onLoad);
|
||||
handleMatrixComplete();
|
||||
}, MAX_MATRIX_DURATION);
|
||||
}
|
||||
|
||||
return () => {
|
||||
window.removeEventListener('load', onLoad);
|
||||
clearTimeout(maxTimerId);
|
||||
clearTimeout(minTimerId);
|
||||
};
|
||||
}, [phase]);
|
||||
|
||||
useEffect(() => {
|
||||
if (phase === 'transition') {
|
||||
const id = setTimeout(() => setPhase('terminal'), TRANSITION_DURATION);
|
||||
return () => clearTimeout(id);
|
||||
}
|
||||
}, [phase]);
|
||||
|
||||
// Skip Matrix entirely if user prefers reduced motion
|
||||
useEffect(() => {
|
||||
if (reducedMotion) {
|
||||
setPhase('terminal');
|
||||
setMatrixOpacity(0);
|
||||
setTerminalOpacity(1);
|
||||
}
|
||||
}, [reducedMotion]);
|
||||
|
||||
return (
|
||||
<div className="relative h-[360px] w-full overflow-hidden rounded-2xl sm:h-[400px] lg:h-[440px] xl:h-[480px]">
|
||||
{/* Matrix rain - full area, fades out */}
|
||||
{!reducedMotion && (
|
||||
<div
|
||||
className="absolute inset-0 transition-opacity duration-[600ms] ease-out"
|
||||
style={{ opacity: matrixOpacity }}
|
||||
aria-hidden="true"
|
||||
>
|
||||
<MatrixRain className="h-full w-full" />
|
||||
</div>
|
||||
)}
|
||||
|
||||
{/* Terminal - fades in over Matrix, responsive width */}
|
||||
<div
|
||||
className="relative z-10 mx-auto w-full max-w-2xl px-4 py-6 transition-opacity duration-[600ms] ease-out sm:max-w-3xl lg:max-w-4xl xl:max-w-5xl"
|
||||
style={{ opacity: reducedMotion ? 1 : terminalOpacity }}
|
||||
>
|
||||
<TerminalWindow
|
||||
title={title}
|
||||
tagline={tagline}
|
||||
reducedMotion={reducedMotion}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@@ -52,9 +52,7 @@ export function Hero() {
|
||||
}[];
|
||||
|
||||
return (
|
||||
<section className="motion-card group relative mb-8 overflow-hidden rounded-xl border bg-gradient-to-r from-sky-50 via-indigo-50 to-slate-50 px-6 py-6 shadow-sm dark:border-slate-800 dark:from-slate-900 dark:via-slate-950 dark:to-slate-900">
|
||||
<div className="pointer-events-none absolute -left-16 -top-16 h-40 w-40 rounded-full bg-sky-300/40 blur-3xl mix-blend-soft-light motion-safe:animate-float-soft dark:bg-sky-500/25" />
|
||||
<div className="pointer-events-none absolute -bottom-20 right-[-3rem] h-44 w-44 rounded-full bg-indigo-300/35 blur-3xl mix-blend-soft-light motion-safe:animate-float-soft dark:bg-indigo-500/25" />
|
||||
<section className="motion-card group relative mb-8 overflow-hidden rounded-xl border bg-accent-soft px-6 py-6 shadow-sm dark:border-slate-800">
|
||||
|
||||
<div className="relative flex items-center gap-4 motion-safe:animate-fade-in-up">
|
||||
<div className="flex h-16 w-16 items-center justify-center rounded-full bg-slate-900 text-xl font-semibold text-slate-50 shadow-md transition-transform duration-300 ease-out group-hover:scale-105 dark:bg-slate-100 dark:text-slate-900">
|
||||
|
||||
78
components/homelab-device-hero.tsx
Normal file
78
components/homelab-device-hero.tsx
Normal file
@@ -0,0 +1,78 @@
|
||||
'use client';
|
||||
|
||||
import { SiTruenas, SiProxmox } from 'react-icons/si';
|
||||
import { FiServer } from 'react-icons/fi';
|
||||
|
||||
/**
|
||||
* HomeLab 設備展示:Proxmox VE + VyOS、Switch、NAS (TrueNAS)
|
||||
* 使用純 CSS 藝術,取代 HomeLab 頁的 feature_image
|
||||
*/
|
||||
export function HomeLabDeviceHero() {
|
||||
return (
|
||||
<div
|
||||
className="homelab-device-hero -mx-4 mb-6 flex justify-center py-4 sm:-mx-12 sm:py-6 lg:-mx-20 lg:py-8 group-[.toc-open]:lg:-mx-4"
|
||||
role="img"
|
||||
aria-label="HomeLab 設備:Proxmox VE、VyOS、交換器、NAS (TrueNAS)"
|
||||
>
|
||||
<div className="homelab-device-scene w-full max-w-full">
|
||||
<div className="homelab-rack">
|
||||
{/* Proxmox VE + VyOS Host */}
|
||||
<div className="homelab-router">
|
||||
<div className="homelab-router-body">
|
||||
<div className="homelab-router-leds">
|
||||
<span className="homelab-led homelab-led-power" aria-hidden />
|
||||
<span className="homelab-led homelab-led-wan" aria-hidden />
|
||||
<span className="homelab-led homelab-led-lan" aria-hidden />
|
||||
</div>
|
||||
<div className="homelab-router-logos">
|
||||
<SiProxmox className="homelab-proxmox-logo homelab-logo-svg" aria-label="Proxmox VE" />
|
||||
<FiServer className="homelab-router-icon homelab-logo-svg" aria-label="VyOS Router" />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* 網路線 - 連接 Proxmox 與 Switch */}
|
||||
<div className="homelab-cable" aria-hidden>
|
||||
<span className="homelab-cable-line" />
|
||||
</div>
|
||||
|
||||
{/* Switch */}
|
||||
<div className="homelab-switch">
|
||||
<div className="homelab-switch-body">
|
||||
<div className="homelab-switch-ports">
|
||||
{[1, 2].map((row) => (
|
||||
<div key={row} className="homelab-port-row">
|
||||
{Array.from({ length: 8 }).map((_, i) => (
|
||||
<div key={i} className="homelab-port">
|
||||
<span className="homelab-port-led homelab-port-led-active" aria-hidden />
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* 網路線 - 連接 Switch 與 NAS */}
|
||||
<div className="homelab-cable" aria-hidden>
|
||||
<span className="homelab-cable-line" />
|
||||
</div>
|
||||
|
||||
{/* NAS - TrueNAS */}
|
||||
<div className="homelab-nas">
|
||||
<div className="homelab-nas-body">
|
||||
<div className="homelab-nas-drives">
|
||||
{[1, 2, 3, 4].map((i) => (
|
||||
<div key={i} className="homelab-drive-slot" aria-hidden />
|
||||
))}
|
||||
</div>
|
||||
<div className="homelab-nas-logo" aria-label="TrueNAS logo">
|
||||
<SiTruenas className="homelab-truenas-logo" size={28} />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@@ -1,11 +1,23 @@
|
||||
'use client';
|
||||
|
||||
import { SiteHeader } from './site-header';
|
||||
import { SiteFooter } from './site-footer';
|
||||
import { BackToTop } from './back-to-top';
|
||||
import dynamic from 'next/dynamic';
|
||||
|
||||
export function LayoutShell({ children }: { children: React.ReactNode }) {
|
||||
// Lazy load BackToTop since it's not critical for initial render
|
||||
const BackToTop = dynamic(() => import('./back-to-top').then(mod => ({ default: mod.BackToTop })), {
|
||||
ssr: false,
|
||||
});
|
||||
|
||||
interface LayoutShellProps {
|
||||
children: React.ReactNode;
|
||||
recentPosts?: { title: string; url: string }[];
|
||||
}
|
||||
|
||||
export function LayoutShell({ children, recentPosts = [] }: LayoutShellProps) {
|
||||
return (
|
||||
<div className="flex min-h-screen flex-col">
|
||||
<SiteHeader />
|
||||
<SiteHeader recentPosts={recentPosts} />
|
||||
<main className="flex-1 container mx-auto px-4 py-6">
|
||||
{children}
|
||||
</main>
|
||||
|
||||
@@ -83,10 +83,19 @@ export function MastodonFeed() {
|
||||
{loading ? (
|
||||
<div className="space-y-3">
|
||||
{[...Array(3)].map((_, i) => (
|
||||
<div key={i} className="animate-pulse">
|
||||
<div className="h-3 w-3/4 rounded bg-slate-200 dark:bg-slate-800"></div>
|
||||
<div className="mt-2 h-3 w-full rounded bg-slate-200 dark:bg-slate-800"></div>
|
||||
<div className="mt-2 h-2 w-1/3 rounded bg-slate-200 dark:bg-slate-800"></div>
|
||||
<div key={i}>
|
||||
<div
|
||||
className="mastodon-skeleton-shimmer h-3 w-3/4 rounded"
|
||||
style={{ animationDelay: `${i * 120}ms` }}
|
||||
/>
|
||||
<div
|
||||
className="mastodon-skeleton-shimmer mt-2 h-3 w-full rounded"
|
||||
style={{ animationDelay: `${i * 120}ms` }}
|
||||
/>
|
||||
<div
|
||||
className="mastodon-skeleton-shimmer mt-2 h-2 w-1/3 rounded"
|
||||
style={{ animationDelay: `${i * 120}ms` }}
|
||||
/>
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
@@ -125,10 +134,83 @@ export function MastodonFeed() {
|
||||
{truncated}
|
||||
</p>
|
||||
|
||||
{/* Media indicator */}
|
||||
{/* Media attachments - render images/videos from remote URLs */}
|
||||
{hasMedia && (
|
||||
<div className="type-small text-slate-400 dark:text-slate-500">
|
||||
📎 包含 {displayStatus.media_attachments.length} 個媒體
|
||||
<div
|
||||
className={`mt-1.5 grid gap-1 ${
|
||||
displayStatus.media_attachments.length === 1
|
||||
? 'grid-cols-1'
|
||||
: 'grid-cols-2'
|
||||
}`}
|
||||
>
|
||||
{displayStatus.media_attachments.map((att) => {
|
||||
const src = att.preview_url ?? att.url;
|
||||
if (!src) return null;
|
||||
|
||||
if (att.type === 'image') {
|
||||
return (
|
||||
<img
|
||||
key={att.id}
|
||||
src={src}
|
||||
alt={att.description ?? ''}
|
||||
loading="lazy"
|
||||
className="aspect-video w-full rounded-md object-cover"
|
||||
/>
|
||||
);
|
||||
}
|
||||
if (att.type === 'gifv' && att.url) {
|
||||
return (
|
||||
<div
|
||||
key={att.id}
|
||||
className="overflow-hidden rounded-md"
|
||||
onClick={(e) => e.stopPropagation()}
|
||||
>
|
||||
<video
|
||||
src={att.url}
|
||||
poster={att.preview_url ?? undefined}
|
||||
autoPlay
|
||||
loop
|
||||
muted
|
||||
playsInline
|
||||
className="aspect-video w-full object-cover"
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
if (att.type === 'video' && att.url) {
|
||||
return (
|
||||
<div
|
||||
key={att.id}
|
||||
className="overflow-hidden rounded-md"
|
||||
onClick={(e) => e.stopPropagation()}
|
||||
>
|
||||
<video
|
||||
src={att.url}
|
||||
poster={att.preview_url ?? undefined}
|
||||
controls
|
||||
playsInline
|
||||
className="aspect-video w-full object-cover"
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
if (att.type === 'audio' && att.preview_url) {
|
||||
return (
|
||||
<div
|
||||
key={att.id}
|
||||
className="flex aspect-video w-full items-center justify-center rounded-md bg-slate-200 dark:bg-slate-700"
|
||||
>
|
||||
<img
|
||||
src={att.preview_url}
|
||||
alt={att.description ?? '音訊'}
|
||||
loading="lazy"
|
||||
className="h-full w-full object-cover opacity-80"
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
return null;
|
||||
})}
|
||||
</div>
|
||||
)}
|
||||
|
||||
|
||||
147
components/matrix-rain.tsx
Normal file
147
components/matrix-rain.tsx
Normal file
@@ -0,0 +1,147 @@
|
||||
'use client';
|
||||
|
||||
import { useEffect, useRef } from 'react';
|
||||
|
||||
// Matrix-style characters: katakana, numbers, Latin
|
||||
const CHARS = 'アイウエオカキクケコサシスセソタチツテトナニヌネノハヒフヘホマミムメモヤユヨラリルレロワン0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ';
|
||||
|
||||
interface MatrixRainProps {
|
||||
/** Opacity 0-1 for fade out control */
|
||||
opacity?: number;
|
||||
className?: string;
|
||||
}
|
||||
|
||||
interface Drop {
|
||||
x: number;
|
||||
y: number;
|
||||
speed: number;
|
||||
chars: string[];
|
||||
charIndex: number;
|
||||
}
|
||||
|
||||
export function MatrixRain({
|
||||
opacity = 1,
|
||||
className = '',
|
||||
}: MatrixRainProps) {
|
||||
const canvasRef = useRef<HTMLCanvasElement>(null);
|
||||
|
||||
useEffect(() => {
|
||||
const canvas = canvasRef.current;
|
||||
if (!canvas) return;
|
||||
|
||||
const ctx = canvas.getContext('2d');
|
||||
if (!ctx) return;
|
||||
|
||||
const resize = () => {
|
||||
const rect = canvas.getBoundingClientRect();
|
||||
// Calculate DPR safely - use 1 as fallback
|
||||
const dpr = (typeof window !== 'undefined' && 'devicePixelRatio' in window)
|
||||
? Math.min(window.devicePixelRatio ?? 1, 2)
|
||||
: 1;
|
||||
canvas.width = rect.width * dpr;
|
||||
canvas.height = rect.height * dpr;
|
||||
ctx.scale(dpr, dpr);
|
||||
canvas.style.width = `${rect.width}px`;
|
||||
canvas.style.height = `${rect.height}px`;
|
||||
};
|
||||
|
||||
const handleResize = () => {
|
||||
// Use requestAnimationFrame for smoother resizing
|
||||
requestAnimationFrame(() => {
|
||||
const rect = canvasRef.current?.getBoundingClientRect();
|
||||
if (rect) {
|
||||
const dpr = (typeof window !== 'undefined' && 'devicePixelRatio' in window)
|
||||
? Math.min(window.devicePixelRatio ?? 1, 2)
|
||||
: 1;
|
||||
canvasRef.current!.width = rect.width * dpr;
|
||||
canvasRef.current!.height = rect.height * dpr;
|
||||
if (ctx) {
|
||||
ctx.scale(dpr, dpr);
|
||||
}
|
||||
canvasRef.current!.style.width = `${rect.width}px`;
|
||||
canvasRef.current!.style.height = `${rect.height}px`;
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
resize();
|
||||
window.addEventListener('resize', handleResize, { passive: true, signal: AbortSignal.timeout(60000) });
|
||||
|
||||
const fontSize = 14;
|
||||
const columns = Math.floor(canvas.getBoundingClientRect().width / fontSize);
|
||||
const drops: Drop[] = Array.from({ length: columns }, (_, i) => ({
|
||||
x: i * fontSize,
|
||||
y: Math.random() * -100,
|
||||
speed: 0.15 + Math.random() * 0.4,
|
||||
chars: Array.from({ length: 20 }, () =>
|
||||
CHARS[Math.floor(Math.random() * CHARS.length)]
|
||||
),
|
||||
charIndex: Math.floor(Math.random() * 20),
|
||||
}));
|
||||
|
||||
let animationId: number;
|
||||
let lastTime: number | null = null;
|
||||
|
||||
const draw = (timestamp: number) => {
|
||||
const rect = canvas.getBoundingClientRect();
|
||||
const delta =
|
||||
lastTime !== null ? (timestamp - lastTime) / 1000 : 1 / 60;
|
||||
lastTime = timestamp;
|
||||
|
||||
ctx.fillStyle = 'rgba(15, 23, 42, 0.05)';
|
||||
ctx.fillRect(0, 0, rect.width, rect.height);
|
||||
|
||||
ctx.font = `${fontSize}px "JetBrains Mono", "SF Mono", "Fira Code", monospace`;
|
||||
|
||||
drops.forEach((drop) => {
|
||||
// Bright green for leading char
|
||||
ctx.fillStyle = 'rgba(34, 197, 94, 1)';
|
||||
ctx.fillText(drop.chars[drop.charIndex], drop.x, drop.y);
|
||||
|
||||
// Dimmer trailing chars
|
||||
for (let i = 1; i < 8; i++) {
|
||||
const idx = (drop.charIndex - i + 20) % 20;
|
||||
const alpha = 1 - i * 0.12;
|
||||
ctx.fillStyle = `rgba(34, 197, 94, ${alpha * 0.4})`;
|
||||
ctx.fillText(
|
||||
drop.chars[idx],
|
||||
drop.x,
|
||||
drop.y - i * fontSize
|
||||
);
|
||||
}
|
||||
|
||||
// Frame-rate independent: scale by delta, 60fps as baseline
|
||||
drop.y += drop.speed * fontSize * delta * 60;
|
||||
if (drop.y > rect.height + 100) {
|
||||
drop.y = -50;
|
||||
drop.charIndex = (drop.charIndex + 1) % 20;
|
||||
} else {
|
||||
drop.charIndex = (drop.charIndex + 1) % 20;
|
||||
}
|
||||
});
|
||||
|
||||
animationId = requestAnimationFrame(draw);
|
||||
};
|
||||
|
||||
animationId = requestAnimationFrame(draw);
|
||||
|
||||
return () => {
|
||||
cancelAnimationFrame(animationId);
|
||||
window.removeEventListener('resize', handleResize);
|
||||
};
|
||||
}, []);
|
||||
|
||||
return (
|
||||
<canvas
|
||||
ref={canvasRef}
|
||||
className={className}
|
||||
style={{
|
||||
opacity,
|
||||
transition: 'opacity 0.6s ease-out',
|
||||
background: 'rgb(15, 23, 42)',
|
||||
}}
|
||||
aria-hidden="true"
|
||||
role="img"
|
||||
/>
|
||||
);
|
||||
}
|
||||
312
components/mermaid-renderer.tsx
Normal file
312
components/mermaid-renderer.tsx
Normal file
@@ -0,0 +1,312 @@
|
||||
'use client';
|
||||
|
||||
import { useEffect, useRef, useCallback } from 'react';
|
||||
import { useTheme } from 'next-themes';
|
||||
|
||||
const ZOOM_STEP = 0.2;
|
||||
const ZOOM_MIN = 0.25;
|
||||
const ZOOM_MAX = 5;
|
||||
const WHEEL_ZOOM_FACTOR = 0.001;
|
||||
|
||||
interface ViewState {
|
||||
scale: number;
|
||||
x: number;
|
||||
y: number;
|
||||
}
|
||||
|
||||
function clampScale(s: number) {
|
||||
return Math.min(ZOOM_MAX, Math.max(ZOOM_MIN, s));
|
||||
}
|
||||
|
||||
function attachViewer(wrapper: HTMLDivElement, viewport: HTMLDivElement) {
|
||||
const state: ViewState = { scale: 1, x: 0, y: 0 };
|
||||
let dragging = false;
|
||||
let dragStart = { x: 0, y: 0 };
|
||||
let originAtDragStart = { x: 0, y: 0 };
|
||||
|
||||
// --- Pinch state ---
|
||||
let lastPinchDist = 0;
|
||||
let lastPinchCenter = { x: 0, y: 0 };
|
||||
let pinching = false;
|
||||
|
||||
const levelBtn = wrapper.querySelector<HTMLButtonElement>('.mermaid-zoom-level')!;
|
||||
|
||||
const apply = () => {
|
||||
viewport.style.transform = `translate(${state.x}px, ${state.y}px) scale(${state.scale})`;
|
||||
levelBtn.textContent = `${Math.round(state.scale * 100)}%`;
|
||||
};
|
||||
|
||||
const zoomTo = (newScale: number, cx: number, cy: number) => {
|
||||
const clamped = clampScale(newScale);
|
||||
const rect = viewport.getBoundingClientRect();
|
||||
const wrapRect = wrapper.querySelector<HTMLElement>('.mermaid-canvas')!.getBoundingClientRect();
|
||||
|
||||
// Point under cursor in viewport-local coords
|
||||
const px = cx - wrapRect.left;
|
||||
const py = cy - wrapRect.top;
|
||||
|
||||
// Adjust translate so the point under cursor stays put
|
||||
const ratio = clamped / state.scale;
|
||||
state.x = px - ratio * (px - state.x);
|
||||
state.y = py - ratio * (py - state.y);
|
||||
state.scale = clamped;
|
||||
apply();
|
||||
};
|
||||
|
||||
// --- Mouse drag ---
|
||||
const onMouseDown = (e: MouseEvent) => {
|
||||
if (e.button !== 0) return;
|
||||
dragging = true;
|
||||
dragStart = { x: e.clientX, y: e.clientY };
|
||||
originAtDragStart = { x: state.x, y: state.y };
|
||||
wrapper.classList.add('mermaid-grabbing');
|
||||
e.preventDefault();
|
||||
};
|
||||
|
||||
const onMouseMove = (e: MouseEvent) => {
|
||||
if (!dragging) return;
|
||||
state.x = originAtDragStart.x + (e.clientX - dragStart.x);
|
||||
state.y = originAtDragStart.y + (e.clientY - dragStart.y);
|
||||
apply();
|
||||
};
|
||||
|
||||
const onMouseUp = () => {
|
||||
dragging = false;
|
||||
wrapper.classList.remove('mermaid-grabbing');
|
||||
};
|
||||
|
||||
// --- Wheel zoom ---
|
||||
const onWheel = (e: WheelEvent) => {
|
||||
e.preventDefault();
|
||||
const delta = -e.deltaY * WHEEL_ZOOM_FACTOR;
|
||||
const newScale = clampScale(state.scale * (1 + delta * state.scale));
|
||||
zoomTo(newScale, e.clientX, e.clientY);
|
||||
};
|
||||
|
||||
// --- Touch: pinch-to-zoom + drag ---
|
||||
const pinchDist = (t: TouchList) => {
|
||||
const dx = t[0].clientX - t[1].clientX;
|
||||
const dy = t[0].clientY - t[1].clientY;
|
||||
return Math.hypot(dx, dy);
|
||||
};
|
||||
|
||||
const pinchCenter = (t: TouchList) => ({
|
||||
x: (t[0].clientX + t[1].clientX) / 2,
|
||||
y: (t[0].clientY + t[1].clientY) / 2,
|
||||
});
|
||||
|
||||
const onTouchStart = (e: TouchEvent) => {
|
||||
if (e.touches.length === 2) {
|
||||
pinching = true;
|
||||
lastPinchDist = pinchDist(e.touches);
|
||||
lastPinchCenter = pinchCenter(e.touches);
|
||||
e.preventDefault();
|
||||
} else if (e.touches.length === 1) {
|
||||
dragging = true;
|
||||
dragStart = { x: e.touches[0].clientX, y: e.touches[0].clientY };
|
||||
originAtDragStart = { x: state.x, y: state.y };
|
||||
}
|
||||
};
|
||||
|
||||
const onTouchMove = (e: TouchEvent) => {
|
||||
if (pinching && e.touches.length === 2) {
|
||||
e.preventDefault();
|
||||
const dist = pinchDist(e.touches);
|
||||
const center = pinchCenter(e.touches);
|
||||
const ratio = dist / lastPinchDist;
|
||||
zoomTo(state.scale * ratio, center.x, center.y);
|
||||
lastPinchDist = dist;
|
||||
lastPinchCenter = center;
|
||||
} else if (dragging && e.touches.length === 1) {
|
||||
state.x = originAtDragStart.x + (e.touches[0].clientX - dragStart.x);
|
||||
state.y = originAtDragStart.y + (e.touches[0].clientY - dragStart.y);
|
||||
apply();
|
||||
}
|
||||
};
|
||||
|
||||
const onTouchEnd = (e: TouchEvent) => {
|
||||
if (e.touches.length < 2) pinching = false;
|
||||
if (e.touches.length === 0) dragging = false;
|
||||
};
|
||||
|
||||
// --- Canvas element (the pannable area) ---
|
||||
const canvas = wrapper.querySelector<HTMLElement>('.mermaid-canvas')!;
|
||||
canvas.addEventListener('mousedown', onMouseDown);
|
||||
window.addEventListener('mousemove', onMouseMove);
|
||||
window.addEventListener('mouseup', onMouseUp);
|
||||
canvas.addEventListener('wheel', onWheel, { passive: false });
|
||||
canvas.addEventListener('touchstart', onTouchStart, { passive: false });
|
||||
canvas.addEventListener('touchmove', onTouchMove, { passive: false });
|
||||
canvas.addEventListener('touchend', onTouchEnd);
|
||||
|
||||
// --- Button handlers ---
|
||||
wrapper.querySelector('.mermaid-btn-zoomout')!.addEventListener('click', () => {
|
||||
const rect = canvas.getBoundingClientRect();
|
||||
zoomTo(state.scale - ZOOM_STEP, rect.left + rect.width / 2, rect.top + rect.height / 2);
|
||||
});
|
||||
|
||||
wrapper.querySelector('.mermaid-btn-zoomin')!.addEventListener('click', () => {
|
||||
const rect = canvas.getBoundingClientRect();
|
||||
zoomTo(state.scale + ZOOM_STEP, rect.left + rect.width / 2, rect.top + rect.height / 2);
|
||||
});
|
||||
|
||||
levelBtn.addEventListener('click', () => {
|
||||
state.scale = 1;
|
||||
state.x = 0;
|
||||
state.y = 0;
|
||||
apply();
|
||||
});
|
||||
|
||||
wrapper.querySelector('.mermaid-btn-fit')!.addEventListener('click', () => {
|
||||
const svg = viewport.querySelector('svg');
|
||||
if (!svg) return;
|
||||
const canvasRect = canvas.getBoundingClientRect();
|
||||
const svgW = svg.viewBox.baseVal.width || svg.getBoundingClientRect().width / state.scale;
|
||||
const svgH = svg.viewBox.baseVal.height || svg.getBoundingClientRect().height / state.scale;
|
||||
const padding = 32;
|
||||
const fitScale = Math.min(
|
||||
(canvasRect.width - padding) / svgW,
|
||||
(canvasRect.height - padding) / svgH,
|
||||
ZOOM_MAX
|
||||
);
|
||||
state.scale = clampScale(fitScale);
|
||||
state.x = 0;
|
||||
state.y = 0;
|
||||
apply();
|
||||
});
|
||||
|
||||
wrapper.querySelector('.mermaid-btn-fullscreen')!.addEventListener('click', () => {
|
||||
if (document.fullscreenElement === wrapper) {
|
||||
document.exitFullscreen();
|
||||
} else {
|
||||
wrapper.requestFullscreen().catch(() => {});
|
||||
}
|
||||
});
|
||||
|
||||
// Cleanup
|
||||
return () => {
|
||||
canvas.removeEventListener('mousedown', onMouseDown);
|
||||
window.removeEventListener('mousemove', onMouseMove);
|
||||
window.removeEventListener('mouseup', onMouseUp);
|
||||
canvas.removeEventListener('wheel', onWheel);
|
||||
canvas.removeEventListener('touchstart', onTouchStart);
|
||||
canvas.removeEventListener('touchmove', onTouchMove);
|
||||
canvas.removeEventListener('touchend', onTouchEnd);
|
||||
};
|
||||
}
|
||||
|
||||
function buildShell(): { wrapper: HTMLDivElement; viewport: HTMLDivElement } {
|
||||
const wrapper = document.createElement('div');
|
||||
wrapper.className = 'mermaid-diagram';
|
||||
|
||||
const canvas = document.createElement('div');
|
||||
canvas.className = 'mermaid-canvas';
|
||||
|
||||
const viewport = document.createElement('div');
|
||||
viewport.className = 'mermaid-viewport';
|
||||
|
||||
canvas.appendChild(viewport);
|
||||
|
||||
// Toolbar
|
||||
const bar = document.createElement('div');
|
||||
bar.className = 'mermaid-zoom-bar';
|
||||
|
||||
const btnZoomOut = document.createElement('button');
|
||||
btnZoomOut.className = 'mermaid-zoom-btn mermaid-btn-zoomout';
|
||||
btnZoomOut.textContent = '−';
|
||||
btnZoomOut.ariaLabel = '縮小';
|
||||
|
||||
const btnLevel = document.createElement('button');
|
||||
btnLevel.className = 'mermaid-zoom-btn mermaid-zoom-level';
|
||||
btnLevel.textContent = '100%';
|
||||
btnLevel.ariaLabel = '重置';
|
||||
|
||||
const btnZoomIn = document.createElement('button');
|
||||
btnZoomIn.className = 'mermaid-zoom-btn mermaid-btn-zoomin';
|
||||
btnZoomIn.textContent = '+';
|
||||
btnZoomIn.ariaLabel = '放大';
|
||||
|
||||
const sep1 = document.createElement('span');
|
||||
sep1.className = 'mermaid-sep';
|
||||
|
||||
const btnFit = document.createElement('button');
|
||||
btnFit.className = 'mermaid-zoom-btn mermaid-btn-fit';
|
||||
btnFit.innerHTML = '<svg width="16" height="16" viewBox="0 0 16 16" fill="none" stroke="currentColor" stroke-width="1.5"><rect x="2" y="2" width="12" height="12" rx="2"/><path d="M2 6V2h4M10 2h4v4M14 10v4h-4M6 14H2v-4"/></svg>';
|
||||
btnFit.ariaLabel = '適合畫面';
|
||||
|
||||
const btnFullscreen = document.createElement('button');
|
||||
btnFullscreen.className = 'mermaid-zoom-btn mermaid-btn-fullscreen';
|
||||
btnFullscreen.innerHTML = '<svg width="16" height="16" viewBox="0 0 16 16" fill="none" stroke="currentColor" stroke-width="1.5"><path d="M2 6V2h4M10 2h4v4M14 10v4h-4M6 14H2v-4"/></svg>';
|
||||
btnFullscreen.ariaLabel = '全螢幕';
|
||||
|
||||
bar.append(btnZoomOut, btnLevel, btnZoomIn, sep1, btnFit, btnFullscreen);
|
||||
wrapper.append(canvas, bar);
|
||||
|
||||
return { wrapper, viewport };
|
||||
}
|
||||
|
||||
export function MermaidRenderer() {
|
||||
const { resolvedTheme } = useTheme();
|
||||
const containersRef = useRef<{ viewport: HTMLDivElement; wrapper: HTMLDivElement; source: string }[]>([]);
|
||||
const cleanupRef = useRef<(() => void)[]>([]);
|
||||
|
||||
const renderDiagrams = useCallback(async () => {
|
||||
if (containersRef.current.length === 0) return;
|
||||
|
||||
// Clean up previous event listeners
|
||||
cleanupRef.current.forEach((fn) => fn());
|
||||
cleanupRef.current = [];
|
||||
|
||||
const mermaid = (await import('mermaid')).default;
|
||||
const theme = resolvedTheme === 'dark' ? 'dark' : 'default';
|
||||
|
||||
mermaid.initialize({
|
||||
startOnLoad: false,
|
||||
theme,
|
||||
fontFamily: 'inherit',
|
||||
});
|
||||
|
||||
for (const { viewport, wrapper, source } of containersRef.current) {
|
||||
const id = `mermaid-${Math.random().toString(36).slice(2, 9)}`;
|
||||
try {
|
||||
const { svg } = await mermaid.render(id, source);
|
||||
viewport.innerHTML = svg;
|
||||
wrapper.classList.add('mermaid-rendered');
|
||||
const cleanup = attachViewer(wrapper, viewport);
|
||||
cleanupRef.current.push(cleanup);
|
||||
} catch {
|
||||
viewport.textContent = source;
|
||||
}
|
||||
}
|
||||
}, [resolvedTheme]);
|
||||
|
||||
useEffect(() => {
|
||||
const figures = document.querySelectorAll<HTMLElement>(
|
||||
'figure[data-rehype-pretty-code-figure]'
|
||||
);
|
||||
|
||||
const entries: typeof containersRef.current = [];
|
||||
|
||||
figures.forEach((figure) => {
|
||||
const code = figure.querySelector('code[data-language="mermaid"]');
|
||||
if (!code) return;
|
||||
|
||||
const source = code.textContent?.trim() ?? '';
|
||||
if (!source) return;
|
||||
|
||||
const { wrapper, viewport } = buildShell();
|
||||
figure.replaceWith(wrapper);
|
||||
entries.push({ viewport, wrapper, source });
|
||||
});
|
||||
|
||||
containersRef.current = entries;
|
||||
renderDiagrams();
|
||||
|
||||
return () => {
|
||||
cleanupRef.current.forEach((fn) => fn());
|
||||
cleanupRef.current = [];
|
||||
};
|
||||
}, [renderDiagrams]);
|
||||
|
||||
return null;
|
||||
}
|
||||
@@ -12,11 +12,11 @@ interface MetaItemProps {
|
||||
export function MetaItem({ icon: Icon, children, className, tone = 'default' }: MetaItemProps) {
|
||||
return (
|
||||
<span
|
||||
className={clsx(
|
||||
'inline-flex items-center gap-1.5 text-xs transition-colors duration-180 ease-snappy',
|
||||
tone === 'muted' ? 'text-slate-500 dark:text-slate-400' : 'text-slate-600 dark:text-slate-200',
|
||||
className
|
||||
)}
|
||||
className={clsx(
|
||||
'inline-flex items-center gap-1.5 text-xs transition-colors duration-180 ease-snappy',
|
||||
tone === 'muted' ? 'text-slate-500 dark:text-slate-400' : 'text-slate-600 dark:text-slate-200',
|
||||
className
|
||||
)}
|
||||
>
|
||||
<Icon className="h-3.5 w-3.5 text-slate-400 dark:text-slate-500" />
|
||||
<span>{children}</span>
|
||||
|
||||
20
components/native-link.tsx
Normal file
20
components/native-link.tsx
Normal file
@@ -0,0 +1,20 @@
|
||||
'use client';
|
||||
|
||||
import { useState, useEffect, ReactNode } from 'react';
|
||||
import Link from 'next/link';
|
||||
|
||||
export function NativeLink({ href, children, ...props }: { href: string; children: ReactNode; [key: string]: any }) {
|
||||
const [isSafari18, setIsSafari18] = useState(false);
|
||||
|
||||
useEffect(() => {
|
||||
const isSafari = typeof navigator !== 'undefined' && navigator.userAgent.includes('safari') && !navigator.userAgent.includes('chrome') && !navigator.userAgent.includes('firefox');
|
||||
const hasNativeTransitions = typeof document !== 'undefined' && typeof document.startViewTransition === 'function';
|
||||
setIsSafari18(isSafari && hasNativeTransitions);
|
||||
}, []);
|
||||
|
||||
if (isSafari18) {
|
||||
return <a href={href} {...props}>{children}</a>;
|
||||
}
|
||||
|
||||
return <Link href={href} {...props}>{children}</Link>;
|
||||
}
|
||||
@@ -125,14 +125,14 @@ export function NavMenu({ items }: NavMenuProps) {
|
||||
const renderDesktopChild = (item: NavLinkItem) => {
|
||||
const Icon = ICON_MAP[item.iconKey] ?? FiFile;
|
||||
return item.href ? (
|
||||
<Link
|
||||
<Link
|
||||
key={item.key}
|
||||
href={item.href}
|
||||
className="motion-link inline-flex items-center gap-2 rounded-xl px-3 py-2 text-sm text-slate-600 hover:bg-slate-100 hover:text-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/40 dark:text-slate-200 dark:hover:bg-slate-800"
|
||||
className="motion-link inline-flex items-center gap-2 rounded-xl px-3 py-2 text-sm text-slate-600 hover:bg-slate-100 hover:text-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/40 dark:text-slate-200 dark:hover:bg-slate-800 dark:hover:text-accent"
|
||||
onClick={close}
|
||||
>
|
||||
<Icon className="h-4 w-4 text-slate-400" />
|
||||
<span>{item.label}</span>
|
||||
<Icon className="h-4 w-4 shrink-0 text-slate-400" />
|
||||
<span className="whitespace-nowrap">{item.label}</span>
|
||||
</Link>
|
||||
) : null;
|
||||
};
|
||||
@@ -147,11 +147,11 @@ export function NavMenu({ items }: NavMenuProps) {
|
||||
<div key={item.key} className="flex flex-col">
|
||||
<button
|
||||
onClick={() => toggleMobileItem(item.key)}
|
||||
className="flex w-full items-center justify-between rounded-xl px-4 py-3 text-base font-medium text-slate-700 transition-colors active:bg-slate-100 dark:text-slate-200 dark:active:bg-slate-800"
|
||||
className="flex w-full items-center justify-between rounded-xl px-4 py-3 text-base font-medium text-slate-700 transition-colors active:bg-slate-100 dark:text-slate-200 dark:active:bg-slate-800 dark:hover:text-accent"
|
||||
>
|
||||
<div className="flex items-center gap-3">
|
||||
<Icon className="h-5 w-5 text-slate-400" />
|
||||
<span>{item.label}</span>
|
||||
<Icon className="h-5 w-5 shrink-0 text-slate-400" />
|
||||
<span className="whitespace-nowrap">{item.label}</span>
|
||||
</div>
|
||||
<FiChevronRight
|
||||
className={`h-5 w-5 text-slate-400 transition-transform duration-200 ${isExpanded ? 'rotate-90' : ''}`}
|
||||
@@ -172,14 +172,14 @@ export function NavMenu({ items }: NavMenuProps) {
|
||||
}
|
||||
|
||||
return item.href ? (
|
||||
<Link
|
||||
<Link
|
||||
key={item.key}
|
||||
href={item.href}
|
||||
className="flex w-full items-center gap-3 rounded-xl px-4 py-3 text-base font-medium text-slate-700 transition-colors active:bg-slate-100 dark:text-slate-200 dark:active:bg-slate-800"
|
||||
className="flex w-full items-center gap-3 rounded-xl px-4 py-3 text-base font-medium text-slate-700 transition-colors active:bg-slate-100 dark:text-slate-200 dark:active:bg-slate-800 dark:hover:text-accent"
|
||||
onClick={close}
|
||||
>
|
||||
<Icon className="h-5 w-5 text-slate-400" />
|
||||
<span>{item.label}</span>
|
||||
<Icon className="h-5 w-5 shrink-0 text-slate-400" />
|
||||
<span className="whitespace-nowrap">{item.label}</span>
|
||||
</Link>
|
||||
) : null;
|
||||
};
|
||||
@@ -189,7 +189,7 @@ export function NavMenu({ items }: NavMenuProps) {
|
||||
{/* Mobile Menu Trigger */}
|
||||
<button
|
||||
type="button"
|
||||
className="relative z-50 inline-flex h-10 w-10 items-center justify-center rounded-full text-slate-600 transition-colors hover:bg-slate-100 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/40 dark:text-slate-200 dark:hover:bg-slate-800 sm:hidden"
|
||||
className="relative z-50 inline-flex h-10 w-10 items-center justify-center rounded-full text-slate-600 transition-colors hover:bg-slate-100 hover:text-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/40 dark:text-slate-200 dark:hover:bg-slate-800 dark:hover:text-accent sm:hidden"
|
||||
aria-label={open ? '關閉選單' : '開啟選單'}
|
||||
aria-expanded={open}
|
||||
onClick={toggle}
|
||||
@@ -220,7 +220,7 @@ export function NavMenu({ items }: NavMenuProps) {
|
||||
<div className="flex items-center justify-end px-4 py-3">
|
||||
<button
|
||||
type="button"
|
||||
className="inline-flex h-10 w-10 items-center justify-center rounded-full text-slate-600 transition-colors hover:bg-slate-100 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/40 dark:text-slate-200 dark:hover:bg-slate-800"
|
||||
className="inline-flex h-10 w-10 items-center justify-center rounded-full text-slate-600 transition-colors hover:bg-slate-100 hover:text-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/40 dark:text-slate-200 dark:hover:bg-slate-800 dark:hover:text-accent"
|
||||
onClick={close}
|
||||
aria-label="Close menu"
|
||||
>
|
||||
@@ -259,15 +259,15 @@ export function NavMenu({ items }: NavMenuProps) {
|
||||
onFocus={() => openDropdown(item.key)}
|
||||
onBlur={handleBlur}
|
||||
>
|
||||
<button
|
||||
<button
|
||||
type="button"
|
||||
className="motion-link type-nav inline-flex items-center gap-1.5 rounded-full px-3 py-1 text-slate-600 hover:text-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/40 dark:text-slate-200"
|
||||
aria-haspopup="menu"
|
||||
className="motion-link type-nav inline-flex shrink-0 items-center gap-1.5 rounded-full px-3 py-1 text-slate-600 hover:text-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/40 dark:text-slate-200 dark:hover:text-accent"
|
||||
aria-haspopup="menu"
|
||||
aria-expanded={isOpen}
|
||||
>
|
||||
<Icon className="h-3.5 w-3.5 text-slate-400 transition group-hover:text-accent" />
|
||||
<span>{item.label}</span>
|
||||
<FiChevronDown className="h-3 w-3 text-slate-400 transition group-hover:text-accent" />
|
||||
<Icon className="h-3.5 w-3.5 shrink-0 text-slate-400 transition group-hover:text-accent dark:group-hover:text-accent" />
|
||||
<span className="whitespace-nowrap">{item.label}</span>
|
||||
<FiChevronDown className="h-3 w-3 shrink-0 text-slate-400 transition group-hover:text-accent dark:group-hover:text-accent" />
|
||||
</button>
|
||||
|
||||
<div
|
||||
@@ -290,11 +290,11 @@ export function NavMenu({ items }: NavMenuProps) {
|
||||
<Link
|
||||
key={item.key}
|
||||
href={item.href}
|
||||
className="motion-link type-nav group relative inline-flex items-center gap-1.5 rounded-full px-3 py-1 text-slate-600 hover:text-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/40 dark:text-slate-200"
|
||||
className="motion-link type-nav group relative inline-flex shrink-0 items-center gap-1.5 rounded-full px-3 py-1 text-slate-600 hover:text-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/40 dark:text-slate-200 dark:hover:text-accent"
|
||||
onClick={close}
|
||||
>
|
||||
<Icon className="h-3.5 w-3.5 text-slate-400 transition group-hover:text-accent" />
|
||||
<span>{item.label}</span>
|
||||
<Icon className="h-3.5 w-3.5 shrink-0 text-slate-400 transition group-hover:text-accent" />
|
||||
<span className="whitespace-nowrap">{item.label}</span>
|
||||
<span className="absolute inset-x-3 -bottom-0.5 h-px origin-left scale-x-0 bg-accent transition duration-180 ease-snappy group-hover:scale-x-100" aria-hidden="true" />
|
||||
</Link>
|
||||
) : null;
|
||||
|
||||
@@ -17,10 +17,10 @@ export function PostCard({ post, showTags = true }: PostCardProps) {
|
||||
: undefined;
|
||||
|
||||
return (
|
||||
<article className="motion-card group relative overflow-hidden rounded-xl border bg-white shadow-sm dark:border-slate-800 dark:bg-slate-900">
|
||||
<div className="pointer-events-none absolute inset-x-0 top-0 h-1 origin-left scale-x-0 bg-gradient-to-r from-blue-500 via-sky-400 to-indigo-500 opacity-80 transition-transform duration-300 ease-out group-hover:scale-x-100 dark:from-blue-400 dark:via-sky-300 dark:to-indigo-400" />
|
||||
<article className="motion-card group relative overflow-hidden rounded-xl border bg-white shadow-sm transition-all duration-300 ease-snappy hover:-translate-y-1 hover:shadow-lg dark:border-slate-800 dark:bg-slate-900">
|
||||
<div className="pointer-events-none absolute inset-x-0 top-0 h-1 origin-left scale-x-0 bg-accent opacity-80 transition-transform duration-300 ease-out group-hover:scale-x-100" />
|
||||
{cover && (
|
||||
<div className="relative w-full bg-slate-100 dark:bg-slate-800">
|
||||
<div className="relative w-full bg-slate-100 dark:bg-slate-800 overflow-hidden">
|
||||
<Image
|
||||
src={cover}
|
||||
alt={post.title}
|
||||
@@ -28,7 +28,9 @@ export function PostCard({ post, showTags = true }: PostCardProps) {
|
||||
height={360}
|
||||
sizes="(max-width: 768px) 100vw, (max-width: 1200px) 50vw, 33vw"
|
||||
loading="lazy"
|
||||
className="mx-auto max-h-60 w-full object-contain transition-transform duration-300 ease-out group-hover:scale-105"
|
||||
placeholder="blur"
|
||||
blurDataURL="data:image/jpeg;base64,/9j/4AAQSkZJRgABAQAAAQABAAD/2wBDAAYEBQYFBAYGBQYHBwYIChAKCgkJChQODwwQFxQYGBcUFhYaHSUfGhsjHBYWICwgIyYnKSopGR8tMC0oMCUoKSj/2wBDAQcHBwoIChMKChMoGhYaKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCj/wAARCAAIAAoDASIAAhEBAxEB/8QAFQABAQAAAAAAAAAAAAAAAAAAAAv/xAAhEAACAQMDBQAAAAAAAAAAAAABAgMABAUGIWGRkqGx0f/EABUBAQEAAAAAAAAAAAAAAAAAAAMF/8QAGhEAAgIDAAAAAAAAAAAAAAAAAAECEgMRkf/aAAwDAQACEQMRAD8AltJagyeH0AthI5xdrLcNM91BF5pX2HaH9bcfaSXWGaRmknyJckliyjqTzSlT54b6bk+h0R//2Q=="
|
||||
className="mx-auto w-full object-cover transition-transform duration-300 ease-out group-hover:scale-105"
|
||||
/>
|
||||
</div>
|
||||
)}
|
||||
@@ -48,15 +50,15 @@ export function PostCard({ post, showTags = true }: PostCardProps) {
|
||||
)}
|
||||
</div>
|
||||
<h2 className="text-lg font-semibold leading-snug">
|
||||
<Link
|
||||
<Link
|
||||
href={post.url}
|
||||
className="hover:text-blue-600 dark:hover:text-blue-400"
|
||||
className="hover:text-accent dark:hover:text-accent"
|
||||
>
|
||||
{post.title}
|
||||
</Link>
|
||||
</h2>
|
||||
{post.description && (
|
||||
<p className="line-clamp-3 text-sm text-slate-700 dark:text-slate-100">
|
||||
<p className="line-clamp-3 text-sm text-slate-600 dark:text-slate-300">
|
||||
{post.description}
|
||||
</p>
|
||||
)}
|
||||
|
||||
@@ -3,15 +3,15 @@
|
||||
import { useState, useEffect } from 'react';
|
||||
import { createPortal } from 'react-dom';
|
||||
import { FiList, FiX } from 'react-icons/fi';
|
||||
import { PostToc } from './post-toc';
|
||||
import { clsx, type ClassValue } from 'clsx';
|
||||
import { twMerge } from 'tailwind-merge';
|
||||
import dynamic from 'next/dynamic';
|
||||
import { cn } from '@/lib/utils';
|
||||
|
||||
function cn(...inputs: ClassValue[]) {
|
||||
return twMerge(clsx(inputs));
|
||||
}
|
||||
// Lazy load PostToc since it's not critical for initial render
|
||||
const PostToc = dynamic(() => import('./post-toc').then(mod => ({ default: mod.PostToc })), {
|
||||
ssr: false,
|
||||
});
|
||||
|
||||
export function PostLayout({ children, hasToc = true, contentKey }: { children: React.ReactNode; hasToc?: boolean; contentKey?: string }) {
|
||||
export function PostLayout({ children, hasToc = true, contentKey, wide }: { children: React.ReactNode; hasToc?: boolean; contentKey?: string; wide?: boolean }) {
|
||||
const [isTocOpen, setIsTocOpen] = useState(false); // Default closed on mobile
|
||||
const [isDesktopTocOpen, setIsDesktopTocOpen] = useState(hasToc); // Separate state for desktop
|
||||
const [mounted, setMounted] = useState(false);
|
||||
@@ -84,10 +84,10 @@ export function PostLayout({ children, hasToc = true, contentKey }: { children:
|
||||
const tocButton = hasToc && mounted ? (
|
||||
<button
|
||||
onClick={() => setIsTocOpen(true)}
|
||||
className={cn(
|
||||
"fixed bottom-6 right-16 z-40 flex h-9 items-center gap-2 rounded-full border border-slate-200 bg-white/90 px-3 text-sm font-medium text-slate-600 shadow-md backdrop-blur-sm transition hover:bg-slate-50 dark:border-slate-700 dark:bg-slate-900/90 dark:text-slate-300 dark:hover:bg-slate-800 lg:hidden",
|
||||
className={cn(
|
||||
"fixed bottom-6 right-16 z-40 flex h-9 items-center gap-2 rounded-full border border-slate-200 bg-white/90 px-3 text-sm font-medium text-slate-600 shadow-md backdrop-blur-sm transition hover:bg-slate-50 hover:text-accent dark:border-slate-700 dark:bg-slate-900/90 dark:text-slate-300 dark:hover:bg-slate-800 dark:hover:text-accent lg:hidden",
|
||||
isTocOpen ? "opacity-0 pointer-events-none" : "opacity-100"
|
||||
)}
|
||||
)}
|
||||
aria-label="Open Table of Contents"
|
||||
>
|
||||
<FiList className="h-4 w-4" />
|
||||
@@ -98,9 +98,9 @@ export function PostLayout({ children, hasToc = true, contentKey }: { children:
|
||||
const desktopTocButton = hasToc && mounted ? (
|
||||
<button
|
||||
onClick={() => setIsDesktopTocOpen(!isDesktopTocOpen)}
|
||||
className={cn(
|
||||
"fixed bottom-6 right-16 z-40 hidden h-9 items-center gap-2 rounded-full border border-slate-200 bg-white/90 px-3 text-sm font-medium text-slate-600 shadow-md backdrop-blur-sm transition hover:bg-slate-50 dark:border-slate-700 dark:bg-slate-900/90 dark:text-slate-300 dark:hover:bg-slate-800 lg:flex",
|
||||
)}
|
||||
className={cn(
|
||||
"fixed bottom-6 right-16 z-40 hidden h-9 items-center gap-2 rounded-full border border-slate-200 bg-white/90 px-3 text-sm font-medium text-slate-600 shadow-md backdrop-blur-sm transition hover:bg-slate-50 hover:text-accent dark:border-slate-700 dark:bg-slate-900/90 dark:text-slate-300 dark:hover:bg-slate-800 dark:hover:text-accent lg:flex",
|
||||
)}
|
||||
aria-label={isDesktopTocOpen ? "Close Table of Contents" : "Open Table of Contents"}
|
||||
>
|
||||
<FiList className="h-4 w-4" />
|
||||
@@ -116,7 +116,7 @@ export function PostLayout({ children, hasToc = true, contentKey }: { children:
|
||||
)}>
|
||||
{/* Main Content Area */}
|
||||
<div className="min-w-0">
|
||||
<div className={cn("mx-auto transition-all duration-500 ease-snappy", isDesktopTocOpen && hasToc ? "max-w-3xl" : "max-w-4xl")}>
|
||||
<div className={cn("mx-auto transition-all duration-500 ease-snappy", isDesktopTocOpen && hasToc ? "max-w-3xl" : wide ? "max-w-5xl" : "max-w-4xl")}>
|
||||
{children}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@@ -7,9 +7,10 @@ import { MetaItem } from './meta-item';
|
||||
|
||||
interface Props {
|
||||
post: Post;
|
||||
priority?: boolean;
|
||||
}
|
||||
|
||||
export function PostListItem({ post }: Props) {
|
||||
export function PostListItem({ post, priority = false }: Props) {
|
||||
const cover =
|
||||
post.feature_image && post.feature_image.startsWith('../assets')
|
||||
? post.feature_image.replace('../assets', '/assets')
|
||||
@@ -20,7 +21,7 @@ export function PostListItem({ post }: Props) {
|
||||
|
||||
return (
|
||||
<article className="motion-card group relative flex gap-4 rounded-2xl border border-white/40 bg-white/60 p-5 shadow-lg backdrop-blur-md transition-all hover:scale-[1.01] hover:shadow-xl dark:border-white/10 dark:bg-slate-900/60">
|
||||
<div className="pointer-events-none absolute inset-x-0 top-0 h-0.5 origin-left scale-x-0 bg-gradient-to-r from-blue-500 via-sky-400 to-indigo-500 opacity-80 transition-transform duration-300 ease-out group-hover:scale-x-100 dark:from-blue-400 dark:via-sky-300 dark:to-indigo-400" />
|
||||
<div className="pointer-events-none absolute inset-x-0 top-0 h-0.5 origin-left scale-x-0 bg-accent opacity-80 transition-transform duration-300 ease-out group-hover:scale-x-100" />
|
||||
{cover && (
|
||||
<div className="relative flex h-24 w-24 flex-none overflow-hidden rounded-md bg-slate-100 dark:bg-slate-800 sm:h-auto sm:w-40">
|
||||
<Image
|
||||
@@ -29,7 +30,10 @@ export function PostListItem({ post }: Props) {
|
||||
width={320}
|
||||
height={240}
|
||||
sizes="(max-width: 640px) 96px, 160px"
|
||||
loading="lazy"
|
||||
loading={priority ? undefined : 'lazy'}
|
||||
priority={priority}
|
||||
placeholder="blur"
|
||||
blurDataURL="data:image/jpeg;base64,/9j/4AAQSkZJRgABAQAAAQABAAD/2wBDAAYEBQYFBAYGBQYHBwYIChAKCgkJChQODwwQFxQYGBcUFhYaHSUfGhsjHBYWICwgIyYnKSopGR8tMC0oMCUoKSj/2wBDAQcHBwoIChMKChMoGhYaKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCj/wAARCAAIAAoDASIAAhEBAxEB/8QAFQABAQAAAAAAAAAAAAAAAAAAAAv/xAAhEAACAQMDBQAAAAAAAAAAAAABAgMABAUGIWGRkqGx0f/EABUBAQEAAAAAAAAAAAAAAAAAAAMF/8QAGhEAAgIDAAAAAAAAAAAAAAAAAAECEgMRkf/aAAwDAQACEQMRAD8AltJagyeH0AthI5xdrLcNM91BF5pX2HaH9bcfaSXWGaRmknyJckliyjqTzSlT54b6bk+h0R//2Q=="
|
||||
className="h-full w-full object-cover transition-transform duration-300 ease-out group-hover:scale-105"
|
||||
/>
|
||||
</div>
|
||||
@@ -49,11 +53,11 @@ export function PostListItem({ post }: Props) {
|
||||
</MetaItem>
|
||||
)}
|
||||
</div>
|
||||
<h2 className="text-base font-semibold leading-snug text-slate-900 hover:text-accent sm:text-lg dark:text-slate-50 dark:hover:text-accent">
|
||||
<h2 className="type-body font-semibold leading-snug hover:text-accent sm:type-title">
|
||||
<Link href={post.url}>{post.title}</Link>
|
||||
</h2>
|
||||
{excerpt && (
|
||||
<p className="line-clamp-2 text-sm text-slate-600 group-hover:text-slate-800 dark:text-slate-300 dark:group-hover:text-slate-100">
|
||||
<p className="line-clamp-2 text-sm text-slate-600 dark:text-slate-300">
|
||||
{excerpt}
|
||||
</p>
|
||||
)}
|
||||
|
||||
@@ -113,10 +113,10 @@ export function PostListWithControls({ posts, pageSize }: Props) {
|
||||
<input
|
||||
id="post-search"
|
||||
type="search"
|
||||
placeholder="標題、標籤、摘要關鍵字"
|
||||
placeholder="搜尋文章…"
|
||||
value={searchTerm}
|
||||
onChange={(event) => setSearchTerm(event.target.value)}
|
||||
className="w-full rounded-full border border-slate-200 bg-white py-1.5 pl-9 pr-3 text-sm text-slate-700 shadow-sm transition duration-180 ease-snappy focus:border-blue-500 focus:outline-none focus:ring-2 focus:ring-blue-200 dark:border-slate-700 dark:bg-slate-900 dark:text-slate-100 dark:focus:ring-blue-500"
|
||||
className="w-full rounded-full border border-slate-200 bg-white py-1.5 pl-9 pr-3 text-sm text-slate-700 shadow-sm transition duration-180 ease-snappy focus:border-accent focus:outline-none focus:ring-2 focus:ring-accent/20 dark:border-slate-700 dark:bg-slate-900 dark:text-slate-100 dark:focus:border-accent dark:focus:ring-accent/30"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@@ -1,48 +1,86 @@
|
||||
'use client';
|
||||
|
||||
import { useEffect, useState } from 'react';
|
||||
import { useEffect, useState, useCallback } from 'react';
|
||||
import { createPortal } from 'react-dom';
|
||||
|
||||
function supportsScrollDrivenAnimations(): boolean {
|
||||
if (typeof CSS === 'undefined') return false;
|
||||
return CSS.supports?.('animation-timeline', 'scroll()') ?? false;
|
||||
}
|
||||
|
||||
export function ReadingProgress() {
|
||||
const [mounted, setMounted] = useState(false);
|
||||
const [progress, setProgress] = useState(0);
|
||||
const [useScrollDriven, setUseScrollDriven] = useState(false);
|
||||
|
||||
useEffect(() => {
|
||||
setMounted(true);
|
||||
const updateMode = () => {
|
||||
const prefersReducedMotion = window.matchMedia(
|
||||
'(prefers-reduced-motion: reduce)'
|
||||
).matches;
|
||||
setUseScrollDriven(
|
||||
supportsScrollDrivenAnimations() && !prefersReducedMotion
|
||||
);
|
||||
};
|
||||
updateMode();
|
||||
const mq = window.matchMedia('(prefers-reduced-motion: reduce)');
|
||||
mq.addEventListener('change', updateMode);
|
||||
return () => mq.removeEventListener('change', updateMode);
|
||||
}, []);
|
||||
|
||||
useEffect(() => {
|
||||
if (!mounted) return;
|
||||
const handleScroll = useCallback(() => {
|
||||
if (!mounted || useScrollDriven) return;
|
||||
|
||||
const { scrollTop, scrollHeight, clientHeight } = document.documentElement;
|
||||
const total = scrollHeight - clientHeight;
|
||||
if (total <= 0) {
|
||||
setProgress(0);
|
||||
return;
|
||||
}
|
||||
const value = Math.min(100, Math.max(0, (scrollTop / total) * 100));
|
||||
setProgress(value);
|
||||
}, [mounted, useScrollDriven]);
|
||||
|
||||
const handleScroll = () => {
|
||||
const { scrollTop, scrollHeight, clientHeight } = document.documentElement;
|
||||
const total = scrollHeight - clientHeight;
|
||||
if (total <= 0) {
|
||||
setProgress(0);
|
||||
return;
|
||||
}
|
||||
const value = Math.min(100, Math.max(0, (scrollTop / total) * 100));
|
||||
setProgress(value);
|
||||
};
|
||||
useEffect(() => {
|
||||
if (!mounted || useScrollDriven) return;
|
||||
|
||||
handleScroll();
|
||||
window.addEventListener('scroll', handleScroll, { passive: true });
|
||||
window.addEventListener('scroll', handleScroll, { passive: true, signal: AbortSignal.timeout(60000) });
|
||||
return () => window.removeEventListener('scroll', handleScroll);
|
||||
}, [mounted]);
|
||||
|
||||
}, [mounted, useScrollDriven, handleScroll]);
|
||||
|
||||
if (!mounted) return null;
|
||||
|
||||
return createPortal(
|
||||
<div className="pointer-events-none fixed inset-x-0 top-0 z-[1200] h-1.5 bg-transparent">
|
||||
<div className="relative h-1.5 w-full overflow-visible">
|
||||
<div
|
||||
className="absolute inset-y-0 left-0 w-full origin-left rounded-full bg-gradient-to-r from-[rgba(124,58,237,0.9)] via-[rgba(167,139,250,0.9)] to-[rgba(14,165,233,0.8)] shadow-[0_0_12px_rgba(124,58,237,0.5)] transition-[transform,opacity] duration-300 ease-out"
|
||||
style={{ transform: `scaleX(${progress / 100})`, opacity: progress > 0 ? 1 : 0 }}
|
||||
>
|
||||
<span className="absolute -right-2 top-1/2 h-3 w-3 -translate-y-1/2 rounded-full bg-white/80 blur-[1px] dark:bg-slate-900/80" aria-hidden="true" />
|
||||
</div>
|
||||
<span className="absolute inset-x-0 top-2 h-px bg-gradient-to-r from-transparent via-blue-200/40 to-transparent blur-sm dark:via-blue-900/30" aria-hidden="true" />
|
||||
{useScrollDriven ? (
|
||||
<div aria-hidden="true" className="reading-progress-bar-scroll-driven absolute inset-y-0 left-0 w-full origin-left rounded-full bg-accent">
|
||||
<span
|
||||
className="absolute -right-2 top-1/2 h-3 w-3 -translate-y-1/2 rounded-full bg-white/80 blur-[1px] dark:bg-slate-900/80"
|
||||
aria-hidden="true"
|
||||
/>
|
||||
</div>
|
||||
) : (
|
||||
<div
|
||||
className="absolute inset-y-0 left-0 w-full origin-left rounded-full bg-accent will-change-transform transition-[transform,opacity] duration-300 ease-out"
|
||||
style={{
|
||||
transform: `scaleX(${progress / 100})`,
|
||||
opacity: progress > 0 ? 1 : 0
|
||||
}}
|
||||
aria-hidden="true"
|
||||
>
|
||||
<span
|
||||
className="absolute -right-2 top-1/2 h-3 w-3 -translate-y-1/2 rounded-full bg-white/80 blur-[1px] dark:bg-slate-900/80"
|
||||
aria-hidden="true"
|
||||
/>
|
||||
</div>
|
||||
)}
|
||||
<span
|
||||
className="absolute inset-x-0 top-2 h-px bg-gradient-to-r from-transparent via-accent-soft to-transparent blur-sm"
|
||||
aria-hidden="true"
|
||||
/>
|
||||
</div>
|
||||
</div>,
|
||||
document.body
|
||||
|
||||
64
components/repo-card.tsx
Normal file
64
components/repo-card.tsx
Normal file
@@ -0,0 +1,64 @@
|
||||
import Link from 'next/link';
|
||||
import { FiExternalLink } from 'react-icons/fi';
|
||||
import type { RepoSummary } from '@/lib/github';
|
||||
import { getLanguageColor } from '@/lib/github-lang-colors';
|
||||
|
||||
interface RepoCardProps {
|
||||
repo: RepoSummary;
|
||||
animationDelay?: number;
|
||||
}
|
||||
|
||||
export function RepoCard({ repo, animationDelay = 0 }: RepoCardProps) {
|
||||
const langColor = getLanguageColor(repo.language);
|
||||
|
||||
return (
|
||||
<li
|
||||
className={`motion-card group relative flex h-full flex-col rounded-2xl border border-white/40 bg-white/60 p-5 shadow-lg backdrop-blur-md transition-all hover:scale-[1.01] hover:shadow-xl dark:border-white/10 dark:bg-slate-900/60 ${animationDelay > 0 ? 'repo-card-enter' : ''}`}
|
||||
style={
|
||||
animationDelay > 0 ? { animationDelay: `${animationDelay}ms` } : undefined
|
||||
}
|
||||
>
|
||||
<div className="pointer-events-none absolute inset-x-0 top-0 h-0.5 origin-left scale-x-0 bg-accent opacity-80 transition-transform duration-300 ease-out group-hover:scale-x-100" />
|
||||
<div className="flex items-start justify-between gap-2">
|
||||
<Link
|
||||
href={repo.htmlUrl}
|
||||
prefetch={false}
|
||||
target="_blank"
|
||||
rel="noreferrer"
|
||||
className="type-base inline-flex items-center gap-2 font-semibold text-slate-900 transition-colors hover:text-accent dark:text-slate-50 dark:hover:text-accent"
|
||||
>
|
||||
{repo.name}
|
||||
<FiExternalLink className="h-3.5 w-3.5 opacity-0 transition-opacity group-hover:opacity-100" />
|
||||
</Link>
|
||||
{repo.stargazersCount > 0 && (
|
||||
<span className="inline-flex shrink-0 items-center gap-1 rounded-lg bg-amber-50 px-2 py-0.5 text-xs font-medium text-amber-700 dark:bg-amber-900/40 dark:text-amber-300">
|
||||
★ {repo.stargazersCount}
|
||||
</span>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{repo.description && (
|
||||
<p className="mt-2 flex-1 line-clamp-2 text-sm text-slate-600 dark:text-slate-300">
|
||||
{repo.description}
|
||||
</p>
|
||||
)}
|
||||
|
||||
<div className="mt-3 flex items-center justify-between gap-2 text-xs text-slate-500 dark:text-slate-400">
|
||||
<span className="inline-flex items-center gap-1.5">
|
||||
<span
|
||||
className="h-2 w-2 shrink-0 rounded-full"
|
||||
style={{ backgroundColor: langColor }}
|
||||
aria-hidden
|
||||
/>
|
||||
{repo.language ?? '其他'}
|
||||
</span>
|
||||
<span suppressHydrationWarning>
|
||||
更新於{' '}
|
||||
{repo.updatedAt
|
||||
? new Date(repo.updatedAt).toLocaleDateString('zh-TW')
|
||||
: '未知'}
|
||||
</span>
|
||||
</div>
|
||||
</li>
|
||||
);
|
||||
}
|
||||
@@ -1,13 +1,69 @@
|
||||
'use client';
|
||||
|
||||
import Link from 'next/link';
|
||||
import Image from 'next/image';
|
||||
import { useEffect, useRef, useState } from 'react';
|
||||
import { FaGithub, FaMastodon, FaLinkedin } from 'react-icons/fa';
|
||||
import { FiTrendingUp, FiArrowRight } from 'react-icons/fi';
|
||||
import { siteConfig } from '@/lib/config';
|
||||
import { getAllTagsWithCount } from '@/lib/posts';
|
||||
import { allPages } from 'contentlayer2/generated';
|
||||
import { MastodonFeed } from './mastodon-feed';
|
||||
import dynamic from 'next/dynamic';
|
||||
|
||||
// Lazy load MastodonFeed - only load when sidebar is visible
|
||||
const MastodonFeed = dynamic(() => import('./mastodon-feed').then(mod => ({ default: mod.MastodonFeed })), {
|
||||
ssr: false,
|
||||
loading: () => <div className="h-32 w-full animate-pulse rounded-xl bg-slate-100 dark:bg-slate-800" />,
|
||||
});
|
||||
|
||||
/** Shared sidebar content for desktop aside and mobile drawer */
|
||||
export function RightSidebarContent({ forceLoadFeed = false }: { forceLoadFeed?: boolean }) {
|
||||
const [shouldLoadFeed, setShouldLoadFeed] = useState(forceLoadFeed);
|
||||
const feedRef = useRef<HTMLDivElement>(null);
|
||||
|
||||
useEffect(() => {
|
||||
if (forceLoadFeed) {
|
||||
setShouldLoadFeed(true);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!feedRef.current) return;
|
||||
|
||||
let observer: IntersectionObserver | null = null;
|
||||
let cleanupRequested = false;
|
||||
|
||||
const setupObserver = () => {
|
||||
if (cleanupRequested) return;
|
||||
|
||||
const el = feedRef.current;
|
||||
if (!el) return;
|
||||
|
||||
observer = new IntersectionObserver(
|
||||
(entries) => {
|
||||
if (entries[0].isIntersecting) {
|
||||
setShouldLoadFeed(true);
|
||||
observer?.disconnect();
|
||||
}
|
||||
},
|
||||
{ rootMargin: '100px' }
|
||||
);
|
||||
|
||||
observer.observe(el);
|
||||
};
|
||||
|
||||
// Defer observer setup for better initial performance
|
||||
requestAnimationFrame(() => {
|
||||
if (!cleanupRequested && feedRef.current) {
|
||||
setupObserver();
|
||||
}
|
||||
});
|
||||
|
||||
return () => {
|
||||
cleanupRequested = true;
|
||||
observer?.disconnect();
|
||||
};
|
||||
}, [forceLoadFeed]);
|
||||
|
||||
export function RightSidebar() {
|
||||
const tags = getAllTagsWithCount().slice(0, 5);
|
||||
|
||||
const aboutPage =
|
||||
@@ -38,9 +94,8 @@ export function RightSidebar() {
|
||||
].filter(Boolean) as { key: string; href: string; icon: any; label: string }[];
|
||||
|
||||
return (
|
||||
<aside className="hidden lg:block">
|
||||
<div className="sticky top-20 flex flex-col gap-4">
|
||||
<section className="motion-card group relative overflow-hidden rounded-xl border bg-white px-4 py-4 shadow-sm hover:bg-slate-50 dark:border-slate-800 dark:bg-slate-900 dark:text-slate-100">
|
||||
<div className="flex flex-col gap-4">
|
||||
<section className="motion-card group relative overflow-hidden rounded-xl border bg-white px-4 py-4 shadow-sm hover:bg-slate-50 dark:border-slate-800 dark:bg-slate-900 dark:text-slate-100 dark:hover:bg-slate-800/80">
|
||||
<div className="pointer-events-none absolute -left-10 -top-10 h-24 w-24 rounded-full bg-sky-300/35 blur-3xl mix-blend-soft-light motion-safe:animate-float-soft dark:bg-sky-500/25" />
|
||||
<div className="pointer-events-none absolute -bottom-12 right-[-2.5rem] h-28 w-28 rounded-full bg-indigo-300/30 blur-3xl mix-blend-soft-light motion-safe:animate-float-soft dark:bg-indigo-500/20" />
|
||||
|
||||
@@ -74,7 +129,7 @@ export function RightSidebar() {
|
||||
target="_blank"
|
||||
rel="noopener noreferrer"
|
||||
aria-label={item.label}
|
||||
className="motion-link inline-flex h-8 w-8 items-center justify-center rounded-full bg-slate-100 text-slate-600 hover:-translate-y-0.5 hover:bg-accent-soft hover:text-accent dark:bg-slate-800 dark:text-slate-200"
|
||||
className="motion-link inline-flex h-8 w-8 items-center justify-center rounded-full bg-slate-100 text-slate-600 hover:-translate-y-0.5 hover:bg-accent-soft hover:text-accent dark:bg-slate-800 dark:text-slate-200 dark:hover:text-accent"
|
||||
>
|
||||
<item.icon className="h-4 w-4" />
|
||||
</a>
|
||||
@@ -91,8 +146,10 @@ export function RightSidebar() {
|
||||
</div>
|
||||
</section>
|
||||
|
||||
{/* Mastodon Feed */}
|
||||
<MastodonFeed />
|
||||
{/* Mastodon Feed - Lazy loaded when visible */}
|
||||
<div ref={feedRef}>
|
||||
{shouldLoadFeed && <MastodonFeed />}
|
||||
</div>
|
||||
|
||||
{tags.length > 0 && (
|
||||
<section className="motion-card rounded-xl border bg-white px-4 py-3 shadow-sm dark:border-slate-800 dark:bg-slate-900 dark:text-slate-100">
|
||||
@@ -110,7 +167,7 @@ export function RightSidebar() {
|
||||
<Link
|
||||
key={tag}
|
||||
href={`/tags/${slug}`}
|
||||
className={`${sizeClass} tag-chip rounded-full bg-accent-soft px-2 py-0.5 text-accent-textLight transition hover:bg-accent hover:text-white dark:bg-slate-800 dark:text-slate-100 dark:hover:bg-slate-700`}
|
||||
className={`${sizeClass} tag-chip rounded-full bg-accent-soft px-2 py-0.5 text-accent-textLight transition hover:bg-accent hover:text-white dark:bg-slate-800 dark:text-slate-100 dark:hover:bg-slate-700 dark:hover:text-white`}
|
||||
>
|
||||
{tag}
|
||||
</Link>
|
||||
@@ -124,13 +181,22 @@ export function RightSidebar() {
|
||||
</span>
|
||||
<Link
|
||||
href="/tags"
|
||||
className="motion-link text-accent-textLight hover:text-accent dark:text-accent-textDark"
|
||||
className="motion-link text-accent-textLight hover:text-accent dark:text-accent-textDark dark:hover:text-accent"
|
||||
>
|
||||
前往
|
||||
</Link>
|
||||
</div>
|
||||
</section>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
export function RightSidebar() {
|
||||
return (
|
||||
<aside className="hidden lg:block">
|
||||
<div className="sticky top-20">
|
||||
<RightSidebarContent />
|
||||
</div>
|
||||
</aside>
|
||||
);
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
'use client';
|
||||
|
||||
import { useEffect, useRef } from 'react';
|
||||
import { useEffect, useRef, useCallback } from 'react';
|
||||
import clsx from 'clsx';
|
||||
|
||||
interface ScrollRevealProps {
|
||||
@@ -15,6 +15,20 @@ export function ScrollReveal({
|
||||
once = true
|
||||
}: ScrollRevealProps) {
|
||||
const ref = useRef<HTMLDivElement | null>(null);
|
||||
const observerRef = useRef<IntersectionObserver | null>(null);
|
||||
|
||||
const handleObserver = useCallback((entries: IntersectionObserverEntry[]) => {
|
||||
entries.forEach((entry) => {
|
||||
if (entry.isIntersecting) {
|
||||
entry.target.classList.add('is-visible');
|
||||
if (once && observerRef.current) {
|
||||
observerRef.current.unobserve(entry.target);
|
||||
}
|
||||
} else if (!once) {
|
||||
entry.target.classList.remove('is-visible');
|
||||
}
|
||||
});
|
||||
}, [once]);
|
||||
|
||||
useEffect(() => {
|
||||
const el = ref.current;
|
||||
@@ -26,35 +40,26 @@ export function ScrollReveal({
|
||||
return;
|
||||
}
|
||||
|
||||
const observer = new IntersectionObserver(
|
||||
(entries) => {
|
||||
entries.forEach((entry) => {
|
||||
if (entry.isIntersecting) {
|
||||
entry.target.classList.add('is-visible');
|
||||
if (once) observer.unobserve(entry.target);
|
||||
} else if (!once) {
|
||||
entry.target.classList.remove('is-visible');
|
||||
}
|
||||
});
|
||||
},
|
||||
observerRef.current = new IntersectionObserver(
|
||||
handleObserver,
|
||||
{
|
||||
threshold: 0.05,
|
||||
rootMargin: '0px 0px -20% 0px'
|
||||
}
|
||||
);
|
||||
|
||||
observer.observe(el);
|
||||
observerRef.current.observe(el);
|
||||
|
||||
// Fallback timeout for slow connections
|
||||
const fallback = window.setTimeout(() => {
|
||||
// Fallback timeout for slow connections - reduce to 300ms
|
||||
const fallback = setTimeout(() => {
|
||||
el.classList.add('is-visible');
|
||||
}, 500);
|
||||
}, 300);
|
||||
|
||||
return () => {
|
||||
observer.disconnect();
|
||||
window.clearTimeout(fallback);
|
||||
observerRef.current?.disconnect();
|
||||
clearTimeout(fallback);
|
||||
};
|
||||
}, [once]);
|
||||
}, [handleObserver, once]);
|
||||
|
||||
return (
|
||||
<div
|
||||
|
||||
@@ -1,78 +1,69 @@
|
||||
'use client';
|
||||
|
||||
import { useEffect, useRef, useState } from 'react';
|
||||
import { createPortal } from 'react-dom';
|
||||
import { FiSearch, FiX } from 'react-icons/fi';
|
||||
import { useCallback, useEffect, useRef, useState } from 'react';
|
||||
import { useRouter } from 'next/navigation';
|
||||
import { Command } from 'cmdk';
|
||||
import {
|
||||
FiSearch,
|
||||
FiHome,
|
||||
FiFileText,
|
||||
FiTag,
|
||||
FiBook
|
||||
} from 'react-icons/fi';
|
||||
import { cn } from '@/lib/utils';
|
||||
|
||||
interface PagefindResult {
|
||||
url: string;
|
||||
meta: { title?: string };
|
||||
excerpt?: string;
|
||||
}
|
||||
|
||||
interface QuickAction {
|
||||
id: string;
|
||||
title: string;
|
||||
url: string;
|
||||
icon: React.ReactNode;
|
||||
}
|
||||
|
||||
interface SearchModalProps {
|
||||
isOpen: boolean;
|
||||
onClose: () => void;
|
||||
recentPosts?: { title: string; url: string }[];
|
||||
}
|
||||
|
||||
export function SearchModal({ isOpen, onClose }: SearchModalProps) {
|
||||
const [isLoaded, setIsLoaded] = useState(false);
|
||||
const searchContainerRef = useRef<HTMLDivElement>(null);
|
||||
const pagefindUIRef = useRef<any>(null);
|
||||
export function SearchModal({
|
||||
isOpen,
|
||||
onClose,
|
||||
recentPosts = []
|
||||
}: SearchModalProps) {
|
||||
const router = useRouter();
|
||||
const [search, setSearch] = useState('');
|
||||
const [results, setResults] = useState<PagefindResult[]>([]);
|
||||
const [loading, setLoading] = useState(false);
|
||||
const [pagefindReady, setPagefindReady] = useState(false);
|
||||
const pagefindRef = useRef<{
|
||||
init: () => void;
|
||||
options: (opts: { bundlePath: string }) => Promise<void>;
|
||||
preload: (query: string) => void;
|
||||
debouncedSearch: (
|
||||
query: string,
|
||||
opts: object,
|
||||
debounceMs: number
|
||||
) => Promise<{ results: { data: () => Promise<PagefindResult> }[] } | null>;
|
||||
} | null>(null);
|
||||
|
||||
// Initialize Pagefind when modal opens
|
||||
useEffect(() => {
|
||||
if (!isOpen) return;
|
||||
|
||||
let link: HTMLLinkElement | null = null;
|
||||
let script: HTMLScriptElement | null = null;
|
||||
|
||||
// Load Pagefind UI dynamically when modal opens
|
||||
const loadPagefind = async () => {
|
||||
if (pagefindUIRef.current) {
|
||||
// Already loaded
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
// Load Pagefind UI CSS
|
||||
link = document.createElement('link');
|
||||
link.rel = 'stylesheet';
|
||||
link.href = '/_pagefind/pagefind-ui.css';
|
||||
document.head.appendChild(link);
|
||||
|
||||
// Load Pagefind UI JS
|
||||
script = document.createElement('script');
|
||||
script.src = '/_pagefind/pagefind-ui.js';
|
||||
script.onload = () => {
|
||||
if (searchContainerRef.current && (window as any).PagefindUI) {
|
||||
pagefindUIRef.current = new (window as any).PagefindUI({
|
||||
element: searchContainerRef.current,
|
||||
bundlePath: '/_pagefind/',
|
||||
showSubResults: true,
|
||||
showImages: false,
|
||||
excerptLength: 15,
|
||||
resetStyles: false,
|
||||
autofocus: true,
|
||||
translations: {
|
||||
placeholder: '搜尋文章...',
|
||||
clear_search: '清除',
|
||||
load_more: '載入更多結果',
|
||||
search_label: '搜尋此網站',
|
||||
filters_label: '篩選',
|
||||
zero_results: '找不到 [SEARCH_TERM] 的結果',
|
||||
many_results: '找到 [COUNT] 個 [SEARCH_TERM] 的結果',
|
||||
one_result: '找到 [COUNT] 個 [SEARCH_TERM] 的結果',
|
||||
alt_search: '找不到 [SEARCH_TERM] 的結果。改為顯示 [DIFFERENT_TERM] 的結果',
|
||||
search_suggestion: '找不到 [SEARCH_TERM] 的結果。請嘗試以下搜尋:',
|
||||
searching: '搜尋中...'
|
||||
}
|
||||
});
|
||||
setIsLoaded(true);
|
||||
|
||||
// Auto-focus the search input after a short delay
|
||||
setTimeout(() => {
|
||||
const input = searchContainerRef.current?.querySelector('input[type="search"]') as HTMLInputElement;
|
||||
if (input) {
|
||||
input.focus();
|
||||
}
|
||||
}, 100);
|
||||
}
|
||||
};
|
||||
document.head.appendChild(script);
|
||||
const pagefindUrl = `${window.location.origin}/_pagefind/pagefind.js`;
|
||||
const pagefind = await import(/* webpackIgnore: true */ pagefindUrl);
|
||||
await pagefind.options({ bundlePath: '/_pagefind/' });
|
||||
pagefind.init();
|
||||
pagefindRef.current = pagefind;
|
||||
setPagefindReady(true);
|
||||
} catch (error) {
|
||||
console.error('Failed to load Pagefind:', error);
|
||||
}
|
||||
@@ -80,102 +71,179 @@ export function SearchModal({ isOpen, onClose }: SearchModalProps) {
|
||||
|
||||
loadPagefind();
|
||||
|
||||
// Cleanup function to prevent duplicate initializations
|
||||
return () => {
|
||||
if (link && link.parentNode) {
|
||||
link.parentNode.removeChild(link);
|
||||
}
|
||||
if (script && script.parentNode) {
|
||||
script.parentNode.removeChild(script);
|
||||
}
|
||||
if (pagefindUIRef.current && pagefindUIRef.current.destroy) {
|
||||
pagefindUIRef.current.destroy();
|
||||
pagefindUIRef.current = null;
|
||||
}
|
||||
pagefindRef.current = null;
|
||||
setPagefindReady(false);
|
||||
setSearch('');
|
||||
setResults([]);
|
||||
};
|
||||
}, [isOpen]);
|
||||
|
||||
// Debounced search when user types
|
||||
useEffect(() => {
|
||||
const handleEscape = (e: KeyboardEvent) => {
|
||||
if (e.key === 'Escape' && isOpen) {
|
||||
onClose();
|
||||
}
|
||||
};
|
||||
|
||||
document.addEventListener('keydown', handleEscape);
|
||||
return () => document.removeEventListener('keydown', handleEscape);
|
||||
}, [isOpen, onClose]);
|
||||
|
||||
useEffect(() => {
|
||||
// Prevent body scroll when modal is open
|
||||
if (isOpen) {
|
||||
document.body.style.overflow = 'hidden';
|
||||
} else {
|
||||
document.body.style.overflow = '';
|
||||
const query = search.trim();
|
||||
if (!query || !pagefindRef.current) {
|
||||
setResults([]);
|
||||
setLoading(false);
|
||||
return;
|
||||
}
|
||||
return () => {
|
||||
document.body.style.overflow = '';
|
||||
};
|
||||
}, [isOpen]);
|
||||
|
||||
if (!isOpen) return null;
|
||||
setLoading(true);
|
||||
pagefindRef.current.preload(query);
|
||||
|
||||
// Use portal to render modal at document body level to avoid z-index stacking context issues
|
||||
if (typeof window === 'undefined') return null;
|
||||
const timer = setTimeout(async () => {
|
||||
const pagefind = pagefindRef.current;
|
||||
if (!pagefind) return;
|
||||
|
||||
return createPortal(
|
||||
<div
|
||||
className="fixed inset-0 z-[9999] flex items-start justify-center bg-black/50 backdrop-blur-sm pt-20 px-4"
|
||||
onClick={onClose}
|
||||
const searchResult = await pagefind.debouncedSearch(query, {}, 300);
|
||||
if (searchResult === null) return; // Superseded by newer search
|
||||
|
||||
const dataPromises = searchResult.results.slice(0, 10).map((r) => r.data());
|
||||
const items = await Promise.all(dataPromises);
|
||||
setResults(items);
|
||||
setLoading(false);
|
||||
}, 300);
|
||||
|
||||
return () => clearTimeout(timer);
|
||||
}, [search, pagefindReady]);
|
||||
|
||||
const handleSelect = useCallback(
|
||||
(url: string) => {
|
||||
onClose();
|
||||
router.push(url);
|
||||
},
|
||||
[onClose, router]
|
||||
);
|
||||
|
||||
const navActions: QuickAction[] = [
|
||||
{ id: 'home', title: '首頁', url: '/', icon: <FiHome className="size-4" /> },
|
||||
{
|
||||
id: 'blog',
|
||||
title: '部落格',
|
||||
url: '/blog',
|
||||
icon: <FiFileText className="size-4" />
|
||||
},
|
||||
{
|
||||
id: 'tags',
|
||||
title: '標籤',
|
||||
url: '/tags',
|
||||
icon: <FiTag className="size-4" />
|
||||
}
|
||||
];
|
||||
|
||||
const recentPostActions: QuickAction[] = recentPosts.map((p) => ({
|
||||
id: `post-${p.url}`,
|
||||
title: p.title,
|
||||
url: p.url,
|
||||
icon: <FiBook className="size-4" />
|
||||
}));
|
||||
|
||||
return (
|
||||
<Command.Dialog
|
||||
open={isOpen}
|
||||
onOpenChange={(open) => !open && onClose()}
|
||||
label="全站搜尋"
|
||||
shouldFilter={false}
|
||||
className="fixed left-1/2 top-[20%] z-[9999] w-full max-w-2xl -translate-x-1/2 rounded-2xl border border-white/40 bg-white/95 shadow-2xl backdrop-blur-md dark:border-white/10 dark:bg-slate-900/95"
|
||||
>
|
||||
<div
|
||||
className="w-full max-w-3xl rounded-2xl border border-white/40 bg-white/95 shadow-2xl backdrop-blur-md dark:border-white/10 dark:bg-slate-900/95"
|
||||
onClick={(e) => e.stopPropagation()}
|
||||
>
|
||||
{/* Header */}
|
||||
<div className="flex items-center justify-between border-b border-slate-200 px-6 py-4 dark:border-slate-700">
|
||||
<div className="flex items-center gap-2 text-slate-600 dark:text-slate-300">
|
||||
<FiSearch className="h-5 w-5" />
|
||||
<span className="text-sm font-medium">全站搜尋</span>
|
||||
</div>
|
||||
<button
|
||||
onClick={onClose}
|
||||
className="inline-flex h-8 w-8 items-center justify-center rounded-full text-slate-500 transition hover:bg-slate-100 hover:text-slate-700 dark:text-slate-400 dark:hover:bg-slate-800 dark:hover:text-slate-200"
|
||||
aria-label="關閉搜尋"
|
||||
>
|
||||
<FiX className="h-5 w-5" />
|
||||
</button>
|
||||
</div>
|
||||
|
||||
{/* Search Container */}
|
||||
<div className="max-h-[60vh] overflow-y-auto p-6">
|
||||
<div
|
||||
ref={searchContainerRef}
|
||||
className="pagefind-search"
|
||||
data-pagefind-ui
|
||||
/>
|
||||
{!isLoaded && (
|
||||
<div className="flex items-center justify-center py-12">
|
||||
<div className="text-center">
|
||||
<div className="inline-block h-8 w-8 animate-spin rounded-full border-4 border-solid border-blue-500 border-r-transparent"></div>
|
||||
<p className="mt-4 text-sm text-slate-500 dark:text-slate-400">
|
||||
載入搜尋引擎...
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Footer */}
|
||||
<div className="border-t border-slate-200 px-6 py-3 text-xs text-slate-500 dark:border-slate-700 dark:text-slate-400">
|
||||
<div className="flex items-center justify-between">
|
||||
<span>按 ESC 關閉</span>
|
||||
<span className="text-right">支援中英文全文搜尋</span>
|
||||
</div>
|
||||
</div>
|
||||
<div className="flex items-center border-b border-slate-200 px-4 dark:border-slate-700">
|
||||
<FiSearch className="size-5 shrink-0 text-slate-400" />
|
||||
<Command.Input
|
||||
value={search}
|
||||
onValueChange={setSearch}
|
||||
placeholder="搜尋文章或快速導航…"
|
||||
className="flex h-14 w-full bg-transparent px-3 text-base text-slate-900 placeholder:text-slate-400 focus:outline-none dark:text-slate-100 dark:placeholder:text-slate-500"
|
||||
/>
|
||||
</div>
|
||||
</div>,
|
||||
document.body
|
||||
|
||||
<Command.List className="max-h-[min(60vh,400px)] overflow-y-auto p-2">
|
||||
{loading && (
|
||||
<Command.Loading className="flex items-center justify-center py-8 text-sm text-slate-500 dark:text-slate-400">
|
||||
搜尋中…
|
||||
</Command.Loading>
|
||||
)}
|
||||
|
||||
{!loading && !search.trim() && (
|
||||
<>
|
||||
<Command.Group heading="導航" className="[&_[cmdk-group-heading]]:px-2 [&_[cmdk-group-heading]]:py-1.5 [&_[cmdk-group-heading]]:text-xs [&_[cmdk-group-heading]]:font-medium [&_[cmdk-group-heading]]:text-slate-500 [&_[cmdk-group-heading]]:dark:text-slate-400">
|
||||
{navActions.map((action) => (
|
||||
<Command.Item
|
||||
key={action.id}
|
||||
value={`${action.title} ${action.url}`}
|
||||
onSelect={() => handleSelect(action.url)}
|
||||
className={cn(
|
||||
'flex cursor-pointer items-center gap-3 rounded-lg px-3 py-2.5 text-sm text-slate-700 outline-none transition-colors',
|
||||
'data-[selected=true]:bg-slate-100 data-[selected=true]:text-slate-900',
|
||||
'dark:text-slate-300 dark:data-[selected=true]:bg-slate-800 dark:data-[selected=true]:text-slate-100 dark:hover:text-accent'
|
||||
)}
|
||||
>
|
||||
<span className="flex size-8 shrink-0 items-center justify-center rounded-md bg-slate-100 text-slate-600 dark:bg-slate-800 dark:text-slate-400">
|
||||
{action.icon}
|
||||
</span>
|
||||
<span className="truncate">{action.title}</span>
|
||||
</Command.Item>
|
||||
))}
|
||||
</Command.Group>
|
||||
{recentPostActions.length > 0 && (
|
||||
<Command.Group heading="最近文章" className="[&_[cmdk-group-heading]]:px-2 [&_[cmdk-group-heading]]:py-1.5 [&_[cmdk-group-heading]]:text-xs [&_[cmdk-group-heading]]:font-medium [&_[cmdk-group-heading]]:text-slate-500 [&_[cmdk-group-heading]]:dark:text-slate-400">
|
||||
{recentPostActions.map((action) => (
|
||||
<Command.Item
|
||||
key={action.id}
|
||||
value={`${action.title} ${action.url}`}
|
||||
onSelect={() => handleSelect(action.url)}
|
||||
className={cn(
|
||||
'flex cursor-pointer items-center gap-3 rounded-lg px-3 py-2.5 text-sm text-slate-700 outline-none transition-colors',
|
||||
'data-[selected=true]:bg-slate-100 data-[selected=true]:text-slate-900',
|
||||
'dark:text-slate-300 dark:data-[selected=true]:bg-slate-800 dark:data-[selected=true]:text-slate-100 dark:hover:text-accent'
|
||||
)}
|
||||
>
|
||||
<span className="flex size-8 shrink-0 items-center justify-center rounded-md bg-slate-100 text-slate-600 dark:bg-slate-800 dark:text-slate-400">
|
||||
{action.icon}
|
||||
</span>
|
||||
<span className="truncate">{action.title}</span>
|
||||
</Command.Item>
|
||||
))}
|
||||
</Command.Group>
|
||||
)}
|
||||
</>
|
||||
)}
|
||||
|
||||
{!loading && search.trim() && results.length > 0 && (
|
||||
<Command.Group heading="搜尋結果" className="[&_[cmdk-group-heading]]:px-2 [&_[cmdk-group-heading]]:py-1.5 [&_[cmdk-group-heading]]:text-xs [&_[cmdk-group-heading]]:font-medium [&_[cmdk-group-heading]]:text-slate-500 [&_[cmdk-group-heading]]:dark:text-slate-400">
|
||||
{results.map((result, i) => (
|
||||
<Command.Item
|
||||
key={`${result.url}-${i}`}
|
||||
value={`${result.meta?.title ?? ''} ${result.url}`}
|
||||
onSelect={() => handleSelect(result.url)}
|
||||
className={cn(
|
||||
'flex cursor-pointer flex-col gap-0.5 rounded-lg px-3 py-2.5 outline-none transition-colors',
|
||||
'data-[selected=true]:bg-slate-100 dark:data-[selected=true]:bg-slate-800 dark:hover:text-accent'
|
||||
)}
|
||||
>
|
||||
<span className="truncate text-sm font-medium text-slate-900 dark:text-slate-100">
|
||||
{result.meta?.title ?? result.url}
|
||||
</span>
|
||||
{result.excerpt && (
|
||||
<span
|
||||
className="line-clamp-2 text-xs text-slate-500 dark:text-slate-400 [&_mark]:bg-yellow-200 [&_mark]:font-semibold [&_mark]:text-slate-900 dark:[&_mark]:bg-yellow-600 dark:[&_mark]:text-slate-100"
|
||||
dangerouslySetInnerHTML={{ __html: result.excerpt }}
|
||||
/>
|
||||
)}
|
||||
</Command.Item>
|
||||
))}
|
||||
</Command.Group>
|
||||
)}
|
||||
|
||||
<Command.Empty className="py-8 text-center text-sm text-slate-500 dark:text-slate-400">
|
||||
找不到結果
|
||||
</Command.Empty>
|
||||
</Command.List>
|
||||
|
||||
<div className="border-t border-slate-200 px-4 py-2 text-xs text-slate-500 dark:border-slate-700 dark:text-slate-400">
|
||||
<span>ESC 關閉</span>
|
||||
<span className="ml-4">⌘K 開啟</span>
|
||||
</div>
|
||||
</Command.Dialog>
|
||||
);
|
||||
}
|
||||
|
||||
@@ -195,11 +263,11 @@ export function SearchButton({ onClick }: { onClick: () => void }) {
|
||||
return (
|
||||
<button
|
||||
onClick={onClick}
|
||||
className="motion-link inline-flex h-9 items-center gap-2 rounded-full bg-slate-100 px-3 py-1.5 text-sm text-slate-600 transition hover:bg-slate-200 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/40 dark:bg-slate-800 dark:text-slate-300 dark:hover:bg-slate-700"
|
||||
className="motion-link inline-flex h-9 shrink-0 items-center gap-2 rounded-full bg-slate-100 px-3 py-1.5 text-sm text-slate-600 transition-all duration-260 ease-snappy hover:-translate-y-0.5 hover:bg-slate-200 hover:text-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/40 dark:bg-slate-800 dark:text-slate-300 dark:hover:bg-slate-700 dark:hover:text-accent"
|
||||
aria-label="搜尋 (Cmd+K)"
|
||||
>
|
||||
<FiSearch className="h-3.5 w-3.5" />
|
||||
<span className="hidden sm:inline">搜尋</span>
|
||||
<FiSearch className="h-3.5 w-3.5 shrink-0" />
|
||||
<span className="hidden shrink-0 whitespace-nowrap sm:inline">搜尋</span>
|
||||
<kbd className="hidden rounded bg-white px-1.5 py-0.5 text-xs font-semibold text-slate-500 shadow-sm dark:bg-slate-900 dark:text-slate-400 sm:inline-block">
|
||||
⌘K
|
||||
</kbd>
|
||||
|
||||
@@ -1,10 +1,100 @@
|
||||
import { RightSidebar } from './right-sidebar';
|
||||
'use client';
|
||||
|
||||
import { useState, useEffect } from 'react';
|
||||
import { createPortal } from 'react-dom';
|
||||
import dynamic from 'next/dynamic';
|
||||
import { FiLayout, FiX } from 'react-icons/fi';
|
||||
import { clsx } from 'clsx';
|
||||
|
||||
// Lazy load RightSidebar since it's only visible on lg+ screens
|
||||
const RightSidebar = dynamic(() => import('./right-sidebar').then(mod => ({ default: mod.RightSidebar })), {
|
||||
ssr: false,
|
||||
});
|
||||
|
||||
const RightSidebarContent = dynamic(() => import('./right-sidebar').then(mod => ({ default: mod.RightSidebarContent })), {
|
||||
ssr: false,
|
||||
});
|
||||
|
||||
export function SidebarLayout({ children }: { children: React.ReactNode }) {
|
||||
return (
|
||||
<div className="grid gap-8 lg:grid-cols-[minmax(0,3fr)_minmax(0,1.4fr)]">
|
||||
<div>{children}</div>
|
||||
<RightSidebar />
|
||||
const [mobileSidebarOpen, setMobileSidebarOpen] = useState(false);
|
||||
const [mounted, setMounted] = useState(false);
|
||||
|
||||
useEffect(() => setMounted(true), []);
|
||||
|
||||
useEffect(() => {
|
||||
if (mobileSidebarOpen) {
|
||||
document.body.style.overflow = 'hidden';
|
||||
} else {
|
||||
document.body.style.overflow = '';
|
||||
}
|
||||
return () => { document.body.style.overflow = ''; };
|
||||
}, [mobileSidebarOpen]);
|
||||
|
||||
const mobileDrawer = mounted && createPortal(
|
||||
<>
|
||||
{/* Backdrop */}
|
||||
<div
|
||||
className={clsx(
|
||||
'fixed inset-0 z-[1100] bg-black/40 backdrop-blur-sm transition-opacity duration-300 lg:hidden',
|
||||
mobileSidebarOpen ? 'opacity-100' : 'opacity-0 pointer-events-none'
|
||||
)}
|
||||
onClick={() => setMobileSidebarOpen(false)}
|
||||
aria-hidden="true"
|
||||
/>
|
||||
|
||||
{/* Slide-over panel from right */}
|
||||
<div
|
||||
className={clsx(
|
||||
'fixed top-0 right-0 bottom-0 z-[1110] w-full max-w-sm flex flex-col rounded-l-2xl border-l border-white/20 bg-white/95 shadow-2xl backdrop-blur-xl transition-transform duration-300 ease-snappy dark:border-white/10 dark:bg-slate-900/95 lg:hidden',
|
||||
mobileSidebarOpen ? 'translate-x-0' : 'translate-x-full'
|
||||
)}
|
||||
>
|
||||
<div className="flex items-center justify-between border-b border-slate-200/50 px-6 py-4 dark:border-slate-700/50">
|
||||
<div className="flex items-center gap-2 font-semibold text-slate-900 dark:text-slate-100">
|
||||
<FiLayout className="h-5 w-5 text-slate-500" />
|
||||
<span>側邊欄</span>
|
||||
</div>
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => setMobileSidebarOpen(false)}
|
||||
className="rounded-full p-1 text-slate-500 hover:bg-slate-100 hover:text-accent dark:hover:bg-slate-800 dark:hover:text-accent"
|
||||
aria-label="關閉側邊欄"
|
||||
>
|
||||
<FiX className="h-5 w-5" />
|
||||
</button>
|
||||
</div>
|
||||
);
|
||||
|
||||
<div className="flex-1 overflow-y-auto px-6 py-6">
|
||||
<RightSidebarContent forceLoadFeed={mobileSidebarOpen} />
|
||||
</div>
|
||||
</div>
|
||||
</>,
|
||||
document.body
|
||||
);
|
||||
|
||||
const mobileFab = mounted && (
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => setMobileSidebarOpen(true)}
|
||||
className={clsx(
|
||||
'fixed bottom-6 left-6 z-40 flex h-11 w-11 items-center justify-center rounded-full border border-slate-200 bg-white/90 text-slate-600 shadow-md backdrop-blur-sm transition hover:bg-slate-50 hover:text-accent dark:border-slate-700 dark:bg-slate-900/90 dark:text-slate-300 dark:hover:bg-slate-800 dark:hover:text-accent lg:hidden',
|
||||
mobileSidebarOpen ? 'opacity-0 pointer-events-none' : 'opacity-100'
|
||||
)}
|
||||
aria-label="開啟側邊欄"
|
||||
>
|
||||
<FiLayout className="h-5 w-5" />
|
||||
</button>
|
||||
);
|
||||
|
||||
return (
|
||||
<>
|
||||
<div className="grid gap-8 lg:grid-cols-[minmax(0,3fr)_minmax(0,1.4fr)]">
|
||||
<div>{children}</div>
|
||||
<RightSidebar />
|
||||
</div>
|
||||
|
||||
{mobileDrawer}
|
||||
{mobileFab}
|
||||
</>
|
||||
);
|
||||
}
|
||||
|
||||
@@ -66,7 +66,7 @@ export function SiteFooter() {
|
||||
target="_blank"
|
||||
rel="noopener noreferrer"
|
||||
aria-label={item.label}
|
||||
className="text-slate-500 hover:text-slate-800 dark:text-slate-400 dark:hover:text-slate-100"
|
||||
className="text-slate-500 hover:text-accent dark:text-slate-400 dark:hover:text-accent"
|
||||
>
|
||||
<item.icon className="h-4 w-4" />
|
||||
</a>
|
||||
|
||||
@@ -1,6 +1,5 @@
|
||||
'use client';
|
||||
|
||||
import Link from 'next/link';
|
||||
import { useState } from 'react';
|
||||
import dynamic from 'next/dynamic';
|
||||
import { ThemeToggle } from './theme-toggle';
|
||||
@@ -8,6 +7,7 @@ import { NavMenu, NavLinkItem, IconKey } from './nav-menu';
|
||||
import { SearchButton } from './search-modal';
|
||||
import { siteConfig } from '@/lib/config';
|
||||
import { allPages } from 'contentlayer2/generated';
|
||||
import Link from 'next/link';
|
||||
|
||||
// Dynamically import SearchModal to reduce initial bundle size
|
||||
const SearchModal = dynamic(
|
||||
@@ -15,7 +15,11 @@ const SearchModal = dynamic(
|
||||
{ ssr: false }
|
||||
);
|
||||
|
||||
export function SiteHeader() {
|
||||
interface SiteHeaderProps {
|
||||
recentPosts?: { title: string; url: string }[];
|
||||
}
|
||||
|
||||
export function SiteHeader({ recentPosts = [] }: SiteHeaderProps) {
|
||||
const [isSearchOpen, setIsSearchOpen] = useState(false);
|
||||
const pages = allPages
|
||||
.slice()
|
||||
@@ -23,21 +27,26 @@ export function SiteHeader() {
|
||||
|
||||
const findPage = (title: string) => pages.find((page) => page.title === title);
|
||||
|
||||
const aboutChildren = [
|
||||
{ title: '關於作者', label: '作者' },
|
||||
{ title: '關於本站', label: '本站' }
|
||||
]
|
||||
.map(({ title, label }) => {
|
||||
const page = findPage(title);
|
||||
if (!page) return null;
|
||||
return {
|
||||
key: page._id,
|
||||
href: page.url,
|
||||
label,
|
||||
iconKey: getIconForPage(page.title, page.slug)
|
||||
} satisfies NavLinkItem;
|
||||
})
|
||||
.filter(Boolean) as NavLinkItem[];
|
||||
const aboutChildren: NavLinkItem[] = [
|
||||
...(
|
||||
[
|
||||
{ title: '關於作者', label: '作者' },
|
||||
{ title: '關於本站', label: '本站' }
|
||||
]
|
||||
.map(({ title, label }) => {
|
||||
const page = findPage(title);
|
||||
if (!page) return null;
|
||||
return {
|
||||
key: page._id,
|
||||
href: page.url,
|
||||
label,
|
||||
iconKey: getIconForPage(page.title, page.slug)
|
||||
} satisfies NavLinkItem;
|
||||
})
|
||||
.filter(Boolean) as NavLinkItem[]
|
||||
),
|
||||
{ key: 'projects', href: '/projects', label: '作品', iconKey: 'pen' }
|
||||
];
|
||||
|
||||
const deviceChildren = [
|
||||
{ title: '開發工作環境', label: '開發環境' },
|
||||
@@ -78,7 +87,8 @@ export function SiteHeader() {
|
||||
<div className="container mx-auto flex items-center justify-between px-4 py-3 text-slate-900 dark:text-slate-100">
|
||||
<Link
|
||||
href="/"
|
||||
className="motion-link group relative type-title text-slate-900 hover:text-accent focus-visible:outline-none focus-visible:text-accent dark:text-slate-100"
|
||||
prefetch={true}
|
||||
className="motion-link group relative type-title whitespace-nowrap text-slate-900 hover:text-accent focus-visible:outline-none focus-visible:text-accent dark:text-slate-100 dark:hover:text-accent"
|
||||
>
|
||||
<span className="absolute -bottom-0.5 left-0 h-[2px] w-0 bg-accent transition-all duration-180 ease-snappy group-hover:w-full" aria-hidden="true" />
|
||||
{siteConfig.title}
|
||||
@@ -91,6 +101,7 @@ export function SiteHeader() {
|
||||
<SearchModal
|
||||
isOpen={isSearchOpen}
|
||||
onClose={() => setIsSearchOpen(false)}
|
||||
recentPosts={recentPosts}
|
||||
/>
|
||||
</div>
|
||||
</header>
|
||||
|
||||
225
components/terminal-window.tsx
Normal file
225
components/terminal-window.tsx
Normal file
@@ -0,0 +1,225 @@
|
||||
'use client';
|
||||
|
||||
import { useState, useEffect, useCallback } from 'react';
|
||||
|
||||
// 眼睛 (霍德爾之目) - 雙鷹勾眼
|
||||
const ASCII_ART = [
|
||||
' /\\ /\\',
|
||||
' / \\ / \\',
|
||||
' | > | | > |',
|
||||
' \\ / \\ /',
|
||||
' \\/ \\/',
|
||||
];
|
||||
|
||||
interface TerminalWindowProps {
|
||||
title: string;
|
||||
tagline: string;
|
||||
/** Skip typing animation, show all at once */
|
||||
reducedMotion?: boolean;
|
||||
className?: string;
|
||||
}
|
||||
|
||||
type Phase =
|
||||
| 'prompt'
|
||||
| 'typing-line1'
|
||||
| 'typing-line2'
|
||||
| 'prompt2'
|
||||
| 'typing-ascii'
|
||||
| 'done';
|
||||
|
||||
export function TerminalWindow({
|
||||
title,
|
||||
tagline,
|
||||
reducedMotion = false,
|
||||
className = '',
|
||||
}: TerminalWindowProps) {
|
||||
const [phase, setPhase] = useState<Phase>('prompt');
|
||||
const [displayedPrompt, setDisplayedPrompt] = useState('');
|
||||
const [displayedLine1, setDisplayedLine1] = useState('');
|
||||
const [displayedLine2, setDisplayedLine2] = useState('');
|
||||
const [displayedPrompt2, setDisplayedPrompt2] = useState('');
|
||||
const [displayedAscii, setDisplayedAscii] = useState<string[]>([]);
|
||||
const [showCursor, setShowCursor] = useState(true);
|
||||
|
||||
const prompt = 'cat ~/welcome.txt';
|
||||
const prompt2 = 'fastfetch';
|
||||
const line1 = `${title}`;
|
||||
const line2 = tagline;
|
||||
|
||||
const charDelay = reducedMotion ? 0 : 50;
|
||||
const lineDelay = reducedMotion ? 0 : 400;
|
||||
const asciiLineDelay = reducedMotion ? 0 : 80;
|
||||
|
||||
const typeString = useCallback(
|
||||
(
|
||||
str: string,
|
||||
setter: (s: string) => void,
|
||||
onComplete?: () => void
|
||||
) => {
|
||||
if (reducedMotion) {
|
||||
setter(str);
|
||||
onComplete?.();
|
||||
return;
|
||||
}
|
||||
let i = 0;
|
||||
const id = setInterval(() => {
|
||||
if (i <= str.length) {
|
||||
setter(str.slice(0, i));
|
||||
i++;
|
||||
} else {
|
||||
clearInterval(id);
|
||||
onComplete?.();
|
||||
}
|
||||
}, charDelay);
|
||||
return () => clearInterval(id);
|
||||
},
|
||||
[charDelay, reducedMotion]
|
||||
);
|
||||
|
||||
useEffect(() => {
|
||||
if (phase === 'prompt') {
|
||||
const cleanup = typeString(prompt, setDisplayedPrompt, () => {
|
||||
setTimeout(() => setPhase('typing-line1'), lineDelay);
|
||||
});
|
||||
return cleanup;
|
||||
}
|
||||
}, [phase, prompt, typeString, lineDelay]);
|
||||
|
||||
useEffect(() => {
|
||||
if (phase === 'typing-line1') {
|
||||
const cleanup = typeString(line1, setDisplayedLine1, () => {
|
||||
setTimeout(() => setPhase('typing-line2'), lineDelay);
|
||||
});
|
||||
return cleanup;
|
||||
}
|
||||
}, [phase, line1, typeString, lineDelay]);
|
||||
|
||||
useEffect(() => {
|
||||
if (phase === 'typing-line2') {
|
||||
const cleanup = typeString(line2, setDisplayedLine2, () => {
|
||||
setTimeout(() => setPhase('prompt2'), lineDelay);
|
||||
});
|
||||
return cleanup;
|
||||
}
|
||||
}, [phase, line2, typeString, lineDelay]);
|
||||
|
||||
useEffect(() => {
|
||||
if (phase === 'prompt2') {
|
||||
setDisplayedPrompt2('');
|
||||
const cleanup = typeString(prompt2, setDisplayedPrompt2, () => {
|
||||
setTimeout(() => setPhase('typing-ascii'), lineDelay);
|
||||
});
|
||||
return cleanup;
|
||||
}
|
||||
}, [phase, prompt2, typeString, lineDelay]);
|
||||
|
||||
useEffect(() => {
|
||||
if (phase === 'typing-ascii') {
|
||||
if (reducedMotion) {
|
||||
setDisplayedAscii(ASCII_ART);
|
||||
setTimeout(() => setPhase('done'), lineDelay);
|
||||
return;
|
||||
}
|
||||
let lineIndex = 0;
|
||||
const id = setInterval(() => {
|
||||
if (lineIndex < ASCII_ART.length) {
|
||||
setDisplayedAscii((prev) => [...prev, ASCII_ART[lineIndex]]);
|
||||
lineIndex++;
|
||||
} else {
|
||||
clearInterval(id);
|
||||
setTimeout(() => setPhase('done'), lineDelay);
|
||||
}
|
||||
}, asciiLineDelay);
|
||||
return () => clearInterval(id);
|
||||
}
|
||||
}, [phase, asciiLineDelay, lineDelay, reducedMotion]);
|
||||
|
||||
// Blinking cursor
|
||||
useEffect(() => {
|
||||
if (!reducedMotion && phase !== 'done') {
|
||||
const id = setInterval(() => setShowCursor((c) => !c), 530);
|
||||
return () => clearInterval(id);
|
||||
}
|
||||
setShowCursor(true);
|
||||
}, [phase, reducedMotion]);
|
||||
|
||||
return (
|
||||
<div
|
||||
className={`overflow-hidden rounded-xl border border-slate-300 bg-slate-100 shadow-xl dark:border-slate-700/50 dark:bg-slate-900 ${className}`}
|
||||
role="img"
|
||||
aria-label={`終端機:${title} - ${tagline}`}
|
||||
>
|
||||
{/* macOS-style title bar */}
|
||||
<div className="flex items-center gap-2 border-b border-slate-200 px-4 py-2.5 dark:border-slate-700/50 sm:px-5 sm:py-3 lg:px-6 lg:py-3.5">
|
||||
<div className="flex gap-1.5 sm:gap-2">
|
||||
<span className="h-3 w-3 rounded-full bg-red-500/90 sm:h-3.5 sm:w-3.5 lg:h-4 lg:w-4" />
|
||||
<span className="h-3 w-3 rounded-full bg-amber-500/90 sm:h-3.5 sm:w-3.5 lg:h-4 lg:w-4" />
|
||||
<span className="h-3 w-3 rounded-full bg-emerald-500/90 sm:h-3.5 sm:w-3.5 lg:h-4 lg:w-4" />
|
||||
</div>
|
||||
<span className="ml-4 font-mono text-xs text-slate-500 sm:text-sm dark:text-slate-400 lg:text-base">
|
||||
gbanyan@blog — zsh
|
||||
</span>
|
||||
</div>
|
||||
|
||||
{/* Terminal content */}
|
||||
<div className="px-4 py-4 font-mono text-sm sm:px-5 sm:py-5 sm:text-base lg:px-6 lg:py-6 lg:text-lg">
|
||||
<div className="text-slate-600 dark:text-slate-300">
|
||||
<span className="text-emerald-600 dark:text-emerald-400">~</span>
|
||||
<span className="text-slate-500"> $ </span>
|
||||
<span>{displayedPrompt}</span>
|
||||
{phase === 'prompt' && showCursor && (
|
||||
<span className="ml-0.5 inline-block h-4 w-0.5 animate-pulse bg-emerald-600 dark:bg-emerald-400" />
|
||||
)}
|
||||
</div>
|
||||
|
||||
{displayedLine1 && (
|
||||
<div className="mt-2 text-slate-900 dark:text-slate-100">
|
||||
{displayedLine1}
|
||||
{phase === 'typing-line1' && showCursor && (
|
||||
<span className="ml-0.5 inline-block h-4 w-0.5 animate-pulse bg-emerald-600 dark:bg-emerald-400" />
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{displayedLine2 && (
|
||||
<div className="mt-1 text-slate-600 dark:text-slate-300">
|
||||
{displayedLine2}
|
||||
{phase === 'typing-line2' && showCursor && (
|
||||
<span className="ml-0.5 inline-block h-4 w-0.5 animate-pulse bg-emerald-600 dark:bg-emerald-400" />
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{(phase === 'prompt2' || phase === 'typing-ascii' || displayedPrompt2 || displayedAscii.length > 0) && (
|
||||
<div className="mt-2 text-slate-600 dark:text-slate-300">
|
||||
<span className="text-emerald-600 dark:text-emerald-400">~</span>
|
||||
<span className="text-slate-500"> $ </span>
|
||||
<span>{displayedPrompt2}</span>
|
||||
{phase === 'prompt2' && showCursor && (
|
||||
<span className="ml-0.5 inline-block h-4 w-0.5 animate-pulse bg-emerald-600 dark:bg-emerald-400" />
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{displayedAscii.length > 0 && (
|
||||
<div className="mt-2 whitespace-pre text-emerald-600/90 dark:text-emerald-400/90">
|
||||
{displayedAscii.map((line, i) => (
|
||||
<div key={i}>{line}</div>
|
||||
))}
|
||||
{phase === 'typing-ascii' && showCursor && (
|
||||
<span className="inline-block h-4 w-0.5 animate-pulse bg-emerald-600 dark:bg-emerald-400" />
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{phase === 'done' && (
|
||||
<div className="mt-2 text-slate-600 dark:text-slate-300">
|
||||
<span className="text-emerald-600 dark:text-emerald-400">~</span>
|
||||
<span className="text-slate-500"> $ </span>
|
||||
<span className="inline-block h-4 w-4 animate-pulse border-l-2 border-emerald-600 dark:border-emerald-400" />
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@@ -22,14 +22,14 @@ export function ThemeToggle() {
|
||||
return (
|
||||
<button
|
||||
type="button"
|
||||
className="inline-flex h-9 w-9 items-center justify-center rounded-full text-accent-textLight transition duration-180 ease-snappy hover:-translate-y-0.5 hover:bg-accent-soft hover:text-accent active:scale-95 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/40 dark:text-accent-textDark dark:hover:bg-slate-800 dark:hover:text-accent"
|
||||
className="inline-flex h-9 w-9 items-center justify-center rounded-full text-accent-textLight transition-all duration-300 ease-out-quarter hover:-translate-y-1 hover:scale-110 hover:bg-accent-soft hover:text-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/40 dark:text-accent-textDark dark:hover:bg-accent-soft dark:hover:text-accent"
|
||||
onClick={() => setTheme(next)}
|
||||
aria-label={theme === 'dark' ? '切換為淺色主題' : '切換為深色主題'}
|
||||
>
|
||||
{isDark ? (
|
||||
<FiSun className="h-4 w-4 rotate-0 text-amber-400 transition-transform duration-260 ease-snappy" />
|
||||
<FiSun className="h-4 w-4 rotate-0 text-amber-400 transition-all duration-500 ease-out-expo" />
|
||||
) : (
|
||||
<FiMoon className="h-4 w-4 rotate-180 text-blue-500 transition-transform duration-260 ease-snappy" />
|
||||
<FiMoon className="h-4 w-4 rotate-180 text-blue-500 transition-all duration-500 ease-out-expo" />
|
||||
)}
|
||||
</button>
|
||||
);
|
||||
|
||||
@@ -1,4 +1,6 @@
|
||||
import { Children, ReactNode } from 'react';
|
||||
'use client';
|
||||
|
||||
import {Children, ReactNode, useEffect, useState} from 'react';
|
||||
import clsx from 'clsx';
|
||||
|
||||
interface TimelineWrapperProps {
|
||||
@@ -7,22 +9,34 @@ interface TimelineWrapperProps {
|
||||
}
|
||||
|
||||
export function TimelineWrapper({ children, className }: TimelineWrapperProps) {
|
||||
const [mounted, setMounted] = useState(false);
|
||||
|
||||
useEffect(() => {
|
||||
setMounted(true);
|
||||
}, []);
|
||||
|
||||
const items = Children.toArray(children);
|
||||
|
||||
// Only render decorative elements after mount to prevent layout shift
|
||||
if (!mounted) {
|
||||
return (
|
||||
<div className={clsx('relative pl-6 md:pl-8', className)}>
|
||||
<div className="space-y-4">{items.map((child, index) => <div key={index} className="relative pl-5 sm:pl-8">{child}</div>)}</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<div className={clsx('relative pl-6 md:pl-8', className)}>
|
||||
<span
|
||||
className="pointer-events-none absolute left-2 top-0 h-full w-[2px] rounded-full bg-blue-400 shadow-[0_0_10px_rgba(59,130,246,0.35)] dark:bg-cyan-300 md:left-3"
|
||||
aria-hidden="true"
|
||||
/>
|
||||
<span
|
||||
className="pointer-events-none absolute left-2 top-0 h-full w-[8px] rounded-full bg-blue-500/15 blur-[14px] md:left-3"
|
||||
className="pointer-events-none absolute left-2 top-0 h-full w-[2px] rounded-full bg-accent/40 md:left-3"
|
||||
aria-hidden="true"
|
||||
/>
|
||||
|
||||
<div className="space-y-4">
|
||||
{items.map((child, index) => (
|
||||
<div key={index} className="relative pl-5 sm:pl-8">
|
||||
<span className="pointer-events-none absolute left-0 top-1/2 h-px w-5 -translate-x-full -translate-y-1/2 bg-gradient-to-r from-transparent via-blue-300/80 to-transparent dark:via-cyan-200/80 sm:w-8" aria-hidden="true" />
|
||||
<span className="pointer-events-none absolute left-0 top-1/2 h-px w-5 -translate-x-full -translate-y-1/2 bg-gradient-to-r from-transparent via-accent/30 to-transparent sm:w-8" aria-hidden="true" />
|
||||
{child}
|
||||
</div>
|
||||
))}
|
||||
|
||||
19
components/view-transition-provider.tsx
Normal file
19
components/view-transition-provider.tsx
Normal file
@@ -0,0 +1,19 @@
|
||||
'use client';
|
||||
|
||||
import { useState, useEffect, ReactNode } from 'react';
|
||||
|
||||
export function ViewTransitionProvider({ children }: { children: ReactNode }) {
|
||||
const [isSafari18, setIsSafari18] = useState(false);
|
||||
|
||||
useEffect(() => {
|
||||
const isSafari = typeof navigator !== 'undefined' && navigator.userAgent.includes('safari') && !navigator.userAgent.includes('chrome') && !navigator.userAgent.includes('firefox');
|
||||
const hasNativeTransitions = typeof document !== 'undefined' && typeof document.startViewTransition === 'function';
|
||||
setIsSafari18(isSafari && hasNativeTransitions);
|
||||
}, []);
|
||||
|
||||
if (isSafari18) {
|
||||
return <>{children}</>;
|
||||
}
|
||||
|
||||
return <>{children}</>;
|
||||
}
|
||||
49
components/web-vitals.tsx
Normal file
49
components/web-vitals.tsx
Normal file
@@ -0,0 +1,49 @@
|
||||
'use client';
|
||||
|
||||
import { useEffect } from 'react';
|
||||
|
||||
/**
|
||||
* Web Vitals monitoring component (optional)
|
||||
*
|
||||
* To enable full Web Vitals tracking, install web-vitals package:
|
||||
* npm install web-vitals
|
||||
*
|
||||
* Then uncomment the code below and import from 'web-vitals'
|
||||
*/
|
||||
export function WebVitals() {
|
||||
useEffect(() => {
|
||||
// Only track in production
|
||||
if (process.env.NODE_ENV !== 'production') return;
|
||||
|
||||
// Basic performance monitoring using Performance API
|
||||
if (typeof window !== 'undefined' && 'performance' in window) {
|
||||
// Track page load time
|
||||
window.addEventListener('load', () => {
|
||||
const perfData = performance.getEntriesByType('navigation')[0] as PerformanceNavigationTiming;
|
||||
if (perfData) {
|
||||
const loadTime = perfData.loadEventEnd - perfData.fetchStart;
|
||||
const domContentLoaded = perfData.domContentLoadedEventEnd - perfData.fetchStart;
|
||||
|
||||
// Log metrics (can be sent to analytics service)
|
||||
if (process.env.NODE_ENV === 'development') {
|
||||
console.log('Performance Metrics:', {
|
||||
loadTime: Math.round(loadTime),
|
||||
domContentLoaded: Math.round(domContentLoaded),
|
||||
firstByte: Math.round(perfData.responseStart - perfData.fetchStart),
|
||||
});
|
||||
}
|
||||
|
||||
// Example: Send to analytics service
|
||||
// if (typeof window !== 'undefined' && window.gtag) {
|
||||
// window.gtag('event', 'page_load_time', {
|
||||
// value: Math.round(loadTime),
|
||||
// non_interaction: true,
|
||||
// });
|
||||
// }
|
||||
}
|
||||
});
|
||||
}
|
||||
}, []);
|
||||
|
||||
return null;
|
||||
}
|
||||
2
content
2
content
Submodule content updated: cbde394ac2...7b52c564dc
@@ -103,7 +103,7 @@ export default makeSource({
|
||||
[rehypeAutolinkHeadings, { behavior: 'wrap' }],
|
||||
/**
|
||||
* Rewrite markdown image src from relative "../assets/..." to
|
||||
* absolute "/assets/..." so they are served from Next.js public/.
|
||||
* absolute "/assets/..." and add lazy loading for cross-browser performance.
|
||||
*/
|
||||
() => (tree: any) => {
|
||||
visit(tree, 'element', (node: any) => {
|
||||
@@ -118,6 +118,9 @@ export default makeSource({
|
||||
} else if (src.startsWith('assets/')) {
|
||||
node.properties.src = '/' + src.replace(/^\/?/, '');
|
||||
}
|
||||
// Lazy load images for better LCP and bandwidth (Chrome, Firefox, Safari, Edge)
|
||||
node.properties.loading = 'lazy';
|
||||
node.properties.decoding = 'async';
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
2
env
2
env
@@ -5,7 +5,7 @@ NEXT_PUBLIC_SITE_DESCRIPTION="醫學、科技與生活隨筆。"
|
||||
NEXT_PUBLIC_SITE_URL="http://localhost:3000"
|
||||
NEXT_PUBLIC_SITE_AUTHOR="Gbanyan"
|
||||
NEXT_PUBLIC_SITE_TAGLINE="醫學、科技與生活的隨筆記錄。"
|
||||
NEXT_PUBLIC_POSTS_PER_PAGE="5"
|
||||
NEXT_PUBLIC_POSTS_PER_PAGE="7"
|
||||
NEXT_PUBLIC_DEFAULT_LOCALE="zh-TW"
|
||||
NEXT_PUBLIC_SITE_AVATAR_URL="https://www.gravatar.com/avatar/53f2a6e011d5ececcd0d6e33e8e7329f60efcd23a5b77ba382273d5060a2cffe?s=160&d=identicon"
|
||||
NEXT_PUBLIC_SITE_ABOUT_SHORT="掙扎混亂過日子 \n 對平淡美好日常的期待即是救贖"
|
||||
|
||||
@@ -27,12 +27,12 @@ export const siteConfig = {
|
||||
gitea: process.env.NEXT_PUBLIC_GITEA_URL || ''
|
||||
},
|
||||
theme: {
|
||||
accent: process.env.NEXT_PUBLIC_COLOR_ACCENT || '#2563eb',
|
||||
accentSoft: process.env.NEXT_PUBLIC_COLOR_ACCENT_SOFT || '#dbeafe',
|
||||
accent: process.env.NEXT_PUBLIC_COLOR_ACCENT || '#7c3aed',
|
||||
accentSoft: process.env.NEXT_PUBLIC_COLOR_ACCENT_SOFT || '#f3e8ff',
|
||||
accentTextLight:
|
||||
process.env.NEXT_PUBLIC_COLOR_ACCENT_TEXT_LIGHT || '#1d4ed8',
|
||||
process.env.NEXT_PUBLIC_COLOR_ACCENT_TEXT_LIGHT || '#6d28d9',
|
||||
accentTextDark:
|
||||
process.env.NEXT_PUBLIC_COLOR_ACCENT_TEXT_DARK || '#93c5fd'
|
||||
process.env.NEXT_PUBLIC_COLOR_ACCENT_TEXT_DARK || '#c4b5fd'
|
||||
},
|
||||
navIconOverrides: {
|
||||
titles: {
|
||||
|
||||
43
lib/github-lang-colors.ts
Normal file
43
lib/github-lang-colors.ts
Normal file
@@ -0,0 +1,43 @@
|
||||
/**
|
||||
* GitHub-style language colors for repo cards.
|
||||
* Fallback: #94a3b8 (slate-400) for unknown languages.
|
||||
*/
|
||||
const LANG_COLORS: Record<string, string> = {
|
||||
TypeScript: '#3178c6',
|
||||
JavaScript: '#f1e05a',
|
||||
Python: '#3572A5',
|
||||
Rust: '#dea584',
|
||||
Go: '#00ADD8',
|
||||
Ruby: '#701516',
|
||||
PHP: '#4F5D95',
|
||||
Java: '#b07219',
|
||||
Kotlin: '#A97BFF',
|
||||
Swift: '#F05138',
|
||||
C: '#555555',
|
||||
'C++': '#f34b7d',
|
||||
'C#': '#239120',
|
||||
Shell: '#89e051',
|
||||
HTML: '#e34c26',
|
||||
CSS: '#563d7c',
|
||||
Vue: '#41b883',
|
||||
Svelte: '#ff3e00',
|
||||
Dart: '#00B4AB',
|
||||
Scala: '#c22d40',
|
||||
Elixir: '#6e4a7e',
|
||||
Lua: '#000080',
|
||||
R: '#198CE7',
|
||||
Markdown: '#083fa1',
|
||||
YAML: '#cb171e',
|
||||
JSON: '#292929',
|
||||
};
|
||||
|
||||
const FALLBACK_COLOR = '#94a3b8';
|
||||
|
||||
/**
|
||||
* Returns the GitHub-style hex color for a programming language.
|
||||
* Unknown languages use a neutral slate fallback.
|
||||
*/
|
||||
export function getLanguageColor(lang: string | null): string {
|
||||
if (!lang || !lang.trim()) return FALLBACK_COLOR;
|
||||
return LANG_COLORS[lang] ?? FALLBACK_COLOR;
|
||||
}
|
||||
76
lib/github.ts
Normal file
76
lib/github.ts
Normal file
@@ -0,0 +1,76 @@
|
||||
export type RepoSummary = {
|
||||
id: number;
|
||||
name: string;
|
||||
fullName: string;
|
||||
htmlUrl: string;
|
||||
description: string | null;
|
||||
language: string | null;
|
||||
stargazersCount: number;
|
||||
updatedAt: string;
|
||||
};
|
||||
|
||||
const GITHUB_API_BASE = 'https://api.github.com';
|
||||
|
||||
function getGithubHeaders() {
|
||||
const headers: Record<string, string> = {
|
||||
Accept: 'application/vnd.github+json',
|
||||
'User-Agent': 'blog-nextjs-app',
|
||||
};
|
||||
|
||||
const token = process.env.GITHUB_TOKEN;
|
||||
if (token && token.trim() !== '') {
|
||||
headers.Authorization = `Bearer ${token}`;
|
||||
}
|
||||
|
||||
return headers;
|
||||
}
|
||||
|
||||
/**
|
||||
* Fetch all public repositories for the configured GitHub user.
|
||||
* Excludes forked repositories. Returns an empty array on error instead of
|
||||
* throwing, so the UI can render a graceful fallback.
|
||||
*/
|
||||
export async function fetchPublicRepos(usernameOverride?: string): Promise<RepoSummary[]> {
|
||||
const username = usernameOverride || process.env.GITHUB_USERNAME;
|
||||
|
||||
if (!username) {
|
||||
console.error('GITHUB_USERNAME is not set; cannot fetch GitHub repositories.');
|
||||
return [];
|
||||
}
|
||||
|
||||
const url = `${GITHUB_API_BASE}/users/${encodeURIComponent(
|
||||
username
|
||||
)}/repos?type=public&sort=updated`;
|
||||
|
||||
try {
|
||||
const res = await fetch(url, {
|
||||
headers: getGithubHeaders(),
|
||||
// Use Next.js App Router caching / ISR
|
||||
next: { revalidate: 3600 },
|
||||
});
|
||||
|
||||
if (!res.ok) {
|
||||
console.error('Failed to fetch GitHub repositories:', res.status, res.statusText);
|
||||
return [];
|
||||
}
|
||||
|
||||
const data = (await res.json()) as any[];
|
||||
|
||||
return data
|
||||
.filter((repo) => !repo.fork)
|
||||
.map((repo) => ({
|
||||
id: repo.id,
|
||||
name: repo.name,
|
||||
fullName: repo.full_name,
|
||||
htmlUrl: repo.html_url,
|
||||
description: repo.description,
|
||||
language: repo.language,
|
||||
stargazersCount: repo.stargazers_count,
|
||||
updatedAt: repo.updated_at,
|
||||
}));
|
||||
} catch (error) {
|
||||
console.error('Error while fetching GitHub repositories:', error);
|
||||
return [];
|
||||
}
|
||||
}
|
||||
|
||||
@@ -14,9 +14,12 @@ export interface MastodonStatus {
|
||||
avatar: string;
|
||||
};
|
||||
media_attachments: Array<{
|
||||
type: string;
|
||||
url: string;
|
||||
preview_url: string;
|
||||
id: string;
|
||||
type: 'image' | 'video' | 'gifv' | 'audio' | 'unknown';
|
||||
url: string | null;
|
||||
preview_url: string | null;
|
||||
description: string | null;
|
||||
blurhash?: string | null;
|
||||
}>;
|
||||
}
|
||||
|
||||
|
||||
49
lib/posts.ts
49
lib/posts.ts
@@ -1,11 +1,18 @@
|
||||
import { allPosts, allPages, Post, Page } from 'contentlayer2/generated';
|
||||
|
||||
let _sortedCache: Post[] | null = null;
|
||||
let _relatedCache: Map<string, Post[]> = new Map();
|
||||
let _neighborsCache: Map<string, { newer?: Post; older?: Post }> = new Map();
|
||||
let _tagsCache: { tag: string; slug: string; count: number }[] | null = null;
|
||||
|
||||
export function getAllPostsSorted(): Post[] {
|
||||
return [...allPosts].sort((a, b) => {
|
||||
if (_sortedCache) return _sortedCache;
|
||||
_sortedCache = [...allPosts].sort((a, b) => {
|
||||
const aDate = a.published_at ? new Date(a.published_at).getTime() : 0;
|
||||
const bDate = b.published_at ? new Date(b.published_at).getTime() : 0;
|
||||
return bDate - aDate;
|
||||
});
|
||||
return _sortedCache;
|
||||
}
|
||||
|
||||
export function getPostBySlug(slug: string): Post | undefined {
|
||||
@@ -38,24 +45,31 @@ export function getTagSlug(tag: string): string {
|
||||
}
|
||||
|
||||
export function getAllTagsWithCount(): { tag: string; slug: string; count: number }[] {
|
||||
const map = new Map<string, number>();
|
||||
if (_tagsCache) return _tagsCache;
|
||||
|
||||
const map = new Map<string, number>();
|
||||
for (const post of allPosts) {
|
||||
if (!post.tags) continue;
|
||||
for (const tag of post.tags) {
|
||||
map.set(tag, (map.get(tag) ?? 0) + 1);
|
||||
for (const postTag of post.tags) {
|
||||
map.set(postTag, (map.get(postTag) ?? 0) + 1);
|
||||
}
|
||||
}
|
||||
|
||||
return Array.from(map.entries())
|
||||
_tagsCache = Array.from(map.entries())
|
||||
.map(([tag, count]) => ({ tag, slug: getTagSlug(tag), count }))
|
||||
.sort((a, b) => {
|
||||
if (b.count === a.count) return a.tag.localeCompare(b.tag);
|
||||
return b.count - a.count;
|
||||
});
|
||||
return _tagsCache;
|
||||
}
|
||||
|
||||
export function getRelatedPosts(target: Post, limit = 3): Post[] {
|
||||
const cacheKey = `${target._id}-${limit}`;
|
||||
if (_relatedCache.has(cacheKey)) {
|
||||
return _relatedCache.get(cacheKey)!;
|
||||
}
|
||||
|
||||
const targetTags = new Set(target.tags?.map((tag) => tag.toLowerCase()) ?? []);
|
||||
const candidates = getAllPostsSorted().filter((post) => post._id !== target._id);
|
||||
|
||||
@@ -84,28 +98,39 @@ export function getRelatedPosts(target: Post, limit = 3): Post[] {
|
||||
.slice(0, limit)
|
||||
.map((entry) => entry.post);
|
||||
|
||||
let result: Post[];
|
||||
if (scored.length >= limit) {
|
||||
return scored;
|
||||
result = scored;
|
||||
} else {
|
||||
const fallback = candidates.filter(
|
||||
(post) => !scored.some((existing) => existing._id === post._id)
|
||||
);
|
||||
result = [...scored, ...fallback.slice(0, limit - scored.length)].slice(0, limit);
|
||||
}
|
||||
|
||||
const fallback = candidates.filter(
|
||||
(post) => !scored.some((existing) => existing._id === post._id)
|
||||
);
|
||||
|
||||
return [...scored, ...fallback.slice(0, limit - scored.length)].slice(0, limit);
|
||||
_relatedCache.set(cacheKey, result);
|
||||
return result;
|
||||
}
|
||||
|
||||
export function getPostNeighbors(target: Post): {
|
||||
newer?: Post;
|
||||
older?: Post;
|
||||
} {
|
||||
const cacheKey = target._id;
|
||||
if (_neighborsCache.has(cacheKey)) {
|
||||
return _neighborsCache.get(cacheKey)!;
|
||||
}
|
||||
|
||||
const sorted = getAllPostsSorted();
|
||||
const index = sorted.findIndex((post) => post._id === target._id);
|
||||
|
||||
if (index === -1) return {};
|
||||
|
||||
return {
|
||||
const result = {
|
||||
newer: index > 0 ? sorted[index - 1] : undefined,
|
||||
older: index < sorted.length - 1 ? sorted[index + 1] : undefined
|
||||
};
|
||||
|
||||
_neighborsCache.set(cacheKey, result);
|
||||
return result;
|
||||
}
|
||||
|
||||
6
lib/utils.ts
Normal file
6
lib/utils.ts
Normal file
@@ -0,0 +1,6 @@
|
||||
import { clsx, type ClassValue } from 'clsx';
|
||||
import { twMerge } from 'tailwind-merge';
|
||||
|
||||
export function cn(...inputs: ClassValue[]) {
|
||||
return twMerge(clsx(inputs));
|
||||
}
|
||||
2
next-env.d.ts
vendored
2
next-env.d.ts
vendored
@@ -1,6 +1,6 @@
|
||||
/// <reference types="next" />
|
||||
/// <reference types="next/image-types/global" />
|
||||
import "./.next/dev/types/routes.d.ts";
|
||||
import "./.next/types/routes.d.ts";
|
||||
|
||||
// NOTE: This file should not be edited
|
||||
// see https://nextjs.org/docs/app/api-reference/config/typescript for more information.
|
||||
|
||||
@@ -4,12 +4,13 @@ const nextConfig = {
|
||||
images: {
|
||||
remotePatterns: [],
|
||||
formats: ['image/avif', 'image/webp'],
|
||||
deviceSizes: [640, 750, 828, 1080, 1200, 1920, 2048, 3840],
|
||||
imageSizes: [16, 32, 48, 64, 96, 128, 256, 384],
|
||||
// Optimized sizes for better performance
|
||||
deviceSizes: [640, 750, 828, 1080, 1200, 1920],
|
||||
imageSizes: [16, 32, 48, 64, 96, 128, 256],
|
||||
// Enable image optimization
|
||||
minimumCacheTTL: 60,
|
||||
},
|
||||
|
||||
// Enable Partial Prerendering (PPR) via cacheComponents in Next.js 16
|
||||
cacheComponents: true,
|
||||
|
||||
// Compiler optimizations
|
||||
compiler: {
|
||||
@@ -31,6 +32,33 @@ const nextConfig = {
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
source: '/:path*',
|
||||
headers: [
|
||||
{
|
||||
key: 'Cache-Control',
|
||||
value: 'public, max-age=0, must-revalidate',
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
source: '/blog/:slug*',
|
||||
headers: [
|
||||
{
|
||||
key: 'Cache-Control',
|
||||
value: 'public, max-age=3600, stale-while-revalidate=86400',
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
source: '/pages/:slug*',
|
||||
headers: [
|
||||
{
|
||||
key: 'Cache-Control',
|
||||
value: 'public, max-age=3600, stale-while-revalidate=86400',
|
||||
},
|
||||
],
|
||||
},
|
||||
];
|
||||
},
|
||||
};
|
||||
|
||||
2994
package-lock.json
generated
2994
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
10
package.json
10
package.json
@@ -15,16 +15,26 @@
|
||||
"author": "",
|
||||
"license": "ISC",
|
||||
"type": "module",
|
||||
"browserslist": [
|
||||
"chrome 111",
|
||||
"edge 111",
|
||||
"firefox 111",
|
||||
"safari 16.4"
|
||||
],
|
||||
"dependencies": {
|
||||
"@emotion/is-prop-valid": "^1.4.0",
|
||||
"@radix-ui/react-dialog": "^1.1.15",
|
||||
"@vercel/og": "^0.8.5",
|
||||
"clsx": "^2.1.1",
|
||||
"cmdk": "^1.1.1",
|
||||
"contentlayer2": "^0.5.8",
|
||||
"gray-matter": "^4.0.3",
|
||||
"markdown-wasm": "^1.2.0",
|
||||
"mermaid": "^11.12.3",
|
||||
"next": "^16.0.7",
|
||||
"next-contentlayer2": "^0.5.8",
|
||||
"next-themes": "^0.4.6",
|
||||
"nextjs-toploader": "^3.9.17",
|
||||
"react": "^19.2.1",
|
||||
"react-dom": "^19.2.1",
|
||||
"react-icons": "^5.5.0",
|
||||
|
||||
BIN
prod-homepage.png
Normal file
BIN
prod-homepage.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 137 KiB |
66
scripts/subset-font.mjs
Normal file
66
scripts/subset-font.mjs
Normal file
@@ -0,0 +1,66 @@
|
||||
#!/usr/bin/env node
|
||||
/**
|
||||
* 字体子集化脚本
|
||||
* 使用 pyftsubset (需要安装 fonttools: pip install fonttools brotli)
|
||||
*
|
||||
* 用法: node scripts/subset-font.mjs
|
||||
*
|
||||
* 注意:此脚本需要先下载字体文件,或使用 Google Fonts API
|
||||
* 由于我们使用 next/font/google,Next.js 会自动优化字体加载
|
||||
* 此脚本主要用于本地字体文件的子集化
|
||||
*/
|
||||
|
||||
import { execSync } from 'child_process';
|
||||
import { existsSync, mkdirSync } from 'fs';
|
||||
import { join } from 'path';
|
||||
|
||||
// 常用繁体中文字符集(约3000-5000字)
|
||||
const commonTCChars = `
|
||||
的一是在不了有和人這中大為上個國我以要他時來用們生到作地於出就分對成會可主發年動同工也能下過子說產種面而方後多定行學法所民得經十三之進著等部度家電力裡如水化高自二理起小物實現加量都兩體制機當使點從業本去把性好應開它合還因由其些然前外天政四日那社義事平形相全表間樣與關各重新線內數正心反你明看原又麼利比或但質氣第向道命此變條只沒結解問意建月公無系軍很情者最立代想已通並提直題黨程展五果料象員革位入常文總次品式活設及管特件長求老頭基資邊流路級少圖山統接知較將組見計別她手角期根論運農指幾九區強放決西被幹做必戰先回則任取據處隊南給色光門即保治北造百規熱領七海口東導器壓志世金增爭濟階油思術極交受聯什認六共權收證改清己美再轉更單風切打白教速花帶安場身車例真務具萬每目至達走積示議聲報鬥完類離離戶科懸空需廠商校連斷深難近礦千週委素技備半辦青省列習響約支般史感勞便團往酸歷市克何除消構府稱太準精值號率族維劃選標寫存候毛親快效斯院查江型眼王按格養易置派層片始卻專狀育廠京識適屬圓包火住調滿縣局照參紅細引聽該鐵價嚴龍飛
|
||||
`.trim().replace(/\s+/g, '');
|
||||
|
||||
async function subsetFont() {
|
||||
console.log('开始字体子集化...');
|
||||
|
||||
// 注意:这个脚本需要你先下载字体文件到 fonts/ 目录
|
||||
// 或者使用 Google Fonts API 下载
|
||||
const fontsDir = join(process.cwd(), 'fonts');
|
||||
const fontPath = join(fontsDir, 'LXGWWenKaiTC-Regular.ttf');
|
||||
const outputDir = join(process.cwd(), 'public', 'fonts');
|
||||
|
||||
if (!existsSync(fontPath)) {
|
||||
console.log('⚠️ 字体文件不存在,跳过子集化');
|
||||
console.log(` 预期路径: ${fontPath}`);
|
||||
console.log(' 提示: 由于使用 next/font/google,Next.js 会自动优化字体加载');
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
// 检查 pyftsubset 是否安装
|
||||
execSync('which pyftsubset', { stdio: 'ignore' });
|
||||
|
||||
const outputPath = join(outputDir, 'LXGWWenKaiTC-Regular-subset.woff2');
|
||||
|
||||
// 创建输出目录
|
||||
if (!existsSync(outputDir)) {
|
||||
mkdirSync(outputDir, { recursive: true });
|
||||
}
|
||||
|
||||
// 生成 Unicode 范围
|
||||
const unicodes = Array.from(new Set(commonTCChars))
|
||||
.map(c => `U+${c.charCodeAt(0).toString(16).toUpperCase().padStart(4, '0')}`)
|
||||
.join(',');
|
||||
|
||||
// 执行子集化
|
||||
const command = `pyftsubset "${fontPath}" --unicodes="${unicodes}" --flavor=woff2 --output-file="${outputPath}"`;
|
||||
|
||||
execSync(command, { stdio: 'inherit' });
|
||||
|
||||
console.log(`✅ 子集化完成: ${outputPath}`);
|
||||
} catch (error) {
|
||||
console.error('❌ 子集化失败:', error.message);
|
||||
console.log('\n提示: 需要安装 fonttools: pip install fonttools brotli');
|
||||
}
|
||||
}
|
||||
|
||||
subsetFont();
|
||||
1564
styles/globals.css
1564
styles/globals.css
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user