@hutusi/amytis 1.5.6 → 1.7.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (65) hide show
  1. package/CHANGELOG.md +94 -0
  2. package/CLAUDE.md +3 -2
  3. package/GEMINI.md +13 -6
  4. package/README.md +1 -1
  5. package/TODO.md +21 -76
  6. package/bun.lock +18 -3
  7. package/content/about.mdx +1 -0
  8. package/content/about.zh.mdx +21 -0
  9. package/content/flows/2026/02/20.md +16 -0
  10. package/content/links.mdx +42 -0
  11. package/content/links.zh.mdx +41 -0
  12. package/content/posts/2026-02-20-i18n-routing-considerations.mdx +150 -0
  13. package/content/posts/multimedia-showcase/index.mdx +261 -0
  14. package/content/privacy.mdx +32 -0
  15. package/content/privacy.zh.mdx +32 -0
  16. package/docs/ARCHITECTURE.md +11 -2
  17. package/docs/CONTRIBUTING.md +4 -2
  18. package/docs/deployment.md +9 -1
  19. package/eslint.config.mjs +2 -0
  20. package/package.json +5 -4
  21. package/public/next-image-export-optimizer-hashes.json +0 -3
  22. package/scripts/copy-assets.ts +1 -1
  23. package/site.config.ts +126 -44
  24. package/src/app/[slug]/page.tsx +0 -10
  25. package/src/app/archive/page.tsx +38 -10
  26. package/src/app/books/[slug]/page.tsx +18 -0
  27. package/src/app/flows/[year]/[month]/[day]/page.tsx +21 -4
  28. package/src/app/layout.tsx +48 -21
  29. package/src/app/page.tsx +135 -72
  30. package/src/app/posts/[slug]/page.tsx +6 -12
  31. package/src/app/search.json/route.ts +4 -0
  32. package/src/app/series/[slug]/page.tsx +18 -0
  33. package/src/app/subscribe/page.tsx +17 -0
  34. package/src/app/tags/[tag]/page.tsx +9 -26
  35. package/src/app/tags/page.tsx +3 -8
  36. package/src/components/AuthorCard.tsx +43 -0
  37. package/src/components/Comments.tsx +20 -4
  38. package/src/components/ExternalLinks.tsx +6 -2
  39. package/src/components/Footer.tsx +35 -26
  40. package/src/components/LanguageProvider.tsx +0 -5
  41. package/src/components/LanguageSwitch.tsx +117 -6
  42. package/src/components/LocaleSwitch.tsx +33 -0
  43. package/src/components/Navbar.tsx +31 -8
  44. package/src/components/PostNavigation.tsx +55 -0
  45. package/src/components/PostSidebar.tsx +172 -126
  46. package/src/components/ReadingProgressBar.tsx +6 -21
  47. package/src/components/RelatedPosts.tsx +1 -1
  48. package/src/components/Search.tsx +420 -70
  49. package/src/components/SelectedBooksSection.tsx +12 -6
  50. package/src/components/ShareBar.tsx +115 -0
  51. package/src/components/SimpleLayoutHeader.tsx +5 -14
  52. package/src/components/SubscribePage.tsx +298 -0
  53. package/src/components/TagContentTabs.tsx +103 -0
  54. package/src/components/TagPageHeader.tsx +7 -13
  55. package/src/components/TagSidebar.tsx +142 -0
  56. package/src/components/TagsIndexClient.tsx +156 -0
  57. package/src/hooks/useScrollY.ts +41 -0
  58. package/src/i18n/translations.ts +110 -2
  59. package/src/layouts/PostLayout.tsx +34 -7
  60. package/src/layouts/SimpleLayout.tsx +53 -15
  61. package/src/lib/markdown.ts +71 -15
  62. package/src/lib/search-utils.test.ts +163 -0
  63. package/src/lib/search-utils.ts +39 -0
  64. package/src/types/pagefind.d.ts +42 -0
  65. package/src/components/TableOfContents.tsx +0 -158
@@ -0,0 +1,142 @@
1
+ 'use client';
2
+
3
+ import Link from 'next/link';
4
+ import { useState } from 'react';
5
+ import { t, tWith } from '@/lib/i18n';
6
+ import { LuTag, LuX, LuSearch } from 'react-icons/lu';
7
+
8
+ const INITIAL_SHOW = 12;
9
+
10
+ interface TagSidebarProps {
11
+ tags: Record<string, number>;
12
+ activeTag: string;
13
+ }
14
+
15
+ export default function TagSidebar({ tags, activeTag }: TagSidebarProps) {
16
+ const [filter, setFilter] = useState('');
17
+ const [expanded, setExpanded] = useState(false);
18
+
19
+ const totalCount = Object.keys(tags).length;
20
+
21
+ const sortedTags = Object.entries(tags)
22
+ .sort((a, b) => b[1] - a[1])
23
+ .filter(([tag]) => !filter || tag.toLowerCase().includes(filter.toLowerCase()));
24
+
25
+ const activeIndex = sortedTags.findIndex(([tag]) => tag === activeTag);
26
+
27
+ // Compute visible tags:
28
+ // - Filtering or expanded: show all
29
+ // - Default: show first INITIAL_SHOW, then append active tag (with a separator)
30
+ // if it falls beyond the initial slice — keeps sidebar compact while
31
+ // always making the selected tag visible
32
+ const getVisibleTags = (): { entries: [string, number][]; appendedAt: number | null } => {
33
+ if (filter || expanded) return { entries: sortedTags, appendedAt: null };
34
+ const initial = sortedTags.slice(0, INITIAL_SHOW);
35
+ if (activeIndex >= INITIAL_SHOW) {
36
+ return { entries: [...initial, sortedTags[activeIndex]], appendedAt: INITIAL_SHOW };
37
+ }
38
+ return { entries: initial, appendedAt: null };
39
+ };
40
+
41
+ const { entries: visibleTags, appendedAt } = getVisibleTags();
42
+ const remainingCount = sortedTags.length - visibleTags.length;
43
+ const showExpandButton = !filter && !expanded && remainingCount > 0;
44
+ // Only allow collapsing if it won't hide the active tag
45
+ const showCollapseButton = expanded && !filter && (activeIndex === -1 || activeIndex < INITIAL_SHOW);
46
+
47
+ return (
48
+ <aside className="hidden lg:block flex-shrink-0">
49
+ <div className="sticky top-24">
50
+
51
+ {/* Section heading → links to all tags, shows total count */}
52
+ <Link
53
+ href="/tags"
54
+ className="flex items-center gap-1.5 text-[10px] font-sans font-bold uppercase tracking-widest text-muted hover:text-accent transition-colors no-underline mb-3"
55
+ >
56
+ <LuTag className="w-3 h-3" />
57
+ <span>{t('tags')}</span>
58
+ <span className="ml-auto font-mono font-normal normal-case tracking-normal text-muted/50">
59
+ {totalCount}
60
+ </span>
61
+ </Link>
62
+
63
+ {/* Filter input with clear button */}
64
+ <div className="relative mb-3">
65
+ <LuSearch className="absolute left-2.5 top-1/2 -translate-y-1/2 w-3 h-3 text-muted/40 pointer-events-none" />
66
+ <input
67
+ type="text"
68
+ value={filter}
69
+ onChange={(e) => setFilter(e.target.value)}
70
+ placeholder="Filter…"
71
+ aria-label={t('filter_tags')}
72
+ className="w-full pl-8 pr-7 py-1.5 text-xs bg-muted/5 border border-muted/15 rounded-lg outline-none focus:border-accent/40 text-foreground placeholder:text-muted/40 transition-colors"
73
+ />
74
+ {filter && (
75
+ <button
76
+ onClick={() => setFilter('')}
77
+ className="absolute right-2 top-1/2 -translate-y-1/2 text-muted/40 hover:text-muted transition-colors p-0.5 rounded"
78
+ aria-label="Clear filter"
79
+ >
80
+ <LuX className="w-3 h-3" />
81
+ </button>
82
+ )}
83
+ </div>
84
+
85
+ {/* Tag list — no overflow, no scrollbar */}
86
+ <nav className="space-y-0.5">
87
+ {visibleTags.map(([tag, count], index) => {
88
+ const isActive = tag === activeTag;
89
+ // Thin separator before the appended active tag
90
+ const showSeparator = appendedAt !== null && index === appendedAt;
91
+ return (
92
+ <div key={tag}>
93
+ {showSeparator && <div className="my-1.5 h-px bg-muted/10" />}
94
+ <Link
95
+ href={`/tags/${encodeURIComponent(tag)}`}
96
+ className={`flex items-center justify-between px-2.5 py-1.5 rounded-lg text-sm no-underline transition-colors ${
97
+ isActive
98
+ ? 'bg-accent/10 text-accent font-medium'
99
+ : 'text-foreground/70 hover:text-foreground hover:bg-muted/10'
100
+ }`}
101
+ >
102
+ <span className="truncate">{tag}</span>
103
+ <span className={`ml-2 text-xs font-mono flex-shrink-0 ${isActive ? 'text-accent/70' : 'text-muted/50'}`}>
104
+ {count}
105
+ </span>
106
+ </Link>
107
+ </div>
108
+ );
109
+ })}
110
+
111
+ {/* Expand button */}
112
+ {showExpandButton && (
113
+ <button
114
+ onClick={() => setExpanded(true)}
115
+ aria-expanded={false}
116
+ className="w-full text-left px-2.5 py-1.5 text-xs text-muted/50 hover:text-accent transition-colors"
117
+ >
118
+ {tWith('more_tags', { count: remainingCount })}
119
+ </button>
120
+ )}
121
+
122
+ {/* Collapse button */}
123
+ {showCollapseButton && (
124
+ <button
125
+ onClick={() => setExpanded(false)}
126
+ aria-expanded={true}
127
+ className="w-full text-left px-2.5 py-1.5 text-xs text-muted/50 hover:text-accent transition-colors"
128
+ >
129
+ {t('collapse_tags')}
130
+ </button>
131
+ )}
132
+
133
+ {/* Empty state */}
134
+ {visibleTags.length === 0 && (
135
+ <p className="text-xs text-muted/60 italic px-2.5 py-2">{t('no_tags_found')}</p>
136
+ )}
137
+ </nav>
138
+
139
+ </div>
140
+ </aside>
141
+ );
142
+ }
@@ -0,0 +1,156 @@
1
+ 'use client';
2
+
3
+ import Link from 'next/link';
4
+ import { useState } from 'react';
5
+ import { LuSearch, LuX } from 'react-icons/lu';
6
+ import { useLanguage } from './LanguageProvider';
7
+
8
+ interface TagsIndexClientProps {
9
+ tags: Record<string, number>;
10
+ }
11
+
12
+ type SortMode = 'popular' | 'alpha';
13
+
14
+ function getTagClasses(count: number, min: number, max: number): string {
15
+ const ratio = max === min ? 0.5 : (count - min) / (max - min);
16
+ if (ratio >= 0.8) return 'text-xl font-bold px-5 py-2.5';
17
+ if (ratio >= 0.6) return 'text-lg font-semibold px-5 py-2';
18
+ if (ratio >= 0.4) return 'text-base font-medium px-4 py-2';
19
+ if (ratio >= 0.2) return 'text-sm px-3.5 py-1.5';
20
+ return 'text-xs px-3 py-1.5';
21
+ }
22
+
23
+ function TagLink({ tag, count, min, max }: { tag: string; count: number; min: number; max: number }) {
24
+ return (
25
+ <Link
26
+ href={`/tags/${encodeURIComponent(tag)}`}
27
+ className={`group inline-flex items-baseline gap-1.5 rounded-xl border border-muted/20 bg-muted/5 hover:bg-background hover:border-accent hover:shadow-md hover:shadow-accent/5 no-underline transition-all duration-200 ${getTagClasses(count, min, max)}`}
28
+ >
29
+ <span className="text-foreground group-hover:text-accent transition-colors">{tag}</span>
30
+ <span className="font-mono text-muted/50 group-hover:text-accent/50 transition-colors" style={{ fontSize: '0.7em' }}>{count}</span>
31
+ </Link>
32
+ );
33
+ }
34
+
35
+ export default function TagsIndexClient({ tags }: TagsIndexClientProps) {
36
+ const { t, tWith } = useLanguage();
37
+ const [filter, setFilter] = useState('');
38
+ const [sort, setSort] = useState<SortMode>('popular');
39
+
40
+ const total = Object.keys(tags).length;
41
+ const allEntries = Object.entries(tags);
42
+ const counts = allEntries.map(([, c]) => c);
43
+ const min = Math.min(...counts);
44
+ const max = Math.max(...counts);
45
+
46
+ const filtered = allEntries
47
+ .filter(([tag]) => !filter || tag.toLowerCase().includes(filter.toLowerCase()))
48
+ .sort((a, b) =>
49
+ sort === 'popular'
50
+ ? b[1] - a[1]
51
+ : a[0].localeCompare(b[0])
52
+ );
53
+
54
+ // Group by first letter for A-Z mode
55
+ const letterGroups = sort === 'alpha'
56
+ ? filtered.reduce<Record<string, [string, number][]>>((acc, entry) => {
57
+ const letter = /^[a-zA-Z]/.test(entry[0]) ? entry[0][0].toUpperCase() : '#';
58
+ if (!acc[letter]) acc[letter] = [];
59
+ acc[letter].push(entry);
60
+ return acc;
61
+ }, {})
62
+ : null;
63
+
64
+ const sortedLetters = letterGroups
65
+ ? Object.keys(letterGroups).sort((a, b) => a === '#' ? 1 : b === '#' ? -1 : a.localeCompare(b))
66
+ : null;
67
+
68
+ return (
69
+ <div>
70
+ {/* Controls */}
71
+ <div className="flex flex-col sm:flex-row gap-3 mb-10">
72
+ <div className="relative flex-1 max-w-sm">
73
+ <LuSearch className="absolute left-3 top-1/2 -translate-y-1/2 w-3.5 h-3.5 text-muted/50 pointer-events-none" />
74
+ <input
75
+ type="text"
76
+ value={filter}
77
+ onChange={(e) => setFilter(e.target.value)}
78
+ placeholder="Filter tags…"
79
+ aria-label={t('filter_tags')}
80
+ className="w-full pl-9 pr-8 py-2 text-sm bg-muted/5 border border-muted/15 rounded-lg outline-none focus:border-accent/40 text-foreground placeholder:text-muted/40 transition-colors"
81
+ />
82
+ {filter && (
83
+ <button
84
+ onClick={() => setFilter('')}
85
+ className="absolute right-2.5 top-1/2 -translate-y-1/2 text-muted/40 hover:text-muted transition-colors p-0.5 rounded"
86
+ aria-label="Clear filter"
87
+ >
88
+ <LuX className="w-3.5 h-3.5" />
89
+ </button>
90
+ )}
91
+ </div>
92
+
93
+ <div className="flex rounded-lg border border-muted/15 overflow-hidden text-xs font-sans font-semibold self-start">
94
+ <button
95
+ type="button"
96
+ onClick={() => setSort('popular')}
97
+ aria-pressed={sort === 'popular'}
98
+ className={`px-4 py-2 transition-colors ${sort === 'popular' ? 'bg-accent/10 text-accent' : 'text-muted hover:text-foreground hover:bg-muted/5'}`}
99
+ >
100
+ {t('sort_popular')}
101
+ </button>
102
+ <button
103
+ type="button"
104
+ onClick={() => setSort('alpha')}
105
+ aria-pressed={sort === 'alpha'}
106
+ className={`px-4 py-2 border-l border-muted/15 transition-colors ${sort === 'alpha' ? 'bg-accent/10 text-accent' : 'text-muted hover:text-foreground hover:bg-muted/5'}`}
107
+ >
108
+ {t('sort_az')}
109
+ </button>
110
+ </div>
111
+ </div>
112
+
113
+ {/* Result count when filtering */}
114
+ {filter && (
115
+ <p className="text-xs font-mono text-muted mb-6">
116
+ {tWith('tags_count', { shown: filtered.length, total })}
117
+ </p>
118
+ )}
119
+
120
+ {/* Popular mode: flat size-scaled cloud */}
121
+ {sort === 'popular' && (
122
+ <div className="flex flex-wrap gap-3 items-baseline">
123
+ {filtered.map(([tag, count]) => (
124
+ <TagLink key={tag} tag={tag} count={count} min={min} max={max} />
125
+ ))}
126
+ {filtered.length === 0 && (
127
+ <p className="text-sm text-muted italic">{tWith('tags_no_match', { filter })}</p>
128
+ )}
129
+ </div>
130
+ )}
131
+
132
+ {/* A-Z mode: grouped under letter section headers */}
133
+ {sort === 'alpha' && sortedLetters && (
134
+ <div>
135
+ {sortedLetters.length === 0 ? (
136
+ <p className="text-sm text-muted italic">{tWith('tags_no_match', { filter })}</p>
137
+ ) : (
138
+ sortedLetters.map((letter, i) => (
139
+ <div key={letter} className={i > 0 ? 'mt-10' : ''}>
140
+ <div className="flex items-center gap-3 mb-4">
141
+ <span className="text-xs font-mono font-bold text-muted/40 w-4">{letter}</span>
142
+ <div className="flex-1 h-px bg-muted/10" />
143
+ </div>
144
+ <div className="flex flex-wrap gap-3 items-baseline">
145
+ {letterGroups![letter].map(([tag, count]) => (
146
+ <TagLink key={tag} tag={tag} count={count} min={min} max={max} />
147
+ ))}
148
+ </div>
149
+ </div>
150
+ ))
151
+ )}
152
+ </div>
153
+ )}
154
+ </div>
155
+ );
156
+ }
@@ -0,0 +1,41 @@
1
+ 'use client';
2
+
3
+ import { useState, useEffect } from 'react';
4
+
5
+ /**
6
+ * Module-level subscriber set — only one DOM scroll listener exists
7
+ * regardless of how many components call this hook.
8
+ */
9
+ const listeners = new Set<(y: number) => void>();
10
+
11
+ function onScroll() {
12
+ const y = window.scrollY;
13
+ listeners.forEach(fn => fn(y));
14
+ }
15
+
16
+ /**
17
+ * Returns the current window.scrollY, updating on every scroll event.
18
+ * A single passive scroll listener is shared across all consumers.
19
+ */
20
+ export function useScrollY(): number {
21
+ const [scrollY, setScrollY] = useState(0);
22
+
23
+ useEffect(() => {
24
+ if (listeners.size === 0) {
25
+ window.addEventListener('scroll', onScroll, { passive: true });
26
+ }
27
+ listeners.add(setScrollY);
28
+ // Sync on mount via RAF to avoid cascading render error
29
+ const rafId = requestAnimationFrame(() => setScrollY(window.scrollY));
30
+
31
+ return () => {
32
+ cancelAnimationFrame(rafId);
33
+ listeners.delete(setScrollY);
34
+ if (listeners.size === 0) {
35
+ window.removeEventListener('scroll', onScroll);
36
+ }
37
+ };
38
+ }, []);
39
+
40
+ return scrollY;
41
+ }
@@ -6,7 +6,7 @@ export const translations = {
6
6
  tags: "Tags",
7
7
  about: "About",
8
8
  search: "Search...",
9
- search_placeholder: "Search posts...",
9
+ search_placeholder: "Search...",
10
10
  no_results: "No results found.",
11
11
  latest_writing: "Latest Writing",
12
12
  curated_series: "Curated Series",
@@ -25,8 +25,12 @@ export const translations = {
25
25
  categories: "Categories",
26
26
  articles: "Articles",
27
27
  posts: "Posts",
28
+ links: "Links",
28
29
  explore: "Explore",
29
30
  connect: "Connect",
31
+ rss_feed: "RSS Feed",
32
+ privacy: "Privacy",
33
+ built_with: "Built with Amytis",
30
34
  on_this_page: "On this page",
31
35
  back_to_top: "Back to top",
32
36
  archive_subtitle: "{count} posts across {years} years.",
@@ -63,6 +67,56 @@ export const translations = {
63
67
  flows_in_month: "Notes in {month}",
64
68
  browse: "Browse",
65
69
  clear: "Clear",
70
+ search_all: "All",
71
+ recent_searches: "Recent",
72
+ search_showing: "Showing {shown} of {total} results",
73
+ search_results_found: "{total} results for \"{query}\"",
74
+ search_no_results_for: "No results for \"{query}\"",
75
+ search_type_post: "Post",
76
+ search_type_flow: "Flow",
77
+ search_type_book: "Book",
78
+ search_tips: "Tips",
79
+ search_tip_phrase: "Quoted string for exact matching (\" \" = Exact match)",
80
+ search_tip_and: "Use spaces to combine keywords (Space = AND)",
81
+ search_tip_exclude: "Exclude a term (- = Exclude)",
82
+ discuss_post: "Discuss this post",
83
+ share_post: "Share",
84
+ copy_link: "Copy link",
85
+ link_copied: "Copied!",
86
+ flow_notes: "Flow Notes",
87
+ tag_post_count: "{count} posts",
88
+ tag_post_count_one: "1 post",
89
+ tag_flow_count: "{count} flow notes",
90
+ tag_flow_count_one: "1 flow note",
91
+ subscribe: "Subscribe",
92
+ subscribe_subtitle: "Stay updated with new posts and notes via your preferred channel.",
93
+ rss_readers: "RSS Readers",
94
+ rss_description: "Subscribe with any RSS reader for automatic updates when new content is published.",
95
+ email_newsletter: "Email Newsletter",
96
+ email_newsletter_description: "Get new posts delivered directly to your inbox.",
97
+ telegram_channel: "Telegram",
98
+ telegram_channel_description: "Instant updates via Telegram channel.",
99
+ wechat_official: "WeChat Official Account",
100
+ wechat_description: "Follow on WeChat for updates.",
101
+ scan_qr_code: "Scan to follow",
102
+ copy_feed_url: "Copy feed URL",
103
+ feed_url_copied: "Copied!",
104
+ join_channel: "Join Channel",
105
+ subscribe_on_substack: "Subscribe on Substack",
106
+ subscribe_via_email: "Subscribe via Email",
107
+ social_connections: "Social",
108
+ older: "Older",
109
+ newer: "Newer",
110
+ tab_all: "All",
111
+ post_navigation: "Post navigation",
112
+ filter_tags: "Filter tags",
113
+ no_tags_found: "No tags found",
114
+ more_tags: "+ {count} more",
115
+ collapse_tags: "Show less",
116
+ sort_popular: "Popular",
117
+ sort_az: "A–Z",
118
+ tags_count: "{shown} / {total} tags",
119
+ tags_no_match: "No tags match \"{filter}\"",
66
120
  },
67
121
  zh: {
68
122
  home: "首页",
@@ -71,7 +125,7 @@ export const translations = {
71
125
  tags: "标签",
72
126
  about: "关于",
73
127
  search: "搜索...",
74
- search_placeholder: "搜索文章...",
128
+ search_placeholder: "搜索...",
75
129
  no_results: "未找到结果。",
76
130
  latest_writing: "最新文章",
77
131
  curated_series: "精选系列",
@@ -90,8 +144,12 @@ export const translations = {
90
144
  categories: "分类",
91
145
  articles: "文章",
92
146
  posts: "文章",
147
+ links: "链接",
93
148
  explore: "探索",
94
149
  connect: "连接",
150
+ rss_feed: "RSS 订阅",
151
+ privacy: "隐私政策",
152
+ built_with: "基于 Amytis 构建",
95
153
  on_this_page: "本页目录",
96
154
  back_to_top: "返回顶部",
97
155
  archive_subtitle: "横跨 {years} 年,共 {count} 篇文章。",
@@ -128,6 +186,56 @@ export const translations = {
128
186
  flows_in_month: "{month} 随笔",
129
187
  browse: "浏览",
130
188
  clear: "清除",
189
+ search_all: "全部",
190
+ recent_searches: "最近",
191
+ search_showing: "显示 {shown} / {total} 条结果",
192
+ search_results_found: "找到 {total} 条\"{query}\"的结果",
193
+ search_no_results_for: "未找到\"{query}\"的相关结果",
194
+ search_type_post: "文章",
195
+ search_type_flow: "随笔",
196
+ search_type_book: "书籍",
197
+ search_tips: "搜索技巧",
198
+ search_tip_phrase: "加引号精确匹配短语(\" \" 表示精确匹配)",
199
+ search_tip_and: "使用空格组合关键词(空格表示 AND)",
200
+ search_tip_exclude: "排除关键词(- 表示排除)",
201
+ discuss_post: "讨论这篇文章",
202
+ share_post: "分享",
203
+ copy_link: "复制链接",
204
+ link_copied: "已复制",
205
+ flow_notes: "随笔",
206
+ tag_post_count: "{count} 篇文章",
207
+ tag_post_count_one: "1 篇文章",
208
+ tag_flow_count: "{count} 条随笔",
209
+ tag_flow_count_one: "1 条随笔",
210
+ subscribe: "订阅",
211
+ subscribe_subtitle: "通过您喜爱的方式订阅,及时获取新文章和随笔。",
212
+ rss_readers: "RSS 阅读器",
213
+ rss_description: "通过 RSS 阅读器订阅,发布新内容时自动获取更新。",
214
+ email_newsletter: "邮件订阅",
215
+ email_newsletter_description: "将新文章直接发送到您的邮箱。",
216
+ telegram_channel: "Telegram 频道",
217
+ telegram_channel_description: "通过 Telegram 频道获取即时更新。",
218
+ wechat_official: "微信公众号",
219
+ wechat_description: "关注微信公众号获取更新。",
220
+ scan_qr_code: "扫码关注",
221
+ copy_feed_url: "复制订阅链接",
222
+ feed_url_copied: "已复制!",
223
+ join_channel: "加入频道",
224
+ subscribe_on_substack: "在 Substack 订阅",
225
+ subscribe_via_email: "邮件订阅",
226
+ social_connections: "社交媒体",
227
+ older: "更早",
228
+ newer: "更新",
229
+ tab_all: "全部",
230
+ post_navigation: "文章导航",
231
+ filter_tags: "筛选标签",
232
+ no_tags_found: "未找到标签",
233
+ more_tags: "还有 {count} 个",
234
+ collapse_tags: "收起",
235
+ sort_popular: "热门",
236
+ sort_az: "A–Z",
237
+ tags_count: "{shown} / {total} 个标签",
238
+ tags_no_match: "未找到匹配\"{filter}\"的标签",
131
239
  },
132
240
  };
133
241
 
@@ -8,6 +8,9 @@ import Comments from '@/components/Comments';
8
8
  import ExternalLinks from '@/components/ExternalLinks';
9
9
  import Tag from '@/components/Tag';
10
10
  import ReadingProgressBar from '@/components/ReadingProgressBar';
11
+ import PostNavigation from '@/components/PostNavigation';
12
+ import AuthorCard from '@/components/AuthorCard';
13
+ import ShareBar from '@/components/ShareBar';
11
14
  import { siteConfig } from '../../site.config';
12
15
  import { t } from '@/lib/i18n';
13
16
 
@@ -16,15 +19,18 @@ interface PostLayoutProps {
16
19
  relatedPosts?: PostData[];
17
20
  seriesPosts?: PostData[];
18
21
  seriesTitle?: string;
22
+ prevPost?: PostData | null;
23
+ nextPost?: PostData | null;
19
24
  }
20
25
 
21
- export default function PostLayout({ post, relatedPosts, seriesPosts, seriesTitle }: PostLayoutProps) {
22
- const showToc = siteConfig.toc !== false && post.toc !== false && post.headings && post.headings.length > 0;
26
+ export default function PostLayout({ post, relatedPosts, seriesPosts, seriesTitle, prevPost, nextPost }: PostLayoutProps) {
27
+ const showToc = siteConfig.posts?.toc !== false && post.toc !== false && post.headings && post.headings.length > 0;
23
28
  const hasSeries = !!(post.series && seriesPosts && seriesPosts.length > 0);
24
29
  const showSidebar = showToc || hasSeries;
30
+ const postUrl = `${siteConfig.baseUrl}/posts/${post.slug}`;
25
31
 
26
32
  return (
27
- <div className={`layout-container ${showSidebar ? 'lg:max-w-7xl' : 'lg:max-w-6xl'}`}>
33
+ <div className="layout-container">
28
34
  <ReadingProgressBar />
29
35
  <div className={showSidebar
30
36
  ? 'grid grid-cols-1 lg:grid-cols-[280px_minmax(0,1fr)] gap-8 items-start'
@@ -38,11 +44,13 @@ export default function PostLayout({ post, relatedPosts, seriesPosts, seriesTitl
38
44
  posts={hasSeries ? seriesPosts : undefined}
39
45
  currentSlug={post.slug}
40
46
  headings={showToc ? post.headings : []}
47
+ shareUrl={postUrl}
48
+ shareTitle={post.title}
41
49
  />
42
50
  )}
43
51
 
44
- <article className="min-w-0 max-w-3xl">
45
- <header className="mb-16 border-b border-muted/10 pb-12">
52
+ <article className="min-w-0 max-w-3xl mx-auto">
53
+ <header className="mb-16 border-b border-muted/10 pb-8">
46
54
  {post.draft && (
47
55
  <div className="mb-4">
48
56
  <span className="text-xs font-bold text-red-500 bg-red-100 dark:bg-red-900/30 px-2 py-1 rounded tracking-widest inline-block">
@@ -55,7 +63,7 @@ export default function PostLayout({ post, relatedPosts, seriesPosts, seriesTitl
55
63
  {post.category}
56
64
  </span>
57
65
  <span className="w-1 h-1 rounded-full bg-muted/30" />
58
- <time className="font-mono">{post.date}</time>
66
+ <time className="font-mono" data-pagefind-meta="date[content]">{post.date}</time>
59
67
  <span className="w-1 h-1 rounded-full bg-muted/30" />
60
68
  <span className="font-mono">{post.readingTime}</span>
61
69
  </div>
@@ -104,13 +112,32 @@ export default function PostLayout({ post, relatedPosts, seriesPosts, seriesTitl
104
112
 
105
113
  <MarkdownRenderer content={post.content} latex={post.latex} slug={post.slug} />
106
114
 
115
+ {post.tags && post.tags.length > 0 && (
116
+ <div className="mt-12 pt-12 border-t border-muted/20 flex flex-wrap items-center gap-2">
117
+ <span className="text-[10px] font-sans font-bold uppercase tracking-widest text-muted mr-1">{t('tags')}</span>
118
+ {post.tags.map((tag) => (
119
+ <Tag key={tag} tag={tag} variant="default" />
120
+ ))}
121
+ </div>
122
+ )}
123
+
107
124
  {post.externalLinks && post.externalLinks.length > 0 && (
108
125
  <ExternalLinks links={post.externalLinks} />
109
126
  )}
110
127
 
111
- <RelatedPosts posts={relatedPosts || []} />
128
+ <ShareBar
129
+ url={postUrl}
130
+ title={post.title}
131
+ className={showSidebar ? 'mt-8 lg:hidden' : 'mt-8'}
132
+ />
133
+
134
+ <AuthorCard authors={post.authors} />
135
+
136
+ <PostNavigation prev={prevPost ?? null} next={nextPost ?? null} />
112
137
 
113
138
  <Comments slug={post.slug} />
139
+
140
+ <RelatedPosts posts={relatedPosts || []} />
114
141
  </article>
115
142
  </div>
116
143
  </div>
@@ -1,31 +1,69 @@
1
1
  import { PostData } from '@/lib/markdown';
2
2
  import MarkdownRenderer from '@/components/MarkdownRenderer';
3
3
  import SimpleLayoutHeader from '@/components/SimpleLayoutHeader';
4
+ import LocaleSwitch from '@/components/LocaleSwitch';
5
+ import PostSidebar from '@/components/PostSidebar';
4
6
  import { TranslationKey } from '@/i18n/translations';
7
+ import { siteConfig } from '../../site.config';
5
8
 
6
9
  interface SimpleLayoutProps {
7
10
  post: PostData;
8
11
  titleKey?: TranslationKey;
9
12
  subtitleKey?: TranslationKey;
10
- titleOverride?: string | Record<string, string>;
11
- subtitleOverride?: string | Record<string, string>;
12
13
  }
13
14
 
14
- export default function SimpleLayout({ post, titleKey, subtitleKey, titleOverride, subtitleOverride }: SimpleLayoutProps) {
15
- return (
16
- <div className="layout-main">
17
- <article className="max-w-3xl mx-auto">
18
- <SimpleLayoutHeader
19
- title={post.title}
20
- excerpt={post.excerpt}
21
- titleKey={titleKey}
22
- subtitleKey={subtitleKey}
23
- titleOverride={titleOverride}
24
- subtitleOverride={subtitleOverride}
25
- />
15
+ export default function SimpleLayout({ post, titleKey, subtitleKey }: SimpleLayoutProps) {
16
+ const defaultLocale = siteConfig.i18n.defaultLocale;
17
+ const localeEntries = Object.entries(post.contentLocales ?? {});
18
+ const showToc = siteConfig.posts?.toc !== false && post.toc !== false && post.headings?.length > 0;
19
+ const localeHeadings = post.contentLocales
20
+ ? Object.fromEntries(
21
+ Object.entries(post.contentLocales)
22
+ .filter(([, data]) => data.headings && data.headings.length > 0)
23
+ .map(([locale, data]) => [locale, data.headings!])
24
+ )
25
+ : undefined;
26
26
 
27
+ const articleContent = (
28
+ <>
29
+ <SimpleLayoutHeader
30
+ title={post.title}
31
+ excerpt={post.excerpt}
32
+ titleKey={titleKey}
33
+ subtitleKey={subtitleKey}
34
+ contentLocales={post.contentLocales}
35
+ />
36
+ {localeEntries.length > 0 ? (
37
+ <LocaleSwitch>
38
+ <div data-locale={defaultLocale}>
39
+ <MarkdownRenderer content={post.content} latex={post.latex} slug={post.slug} />
40
+ </div>
41
+ {localeEntries.map(([locale, data]) => (
42
+ <div key={locale} data-locale={locale} style={{ display: 'none' }}>
43
+ <MarkdownRenderer content={data.content} latex={post.latex} slug={post.slug} />
44
+ </div>
45
+ ))}
46
+ </LocaleSwitch>
47
+ ) : (
27
48
  <MarkdownRenderer content={post.content} latex={post.latex} slug={post.slug} />
28
- </article>
49
+ )}
50
+ </>
51
+ );
52
+
53
+ return (
54
+ <div className="layout-main">
55
+ {showToc ? (
56
+ <div className="grid grid-cols-1 lg:grid-cols-[280px_minmax(0,1fr)] gap-8 items-start">
57
+ <PostSidebar currentSlug={post.slug} headings={post.headings} localeHeadings={localeHeadings} />
58
+ <article className="min-w-0 max-w-3xl">
59
+ {articleContent}
60
+ </article>
61
+ </div>
62
+ ) : (
63
+ <article className="max-w-3xl mx-auto">
64
+ {articleContent}
65
+ </article>
66
+ )}
29
67
  </div>
30
68
  );
31
69
  }