@jant/core 0.3.25 → 0.3.26

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 (131) hide show
  1. package/dist/app.js +67 -562
  2. package/dist/client.js +1 -0
  3. package/dist/i18n/locales/en.js +1 -1
  4. package/dist/i18n/locales/zh-Hans.js +1 -1
  5. package/dist/i18n/locales/zh-Hant.js +1 -1
  6. package/dist/lib/avatar-upload.js +134 -0
  7. package/dist/lib/config.js +39 -0
  8. package/dist/lib/constants.js +10 -10
  9. package/dist/lib/favicon.js +102 -0
  10. package/dist/lib/image.js +13 -17
  11. package/dist/lib/media-helpers.js +2 -2
  12. package/dist/lib/navigation.js +23 -3
  13. package/dist/lib/render.js +10 -1
  14. package/dist/lib/schemas.js +31 -0
  15. package/dist/lib/timezones.js +388 -0
  16. package/dist/lib/view.js +1 -1
  17. package/dist/routes/api/posts.js +1 -1
  18. package/dist/routes/api/upload.js +3 -3
  19. package/dist/routes/auth/reset.js +221 -0
  20. package/dist/routes/auth/setup.js +194 -0
  21. package/dist/routes/auth/signin.js +176 -0
  22. package/dist/routes/dash/collections.js +23 -415
  23. package/dist/routes/dash/media.js +12 -392
  24. package/dist/routes/dash/pages.js +7 -330
  25. package/dist/routes/dash/redirects.js +18 -12
  26. package/dist/routes/dash/settings.js +198 -577
  27. package/dist/routes/feed/rss.js +2 -1
  28. package/dist/routes/feed/sitemap.js +4 -2
  29. package/dist/routes/pages/featured.js +5 -1
  30. package/dist/routes/pages/home.js +26 -1
  31. package/dist/routes/pages/latest.js +45 -0
  32. package/dist/services/post.js +30 -50
  33. package/dist/types/bindings.js +3 -0
  34. package/dist/types/config.js +147 -0
  35. package/dist/types/constants.js +27 -0
  36. package/dist/types/entities.js +3 -0
  37. package/dist/types/operations.js +3 -0
  38. package/dist/types/props.js +3 -0
  39. package/dist/types/views.js +5 -0
  40. package/dist/types.js +8 -111
  41. package/dist/ui/color-themes.js +33 -33
  42. package/dist/ui/compose/ComposeDialog.js +36 -21
  43. package/dist/ui/dash/PageForm.js +21 -15
  44. package/dist/ui/dash/PostForm.js +22 -16
  45. package/dist/ui/dash/collections/CollectionForm.js +152 -0
  46. package/dist/ui/dash/collections/CollectionsListContent.js +68 -0
  47. package/dist/ui/dash/collections/ViewCollectionContent.js +96 -0
  48. package/dist/ui/dash/media/MediaListContent.js +166 -0
  49. package/dist/ui/dash/media/ViewMediaContent.js +212 -0
  50. package/dist/ui/dash/pages/LinkFormContent.js +130 -0
  51. package/dist/ui/dash/pages/UnifiedPagesContent.js +193 -0
  52. package/dist/ui/dash/settings/AccountContent.js +209 -0
  53. package/dist/ui/dash/settings/AppearanceContent.js +259 -0
  54. package/dist/ui/dash/settings/GeneralContent.js +536 -0
  55. package/dist/ui/dash/settings/SettingsNav.js +41 -0
  56. package/dist/ui/font-themes.js +36 -0
  57. package/dist/ui/layouts/BaseLayout.js +24 -2
  58. package/dist/ui/layouts/SiteLayout.js +47 -19
  59. package/package.json +1 -1
  60. package/src/app.tsx +93 -553
  61. package/src/client.ts +1 -0
  62. package/src/i18n/locales/en.po +240 -175
  63. package/src/i18n/locales/en.ts +1 -1
  64. package/src/i18n/locales/zh-Hans.po +240 -175
  65. package/src/i18n/locales/zh-Hans.ts +1 -1
  66. package/src/i18n/locales/zh-Hant.po +240 -175
  67. package/src/i18n/locales/zh-Hant.ts +1 -1
  68. package/src/lib/__tests__/config.test.ts +192 -0
  69. package/src/lib/__tests__/favicon.test.ts +151 -0
  70. package/src/lib/__tests__/image.test.ts +2 -6
  71. package/src/lib/__tests__/timezones.test.ts +61 -0
  72. package/src/lib/__tests__/view.test.ts +2 -2
  73. package/src/lib/avatar-upload.ts +165 -0
  74. package/src/lib/config.ts +47 -0
  75. package/src/lib/constants.ts +19 -11
  76. package/src/lib/favicon.ts +115 -0
  77. package/src/lib/image.ts +13 -21
  78. package/src/lib/media-helpers.ts +2 -2
  79. package/src/lib/navigation.ts +33 -2
  80. package/src/lib/render.tsx +15 -1
  81. package/src/lib/schemas.ts +39 -0
  82. package/src/lib/timezones.ts +325 -0
  83. package/src/lib/view.ts +1 -1
  84. package/src/routes/api/posts.ts +1 -1
  85. package/src/routes/api/upload.ts +2 -3
  86. package/src/routes/auth/reset.tsx +239 -0
  87. package/src/routes/auth/setup.tsx +189 -0
  88. package/src/routes/auth/signin.tsx +163 -0
  89. package/src/routes/dash/__tests__/settings-avatar.test.ts +89 -0
  90. package/src/routes/dash/collections.tsx +17 -366
  91. package/src/routes/dash/media.tsx +12 -414
  92. package/src/routes/dash/pages.tsx +8 -348
  93. package/src/routes/dash/redirects.tsx +20 -14
  94. package/src/routes/dash/settings.tsx +243 -534
  95. package/src/routes/feed/__tests__/rss.test.ts +141 -0
  96. package/src/routes/feed/rss.ts +3 -1
  97. package/src/routes/feed/sitemap.ts +4 -2
  98. package/src/routes/pages/featured.tsx +7 -1
  99. package/src/routes/pages/home.tsx +25 -2
  100. package/src/routes/pages/latest.tsx +59 -0
  101. package/src/services/post.ts +34 -66
  102. package/src/styles/components.css +0 -65
  103. package/src/styles/tokens.css +1 -1
  104. package/src/styles/ui.css +24 -40
  105. package/src/types/bindings.ts +30 -0
  106. package/src/types/config.ts +183 -0
  107. package/src/types/constants.ts +26 -0
  108. package/src/types/entities.ts +109 -0
  109. package/src/types/operations.ts +88 -0
  110. package/src/types/props.ts +115 -0
  111. package/src/types/views.ts +172 -0
  112. package/src/types.ts +8 -644
  113. package/src/ui/__tests__/font-themes.test.ts +34 -0
  114. package/src/ui/color-themes.ts +34 -34
  115. package/src/ui/compose/ComposeDialog.tsx +40 -21
  116. package/src/ui/dash/PageForm.tsx +25 -19
  117. package/src/ui/dash/PostForm.tsx +26 -20
  118. package/src/ui/dash/collections/CollectionForm.tsx +153 -0
  119. package/src/ui/dash/collections/CollectionsListContent.tsx +85 -0
  120. package/src/ui/dash/collections/ViewCollectionContent.tsx +92 -0
  121. package/src/ui/dash/media/MediaListContent.tsx +201 -0
  122. package/src/ui/dash/media/ViewMediaContent.tsx +208 -0
  123. package/src/ui/dash/pages/LinkFormContent.tsx +119 -0
  124. package/src/ui/dash/pages/UnifiedPagesContent.tsx +203 -0
  125. package/src/ui/dash/settings/AccountContent.tsx +176 -0
  126. package/src/ui/dash/settings/AppearanceContent.tsx +254 -0
  127. package/src/ui/dash/settings/GeneralContent.tsx +533 -0
  128. package/src/ui/dash/settings/SettingsNav.tsx +56 -0
  129. package/src/ui/font-themes.ts +54 -0
  130. package/src/ui/layouts/BaseLayout.tsx +17 -0
  131. package/src/ui/layouts/SiteLayout.tsx +45 -31
@@ -0,0 +1,533 @@
1
+ /**
2
+ * General settings form
3
+ */
4
+
5
+ import { useLingui } from "@lingui/react/macro";
6
+ import type { TimezoneEntry } from "../../../lib/timezones.js";
7
+ import { SettingsNav } from "./SettingsNav.js";
8
+
9
+ /**
10
+ * Build data-signals JSON with `_orig_<key>` duplicates for cancel/reset.
11
+ * Private `_orig_*` signals store original values so Cancel can revert.
12
+ * The `dirty` signal tracks whether the user has made any changes.
13
+ */
14
+ function buildSignals(fields: Record<string, string>, dirty: string): string {
15
+ const signals: Record<string, string | boolean> = {};
16
+ for (const [key, value] of Object.entries(fields)) {
17
+ signals[key] = value;
18
+ signals[`_orig_${key}`] = value;
19
+ }
20
+ signals[dirty] = false;
21
+ return JSON.stringify(signals).replace(/</g, "\\u003c");
22
+ }
23
+
24
+ /** Spinner SVG shown inside buttons during loading */
25
+ function Spinner({ show }: { show: string }) {
26
+ return (
27
+ <svg
28
+ data-show={show}
29
+ style="display:none"
30
+ class="animate-spin size-4"
31
+ xmlns="http://www.w3.org/2000/svg"
32
+ viewBox="0 0 24 24"
33
+ fill="none"
34
+ stroke="currentColor"
35
+ stroke-width="2"
36
+ stroke-linecap="round"
37
+ stroke-linejoin="round"
38
+ role="status"
39
+ >
40
+ <path d="M21 12a9 9 0 1 1-6.219-8.56" />
41
+ </svg>
42
+ );
43
+ }
44
+
45
+ /**
46
+ * Save + Cancel button pair.
47
+ * Both are disabled when no changes (`!dirty`) or during loading.
48
+ * Cancel resets all signals to originals and clears dirty.
49
+ */
50
+ function FormActions({
51
+ indicator,
52
+ dirty,
53
+ fields,
54
+ }: {
55
+ indicator: string;
56
+ dirty: string;
57
+ fields: string[];
58
+ }) {
59
+ const { t } = useLingui();
60
+ const resetExpr = [
61
+ ...fields.map((f) => `$${f} = $_orig_${f}`),
62
+ `$${dirty} = false`,
63
+ ].join("; ");
64
+
65
+ return (
66
+ <div class="flex gap-2 mt-4">
67
+ <button
68
+ type="submit"
69
+ class="btn"
70
+ disabled
71
+ data-attr:disabled={`$${indicator} || !$${dirty}`}
72
+ >
73
+ <Spinner show={`$${indicator}`} />
74
+ {t({
75
+ message: "Save",
76
+ comment: "@context: Button to save settings",
77
+ })}
78
+ </button>
79
+ <button
80
+ type="button"
81
+ class="btn-outline"
82
+ disabled
83
+ data-attr:disabled={`$${indicator} || !$${dirty}`}
84
+ data-on:click={resetExpr}
85
+ >
86
+ {t({
87
+ message: "Cancel",
88
+ comment:
89
+ "@context: Button to cancel unsaved changes and revert to original values",
90
+ })}
91
+ </button>
92
+ </div>
93
+ );
94
+ }
95
+
96
+ export function GeneralContent({
97
+ siteName,
98
+ siteDescription,
99
+ siteLanguage,
100
+ homeDefaultView,
101
+ siteNameFallback,
102
+ siteDescriptionFallback,
103
+ siteAvatarUrl,
104
+ showHeaderAvatar,
105
+ timeZone,
106
+ siteFooter,
107
+ noindex,
108
+ timezones,
109
+ }: {
110
+ siteName: string;
111
+ siteDescription: string;
112
+ siteLanguage: string;
113
+ homeDefaultView: string;
114
+ siteNameFallback: string;
115
+ siteDescriptionFallback: string;
116
+ siteAvatarUrl: string;
117
+ showHeaderAvatar: boolean;
118
+ timeZone: string;
119
+ siteFooter: string;
120
+ noindex: boolean;
121
+ timezones: TimezoneEntry[];
122
+ }) {
123
+ const { t } = useLingui();
124
+
125
+ const generalSignals = buildSignals(
126
+ {
127
+ siteName,
128
+ siteDescription,
129
+ siteLanguage,
130
+ homeDefaultView,
131
+ timeZone,
132
+ },
133
+ "_generalDirty",
134
+ );
135
+
136
+ const footerSignals = buildSignals({ siteFooter }, "_footerDirty");
137
+
138
+ const seoSignals = buildSignals(
139
+ { noindex: noindex ? "" : "true" },
140
+ "_seoDirty",
141
+ );
142
+
143
+ const avatarSignals = buildSignals(
144
+ { showHeaderAvatar: showHeaderAvatar ? "true" : "" },
145
+ "_avatarDisplayDirty",
146
+ );
147
+
148
+ return (
149
+ <>
150
+ <h1 class="text-2xl font-semibold mb-2">
151
+ {t({ message: "Settings", comment: "@context: Dashboard heading" })}
152
+ </h1>
153
+ <SettingsNav currentTab="general" />
154
+
155
+ <div class="flex flex-col gap-6 max-w-lg">
156
+ {/* Blog Avatar */}
157
+ <div class="card">
158
+ <header>
159
+ <h2>
160
+ {t({
161
+ message: "Blog Avatar",
162
+ comment: "@context: Settings section heading for avatar",
163
+ })}
164
+ </h2>
165
+ </header>
166
+ <section class="flex flex-col gap-4">
167
+ <div class="flex items-center gap-4">
168
+ {siteAvatarUrl ? (
169
+ <img
170
+ src={siteAvatarUrl}
171
+ alt=""
172
+ class="rounded-full object-cover"
173
+ style="width:64px;height:64px"
174
+ />
175
+ ) : (
176
+ <div
177
+ class="rounded-full bg-muted flex items-center justify-center text-muted-foreground"
178
+ style="width:64px;height:64px"
179
+ >
180
+ <svg
181
+ xmlns="http://www.w3.org/2000/svg"
182
+ width="24"
183
+ height="24"
184
+ viewBox="0 0 24 24"
185
+ fill="none"
186
+ stroke="currentColor"
187
+ stroke-width="2"
188
+ stroke-linecap="round"
189
+ stroke-linejoin="round"
190
+ >
191
+ <rect width="18" height="18" x="3" y="3" rx="2" ry="2" />
192
+ <circle cx="9" cy="9" r="2" />
193
+ <path d="m21 15-3.086-3.086a2 2 0 0 0-2.828 0L6 21" />
194
+ </svg>
195
+ </div>
196
+ )}
197
+ <div class="flex flex-col gap-2">
198
+ <form
199
+ action="/dash/settings/avatar"
200
+ method="post"
201
+ enctype="multipart/form-data"
202
+ class="inline"
203
+ >
204
+ <label class="btn text-sm cursor-pointer">
205
+ {t({
206
+ message: "Upload Avatar",
207
+ comment: "@context: Button to upload avatar image",
208
+ })}
209
+ <input
210
+ type="file"
211
+ name="file"
212
+ accept="image/jpeg,image/png,image/gif,image/webp,image/svg+xml"
213
+ class="hidden"
214
+ data-avatar-upload
215
+ data-text-processing={t({
216
+ message: "Processing...",
217
+ comment:
218
+ "@context: Avatar upload button text while generating favicon variants",
219
+ })}
220
+ data-text-uploading={t({
221
+ message: "Uploading...",
222
+ comment:
223
+ "@context: Avatar upload button text while uploading",
224
+ })}
225
+ data-text-error={t({
226
+ message: "Upload failed. Please try again.",
227
+ comment:
228
+ "@context: Error message when avatar upload fails",
229
+ })}
230
+ />
231
+ </label>
232
+ </form>
233
+ {siteAvatarUrl && (
234
+ <form
235
+ data-on:submit__prevent="@post('/dash/settings/avatar/remove')"
236
+ data-indicator="_removeAvatarLoading"
237
+ >
238
+ <button
239
+ type="submit"
240
+ class="btn-outline text-sm"
241
+ data-attr:disabled="$_removeAvatarLoading"
242
+ >
243
+ {t({
244
+ message: "Remove",
245
+ comment: "@context: Button to remove the blog avatar",
246
+ })}
247
+ </button>
248
+ </form>
249
+ )}
250
+ </div>
251
+ </div>
252
+ <p class="text-sm text-muted-foreground">
253
+ {t({
254
+ message:
255
+ "This is used for your favicon and apple-touch-icon. For best results, upload a square image at least 180x180 pixels.",
256
+ comment: "@context: Help text for avatar upload",
257
+ })}
258
+ </p>
259
+ <form
260
+ data-signals={avatarSignals}
261
+ data-on:submit__prevent="@post('/dash/settings/avatar/display')"
262
+ data-indicator="_avatarDisplayLoading"
263
+ >
264
+ <label class="flex items-center gap-2 cursor-pointer">
265
+ <input
266
+ type="checkbox"
267
+ class="checkbox"
268
+ data-bind="showHeaderAvatar"
269
+ data-on:change="$_avatarDisplayDirty = true"
270
+ checked={showHeaderAvatar || undefined}
271
+ value="true"
272
+ />
273
+ <span>
274
+ {t({
275
+ message: "Display avatar in my site header",
276
+ comment:
277
+ "@context: Checkbox to show avatar in the site header",
278
+ })}
279
+ </span>
280
+ </label>
281
+ <FormActions
282
+ indicator="_avatarDisplayLoading"
283
+ dirty="_avatarDisplayDirty"
284
+ fields={["showHeaderAvatar"]}
285
+ />
286
+ </form>
287
+ </section>
288
+ </div>
289
+
290
+ {/* General settings */}
291
+ <form
292
+ data-signals={generalSignals}
293
+ data-on:submit__prevent="@post('/dash/settings')"
294
+ data-indicator="_generalLoading"
295
+ >
296
+ <div class="card">
297
+ <header>
298
+ <h2>
299
+ {t({
300
+ message: "General",
301
+ comment: "@context: Settings section heading",
302
+ })}
303
+ </h2>
304
+ </header>
305
+ <section class="flex flex-col gap-4">
306
+ <div class="field">
307
+ <label class="label">
308
+ {t({
309
+ message: "Site Name",
310
+ comment: "@context: Settings form field",
311
+ })}
312
+ </label>
313
+ <input
314
+ type="text"
315
+ data-bind="siteName"
316
+ data-on:input="$_generalDirty = true"
317
+ class="input"
318
+ placeholder={siteNameFallback}
319
+ />
320
+ </div>
321
+
322
+ <div class="field">
323
+ <label class="label">
324
+ {t({
325
+ message: "About this blog",
326
+ comment:
327
+ "@context: Settings form field for site description",
328
+ })}
329
+ </label>
330
+ <textarea
331
+ data-bind="siteDescription"
332
+ data-on:input="$_generalDirty = true"
333
+ class="textarea"
334
+ rows={3}
335
+ placeholder={siteDescriptionFallback}
336
+ >
337
+ {siteDescription}
338
+ </textarea>
339
+ <p class="text-sm text-muted-foreground mt-1">
340
+ {t({
341
+ message:
342
+ "This is displayed above your blog posts on your default home page. This is also used for the meta description on your home page.",
343
+ comment: "@context: Help text for site description field",
344
+ })}
345
+ </p>
346
+ </div>
347
+
348
+ <div class="field">
349
+ <label class="label">
350
+ {t({
351
+ message: "Language",
352
+ comment: "@context: Settings form field",
353
+ })}
354
+ </label>
355
+ <select
356
+ data-bind="siteLanguage"
357
+ data-on:change="$_generalDirty = true"
358
+ class="select"
359
+ >
360
+ <option value="en" selected={siteLanguage === "en"}>
361
+ English
362
+ </option>
363
+ <option value="zh-Hans" selected={siteLanguage === "zh-Hans"}>
364
+ 简体中文
365
+ </option>
366
+ <option value="zh-Hant" selected={siteLanguage === "zh-Hant"}>
367
+ 繁體中文
368
+ </option>
369
+ </select>
370
+ </div>
371
+
372
+ <div class="field">
373
+ <label class="label">
374
+ {t({
375
+ message: "Default Homepage View",
376
+ comment: "@context: Settings form field",
377
+ })}
378
+ </label>
379
+ <select
380
+ data-bind="homeDefaultView"
381
+ data-on:change="$_generalDirty = true"
382
+ class="select"
383
+ >
384
+ <option
385
+ value="latest"
386
+ selected={homeDefaultView === "latest"}
387
+ >
388
+ {t({
389
+ message: "Latest",
390
+ comment:
391
+ "@context: Homepage view option - show latest posts",
392
+ })}
393
+ </option>
394
+ <option
395
+ value="featured"
396
+ selected={homeDefaultView === "featured"}
397
+ >
398
+ {t({
399
+ message: "Featured",
400
+ comment:
401
+ "@context: Homepage view option - show featured posts",
402
+ })}
403
+ </option>
404
+ </select>
405
+ </div>
406
+
407
+ <div class="field">
408
+ <label class="label">
409
+ {t({
410
+ message: "Time Zone",
411
+ comment: "@context: Settings form field",
412
+ })}
413
+ </label>
414
+ <select
415
+ data-bind="timeZone"
416
+ data-on:change="$_generalDirty = true"
417
+ class="select"
418
+ >
419
+ {timezones.map((tz) => (
420
+ <option
421
+ key={tz.value}
422
+ value={tz.value}
423
+ selected={timeZone === tz.value}
424
+ >
425
+ {tz.label}
426
+ </option>
427
+ ))}
428
+ </select>
429
+ </div>
430
+ <FormActions
431
+ indicator="_generalLoading"
432
+ dirty="_generalDirty"
433
+ fields={[
434
+ "siteName",
435
+ "siteDescription",
436
+ "siteLanguage",
437
+ "homeDefaultView",
438
+ "timeZone",
439
+ ]}
440
+ />
441
+ </section>
442
+ </div>
443
+ </form>
444
+
445
+ {/* Site Footer */}
446
+ <form
447
+ data-signals={footerSignals}
448
+ data-on:submit__prevent="@post('/dash/settings/footer')"
449
+ data-indicator="_footerLoading"
450
+ >
451
+ <div class="card">
452
+ <header>
453
+ <h2>
454
+ {t({
455
+ message: "Site Footer",
456
+ comment: "@context: Settings section heading for site footer",
457
+ })}
458
+ </h2>
459
+ </header>
460
+ <section class="flex flex-col gap-4">
461
+ <textarea
462
+ data-bind="siteFooter"
463
+ data-on:input="$_footerDirty = true"
464
+ class="textarea font-mono text-sm"
465
+ rows={4}
466
+ placeholder={t({
467
+ message: "Markdown supported",
468
+ comment: "@context: Placeholder for footer textarea",
469
+ })}
470
+ >
471
+ {siteFooter}
472
+ </textarea>
473
+ <p class="text-sm text-muted-foreground">
474
+ {t({
475
+ message:
476
+ "This is displayed at the bottom of all of your posts and pages. Markdown is supported.",
477
+ comment: "@context: Help text for site footer field",
478
+ })}
479
+ </p>
480
+ <FormActions
481
+ indicator="_footerLoading"
482
+ dirty="_footerDirty"
483
+ fields={["siteFooter"]}
484
+ />
485
+ </section>
486
+ </div>
487
+ </form>
488
+
489
+ {/* SEO */}
490
+ <form
491
+ data-signals={seoSignals}
492
+ data-on:submit__prevent="@post('/dash/settings/seo')"
493
+ data-indicator="_seoLoading"
494
+ >
495
+ <div class="card">
496
+ <header>
497
+ <h2>
498
+ {t({
499
+ message: "SEO",
500
+ comment: "@context: Settings section heading for SEO",
501
+ })}
502
+ </h2>
503
+ </header>
504
+ <section>
505
+ <label class="flex items-center gap-2 cursor-pointer">
506
+ <input
507
+ type="checkbox"
508
+ class="checkbox"
509
+ data-bind="noindex"
510
+ data-on:change="$_seoDirty = true"
511
+ checked={!noindex || undefined}
512
+ value="true"
513
+ />
514
+ <span>
515
+ {t({
516
+ message: "It's OK for search engines to index my site",
517
+ comment:
518
+ "@context: Checkbox for allowing search engine indexing",
519
+ })}
520
+ </span>
521
+ </label>
522
+ <FormActions
523
+ indicator="_seoLoading"
524
+ dirty="_seoDirty"
525
+ fields={["noindex"]}
526
+ />
527
+ </section>
528
+ </div>
529
+ </form>
530
+ </div>
531
+ </>
532
+ );
533
+ }
@@ -0,0 +1,56 @@
1
+ /**
2
+ * Settings sub-navigation tabs
3
+ */
4
+
5
+ import { useLingui } from "@lingui/react/macro";
6
+
7
+ export type SettingsTab = "general" | "appearance" | "account";
8
+
9
+ export function SettingsNav({ currentTab }: { currentTab: SettingsTab }) {
10
+ const { t } = useLingui();
11
+
12
+ const tabs: { id: SettingsTab; label: string; href: string }[] = [
13
+ {
14
+ id: "general",
15
+ label: t({
16
+ message: "General",
17
+ comment: "@context: Settings sub-navigation tab",
18
+ }),
19
+ href: "/dash/settings",
20
+ },
21
+ {
22
+ id: "appearance",
23
+ label: t({
24
+ message: "Appearance",
25
+ comment: "@context: Settings sub-navigation tab",
26
+ }),
27
+ href: "/dash/settings/appearance",
28
+ },
29
+ {
30
+ id: "account",
31
+ label: t({
32
+ message: "Account",
33
+ comment: "@context: Settings sub-navigation tab",
34
+ }),
35
+ href: "/dash/settings/account",
36
+ },
37
+ ];
38
+
39
+ return (
40
+ <nav class="flex gap-1 mb-6">
41
+ {tabs.map((tab) => (
42
+ <a
43
+ key={tab.id}
44
+ href={tab.href}
45
+ class={`px-3 py-2 text-sm rounded-md ${
46
+ tab.id === currentTab
47
+ ? "bg-accent text-accent-foreground font-medium"
48
+ : "text-muted-foreground hover:bg-accent hover:text-accent-foreground"
49
+ }`}
50
+ >
51
+ {tab.label}
52
+ </a>
53
+ ))}
54
+ </nav>
55
+ );
56
+ }
@@ -0,0 +1,54 @@
1
+ /**
2
+ * Built-in Font Themes
3
+ *
4
+ * System-font-only presets — no external font loading required.
5
+ */
6
+
7
+ /**
8
+ * A font theme definition with display metadata.
9
+ */
10
+ export interface FontTheme {
11
+ /** Stored in DB settings, e.g. "serif" */
12
+ id: string;
13
+ /** Display name, e.g. "Serif" */
14
+ name: string;
15
+ /** CSS font-family stack */
16
+ fontFamily: string;
17
+ /** Short description for the picker UI */
18
+ description: string;
19
+ }
20
+
21
+ export const BUILTIN_FONT_THEMES: FontTheme[] = [
22
+ {
23
+ id: "default",
24
+ name: "System Default",
25
+ // 现代系统字体栈:先英文,后 Mac/iOS 中文,再 Win 中文
26
+ fontFamily:
27
+ 'system-ui, -apple-system, "Segoe UI", Roboto, Helvetica, Arial, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", "Source Han Sans CN", sans-serif',
28
+ description: "与你的操作系统保持一致,最稳定的阅读体验",
29
+ },
30
+ {
31
+ id: "serif",
32
+ name: "Classic Serif",
33
+ // Charter 是 Apple 系统自带的极品衬线体
34
+ fontFamily:
35
+ 'Charter, "Bitstream Charter", "Sitka Text", Georgia, "Songti SC", "Source Han Serif CN", "STSong", "SimSun", serif',
36
+ description: "传统的衬线体,适合深度长文阅读",
37
+ },
38
+ {
39
+ id: "humanist",
40
+ name: "Humanist",
41
+ // Optima 具有书法韵味,Candara 是 Windows 上的优质人文体
42
+ fontFamily:
43
+ 'Optima, Candara, "Noto Sans", "Segoe UI", Roboto, "PingFang SC", "Hiragino Sans GB", "Microsoft YaHei", sans-serif',
44
+ description: "温润如玉的字体风格,兼具现代感与书法美感",
45
+ },
46
+ {
47
+ id: "mono",
48
+ name: "Monospace",
49
+ // 优先使用 JetBrains Mono 或 SF Mono
50
+ fontFamily:
51
+ '"JetBrains Mono", "SF Mono", Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", "PingFang SC", "Microsoft YaHei", monospace',
52
+ description: "等宽字体,适合技术内容或代码展示",
53
+ },
54
+ ];
@@ -23,6 +23,8 @@ export interface BaseLayoutProps {
23
23
  lang?: string;
24
24
  c?: Context;
25
25
  toast?: ToastProps;
26
+ faviconUrl?: string;
27
+ noindex?: boolean;
26
28
  }
27
29
 
28
30
  export const BaseLayout: FC<PropsWithChildren<BaseLayoutProps>> = ({
@@ -31,11 +33,19 @@ export const BaseLayout: FC<PropsWithChildren<BaseLayoutProps>> = ({
31
33
  lang,
32
34
  c,
33
35
  toast,
36
+ faviconUrl,
37
+ noindex,
34
38
  children,
35
39
  }) => {
36
40
  // Read lang from Hono context if available, otherwise use prop or default
37
41
  const resolvedLang = lang ?? (c ? c.get("lang") : "en");
38
42
 
43
+ // Read faviconUrl from context when not provided as prop (fixes dashboard favicon)
44
+ const resolvedFaviconUrl = faviconUrl ?? (c ? c.get("faviconUrl") : undefined);
45
+
46
+ // Read noindex from context when not provided as prop
47
+ const resolvedNoindex = noindex ?? (c ? c.get("noindex") : undefined);
48
+
39
49
  // Automatically wrap with I18nProvider if Context is provided
40
50
  const content = c ? <I18nProvider c={c}>{children}</I18nProvider> : children;
41
51
 
@@ -55,6 +65,13 @@ export const BaseLayout: FC<PropsWithChildren<BaseLayoutProps>> = ({
55
65
  <meta name="viewport" content="width=device-width, initial-scale=1.0" />
56
66
  <title>{title}</title>
57
67
  {description && <meta name="description" content={description} />}
68
+ {resolvedNoindex && <meta name="robots" content="noindex, nofollow" />}
69
+ {resolvedFaviconUrl && (
70
+ <>
71
+ <link rel="icon" href="/favicon.ico" sizes="16x16 32x32" />
72
+ <link rel="apple-touch-icon" href="/apple-touch-icon.png" sizes="180x180" />
73
+ </>
74
+ )}
58
75
  <ViteClient />
59
76
  <Link href="/src/style.css" rel="stylesheet" />
60
77
  {themeStyle && <style>{themeStyle}</style>}