@jant/core 0.3.25 → 0.3.27
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.
- package/dist/app.js +70 -563
- package/dist/auth.js +3 -0
- package/dist/client.js +1 -0
- package/dist/i18n/locales/en.js +1 -1
- package/dist/i18n/locales/zh-Hans.js +1 -1
- package/dist/i18n/locales/zh-Hant.js +1 -1
- package/dist/lib/avatar-upload.js +134 -0
- package/dist/lib/config.js +39 -0
- package/dist/lib/constants.js +10 -10
- package/dist/lib/favicon.js +102 -0
- package/dist/lib/image.js +13 -17
- package/dist/lib/media-helpers.js +2 -2
- package/dist/lib/navigation.js +23 -3
- package/dist/lib/render.js +10 -1
- package/dist/lib/schemas.js +31 -0
- package/dist/lib/timezones.js +388 -0
- package/dist/lib/view.js +1 -1
- package/dist/routes/api/posts.js +1 -1
- package/dist/routes/api/upload.js +3 -3
- package/dist/routes/auth/reset.js +221 -0
- package/dist/routes/auth/setup.js +194 -0
- package/dist/routes/auth/signin.js +176 -0
- package/dist/routes/dash/collections.js +23 -415
- package/dist/routes/dash/media.js +12 -392
- package/dist/routes/dash/pages.js +7 -330
- package/dist/routes/dash/redirects.js +18 -12
- package/dist/routes/dash/settings.js +198 -577
- package/dist/routes/feed/rss.js +2 -1
- package/dist/routes/feed/sitemap.js +4 -2
- package/dist/routes/pages/featured.js +5 -1
- package/dist/routes/pages/home.js +26 -1
- package/dist/routes/pages/latest.js +45 -0
- package/dist/services/post.js +30 -50
- package/dist/types/bindings.js +3 -0
- package/dist/types/config.js +147 -0
- package/dist/types/constants.js +27 -0
- package/dist/types/entities.js +3 -0
- package/dist/types/operations.js +3 -0
- package/dist/types/props.js +3 -0
- package/dist/types/views.js +5 -0
- package/dist/types.js +8 -111
- package/dist/ui/color-themes.js +33 -33
- package/dist/ui/compose/ComposeDialog.js +36 -21
- package/dist/ui/dash/PageForm.js +21 -15
- package/dist/ui/dash/PostForm.js +22 -16
- package/dist/ui/dash/collections/CollectionForm.js +152 -0
- package/dist/ui/dash/collections/CollectionsListContent.js +68 -0
- package/dist/ui/dash/collections/ViewCollectionContent.js +96 -0
- package/dist/ui/dash/media/MediaListContent.js +166 -0
- package/dist/ui/dash/media/ViewMediaContent.js +212 -0
- package/dist/ui/dash/pages/LinkFormContent.js +130 -0
- package/dist/ui/dash/pages/UnifiedPagesContent.js +193 -0
- package/dist/ui/dash/settings/AccountContent.js +209 -0
- package/dist/ui/dash/settings/AppearanceContent.js +259 -0
- package/dist/ui/dash/settings/GeneralContent.js +536 -0
- package/dist/ui/dash/settings/SettingsNav.js +41 -0
- package/dist/ui/font-themes.js +36 -0
- package/dist/ui/layouts/BaseLayout.js +24 -2
- package/dist/ui/layouts/SiteLayout.js +47 -19
- package/package.json +1 -1
- package/src/app.tsx +95 -553
- package/src/auth.ts +4 -1
- package/src/client.ts +1 -0
- package/src/i18n/locales/en.po +240 -175
- package/src/i18n/locales/en.ts +1 -1
- package/src/i18n/locales/zh-Hans.po +240 -175
- package/src/i18n/locales/zh-Hans.ts +1 -1
- package/src/i18n/locales/zh-Hant.po +240 -175
- package/src/i18n/locales/zh-Hant.ts +1 -1
- package/src/lib/__tests__/config.test.ts +192 -0
- package/src/lib/__tests__/favicon.test.ts +151 -0
- package/src/lib/__tests__/image.test.ts +2 -6
- package/src/lib/__tests__/timezones.test.ts +61 -0
- package/src/lib/__tests__/view.test.ts +2 -2
- package/src/lib/avatar-upload.ts +165 -0
- package/src/lib/config.ts +47 -0
- package/src/lib/constants.ts +19 -11
- package/src/lib/favicon.ts +115 -0
- package/src/lib/image.ts +13 -21
- package/src/lib/media-helpers.ts +2 -2
- package/src/lib/navigation.ts +33 -2
- package/src/lib/render.tsx +15 -1
- package/src/lib/schemas.ts +39 -0
- package/src/lib/timezones.ts +325 -0
- package/src/lib/view.ts +1 -1
- package/src/routes/api/posts.ts +1 -1
- package/src/routes/api/upload.ts +2 -3
- package/src/routes/auth/reset.tsx +239 -0
- package/src/routes/auth/setup.tsx +189 -0
- package/src/routes/auth/signin.tsx +163 -0
- package/src/routes/dash/__tests__/settings-avatar.test.ts +89 -0
- package/src/routes/dash/collections.tsx +17 -366
- package/src/routes/dash/media.tsx +12 -414
- package/src/routes/dash/pages.tsx +8 -348
- package/src/routes/dash/redirects.tsx +20 -14
- package/src/routes/dash/settings.tsx +243 -534
- package/src/routes/feed/__tests__/rss.test.ts +141 -0
- package/src/routes/feed/rss.ts +3 -1
- package/src/routes/feed/sitemap.ts +4 -2
- package/src/routes/pages/featured.tsx +7 -1
- package/src/routes/pages/home.tsx +25 -2
- package/src/routes/pages/latest.tsx +59 -0
- package/src/services/post.ts +34 -66
- package/src/styles/components.css +0 -65
- package/src/styles/tokens.css +1 -1
- package/src/styles/ui.css +24 -40
- package/src/types/bindings.ts +30 -0
- package/src/types/config.ts +183 -0
- package/src/types/constants.ts +26 -0
- package/src/types/entities.ts +109 -0
- package/src/types/operations.ts +88 -0
- package/src/types/props.ts +115 -0
- package/src/types/views.ts +172 -0
- package/src/types.ts +8 -644
- package/src/ui/__tests__/font-themes.test.ts +34 -0
- package/src/ui/color-themes.ts +34 -34
- package/src/ui/compose/ComposeDialog.tsx +40 -21
- package/src/ui/dash/PageForm.tsx +25 -19
- package/src/ui/dash/PostForm.tsx +26 -20
- package/src/ui/dash/collections/CollectionForm.tsx +153 -0
- package/src/ui/dash/collections/CollectionsListContent.tsx +85 -0
- package/src/ui/dash/collections/ViewCollectionContent.tsx +92 -0
- package/src/ui/dash/media/MediaListContent.tsx +201 -0
- package/src/ui/dash/media/ViewMediaContent.tsx +208 -0
- package/src/ui/dash/pages/LinkFormContent.tsx +119 -0
- package/src/ui/dash/pages/UnifiedPagesContent.tsx +203 -0
- package/src/ui/dash/settings/AccountContent.tsx +176 -0
- package/src/ui/dash/settings/AppearanceContent.tsx +254 -0
- package/src/ui/dash/settings/GeneralContent.tsx +533 -0
- package/src/ui/dash/settings/SettingsNav.tsx +56 -0
- package/src/ui/font-themes.ts +54 -0
- package/src/ui/layouts/BaseLayout.tsx +17 -0
- 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>}
|