@jant/core 0.6.7 → 0.6.8

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 (99) hide show
  1. package/bin/commands/uploads/cleanup.js +1 -0
  2. package/dist/{app-L1UPUArB.js → app-9P4rVCe2.js} +338 -117
  3. package/dist/app-DaxS_Cz-.js +6 -0
  4. package/dist/client/.vite/manifest.json +3 -3
  5. package/dist/client/_assets/client-C6peCkkD.css +2 -0
  6. package/dist/client/_assets/{client-B0MvB2r0.js → client-CXnEhyyv.js} +2 -2
  7. package/dist/client/_assets/{client-auth-CwwuucF_.js → client-auth-CSItbyU8.js} +357 -355
  8. package/dist/{env-CoSe-1y4.js → env-OHRKGcMj.js} +1 -1
  9. package/dist/{export-DLukCOO3.js → export-Be082J0n.js} +33 -8
  10. package/dist/{github-api-UD4u_7fa.js → github-api-BgSiE71w.js} +1 -1
  11. package/dist/{github-app-DeX6Td1O.js → github-app-BbklkFmU.js} +1 -1
  12. package/dist/{github-sync-BtHY2AST.js → github-sync-D1Cw8mOY.js} +3 -3
  13. package/dist/{github-sync-BeDecPen.js → github-sync-_kPWM4m9.js} +3 -3
  14. package/dist/index.js +5 -5
  15. package/dist/node.js +6 -6
  16. package/dist/{url-XF0GbKGO.js → url-BMYO-Zlt.js} +42 -2
  17. package/package.json +1 -1
  18. package/src/__tests__/bin/uploads-cleanup.test.ts +2 -0
  19. package/src/client/__tests__/compose-bridge.test.ts +105 -0
  20. package/src/client/__tests__/hydrate-partial.test.ts +27 -0
  21. package/src/client/__tests__/note-expand.test.ts +130 -0
  22. package/src/client/archive-nav.js +2 -1
  23. package/src/client/audio-player.ts +7 -3
  24. package/src/client/components/__tests__/compose-format-convert.test.ts +357 -0
  25. package/src/client/components/__tests__/jant-compose-dialog.test.ts +313 -0
  26. package/src/client/components/__tests__/jant-compose-editor.test.ts +2 -0
  27. package/src/client/components/__tests__/jant-compose-fullscreen.test.ts +2 -0
  28. package/src/client/components/compose-format-convert.ts +255 -0
  29. package/src/client/components/compose-types.ts +2 -0
  30. package/src/client/components/jant-compose-dialog.ts +98 -44
  31. package/src/client/components/jant-compose-editor.ts +64 -11
  32. package/src/client/compose-bridge.ts +17 -0
  33. package/src/client/feed-video-player.ts +1 -1
  34. package/src/client/hydrate-partial.ts +25 -0
  35. package/src/client/note-expand.ts +63 -0
  36. package/src/client.ts +1 -0
  37. package/src/i18n/locales/public/en.po +41 -0
  38. package/src/i18n/locales/public/en.ts +1 -1
  39. package/src/i18n/locales/public/zh-Hans.po +41 -0
  40. package/src/i18n/locales/public/zh-Hans.ts +1 -1
  41. package/src/i18n/locales/public/zh-Hant.po +41 -0
  42. package/src/i18n/locales/public/zh-Hant.ts +1 -1
  43. package/src/i18n/locales/settings/en.po +12 -12
  44. package/src/i18n/locales/settings/en.ts +1 -1
  45. package/src/i18n/locales/settings/zh-Hans.po +12 -12
  46. package/src/i18n/locales/settings/zh-Hans.ts +1 -1
  47. package/src/i18n/locales/settings/zh-Hant.po +12 -12
  48. package/src/i18n/locales/settings/zh-Hant.ts +1 -1
  49. package/src/lib/__tests__/markdown-to-tiptap.test.ts +1 -1
  50. package/src/lib/__tests__/markdown.test.ts +1 -1
  51. package/src/lib/__tests__/summary.test.ts +87 -0
  52. package/src/lib/__tests__/timeline.test.ts +48 -1
  53. package/src/lib/__tests__/tiptap-render.test.ts +4 -4
  54. package/src/lib/__tests__/url.test.ts +44 -0
  55. package/src/lib/__tests__/view.test.ts +168 -1
  56. package/src/lib/navigation.ts +1 -0
  57. package/src/lib/resolve-config.ts +2 -2
  58. package/src/lib/summary.ts +42 -3
  59. package/src/lib/tiptap-render.ts +6 -2
  60. package/src/lib/upload.ts +2 -2
  61. package/src/lib/url.ts +41 -0
  62. package/src/lib/view.ts +102 -40
  63. package/src/preset.css +7 -1
  64. package/src/routes/api/internal/__tests__/uploads.test.ts +68 -0
  65. package/src/routes/api/internal/sites.ts +44 -1
  66. package/src/routes/api/public/__tests__/archive.test.ts +66 -0
  67. package/src/routes/api/public/archive.ts +22 -6
  68. package/src/routes/api/telegram.ts +2 -1
  69. package/src/routes/dash/custom-urls.tsx +1 -1
  70. package/src/routes/dash/settings.tsx +8 -5
  71. package/src/routes/pages/__tests__/archive-params.test.ts +135 -0
  72. package/src/routes/pages/archive.tsx +116 -20
  73. package/src/routes/pages/collections.tsx +1 -0
  74. package/src/services/__tests__/media.test.ts +83 -0
  75. package/src/services/__tests__/post.test.ts +81 -0
  76. package/src/services/export-theme/assets/client-site.js +1 -1
  77. package/src/services/export-theme/styles/main.css +49 -15
  78. package/src/services/media.ts +31 -1
  79. package/src/services/post.ts +22 -2
  80. package/src/services/search.ts +4 -4
  81. package/src/services/upload-session.ts +18 -0
  82. package/src/styles/tokens.css +1 -1
  83. package/src/styles/ui.css +163 -34
  84. package/src/types/config.ts +1 -1
  85. package/src/types/props.ts +3 -0
  86. package/src/ui/compose/ComposeDialog.tsx +13 -0
  87. package/src/ui/dash/settings/AccountMenuContent.tsx +0 -39
  88. package/src/ui/dash/settings/SettingsDirectory.tsx +26 -1
  89. package/src/ui/dash/settings/SettingsRootContent.tsx +46 -1
  90. package/src/ui/dash/settings/__tests__/SettingsRootContent.test.tsx +55 -0
  91. package/src/ui/feed/NoteCard.tsx +54 -5
  92. package/src/ui/feed/__tests__/timeline-cards.test.ts +73 -0
  93. package/src/ui/pages/ArchivePage.tsx +89 -6
  94. package/src/ui/pages/CollectionsPage.tsx +7 -1
  95. package/src/ui/pages/__tests__/ArchivePage.test.tsx +37 -0
  96. package/src/ui/shared/CollectionDirectory.tsx +13 -3
  97. package/src/ui/shared/CollectionsManager.tsx +3 -0
  98. package/dist/app-C1QgMNRY.js +0 -6
  99. package/dist/client/_assets/client-BMPMuwvV.css +0 -2
@@ -1,9 +1,9 @@
1
- import { _ as toPublicPath, a as getSitePathPrefix, c as normalizePath, d as sanitizeUrl, f as slugify, g as toPublicHref, h as toAbsoluteSiteUrl, i as getSiteOrigin, m as toAbsoluteAssetUrl, n as extractDisplayDomain, o as isFullUrl, p as stripSitePathPrefix, r as extractDomain, s as isSafeInternalRedirect, t as buildSiteUrl, u as normalizeSiteUrl, y as __exportAll } from "./url-XF0GbKGO.js";
2
- import { A as JANT_POSITIVE_LOGO_PNG_FILENAME, B as getJantLogoHref, C as formatYearMonth, D as HOME_BRANDING_LINK_LABEL, E as toISOString, F as getJantBundledAsset, G as base64ToUint8Array, H as JANT_LOGO_PATH_DATA, I as getJantIconFilename, L as getJantIconHref, M as getDefaultJantAppleTouchIconBytes, N as getDefaultJantFaviconIcoBytes, O as HOME_BRANDING_PREFIX, P as getJantBrandPackHref, R as getJantLogoFilename, S as formatTime, U as JANT_LOGO_VIEW_BOX, V as getJantPositiveLogoPngHref, W as arrayBufferToBase64, _ as getMediaUrl, b as formatRelativeAge, d as extractSummaryHtml, f as renderTiptapDocument, g as getImageUrl, h as escapeHtml, i as tiptapJsonToMarkdown, j as JANT_REPO_URL, k as JANT_BRAND_PACK_FILENAME, l as extractBodyText, m as trimTiptapBody, o as render, p as renderTiptapJson, s as toPlainText, t as createExportService, u as extractSummary, v as getPublicUrlForProvider, w as now, x as formatRelativeTime, y as formatDate, z as getJantLogoFills } from "./export-DLukCOO3.js";
3
- import { S as getTelegramWebhookSecret, T as coalesceDisplayText, _ as getInternalAdminToken, a as getConfiguredSingleSiteUrl, b as getSiteResolutionMode, c as getDevApiToken, d as getHostedControlPlaneBaseUrl, f as getHostedControlPlaneDomainCheckSecret, g as getHostedControlPlaneSsoSecret, h as getHostedControlPlaneProviderLabel$1, i as getConfiguredSingleSitePathPrefix, l as getEnvString, m as getHostedControlPlaneInternalToken, n as getAuthSecret, o as getConfiguredStorageDriver, p as getHostedControlPlaneInternalBaseUrl, r as getConfiguredSingleSiteOrigin, s as getCorsOrigins, u as getGitHubAppConfig, v as getLocalStoragePath, w as shouldUseSecureCookies, x as getTelegramBotPool } from "./env-CoSe-1y4.js";
4
- import { l as markdownToTiptapJson, o as createGitHubSyncService } from "./github-sync-BtHY2AST.js";
5
- import { a as listInstallationReposPage, n as getInstallation, o as searchInstallationRepos, t as buildInstallUrl } from "./github-app-DeX6Td1O.js";
6
- import { r as parseRepoSlug, t as createGitHubClient } from "./github-api-UD4u_7fa.js";
1
+ import { _ as toPublicPath, a as getSitePathPrefix, b as __exportAll, c as normalizePath, d as sanitizeUrl, f as slugify, g as toPublicHref, h as toAbsoluteSiteUrl, i as getSiteOrigin, m as toAbsoluteAssetUrl, n as extractDisplayDomain, o as isFullUrl, p as stripSitePathPrefix, r as extractDomain, s as isSafeInternalRedirect, t as buildSiteUrl, u as normalizeSiteUrl, v as toSameSitePath } from "./url-BMYO-Zlt.js";
2
+ import { A as JANT_POSITIVE_LOGO_PNG_FILENAME, B as getJantLogoHref, C as formatYearMonth, D as HOME_BRANDING_LINK_LABEL, E as toISOString, F as getJantBundledAsset, G as base64ToUint8Array, H as JANT_LOGO_PATH_DATA, I as getJantIconFilename, L as getJantIconHref, M as getDefaultJantAppleTouchIconBytes, N as getDefaultJantFaviconIcoBytes, O as HOME_BRANDING_PREFIX, P as getJantBrandPackHref, R as getJantLogoFilename, S as formatTime, U as JANT_LOGO_VIEW_BOX, V as getJantPositiveLogoPngHref, W as arrayBufferToBase64, _ as getMediaUrl, b as formatRelativeAge, d as extractSummaryHtml, f as renderTiptapDocument, g as getImageUrl, h as escapeHtml, i as tiptapJsonToMarkdown, j as JANT_REPO_URL, k as JANT_BRAND_PACK_FILENAME, l as extractBodyText, m as trimTiptapBody, o as render, p as renderTiptapJson, s as toPlainText, t as createExportService, u as extractSummary, v as getPublicUrlForProvider, w as now, x as formatRelativeTime, y as formatDate, z as getJantLogoFills } from "./export-Be082J0n.js";
3
+ import { S as getTelegramWebhookSecret, T as coalesceDisplayText, _ as getInternalAdminToken, a as getConfiguredSingleSiteUrl, b as getSiteResolutionMode, c as getDevApiToken, d as getHostedControlPlaneBaseUrl, f as getHostedControlPlaneDomainCheckSecret, g as getHostedControlPlaneSsoSecret, h as getHostedControlPlaneProviderLabel$1, i as getConfiguredSingleSitePathPrefix, l as getEnvString, m as getHostedControlPlaneInternalToken, n as getAuthSecret, o as getConfiguredStorageDriver, p as getHostedControlPlaneInternalBaseUrl, r as getConfiguredSingleSiteOrigin, s as getCorsOrigins, u as getGitHubAppConfig, v as getLocalStoragePath, w as shouldUseSecureCookies, x as getTelegramBotPool } from "./env-OHRKGcMj.js";
4
+ import { l as markdownToTiptapJson, o as createGitHubSyncService } from "./github-sync-D1Cw8mOY.js";
5
+ import { a as listInstallationReposPage, n as getInstallation, o as searchInstallationRepos, t as buildInstallUrl } from "./github-app-BbklkFmU.js";
6
+ import { r as parseRepoSlug, t as createGitHubClient } from "./github-api-BgSiE71w.js";
7
7
  import { I18n } from "@lingui/core";
8
8
  import * as lucideIcons from "lucide-static";
9
9
  import { z } from "zod";
@@ -1913,16 +1913,16 @@ var Hono = class extends Hono$1 {
1913
1913
  }
1914
1914
  //#endregion
1915
1915
  //#region src/i18n/locales/public/en.ts
1916
- var messages$3 = JSON.parse("{\"+4u2g6\":[\"A ready-made 1:1 PNG for decks, mockups, directories, and other square placements.\"],\"+DPYOZ\":[\"Add a link to your main RSS feed. Change what /feed returns in General.\"],\"+G8qqW\":[\"Collection saved.\"],\"+IJm1Z\":[\"Muted\"],\"+Irvp3\":[\"Everything on this page is ready to use for articles, launch posts, directories, and product coverage.\"],\"+Qaboy\":[\"Favicon\"],\"+fWu2O\":[\"A calmer, warmer accent makes the default theme feel quieter and more intentional.\"],\"+nHhRH\":[\"Use \",[\"brandColorName\"]],\"+siMqD\":[\"Journal\"],\"/DFKdU\":[\"Type the quote...\"],\"/PfPLc\":[\"Label (optional)\"],\"/PoNoq\":[\"Edit link\"],\"/Ui2OV\":[\"Use the reverse logo on dark backgrounds.\"],\"/Ybds4\":[\"Primary Jant logo for websites, docs, press coverage, and editorial layouts.\"],\"/rTz0M\":[\"Audio\"],\"0EcUWz\":[\"Discard changes?\"],\"0Lj7or\":[\"Save text attachment?\"],\"0XDp7X\":[\"Links should read clearly without glowing against the page.\"],\"0ieXE7\":[\"Highest rated\"],\"11h9eK\":[\"Includes\"],\"15++NM\":[\"Inline emphasis\"],\"1DBGsz\":[\"Notes\"],\"1NeeWI\":[\"Square assets for avatars, apps, browsers, and shared links\"],\"1THMr2\":[\"Brand pack\"],\"1njn7W\":[\"Light\"],\"2B7HLH\":[\"New post\"],\"2C7mSG\":[\"Collection link\"],\"2ETv7R\":[\"Tune color in a real reading context\"],\"2HbvFp\":[\"Real post components\"],\"2MXb5X\":[\"Field notes on quiet design\"],\"2koDOQ\":[\"Thread accents\"],\"2lKpcz\":[\"Write your first post to get started.\"],\"2q/Q7x\":[\"Visibility\"],\"2sCqzD\":[\"Use this for websites, docs, articles, and other light or neutral surfaces.\"],\"33DClx\":[\"Link to your latest posts. If it comes before Featured, the homepage opens here.\"],\"3Cw1AI\":[\"Add Collection\"],\"3lJk5u\":[\"Keep the artwork unchanged.\"],\"3mdteM\":[\"before deciding whether the accent is carrying too much product energy.\"],\"3neqtf\":[\"Thread accent\"],\"3qkggm\":[\"Fullscreen\"],\"3vMdv3\":[\"This link is reserved. Choose something else.\"],\"3wKq0C\":[\"Couldn't save. Try again in a moment.\"],\"3xi01/\":[\"Look at the footer metadata last, to make sure the accent is not fighting the typography.\"],\"47iMgt\":[\"Editorial interfaces worth borrowing from\"],\"4D09NB\":[\"Link to your collections page\"],\"4HLTdq\":[\"without media\"],\"4J/OYU\":[\"Collection created.\"],\"4eiXo+\":[\"Leave blank to generate one automatically.\"],\"4pV0kE\":[\"Avatar-ready\"],\"51EYZX\":[\"without title\"],\"5dcjwM\":[\"Choose today or an earlier date, or leave it blank to publish now.\"],\"5pAjd8\":[\"Accent should feel present, not loud.\"],\"5sEkBi\":[\"Open raw asset\"],\"6UTABI\":[\"Collection order updated.\"],\"6WAK+2\":[\"Use current date\"],\"6Y4BBO\":[\"An abstract editorial layout in warm paper colors\"],\"6cjUDB\":[\"Brand assets\"],\"6lGV3K\":[\"Show less\"],\"6p0JeQ\":[\"to make sure both still feel like they belong to the same product.\"],\"6sVyMq\":[\"Add a URL before posting this link.\"],\"6yCv8j\":[\"Save these changes to the text attachment, discard them, or keep editing.\"],\"74kJNs\":[\"View earlier notes in this thread\"],\"7DvUqV\":[\"Read the page from top to bottom without looking at the swatches.\"],\"7aris6\":[\"March 15\"],\"7d1a0d\":[\"Public\"],\"7hYXO0\":[\"Use this on dark backgrounds, image-backed surfaces, and any placement where the green logo would lose contrast.\"],\"7kMW54\":[\"Open raw SVG\"],\"7nGhhM\":[\"What's on your mind?\"],\"7vhWI8\":[\"New Password\"],\"87a/t/\":[\"Label\"],\"8Btgys\":[\"Draft deleted.\"],\"8WX0J+\":[\"Your thoughts (optional)\"],\"8ZsakT\":[\"Password\"],\"8bpHix\":[\"Couldn't create your account. Check the details and try again.\"],\"8eC78s\":[\"A calmer accent makes\"],\"8tM8+a\":[\"Save as draft\"],\"90IRF2\":[\"This article is here to answer a specific question: does the default accent still feel calm once it has to carry a full reading experience?\"],\"9SHZas\":[\"Shows 'Settings' when logged in, 'Sign in' when logged out\"],\"9aloPG\":[\"References\"],\"9dr9Nh\":[\"Open external link\"],\"9qWoxS\":[\"Feed\"],\"A1D8Yt\":[\"What the accent should do\"],\"A1taO8\":[\"Search\"],\"A2Vg/u\":[\"Navigation and reading states\"],\"AjHkcv\":[\"Default preview image for social shares and link unfurls.\"],\"AyHO4m\":[\"What's this collection about?\"],\"B1FFMj\":[\"Download Brand Pack\"],\"B495Gs\":[\"Archive\"],\"BdjLtf\":[\"thread\"],\"Bmaby2\":[\"All formats\"],\"C+9df9\":[\"Quoted or highlighted passages should feel like annotations, not warnings.\"],\"C0/57J\":[\"This is the last part of the collection link.\"],\"CAh1km\":[\"collections\"],\"CH3bgf\":[\"RSS feed for this view\"],\"CT7H2e\":[\"Link to your featured posts. If it comes before Latest, the homepage opens here.\"],\"CmBCXY\":[\"Link updated.\"],\"D4em/+\":[\"Logos\"],\"DHhJ7s\":[\"Previous\"],\"DJLY+/\":[\" and try again.\"],\"DOx286\":[\"Draft restored.\"],\"DPfwMq\":[\"Done\"],\"DSJXZM\":[\"Enter a valid date.\"],\"DYlMYF\":[\"Built-in background\"],\"DoJzLz\":[\"Collections\"],\"Du2B9f\":[\"The default accent should support reading first. Start by comparing it against the\"],\"DxwUcG\":[\"Read the palette as content first\"],\"E3NcGH\":[\"Square logo PNG\"],\"EEYbdt\":[\"Publish\"],\"EGwzOK\":[\"Complete Setup\"],\"EHWwm1\":[\"The default accent should feel written, not branded.\"],\"EO3I6h\":[\"Upload didn't go through. Try again in a moment.\"],\"EQNPYo\":[\"Featured on \",[\"date\"],\" at \",[\"time\"]],\"EQtz4D\":[\"Open a few links and check whether they still feel native to the page.\"],\"EU3tBD\":[\"Link removed.\"],\"EetoJL\":[\"Guide the eye without taking over the layout.\"],\"Eiv3bO\":[\"Buttons can stay steady, but links, thread markers, and subtle emphasis should feel closer to ink on paper than dashboard chrome.\"],\"ElTnWL\":[\"Published on\"],\"EmQw8O\":[\"If this article still feels like a page you want to keep reading, the palette is probably close.\"],\"EsJdRp\":[\"Save theme\"],\"FESYvt\":[\"Describe this for people with visual impairments...\"],\"FEr96N\":[\"Theme\"],\"FGySZL\":[\"The default accent works best when it reads like a fountain-pen underline. Compare it against the\"],\"FM+KeU\":[\"No drafts yet. Save a draft to find it here.\"],\"Fdv5k7\":[\"What to look for while tuning it\"],\"FkMol5\":[\"Featured\"],\"Fxf4jq\":[\"Description (optional)\"],\"G2u/aQ\":[\"Download official Jant logos, icons, and preview assets.\"],\"GBJzTZ\":[\"Archive\"],\"GX2VMa\":[\"Create your admin account.\"],\"GY/1J4\":[\"Jant fallback canary string\"],\"GbIOhd\":[\"Reply quietly\"],\"GiRWtR\":[\"Why the default accent should feel written, not branded\"],\"GkpIs2\":[\"Remove this link from Collections? The destination won't change.\"],\"GorKul\":[\"Welcome to Jant\"],\"GxkJXS\":[\"Uploading...\"],\"H29JXm\":[\"+ ALT\"],\"H4lgRd\":[\"Authentication isn't set up. Check your server config.\"],\"HFPGej\":[\"No threads match these filters. Try adjusting your selection or clear all filters.\"],\"HG79RB\":[\"Post as Private\"],\"HNEHJP\":[\"Demo credentials are pre-filled — hit Sign In to continue.\"],\"HbAIQc\":[\"A reference link for checking whether the accent feels editorial instead of promotional.\"],\"Ht1V3q\":[\"For the same reason, inline code should stay neutral. Something like theme.siteAccent = soften(green, 12%) should not suddenly become the loudest thing on the page.\"],\"I22eN0\":[\"Shared links\"],\"I6zLrz\":[\"Use these when you need a transparent square logo, a shaped tile with a built-in background, a browser icon, or a default preview image.\"],\"ICsA6P\":[\"You have unsaved changes\"],\"IUX7p+\":[\"White logo on the Jant green rounded tile for app icon mockups, touch icons, directory listings, and other square placements that should feel softer.\"],\"IagCbF\":[\"URL\"],\"IjnQHI\":[\"with title\"],\"ImOQa9\":[\"Reply\"],\"IsI3kE\":[\"Nothing here yet. Add posts to one of these collections to fill this view.\"],\"J+2Rls\":[\"Leave blank to publish now. Use an earlier date when importing older posts.\"],\"J4tAHl\":[\"Headings should keep their hierarchy even when the accent gets softer.\"],\"JYj5R2\":[\"Browse files\"],\"JcD7qf\":[\"More actions\"],\"JqJ5Xv\":[\"Latest\"],\"JuN5GC\":[\"No file selected. Choose a file to upload.\"],\"JwLPQ/\":[\"This sign-in link has expired. Return to \"],\"KOqvXP\":[\"Do not recolor, stretch, rotate, outline, or add effects to the logo.\"],\"KbS2K9\":[\"Reset Password\"],\"KdSsVl\":[\"Author (optional)\"],\"Khu3PV\":[\"Publish settings\"],\"KiJn9B\":[\"Note\"],\"KlZ+t+\":[\"%name% + %count% more\"],\"KsvRin\":[\"Hide from Latest\"],\"KzmC5L\":[\"Controls\"],\"L7svJg\":[\"Reading\"],\"Lbkbwy\":[\"A quote card for judging accent color against softer, citation-heavy content.\"],\"LcvzvX\":[\"Tap to retry\"],\"LkA8jz\":[\"Add alt text\"],\"LxRg6f\":[\"live theme controls\"],\"M4tzVU\":[\"Latest posts\"],\"M8kJqa\":[\"Drafts\"],\"MHrjPM\":[\"Title\"],\"MILa7n\":[\"Square tile\"],\"MSc/Yq\":[\"Do you want to publish your changes or discard them?\"],\"Mc7+6G\":[\"Enter a valid URL starting with http://, https://, or mailto:.\"],\"MdMyne\":[\"Source link (optional)\"],\"MiMY3Q\":[\"Apple touch icon\"],\"MiyoI7\":[\"default note sample\"],\"MqghUt\":[\"Search posts...\"],\"Myqkib\":[\"Create a collection to get started.\"],\"N8UzTV\":[\"Replies\"],\"NAFbuE\":[\"Search snippet\"],\"NH9Z1R\":[\"Start here\"],\"NqsRbb\":[\"Jant logo\"],\"NvXuWk\":[\"Won't move the thread to the top of latest.\"],\"O1367B\":[\"All collections\"],\"O3oNi5\":[\"Email\"],\"OEdMhi\":[\"The best default color is the one you notice only after reading for a while.\"],\"OEt/to\":[\"Guidelines\"],\"OJxdgi\":[\"Keep this link under 200 characters.\"],\"OaoJcz\":[\"Social preview\"],\"OmfDbR\":[\"Site accent\"],\"Ovks1h\":[\"A softer blue feels more like ink than product chrome.\"],\"P/sHNL\":[\"Use this page to judge buttons, links, cards, forms, thread accents, and quiet surfaces before changing a theme globally.\"],\"Q/uoSA\":[\"Quiet here for now.\"],\"Q2mGA7\":[\"Clear filter\"],\"QBqVyM\":[\"Home screen icon for iPhone and iPad shortcuts.\"],\"QebAts\":[\"Link added.\"],\"Qgbxdw\":[\"Designing a calmer default accent for Jant\"],\"Qn9Ao8\":[\"Circle tile\"],\"QyDt3L\":[\"File uploaded.\"],\"R5CMuK\":[\"Jant looks best when the accent feels editorial. Buttons can stay sturdy, but inline emphasis should feel like a pen mark, not a dashboard highlight.\"],\"R8AthW\":[\"Divider\"],\"R9Khdg\":[\"Auto\"],\"RAv3u7\":[\"Compare it against the theme controls\"],\"ROa4Ti\":[\"Interfaces for reading should guide the eye, not keep asking for attention.\"],\"RZOWDv\":[\"Add a custom shortcut to any page or site.\"],\"RdmNnl\":[\"Browser tab\"],\"RfGczC\":[\"Square logo\"],\"Rj01Fz\":[\"Links\"],\"S37om9\":[\"Included assets\"],\"S8NCfs\":[\"Save to drafts to edit and post at a later time.\"],\"SJGVAw\":[\"Feel editorial and slightly quieter.\"],\"SJmfuf\":[\"Site Name\"],\"SaNhJE\":[\"feel deliberate instead of washed out.\"],\"SpTWH3\":[\"Download SVG\"],\"SvRuJt\":[\"Field Notes on Interface Tone\"],\"T/R+Qz\":[\"Primary\"],\"TNZKpI\":[\"Danger\"],\"TvaTxw\":[\"Doesn't appear in Latest. Still appears in collections you add it to.\"],\"UIMXHD\":[\"Remove Divider\"],\"UaZwcz\":[\"More options are available after you create it.\"],\"Uc5y7o\":[\"Choose the standard logo for websites, docs, directories, and editorial layouts.\"],\"V18SVO\":[\"Use the logo on light backgrounds.\"],\"V4WsyL\":[\"Add Link\"],\"VCA6B2\":[\"These are actual feed components with real footers, summaries, and inline links. Use this section to judge whether the theme still feels calm once it is applied to realistic content.\"],\"VNqFYa\":[\"Loading post...\"],\"WCOanD\":[\"This reference is useful because it treats links and citations as part of the reading rhythm. Keep that in mind while tuning the\"],\"WbIbzR\":[\"Checking link...\"],\"WcWS//\":[\"Download file\"],\"WhsN3P\":[\"A good default accent in Jant should feel like editorial structure, not product branding. That means links, emphasis, and thread cues can be visible without turning the page into UI chrome.\"],\"Wn+/rH\":[\"Transparent square\"],\"XU7b+L\":[\"Primary logo files\"],\"XV1mAn\":[\"Only visible when signed in.\"],\"XrnWzN\":[\"Published!\"],\"YIix5Y\":[\"Search...\"],\"YUglt2\":[\"Generating a link...\"],\"YXiA6e\":[\"Primary button\"],\"Ygx3Yl\":[\"Small browser icon used in tabs and bookmarks.\"],\"Z6NwTi\":[\"Save as Draft\"],\"ZGs2so\":[\"Delete this collection permanently? Posts inside won't be removed.\"],\"ZV5ykW\":[\"Download PNG\"],\"ZhhOwV\":[\"Quote\"],\"ZmSeP+\":[\"Save to drafts?\"],\"ZxFuun\":[[\"count\",\"plural\",{\"one\":[\"Found \",\"#\",\" result\"],\"other\":[\"Found \",\"#\",\" results\"]}]],\"a5j82I\":[\"No collections match that search. Try a different name.\"],\"aHTB7P\":[\"Supplementary content attached to your post\"],\"aMEyv0\":[\"Stay sturdy and readable.\"],\"aN6wx0\":[\"Nothing in Featured yet. Mark a post as featured to show it here.\"],\"aYpXKS\":[\"and checking whether the accent is guiding attention or pulling too hard.\"],\"aaGV/9\":[\"New Link\"],\"af+9p6\":[\"Quiet metadata\"],\"an5hVd\":[\"Images\"],\"ao77hr\":[[\"count\",\"plural\",{\"one\":[\"#\",\" hidden post\"],\"other\":[\"#\",\" hidden posts\"]}]],\"auFlOr\":[\"Icons and previews\"],\"avuFKG\":[\"threads\"],\"bFpC86\":[\"Everything in one download\"],\"bGtMpA\":[\"Add a label and URL.\"],\"bHOiy1\":[\"Password changes are off in demo mode. Sign in with the shared demo credentials.\"],\"bbdNeX\":[\"Sign in\"],\"bfCbdi\":[\"Current post\"],\"bkBJmZ\":[\"This is useful as a color check because it puts the accent next to quotation styling, metadata, and a quieter explanatory paragraph. Compare it back to the\"],\"bzSI52\":[\"Discard\"],\"c2JRUS\":[\"Generate automatically\"],\"cIoW7X\":[\"Inline link\"],\"cTUByn\":[\"Newest first\"],\"cb7FR8\":[\"White logo on the Jant green square tile for platforms and layouts that expect a true edge-to-edge square.\"],\"cgmi4V\":[\"Delete Draft\"],\"cnGeoo\":[\"Delete\"],\"d+F4pf\":[\"The image should sit quietly inside the article instead of feeling like a card preview.\"],\"d/o/BH\":[\"Couldn't publish. Saved as draft.\"],\"dD7NPy\":[\"Outline\"],\"dEgA5A\":[\"Cancel\"],\"dUsGbd\":[\"The right accent should disappear into the writing until you need it.\"],\"dXoieq\":[\"Summary\"],\"dYKrp3\":[\"Hidden from Latest\"],\"dbUuAj\":[\"Appears in Latest.\"],\"df4a/r\":[\"Couldn't load this post. Try again.\"],\"ePK91l\":[\"Edit\"],\"eWLklq\":[\"Quotes\"],\"f4MAoA\":[\"Some uploads failed. Saved as draft.\"],\"f5s9EI\":[\"Press N to write\"],\"f6Hub0\":[\"Sort\"],\"f8fH8W\":[\"Design\"],\"fD+f7T\":[\"RSS feed\"],\"fKrDxS\":[\"Brand tile\"],\"fMPkxb\":[\"Show more\"],\"fqDzSu\":[\"Rate\"],\"fttd2R\":[\"My Collection\"],\"gCcxP/\":[\"Threads can include up to \",[\"count\"],\" posts.\"],\"gFdWl+\":[\"A long-form article sample for checking the default palette in a true reading context.\"],\"gNKz6Z\":[\"Collection deleted.\"],\"gXH9r/\":[\"Open raw PNG\"],\"gj52YE\":[\"This collection is empty. Add posts from the editor.\"],\"gpaPhA\":[\"Helps screen readers describe the image\"],\"h5RcXU\":[\"Post hidden\"],\"hLlWo5\":[\"A few simple rules.\"],\"hWpUeY\":[\"Auto link\"],\"hXzOVo\":[\"Next\"],\"heSQoS\":[\"Paste a URL...\"],\"hrkGms\":[\"Search\"],\"i0vDGK\":[\"Sort Order\"],\"i5+Y7d\":[\"Download the official Jant logo, icons, and preview files.\"],\"i6kro6\":[\"Edit custom link\"],\"i6nDCI\":[\"Choose a new password.\"],\"iG7KNr\":[\"Logo\"],\"iH8pgl\":[\"Back\"],\"ilSmIt\":[\"Hard edge\"],\"iu7tUI\":[\"Breadcrumb\"],\"jAXE5p\":[\"Reverse logo\"],\"jAqB/k\":[\"Post privately\"],\"jQflRT\":[\"This uses the real single-post detail rendering with a longer article, inline image, tables, lists, quotes, and code. The content column stays at the same width as the live site.\"],\"jd+8Mm\":[\"Social preview image\"],\"jdJOV1\":[\"Settings\"],\"ji7oVU\":[\"Edit post\"],\"jpctdh\":[\"View\"],\"jrsUoG\":[\"Type / for commands\"],\"jvyYZG\":[\"What's on your mind...\"],\"k3Iw35\":[\"Switch to the white logo when the standard green version would lose contrast.\"],\"kPMIr+\":[\"Give it a title...\"],\"kj6ppi\":[\"entry\"],\"kr39oD\":[\"No collections yet. Start one to organize posts by topic.\"],\"kzvWob\":[\"Link to the post archive\"],\"laT1IJ\":[\"iOS home screen\"],\"lb+Xwx\":[\"Custom link\"],\"m16xKo\":[\"Add\"],\"mKT7g0\":[\"Text attachment\"],\"mc/vLq\":[\"This link is already in use. Choose something else.\"],\"muKqfV\":[\"Featured\"],\"n1ekoW\":[\"Sign In\"],\"n3ReIn\":[\"Collections\"],\"n6QD94\":[\"Oldest first\"],\"nFukaP\":[\"Wrong email or password. Check your credentials and try again.\"],\"nV6twc\":[\"Organize\"],\"nd8Puv\":[\"White logo on the Jant green circle for profile images, badges, and other round placements where you want a ready-made asset.\"],\"ndrEYW\":[\"When the accent is slightly warmer and less literal, the whole page feels more like a writing space and less like product UI.\"],\"o21Y+P\":[\"entries\"],\"oO0hKx\":[[\"count\",\"plural\",{\"one\":[\"#\",\" more post\"],\"other\":[\"#\",\" more posts\"]}]],\"oTu7Wt\":[\"Combined Collections\"],\"ode0+L\":[\"Theme sample\"],\"ogssnn\":[\"with media\"],\"ovBPCi\":[\"Default\"],\"p1Z67P\":[\"When primary is too rigid, the whole page starts reading like product UI instead of writing space.\"],\"p2/GCq\":[\"Confirm Password\"],\"pB0OKE\":[\"New Divider\"],\"pBHx39\":[\"Dark backgrounds\"],\"pVrU5x\":[\"If this page feels too branded, the first place to soften is the default theme’s site accent, not the border or body text.\"],\"pvnfJD\":[\"Dark\"],\"q+hNag\":[\"Collection\"],\"q5YRzz\":[\"Color check\"],\"q8RviX\":[\"Titled\"],\"qcawwg\":[\"Publish now\"],\"qiN9NB\":[\"Surface\"],\"qt89I8\":[\"Draft saved.\"],\"quvfGs\":[\"instead of judging it as an isolated swatch.\"],\"r7kcaA\":[\"Drag collections, links, and dividers into the order you want.\"],\"rA2TFI\":[\"Switch the palette and mode without opening settings or changing the active site theme.\"],\"rV8ZnP\":[\"Edit publish date\"],\"rdUucN\":[\"Preview\"],\"s8G5Or\":[\"This upload would exceed your shared hosted media limit. Remove files or upgrade storage to continue.\"],\"s9gHf5\":[\"your-post-link\"],\"sER+bs\":[\"Files\"],\"sQpDn6\":[\"Exit fullscreen\"],\"sgr2wQ\":[\"collection\"],\"slujBW\":[\"Use lowercase letters, numbers, and hyphens only.\"],\"syiAKf\":[\"note treatment\"],\"t42hIC\":[\"Everything most people need is in one ZIP.\"],\"tCctex\":[\"The brand pack includes SVG logos, a transparent square PNG, rounded, square, and circle tiles, plus favicon, Apple touch icon, and the default social preview image.\"],\"tKlWWY\":[\"Emoji\"],\"tSWVu5\":[\"Published on \",[\"date\"],\" at \",[\"time\"]],\"tfDRzk\":[\"Save\"],\"tg5MRw\":[\"Sign in to start writing.\"],\"tgSBSE\":[\"Remove Link\"],\"uowbPn\":[\"Remove attachment\"],\"v3E8iS\":[\"A practical checklist\"],\"vSJd18\":[\"Video\"],\"vSYKYI\":[\"Main feed\"],\"vXCC6J\":[\"Something doesn't look right. Check the form and try again.\"],\"vcpc5o\":[\"Close menu\"],\"vdFnYM\":[\"Reset link\"],\"vdvpU5\":[\"/archive?format=quote or https://example.com\"],\"vgpfCi\":[\"Save draft\"],\"vpSPA1\":[\"Auth secret is missing. Check your environment variables.\"],\"vzU4k9\":[\"New Collection\"],\"w0Emel\":[\"Suggested link\"],\"w6mlns\":[\"Article detail page\"],\"wJ+GRy\":[\"All visibility\"],\"wL3cK8\":[\"Latest\"],\"wja8aL\":[\"Untitled\"],\"wlnK1t\":[\"A single ZIP with the main logo, reverse logo, square PNG, rounded, square, and circle tiles, plus favicon, Apple touch icon, and social preview image.\"],\"wm3Zlr\":[\"All years\"],\"xCWek4\":[\"File storage isn't set up. Check your server config.\"],\"xVrkxi\":[\"quiet design\"],\"xVvw1i\":[\"This reset link is no longer valid. Request a new one to continue.\"],\"xYilR2\":[\"Media\"],\"xeiujy\":[\"Text\"],\"xhTx3y\":[\"Choose the standard logo for most placements and the reverse logo when you need more contrast.\"],\"y28hnO\":[\"Post\"],\"y2o/Y0\":[\"This Link Has Expired\"],\"yGZVl1\":[\"More\"],\"yQ2kGp\":[\"Load more\"],\"yUtAh2\":[\"New Thread\"],\"ycM1Xg\":[\"No results. Try different keywords.\"],\"ynMAhG\":[\"Default logo\"],\"yzF66j\":[\"Link\"],\"zBFr9G\":[\"Paste a long article, AI response, or any text...\\n\\nMarkdown formatting will be preserved.\"],\"zJDAbh\":[\"Don't save\"],\"zcDmsG\":[\"Featured posts\"],\"zoK+eO\":[\"Add a title before posting this link.\"],\"zucql+\":[\"Menu\"],\"zwBp5t\":[\"Private\"]}");
1916
+ var messages$3 = JSON.parse("{\"+4u2g6\":[\"A ready-made 1:1 PNG for decks, mockups, directories, and other square placements.\"],\"+DPYOZ\":[\"Add a link to your main RSS feed. Change what /feed returns in General.\"],\"+G8qqW\":[\"Collection saved.\"],\"+IJm1Z\":[\"Muted\"],\"+Irvp3\":[\"Everything on this page is ready to use for articles, launch posts, directories, and product coverage.\"],\"+Qaboy\":[\"Favicon\"],\"+fWu2O\":[\"A calmer, warmer accent makes the default theme feel quieter and more intentional.\"],\"+nHhRH\":[\"Use \",[\"brandColorName\"]],\"+siMqD\":[\"Journal\"],\"/DFKdU\":[\"Type the quote...\"],\"/PfPLc\":[\"Label (optional)\"],\"/PoNoq\":[\"Edit link\"],\"/Ui2OV\":[\"Use the reverse logo on dark backgrounds.\"],\"/Ybds4\":[\"Primary Jant logo for websites, docs, press coverage, and editorial layouts.\"],\"/rTz0M\":[\"Audio\"],\"0EcUWz\":[\"Discard changes?\"],\"0Lj7or\":[\"Save text attachment?\"],\"0XDp7X\":[\"Links should read clearly without glowing against the page.\"],\"0ieXE7\":[\"Highest rated\"],\"11h9eK\":[\"Includes\"],\"15++NM\":[\"Inline emphasis\"],\"1DBGsz\":[\"Notes\"],\"1NeeWI\":[\"Square assets for avatars, apps, browsers, and shared links\"],\"1THMr2\":[\"Brand pack\"],\"1njn7W\":[\"Light\"],\"2B7HLH\":[\"New post\"],\"2C7mSG\":[\"Collection link\"],\"2ETv7R\":[\"Tune color in a real reading context\"],\"2HbvFp\":[\"Real post components\"],\"2MXb5X\":[\"Field notes on quiet design\"],\"2koDOQ\":[\"Thread accents\"],\"2lKpcz\":[\"Write your first post to get started.\"],\"2q/Q7x\":[\"Visibility\"],\"2sCqzD\":[\"Use this for websites, docs, articles, and other light or neutral surfaces.\"],\"33DClx\":[\"Link to your latest posts. If it comes before Featured, the homepage opens here.\"],\"3Cw1AI\":[\"Add Collection\"],\"3lJk5u\":[\"Keep the artwork unchanged.\"],\"3mdteM\":[\"before deciding whether the accent is carrying too much product energy.\"],\"3neqtf\":[\"Thread accent\"],\"3qkggm\":[\"Fullscreen\"],\"3vMdv3\":[\"This link is reserved. Choose something else.\"],\"3wKq0C\":[\"Couldn't save. Try again in a moment.\"],\"3xi01/\":[\"Look at the footer metadata last, to make sure the accent is not fighting the typography.\"],\"47iMgt\":[\"Editorial interfaces worth borrowing from\"],\"4D09NB\":[\"Link to your collections page\"],\"4HLTdq\":[\"without media\"],\"4J/OYU\":[\"Collection created.\"],\"4eiXo+\":[\"Leave blank to generate one automatically.\"],\"4pV0kE\":[\"Avatar-ready\"],\"51EYZX\":[\"without title\"],\"5dcjwM\":[\"Choose today or an earlier date, or leave it blank to publish now.\"],\"5pAjd8\":[\"Accent should feel present, not loud.\"],\"5sEkBi\":[\"Open raw asset\"],\"6UTABI\":[\"Collection order updated.\"],\"6WAK+2\":[\"Use current date\"],\"6Y4BBO\":[\"An abstract editorial layout in warm paper colors\"],\"6cjUDB\":[\"Brand assets\"],\"6lGV3K\":[\"Show less\"],\"6p0JeQ\":[\"to make sure both still feel like they belong to the same product.\"],\"6sVyMq\":[\"Add a URL before posting this link.\"],\"6yCv8j\":[\"Save these changes to the text attachment, discard them, or keep editing.\"],\"74kJNs\":[\"View earlier notes in this thread\"],\"7DvUqV\":[\"Read the page from top to bottom without looking at the swatches.\"],\"7aris6\":[\"March 15\"],\"7d1a0d\":[\"Public\"],\"7hYXO0\":[\"Use this on dark backgrounds, image-backed surfaces, and any placement where the green logo would lose contrast.\"],\"7kMW54\":[\"Open raw SVG\"],\"7nGhhM\":[\"What's on your mind?\"],\"7vhWI8\":[\"New Password\"],\"87a/t/\":[\"Label\"],\"8Btgys\":[\"Draft deleted.\"],\"8WX0J+\":[\"Your thoughts (optional)\"],\"8ZsakT\":[\"Password\"],\"8bpHix\":[\"Couldn't create your account. Check the details and try again.\"],\"8eC78s\":[\"A calmer accent makes\"],\"8tM8+a\":[\"Save as draft\"],\"90IRF2\":[\"This article is here to answer a specific question: does the default accent still feel calm once it has to carry a full reading experience?\"],\"9SHZas\":[\"Shows 'Settings' when logged in, 'Sign in' when logged out\"],\"9aloPG\":[\"References\"],\"9dr9Nh\":[\"Open external link\"],\"9qWoxS\":[\"Feed\"],\"A1D8Yt\":[\"What the accent should do\"],\"A1taO8\":[\"Search\"],\"A2Vg/u\":[\"Navigation and reading states\"],\"AjHkcv\":[\"Default preview image for social shares and link unfurls.\"],\"AyHO4m\":[\"What's this collection about?\"],\"B1FFMj\":[\"Download Brand Pack\"],\"B495Gs\":[\"Archive\"],\"BdjLtf\":[\"thread\"],\"Bmaby2\":[\"All formats\"],\"C+9df9\":[\"Quoted or highlighted passages should feel like annotations, not warnings.\"],\"C0/57J\":[\"This is the last part of the collection link.\"],\"C4TjpG\":[\"Read less\"],\"CAh1km\":[\"collections\"],\"CH3bgf\":[\"RSS feed for this view\"],\"CT7H2e\":[\"Link to your featured posts. If it comes before Latest, the homepage opens here.\"],\"CmBCXY\":[\"Link updated.\"],\"D4em/+\":[\"Logos\"],\"DHhJ7s\":[\"Previous\"],\"DJLY+/\":[\" and try again.\"],\"DOx286\":[\"Draft restored.\"],\"DPfwMq\":[\"Done\"],\"DSJXZM\":[\"Enter a valid date.\"],\"DYlMYF\":[\"Built-in background\"],\"DoJzLz\":[\"Collections\"],\"Du2B9f\":[\"The default accent should support reading first. Start by comparing it against the\"],\"DxwUcG\":[\"Read the palette as content first\"],\"E3NcGH\":[\"Square logo PNG\"],\"EEYbdt\":[\"Publish\"],\"EGwzOK\":[\"Complete Setup\"],\"EHWwm1\":[\"The default accent should feel written, not branded.\"],\"EO3I6h\":[\"Upload didn't go through. Try again in a moment.\"],\"EQNPYo\":[\"Featured on \",[\"date\"],\" at \",[\"time\"]],\"EQtz4D\":[\"Open a few links and check whether they still feel native to the page.\"],\"EU3tBD\":[\"Link removed.\"],\"EetoJL\":[\"Guide the eye without taking over the layout.\"],\"Eiv3bO\":[\"Buttons can stay steady, but links, thread markers, and subtle emphasis should feel closer to ink on paper than dashboard chrome.\"],\"ElTnWL\":[\"Published on\"],\"EmQw8O\":[\"If this article still feels like a page you want to keep reading, the palette is probably close.\"],\"EsJdRp\":[\"Save theme\"],\"FESYvt\":[\"Describe this for people with visual impairments...\"],\"FEr96N\":[\"Theme\"],\"FGySZL\":[\"The default accent works best when it reads like a fountain-pen underline. Compare it against the\"],\"FM+KeU\":[\"No drafts yet. Save a draft to find it here.\"],\"Fdv5k7\":[\"What to look for while tuning it\"],\"FkMol5\":[\"Featured\"],\"FqCHF/\":[\"Threads\"],\"Fxf4jq\":[\"Description (optional)\"],\"G2u/aQ\":[\"Download official Jant logos, icons, and preview assets.\"],\"GBJzTZ\":[\"Archive\"],\"GX2VMa\":[\"Create your admin account.\"],\"GY/1J4\":[\"Jant fallback canary string\"],\"GbIOhd\":[\"Reply quietly\"],\"GiRWtR\":[\"Why the default accent should feel written, not branded\"],\"GkpIs2\":[\"Remove this link from Collections? The destination won't change.\"],\"GorKul\":[\"Welcome to Jant\"],\"GxkJXS\":[\"Uploading...\"],\"H29JXm\":[\"+ ALT\"],\"H4lgRd\":[\"Authentication isn't set up. Check your server config.\"],\"HFPGej\":[\"No threads match these filters. Try adjusting your selection or clear all filters.\"],\"HG79RB\":[\"Post as Private\"],\"HNEHJP\":[\"Demo credentials are pre-filled — hit Sign In to continue.\"],\"HbAIQc\":[\"A reference link for checking whether the accent feels editorial instead of promotional.\"],\"HrC0ab\":[\"All posts\"],\"Ht1V3q\":[\"For the same reason, inline code should stay neutral. Something like theme.siteAccent = soften(green, 12%) should not suddenly become the loudest thing on the page.\"],\"I22eN0\":[\"Shared links\"],\"I6zLrz\":[\"Use these when you need a transparent square logo, a shaped tile with a built-in background, a browser icon, or a default preview image.\"],\"ICsA6P\":[\"You have unsaved changes\"],\"IUX7p+\":[\"White logo on the Jant green rounded tile for app icon mockups, touch icons, directory listings, and other square placements that should feel softer.\"],\"IagCbF\":[\"URL\"],\"IjnQHI\":[\"with title\"],\"ImOQa9\":[\"Reply\"],\"IsI3kE\":[\"Nothing here yet. Add posts to one of these collections to fill this view.\"],\"J+2Rls\":[\"Leave blank to publish now. Use an earlier date when importing older posts.\"],\"J4tAHl\":[\"Headings should keep their hierarchy even when the accent gets softer.\"],\"JYj5R2\":[\"Browse files\"],\"JcD7qf\":[\"More actions\"],\"JqJ5Xv\":[\"Latest\"],\"JuN5GC\":[\"No file selected. Choose a file to upload.\"],\"JwLPQ/\":[\"This sign-in link has expired. Return to \"],\"KOqvXP\":[\"Do not recolor, stretch, rotate, outline, or add effects to the logo.\"],\"KbS2K9\":[\"Reset Password\"],\"KdSsVl\":[\"Author (optional)\"],\"Khu3PV\":[\"Publish settings\"],\"KiJn9B\":[\"Note\"],\"KlZ+t+\":[\"%name% + %count% more\"],\"KsvRin\":[\"Hide from Latest\"],\"KzmC5L\":[\"Controls\"],\"L7svJg\":[\"Reading\"],\"Lbkbwy\":[\"A quote card for judging accent color against softer, citation-heavy content.\"],\"LcvzvX\":[\"Tap to retry\"],\"LkA8jz\":[\"Add alt text\"],\"LxRg6f\":[\"live theme controls\"],\"M4tzVU\":[\"Latest posts\"],\"M8kJqa\":[\"Drafts\"],\"MHrjPM\":[\"Title\"],\"MILa7n\":[\"Square tile\"],\"MSc/Yq\":[\"Do you want to publish your changes or discard them?\"],\"Mc7+6G\":[\"Enter a valid URL starting with http://, https://, or mailto:.\"],\"MdMyne\":[\"Source link (optional)\"],\"MiMY3Q\":[\"Apple touch icon\"],\"MiyoI7\":[\"default note sample\"],\"MqghUt\":[\"Search posts...\"],\"Myqkib\":[\"Create a collection to get started.\"],\"N8UzTV\":[\"Replies\"],\"NAFbuE\":[\"Search snippet\"],\"NH9Z1R\":[\"Start here\"],\"NqsRbb\":[\"Jant logo\"],\"NvXuWk\":[\"Won't move the thread to the top of latest.\"],\"O1367B\":[\"All collections\"],\"O3oNi5\":[\"Email\"],\"OEdMhi\":[\"The best default color is the one you notice only after reading for a while.\"],\"OEt/to\":[\"Guidelines\"],\"OJxdgi\":[\"Keep this link under 200 characters.\"],\"OaoJcz\":[\"Social preview\"],\"OmfDbR\":[\"Site accent\"],\"Ovks1h\":[\"A softer blue feels more like ink than product chrome.\"],\"P/sHNL\":[\"Use this page to judge buttons, links, cards, forms, thread accents, and quiet surfaces before changing a theme globally.\"],\"Q/uoSA\":[\"Quiet here for now.\"],\"Q2mGA7\":[\"Clear filter\"],\"QBqVyM\":[\"Home screen icon for iPhone and iPad shortcuts.\"],\"QebAts\":[\"Link added.\"],\"Qgbxdw\":[\"Designing a calmer default accent for Jant\"],\"Qn9Ao8\":[\"Circle tile\"],\"Qoq+GP\":[\"Read more\"],\"QyDt3L\":[\"File uploaded.\"],\"R5CMuK\":[\"Jant looks best when the accent feels editorial. Buttons can stay sturdy, but inline emphasis should feel like a pen mark, not a dashboard highlight.\"],\"R8AthW\":[\"Divider\"],\"R9Khdg\":[\"Auto\"],\"RAv3u7\":[\"Compare it against the theme controls\"],\"ROa4Ti\":[\"Interfaces for reading should guide the eye, not keep asking for attention.\"],\"RZOWDv\":[\"Add a custom shortcut to any page or site.\"],\"RdmNnl\":[\"Browser tab\"],\"RfGczC\":[\"Square logo\"],\"Rj01Fz\":[\"Links\"],\"S37om9\":[\"Included assets\"],\"S8NCfs\":[\"Save to drafts to edit and post at a later time.\"],\"SJGVAw\":[\"Feel editorial and slightly quieter.\"],\"SJmfuf\":[\"Site Name\"],\"SaNhJE\":[\"feel deliberate instead of washed out.\"],\"SpTWH3\":[\"Download SVG\"],\"SvRuJt\":[\"Field Notes on Interface Tone\"],\"T/R+Qz\":[\"Primary\"],\"TNZKpI\":[\"Danger\"],\"TvaTxw\":[\"Doesn't appear in Latest. Still appears in collections you add it to.\"],\"UIMXHD\":[\"Remove Divider\"],\"UaZwcz\":[\"More options are available after you create it.\"],\"Uc5y7o\":[\"Choose the standard logo for websites, docs, directories, and editorial layouts.\"],\"V18SVO\":[\"Use the logo on light backgrounds.\"],\"V4WsyL\":[\"Add Link\"],\"VCA6B2\":[\"These are actual feed components with real footers, summaries, and inline links. Use this section to judge whether the theme still feels calm once it is applied to realistic content.\"],\"VNqFYa\":[\"Loading post...\"],\"WCOanD\":[\"This reference is useful because it treats links and citations as part of the reading rhythm. Keep that in mind while tuning the\"],\"WbIbzR\":[\"Checking link...\"],\"WcWS//\":[\"Download file\"],\"WhsN3P\":[\"A good default accent in Jant should feel like editorial structure, not product branding. That means links, emphasis, and thread cues can be visible without turning the page into UI chrome.\"],\"Wn+/rH\":[\"Transparent square\"],\"XU7b+L\":[\"Primary logo files\"],\"XV1mAn\":[\"Only visible when signed in.\"],\"XrnWzN\":[\"Published!\"],\"YIix5Y\":[\"Search...\"],\"YUglt2\":[\"Generating a link...\"],\"YXiA6e\":[\"Primary button\"],\"Ygx3Yl\":[\"Small browser icon used in tabs and bookmarks.\"],\"Z6NwTi\":[\"Save as Draft\"],\"ZGs2so\":[\"Delete this collection permanently? Posts inside won't be removed.\"],\"ZV5ykW\":[\"Download PNG\"],\"ZhhOwV\":[\"Quote\"],\"ZmSeP+\":[\"Save to drafts?\"],\"ZxFuun\":[[\"count\",\"plural\",{\"one\":[\"Found \",\"#\",\" result\"],\"other\":[\"Found \",\"#\",\" results\"]}]],\"a5j82I\":[\"No collections match that search. Try a different name.\"],\"aHTB7P\":[\"Supplementary content attached to your post\"],\"aMEyv0\":[\"Stay sturdy and readable.\"],\"aN6wx0\":[\"Nothing in Featured yet. Mark a post as featured to show it here.\"],\"aYpXKS\":[\"and checking whether the accent is guiding attention or pulling too hard.\"],\"aaGV/9\":[\"New Link\"],\"af+9p6\":[\"Quiet metadata\"],\"an5hVd\":[\"Images\"],\"ao77hr\":[[\"count\",\"plural\",{\"one\":[\"#\",\" hidden post\"],\"other\":[\"#\",\" hidden posts\"]}]],\"auFlOr\":[\"Icons and previews\"],\"avuFKG\":[\"threads\"],\"bFpC86\":[\"Everything in one download\"],\"bGtMpA\":[\"Add a label and URL.\"],\"bHOiy1\":[\"Password changes are off in demo mode. Sign in with the shared demo credentials.\"],\"bbdNeX\":[\"Sign in\"],\"bfCbdi\":[\"Current post\"],\"bkBJmZ\":[\"This is useful as a color check because it puts the accent next to quotation styling, metadata, and a quieter explanatory paragraph. Compare it back to the\"],\"bzSI52\":[\"Discard\"],\"c2JRUS\":[\"Generate automatically\"],\"cIoW7X\":[\"Inline link\"],\"cTUByn\":[\"Newest first\"],\"cb7FR8\":[\"White logo on the Jant green square tile for platforms and layouts that expect a true edge-to-edge square.\"],\"cgmi4V\":[\"Delete Draft\"],\"cnGeoo\":[\"Delete\"],\"d+F4pf\":[\"The image should sit quietly inside the article instead of feeling like a card preview.\"],\"d/o/BH\":[\"Couldn't publish. Saved as draft.\"],\"dD7NPy\":[\"Outline\"],\"dEgA5A\":[\"Cancel\"],\"dHko2w\":[\"single posts\"],\"dUsGbd\":[\"The right accent should disappear into the writing until you need it.\"],\"dXoieq\":[\"Summary\"],\"dYKrp3\":[\"Hidden from Latest\"],\"dbUuAj\":[\"Appears in Latest.\"],\"df4a/r\":[\"Couldn't load this post. Try again.\"],\"ePK91l\":[\"Edit\"],\"eWLklq\":[\"Quotes\"],\"f4MAoA\":[\"Some uploads failed. Saved as draft.\"],\"f5s9EI\":[\"Press N to write\"],\"f6Hub0\":[\"Sort\"],\"f8fH8W\":[\"Design\"],\"fD+f7T\":[\"RSS feed\"],\"fKrDxS\":[\"Brand tile\"],\"fMPkxb\":[\"Show more\"],\"fqDzSu\":[\"Rate\"],\"fttd2R\":[\"My Collection\"],\"gCcxP/\":[\"Threads can include up to \",[\"count\"],\" posts.\"],\"gFdWl+\":[\"A long-form article sample for checking the default palette in a true reading context.\"],\"gNKz6Z\":[\"Collection deleted.\"],\"gXH9r/\":[\"Open raw PNG\"],\"gj52YE\":[\"This collection is empty. Add posts from the editor.\"],\"gpaPhA\":[\"Helps screen readers describe the image\"],\"h5RcXU\":[\"Post hidden\"],\"hLlWo5\":[\"A few simple rules.\"],\"hWpUeY\":[\"Auto link\"],\"hXzOVo\":[\"Next\"],\"heSQoS\":[\"Paste a URL...\"],\"hqeXKW\":[\"Single posts\"],\"hrkGms\":[\"Search\"],\"i0vDGK\":[\"Sort Order\"],\"i5+Y7d\":[\"Download the official Jant logo, icons, and preview files.\"],\"i6kro6\":[\"Edit custom link\"],\"i6nDCI\":[\"Choose a new password.\"],\"iG7KNr\":[\"Logo\"],\"iH8pgl\":[\"Back\"],\"ilSmIt\":[\"Hard edge\"],\"iu7tUI\":[\"Breadcrumb\"],\"jAXE5p\":[\"Reverse logo\"],\"jAqB/k\":[\"Post privately\"],\"jQflRT\":[\"This uses the real single-post detail rendering with a longer article, inline image, tables, lists, quotes, and code. The content column stays at the same width as the live site.\"],\"jd+8Mm\":[\"Social preview image\"],\"jdJOV1\":[\"Settings\"],\"ji7oVU\":[\"Edit post\"],\"jpctdh\":[\"View\"],\"jrsUoG\":[\"Type / for commands\"],\"jvyYZG\":[\"What's on your mind...\"],\"k3Iw35\":[\"Switch to the white logo when the standard green version would lose contrast.\"],\"kPMIr+\":[\"Give it a title...\"],\"kj6ppi\":[\"entry\"],\"kr39oD\":[\"No collections yet. Start one to organize posts by topic.\"],\"kzvWob\":[\"Link to the post archive\"],\"laT1IJ\":[\"iOS home screen\"],\"lb+Xwx\":[\"Custom link\"],\"m16xKo\":[\"Add\"],\"mKT7g0\":[\"Text attachment\"],\"mc/vLq\":[\"This link is already in use. Choose something else.\"],\"muKqfV\":[\"Featured\"],\"n1ekoW\":[\"Sign In\"],\"n3ReIn\":[\"Collections\"],\"n6QD94\":[\"Oldest first\"],\"nFukaP\":[\"Wrong email or password. Check your credentials and try again.\"],\"nV6twc\":[\"Organize\"],\"nd8Puv\":[\"White logo on the Jant green circle for profile images, badges, and other round placements where you want a ready-made asset.\"],\"ndrEYW\":[\"When the accent is slightly warmer and less literal, the whole page feels more like a writing space and less like product UI.\"],\"o21Y+P\":[\"entries\"],\"oO0hKx\":[[\"count\",\"plural\",{\"one\":[\"#\",\" more post\"],\"other\":[\"#\",\" more posts\"]}]],\"oTu7Wt\":[\"Combined Collections\"],\"ode0+L\":[\"Theme sample\"],\"ogssnn\":[\"with media\"],\"ovBPCi\":[\"Default\"],\"p1Z67P\":[\"When primary is too rigid, the whole page starts reading like product UI instead of writing space.\"],\"p2/GCq\":[\"Confirm Password\"],\"pB0OKE\":[\"New Divider\"],\"pBHx39\":[\"Dark backgrounds\"],\"pVrU5x\":[\"If this page feels too branded, the first place to soften is the default theme’s site accent, not the border or body text.\"],\"pvnfJD\":[\"Dark\"],\"q+hNag\":[\"Collection\"],\"q5YRzz\":[\"Color check\"],\"q8RviX\":[\"Titled\"],\"qcawwg\":[\"Publish now\"],\"qiN9NB\":[\"Surface\"],\"qt89I8\":[\"Draft saved.\"],\"quvfGs\":[\"instead of judging it as an isolated swatch.\"],\"r7kcaA\":[\"Drag collections, links, and dividers into the order you want.\"],\"rA2TFI\":[\"Switch the palette and mode without opening settings or changing the active site theme.\"],\"rV8ZnP\":[\"Edit publish date\"],\"rdUucN\":[\"Preview\"],\"s8G5Or\":[\"This upload would exceed your shared hosted media limit. Remove files or upgrade storage to continue.\"],\"s9gHf5\":[\"your-post-link\"],\"sER+bs\":[\"Files\"],\"sQpDn6\":[\"Exit fullscreen\"],\"sgr2wQ\":[\"collection\"],\"slujBW\":[\"Use lowercase letters, numbers, and hyphens only.\"],\"syiAKf\":[\"note treatment\"],\"t42hIC\":[\"Everything most people need is in one ZIP.\"],\"tCctex\":[\"The brand pack includes SVG logos, a transparent square PNG, rounded, square, and circle tiles, plus favicon, Apple touch icon, and the default social preview image.\"],\"tKlWWY\":[\"Emoji\"],\"tSWVu5\":[\"Published on \",[\"date\"],\" at \",[\"time\"]],\"tfDRzk\":[\"Save\"],\"tg5MRw\":[\"Sign in to start writing.\"],\"tgSBSE\":[\"Remove Link\"],\"uowbPn\":[\"Remove attachment\"],\"v3E8iS\":[\"A practical checklist\"],\"vSJd18\":[\"Video\"],\"vSYKYI\":[\"Main feed\"],\"vXCC6J\":[\"Something doesn't look right. Check the form and try again.\"],\"vcpc5o\":[\"Close menu\"],\"vdFnYM\":[\"Reset link\"],\"vdvpU5\":[\"/archive?format=quote or https://example.com\"],\"vgpfCi\":[\"Save draft\"],\"vpSPA1\":[\"Auth secret is missing. Check your environment variables.\"],\"vzU4k9\":[\"New Collection\"],\"w0Emel\":[\"Suggested link\"],\"w6mlns\":[\"Article detail page\"],\"wJ+GRy\":[\"All visibility\"],\"wL3cK8\":[\"Latest\"],\"wja8aL\":[\"Untitled\"],\"wlnK1t\":[\"A single ZIP with the main logo, reverse logo, square PNG, rounded, square, and circle tiles, plus favicon, Apple touch icon, and social preview image.\"],\"wm3Zlr\":[\"All years\"],\"xCWek4\":[\"File storage isn't set up. Check your server config.\"],\"xVrkxi\":[\"quiet design\"],\"xVvw1i\":[\"This reset link is no longer valid. Request a new one to continue.\"],\"xYilR2\":[\"Media\"],\"xeiujy\":[\"Text\"],\"xhTx3y\":[\"Choose the standard logo for most placements and the reverse logo when you need more contrast.\"],\"xjHB3/\":[\"Continue →\"],\"y28hnO\":[\"Post\"],\"y2o/Y0\":[\"This Link Has Expired\"],\"yGZVl1\":[\"More\"],\"yQ2kGp\":[\"Load more\"],\"yUtAh2\":[\"New Thread\"],\"ycM1Xg\":[\"No results. Try different keywords.\"],\"ynMAhG\":[\"Default logo\"],\"yzF66j\":[\"Link\"],\"zBFr9G\":[\"Paste a long article, AI response, or any text...\\n\\nMarkdown formatting will be preserved.\"],\"zJDAbh\":[\"Don't save\"],\"zcDmsG\":[\"Featured posts\"],\"zoK+eO\":[\"Add a title before posting this link.\"],\"zucql+\":[\"Menu\"],\"zwBp5t\":[\"Private\"]}");
1917
1917
  //#endregion
1918
1918
  //#region src/i18n/locales/settings/en.ts
1919
- var messages$2 = JSON.parse("{\"+4Z6iP\":[\"Create the repository on GitHub first — it can be empty.\"],\"+9JI/F\":[\"Connecting will sync your site onto \",[\"repo\"],\"'s default branch on top of its existing history. Existing files outside Jant's managed paths are kept. This can't be undone.\"],\"+AXdXp\":[\"Label and URL are required\"],\"+K0AvT\":[\"Disconnect\"],\"+zy2Nq\":[\"Type\"],\"/3H2/s\":[\"This hosted site signs in through \",[\"providerLabel\"],\". Manage password and hosted access there.\"],\"/JnyjR\":[\"Toggle built-in navigation items. Their order controls what shows in the header and which feed the homepage opens first.\"],\"/ODeyS\":[\"Sets the content language announced to readers (HTML lang, RSS) and the dashboard language. Any BCP 47 tag is accepted; tags without a dashboard translation fall back to English.\"],\"/zOUxl\":[\"QR code linking to the Telegram bot\"],\"0OGSSc\":[\"Avatar display updated.\"],\"0UzCUX\":[\"Update the password you use to sign in\"],\"0bdA9b\":[\"Open Telegram to connect\"],\"10UtuM\":[\"CJK Font\"],\"14BEca\":[\"Read why\"],\"1F6Mzc\":[\"No navigation items yet. Add links or enable system items below.\"],\"1mbBbL\":[\"Connect manually\"],\"1njn7W\":[\"Light\"],\"2B7t+s\":[\"Sessions and password\"],\"2DoBvq\":[\"Feeds\"],\"2FYpfJ\":[\"More\"],\"2Ithfh\":[\"Message the bot any text and it's published as a note.\"],\"2MXb5X\":[\"Field notes on quiet design\"],\"2PTjMB\":[\"I want to delete \",[\"siteName\"]],\"2cFU6q\":[\"Site Footer\"],\"2oWZo7\":[\"Last commit\"],\"2uuy4H\":[\"Connected via Personal Access Token\"],\"35x8eZ\":[\"Showing \",[\"shown\"],\" of \",[\"total\"]],\"39QGku\":[\"Open the bot and send the binding code, then anything you message it becomes a note.\"],\"3Cw1AI\":[\"Add Collection\"],\"3VrybB\":[\"Redirect\"],\"3Yvsaz\":[\"302 (Temporary)\"],\"3n0zbB\":[\"Session management is off in demo mode. Use the shared demo session instead.\"],\"3sYJi5\":[\"Download a Hugo-compatible archive — host it statically or move to another Jant.\"],\"3wKq0C\":[\"Couldn't save. Try again in a moment.\"],\"49Bsal\":[\"Feed settings updated.\"],\"4Jge8E\":[\"Active Sessions\"],\"4KIa+q\":[\"Export downloaded.\"],\"4cEClj\":[\"Sessions\"],\"4zGJ5E\":[\"Delete Account Permanently\"],\"5QlUIt\":[\"Empty repository. Ready to connect.\"],\"5VQnR3\":[\"Use these when you want a feed URL that never changes.\"],\"5dpcN1\":[\"type to search all\"],\"5f1Wo9\":[\"Connected as \",[\"account\"]],\"69OXZB\":[\"Delete Hosted Site\"],\"6ArdBh\":[\"Uses featured posts for /feed.\"],\"6DjeBT\":[\"Demo sites always stay hidden from search engines.\"],\"6FFB7q\":[\"Uses the latest public posts for /feed.\"],\"6K1Vef\":[\"Delete this blog permanently? This cannot be undone.\"],\"6NpNLc\":[\"This repository has existing content.\"],\"6V3Ea3\":[\"Copied\"],\"71WIgc\":[\"Get a new code\"],\"746NHh\":[\"this blog\"],\"7811AW\":[\"This repository is already backing up this site.\"],\"7FaY4u\":[\"Usage\"],\"7G9YLi\":[\"Allow search engines to index my site\"],\"7GISOt\":[\"Save bot token\"],\"7MZxzw\":[\"Password changed.\"],\"7vhWI8\":[\"New Password\"],\"7z05Pf\":[\"Open the hosted site controls in \",[\"providerLabel\"],\" to cancel billing or permanently delete this site.\"],\"81nFIS\":[\"Passwords don't match. Make sure both fields are identical.\"],\"87a/t/\":[\"Label\"],\"89Upyo\":[\"That theme isn't available. Pick another one.\"],\"8BfEpW\":[\"Hosted account\"],\"8N/Mcp\":[\"Archive filter parameters (e.g. format=note&view=list)\"],\"8T46pB\":[\"Bot token\"],\"8U2Z7f\":[\"New Custom URL\"],\"8ZsakT\":[\"Password\"],\"9+vGLh\":[\"Custom CSS\"],\"9As8Nu\":[\"Create one on GitHub\"],\"9Lsvt5\":[\"Signed in \",[\"date\"]],\"9T7Cwm\":[\"Redirects, vanity paths, and URL control\"],\"9aUyym\":[\"See where you're signed in and revoke old sessions\"],\"A1taO8\":[\"Search\"],\"AeXO77\":[\"Account\"],\"AnY+O9\":[\"Show \\\"Build with Jant\\\" at the bottom of the home page\"],\"ApZDMk\":[\"This is used for your favicon and apple-touch-icon. For best results, upload a square PNG with a solid background at least 512x512 pixels.\"],\"B495Gs\":[\"Archive\"],\"B4ESok\":[\"API reference\"],\"BzEFor\":[\"or\"],\"CDAdlf\":[\"Remove bot\"],\"CTAEes\":[\"Select a repository\"],\"CjZZgz\":[\"This repository already has commits\"],\"D8k2s6\":[\"Connect Telegram\"],\"DCKkhU\":[\"Current Password\"],\"DKKKeF\":[\"Manage password and hosted access in \",[\"providerLabel\"]],\"EO3I6h\":[\"Upload didn't go through. Try again in a moment.\"],\"Enslfm\":[\"Destination\"],\"F7FKwe\":[\"Anything you paste here has full access to your visitors' browsers. Only use code from sources you trust.\"],\"FkMol5\":[\"Featured\"],\"G/1oP+\":[\"Remove the webhook and stop syncing. Your repository content will not be deleted.\"],\"G0qJsQ\":[\"Security token missing. Refresh the page and try again.\"],\"G39wnK\":[\"Back up and sync content with a GitHub repository\"],\"GMMWcy\":[\"Name, metadata, language, and search defaults\"],\"GXsAby\":[\"Revoke\"],\"GxkJXS\":[\"Uploading...\"],\"GzKzUa\":[\"Demo limits\"],\"HKH+W+\":[\"Data\"],\"Hp1l6f\":[\"Current\"],\"HxlY7t\":[\"Changing this updates what subscribers get from /feed.\"],\"HxuOlm\":[\"Site Header\"],\"I6gXOa\":[\"Path\"],\"ID38tA\":[\"Account deletion is off in demo mode. The shared demo resets separately.\"],\"IF9tPu\":[\"When to use site export, database backups, and recovery drills.\"],\"IW5PBo\":[\"Copy Token\"],\"IagCbF\":[\"URL\"],\"IreQBq\":[\"Repository\"],\"J6bLeg\":[\"Add a custom link to any URL\"],\"JL7LF5\":[\"available CSS variables, data attributes, and examples.\"],\"JTviaO\":[\"Manage sign-in security, exports, and irreversible actions.\"],\"JcD7qf\":[\"More actions\"],\"JjX0OO\":[\"Copy your token now — it won't be shown again.\"],\"JrFTcr\":[\"Connecting…\"],\"JuN5GC\":[\"No file selected. Choose a file to upload.\"],\"KDw4GX\":[\"Try again\"],\"KSgo21\":[\"Pick a repository\"],\"KVVYBh\":[\"Add collection to navigation\"],\"KiJn9B\":[\"Note\"],\"L3DEwT\":[\"Remove this avatar? Your favicon and header icon will go back to the default.\"],\"L4t4/q\":[\"March 14\"],\"LdyooL\":[\"link\"],\"M/D8PK\":[\"+ Install on another account\"],\"M/haSd\":[\"Always show the light version of the theme.\"],\"M2kIWU\":[\"Font theme\"],\"M6CbAU\":[\"Toggle edit panel\"],\"MaYYE6\":[\"Post notes by messaging a Telegram bot\"],\"Me5t5H\":[\"Connect a GitHub repository to automatically back up your posts as Markdown files. Edits on GitHub sync back to your site.\"],\"Mr4QPw\":[\"Disconnect Telegram? You can reconnect any time with a new binding code.\"],\"MtENL9\":[\"Tune how your site looks, reads, and runs.\"],\"N/8NPV\":[\"Before deleting, download a site export. You won't be able to recover this account after deletion.\"],\"N7UNHY\":[\"Featured feed\"],\"NHnUHF\":[\"Favicon and the profile mark in your header\"],\"NU2Fqi\":[\"Save CSS\"],\"Nldjdr\":[\"No custom URLs yet. Create one to add redirects or custom paths for posts.\"],\"O7rgs6\":[\"Header RSS points to your \",[\"feed\"],\" feed (/feed). Change what /feed returns in General.\"],\"OSJXFg\":[\"Applies to your entire site, including admin pages. Pick a palette, then choose whether it follows the system or stays fixed.\"],\"Ox3+3h\":[\"No matches.\"],\"PEUV5I\":[\"Code injection updated.\"],\"PXj9lw\":[\"Stop accepting posts from Telegram. Your existing notes stay published.\"],\"PZ7HJ8\":[\"Blog Avatar\"],\"Pwqkdw\":[\"Loading…\"],\"PxJ9W6\":[\"Generate Token\"],\"Q/6Y+2\":[\"Needs Contents (read/write) and Webhooks (read/write) on the target repository.\"],\"Q30z/l\":[\"Remove this collection from navigation? The collection itself won't be deleted.\"],\"Q99OtV\":[\"Pin a collection to your navigation bar. An asterisk (*) appears next to collections updated in the last 48 hours.\"],\"QZmz0H\":[\"Built-in links\"],\"Qnrzvb\":[\"Active Tokens\"],\"R6Z4LE\":[\"Download failed. Please try again.\"],\"R9Khdg\":[\"Auto\"],\"RcdDOS\":[\"Create a bot by messaging @BotFather on Telegram, then paste the token it gives you.\"],\"RxsRD6\":[\"Time Zone\"],\"SJmfuf\":[\"Site Name\"],\"SKZhW9\":[\"Token name\"],\"SVQQPe\":[\"Couldn't connect. Check the error and try again.\"],\"SchpMp\":[\"Telegram\"],\"TpF3v+\":[\"Injected before </head>. Use for analytics, custom meta tags, and styles that must load early.\"],\"Tz0i8g\":[\"Settings\"],\"UFK415\":[\"Site-wide HTML for analytics and widgets\"],\"UTvFQq\":[\"Open \",[\"linkOpen\"],\"@\",[\"botUsername\"],[\"linkClose\"],\" and send:\"],\"Uj/btJ\":[\"Display avatar in my site header\"],\"UsODUn\":[\"Select an account\"],\"UxKoFf\":[\"Navigation\"],\"V+bhUy\":[\"Install GitHub App\"],\"V4WsyL\":[\"Add Link\"],\"V5pZwT\":[\"Search settings updated.\"],\"VXUPla\":[\"Connect with GitHub App\"],\"VhMDMg\":[\"Change Password\"],\"Vn3jYy\":[\"Navigation items\"],\"VoZYGU\":[\"This will permanently delete all your data — posts, media, collections, settings, and your account. Your blog will be reset to its initial setup state. This cannot be undone.\"],\"Weq9zb\":[\"General\"],\"Wi9i06\":[\"Follow each visitor's system preference.\"],\"Wx1M8N\":[\"Install the GitHub App to grant access without managing personal tokens. Permissions are scoped per repository and revocable from GitHub.\"],\"X+8FMk\":[\"Current password doesn't match. Try again.\"],\"X1G9eY\":[\"Navigation Preview\"],\"X9Hujr\":[\"Manual Push\"],\"XtBJV8\":[\"Checking repository…\"],\"Xtc16w\":[\"Refresh repository list\"],\"Y/F35r\":[\"Create a post with curl:\"],\"YF6zHf\":[\"Site settings updated.\"],\"YdG2RF\":[\"Export Site\"],\"YkgZi7\":[\"Connect a Telegram bot, then anything you message it gets published as a note.\"],\"YwhjRx\":[\"Manage Account\"],\"ZDY7Fy\":[\"Syncing…\"],\"ZQKLI1\":[\"Danger Zone\"],\"ZS/CBL\":[\"Delete this navigation link? Visitors won't see it in your site header anymore.\"],\"ZhhOwV\":[\"Quote\"],\"ZiooJI\":[\"API Tokens\"],\"Zm7Qb0\":[\"Backup & Restore Guide\"],\"ZmUkwN\":[\"Add custom link to navigation\"],\"a14mj8\":[\"Unknown device\"],\"a3LDKx\":[\"Security\"],\"aAIQg2\":[\"Appearance\"],\"aFkzVF\":[\"The slug of the target post or collection\"],\"alKG0+\":[\"Font Theme\"],\"anibOb\":[\"About this blog\"],\"any7NR\":[\"Theming guide\"],\"b+/jO6\":[\"301 (Permanent)\"],\"bHOiy1\":[\"Password changes are off in demo mode. Sign in with the shared demo credentials.\"],\"bHYIks\":[\"Sign Out\"],\"bmrL08\":[\"Demo mode hides sessions, password changes, and account deletion. Export still works.\"],\"c3MN2z\":[\"all available endpoints and request formats.\"],\"cS7/bk\":[\"Remove the saved bot token? Its webhook is deleted and any connected account is disconnected.\"],\"cSDy01\":[\"Custom CSS updated.\"],\"clzoNp\":[\"Always show the dark version of the theme.\"],\"cnGeoo\":[\"Delete\"],\"d3FRkY\":[\"Could not copy. Try again.\"],\"d5oGUo\":[\"Create a new repository on GitHub\"],\"dEgA5A\":[\"Cancel\"],\"dTXUY+\":[\"Confirm account deletion\"],\"dYKrp3\":[\"Hidden from Latest\"],\"dk7TCH\":[\"Permanently delete all data and reset the blog\"],\"drodVV\":[\"No collections yet. Create one first, then add it to your navigation.\"],\"dsWkIw\":[\"Disconnect from GitHub? The webhook will be removed. Your repository content will not be deleted.\"],\"e/tSI5\":[\"Navigation order updated.\"],\"ePK91l\":[\"Edit\"],\"ebQKK7\":[\"Site\"],\"egK+Yy\":[\"Bearer tokens for scripts and automation\"],\"ehj/zN\":[\"Redirect Type\"],\"eneWvv\":[\"Draft\"],\"erTMh7\":[\"Last synced\"],\"f+m8jj\":[\"Feed URL copied.\"],\"f8fH8W\":[\"Design\"],\"fWYqkz\":[\"Code Injection\"],\"gOWiTY\":[\"Load a serif font optimized for Chinese, Japanese, or Korean content.\"],\"gZ5owP\":[\"Search repositories\"],\"gbqbh6\":[\"Safe to leave this page — syncing continues in the background.\"],\"gkFvVN\":[\"Injected before </body>. Use for chat widgets and scripts that should not block page load.\"],\"gtQsRO\":[\"Create Custom URL\"],\"hBO/y4\":[\"Security token expired. Refresh the page and try again.\"],\"hGmyDl\":[\"Tokens let you access the API from scripts, shortcuts, and other tools without signing in.\"],\"hIHkRy\":[\"Connected via GitHub App\"],\"hdSi1b\":[\"Type \",[\"repo\"],\" to confirm\"],\"he3ygx\":[\"Copy\"],\"i0qMbr\":[\"Home\"],\"iEUzMn\":[\"system\"],\"iSLIjg\":[\"Connect\"],\"iVOMRi\":[\"Home settings updated.\"],\"icB4Cv\":[\"Drag links here to show them under the More menu\"],\"id3vuh\":[\"Telegram is set up, but the bot couldn't be reached. Check the bot token and try again.\"],\"ihn4zD\":[\"Search…\"],\"iiDXZc\":[\"Displayed at the bottom of all posts and pages.\"],\"j4VrG6\":[\"Download Export ZIP\"],\"j5nQL2\":[\"e.g. iOS Shortcuts\"],\"jUV7CU\":[\"Upload Avatar\"],\"jVUmOK\":[\"Markdown supported\"],\"jgBjXJ\":[\"Revoke this token? Any scripts using it will stop working.\"],\"jpctdh\":[\"View\"],\"k1ifdL\":[\"Processing...\"],\"kMXclu\":[\"Download a site export\"],\"kNiQp6\":[\"Pinned\"],\"kRhzWq\":[\"GitHub Sync\"],\"kVQs7s\":[\"Fine-grained styling overrides\"],\"ke1gWS\":[\"Custom URLs\"],\"kfcRb0\":[\"Avatar\"],\"kxDZ2i\":[\"This code runs on every page of your site.\"],\"l2Op2p\":[\"Query Parameters\"],\"lLW3vJ\":[\"Target Slug\"],\"lYHJih\":[\"Revoke this session? That device will need to sign in again.\"],\"mLOk1i\":[\"Push all posts to GitHub right now instead of waiting for the next automatic sync.\"],\"mSNmrX\":[\"List posts:\"],\"nK07ni\":[\"Choose a typographic direction for your site. Each theme changes both the font pairing and the reading rhythm.\"],\"nbfdhU\":[\"Integrations\"],\"o/vNDE\":[\"lets you override any theme variable.\"],\"oGC9uP\":[\"owner/repo\"],\"oH2JHg\":[\"We'll prefill the name \",[\"name\"],\". The list refreshes on return.\"],\"oKOOsY\":[\"Color Theme\"],\"oL535e\":[\"Not synced yet\"],\"oNA4If\":[\"All collections are already in your navigation.\"],\"pZq3aX\":[\"Upload failed. Please try again.\"],\"pgTIrt\":[\"Choose the GitHub account and repository to sync with this site.\"],\"psoxDF\":[\"That font theme isn't available. Pick another one.\"],\"pvnfJD\":[\"Dark\"],\"q+hNag\":[\"Collection\"],\"qdcESc\":[\"Create a new repository\"],\"r5EW6f\":[\"This repository is already backing up another Jant site (\",[\"host\"],\"). Pick a different repository.\"],\"rEspiY\":[\"Navigation placement updated.\"],\"rFmBG3\":[\"Color theme\"],\"rlonmB\":[\"Couldn't delete. Try again in a moment.\"],\"satWc6\":[\"Main RSS feed\"],\"sgr2wQ\":[\"collection\"],\"sqxcaY\":[\"Created \",[\"date\"]],\"sxkWRg\":[\"Advanced\"],\"t/YqKh\":[\"Remove\"],\"t3hvHq\":[\"Sync Now\"],\"tJ4H0O\":[\"your Telegram account\"],\"tfDRzk\":[\"Save\"],\"tvgAq5\":[\"No accounts authorized yet\"],\"u1VTd3\":[\"Palette, surface tone, and overall mood\"],\"u3wRF+\":[\"Published\"],\"u6KOjV\":[\"Want more control?\"],\"udPwLB\":[\"Header\"],\"ui6aMF\":[\"These devices are currently signed in to your account. Revoke any session you don't recognize.\"],\"vBEKwo\":[\"Manage this site's active sessions here. Password and hosted access are managed through \",[\"providerLabel\"],\".\"],\"vRldcl\":[\"Typography choices and reading texture\"],\"vSYKYI\":[\"Main feed\"],\"vTuib7\":[\"This controls what /feed returns.\"],\"vXIe7J\":[\"Language\"],\"vmQmHx\":[\"Add custom CSS to override any styles. Use data attributes like [data-page], [data-post], [data-format] to target specific elements.\"],\"vzX5FB\":[\"Delete Account\"],\"w8Rv8T\":[\"Label is required\"],\"wL3cK8\":[\"Latest\"],\"wPmHHc\":[\"Quiet surfaces let writing lead.\"],\"wW6NCp\":[\"Last error\"],\"wc+17X\":[\"/* Your custom CSS here */\"],\"wuLtXn\":[\"No active sessions right now. Signed-in devices show up here.\"],\"xCWek4\":[\"File storage isn't set up. Check your server config.\"],\"xHt036\":[\"Personal Access Token\"],\"xbN8dp\":[\"Soft color should still carry a clear reading rhythm.\"],\"y28hnO\":[\"Post\"],\"y8Md/V\":[\"Language and time updated.\"],\"yNCqOt\":[\"Latest feed\"],\"yQ3kNF\":[\"Type the following phrase to confirm:\"],\"ydq1k2\":[\"Pick an account first\"],\"yjjCV8\":[\"Fixed feed URLs\"],\"yjkELF\":[\"Confirm New Password\"],\"yzF66j\":[\"Link\"],\"z6wakA\":[\"A short intro shown on your home page.\"],\"zEizrk\":[\"Last used \",[\"date\"]],\"zSURJW\":[\"No repositories match.\"],\"zXH2jX\":[\"Language & Time\"],\"zlcDd2\":[\"Delete this custom URL? Visitors using it won't be redirected anymore.\"],\"zwBp5t\":[\"Private\"],\"zxRN6H\":[\"Header links, home feed, and overflow menu\"]}");
1919
+ var messages$2 = JSON.parse("{\"+4Z6iP\":[\"Create the repository on GitHub first — it can be empty.\"],\"+9JI/F\":[\"Connecting will sync your site onto \",[\"repo\"],\"'s default branch on top of its existing history. Existing files outside Jant's managed paths are kept. This can't be undone.\"],\"+AXdXp\":[\"Label and URL are required\"],\"+K0AvT\":[\"Disconnect\"],\"+zy2Nq\":[\"Type\"],\"/3H2/s\":[\"This hosted site signs in through \",[\"providerLabel\"],\". Manage password and hosted access there.\"],\"/JnyjR\":[\"Toggle built-in navigation items. Their order controls what shows in the header and which feed the homepage opens first.\"],\"/ODeyS\":[\"Sets the content language announced to readers (HTML lang, RSS) and the dashboard language. Any BCP 47 tag is accepted; tags without a dashboard translation fall back to English.\"],\"/zOUxl\":[\"QR code linking to the Telegram bot\"],\"0OGSSc\":[\"Avatar display updated.\"],\"0UzCUX\":[\"Update the password you use to sign in\"],\"0bdA9b\":[\"Open Telegram to connect\"],\"10UtuM\":[\"CJK Font\"],\"14BEca\":[\"Read why\"],\"1F6Mzc\":[\"No navigation items yet. Add links or enable system items below.\"],\"1mbBbL\":[\"Connect manually\"],\"1njn7W\":[\"Light\"],\"2B7t+s\":[\"Sessions and password\"],\"2DoBvq\":[\"Feeds\"],\"2FYpfJ\":[\"More\"],\"2Ithfh\":[\"Message the bot any text and it's published as a note.\"],\"2MXb5X\":[\"Field notes on quiet design\"],\"2PTjMB\":[\"I want to delete \",[\"siteName\"]],\"2cFU6q\":[\"Site Footer\"],\"2oWZo7\":[\"Last commit\"],\"2uuy4H\":[\"Connected via Personal Access Token\"],\"35x8eZ\":[\"Showing \",[\"shown\"],\" of \",[\"total\"]],\"39QGku\":[\"Open the bot and send the binding code, then anything you message it becomes a note.\"],\"3Cw1AI\":[\"Add Collection\"],\"3VrybB\":[\"Redirect\"],\"3Yvsaz\":[\"302 (Temporary)\"],\"3n0zbB\":[\"Session management is off in demo mode. Use the shared demo session instead.\"],\"3sYJi5\":[\"Download a Hugo-compatible archive — host it statically or move to another Jant.\"],\"3wKq0C\":[\"Couldn't save. Try again in a moment.\"],\"49Bsal\":[\"Feed settings updated.\"],\"4Jge8E\":[\"Active Sessions\"],\"4KIa+q\":[\"Export downloaded.\"],\"4cEClj\":[\"Sessions\"],\"4zGJ5E\":[\"Delete Account Permanently\"],\"5QlUIt\":[\"Empty repository. Ready to connect.\"],\"5VQnR3\":[\"Use these when you want a feed URL that never changes.\"],\"5dpcN1\":[\"type to search all\"],\"5f1Wo9\":[\"Connected as \",[\"account\"]],\"6ArdBh\":[\"Uses featured posts for /feed.\"],\"6DjeBT\":[\"Demo sites always stay hidden from search engines.\"],\"6E3aK4\":[\"Manage Hosting\"],\"6FFB7q\":[\"Uses the latest public posts for /feed.\"],\"6K1Vef\":[\"Delete this blog permanently? This cannot be undone.\"],\"6NpNLc\":[\"This repository has existing content.\"],\"6V3Ea3\":[\"Copied\"],\"71WIgc\":[\"Get a new code\"],\"746NHh\":[\"this blog\"],\"7811AW\":[\"This repository is already backing up this site.\"],\"7FaY4u\":[\"Usage\"],\"7G9YLi\":[\"Allow search engines to index my site\"],\"7GISOt\":[\"Save bot token\"],\"7MZxzw\":[\"Password changed.\"],\"7vhWI8\":[\"New Password\"],\"81nFIS\":[\"Passwords don't match. Make sure both fields are identical.\"],\"87a/t/\":[\"Label\"],\"89Upyo\":[\"That theme isn't available. Pick another one.\"],\"8BfEpW\":[\"Hosted account\"],\"8N/Mcp\":[\"Archive filter parameters (e.g. format=note&view=list)\"],\"8T46pB\":[\"Bot token\"],\"8U2Z7f\":[\"New Custom URL\"],\"8ZsakT\":[\"Password\"],\"9+vGLh\":[\"Custom CSS\"],\"9As8Nu\":[\"Create one on GitHub\"],\"9Lsvt5\":[\"Signed in \",[\"date\"]],\"9T7Cwm\":[\"Redirects, vanity paths, and URL control\"],\"9aUyym\":[\"See where you're signed in and revoke old sessions\"],\"A1taO8\":[\"Search\"],\"AeXO77\":[\"Account\"],\"AnY+O9\":[\"Show \\\"Build with Jant\\\" at the bottom of the home page\"],\"ApZDMk\":[\"This is used for your favicon and apple-touch-icon. For best results, upload a square PNG with a solid background at least 512x512 pixels.\"],\"B495Gs\":[\"Archive\"],\"B4ESok\":[\"API reference\"],\"BzEFor\":[\"or\"],\"CDAdlf\":[\"Remove bot\"],\"CTAEes\":[\"Select a repository\"],\"CjZZgz\":[\"This repository already has commits\"],\"D8k2s6\":[\"Connect Telegram\"],\"DCKkhU\":[\"Current Password\"],\"DKKKeF\":[\"Manage password and hosted access in \",[\"providerLabel\"]],\"EO3I6h\":[\"Upload didn't go through. Try again in a moment.\"],\"Enslfm\":[\"Destination\"],\"F7FKwe\":[\"Anything you paste here has full access to your visitors' browsers. Only use code from sources you trust.\"],\"FkMol5\":[\"Featured\"],\"G/1oP+\":[\"Remove the webhook and stop syncing. Your repository content will not be deleted.\"],\"G0qJsQ\":[\"Security token missing. Refresh the page and try again.\"],\"G39wnK\":[\"Back up and sync content with a GitHub repository\"],\"GMMWcy\":[\"Name, metadata, language, and search defaults\"],\"GXsAby\":[\"Revoke\"],\"GxkJXS\":[\"Uploading...\"],\"GzKzUa\":[\"Demo limits\"],\"HKH+W+\":[\"Data\"],\"Hp1l6f\":[\"Current\"],\"HxlY7t\":[\"Changing this updates what subscribers get from /feed.\"],\"HxuOlm\":[\"Site Header\"],\"I6gXOa\":[\"Path\"],\"ID38tA\":[\"Account deletion is off in demo mode. The shared demo resets separately.\"],\"IF9tPu\":[\"When to use site export, database backups, and recovery drills.\"],\"IW5PBo\":[\"Copy Token\"],\"IagCbF\":[\"URL\"],\"IreQBq\":[\"Repository\"],\"J6bLeg\":[\"Add a custom link to any URL\"],\"JL7LF5\":[\"available CSS variables, data attributes, and examples.\"],\"JTviaO\":[\"Manage sign-in security, exports, and irreversible actions.\"],\"JcD7qf\":[\"More actions\"],\"JjX0OO\":[\"Copy your token now — it won't be shown again.\"],\"JrFTcr\":[\"Connecting…\"],\"JuN5GC\":[\"No file selected. Choose a file to upload.\"],\"KDw4GX\":[\"Try again\"],\"KSgo21\":[\"Pick a repository\"],\"KVVYBh\":[\"Add collection to navigation\"],\"KiJn9B\":[\"Note\"],\"L3DEwT\":[\"Remove this avatar? Your favicon and header icon will go back to the default.\"],\"L4t4/q\":[\"March 14\"],\"LdyooL\":[\"link\"],\"M/D8PK\":[\"+ Install on another account\"],\"M/haSd\":[\"Always show the light version of the theme.\"],\"M2kIWU\":[\"Font theme\"],\"M6CbAU\":[\"Toggle edit panel\"],\"MaYYE6\":[\"Post notes by messaging a Telegram bot\"],\"Me5t5H\":[\"Connect a GitHub repository to automatically back up your posts as Markdown files. Edits on GitHub sync back to your site.\"],\"Mr4QPw\":[\"Disconnect Telegram? You can reconnect any time with a new binding code.\"],\"MtENL9\":[\"Tune how your site looks, reads, and runs.\"],\"N/8NPV\":[\"Before deleting, download a site export. You won't be able to recover this account after deletion.\"],\"N7UNHY\":[\"Featured feed\"],\"NHnUHF\":[\"Favicon and the profile mark in your header\"],\"NU2Fqi\":[\"Save CSS\"],\"Nldjdr\":[\"No custom URLs yet. Create one to add redirects or custom paths for posts.\"],\"O7rgs6\":[\"Header RSS points to your \",[\"feed\"],\" feed (/feed). Change what /feed returns in General.\"],\"OSJXFg\":[\"Applies to your entire site, including admin pages. Pick a palette, then choose whether it follows the system or stays fixed.\"],\"Ox3+3h\":[\"No matches.\"],\"PEUV5I\":[\"Code injection updated.\"],\"PXj9lw\":[\"Stop accepting posts from Telegram. Your existing notes stay published.\"],\"PZ7HJ8\":[\"Blog Avatar\"],\"Pwqkdw\":[\"Loading…\"],\"PxJ9W6\":[\"Generate Token\"],\"Q/6Y+2\":[\"Needs Contents (read/write) and Webhooks (read/write) on the target repository.\"],\"Q30z/l\":[\"Remove this collection from navigation? The collection itself won't be deleted.\"],\"Q99OtV\":[\"Pin a collection to your navigation bar. An asterisk (*) appears next to collections updated in the last 48 hours.\"],\"QZmz0H\":[\"Built-in links\"],\"Qnrzvb\":[\"Active Tokens\"],\"R6Z4LE\":[\"Download failed. Please try again.\"],\"R9Khdg\":[\"Auto\"],\"RcdDOS\":[\"Create a bot by messaging @BotFather on Telegram, then paste the token it gives you.\"],\"RxsRD6\":[\"Time Zone\"],\"SJmfuf\":[\"Site Name\"],\"SKZhW9\":[\"Token name\"],\"SVQQPe\":[\"Couldn't connect. Check the error and try again.\"],\"SchpMp\":[\"Telegram\"],\"TpF3v+\":[\"Injected before </head>. Use for analytics, custom meta tags, and styles that must load early.\"],\"Tz0i8g\":[\"Settings\"],\"UFK415\":[\"Site-wide HTML for analytics and widgets\"],\"UTvFQq\":[\"Open \",[\"linkOpen\"],\"@\",[\"botUsername\"],[\"linkClose\"],\" and send:\"],\"Uj/btJ\":[\"Display avatar in my site header\"],\"UsODUn\":[\"Select an account\"],\"UxKoFf\":[\"Navigation\"],\"V+bhUy\":[\"Install GitHub App\"],\"V4WsyL\":[\"Add Link\"],\"V5pZwT\":[\"Search settings updated.\"],\"VXUPla\":[\"Connect with GitHub App\"],\"VhMDMg\":[\"Change Password\"],\"Vn3jYy\":[\"Navigation items\"],\"VoZYGU\":[\"This will permanently delete all your data — posts, media, collections, settings, and your account. Your blog will be reset to its initial setup state. This cannot be undone.\"],\"Weq9zb\":[\"General\"],\"Wi9i06\":[\"Follow each visitor's system preference.\"],\"Wx1M8N\":[\"Install the GitHub App to grant access without managing personal tokens. Permissions are scoped per repository and revocable from GitHub.\"],\"X+8FMk\":[\"Current password doesn't match. Try again.\"],\"X1G9eY\":[\"Navigation Preview\"],\"X9Hujr\":[\"Manual Push\"],\"XtBJV8\":[\"Checking repository…\"],\"Xtc16w\":[\"Refresh repository list\"],\"Y/F35r\":[\"Create a post with curl:\"],\"YF6zHf\":[\"Site settings updated.\"],\"YdG2RF\":[\"Export Site\"],\"YkgZi7\":[\"Connect a Telegram bot, then anything you message it gets published as a note.\"],\"YwhjRx\":[\"Manage Account\"],\"ZDY7Fy\":[\"Syncing…\"],\"ZQKLI1\":[\"Danger Zone\"],\"ZS/CBL\":[\"Delete this navigation link? Visitors won't see it in your site header anymore.\"],\"ZhhOwV\":[\"Quote\"],\"ZiooJI\":[\"API Tokens\"],\"Zm7Qb0\":[\"Backup & Restore Guide\"],\"ZmUkwN\":[\"Add custom link to navigation\"],\"a14mj8\":[\"Unknown device\"],\"a3LDKx\":[\"Security\"],\"aAIQg2\":[\"Appearance\"],\"aFkzVF\":[\"The slug of the target post or collection\"],\"alKG0+\":[\"Font Theme\"],\"anibOb\":[\"About this blog\"],\"any7NR\":[\"Theming guide\"],\"b+/jO6\":[\"301 (Permanent)\"],\"bHOiy1\":[\"Password changes are off in demo mode. Sign in with the shared demo credentials.\"],\"bHYIks\":[\"Sign Out\"],\"bmrL08\":[\"Demo mode hides sessions, password changes, and account deletion. Export still works.\"],\"c3MN2z\":[\"all available endpoints and request formats.\"],\"cS7/bk\":[\"Remove the saved bot token? Its webhook is deleted and any connected account is disconnected.\"],\"cSDy01\":[\"Custom CSS updated.\"],\"clzoNp\":[\"Always show the dark version of the theme.\"],\"cnGeoo\":[\"Delete\"],\"d3FRkY\":[\"Could not copy. Try again.\"],\"d5oGUo\":[\"Create a new repository on GitHub\"],\"dEgA5A\":[\"Cancel\"],\"dTXUY+\":[\"Confirm account deletion\"],\"dYKrp3\":[\"Hidden from Latest\"],\"dk7TCH\":[\"Permanently delete all data and reset the blog\"],\"drodVV\":[\"No collections yet. Create one first, then add it to your navigation.\"],\"dsWkIw\":[\"Disconnect from GitHub? The webhook will be removed. Your repository content will not be deleted.\"],\"e/tSI5\":[\"Navigation order updated.\"],\"ePK91l\":[\"Edit\"],\"ebQKK7\":[\"Site\"],\"egK+Yy\":[\"Bearer tokens for scripts and automation\"],\"ehj/zN\":[\"Redirect Type\"],\"eneWvv\":[\"Draft\"],\"erTMh7\":[\"Last synced\"],\"f+m8jj\":[\"Feed URL copied.\"],\"f8fH8W\":[\"Design\"],\"fWYqkz\":[\"Code Injection\"],\"gOWiTY\":[\"Load a serif font optimized for Chinese, Japanese, or Korean content.\"],\"gZ5owP\":[\"Search repositories\"],\"gbqbh6\":[\"Safe to leave this page — syncing continues in the background.\"],\"gkFvVN\":[\"Injected before </body>. Use for chat widgets and scripts that should not block page load.\"],\"gtQsRO\":[\"Create Custom URL\"],\"hBO/y4\":[\"Security token expired. Refresh the page and try again.\"],\"hGmyDl\":[\"Tokens let you access the API from scripts, shortcuts, and other tools without signing in.\"],\"hIHkRy\":[\"Connected via GitHub App\"],\"hdSi1b\":[\"Type \",[\"repo\"],\" to confirm\"],\"he3ygx\":[\"Copy\"],\"i0qMbr\":[\"Home\"],\"iEUzMn\":[\"system\"],\"iSLIjg\":[\"Connect\"],\"iVOMRi\":[\"Home settings updated.\"],\"icB4Cv\":[\"Drag links here to show them under the More menu\"],\"id3vuh\":[\"Telegram is set up, but the bot couldn't be reached. Check the bot token and try again.\"],\"ihn4zD\":[\"Search…\"],\"iiDXZc\":[\"Displayed at the bottom of all posts and pages.\"],\"j4VrG6\":[\"Download Export ZIP\"],\"j5nQL2\":[\"e.g. iOS Shortcuts\"],\"jUV7CU\":[\"Upload Avatar\"],\"jVUmOK\":[\"Markdown supported\"],\"jgBjXJ\":[\"Revoke this token? Any scripts using it will stop working.\"],\"jpctdh\":[\"View\"],\"k1ifdL\":[\"Processing...\"],\"kMXclu\":[\"Download a site export\"],\"kNiQp6\":[\"Pinned\"],\"kRhzWq\":[\"GitHub Sync\"],\"kVQs7s\":[\"Fine-grained styling overrides\"],\"ke1gWS\":[\"Custom URLs\"],\"kfcRb0\":[\"Avatar\"],\"kxDZ2i\":[\"This code runs on every page of your site.\"],\"l2Op2p\":[\"Query Parameters\"],\"lLW3vJ\":[\"Target Slug\"],\"lYHJih\":[\"Revoke this session? That device will need to sign in again.\"],\"mLOk1i\":[\"Push all posts to GitHub right now instead of waiting for the next automatic sync.\"],\"mSNmrX\":[\"List posts:\"],\"nK07ni\":[\"Choose a typographic direction for your site. Each theme changes both the font pairing and the reading rhythm.\"],\"nbfdhU\":[\"Integrations\"],\"ntJYyh\":[\"Domains, plan, and billing in \",[\"providerLabel\"]],\"o/vNDE\":[\"lets you override any theme variable.\"],\"oGC9uP\":[\"owner/repo\"],\"oH2JHg\":[\"We'll prefill the name \",[\"name\"],\". The list refreshes on return.\"],\"oKOOsY\":[\"Color Theme\"],\"oL535e\":[\"Not synced yet\"],\"oNA4If\":[\"All collections are already in your navigation.\"],\"pZq3aX\":[\"Upload failed. Please try again.\"],\"pgTIrt\":[\"Choose the GitHub account and repository to sync with this site.\"],\"psoxDF\":[\"That font theme isn't available. Pick another one.\"],\"pvnfJD\":[\"Dark\"],\"q+hNag\":[\"Collection\"],\"qdcESc\":[\"Create a new repository\"],\"r5EW6f\":[\"This repository is already backing up another Jant site (\",[\"host\"],\"). Pick a different repository.\"],\"rEspiY\":[\"Navigation placement updated.\"],\"rFmBG3\":[\"Color theme\"],\"rlonmB\":[\"Couldn't delete. Try again in a moment.\"],\"satWc6\":[\"Main RSS feed\"],\"sgr2wQ\":[\"collection\"],\"sqxcaY\":[\"Created \",[\"date\"]],\"sxkWRg\":[\"Advanced\"],\"t/YqKh\":[\"Remove\"],\"t3hvHq\":[\"Sync Now\"],\"tJ4H0O\":[\"your Telegram account\"],\"tfDRzk\":[\"Save\"],\"tvgAq5\":[\"No accounts authorized yet\"],\"u1VTd3\":[\"Palette, surface tone, and overall mood\"],\"u3wRF+\":[\"Published\"],\"u6KOjV\":[\"Want more control?\"],\"udPwLB\":[\"Header\"],\"ui6aMF\":[\"These devices are currently signed in to your account. Revoke any session you don't recognize.\"],\"vBEKwo\":[\"Manage this site's active sessions here. Password and hosted access are managed through \",[\"providerLabel\"],\".\"],\"vRldcl\":[\"Typography choices and reading texture\"],\"vSYKYI\":[\"Main feed\"],\"vTuib7\":[\"This controls what /feed returns.\"],\"vXIe7J\":[\"Language\"],\"vmQmHx\":[\"Add custom CSS to override any styles. Use data attributes like [data-page], [data-post], [data-format] to target specific elements.\"],\"vzX5FB\":[\"Delete Account\"],\"w8Rv8T\":[\"Label is required\"],\"wL3cK8\":[\"Latest\"],\"wPmHHc\":[\"Quiet surfaces let writing lead.\"],\"wW6NCp\":[\"Last error\"],\"wc+17X\":[\"/* Your custom CSS here */\"],\"wuLtXn\":[\"No active sessions right now. Signed-in devices show up here.\"],\"xCWek4\":[\"File storage isn't set up. Check your server config.\"],\"xHt036\":[\"Personal Access Token\"],\"xbN8dp\":[\"Soft color should still carry a clear reading rhythm.\"],\"y28hnO\":[\"Post\"],\"y8Md/V\":[\"Language and time updated.\"],\"yNCqOt\":[\"Latest feed\"],\"yQ3kNF\":[\"Type the following phrase to confirm:\"],\"ydq1k2\":[\"Pick an account first\"],\"yjjCV8\":[\"Fixed feed URLs\"],\"yjkELF\":[\"Confirm New Password\"],\"yzF66j\":[\"Link\"],\"z6wakA\":[\"A short intro shown on your home page.\"],\"zEizrk\":[\"Last used \",[\"date\"]],\"zSURJW\":[\"No repositories match.\"],\"zXH2jX\":[\"Language & Time\"],\"zlcDd2\":[\"Delete this custom URL? Visitors using it won't be redirected anymore.\"],\"zwBp5t\":[\"Private\"],\"zxRN6H\":[\"Header links, home feed, and overflow menu\"]}");
1920
1920
  //#endregion
1921
1921
  //#region src/i18n/locales/settings/zh-Hans.ts
1922
- var messages$1 = JSON.parse("{\"+4Z6iP\":[\"请先在 GitHub 上创建仓库 — 可以为空。\"],\"+9JI/F\":[\"连接后会将你的网站同步到 \",[\"repo\"],\" 的默认分支,并追加到其现有历史之上。Jant 管理路径之外的现有文件会被保留。此操作无法撤销。\"],\"+AXdXp\":[\"标签和 URL 为必填项\"],\"+K0AvT\":[\"断开连接\"],\"+zy2Nq\":[\"类型\"],\"/3H2/s\":[\"此托管站点通过 \",[\"providerLabel\"],\" 登录。请在那里管理密码和托管访问权限。\"],\"/JnyjR\":[\"切换内置导航项。它们的顺序决定页眉显示的内容以及首页显示哪个视图。\"],\"/ODeyS\":[\"声明给读者的内容语言(HTML lang、RSS),同时驱动后台界面的语言。可填任意 BCP 47 标签;没有对应翻译时后台会回退为英文。\"],\"/zOUxl\":[\"链接到 Telegram 机器人的二维码\"],\"0OGSSc\":[\"头像显示已更新.\"],\"0UzCUX\":[\"更新您用于登录的密码\"],\"0bdA9b\":[\"打开 Telegram 以连接\"],\"10UtuM\":[\"CJK 字体\"],\"14BEca\":[\"了解原因\"],\"1F6Mzc\":[\"当前还没有导航项目。添加链接或在下方启用系统项目。\"],\"1mbBbL\":[\"手动绑定\"],\"1njn7W\":[\"浅色\"],\"2B7t+s\":[\"会话与密码\"],\"2DoBvq\":[\"订阅源\"],\"2FYpfJ\":[\"更多\"],\"2Ithfh\":[\"向机器人发送任意文本,它会作为笔记已发布。\"],\"2MXb5X\":[\"关于静谧设计的笔记\"],\"2PTjMB\":[\"我想删除 \",[\"siteName\"]],\"2cFU6q\":[\"网站页脚\"],\"2oWZo7\":[\"最近一次提交\"],\"2uuy4H\":[\"通过个人访问令牌连接\"],\"35x8eZ\":[\"显示 \",[\"shown\"],\" 共 \",[\"total\"]],\"39QGku\":[\"打开机器人并发送绑定码, 然后你发送给它的任何消息都会变成一条笔记。\"],\"3Cw1AI\":[\"添加合集\"],\"3VrybB\":[\"重定向\"],\"3Yvsaz\":[\"302 (临时)\"],\"3n0zbB\":[\"会话管理在演示模式下已关闭。请使用共享的演示会话。\"],\"3sYJi5\":[\"下载一个与 Hugo 兼容的归档 — 将其静态托管或迁移到另一个 Jant。\"],\"3wKq0C\":[\"保存失败。请稍后再试。\"],\"49Bsal\":[\"订阅源设置已更新。\"],\"4Jge8E\":[\"活动会话\"],\"4KIa+q\":[\"导出文件已下载.\"],\"4cEClj\":[\"会话\"],\"4zGJ5E\":[\"永久删除账号\"],\"5QlUIt\":[\"仓库为空。准备连接。\"],\"5VQnR3\":[\"当你想要一个永远不变的订阅源 URL 时使用它们。\"],\"5dpcN1\":[\"输入以搜索全部\"],\"5f1Wo9\":[\"已连接为 \",[\"account\"]],\"69OXZB\":[\"删除托管站点\"],\"6ArdBh\":[\"将精选文章用于 /feed。\"],\"6DjeBT\":[\"演示站点始终对搜索引擎隐藏。\"],\"6FFB7q\":[\"为 /feed 使用最新的公开帖子。\"],\"6K1Vef\":[\"永久删除此博客?此操作不可撤销。\"],\"6NpNLc\":[\"此仓库已有内容。\"],\"6V3Ea3\":[\"已复制\"],\"71WIgc\":[\"获取新绑定码\"],\"746NHh\":[\"此博客\"],\"7811AW\":[\"此仓库已在备份此站点。\"],\"7FaY4u\":[\"用法\"],\"7G9YLi\":[\"允许搜索引擎索引我的网站\"],\"7GISOt\":[\"保存机器人令牌\"],\"7MZxzw\":[\"密码已更改.\"],\"7vhWI8\":[\"新密码\"],\"7z05Pf\":[\"在 \",[\"providerLabel\"],\" 打开托管站点控制以取消计费或永久删除此站点。\"],\"81nFIS\":[\"密码不匹配。请确保两个字段相同。\"],\"87a/t/\":[\"标签\"],\"89Upyo\":[\"该主题不可用。请选择其他主题。\"],\"8BfEpW\":[\"托管账户\"],\"8N/Mcp\":[\"归档 筛选参数 (例如 format=笔记&view=list)\"],\"8T46pB\":[\"机器人令牌\"],\"8U2Z7f\":[\"新建自定义 URL\"],\"8ZsakT\":[\"密码\"],\"9+vGLh\":[\"自定义 CSS\"],\"9As8Nu\":[\"在 GitHub 上创建一个\"],\"9Lsvt5\":[\"于 \",[\"date\"],\" 登录\"],\"9T7Cwm\":[\"重定向、个性化路径和 URL 控制\"],\"9aUyym\":[\"查看你的登录位置并撤销旧会话\"],\"A1taO8\":[\"搜索\"],\"AeXO77\":[\"账户\"],\"AnY+O9\":[\"在主页底部显示 \\\"Build with Jant\\\"\"],\"ApZDMk\":[\"此图用于您的 favicon 和 apple-touch-icon。为获得最佳效果,请上传至少 512×512 像素、纯色背景的方形 PNG。\"],\"B495Gs\":[\"归档\"],\"B4ESok\":[\"API 参考\"],\"BzEFor\":[\"或\"],\"CDAdlf\":[\"移除机器人\"],\"CTAEes\":[\"选择仓库\"],\"CjZZgz\":[\"该仓库已有提交\"],\"D8k2s6\":[\"连接 Telegram\"],\"DCKkhU\":[\"当前密码\"],\"DKKKeF\":[\"在 \",[\"providerLabel\"],\" 管理密码和托管访问\"],\"EO3I6h\":[\"上传未成功。请稍后再试。\"],\"Enslfm\":[\"目标地址\"],\"F7FKwe\":[\"你在此处粘贴的任何内容都将完全访问访客的浏览器。仅使用来自你信任的来源的代码。\"],\"FkMol5\":[\"精选\"],\"G/1oP+\":[\"移除 webhook 并停止同步。您的仓库内容不会被删除。\"],\"G0qJsQ\":[\"缺少安全令牌。刷新页面后重试。\"],\"G39wnK\":[\"将内容备份并与 GitHub 仓库同步\"],\"GMMWcy\":[\"名称、元数据、语言和搜索默认设置\"],\"GXsAby\":[\"撤销\"],\"GxkJXS\":[\"正在上传...\"],\"GzKzUa\":[\"演示限制\"],\"HKH+W+\":[\"数据\"],\"Hp1l6f\":[\"当前\"],\"HxlY7t\":[\"更改此项会更新订阅者从 /feed 获取的内容。\"],\"HxuOlm\":[\"网站头部\"],\"I6gXOa\":[\"路径\"],\"ID38tA\":[\"演示模式下已禁用账号删除。共享演示会单独重置。\"],\"IF9tPu\":[\"何时使用站点导出、数据库备份和恢复演练。\"],\"IW5PBo\":[\"复制令牌\"],\"IagCbF\":[\"网址\"],\"IreQBq\":[\"仓库\"],\"J6bLeg\":[\"向任意 URL 添加自定义链接\"],\"JL7LF5\":[\"可用的 CSS 变量, 数据属性, 和 示例.\"],\"JTviaO\":[\"管理登录安全、导出和不可逆操作.\"],\"JcD7qf\":[\"更多操作\"],\"JjX0OO\":[\"现在复制您的令牌 — 它不会再次显示。\"],\"JrFTcr\":[\"连接中…\"],\"JuN5GC\":[\"未选择文件。请选择要上传的文件。\"],\"KDw4GX\":[\"重试\"],\"KSgo21\":[\"选择仓库\"],\"KVVYBh\":[\"向导航添加合集\"],\"KiJn9B\":[\"笔记\"],\"L3DEwT\":[\"移除此头像?您的网站图标和页眉图标将恢复为默认设置。\"],\"L4t4/q\":[\"3月14日\"],\"LdyooL\":[\"链接\"],\"M/D8PK\":[\"+ 在其他帐户上安装\"],\"M/haSd\":[\"始终显示浅色主题。\"],\"M2kIWU\":[\"字体主题\"],\"M6CbAU\":[\"切换编辑面板\"],\"MaYYE6\":[\"通过向 Telegram 机器人发送消息来发布笔记\"],\"Me5t5H\":[\"将 GitHub 仓库连接以自动将您的文章备份为 Markdown 文件。您在 GitHub 上的编辑会同步回您的网站。\"],\"Mr4QPw\":[\"断开 Telegram?您可以随时使用新的绑定码重新连接。\"],\"MtENL9\":[\"调整你的网站的外观、阅读体验和运行方式。\"],\"N/8NPV\":[\"在删除之前,请下载站点导出。删除后将无法恢复此账户。\"],\"N7UNHY\":[\"精选订阅源\"],\"NHnUHF\":[\"Favicon 和页眉中的个人标识\"],\"NU2Fqi\":[\"保存 CSS\"],\"Nldjdr\":[\"尚无自定义 URL。创建一个以便为文章添加重定向或自定义路径。\"],\"O7rgs6\":[\"页眉 RSS 指向你的 \",[\"feed\"],\" 源 (/feed)。在 常规 中更改 /feed 返回的内容。\"],\"OSJXFg\":[\"应用于整个站点,包括管理页面。选择一个调色板,然后选择它是随系统变化还是保持固定。\"],\"Ox3+3h\":[\"无匹配结果。\"],\"PEUV5I\":[\"代码注入已更新。\"],\"PXj9lw\":[\"停止接受来自 Telegram 的帖子。您的现有笔记将保持已发布。\"],\"PZ7HJ8\":[\"博客头像\"],\"Pwqkdw\":[\"正在加载…\"],\"PxJ9W6\":[\"生成令牌\"],\"Q/6Y+2\":[\"需要对目标仓库的 Contents(读/写)和 Webhooks(读/写)。\"],\"Q30z/l\":[\"要从导航中移除此合集吗?合集本身不会被删除。\"],\"Q99OtV\":[\"将合集固定到导航栏。在过去 48 小时内更新的合集旁会出现一个 * 号。\"],\"QZmz0H\":[\"内置链接\"],\"Qnrzvb\":[\"活动令牌\"],\"R6Z4LE\":[\"下载失败。请重试。\"],\"R9Khdg\":[\"自动\"],\"RcdDOS\":[\"在 Telegram 上给 @BotFather 发送消息创建一个机器人,然后粘贴它提供的令牌。\"],\"RxsRD6\":[\"时区\"],\"SJmfuf\":[\"站点名称\"],\"SKZhW9\":[\"令牌名称\"],\"SVQQPe\":[\"无法连接。检查错误并重试。\"],\"SchpMp\":[\"Telegram\"],\"TpF3v+\":[\"注入到 </head> 之前。用于分析、自定义元标签以及必须尽早加载的样式。\"],\"Tz0i8g\":[\"设置\"],\"UFK415\":[\"用于分析和小部件的全站 HTML\"],\"UTvFQq\":[\"打开 \",[\"linkOpen\"],\"@\",[\"botUsername\"],[\"linkClose\"],\" 并发送:\"],\"Uj/btJ\":[\"在我的站点页眉显示头像\"],\"UsODUn\":[\"选择账户\"],\"UxKoFf\":[\"导航\"],\"V+bhUy\":[\"安装 GitHub App\"],\"V4WsyL\":[\"添加链接\"],\"V5pZwT\":[\"搜索设置已更新。\"],\"VXUPla\":[\"使用 GitHub 应用连接\"],\"VhMDMg\":[\"更改密码\"],\"Vn3jYy\":[\"导航项\"],\"VoZYGU\":[\"这将永久删除您所有的数据 — 帖子、媒体、合集、设置和您的账户。您的博客将重置为初始设置状态。此操作不可撤销。\"],\"Weq9zb\":[\"常规\"],\"Wi9i06\":[\"遵循每位访客的系统偏好。\"],\"Wx1M8N\":[\"安装 GitHub App 以在无需管理个人令牌的情况下授予访问权限。权限按仓库范围授予,可在 GitHub 上撤销。\"],\"X+8FMk\":[\"当前密码不正确。请重试。\"],\"X1G9eY\":[\"导航预览\"],\"X9Hujr\":[\"手动推送\"],\"XtBJV8\":[\"正在检查仓库…\"],\"Xtc16w\":[\"刷新仓库列表\"],\"Y/F35r\":[\"使用 curl 创建帖子:\"],\"YF6zHf\":[\"站点设置已更新.\"],\"YdG2RF\":[\"导出站点\"],\"YkgZi7\":[\"连接一个 Telegram 机器人,然后你发送给它的任何消息都会作为笔记已发布。\"],\"YwhjRx\":[\"管理账户\"],\"ZDY7Fy\":[\"正在同步…\"],\"ZQKLI1\":[\"危险操作\"],\"ZS/CBL\":[\"删除此导航链接? 访客将不再在您网站的页眉中看到它。\"],\"ZhhOwV\":[\"引用 (Jant 的帖子格式之一。)\"],\"ZiooJI\":[\"API 令牌\"],\"Zm7Qb0\":[\"备份与恢复指南\"],\"ZmUkwN\":[\"向导航添加自定义链接\"],\"a14mj8\":[\"未知设备\"],\"a3LDKx\":[\"安全\"],\"aAIQg2\":[\"外观\"],\"aFkzVF\":[\"目标文章或合集的 slug\"],\"alKG0+\":[\"字体主题\"],\"anibOb\":[\"关于本博客\"],\"any7NR\":[\"主题指南\"],\"b+/jO6\":[\"301 (永久)\"],\"bHOiy1\":[\"演示模式下已禁用密码更改。请使用共享的演示凭证登录。\"],\"bHYIks\":[\"退出登录\"],\"bmrL08\":[\"演示模式会隐藏会话、密码更改和账号删除。导出仍然可用.\"],\"c3MN2z\":[\"所有可用的端点和请求格式。\"],\"cS7/bk\":[\"删除保存的机器人令牌?其 webhook 会被删除,任何已连接的账号将被断开连接。\"],\"cSDy01\":[\"自定义 CSS 已更新.\"],\"clzoNp\":[\"始终显示暗色主题.\"],\"cnGeoo\":[\"删除\"],\"d3FRkY\":[\"无法复制。请再试一次。\"],\"d5oGUo\":[\"在 GitHub 上创建新仓库\"],\"dEgA5A\":[\"取消\"],\"dTXUY+\":[\"确认删除账户\"],\"dYKrp3\":[\"从最新中隐藏 (不出现在 Latest 中,但仍可通过固定链接和集合访问的状态。)\"],\"dk7TCH\":[\"永久删除所有数据并重置博客\"],\"drodVV\":[\"还没有合集。请先创建一个,然后将其添加到你的导航中。\"],\"dsWkIw\":[\"要与 GitHub 断开连接吗?该 webhook 将被移除。您的仓库内容不会被删除。\"],\"e/tSI5\":[\"导航顺序已更新。\"],\"ePK91l\":[\"编辑\"],\"ebQKK7\":[\"站点\"],\"egK+Yy\":[\"用于脚本和自动化的 Bearer 令牌\"],\"ehj/zN\":[\"重定向类型\"],\"eneWvv\":[\"草稿\"],\"erTMh7\":[\"上次同步\"],\"f+m8jj\":[\"订阅源 URL 已复制。\"],\"f8fH8W\":[\"设计\"],\"fWYqkz\":[\"代码注入\"],\"gOWiTY\":[\"加载针对中文、日文或韩文内容优化的衬线字体。\"],\"gZ5owP\":[\"搜索仓库\"],\"gbqbh6\":[\"放心离开此页面 — 同步会在后台继续进行。\"],\"gkFvVN\":[\"注入到 </body> 之前。用于聊天小部件和不应阻塞页面加载的脚本。\"],\"gtQsRO\":[\"创建自定义 URL\"],\"hBO/y4\":[\"安全令牌已过期。刷新页面后重试。\"],\"hGmyDl\":[\"令牌让您无需登录即可从脚本、快捷方式和其他工具访问 API。\"],\"hIHkRy\":[\"已通过 GitHub App 连接\"],\"hdSi1b\":[\"输入 \",[\"repo\"],\" 以确认\"],\"he3ygx\":[\"复制\"],\"i0qMbr\":[\"首页\"],\"iEUzMn\":[\"系统\"],\"iSLIjg\":[\"连接\"],\"iVOMRi\":[\"主页设置已更新.\"],\"icB4Cv\":[\"将链接拖到此处以在更多菜单下显示它们\"],\"id3vuh\":[\"Telegram 已设置,但无法连接到机器人。请检查机器人令牌并重试。\"],\"ihn4zD\":[\"搜索…\"],\"iiDXZc\":[\"显示在所有文章和页面的底部。\"],\"j4VrG6\":[\"下载导出 ZIP\"],\"j5nQL2\":[\"例如 iOS Shortcuts\"],\"jUV7CU\":[\"上传头像\"],\"jVUmOK\":[\"支持 Markdown\"],\"jgBjXJ\":[\"撤销此令牌?任何使用它的脚本将停止工作。\"],\"jpctdh\":[\"查看\"],\"k1ifdL\":[\"处理中...\"],\"kMXclu\":[\"下载网站导出\"],\"kNiQp6\":[\"已置顶\"],\"kRhzWq\":[\"GitHub 同步\"],\"kVQs7s\":[\"细粒度样式覆盖\"],\"ke1gWS\":[\"自定义 URL\"],\"kfcRb0\":[\"头像\"],\"kxDZ2i\":[\"此代码将在您网站的每个页面上运行。\"],\"l2Op2p\":[\"查询参数\"],\"lLW3vJ\":[\"目标 slug\"],\"lYHJih\":[\"撤销此会话? 该设备需要重新登录.\"],\"mLOk1i\":[\"立即将所有文章推送到 GitHub,而不是等待下一次自动同步。\"],\"mSNmrX\":[\"列出帖子:\"],\"nK07ni\":[\"为您的网站选择一种排版方向。每个主题都会同时改变字体搭配和阅读节奏。\"],\"nbfdhU\":[\"集成\"],\"o/vNDE\":[\"允许您覆盖任何主题变量.\"],\"oGC9uP\":[\"owner/repo\"],\"oH2JHg\":[\"我们会预先填充名称 \",[\"name\"],\"。返回时列表会刷新。\"],\"oKOOsY\":[\"颜色主题\"],\"oL535e\":[\"尚未同步\"],\"oNA4If\":[\"所有合集已在您的导航中。\"],\"pZq3aX\":[\"上传失败。请重试。\"],\"pgTIrt\":[\"选择要与此站点同步的 GitHub 账户和仓库。\"],\"psoxDF\":[\"该字体主题不可用。请选择另一个。\"],\"pvnfJD\":[\"深色\"],\"q+hNag\":[\"合集\"],\"qdcESc\":[\"创建新仓库\"],\"r5EW6f\":[\"此仓库已在为另一个 Jant 站点 (\",[\"host\"],\") 进行备份。请选择其他仓库。\"],\"rEspiY\":[\"导航位置已更新.\"],\"rFmBG3\":[\"配色主题\"],\"rlonmB\":[\"删除失败。请稍后再试。\"],\"satWc6\":[\"主 RSS 源\"],\"sgr2wQ\":[\"合集\"],\"sqxcaY\":[\"创建于 \",[\"date\"]],\"sxkWRg\":[\"高级\"],\"t/YqKh\":[\"移除\"],\"t3hvHq\":[\"立即同步\"],\"tJ4H0O\":[\"你的 Telegram 账号\"],\"tfDRzk\":[\"保存\"],\"tvgAq5\":[\"尚未授权任何账户\"],\"u1VTd3\":[\"调色板、表面色调与整体氛围\"],\"u3wRF+\":[\"已发布\"],\"u6KOjV\":[\"想要更多控制?\"],\"udPwLB\":[\"页眉\"],\"ui6aMF\":[\"以下设备当前已登录到您的帐户。撤销任何您不认识的会话。\"],\"vBEKwo\":[\"在此管理本站点的活动会话。密码和托管访问通过 \",[\"providerLabel\"],\" 管理。\"],\"vRldcl\":[\"排版选项与阅读质感\"],\"vSYKYI\":[\"主订阅源\"],\"vTuib7\":[\"这将控制 /feed 返回的内容。\"],\"vXIe7J\":[\"语言\"],\"vmQmHx\":[\"添加自定义 CSS 以覆盖任何样式. 使用数据属性如 [data-page], [data-post], [data-format] 来定位特定元素.\"],\"vzX5FB\":[\"删除账号\"],\"w8Rv8T\":[\"标签为必填项\"],\"wL3cK8\":[\"最新\"],\"wPmHHc\":[\"低调的界面让写作成为焦点.\"],\"wW6NCp\":[\"上次错误\"],\"wc+17X\":[\"/* 在此填写您的自定义 CSS */\"],\"wuLtXn\":[\"当前没有活动会话。 已登录的设备会显示在此处。\"],\"xCWek4\":[\"文件存储尚未设置。请检查服务器配置。\"],\"xHt036\":[\"个人访问令牌\"],\"xbN8dp\":[\"柔和的颜色仍应保持清晰的阅读节奏。\"],\"y28hnO\":[\"文章\"],\"y8Md/V\":[\"语言和时间已更新。\"],\"yNCqOt\":[\"最新订阅源\"],\"yQ3kNF\":[\"输入以下短语以确认:\"],\"ydq1k2\":[\"请先选择一个账号\"],\"yjjCV8\":[\"固定订阅源 URL\"],\"yjkELF\":[\"确认新密码\"],\"yzF66j\":[\"链接\"],\"z6wakA\":[\"在您的主页上显示的简短介绍.\"],\"zEizrk\":[\"上次使用 \",[\"date\"]],\"zSURJW\":[\"没有匹配的仓库。\"],\"zXH2jX\":[\"语言与时间\"],\"zlcDd2\":[\"要删除此自定义 URL 吗?使用它的访问者将不再被重定向。\"],\"zwBp5t\":[\"私有\"],\"zxRN6H\":[\"页眉链接、首页信息流和溢出菜单\"]}");
1922
+ var messages$1 = JSON.parse("{\"+4Z6iP\":[\"请先在 GitHub 上创建仓库 — 可以为空。\"],\"+9JI/F\":[\"连接后会将你的网站同步到 \",[\"repo\"],\" 的默认分支,并追加到其现有历史之上。Jant 管理路径之外的现有文件会被保留。此操作无法撤销。\"],\"+AXdXp\":[\"标签和 URL 为必填项\"],\"+K0AvT\":[\"断开连接\"],\"+zy2Nq\":[\"类型\"],\"/3H2/s\":[\"此托管站点通过 \",[\"providerLabel\"],\" 登录。请在那里管理密码和托管访问权限。\"],\"/JnyjR\":[\"切换内置导航项。它们的顺序决定页眉显示的内容以及首页显示哪个视图。\"],\"/ODeyS\":[\"声明给读者的内容语言(HTML lang、RSS),同时驱动后台界面的语言。可填任意 BCP 47 标签;没有对应翻译时后台会回退为英文。\"],\"/zOUxl\":[\"链接到 Telegram 机器人的二维码\"],\"0OGSSc\":[\"头像显示已更新.\"],\"0UzCUX\":[\"更新您用于登录的密码\"],\"0bdA9b\":[\"打开 Telegram 以连接\"],\"10UtuM\":[\"CJK 字体\"],\"14BEca\":[\"了解原因\"],\"1F6Mzc\":[\"当前还没有导航项目。添加链接或在下方启用系统项目。\"],\"1mbBbL\":[\"手动绑定\"],\"1njn7W\":[\"浅色\"],\"2B7t+s\":[\"会话与密码\"],\"2DoBvq\":[\"订阅源\"],\"2FYpfJ\":[\"更多\"],\"2Ithfh\":[\"向机器人发送任意文本,它会作为笔记已发布。\"],\"2MXb5X\":[\"关于静谧设计的笔记\"],\"2PTjMB\":[\"我想删除 \",[\"siteName\"]],\"2cFU6q\":[\"网站页脚\"],\"2oWZo7\":[\"最近一次提交\"],\"2uuy4H\":[\"通过个人访问令牌连接\"],\"35x8eZ\":[\"显示 \",[\"shown\"],\" 共 \",[\"total\"]],\"39QGku\":[\"打开机器人并发送绑定码, 然后你发送给它的任何消息都会变成一条笔记。\"],\"3Cw1AI\":[\"添加合集\"],\"3VrybB\":[\"重定向\"],\"3Yvsaz\":[\"302 (临时)\"],\"3n0zbB\":[\"会话管理在演示模式下已关闭。请使用共享的演示会话。\"],\"3sYJi5\":[\"下载一个与 Hugo 兼容的归档 — 将其静态托管或迁移到另一个 Jant。\"],\"3wKq0C\":[\"保存失败。请稍后再试。\"],\"49Bsal\":[\"订阅源设置已更新。\"],\"4Jge8E\":[\"活动会话\"],\"4KIa+q\":[\"导出文件已下载.\"],\"4cEClj\":[\"会话\"],\"4zGJ5E\":[\"永久删除账号\"],\"5QlUIt\":[\"仓库为空。准备连接。\"],\"5VQnR3\":[\"当你想要一个永远不变的订阅源 URL 时使用它们。\"],\"5dpcN1\":[\"输入以搜索全部\"],\"5f1Wo9\":[\"已连接为 \",[\"account\"]],\"6ArdBh\":[\"将精选文章用于 /feed。\"],\"6DjeBT\":[\"演示站点始终对搜索引擎隐藏。\"],\"6E3aK4\":[\"管理托管\"],\"6FFB7q\":[\"为 /feed 使用最新的公开帖子。\"],\"6K1Vef\":[\"永久删除此博客?此操作不可撤销。\"],\"6NpNLc\":[\"此仓库已有内容。\"],\"6V3Ea3\":[\"已复制\"],\"71WIgc\":[\"获取新绑定码\"],\"746NHh\":[\"此博客\"],\"7811AW\":[\"此仓库已在备份此站点。\"],\"7FaY4u\":[\"用法\"],\"7G9YLi\":[\"允许搜索引擎索引我的网站\"],\"7GISOt\":[\"保存机器人令牌\"],\"7MZxzw\":[\"密码已更改.\"],\"7vhWI8\":[\"新密码\"],\"81nFIS\":[\"密码不匹配。请确保两个字段相同。\"],\"87a/t/\":[\"标签\"],\"89Upyo\":[\"该主题不可用。请选择其他主题。\"],\"8BfEpW\":[\"托管账户\"],\"8N/Mcp\":[\"归档 筛选参数 (例如 format=笔记&view=list)\"],\"8T46pB\":[\"机器人令牌\"],\"8U2Z7f\":[\"新建自定义 URL\"],\"8ZsakT\":[\"密码\"],\"9+vGLh\":[\"自定义 CSS\"],\"9As8Nu\":[\"在 GitHub 上创建一个\"],\"9Lsvt5\":[\"于 \",[\"date\"],\" 登录\"],\"9T7Cwm\":[\"重定向、个性化路径和 URL 控制\"],\"9aUyym\":[\"查看你的登录位置并撤销旧会话\"],\"A1taO8\":[\"搜索\"],\"AeXO77\":[\"账户\"],\"AnY+O9\":[\"在主页底部显示 \\\"Build with Jant\\\"\"],\"ApZDMk\":[\"此图用于您的 favicon 和 apple-touch-icon。为获得最佳效果,请上传至少 512×512 像素、纯色背景的方形 PNG。\"],\"B495Gs\":[\"归档\"],\"B4ESok\":[\"API 参考\"],\"BzEFor\":[\"或\"],\"CDAdlf\":[\"移除机器人\"],\"CTAEes\":[\"选择仓库\"],\"CjZZgz\":[\"该仓库已有提交\"],\"D8k2s6\":[\"连接 Telegram\"],\"DCKkhU\":[\"当前密码\"],\"DKKKeF\":[\"在 \",[\"providerLabel\"],\" 管理密码和托管访问\"],\"EO3I6h\":[\"上传未成功。请稍后再试。\"],\"Enslfm\":[\"目标地址\"],\"F7FKwe\":[\"你在此处粘贴的任何内容都将完全访问访客的浏览器。仅使用来自你信任的来源的代码。\"],\"FkMol5\":[\"精选\"],\"G/1oP+\":[\"移除 webhook 并停止同步。您的仓库内容不会被删除。\"],\"G0qJsQ\":[\"缺少安全令牌。刷新页面后重试。\"],\"G39wnK\":[\"将内容备份并与 GitHub 仓库同步\"],\"GMMWcy\":[\"名称、元数据、语言和搜索默认设置\"],\"GXsAby\":[\"撤销\"],\"GxkJXS\":[\"正在上传...\"],\"GzKzUa\":[\"演示限制\"],\"HKH+W+\":[\"数据\"],\"Hp1l6f\":[\"当前\"],\"HxlY7t\":[\"更改此项会更新订阅者从 /feed 获取的内容。\"],\"HxuOlm\":[\"网站头部\"],\"I6gXOa\":[\"路径\"],\"ID38tA\":[\"演示模式下已禁用账号删除。共享演示会单独重置。\"],\"IF9tPu\":[\"何时使用站点导出、数据库备份和恢复演练。\"],\"IW5PBo\":[\"复制令牌\"],\"IagCbF\":[\"网址\"],\"IreQBq\":[\"仓库\"],\"J6bLeg\":[\"向任意 URL 添加自定义链接\"],\"JL7LF5\":[\"可用的 CSS 变量, 数据属性, 和 示例.\"],\"JTviaO\":[\"管理登录安全、导出和不可逆操作.\"],\"JcD7qf\":[\"更多操作\"],\"JjX0OO\":[\"现在复制您的令牌 — 它不会再次显示。\"],\"JrFTcr\":[\"连接中…\"],\"JuN5GC\":[\"未选择文件。请选择要上传的文件。\"],\"KDw4GX\":[\"重试\"],\"KSgo21\":[\"选择仓库\"],\"KVVYBh\":[\"向导航添加合集\"],\"KiJn9B\":[\"笔记\"],\"L3DEwT\":[\"移除此头像?您的网站图标和页眉图标将恢复为默认设置。\"],\"L4t4/q\":[\"3月14日\"],\"LdyooL\":[\"链接\"],\"M/D8PK\":[\"+ 在其他帐户上安装\"],\"M/haSd\":[\"始终显示浅色主题。\"],\"M2kIWU\":[\"字体主题\"],\"M6CbAU\":[\"切换编辑面板\"],\"MaYYE6\":[\"通过向 Telegram 机器人发送消息来发布笔记\"],\"Me5t5H\":[\"将 GitHub 仓库连接以自动将您的文章备份为 Markdown 文件。您在 GitHub 上的编辑会同步回您的网站。\"],\"Mr4QPw\":[\"断开 Telegram?您可以随时使用新的绑定码重新连接。\"],\"MtENL9\":[\"调整你的网站的外观、阅读体验和运行方式。\"],\"N/8NPV\":[\"在删除之前,请下载站点导出。删除后将无法恢复此账户。\"],\"N7UNHY\":[\"精选订阅源\"],\"NHnUHF\":[\"Favicon 和页眉中的个人标识\"],\"NU2Fqi\":[\"保存 CSS\"],\"Nldjdr\":[\"尚无自定义 URL。创建一个以便为文章添加重定向或自定义路径。\"],\"O7rgs6\":[\"页眉 RSS 指向你的 \",[\"feed\"],\" 源 (/feed)。在 常规 中更改 /feed 返回的内容。\"],\"OSJXFg\":[\"应用于整个站点,包括管理页面。选择一个调色板,然后选择它是随系统变化还是保持固定。\"],\"Ox3+3h\":[\"无匹配结果。\"],\"PEUV5I\":[\"代码注入已更新。\"],\"PXj9lw\":[\"停止接受来自 Telegram 的帖子。您的现有笔记将保持已发布。\"],\"PZ7HJ8\":[\"博客头像\"],\"Pwqkdw\":[\"正在加载…\"],\"PxJ9W6\":[\"生成令牌\"],\"Q/6Y+2\":[\"需要对目标仓库的 Contents(读/写)和 Webhooks(读/写)。\"],\"Q30z/l\":[\"要从导航中移除此合集吗?合集本身不会被删除。\"],\"Q99OtV\":[\"将合集固定到导航栏。在过去 48 小时内更新的合集旁会出现一个 * 号。\"],\"QZmz0H\":[\"内置链接\"],\"Qnrzvb\":[\"活动令牌\"],\"R6Z4LE\":[\"下载失败。请重试。\"],\"R9Khdg\":[\"自动\"],\"RcdDOS\":[\"在 Telegram 上给 @BotFather 发送消息创建一个机器人,然后粘贴它提供的令牌。\"],\"RxsRD6\":[\"时区\"],\"SJmfuf\":[\"站点名称\"],\"SKZhW9\":[\"令牌名称\"],\"SVQQPe\":[\"无法连接。检查错误并重试。\"],\"SchpMp\":[\"Telegram\"],\"TpF3v+\":[\"注入到 </head> 之前。用于分析、自定义元标签以及必须尽早加载的样式。\"],\"Tz0i8g\":[\"设置\"],\"UFK415\":[\"用于分析和小部件的全站 HTML\"],\"UTvFQq\":[\"打开 \",[\"linkOpen\"],\"@\",[\"botUsername\"],[\"linkClose\"],\" 并发送:\"],\"Uj/btJ\":[\"在我的站点页眉显示头像\"],\"UsODUn\":[\"选择账户\"],\"UxKoFf\":[\"导航\"],\"V+bhUy\":[\"安装 GitHub App\"],\"V4WsyL\":[\"添加链接\"],\"V5pZwT\":[\"搜索设置已更新。\"],\"VXUPla\":[\"使用 GitHub 应用连接\"],\"VhMDMg\":[\"更改密码\"],\"Vn3jYy\":[\"导航项\"],\"VoZYGU\":[\"这将永久删除您所有的数据 — 帖子、媒体、合集、设置和您的账户。您的博客将重置为初始设置状态。此操作不可撤销。\"],\"Weq9zb\":[\"常规\"],\"Wi9i06\":[\"遵循每位访客的系统偏好。\"],\"Wx1M8N\":[\"安装 GitHub App 以在无需管理个人令牌的情况下授予访问权限。权限按仓库范围授予,可在 GitHub 上撤销。\"],\"X+8FMk\":[\"当前密码不正确。请重试。\"],\"X1G9eY\":[\"导航预览\"],\"X9Hujr\":[\"手动推送\"],\"XtBJV8\":[\"正在检查仓库…\"],\"Xtc16w\":[\"刷新仓库列表\"],\"Y/F35r\":[\"使用 curl 创建帖子:\"],\"YF6zHf\":[\"站点设置已更新.\"],\"YdG2RF\":[\"导出站点\"],\"YkgZi7\":[\"连接一个 Telegram 机器人,然后你发送给它的任何消息都会作为笔记已发布。\"],\"YwhjRx\":[\"管理账户\"],\"ZDY7Fy\":[\"正在同步…\"],\"ZQKLI1\":[\"危险操作\"],\"ZS/CBL\":[\"删除此导航链接? 访客将不再在您网站的页眉中看到它。\"],\"ZhhOwV\":[\"引用 (Jant 的帖子格式之一。)\"],\"ZiooJI\":[\"API 令牌\"],\"Zm7Qb0\":[\"备份与恢复指南\"],\"ZmUkwN\":[\"向导航添加自定义链接\"],\"a14mj8\":[\"未知设备\"],\"a3LDKx\":[\"安全\"],\"aAIQg2\":[\"外观\"],\"aFkzVF\":[\"目标文章或合集的 slug\"],\"alKG0+\":[\"字体主题\"],\"anibOb\":[\"关于本博客\"],\"any7NR\":[\"主题指南\"],\"b+/jO6\":[\"301 (永久)\"],\"bHOiy1\":[\"演示模式下已禁用密码更改。请使用共享的演示凭证登录。\"],\"bHYIks\":[\"退出登录\"],\"bmrL08\":[\"演示模式会隐藏会话、密码更改和账号删除。导出仍然可用.\"],\"c3MN2z\":[\"所有可用的端点和请求格式。\"],\"cS7/bk\":[\"删除保存的机器人令牌?其 webhook 会被删除,任何已连接的账号将被断开连接。\"],\"cSDy01\":[\"自定义 CSS 已更新.\"],\"clzoNp\":[\"始终显示暗色主题.\"],\"cnGeoo\":[\"删除\"],\"d3FRkY\":[\"无法复制。请再试一次。\"],\"d5oGUo\":[\"在 GitHub 上创建新仓库\"],\"dEgA5A\":[\"取消\"],\"dTXUY+\":[\"确认删除账户\"],\"dYKrp3\":[\"从最新中隐藏 (不出现在 Latest 中,但仍可通过固定链接和集合访问的状态。)\"],\"dk7TCH\":[\"永久删除所有数据并重置博客\"],\"drodVV\":[\"还没有合集。请先创建一个,然后将其添加到你的导航中。\"],\"dsWkIw\":[\"要与 GitHub 断开连接吗?该 webhook 将被移除。您的仓库内容不会被删除。\"],\"e/tSI5\":[\"导航顺序已更新。\"],\"ePK91l\":[\"编辑\"],\"ebQKK7\":[\"站点\"],\"egK+Yy\":[\"用于脚本和自动化的 Bearer 令牌\"],\"ehj/zN\":[\"重定向类型\"],\"eneWvv\":[\"草稿\"],\"erTMh7\":[\"上次同步\"],\"f+m8jj\":[\"订阅源 URL 已复制。\"],\"f8fH8W\":[\"设计\"],\"fWYqkz\":[\"代码注入\"],\"gOWiTY\":[\"加载针对中文、日文或韩文内容优化的衬线字体。\"],\"gZ5owP\":[\"搜索仓库\"],\"gbqbh6\":[\"放心离开此页面 — 同步会在后台继续进行。\"],\"gkFvVN\":[\"注入到 </body> 之前。用于聊天小部件和不应阻塞页面加载的脚本。\"],\"gtQsRO\":[\"创建自定义 URL\"],\"hBO/y4\":[\"安全令牌已过期。刷新页面后重试。\"],\"hGmyDl\":[\"令牌让您无需登录即可从脚本、快捷方式和其他工具访问 API。\"],\"hIHkRy\":[\"已通过 GitHub App 连接\"],\"hdSi1b\":[\"输入 \",[\"repo\"],\" 以确认\"],\"he3ygx\":[\"复制\"],\"i0qMbr\":[\"首页\"],\"iEUzMn\":[\"系统\"],\"iSLIjg\":[\"连接\"],\"iVOMRi\":[\"主页设置已更新.\"],\"icB4Cv\":[\"将链接拖到此处以在更多菜单下显示它们\"],\"id3vuh\":[\"Telegram 已设置,但无法连接到机器人。请检查机器人令牌并重试。\"],\"ihn4zD\":[\"搜索…\"],\"iiDXZc\":[\"显示在所有文章和页面的底部。\"],\"j4VrG6\":[\"下载导出 ZIP\"],\"j5nQL2\":[\"例如 iOS Shortcuts\"],\"jUV7CU\":[\"上传头像\"],\"jVUmOK\":[\"支持 Markdown\"],\"jgBjXJ\":[\"撤销此令牌?任何使用它的脚本将停止工作。\"],\"jpctdh\":[\"查看\"],\"k1ifdL\":[\"处理中...\"],\"kMXclu\":[\"下载网站导出\"],\"kNiQp6\":[\"已置顶\"],\"kRhzWq\":[\"GitHub 同步\"],\"kVQs7s\":[\"细粒度样式覆盖\"],\"ke1gWS\":[\"自定义 URL\"],\"kfcRb0\":[\"头像\"],\"kxDZ2i\":[\"此代码将在您网站的每个页面上运行。\"],\"l2Op2p\":[\"查询参数\"],\"lLW3vJ\":[\"目标 slug\"],\"lYHJih\":[\"撤销此会话? 该设备需要重新登录.\"],\"mLOk1i\":[\"立即将所有文章推送到 GitHub,而不是等待下一次自动同步。\"],\"mSNmrX\":[\"列出帖子:\"],\"nK07ni\":[\"为您的网站选择一种排版方向。每个主题都会同时改变字体搭配和阅读节奏。\"],\"nbfdhU\":[\"集成\"],\"ntJYyh\":[\"在 \",[\"providerLabel\"],\" 管理域名、套餐和计费\"],\"o/vNDE\":[\"允许您覆盖任何主题变量.\"],\"oGC9uP\":[\"owner/repo\"],\"oH2JHg\":[\"我们会预先填充名称 \",[\"name\"],\"。返回时列表会刷新。\"],\"oKOOsY\":[\"颜色主题\"],\"oL535e\":[\"尚未同步\"],\"oNA4If\":[\"所有合集已在您的导航中。\"],\"pZq3aX\":[\"上传失败。请重试。\"],\"pgTIrt\":[\"选择要与此站点同步的 GitHub 账户和仓库。\"],\"psoxDF\":[\"该字体主题不可用。请选择另一个。\"],\"pvnfJD\":[\"深色\"],\"q+hNag\":[\"合集\"],\"qdcESc\":[\"创建新仓库\"],\"r5EW6f\":[\"此仓库已在为另一个 Jant 站点 (\",[\"host\"],\") 进行备份。请选择其他仓库。\"],\"rEspiY\":[\"导航位置已更新.\"],\"rFmBG3\":[\"配色主题\"],\"rlonmB\":[\"删除失败。请稍后再试。\"],\"satWc6\":[\"主 RSS 源\"],\"sgr2wQ\":[\"合集\"],\"sqxcaY\":[\"创建于 \",[\"date\"]],\"sxkWRg\":[\"高级\"],\"t/YqKh\":[\"移除\"],\"t3hvHq\":[\"立即同步\"],\"tJ4H0O\":[\"你的 Telegram 账号\"],\"tfDRzk\":[\"保存\"],\"tvgAq5\":[\"尚未授权任何账户\"],\"u1VTd3\":[\"调色板、表面色调与整体氛围\"],\"u3wRF+\":[\"已发布\"],\"u6KOjV\":[\"想要更多控制?\"],\"udPwLB\":[\"页眉\"],\"ui6aMF\":[\"以下设备当前已登录到您的帐户。撤销任何您不认识的会话。\"],\"vBEKwo\":[\"在此管理本站点的活动会话。密码和托管访问通过 \",[\"providerLabel\"],\" 管理。\"],\"vRldcl\":[\"排版选项与阅读质感\"],\"vSYKYI\":[\"主订阅源\"],\"vTuib7\":[\"这将控制 /feed 返回的内容。\"],\"vXIe7J\":[\"语言\"],\"vmQmHx\":[\"添加自定义 CSS 以覆盖任何样式. 使用数据属性如 [data-page], [data-post], [data-format] 来定位特定元素.\"],\"vzX5FB\":[\"删除账号\"],\"w8Rv8T\":[\"标签为必填项\"],\"wL3cK8\":[\"最新\"],\"wPmHHc\":[\"低调的界面让写作成为焦点.\"],\"wW6NCp\":[\"上次错误\"],\"wc+17X\":[\"/* 在此填写您的自定义 CSS */\"],\"wuLtXn\":[\"当前没有活动会话。 已登录的设备会显示在此处。\"],\"xCWek4\":[\"文件存储尚未设置。请检查服务器配置。\"],\"xHt036\":[\"个人访问令牌\"],\"xbN8dp\":[\"柔和的颜色仍应保持清晰的阅读节奏。\"],\"y28hnO\":[\"文章\"],\"y8Md/V\":[\"语言和时间已更新。\"],\"yNCqOt\":[\"最新订阅源\"],\"yQ3kNF\":[\"输入以下短语以确认:\"],\"ydq1k2\":[\"请先选择一个账号\"],\"yjjCV8\":[\"固定订阅源 URL\"],\"yjkELF\":[\"确认新密码\"],\"yzF66j\":[\"链接\"],\"z6wakA\":[\"在您的主页上显示的简短介绍.\"],\"zEizrk\":[\"上次使用 \",[\"date\"]],\"zSURJW\":[\"没有匹配的仓库。\"],\"zXH2jX\":[\"语言与时间\"],\"zlcDd2\":[\"要删除此自定义 URL 吗?使用它的访问者将不再被重定向。\"],\"zwBp5t\":[\"私有\"],\"zxRN6H\":[\"页眉链接、首页信息流和溢出菜单\"]}");
1923
1923
  //#endregion
1924
1924
  //#region src/i18n/locales/settings/zh-Hant.ts
1925
- var messages = JSON.parse("{\"+4Z6iP\":[\"先在 GitHub 上建立儲存庫 — 可以是空的。\"],\"+9JI/F\":[\"連線後會將您的網站同步到 \",[\"repo\"],\" 的預設分支,並疊加在其現有歷史之上。Jant 管理路徑外的既有檔案會保留。此操作無法復原。\"],\"+AXdXp\":[\"標籤與 URL 為必填\"],\"+K0AvT\":[\"解除連線\"],\"+zy2Nq\":[\"類型\"],\"/3H2/s\":[\"此託管網站透過 \",[\"providerLabel\"],\" 登入。請在該處管理密碼與託管存取權限。\"],\"/JnyjR\":[\"切換內建導覽項目。它們的順序決定頁首顯示哪些項目,以及首頁會先開啟哪一個 feed。\"],\"/ODeyS\":[\"聲明給讀者的內容語言(HTML lang、RSS),同時驅動後台介面的語言。可填任意 BCP 47 標籤;沒有對應翻譯時後台會回退為英文。\"],\"/zOUxl\":[\"連結至 Telegram 機器人的 QR 碼\"],\"0OGSSc\":[\"頭像顯示已更新.\"],\"0UzCUX\":[\"更新您用來登入的密碼\"],\"0bdA9b\":[\"開啟 Telegram 以連線\"],\"10UtuM\":[\"CJK 字體\"],\"14BEca\":[\"瞭解原因\"],\"1F6Mzc\":[\"目前尚無導覽項目。請新增連結或在下方啟用系統項目。\"],\"1mbBbL\":[\"手動連接\"],\"1njn7W\":[\"淺色\"],\"2B7t+s\":[\"工作階段與密碼\"],\"2DoBvq\":[\"訂閱來源\"],\"2FYpfJ\":[\"更多\"],\"2Ithfh\":[\"傳送任何文字給機器人,該文字會作為筆記已發佈。\"],\"2MXb5X\":[\"關於靜謐設計的實地筆記\"],\"2PTjMB\":[\"我想刪除 \",[\"siteName\"]],\"2cFU6q\":[\"網站頁尾\"],\"2oWZo7\":[\"最近一次提交\"],\"2uuy4H\":[\"已透過個人存取權杖連線\"],\"35x8eZ\":[\"顯示 \",[\"shown\"],\" 共 \",[\"total\"]],\"39QGku\":[\"開啟機器人並傳送綁定碼,之後你傳給它的任何訊息都會成為筆記。\"],\"3Cw1AI\":[\"新增選集\"],\"3VrybB\":[\"重新導向\"],\"3Yvsaz\":[\"302 (暫時)\"],\"3n0zbB\":[\"在示範模式中已停用工作階段管理。請改用共用示範工作階段。\"],\"3sYJi5\":[\"下載與 Hugo 相容的封存 — 以靜態方式託管或移轉到另一個 Jant\"],\"3wKq0C\":[\"無法儲存。請稍後再試。\"],\"49Bsal\":[\"Feed 設定已更新。\"],\"4Jge8E\":[\"目前的工作階段\"],\"4KIa+q\":[\"已下載匯出檔案。\"],\"4cEClj\":[\"工作階段\"],\"4zGJ5E\":[\"永久刪除帳戶\"],\"5QlUIt\":[\"倉庫為空。準備連線。\"],\"5VQnR3\":[\"當你想要一個永遠不會改變的 feed URL 時,請使用這些。\"],\"5dpcN1\":[\"輸入以搜尋全部\"],\"5f1Wo9\":[\"已以 \",[\"account\"],\" 連線\"],\"69OXZB\":[\"刪除託管網站\"],\"6ArdBh\":[\"將精選文章用於 /feed.\"],\"6DjeBT\":[\"示範網站會始終對搜尋引擎保持隱藏。\"],\"6FFB7q\":[\"使用最新的公開貼文作為 /feed。\"],\"6K1Vef\":[\"確定要永久刪除此部落格嗎?此動作無法復原。\"],\"6NpNLc\":[\"此儲存庫已有內容。\"],\"6V3Ea3\":[\"已複製\"],\"71WIgc\":[\"取得新代碼\"],\"746NHh\":[\"此部落格\"],\"7811AW\":[\"此儲存庫已在備份本網站。\"],\"7FaY4u\":[\"用法\"],\"7G9YLi\":[\"允許搜尋引擎收錄我的網站\"],\"7GISOt\":[\"儲存機器人權杖\"],\"7MZxzw\":[\"密碼已變更.\"],\"7vhWI8\":[\"新密碼\"],\"7z05Pf\":[\"在 \",[\"providerLabel\"],\" 開啟託管站點控制項以取消計費或永久刪除此站點。\"],\"81nFIS\":[\"密碼不符。請確認兩個欄位相同。\"],\"87a/t/\":[\"標籤\"],\"89Upyo\":[\"該主題目前不可用。請選擇其他主題。\"],\"8BfEpW\":[\"託管帳號\"],\"8N/Mcp\":[\"封存篩選參數 (例如 format=note&view=list)\"],\"8T46pB\":[\"機器人權杖\"],\"8U2Z7f\":[\"新增自訂 URL\"],\"8ZsakT\":[\"密碼\"],\"9+vGLh\":[\"自訂 CSS\"],\"9As8Nu\":[\"在 GitHub 上建立一個\"],\"9Lsvt5\":[\"已於 \",[\"date\"],\" 登入\"],\"9T7Cwm\":[\"重新導向、自訂路徑與網址控制\"],\"9aUyym\":[\"查看您在哪裡已登入,並撤銷舊的工作階段\"],\"A1taO8\":[\"搜尋\"],\"AeXO77\":[\"帳戶\"],\"AnY+O9\":[\"在首頁底部顯示「Build with Jant」\"],\"ApZDMk\":[\"此圖會用於您的 favicon 和 apple-touch-icon。為達最佳效果,請上傳至少 512x512 像素、背景為純色的正方形 PNG。\"],\"B495Gs\":[\"封存\"],\"B4ESok\":[\"API 參考\"],\"BzEFor\":[\"或\"],\"CDAdlf\":[\"移除機器人\"],\"CTAEes\":[\"選擇儲存庫\"],\"CjZZgz\":[\"此儲存庫已有提交紀錄\"],\"D8k2s6\":[\"連接 Telegram\"],\"DCKkhU\":[\"目前密碼\"],\"DKKKeF\":[\"在 \",[\"providerLabel\"],\" 管理密碼與託管存取\"],\"EO3I6h\":[\"上傳未成功. 請稍後再試.\"],\"Enslfm\":[\"目標網址\"],\"F7FKwe\":[\"你在此處貼上的任何內容都能完全存取訪客的瀏覽器。僅使用來自你信任來源的程式碼。\"],\"FkMol5\":[\"精選\"],\"G/1oP+\":[\"移除 webhook 並停止同步。您的儲存庫內容不會被刪除。\"],\"G0qJsQ\":[\"找不到安全權杖。請重新整理頁面後再試一次。\"],\"G39wnK\":[\"將內容備份並與 GitHub 儲存庫同步\"],\"GMMWcy\":[\"名稱, 中繼資料, 語言, 和 搜尋預設值\"],\"GXsAby\":[\"撤銷\"],\"GxkJXS\":[\"上傳中...\"],\"GzKzUa\":[\"試用限制\"],\"HKH+W+\":[\"資料\"],\"Hp1l6f\":[\"目前\"],\"HxlY7t\":[\"變更此設定會更新訂閱者從 /feed 取得的內容。\"],\"HxuOlm\":[\"網站頁首\"],\"I6gXOa\":[\"路徑\"],\"ID38tA\":[\"示範模式下帳號刪除已停用。共用示範會另行重置。\"],\"IF9tPu\":[\"何時使用網站匯出、資料庫備份與復原演練。\"],\"IW5PBo\":[\"複製權杖\"],\"IagCbF\":[\"URL\"],\"IreQBq\":[\"儲存庫\"],\"J6bLeg\":[\"新增自訂連結到任何 URL\"],\"JL7LF5\":[\"可用的 CSS 變數、data 屬性與範例.\"],\"JTviaO\":[\"管理登入安全、匯出,以及不可逆的操作。\"],\"JcD7qf\":[\"更多操作\"],\"JjX0OO\":[\"請立即複製您的權杖 — 它不會再顯示。\"],\"JrFTcr\":[\"連線中…\"],\"JuN5GC\":[\"未選取檔案。請選擇要上傳的檔案。\"],\"KDw4GX\":[\"重試\"],\"KSgo21\":[\"選擇一個儲存庫\"],\"KVVYBh\":[\"新增選集到導覽\"],\"KiJn9B\":[\"筆記\"],\"L3DEwT\":[\"移除這個頭像?您的 favicon 與頁首圖示會回復為預設。\"],\"L4t4/q\":[\"3月14日\"],\"LdyooL\":[\"連結\"],\"M/D8PK\":[\"+ 安裝到其他帳戶\"],\"M/haSd\":[\"永遠顯示主題的淺色版本。\"],\"M2kIWU\":[\"字型主題\"],\"M6CbAU\":[\"切換編輯面板\"],\"MaYYE6\":[\"透過向 Telegram 機器人傳送訊息來發佈筆記\"],\"Me5t5H\":[\"連接 GitHub 倉庫,自動將你的文章備份為 Markdown 檔案。GitHub 上的編輯會同步回你的網站。\"],\"Mr4QPw\":[\"要斷開 Telegram?您可以隨時使用新的綁定代碼重新連接。\"],\"MtENL9\":[\"調整您的網站外觀、可讀性與執行效能。\"],\"N/8NPV\":[\"在刪除前,請先下載網站匯出檔案。刪除後無法恢復此帳號。\"],\"N7UNHY\":[\"精選 RSS 來源\"],\"NHnUHF\":[\"頁首上的網站圖示與個人標記\"],\"NU2Fqi\":[\"儲存 CSS\"],\"Nldjdr\":[\"尚未有自訂 URL。建立一個自訂 URL 以為文章新增重新導向或自訂路徑。\"],\"O7rgs6\":[\"頁首 RSS 指向你的 \",[\"feed\"],\" 訂閱 (/feed)。在「一般」中變更 /feed 回傳的內容。\"],\"OSJXFg\":[\"套用於整個網站, 包括管理頁面. 選擇一個調色盤, 然後決定它是跟隨系統還是維持固定.\"],\"Ox3+3h\":[\"無相符結果。\"],\"PEUV5I\":[\"程式碼注入已更新。\"],\"PXj9lw\":[\"停止接受來自 Telegram 的貼文。您現有的筆記會保持已發佈。\"],\"PZ7HJ8\":[\"部落格大頭貼\"],\"Pwqkdw\":[\"載入中…\"],\"PxJ9W6\":[\"產生權杖\"],\"Q/6Y+2\":[\"需要在目標儲存庫上擁有 Contents(讀/寫)和 Webhooks(讀/寫)權限。\"],\"Q30z/l\":[\"要從導覽移除這個選集嗎?選集本身不會被刪除。\"],\"Q99OtV\":[\"將選集釘選到導覽列。最近 48 小時內更新的選集旁會顯示一個 * 號。\"],\"QZmz0H\":[\"內建連結\"],\"Qnrzvb\":[\"已啟用的權杖\"],\"R6Z4LE\":[\"下載失敗。請再試一次。\"],\"R9Khdg\":[\"自動\"],\"RcdDOS\":[\"在 Telegram 上向 @BotFather 發送訊息以建立機器人,然後貼上它提供給你的權杖\"],\"RxsRD6\":[\"時區\"],\"SJmfuf\":[\"網站名稱\"],\"SKZhW9\":[\"權杖名稱 (API 權杖名稱欄位。)\"],\"SVQQPe\":[\"無法連線。請檢查錯誤並再試一次。\"],\"SchpMp\":[\"Telegram\"],\"TpF3v+\":[\"在 </head> 之前注入。用於分析、自訂 meta 標籤,以及必須提前載入的樣式。\"],\"Tz0i8g\":[\"設定\"],\"UFK415\":[\"用於分析與小工具的網站全域 HTML\"],\"UTvFQq\":[\"開啟 \",[\"linkOpen\"],\"@\",[\"botUsername\"],[\"linkClose\"],\" 並傳送:\"],\"Uj/btJ\":[\"在我的網站頁首顯示大頭貼\"],\"UsODUn\":[\"選擇一個帳戶\"],\"UxKoFf\":[\"導覽\"],\"V+bhUy\":[\"安裝 GitHub App\"],\"V4WsyL\":[\"新增連結\"],\"V5pZwT\":[\"搜尋設定已更新。\"],\"VXUPla\":[\"使用 GitHub App 連線\"],\"VhMDMg\":[\"變更密碼\"],\"Vn3jYy\":[\"導覽項目\"],\"VoZYGU\":[\"這會永久刪除您所有的資料 — 文章、媒體、選集、設定,以及您的帳戶。您的部落格將被重設為初始設定狀態。此操作無法復原。\"],\"Weq9zb\":[\"一般\"],\"Wi9i06\":[\"依照每位訪客的系統偏好。\"],\"Wx1M8N\":[\"安裝 GitHub App,以授予存取權而無需管理個人權杖。權限以每個儲存庫為範圍,並可在 GitHub 上撤銷。\"],\"X+8FMk\":[\"目前的密碼不符。請再試一次。\"],\"X1G9eY\":[\"導覽預覽\"],\"X9Hujr\":[\"手動推送\"],\"XtBJV8\":[\"正在檢查儲存庫…\"],\"Xtc16w\":[\"重新整理儲存庫清單\"],\"Y/F35r\":[\"使用 curl 建立貼文:\"],\"YF6zHf\":[\"網站設定已更新.\"],\"YdG2RF\":[\"匯出網站\"],\"YkgZi7\":[\"連接一個 Telegram 機器人,之後你傳給它的任何訊息都會以筆記形式已發佈。\"],\"YwhjRx\":[\"管理帳戶\"],\"ZDY7Fy\":[\"同步中…\"],\"ZQKLI1\":[\"危險區域\"],\"ZS/CBL\":[\"刪除此導覽連結?訪客將不再在您的網站頁首看到它。\"],\"ZhhOwV\":[\"引用 (Jant 的貼文格式之一。)\"],\"ZiooJI\":[\"API 權杖\"],\"Zm7Qb0\":[\"備份與還原指南\"],\"ZmUkwN\":[\"新增自訂連結到導覽\"],\"a14mj8\":[\"未知裝置\"],\"a3LDKx\":[\"安全性\"],\"aAIQg2\":[\"外觀\"],\"aFkzVF\":[\"目標文章或選集的 slug\"],\"alKG0+\":[\"字型主題\"],\"anibOb\":[\"關於本部落格\"],\"any7NR\":[\"主題指南\"],\"b+/jO6\":[\"301 (永久)\"],\"bHOiy1\":[\"示範模式已停用變更密碼功能。請使用共用示範帳號登入。\"],\"bHYIks\":[\"登出\"],\"bmrL08\":[\"示範模式會隱藏會話、密碼更改與帳號刪除。匯出功能仍可使用。\"],\"c3MN2z\":[\"所有可用的端點與請求格式。\"],\"cS7/bk\":[\"移除已儲存的機器人權杖?其 webhook 會被刪除,任何已連接的帳號將會被斷線。\"],\"cSDy01\":[\"自訂 CSS 已更新.\"],\"clzoNp\":[\"始終顯示深色主題。\"],\"cnGeoo\":[\"刪除\"],\"d3FRkY\":[\"無法複製. 請再試一次.\"],\"d5oGUo\":[\"在 GitHub 建立新儲存庫\"],\"dEgA5A\":[\"取消\"],\"dTXUY+\":[\"確認刪除帳號\"],\"dYKrp3\":[\"從最新中隱藏 (不出現在 Latest 中,但仍可透過固定連結和集合存取的狀態。)\"],\"dk7TCH\":[\"永久刪除所有資料並重設部落格\"],\"drodVV\":[\"目前還沒有選集。請先建立一個,然後將它加入您的導覽。\"],\"dsWkIw\":[\"要與 GitHub 斷開連線嗎? webhook 將會被移除。您的儲存庫內容不會被刪除。\"],\"e/tSI5\":[\"導覽順序已更新。\"],\"ePK91l\":[\"編輯\"],\"ebQKK7\":[\"網站\"],\"egK+Yy\":[\"供腳本與自動化使用的 Bearer 權杖\"],\"ehj/zN\":[\"重新導向類型\"],\"eneWvv\":[\"草稿\"],\"erTMh7\":[\"上次同步\"],\"f+m8jj\":[\"已複製訂閱網址。\"],\"f8fH8W\":[\"設計\"],\"fWYqkz\":[\"程式碼注入\"],\"gOWiTY\":[\"載入為中文、日文或韓文內容最佳化的襯線字型。\"],\"gZ5owP\":[\"搜尋儲存庫\"],\"gbqbh6\":[\"可以放心離開此頁面 — 同步會在背景繼續進行。\"],\"gkFvVN\":[\"在 </body> 之前注入。用於聊天小工具和不應阻塞頁面載入的腳本。\"],\"gtQsRO\":[\"建立自訂網址\"],\"hBO/y4\":[\"安全權杖已過期。請重新整理頁面並再試一次。\"],\"hGmyDl\":[\"權杖讓您從腳本, 捷徑和其他工具存取 API 無需登入\"],\"hIHkRy\":[\"已透過 GitHub 應用程式連線\"],\"hdSi1b\":[\"輸入 \",[\"repo\"],\" 以確認\"],\"he3ygx\":[\"複製\"],\"i0qMbr\":[\"首頁\"],\"iEUzMn\":[\"系統\"],\"iSLIjg\":[\"連接\"],\"iVOMRi\":[\"首頁設定已更新。\"],\"icB4Cv\":[\"將連結拖到此處以顯示於「更多」選單下方\"],\"id3vuh\":[\"已設定 Telegram,但無法連線到機器人。請檢查機器人權杖並再試一次。\"],\"ihn4zD\":[\"搜尋…\"],\"iiDXZc\":[\"顯示於所有文章與頁面的底部。\"],\"j4VrG6\":[\"下載匯出 ZIP\"],\"j5nQL2\":[\"例如 iOS 捷徑\"],\"jUV7CU\":[\"上傳大頭貼\"],\"jVUmOK\":[\"支援 Markdown\"],\"jgBjXJ\":[\"要撤銷這個權杖嗎?任何使用它的腳本都會停止運作。\"],\"jpctdh\":[\"檢視\"],\"k1ifdL\":[\"處理中...\"],\"kMXclu\":[\"下載網站匯出檔案\"],\"kNiQp6\":[\"已釘選\"],\"kRhzWq\":[\"GitHub 同步\"],\"kVQs7s\":[\"細緻的樣式覆寫\"],\"ke1gWS\":[\"自訂 URL\"],\"kfcRb0\":[\"頭像\"],\"kxDZ2i\":[\"此程式碼會在您網站的每個頁面上執行。\"],\"l2Op2p\":[\"查詢參數\"],\"lLW3vJ\":[\"目標 slug\"],\"lYHJih\":[\"撤銷此工作階段?該裝置將需要重新登入。\"],\"mLOk1i\":[\"立即將所有文章推送到 GitHub,而不是等待下一次自動同步。\"],\"mSNmrX\":[\"列出貼文:\"],\"nK07ni\":[\"為您的網站選擇一種排版風格。每個主題會同時改變字體配對與閱讀節奏。\"],\"nbfdhU\":[\"第三方整合\"],\"o/vNDE\":[\"讓您覆寫任何主題變數。\"],\"oGC9uP\":[\"owner/repo\"],\"oH2JHg\":[\"我們會預先填入名稱 \",[\"name\"],\"。返回時清單會重新整理。\"],\"oKOOsY\":[\"色彩主題\"],\"oL535e\":[\"尚未同步\"],\"oNA4If\":[\"所有選集已經在您的導覽中。\"],\"pZq3aX\":[\"上傳失敗。請再試一次。\"],\"pgTIrt\":[\"選擇要與此網站同步的 GitHub 帳號和儲存庫.\"],\"psoxDF\":[\"該字型主題無法使用. 請選擇其他主題.\"],\"pvnfJD\":[\"深色\"],\"q+hNag\":[\"選集\"],\"qdcESc\":[\"建立新儲存庫\"],\"r5EW6f\":[\"此儲存庫已在備份另一個 Jant 網站 (\",[\"host\"],\"). 請選擇其他儲存庫.\"],\"rEspiY\":[\"導覽位置已更新。\"],\"rFmBG3\":[\"色彩主題\"],\"rlonmB\":[\"無法刪除。請稍後再試。\"],\"satWc6\":[\"主要 RSS 訂閱\"],\"sgr2wQ\":[\"選集\"],\"sqxcaY\":[\"建立於 \",[\"date\"]],\"sxkWRg\":[\"進階\"],\"t/YqKh\":[\"移除\"],\"t3hvHq\":[\"立即同步\"],\"tJ4H0O\":[\"您的 Telegram 帳號\"],\"tfDRzk\":[\"儲存\"],\"tvgAq5\":[\"尚未授權任何帳戶\"],\"u1VTd3\":[\"調色盤、表面色調與整體氛圍\"],\"u3wRF+\":[\"已發佈\"],\"u6KOjV\":[\"想要更細緻的控制?\"],\"udPwLB\":[\"頁首\"],\"ui6aMF\":[\"這些裝置目前已登入您的帳號。撤銷任何您不認識的工作階段。\"],\"vBEKwo\":[\"在此管理本網站的活動工作階段。密碼與託管存取由 \",[\"providerLabel\"],\" 管理。\"],\"vRldcl\":[\"字體選擇與閱讀質感\"],\"vSYKYI\":[\"主要訂閱來源\"],\"vTuib7\":[\"這會控制 /feed 回傳的內容。\"],\"vXIe7J\":[\"語言\"],\"vmQmHx\":[\"新增自訂 CSS 以覆寫任何樣式。使用像 [data-page]、[data-post]、[data-format] 這類資料屬性來選取特定元素。\"],\"vzX5FB\":[\"刪除帳號\"],\"w8Rv8T\":[\"標籤為必填\"],\"wL3cK8\":[\"最新\"],\"wPmHHc\":[\"低調的介面讓文字成為主角。\"],\"wW6NCp\":[\"上次錯誤\"],\"wc+17X\":[\"/* 在此放入您的自訂 CSS */\"],\"wuLtXn\":[\"目前沒有任何活動中的工作階段。已登入的裝置會顯示在此處。\"],\"xCWek4\":[\"檔案儲存尚未設定。請檢查您的伺服器設定。\"],\"xHt036\":[\"個人存取權杖\"],\"xbN8dp\":[\"柔和的色彩仍應保有清晰的閱讀節奏。\"],\"y28hnO\":[\"文章\"],\"y8Md/V\":[\"語言與時間已更新。\"],\"yNCqOt\":[\"最新 RSS 來源\"],\"yQ3kNF\":[\"請輸入以下短語以確認:\"],\"ydq1k2\":[\"請先選擇一個帳戶\"],\"yjjCV8\":[\"固定的 RSS 檔案網址\"],\"yjkELF\":[\"確認新密碼\"],\"yzF66j\":[\"連結 (Jant 的貼文格式之一。)\"],\"z6wakA\":[\"顯示在您的主頁上的簡短介紹。\"],\"zEizrk\":[\"最後使用於 \",[\"date\"]],\"zSURJW\":[\"沒有符合的儲存庫.\"],\"zXH2jX\":[\"語言與時區\"],\"zlcDd2\":[\"刪除此自訂 URL?使用該 URL 的訪客將不會再被重新導向。\"],\"zwBp5t\":[\"私有\"],\"zxRN6H\":[\"頁首連結、首頁動態與更多選單\"]}");
1925
+ var messages = JSON.parse("{\"+4Z6iP\":[\"先在 GitHub 上建立儲存庫 — 可以是空的。\"],\"+9JI/F\":[\"連線後會將您的網站同步到 \",[\"repo\"],\" 的預設分支,並疊加在其現有歷史之上。Jant 管理路徑外的既有檔案會保留。此操作無法復原。\"],\"+AXdXp\":[\"標籤與 URL 為必填\"],\"+K0AvT\":[\"解除連線\"],\"+zy2Nq\":[\"類型\"],\"/3H2/s\":[\"此託管網站透過 \",[\"providerLabel\"],\" 登入。請在該處管理密碼與託管存取權限。\"],\"/JnyjR\":[\"切換內建導覽項目。它們的順序決定頁首顯示哪些項目,以及首頁會先開啟哪一個 feed。\"],\"/ODeyS\":[\"聲明給讀者的內容語言(HTML lang、RSS),同時驅動後台介面的語言。可填任意 BCP 47 標籤;沒有對應翻譯時後台會回退為英文。\"],\"/zOUxl\":[\"連結至 Telegram 機器人的 QR 碼\"],\"0OGSSc\":[\"頭像顯示已更新.\"],\"0UzCUX\":[\"更新您用來登入的密碼\"],\"0bdA9b\":[\"開啟 Telegram 以連線\"],\"10UtuM\":[\"CJK 字體\"],\"14BEca\":[\"瞭解原因\"],\"1F6Mzc\":[\"目前尚無導覽項目。請新增連結或在下方啟用系統項目。\"],\"1mbBbL\":[\"手動連接\"],\"1njn7W\":[\"淺色\"],\"2B7t+s\":[\"工作階段與密碼\"],\"2DoBvq\":[\"訂閱來源\"],\"2FYpfJ\":[\"更多\"],\"2Ithfh\":[\"傳送任何文字給機器人,該文字會作為筆記已發佈。\"],\"2MXb5X\":[\"關於靜謐設計的實地筆記\"],\"2PTjMB\":[\"我想刪除 \",[\"siteName\"]],\"2cFU6q\":[\"網站頁尾\"],\"2oWZo7\":[\"最近一次提交\"],\"2uuy4H\":[\"已透過個人存取權杖連線\"],\"35x8eZ\":[\"顯示 \",[\"shown\"],\" 共 \",[\"total\"]],\"39QGku\":[\"開啟機器人並傳送綁定碼,之後你傳給它的任何訊息都會成為筆記。\"],\"3Cw1AI\":[\"新增選集\"],\"3VrybB\":[\"重新導向\"],\"3Yvsaz\":[\"302 (暫時)\"],\"3n0zbB\":[\"在示範模式中已停用工作階段管理。請改用共用示範工作階段。\"],\"3sYJi5\":[\"下載與 Hugo 相容的封存 — 以靜態方式託管或移轉到另一個 Jant\"],\"3wKq0C\":[\"無法儲存。請稍後再試。\"],\"49Bsal\":[\"Feed 設定已更新。\"],\"4Jge8E\":[\"目前的工作階段\"],\"4KIa+q\":[\"已下載匯出檔案。\"],\"4cEClj\":[\"工作階段\"],\"4zGJ5E\":[\"永久刪除帳戶\"],\"5QlUIt\":[\"倉庫為空。準備連線。\"],\"5VQnR3\":[\"當你想要一個永遠不會改變的 feed URL 時,請使用這些。\"],\"5dpcN1\":[\"輸入以搜尋全部\"],\"5f1Wo9\":[\"已以 \",[\"account\"],\" 連線\"],\"6ArdBh\":[\"將精選文章用於 /feed.\"],\"6DjeBT\":[\"示範網站會始終對搜尋引擎保持隱藏。\"],\"6E3aK4\":[\"管理託管\"],\"6FFB7q\":[\"使用最新的公開貼文作為 /feed。\"],\"6K1Vef\":[\"確定要永久刪除此部落格嗎?此動作無法復原。\"],\"6NpNLc\":[\"此儲存庫已有內容。\"],\"6V3Ea3\":[\"已複製\"],\"71WIgc\":[\"取得新代碼\"],\"746NHh\":[\"此部落格\"],\"7811AW\":[\"此儲存庫已在備份本網站。\"],\"7FaY4u\":[\"用法\"],\"7G9YLi\":[\"允許搜尋引擎收錄我的網站\"],\"7GISOt\":[\"儲存機器人權杖\"],\"7MZxzw\":[\"密碼已變更.\"],\"7vhWI8\":[\"新密碼\"],\"81nFIS\":[\"密碼不符。請確認兩個欄位相同。\"],\"87a/t/\":[\"標籤\"],\"89Upyo\":[\"該主題目前不可用。請選擇其他主題。\"],\"8BfEpW\":[\"託管帳號\"],\"8N/Mcp\":[\"封存篩選參數 (例如 format=note&view=list)\"],\"8T46pB\":[\"機器人權杖\"],\"8U2Z7f\":[\"新增自訂 URL\"],\"8ZsakT\":[\"密碼\"],\"9+vGLh\":[\"自訂 CSS\"],\"9As8Nu\":[\"在 GitHub 上建立一個\"],\"9Lsvt5\":[\"已於 \",[\"date\"],\" 登入\"],\"9T7Cwm\":[\"重新導向、自訂路徑與網址控制\"],\"9aUyym\":[\"查看您在哪裡已登入,並撤銷舊的工作階段\"],\"A1taO8\":[\"搜尋\"],\"AeXO77\":[\"帳戶\"],\"AnY+O9\":[\"在首頁底部顯示「Build with Jant」\"],\"ApZDMk\":[\"此圖會用於您的 favicon 和 apple-touch-icon。為達最佳效果,請上傳至少 512x512 像素、背景為純色的正方形 PNG。\"],\"B495Gs\":[\"封存\"],\"B4ESok\":[\"API 參考\"],\"BzEFor\":[\"或\"],\"CDAdlf\":[\"移除機器人\"],\"CTAEes\":[\"選擇儲存庫\"],\"CjZZgz\":[\"此儲存庫已有提交紀錄\"],\"D8k2s6\":[\"連接 Telegram\"],\"DCKkhU\":[\"目前密碼\"],\"DKKKeF\":[\"在 \",[\"providerLabel\"],\" 管理密碼與託管存取\"],\"EO3I6h\":[\"上傳未成功. 請稍後再試.\"],\"Enslfm\":[\"目標網址\"],\"F7FKwe\":[\"你在此處貼上的任何內容都能完全存取訪客的瀏覽器。僅使用來自你信任來源的程式碼。\"],\"FkMol5\":[\"精選\"],\"G/1oP+\":[\"移除 webhook 並停止同步。您的儲存庫內容不會被刪除。\"],\"G0qJsQ\":[\"找不到安全權杖。請重新整理頁面後再試一次。\"],\"G39wnK\":[\"將內容備份並與 GitHub 儲存庫同步\"],\"GMMWcy\":[\"名稱, 中繼資料, 語言, 和 搜尋預設值\"],\"GXsAby\":[\"撤銷\"],\"GxkJXS\":[\"上傳中...\"],\"GzKzUa\":[\"試用限制\"],\"HKH+W+\":[\"資料\"],\"Hp1l6f\":[\"目前\"],\"HxlY7t\":[\"變更此設定會更新訂閱者從 /feed 取得的內容。\"],\"HxuOlm\":[\"網站頁首\"],\"I6gXOa\":[\"路徑\"],\"ID38tA\":[\"示範模式下帳號刪除已停用。共用示範會另行重置。\"],\"IF9tPu\":[\"何時使用網站匯出、資料庫備份與復原演練。\"],\"IW5PBo\":[\"複製權杖\"],\"IagCbF\":[\"URL\"],\"IreQBq\":[\"儲存庫\"],\"J6bLeg\":[\"新增自訂連結到任何 URL\"],\"JL7LF5\":[\"可用的 CSS 變數、data 屬性與範例.\"],\"JTviaO\":[\"管理登入安全、匯出,以及不可逆的操作。\"],\"JcD7qf\":[\"更多操作\"],\"JjX0OO\":[\"請立即複製您的權杖 — 它不會再顯示。\"],\"JrFTcr\":[\"連線中…\"],\"JuN5GC\":[\"未選取檔案。請選擇要上傳的檔案。\"],\"KDw4GX\":[\"重試\"],\"KSgo21\":[\"選擇一個儲存庫\"],\"KVVYBh\":[\"新增選集到導覽\"],\"KiJn9B\":[\"筆記\"],\"L3DEwT\":[\"移除這個頭像?您的 favicon 與頁首圖示會回復為預設。\"],\"L4t4/q\":[\"3月14日\"],\"LdyooL\":[\"連結\"],\"M/D8PK\":[\"+ 安裝到其他帳戶\"],\"M/haSd\":[\"永遠顯示主題的淺色版本。\"],\"M2kIWU\":[\"字型主題\"],\"M6CbAU\":[\"切換編輯面板\"],\"MaYYE6\":[\"透過向 Telegram 機器人傳送訊息來發佈筆記\"],\"Me5t5H\":[\"連接 GitHub 倉庫,自動將你的文章備份為 Markdown 檔案。GitHub 上的編輯會同步回你的網站。\"],\"Mr4QPw\":[\"要斷開 Telegram?您可以隨時使用新的綁定代碼重新連接。\"],\"MtENL9\":[\"調整您的網站外觀、可讀性與執行效能。\"],\"N/8NPV\":[\"在刪除前,請先下載網站匯出檔案。刪除後無法恢復此帳號。\"],\"N7UNHY\":[\"精選 RSS 來源\"],\"NHnUHF\":[\"頁首上的網站圖示與個人標記\"],\"NU2Fqi\":[\"儲存 CSS\"],\"Nldjdr\":[\"尚未有自訂 URL。建立一個自訂 URL 以為文章新增重新導向或自訂路徑。\"],\"O7rgs6\":[\"頁首 RSS 指向你的 \",[\"feed\"],\" 訂閱 (/feed)。在「一般」中變更 /feed 回傳的內容。\"],\"OSJXFg\":[\"套用於整個網站, 包括管理頁面. 選擇一個調色盤, 然後決定它是跟隨系統還是維持固定.\"],\"Ox3+3h\":[\"無相符結果。\"],\"PEUV5I\":[\"程式碼注入已更新。\"],\"PXj9lw\":[\"停止接受來自 Telegram 的貼文。您現有的筆記會保持已發佈。\"],\"PZ7HJ8\":[\"部落格大頭貼\"],\"Pwqkdw\":[\"載入中…\"],\"PxJ9W6\":[\"產生權杖\"],\"Q/6Y+2\":[\"需要在目標儲存庫上擁有 Contents(讀/寫)和 Webhooks(讀/寫)權限。\"],\"Q30z/l\":[\"要從導覽移除這個選集嗎?選集本身不會被刪除。\"],\"Q99OtV\":[\"將選集釘選到導覽列。最近 48 小時內更新的選集旁會顯示一個 * 號。\"],\"QZmz0H\":[\"內建連結\"],\"Qnrzvb\":[\"已啟用的權杖\"],\"R6Z4LE\":[\"下載失敗。請再試一次。\"],\"R9Khdg\":[\"自動\"],\"RcdDOS\":[\"在 Telegram 上向 @BotFather 發送訊息以建立機器人,然後貼上它提供給你的權杖\"],\"RxsRD6\":[\"時區\"],\"SJmfuf\":[\"網站名稱\"],\"SKZhW9\":[\"權杖名稱 (API 權杖名稱欄位。)\"],\"SVQQPe\":[\"無法連線。請檢查錯誤並再試一次。\"],\"SchpMp\":[\"Telegram\"],\"TpF3v+\":[\"在 </head> 之前注入。用於分析、自訂 meta 標籤,以及必須提前載入的樣式。\"],\"Tz0i8g\":[\"設定\"],\"UFK415\":[\"用於分析與小工具的網站全域 HTML\"],\"UTvFQq\":[\"開啟 \",[\"linkOpen\"],\"@\",[\"botUsername\"],[\"linkClose\"],\" 並傳送:\"],\"Uj/btJ\":[\"在我的網站頁首顯示大頭貼\"],\"UsODUn\":[\"選擇一個帳戶\"],\"UxKoFf\":[\"導覽\"],\"V+bhUy\":[\"安裝 GitHub App\"],\"V4WsyL\":[\"新增連結\"],\"V5pZwT\":[\"搜尋設定已更新。\"],\"VXUPla\":[\"使用 GitHub App 連線\"],\"VhMDMg\":[\"變更密碼\"],\"Vn3jYy\":[\"導覽項目\"],\"VoZYGU\":[\"這會永久刪除您所有的資料 — 文章、媒體、選集、設定,以及您的帳戶。您的部落格將被重設為初始設定狀態。此操作無法復原。\"],\"Weq9zb\":[\"一般\"],\"Wi9i06\":[\"依照每位訪客的系統偏好。\"],\"Wx1M8N\":[\"安裝 GitHub App,以授予存取權而無需管理個人權杖。權限以每個儲存庫為範圍,並可在 GitHub 上撤銷。\"],\"X+8FMk\":[\"目前的密碼不符。請再試一次。\"],\"X1G9eY\":[\"導覽預覽\"],\"X9Hujr\":[\"手動推送\"],\"XtBJV8\":[\"正在檢查儲存庫…\"],\"Xtc16w\":[\"重新整理儲存庫清單\"],\"Y/F35r\":[\"使用 curl 建立貼文:\"],\"YF6zHf\":[\"網站設定已更新.\"],\"YdG2RF\":[\"匯出網站\"],\"YkgZi7\":[\"連接一個 Telegram 機器人,之後你傳給它的任何訊息都會以筆記形式已發佈。\"],\"YwhjRx\":[\"管理帳戶\"],\"ZDY7Fy\":[\"同步中…\"],\"ZQKLI1\":[\"危險區域\"],\"ZS/CBL\":[\"刪除此導覽連結?訪客將不再在您的網站頁首看到它。\"],\"ZhhOwV\":[\"引用 (Jant 的貼文格式之一。)\"],\"ZiooJI\":[\"API 權杖\"],\"Zm7Qb0\":[\"備份與還原指南\"],\"ZmUkwN\":[\"新增自訂連結到導覽\"],\"a14mj8\":[\"未知裝置\"],\"a3LDKx\":[\"安全性\"],\"aAIQg2\":[\"外觀\"],\"aFkzVF\":[\"目標文章或選集的 slug\"],\"alKG0+\":[\"字型主題\"],\"anibOb\":[\"關於本部落格\"],\"any7NR\":[\"主題指南\"],\"b+/jO6\":[\"301 (永久)\"],\"bHOiy1\":[\"示範模式已停用變更密碼功能。請使用共用示範帳號登入。\"],\"bHYIks\":[\"登出\"],\"bmrL08\":[\"示範模式會隱藏會話、密碼更改與帳號刪除。匯出功能仍可使用。\"],\"c3MN2z\":[\"所有可用的端點與請求格式。\"],\"cS7/bk\":[\"移除已儲存的機器人權杖?其 webhook 會被刪除,任何已連接的帳號將會被斷線。\"],\"cSDy01\":[\"自訂 CSS 已更新.\"],\"clzoNp\":[\"始終顯示深色主題。\"],\"cnGeoo\":[\"刪除\"],\"d3FRkY\":[\"無法複製. 請再試一次.\"],\"d5oGUo\":[\"在 GitHub 建立新儲存庫\"],\"dEgA5A\":[\"取消\"],\"dTXUY+\":[\"確認刪除帳號\"],\"dYKrp3\":[\"從最新中隱藏 (不出現在 Latest 中,但仍可透過固定連結和集合存取的狀態。)\"],\"dk7TCH\":[\"永久刪除所有資料並重設部落格\"],\"drodVV\":[\"目前還沒有選集。請先建立一個,然後將它加入您的導覽。\"],\"dsWkIw\":[\"要與 GitHub 斷開連線嗎? webhook 將會被移除。您的儲存庫內容不會被刪除。\"],\"e/tSI5\":[\"導覽順序已更新。\"],\"ePK91l\":[\"編輯\"],\"ebQKK7\":[\"網站\"],\"egK+Yy\":[\"供腳本與自動化使用的 Bearer 權杖\"],\"ehj/zN\":[\"重新導向類型\"],\"eneWvv\":[\"草稿\"],\"erTMh7\":[\"上次同步\"],\"f+m8jj\":[\"已複製訂閱網址。\"],\"f8fH8W\":[\"設計\"],\"fWYqkz\":[\"程式碼注入\"],\"gOWiTY\":[\"載入為中文、日文或韓文內容最佳化的襯線字型。\"],\"gZ5owP\":[\"搜尋儲存庫\"],\"gbqbh6\":[\"可以放心離開此頁面 — 同步會在背景繼續進行。\"],\"gkFvVN\":[\"在 </body> 之前注入。用於聊天小工具和不應阻塞頁面載入的腳本。\"],\"gtQsRO\":[\"建立自訂網址\"],\"hBO/y4\":[\"安全權杖已過期。請重新整理頁面並再試一次。\"],\"hGmyDl\":[\"權杖讓您從腳本, 捷徑和其他工具存取 API 無需登入\"],\"hIHkRy\":[\"已透過 GitHub 應用程式連線\"],\"hdSi1b\":[\"輸入 \",[\"repo\"],\" 以確認\"],\"he3ygx\":[\"複製\"],\"i0qMbr\":[\"首頁\"],\"iEUzMn\":[\"系統\"],\"iSLIjg\":[\"連接\"],\"iVOMRi\":[\"首頁設定已更新。\"],\"icB4Cv\":[\"將連結拖到此處以顯示於「更多」選單下方\"],\"id3vuh\":[\"已設定 Telegram,但無法連線到機器人。請檢查機器人權杖並再試一次。\"],\"ihn4zD\":[\"搜尋…\"],\"iiDXZc\":[\"顯示於所有文章與頁面的底部。\"],\"j4VrG6\":[\"下載匯出 ZIP\"],\"j5nQL2\":[\"例如 iOS 捷徑\"],\"jUV7CU\":[\"上傳大頭貼\"],\"jVUmOK\":[\"支援 Markdown\"],\"jgBjXJ\":[\"要撤銷這個權杖嗎?任何使用它的腳本都會停止運作。\"],\"jpctdh\":[\"檢視\"],\"k1ifdL\":[\"處理中...\"],\"kMXclu\":[\"下載網站匯出檔案\"],\"kNiQp6\":[\"已釘選\"],\"kRhzWq\":[\"GitHub 同步\"],\"kVQs7s\":[\"細緻的樣式覆寫\"],\"ke1gWS\":[\"自訂 URL\"],\"kfcRb0\":[\"頭像\"],\"kxDZ2i\":[\"此程式碼會在您網站的每個頁面上執行。\"],\"l2Op2p\":[\"查詢參數\"],\"lLW3vJ\":[\"目標 slug\"],\"lYHJih\":[\"撤銷此工作階段?該裝置將需要重新登入。\"],\"mLOk1i\":[\"立即將所有文章推送到 GitHub,而不是等待下一次自動同步。\"],\"mSNmrX\":[\"列出貼文:\"],\"nK07ni\":[\"為您的網站選擇一種排版風格。每個主題會同時改變字體配對與閱讀節奏。\"],\"nbfdhU\":[\"第三方整合\"],\"ntJYyh\":[\"在 \",[\"providerLabel\"],\" 管理網域、方案和計費\"],\"o/vNDE\":[\"讓您覆寫任何主題變數。\"],\"oGC9uP\":[\"owner/repo\"],\"oH2JHg\":[\"我們會預先填入名稱 \",[\"name\"],\"。返回時清單會重新整理。\"],\"oKOOsY\":[\"色彩主題\"],\"oL535e\":[\"尚未同步\"],\"oNA4If\":[\"所有選集已經在您的導覽中。\"],\"pZq3aX\":[\"上傳失敗。請再試一次。\"],\"pgTIrt\":[\"選擇要與此網站同步的 GitHub 帳號和儲存庫.\"],\"psoxDF\":[\"該字型主題無法使用. 請選擇其他主題.\"],\"pvnfJD\":[\"深色\"],\"q+hNag\":[\"選集\"],\"qdcESc\":[\"建立新儲存庫\"],\"r5EW6f\":[\"此儲存庫已在備份另一個 Jant 網站 (\",[\"host\"],\"). 請選擇其他儲存庫.\"],\"rEspiY\":[\"導覽位置已更新。\"],\"rFmBG3\":[\"色彩主題\"],\"rlonmB\":[\"無法刪除。請稍後再試。\"],\"satWc6\":[\"主要 RSS 訂閱\"],\"sgr2wQ\":[\"選集\"],\"sqxcaY\":[\"建立於 \",[\"date\"]],\"sxkWRg\":[\"進階\"],\"t/YqKh\":[\"移除\"],\"t3hvHq\":[\"立即同步\"],\"tJ4H0O\":[\"您的 Telegram 帳號\"],\"tfDRzk\":[\"儲存\"],\"tvgAq5\":[\"尚未授權任何帳戶\"],\"u1VTd3\":[\"調色盤、表面色調與整體氛圍\"],\"u3wRF+\":[\"已發佈\"],\"u6KOjV\":[\"想要更細緻的控制?\"],\"udPwLB\":[\"頁首\"],\"ui6aMF\":[\"這些裝置目前已登入您的帳號。撤銷任何您不認識的工作階段。\"],\"vBEKwo\":[\"在此管理本網站的活動工作階段。密碼與託管存取由 \",[\"providerLabel\"],\" 管理。\"],\"vRldcl\":[\"字體選擇與閱讀質感\"],\"vSYKYI\":[\"主要訂閱來源\"],\"vTuib7\":[\"這會控制 /feed 回傳的內容。\"],\"vXIe7J\":[\"語言\"],\"vmQmHx\":[\"新增自訂 CSS 以覆寫任何樣式。使用像 [data-page]、[data-post]、[data-format] 這類資料屬性來選取特定元素。\"],\"vzX5FB\":[\"刪除帳號\"],\"w8Rv8T\":[\"標籤為必填\"],\"wL3cK8\":[\"最新\"],\"wPmHHc\":[\"低調的介面讓文字成為主角。\"],\"wW6NCp\":[\"上次錯誤\"],\"wc+17X\":[\"/* 在此放入您的自訂 CSS */\"],\"wuLtXn\":[\"目前沒有任何活動中的工作階段。已登入的裝置會顯示在此處。\"],\"xCWek4\":[\"檔案儲存尚未設定。請檢查您的伺服器設定。\"],\"xHt036\":[\"個人存取權杖\"],\"xbN8dp\":[\"柔和的色彩仍應保有清晰的閱讀節奏。\"],\"y28hnO\":[\"文章\"],\"y8Md/V\":[\"語言與時間已更新。\"],\"yNCqOt\":[\"最新 RSS 來源\"],\"yQ3kNF\":[\"請輸入以下短語以確認:\"],\"ydq1k2\":[\"請先選擇一個帳戶\"],\"yjjCV8\":[\"固定的 RSS 檔案網址\"],\"yjkELF\":[\"確認新密碼\"],\"yzF66j\":[\"連結 (Jant 的貼文格式之一。)\"],\"z6wakA\":[\"顯示在您的主頁上的簡短介紹。\"],\"zEizrk\":[\"最後使用於 \",[\"date\"]],\"zSURJW\":[\"沒有符合的儲存庫.\"],\"zXH2jX\":[\"語言與時區\"],\"zlcDd2\":[\"刪除此自訂 URL?使用該 URL 的訪客將不會再被重新導向。\"],\"zwBp5t\":[\"私有\"],\"zxRN6H\":[\"頁首連結、首頁動態與更多選單\"]}");
1926
1926
  //#endregion
1927
1927
  //#region src/i18n/i18n.ts
1928
1928
  /**
@@ -3419,10 +3419,10 @@ function normalizeThemeColorForMeta(color) {
3419
3419
  * internal paths (e.g. `/_assets/client-HASH.js`) embedded by the Worker build
3420
3420
  * from the Vite client manifest. Used only in production (IS_VITE_DEV=false).
3421
3421
  */ var IS_VITE_DEV = typeof __JANT_DEV__ !== "undefined" && __JANT_DEV__ === true;
3422
- var CORE_VERSION = "0.6.7-514e5b96a3f840fe";
3423
- var CLIENT_JS_FILE = "/_assets/client-B0MvB2r0.js";
3424
- var CLIENT_AUTH_JS_FILE = "/_assets/client-auth-CwwuucF_.js";
3425
- var CLIENT_CSS_FILE = "/_assets/client-BMPMuwvV.css";
3422
+ var CORE_VERSION = "0.6.8-49084ab7ac2a90bc";
3423
+ var CLIENT_JS_FILE = "/_assets/client-CXnEhyyv.js";
3424
+ var CLIENT_AUTH_JS_FILE = "/_assets/client-auth-CSItbyU8.js";
3425
+ var CLIENT_CSS_FILE = "/_assets/client-C6peCkkD.css";
3426
3426
  var CLIENT_CJK_CSS_FILE = "/_assets/client-cjk-B7Z0snDu.css";
3427
3427
  var CLIENT_CJK_TC_CSS_FILE = "/_assets/client-cjk-tc-BesJYrb2.css";
3428
3428
  var CLIENT_CJK_JP_CSS_FILE = "/_assets/client-cjk-jp-DZwrTzQC.css";
@@ -3740,7 +3740,7 @@ var IconSprite = () => {
3740
3740
  const cjkSerifFont = appConfig?.cjkSerifFont ?? "off";
3741
3741
  const cjkStylesheetPath = cjkSerifFont === "zh-Hans" ? IS_VITE_DEV ? assetPath("/src/style-cjk.css") : toPublicAssetPath(CLIENT_CJK_CSS_FILE, assetBasePath) : cjkSerifFont === "zh-Hant" ? IS_VITE_DEV ? assetPath("/src/style-cjk-tc.css") : toPublicAssetPath(CLIENT_CJK_TC_CSS_FILE, assetBasePath) : cjkSerifFont === "ja" ? IS_VITE_DEV ? assetPath("/src/style-cjk-jp.css") : toPublicAssetPath(CLIENT_CJK_JP_CSS_FILE, assetBasePath) : cjkSerifFont === "ko" ? IS_VITE_DEV ? assetPath("/src/style-cjk-kr.css") : toPublicAssetPath(CLIENT_CJK_KR_CSS_FILE, assetBasePath) : null;
3742
3742
  const clientScriptPath = IS_VITE_DEV ? resolvedClientBundle === "full" ? assetPath("/src/client-auth.ts") : assetPath("/src/client.ts") : toPublicAssetPath(resolvedClientBundle === "full" ? CLIENT_AUTH_JS_FILE : CLIENT_JS_FILE, assetBasePath);
3743
- const faviconAssetVersion = resolvedFaviconVersion || "0.6.7-514e5b96a3f840fe";
3743
+ const faviconAssetVersion = resolvedFaviconVersion || "0.6.8-49084ab7ac2a90bc";
3744
3744
  const resolvedFaviconHref = faviconHref ?? (faviconAssetVersion ? toPublicPath(`/favicon.ico?v=${faviconAssetVersion}`, sitePathPrefix) : toPublicPath("/favicon.ico", sitePathPrefix));
3745
3745
  const resolvedAppleTouchHref = appleTouchHref ?? (faviconAssetVersion ? toPublicPath(`/apple-touch-icon.png?v=${faviconAssetVersion}`, sitePathPrefix) : toPublicPath("/apple-touch-icon.png", sitePathPrefix));
3746
3746
  const socialImageHref = resolvedSocialImagePath ? toAbsoluteAssetUrl(resolvedSocialImagePath, appConfig?.siteUrl || "", sitePathPrefix) : "";
@@ -4355,7 +4355,7 @@ var STORAGE_DRIVERS = [
4355
4355
  envKeys: ["ASSET_BASE_URL"]
4356
4356
  },
4357
4357
  UPLOAD_MAX_FILE_SIZE_MB: {
4358
- defaultValue: "500",
4358
+ defaultValue: "1024",
4359
4359
  envOnly: true,
4360
4360
  envKeys: ["UPLOAD_MAX_FILE_SIZE_MB"]
4361
4361
  },
@@ -5798,6 +5798,11 @@ function getHostedControlPlaneAccountUrl(env) {
5798
5798
  function getHostedControlPlaneAccountPasswordUrl(env) {
5799
5799
  return buildHostedControlPlaneUrl(env, "/settings/account/password");
5800
5800
  }
5801
+ function getHostedControlPlaneSiteSettingsUrl(env, coreSiteId) {
5802
+ const normalizedCoreSiteId = coreSiteId.trim();
5803
+ if (!normalizedCoreSiteId) return null;
5804
+ return buildHostedControlPlaneUrl(env, `/sites/core/${encodeURIComponent(normalizedCoreSiteId)}/settings`);
5805
+ }
5801
5806
  function getHostedControlPlaneSiteDeleteUrl(env, coreSiteId) {
5802
5807
  const normalizedCoreSiteId = coreSiteId.trim();
5803
5808
  if (!normalizedCoreSiteId) return null;
@@ -6692,6 +6697,40 @@ function extractSearchTerms(query) {
6692
6697
  chars: media.chars ?? void 0
6693
6698
  };
6694
6699
  }
6700
+ /** Feed summary limits for titled, article-style posts (the excerpt is a teaser). */ var ARTICLE_SUMMARY_MAX_BLOCKS = 5;
6701
+ var ARTICLE_SUMMARY_MAX_CHARS = 500;
6702
+ /** Larger feed summary limits for untitled notes — the body itself is the content. */ var NOTE_SUMMARY_MAX_BLOCKS = 10;
6703
+ var NOTE_SUMMARY_MAX_CHARS = 1500;
6704
+ /** Don't truncate an untitled note just to hide a tail under this many chars. */ var NOTE_SUMMARY_MIN_HIDDEN_CHARS = 200;
6705
+ /**
6706
+ * Splice a zero-width marker into rendered body HTML at a summary boundary.
6707
+ *
6708
+ * The summary HTML is not a byte-prefix of `bodyHtml` — structural nodes
6709
+ * (horizontalRule, moreBreak, image) appear in `bodyHtml` but are excluded from
6710
+ * the summary, so slicing `bodyHtml` by summary length lands mid-tag and
6711
+ * corrupts the markup. Instead we render the pre-boundary doc slice and splice
6712
+ * the marker at that exact block boundary.
6713
+ *
6714
+ * @param bodyJson - Tiptap JSON string for the post body
6715
+ * @param bodyHtml - Rendered body HTML to splice the marker into
6716
+ * @param breakAtIndex - Index in `doc.content` where the post-summary content begins
6717
+ * @param markerHtml - Inert marker to insert at the boundary (e.g. an anchor span)
6718
+ * @returns `bodyHtml` with the marker inserted, or null when the split can't be
6719
+ * computed safely (caller should fall back to the untouched body)
6720
+ */ function spliceAtSummaryBoundary(bodyJson, bodyHtml, breakAtIndex, markerHtml) {
6721
+ try {
6722
+ const doc = JSON.parse(bodyJson);
6723
+ if (doc.type !== "doc" || !Array.isArray(doc.content) || breakAtIndex <= 0 || breakAtIndex > doc.content.length) return null;
6724
+ const beforeHtml = renderTiptapDocument({
6725
+ type: "doc",
6726
+ content: doc.content.slice(0, breakAtIndex)
6727
+ });
6728
+ if (!bodyHtml.startsWith(beforeHtml)) return null;
6729
+ return beforeHtml + markerHtml + bodyHtml.slice(beforeHtml.length);
6730
+ } catch {
6731
+ return null;
6732
+ }
6733
+ }
6695
6734
  function normalizePreviewText(text) {
6696
6735
  return (text ?? "").replace(/\s+/g, " ").trim() || void 0;
6697
6736
  }
@@ -6728,21 +6767,22 @@ function clipPreviewText(text, maxChars) {
6728
6767
  let summaryHtml;
6729
6768
  let summaryHasMore;
6730
6769
  let bodyHtmlWithAnchor = post.bodyHtml;
6731
- if (post.title && post.body) {
6732
- const result = extractSummaryHtml(post.body);
6733
- if (result) {
6770
+ if (post.body) {
6771
+ const isArticle = !!post.title;
6772
+ const result = extractSummaryHtml(post.body, isArticle ? ARTICLE_SUMMARY_MAX_BLOCKS : NOTE_SUMMARY_MAX_BLOCKS, isArticle ? ARTICLE_SUMMARY_MAX_CHARS : NOTE_SUMMARY_MAX_CHARS, isArticle ? 0 : NOTE_SUMMARY_MIN_HIDDEN_CHARS);
6773
+ if (result && isArticle) {
6734
6774
  summaryHtml = result.html;
6735
6775
  summaryHasMore = result.hasMore;
6736
- if (result.hasMore && post.bodyHtml) try {
6737
- const doc = JSON.parse(post.body);
6738
- if (doc.type === "doc" && Array.isArray(doc.content) && result.breakAtIndex > 0 && result.breakAtIndex <= doc.content.length) {
6739
- const beforeHtml = renderTiptapDocument({
6740
- type: "doc",
6741
- content: doc.content.slice(0, result.breakAtIndex)
6742
- });
6743
- if (post.bodyHtml.startsWith(beforeHtml)) bodyHtmlWithAnchor = beforeHtml + "<span id=\"continue\"></span>" + post.bodyHtml.slice(beforeHtml.length);
6744
- }
6745
- } catch {}
6776
+ if (result.hasMore && post.bodyHtml) {
6777
+ const spliced = spliceAtSummaryBoundary(post.body, post.bodyHtml, result.breakAtIndex, "<span id=\"continue\"></span>");
6778
+ if (spliced) bodyHtmlWithAnchor = spliced;
6779
+ }
6780
+ } else if (result && result.hasMore && post.bodyHtml) {
6781
+ const spliced = spliceAtSummaryBoundary(post.body, post.bodyHtml, result.breakAtIndex, "<span data-note-break></span>");
6782
+ if (spliced) {
6783
+ bodyHtmlWithAnchor = spliced;
6784
+ summaryHasMore = true;
6785
+ }
6746
6786
  }
6747
6787
  }
6748
6788
  const collections = (postCollections ?? []).map((c) => ({
@@ -6838,7 +6878,7 @@ function clipPreviewText(text, maxChars) {
6838
6878
  * @param item - Raw nav item from database
6839
6879
  * @param currentPath - Current URL path for active state
6840
6880
  * @param isAuthenticated - Whether the user is logged in (affects system settings item)
6841
- */ function toNavItemView(item, currentPath, homeDefaultView = "latest", isAuthenticated = false, sitePathPrefix = "", collectionFreshness) {
6881
+ */ function toNavItemView(item, currentPath, homeDefaultView = "latest", isAuthenticated = false, sitePathPrefix = "", collectionFreshness, siteOrigin = "") {
6842
6882
  let url = item.url;
6843
6883
  let label = item.label;
6844
6884
  if (item.type === "system" && item.systemKey) {
@@ -6851,8 +6891,9 @@ function clipPreviewText(text, maxChars) {
6851
6891
  if (!isAuthenticated) label = "Sign in";
6852
6892
  }
6853
6893
  }
6854
- const isExternal = url.startsWith("http://") || url.startsWith("https://");
6855
- const publicUrl = isExternal ? url : toPublicPath(url, sitePathPrefix);
6894
+ const sameSitePath = toSameSitePath(url, siteOrigin);
6895
+ const isExternal = sameSitePath === null && isFullUrl(url);
6896
+ const publicUrl = isExternal ? url : toPublicPath(sameSitePath ?? url, sitePathPrefix);
6856
6897
  let isActive = false;
6857
6898
  if (!isExternal) if (publicUrl === sitePathPrefix || publicUrl === "/") isActive = currentPath === (sitePathPrefix || "/");
6858
6899
  else isActive = currentPath === publicUrl || currentPath.startsWith(`${publicUrl}/`);
@@ -6878,8 +6919,8 @@ function clipPreviewText(text, maxChars) {
6878
6919
  * @param items - Raw nav items from database
6879
6920
  * @param currentPath - Current URL path for active state
6880
6921
  * @param isAuthenticated - Whether the user is logged in
6881
- */ function toNavItemViews(items, currentPath, homeDefaultView = "latest", isAuthenticated = false, sitePathPrefix = "", collectionFreshness) {
6882
- return items.map((item) => toNavItemView(item, currentPath, homeDefaultView, isAuthenticated, sitePathPrefix, collectionFreshness));
6922
+ */ function toNavItemViews(items, currentPath, homeDefaultView = "latest", isAuthenticated = false, sitePathPrefix = "", collectionFreshness, siteOrigin = "") {
6923
+ return items.map((item) => toNavItemView(item, currentPath, homeDefaultView, isAuthenticated, sitePathPrefix, collectionFreshness, siteOrigin));
6883
6924
  }
6884
6925
  /**
6885
6926
  * Converts a SearchResult to a SearchResultView with PostView.
@@ -7002,7 +7043,7 @@ function clipPreviewText(text, maxChars) {
7002
7043
  const collectionNavIds = [];
7003
7044
  for (const item of items) if (item.type === "collection" && item.collectionId) collectionNavIds.push(item.collectionId);
7004
7045
  const collectionFreshness = collectionNavIds.length > 0 ? await c.var.services.navItems.getCollectionFreshness(collectionNavIds) : void 0;
7005
- const links = toNavItemViews(items, currentPath, homeDefaultView, isAuthenticated, appConfig.sitePathPrefix, collectionFreshness);
7046
+ const links = toNavItemViews(items, currentPath, homeDefaultView, isAuthenticated, appConfig.sitePathPrefix, collectionFreshness, appConfig.siteOrigin);
7006
7047
  if (isAuthenticated) collections = await c.var.services.collections.listByRecentActivity();
7007
7048
  return {
7008
7049
  links,
@@ -7295,6 +7336,8 @@ var getCollectionMutationLabels = (i18n) => ({
7295
7336
  collectionCountLabel: i18n._({ id: "KlZ+t+" }),
7296
7337
  draftRestored: i18n._({ id: "DOx286" }),
7297
7338
  newThread: i18n._({ id: "yUtAh2" }),
7339
+ replyTitle: i18n._({ id: "ImOQa9" }),
7340
+ editTitle: i18n._({ id: "ePK91l" }),
7298
7341
  slashHint: i18n._({ id: "jrsUoG" }),
7299
7342
  collectionFormLabels: getCollectionFormLabels(i18n)
7300
7343
  }).replace(/</g, "\\u003c");
@@ -8587,7 +8630,7 @@ var INLINE_SIGNATURE_MIME_TYPES = new Set([
8587
8630
  * @returns null if valid, error message string if invalid
8588
8631
  * @example
8589
8632
  * ```ts
8590
- * const error = validateUploadFile(file, { maxFileSizeMB: 500 });
8633
+ * const error = validateUploadFile(file, { maxFileSizeMB: 1024 });
8591
8634
  * if (error) return dsToast(error, "error");
8592
8635
  * ```
8593
8636
  */ function validateUploadFile(file, options) {
@@ -8604,7 +8647,7 @@ var INLINE_SIGNATURE_MIME_TYPES = new Set([
8604
8647
  * @returns null if valid, error message string if invalid
8605
8648
  * @example
8606
8649
  * ```ts
8607
- * const error = validateUploadFileMetadata("image/jpeg", 1024000, { maxFileSizeMB: 500 });
8650
+ * const error = validateUploadFileMetadata("image/jpeg", 1024000, { maxFileSizeMB: 1024 });
8608
8651
  * ```
8609
8652
  */ function validateUploadFileMetadata(contentType, size, options) {
8610
8653
  if (options?.imagesOnly) {
@@ -9733,12 +9776,17 @@ function getContinueHref(post) {
9733
9776
  return !!post.threadRootId || !post.isLastInThread ? post.permalink : `${post.permalink}`;
9734
9777
  }
9735
9778
  var NoteCard = ({ post, mode = "feed", display }) => {
9779
+ const { i18n } = useLingui();
9736
9780
  const isCompact = mode === "compact";
9737
9781
  const isDetail = mode === "detail";
9738
9782
  const isArticle = !!post.title;
9739
9783
  const showFullBody = !isCompact && !isDetail && display?.showFullBody === true;
9740
9784
  const fullBodyHtml = showFullBody ? stripContinueAnchor(post.bodyHtml) : post.bodyHtml;
9741
- const displayHtml = isDetail || !isArticle || showFullBody ? fullBodyHtml : post.summaryHtml;
9785
+ const displayHtml = isDetail || showFullBody ? fullBodyHtml : post.summaryHtml ?? fullBodyHtml;
9786
+ const continueLabel = i18n._({ id: "xjHB3/" });
9787
+ const readMoreLabel = i18n._({ id: "Qoq+GP" });
9788
+ const readLessLabel = i18n._({ id: "C4TjpG" });
9789
+ const clampNote = !isArticle && !isDetail && !isCompact && !showFullBody && post.summaryHasMore === true;
9742
9790
  const hasVisibleRating = !!post.rating && post.rating > 0 && !display?.hideRating;
9743
9791
  const showHeaderRating = isDetail && isArticle && hasVisibleRating;
9744
9792
  const footerDisplay = isDetail && isArticle && display?.footer?.hideTimestamp === void 0 ? {
@@ -9787,6 +9835,7 @@ var NoteCard = ({ post, mode = "feed", display }) => {
9787
9835
  displayHtml && /* @__PURE__ */ jsxDEV$1("div", {
9788
9836
  class: `e-content prose ${isCompact ? "prose-sm" : isDetail || showFullBody ? "post-detail-body" : isArticle ? "post-body-summary" : ""}`,
9789
9837
  "data-post-body": true,
9838
+ ...clampNote ? { "data-note-clamp": "" } : {},
9790
9839
  dangerouslySetInnerHTML: { __html: displayHtml }
9791
9840
  }),
9792
9841
  (() => {
@@ -9799,11 +9848,19 @@ var NoteCard = ({ post, mode = "feed", display }) => {
9799
9848
  postPermalink: post.permalink
9800
9849
  })
9801
9850
  }),
9802
- !isDetail && !isCompact && !showFullBody && isArticle && post.summaryHasMore && /* @__PURE__ */ jsxDEV$1("a", {
9851
+ !isDetail && !isCompact && !showFullBody && post.summaryHasMore && (isArticle ? /* @__PURE__ */ jsxDEV$1("a", {
9803
9852
  href: getContinueHref(post),
9804
9853
  class: "feed-continue-link",
9805
- children: "Continue →"
9806
- }),
9854
+ children: continueLabel
9855
+ }) : /* @__PURE__ */ jsxDEV$1("a", {
9856
+ href: getContinueHref(post),
9857
+ class: "feed-continue-link",
9858
+ "data-note-expand": true,
9859
+ "aria-expanded": "false",
9860
+ "data-label-more": readMoreLabel,
9861
+ "data-label-less": readLessLabel,
9862
+ children: readMoreLabel
9863
+ })),
9807
9864
  !isCompact && !showHeaderRating && !display?.hideRating && /* @__PURE__ */ jsxDEV$1(StarRating, { rating: post.rating }),
9808
9865
  /* @__PURE__ */ jsxDEV$1(PostFooter, {
9809
9866
  post,
@@ -12142,6 +12199,10 @@ function createMediaService(db, siteId, databaseSchema = sqliteSchemaBundle, dat
12142
12199
  if (ids.length === 0) return [];
12143
12200
  return (await db.select().from(media).where(and(eq(media.siteId, siteId), inArray(media.id, ids)))).map(toMedia);
12144
12201
  },
12202
+ async listOrphanedMediaIds({ before, limit }) {
12203
+ if (limit <= 0) return [];
12204
+ return (await db.select({ id: media.id }).from(media).where(and(eq(media.siteId, siteId), isNull(media.postId), lt(media.createdAt, before))).orderBy(asc(media.createdAt), asc(media.id)).limit(limit)).map((r) => r.id);
12205
+ },
12145
12206
  async getByPostId(postId) {
12146
12207
  return (await db.select().from(media).where(and(eq(media.siteId, siteId), eq(media.postId, postId))).orderBy(asc(media.position))).map(toMedia);
12147
12208
  },
@@ -12392,9 +12453,10 @@ function createMediaService(db, siteId, databaseSchema = sqliteSchemaBundle, dat
12392
12453
  if (merged.collectionSlug) params.set("collection", merged.collectionSlug);
12393
12454
  if (merged.format) params.set("format", merged.format);
12394
12455
  if (merged.mediaKinds && merged.mediaKinds.length > 0) params.set("media", merged.mediaKinds.join(","));
12395
- if (merged.hasMedia !== void 0) params.set("hasMedia", merged.hasMedia ? "1" : "0");
12396
- if (merged.hasTitle !== void 0) params.set("hasTitle", merged.hasTitle ? "1" : "0");
12397
- if (merged.visibility) params.set("visibility", merged.visibility);
12456
+ else if (merged.hasMedia !== void 0) params.set("media", merged.hasMedia ? "any" : "none");
12457
+ if (merged.hasTitle !== void 0) params.set("title", merged.hasTitle ? "any" : "none");
12458
+ if (merged.hasReplies !== void 0) params.set("replies", merged.hasReplies ? "any" : "none");
12459
+ if (merged.visibility) params.set("visibility", merged.visibility === "latest_hidden" ? "hidden" : merged.visibility);
12398
12460
  if (merged.view && merged.view !== "grid") params.set("view", merged.view);
12399
12461
  const qs = params.toString();
12400
12462
  return qs ? toPublicPath(`/archive?${qs}`, sitePathPrefix) : toPublicPath("/archive", sitePathPrefix);
@@ -12666,9 +12728,9 @@ var ViewToggle = ({ filters, sitePathPrefix = "" }) => {
12666
12728
  };
12667
12729
  var ARCHIVE_VISIBILITIES = [
12668
12730
  "public",
12731
+ "featured",
12669
12732
  "latest_hidden",
12670
- "private",
12671
- "featured"
12733
+ "private"
12672
12734
  ];
12673
12735
  function getVisibilityLabel(v) {
12674
12736
  const { i18n } = useLingui();
@@ -12691,8 +12753,13 @@ var FEATURED_VISIBILITY_ICON_HTML = getFeaturedIconSvg({ className: "icon-fine"
12691
12753
  collection: "monitor",
12692
12754
  format: "shapes",
12693
12755
  media: "video",
12756
+ thread: "git-branch",
12694
12757
  visibility: "scan-eye"
12695
12758
  };
12759
+ /** Icons for the thread filter options. */ var THREAD_ICONS = {
12760
+ threads: "list-tree",
12761
+ single: "git-commit-horizontal"
12762
+ };
12696
12763
  var FilterBar = ({ filters, availableYears, availableCollections, isAuthenticated, sitePathPrefix = "" }) => {
12697
12764
  const { i18n } = useLingui();
12698
12765
  const currentUrl = buildFilterUrl(filters, {}, sitePathPrefix);
@@ -12786,6 +12853,31 @@ var FilterBar = ({ filters, availableYears, availableCollections, isAuthenticate
12786
12853
  ...v === "featured" ? { iconHtml: FEATURED_VISIBILITY_ICON_HTML } : { icon: VISIBILITY_ICONS[v] },
12787
12854
  value: buildFilterUrl(filters, { visibility: v }, sitePathPrefix)
12788
12855
  }))];
12856
+ const threadClearUrl = buildFilterUrl({
12857
+ ...filters,
12858
+ hasReplies: void 0
12859
+ }, { hasReplies: void 0 }, sitePathPrefix);
12860
+ const threadsLabel = i18n._({ id: "FqCHF/" });
12861
+ const singlePostsLabel = i18n._({ id: "hqeXKW" });
12862
+ const threadOptions = [
12863
+ {
12864
+ label: i18n._({ id: "HrC0ab" }),
12865
+ icon: FILTER_ICONS.thread,
12866
+ value: threadClearUrl
12867
+ },
12868
+ {
12869
+ label: threadsLabel,
12870
+ icon: THREAD_ICONS.threads,
12871
+ value: buildFilterUrl(filters, { hasReplies: true }, sitePathPrefix)
12872
+ },
12873
+ {
12874
+ label: singlePostsLabel,
12875
+ icon: THREAD_ICONS.single,
12876
+ value: buildFilterUrl(filters, { hasReplies: false }, sitePathPrefix)
12877
+ }
12878
+ ];
12879
+ const threadActiveLabel = filters.hasReplies === true ? threadsLabel : filters.hasReplies === false ? singlePostsLabel : void 0;
12880
+ const threadActiveIcon = filters.hasReplies === true ? THREAD_ICONS.threads : filters.hasReplies === false ? THREAD_ICONS.single : void 0;
12789
12881
  const activeKinds = filters.mediaKinds ?? [];
12790
12882
  const mediaActiveLabel = filters.hasMedia === false ? i18n._({ id: "xeiujy" }) : activeKinds.length === 1 ? getMediaKindLabel(activeKinds[0]) : activeKinds.length > 1 ? String(activeKinds.length) : void 0;
12791
12883
  const mediaClearUrl = buildFilterUrl({
@@ -12845,6 +12937,16 @@ var FilterBar = ({ filters, availableYears, availableCollections, isAuthenticate
12845
12937
  activeIcon: formatActiveIcon,
12846
12938
  iconOnly: true
12847
12939
  }),
12940
+ /* @__PURE__ */ jsxDEV$1(ChipSelect, {
12941
+ id: "af-thread",
12942
+ icon: FILTER_ICONS.thread,
12943
+ options: threadOptions,
12944
+ currentValue: currentUrl,
12945
+ clearUrl: threadClearUrl,
12946
+ activeLabel: threadActiveLabel,
12947
+ activeIcon: threadActiveIcon,
12948
+ iconOnly: true
12949
+ }),
12848
12950
  /* @__PURE__ */ jsxDEV$1(ChipMediaSelect, {
12849
12951
  id: "af-media",
12850
12952
  icon: FILTER_ICONS.media,
@@ -13422,19 +13524,26 @@ ${entries.map((entry) => {
13422
13524
  const year = yearParam ? parseInt(yearParam, 10) : void 0;
13423
13525
  const validYear = year && !isNaN(year) && year > 1970 ? year : void 0;
13424
13526
  const collectionSlug = q("collection") || void 0;
13527
+ const parsePresence = (param, legacy) => {
13528
+ if (param === "any") return true;
13529
+ if (param === "none") return false;
13530
+ if (legacy === "1") return true;
13531
+ if (legacy === "0") return false;
13532
+ };
13425
13533
  const mediaParam = q("media") || void 0;
13426
- const mediaKinds = mediaParam ? mediaParam.split(",").filter((m) => MEDIA_KINDS.includes(m)) : void 0;
13427
- const hasMediaParam = q("hasMedia");
13428
- const hasMedia = hasMediaParam === "1" ? true : hasMediaParam === "0" ? false : void 0;
13429
- const hasTitleParam = q("hasTitle");
13430
- const hasTitle = hasTitleParam === "1" ? true : hasTitleParam === "0" ? false : void 0;
13534
+ const mediaIsPresence = mediaParam === "any" || mediaParam === "none";
13535
+ const mediaKinds = mediaParam && !mediaIsPresence ? mediaParam.split(",").filter((m) => MEDIA_KINDS.includes(m)) : void 0;
13536
+ const hasMedia = parsePresence(mediaIsPresence ? mediaParam : void 0, q("hasMedia"));
13537
+ const hasTitle = parsePresence(q("title"), q("hasTitle"));
13538
+ const hasReplies = parsePresence(q("replies"), q("hasReplies"));
13431
13539
  const VALID_VISIBILITIES = [
13432
13540
  "public",
13433
13541
  "latest_hidden",
13434
13542
  "private",
13435
13543
  "featured"
13436
13544
  ];
13437
- const visibilityParam = q("visibility");
13545
+ const rawVisibilityParam = q("visibility");
13546
+ const visibilityParam = rawVisibilityParam === "hidden" ? "latest_hidden" : rawVisibilityParam;
13438
13547
  const visibilityAll = visibilityParam === "all";
13439
13548
  const visibility = visibilityParam && VALID_VISIBILITIES.includes(visibilityParam) ? visibilityParam : void 0;
13440
13549
  const viewParam = q("view");
@@ -13448,6 +13557,7 @@ ${entries.map((entry) => {
13448
13557
  mediaKinds: mediaKinds && mediaKinds.length > 0 ? mediaKinds : void 0,
13449
13558
  hasMedia,
13450
13559
  hasTitle,
13560
+ hasReplies,
13451
13561
  visibility,
13452
13562
  visibilityAll,
13453
13563
  view,
@@ -13481,7 +13591,8 @@ ${entries.map((entry) => {
13481
13591
  publishedBefore,
13482
13592
  mediaKinds: params.mediaKinds,
13483
13593
  hasMedia: params.hasMedia,
13484
- hasTitle: params.hasTitle
13594
+ hasTitle: params.hasTitle,
13595
+ hasReplies: params.hasReplies
13485
13596
  };
13486
13597
  }
13487
13598
  /**
@@ -13493,13 +13604,49 @@ ${entries.map((entry) => {
13493
13604
  if (params.validYear) qs.set("year", String(params.validYear));
13494
13605
  if (params.collectionSlug) qs.set("collection", params.collectionSlug);
13495
13606
  if (params.mediaKinds && params.mediaKinds.length > 0) qs.set("media", params.mediaKinds.join(","));
13496
- if (params.hasMedia !== void 0) qs.set("hasMedia", params.hasMedia ? "1" : "0");
13497
- if (params.hasTitle !== void 0) qs.set("hasTitle", params.hasTitle ? "1" : "0");
13607
+ else if (params.hasMedia !== void 0) qs.set("media", params.hasMedia ? "any" : "none");
13608
+ if (params.hasTitle !== void 0) qs.set("title", params.hasTitle ? "any" : "none");
13609
+ if (params.hasReplies !== void 0) qs.set("replies", params.hasReplies ? "any" : "none");
13498
13610
  const str = qs.toString();
13499
13611
  return str ? `?${str}` : "";
13500
13612
  }
13501
13613
  var archiveRoutes = new Hono();
13502
13614
  /**
13615
+ * Build a canonical redirect target when a request uses legacy archive
13616
+ * param spellings (hasMedia/hasTitle/hasReplies=1/0, visibility=latest_hidden).
13617
+ *
13618
+ * Only legacy params are rewritten; everything else (including unknown
13619
+ * params) is preserved. Returns null when the URL is already canonical.
13620
+ * Applies to the /archive page only — feeds and the public API accept
13621
+ * legacy spellings silently, and custom archive URLs (path_registry
13622
+ * query overrides) never reach this path.
13623
+ *
13624
+ * @param c - Hono context
13625
+ * @returns Canonical path + query to redirect to, or null
13626
+ */ function legacyArchiveParamsRedirect(c) {
13627
+ const url = new URL(c.req.url);
13628
+ const params = url.searchParams;
13629
+ let changed = false;
13630
+ for (const [legacy, name] of [
13631
+ ["hasMedia", "media"],
13632
+ ["hasTitle", "title"],
13633
+ ["hasReplies", "replies"]
13634
+ ]) {
13635
+ const value = params.get(legacy);
13636
+ if (value === null) continue;
13637
+ if (!params.has(name) && (value === "1" || value === "0")) params.set(name, value === "1" ? "any" : "none");
13638
+ params.delete(legacy);
13639
+ changed = true;
13640
+ }
13641
+ if (params.get("visibility") === "latest_hidden") {
13642
+ params.set("visibility", "hidden");
13643
+ changed = true;
13644
+ }
13645
+ if (!changed) return null;
13646
+ const qs = params.toString();
13647
+ return `${url.pathname}${qs ? `?${qs}` : ""}`;
13648
+ }
13649
+ /**
13503
13650
  * Render the archive page. Used by both the `/archive` route and custom
13504
13651
  * archive URLs resolved through the path registry.
13505
13652
  *
@@ -13576,6 +13723,7 @@ var archiveRoutes = new Hono();
13576
13723
  mediaKinds: params.mediaKinds,
13577
13724
  hasMedia: params.hasMedia,
13578
13725
  hasTitle: params.hasTitle,
13726
+ hasReplies: params.hasReplies,
13579
13727
  visibility: effectiveVisibility,
13580
13728
  view: params.view
13581
13729
  };
@@ -13603,7 +13751,11 @@ var archiveRoutes = new Hono();
13603
13751
  })
13604
13752
  });
13605
13753
  }
13606
- archiveRoutes.get("/", (c) => renderArchivePage(c));
13754
+ archiveRoutes.get("/", (c) => {
13755
+ const canonical = legacyArchiveParamsRedirect(c);
13756
+ if (canonical) return c.redirect(canonical, 308);
13757
+ return renderArchivePage(c);
13758
+ });
13607
13759
  /**
13608
13760
  * Build a descriptive feed title from active filters.
13609
13761
  *
@@ -13628,6 +13780,8 @@ archiveRoutes.get("/", (c) => renderArchivePage(c));
13628
13780
  else if (params.hasTitle === true) parts.push(i18n._({ id: "IjnQHI" }));
13629
13781
  if (params.hasMedia === true) parts.push(i18n._({ id: "ogssnn" }));
13630
13782
  else if (params.hasMedia === false) parts.push(i18n._({ id: "4HLTdq" }));
13783
+ if (params.hasReplies === true) parts.push(i18n._({ id: "avuFKG" }));
13784
+ else if (params.hasReplies === false) parts.push(i18n._({ id: "dHko2w" }));
13631
13785
  if (params.validYear) parts.push(String(params.validYear));
13632
13786
  const archiveLabel = i18n._({ id: "B495Gs" });
13633
13787
  if (parts.length === 0) return `${siteName} - ${archiveLabel}`;
@@ -13647,6 +13801,7 @@ async function buildArchiveFeedData(c, selfPath) {
13647
13801
  mediaKinds: params.mediaKinds,
13648
13802
  hasMedia: params.hasMedia,
13649
13803
  hasTitle: params.hasTitle,
13804
+ hasReplies: params.hasReplies,
13650
13805
  ...params.validYear ? {
13651
13806
  publishedAfter: Date.UTC(params.validYear, 0, 1) / 1e3,
13652
13807
  publishedBefore: Date.UTC(params.validYear + 1, 0, 1) / 1e3
@@ -15041,7 +15196,7 @@ var hasDirectoryContent = (items) => items.some((item) => item.type === "collect
15041
15196
  } else labels.push("");
15042
15197
  return labels;
15043
15198
  };
15044
- var CollectionDirectory = ({ items, emptyMessage, sitePathPrefix = "" }) => {
15199
+ var CollectionDirectory = ({ items, emptyMessage, sitePathPrefix = "", siteOrigin = "" }) => {
15045
15200
  const { i18n } = useLingui();
15046
15201
  if (!hasDirectoryContent(items)) return /* @__PURE__ */ jsxDEV$1("p", {
15047
15202
  class: "text-muted-foreground",
@@ -15072,7 +15227,9 @@ var CollectionDirectory = ({ items, emptyMessage, sitePathPrefix = "" }) => {
15072
15227
  }
15073
15228
  if (item.type === "link" && item.label && item.url) {
15074
15229
  const sequence = sequenceLabels[index];
15075
- const isExternal = item.url.startsWith("http://") || item.url.startsWith("https://");
15230
+ const sameSitePath = toSameSitePath(item.url, siteOrigin);
15231
+ const linkHref = sameSitePath !== null ? toPublicHref(sameSitePath, sitePathPrefix) : toPublicHref(item.url, sitePathPrefix);
15232
+ const isExternal = sameSitePath === null && (item.url.startsWith("http://") || item.url.startsWith("https://"));
15076
15233
  return /* @__PURE__ */ jsxDEV$1("div", {
15077
15234
  class: "collection-directory-item collection-directory-item-link",
15078
15235
  children: /* @__PURE__ */ jsxDEV$1("div", {
@@ -15086,7 +15243,7 @@ var CollectionDirectory = ({ items, emptyMessage, sitePathPrefix = "" }) => {
15086
15243
  /* @__PURE__ */ jsxDEV$1("div", {
15087
15244
  class: "collection-directory-title-row",
15088
15245
  children: /* @__PURE__ */ jsxDEV$1("a", {
15089
- href: toPublicHref(item.url, sitePathPrefix),
15246
+ href: linkHref,
15090
15247
  class: "collection-directory-title-link",
15091
15248
  ...isExternal ? {
15092
15249
  target: "_blank",
@@ -15187,7 +15344,7 @@ var CollectionDirectory = ({ items, emptyMessage, sitePathPrefix = "" }) => {
15187
15344
  //#endregion
15188
15345
  //#region src/ui/shared/CollectionsManager.tsx
15189
15346
  var escapeJson = (data) => JSON.stringify(data).replace(/</g, "\\u003c");
15190
- var CollectionsManager = ({ items, sitePathPrefix = "" }) => {
15347
+ var CollectionsManager = ({ items, sitePathPrefix = "", siteOrigin = "" }) => {
15191
15348
  const { i18n } = useLingui();
15192
15349
  const collectionsHref = toPublicPath(getCollectionsDirectoryPath(), sitePathPrefix);
15193
15350
  const newCollectionHref = toPublicPath(`${getNewCollectionPath()}?returnTo=${encodeURIComponent(collectionsHref)}`, sitePathPrefix);
@@ -15399,7 +15556,8 @@ var CollectionsManager = ({ items, sitePathPrefix = "" }) => {
15399
15556
  children: /* @__PURE__ */ jsxDEV$1(CollectionDirectory, {
15400
15557
  items,
15401
15558
  emptyMessage: labels.emptyState,
15402
- sitePathPrefix
15559
+ sitePathPrefix,
15560
+ siteOrigin
15403
15561
  })
15404
15562
  })]
15405
15563
  });
@@ -15410,7 +15568,7 @@ var CollectionsManager = ({ items, sitePathPrefix = "" }) => {
15410
15568
  * Collections Listing Page
15411
15569
  *
15412
15570
  * Single-column directory of collections.
15413
- */ var CollectionsPage = ({ items, isAuthenticated, sitePathPrefix = "" }) => {
15571
+ */ var CollectionsPage = ({ items, isAuthenticated, sitePathPrefix = "", siteOrigin = "" }) => {
15414
15572
  const { i18n } = useLingui();
15415
15573
  const emptyMessage = i18n._({ id: "kr39oD" });
15416
15574
  if (isAuthenticated) return /* @__PURE__ */ jsxDEV$1("div", {
@@ -15418,7 +15576,8 @@ var CollectionsManager = ({ items, sitePathPrefix = "" }) => {
15418
15576
  "data-page": "collections",
15419
15577
  children: /* @__PURE__ */ jsxDEV$1(CollectionsManager, {
15420
15578
  items,
15421
- sitePathPrefix
15579
+ sitePathPrefix,
15580
+ siteOrigin
15422
15581
  })
15423
15582
  });
15424
15583
  return /* @__PURE__ */ jsxDEV$1("div", {
@@ -15441,7 +15600,8 @@ var CollectionsManager = ({ items, sitePathPrefix = "" }) => {
15441
15600
  }), /* @__PURE__ */ jsxDEV$1(CollectionDirectory, {
15442
15601
  items,
15443
15602
  emptyMessage,
15444
- sitePathPrefix
15603
+ sitePathPrefix,
15604
+ siteOrigin
15445
15605
  })]
15446
15606
  })
15447
15607
  });
@@ -15480,7 +15640,8 @@ collectionsPageRoutes.get("/", async (c) => {
15480
15640
  content: /* @__PURE__ */ jsxDEV$1(CollectionsPage, {
15481
15641
  items: directoryData.items,
15482
15642
  isAuthenticated: navData.isAuthenticated ?? false,
15483
- sitePathPrefix: navData.sitePathPrefix
15643
+ sitePathPrefix: navData.sitePathPrefix,
15644
+ siteOrigin: c.var.appConfig.siteOrigin
15484
15645
  })
15485
15646
  });
15486
15647
  });
@@ -18291,6 +18452,26 @@ function ChevronRight() {
18291
18452
  children: /* @__PURE__ */ jsxDEV$1("path", { d: "m9 18 6-6-6-6" })
18292
18453
  });
18293
18454
  }
18455
+ function ExternalLinkIndicator() {
18456
+ return /* @__PURE__ */ jsxDEV$1("svg", {
18457
+ class: "settings-directory-item-chevron",
18458
+ xmlns: "http://www.w3.org/2000/svg",
18459
+ width: "16",
18460
+ height: "16",
18461
+ viewBox: "0 0 24 24",
18462
+ fill: "none",
18463
+ stroke: "currentColor",
18464
+ "stroke-width": "2",
18465
+ "stroke-linecap": "round",
18466
+ "stroke-linejoin": "round",
18467
+ "aria-hidden": "true",
18468
+ children: [
18469
+ /* @__PURE__ */ jsxDEV$1("path", { d: "M15 3h6v6" }),
18470
+ /* @__PURE__ */ jsxDEV$1("path", { d: "M10 14 21 3" }),
18471
+ /* @__PURE__ */ jsxDEV$1("path", { d: "M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6" })
18472
+ ]
18473
+ });
18474
+ }
18294
18475
  function SettingsDirectorySection({ title, tone = "default", children }) {
18295
18476
  return /* @__PURE__ */ jsxDEV$1("section", {
18296
18477
  class: "settings-directory-section",
@@ -18304,7 +18485,7 @@ function SettingsDirectorySection({ title, tone = "default", children }) {
18304
18485
  })]
18305
18486
  });
18306
18487
  }
18307
- function SettingsDirectoryItemContent({ icon, name, description }) {
18488
+ function SettingsDirectoryItemContent({ icon, name, description, external = false }) {
18308
18489
  return /* @__PURE__ */ jsxDEV$1(Fragment$1, { children: [
18309
18490
  /* @__PURE__ */ jsxDEV$1("span", {
18310
18491
  class: "settings-directory-item-icon",
@@ -18320,7 +18501,7 @@ function SettingsDirectoryItemContent({ icon, name, description }) {
18320
18501
  children: description
18321
18502
  })]
18322
18503
  }),
18323
- /* @__PURE__ */ jsxDEV$1(ChevronRight, {})
18504
+ external ? /* @__PURE__ */ jsxDEV$1(ExternalLinkIndicator, {}) : /* @__PURE__ */ jsxDEV$1(ChevronRight, {})
18324
18505
  ] });
18325
18506
  }
18326
18507
  function SettingsDirectoryLink({ href, icon, tone = "default", name, description, target, rel }) {
@@ -18333,7 +18514,8 @@ function SettingsDirectoryLink({ href, icon, tone = "default", name, description
18333
18514
  children: /* @__PURE__ */ jsxDEV$1(SettingsDirectoryItemContent, {
18334
18515
  icon,
18335
18516
  name,
18336
- description
18517
+ description,
18518
+ external: target === "_blank"
18337
18519
  })
18338
18520
  });
18339
18521
  }
@@ -18354,11 +18536,13 @@ function SettingsDirectoryLink({ href, icon, tone = "default", name, description
18354
18536
  key: `<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="m15.5 7.5 2.3 2.3a1 1 0 0 0 1.4 0l2.1-2.1a1 1 0 0 0 0-1.4L19 4"/><path d="m21 2-9.6 9.6"/><circle cx="7.5" cy="15.5" r="5.5"/></svg>`,
18355
18537
  shield: `<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M20 13c0 5-3.5 7.5-7.66 8.95a1 1 0 0 1-.67-.01C7.5 20.5 4 18 4 13V6a1 1 0 0 1 1-1c2 0 4.5-1.2 6.24-2.72a1.17 1.17 0 0 1 1.52 0C14.51 3.81 17 5 19 5a1 1 0 0 1 1 1z"/></svg>`,
18356
18538
  gitBranch: `<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><line x1="6" x2="6" y1="3" y2="15"/><circle cx="18" cy="6" r="3"/><circle cx="6" cy="18" r="3"/><path d="M18 9a9 9 0 0 1-9 9"/></svg>`,
18357
- send: `<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M14.536 21.686a.5.5 0 0 0 .937-.024l6.5-19a.496.496 0 0 0-.635-.635l-19 6.5a.5.5 0 0 0-.024.937l7.93 3.18a2 2 0 0 1 1.112 1.11z"/><path d="m21.854 2.147-10.94 10.939"/></svg>`
18539
+ send: `<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M14.536 21.686a.5.5 0 0 0 .937-.024l6.5-19a.496.496 0 0 0-.635-.635l-19 6.5a.5.5 0 0 0-.024.937l7.93 3.18a2 2 0 0 1 1.112 1.11z"/><path d="m21.854 2.147-10.94 10.939"/></svg>`,
18540
+ cloud: `<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M17.5 19H9a7 7 0 1 1 6.71-9h1.79a4.5 4.5 0 1 1 0 9Z"/></svg>`
18358
18541
  };
18359
- function SettingsRootContent({ sitePathPrefix = "", demoMode = false }) {
18542
+ function SettingsRootContent({ sitePathPrefix = "", demoMode = false, hostedControlPlaneSiteSettingsUrl, hostedControlPlaneProviderLabel }) {
18360
18543
  const { i18n } = useLingui();
18361
18544
  const accountDescription = demoMode ? i18n._({ id: "GzKzUa" }) : i18n._({ id: "2B7t+s" });
18545
+ const hostingProviderLabel = hostedControlPlaneSiteSettingsUrl ? coalesceDisplayText(hostedControlPlaneProviderLabel, extractDomain(hostedControlPlaneSiteSettingsUrl)) ?? i18n._({ id: "8BfEpW" }) : null;
18362
18546
  return /* @__PURE__ */ jsxDEV$1("div", {
18363
18547
  class: "settings-root",
18364
18548
  children: [
@@ -18459,13 +18643,21 @@ function SettingsRootContent({ sitePathPrefix = "", demoMode = false }) {
18459
18643
  }),
18460
18644
  /* @__PURE__ */ jsxDEV$1(SettingsDirectorySection, {
18461
18645
  title: i18n._({ id: "AeXO77" }),
18462
- children: /* @__PURE__ */ jsxDEV$1(SettingsDirectoryLink, {
18646
+ children: [/* @__PURE__ */ jsxDEV$1(SettingsDirectoryLink, {
18463
18647
  href: toPublicPath("/settings/account", sitePathPrefix),
18464
18648
  icon: ICONS$1.shield,
18465
18649
  tone: "subtle",
18466
18650
  name: i18n._({ id: "AeXO77" }),
18467
18651
  description: accountDescription
18468
- })
18652
+ }), hostedControlPlaneSiteSettingsUrl && hostingProviderLabel ? /* @__PURE__ */ jsxDEV$1(SettingsDirectoryLink, {
18653
+ href: hostedControlPlaneSiteSettingsUrl,
18654
+ target: "_blank",
18655
+ rel: "noopener noreferrer",
18656
+ icon: ICONS$1.cloud,
18657
+ tone: "subtle",
18658
+ name: i18n._({ id: "6E3aK4" }),
18659
+ description: i18n._({ id: "ntJYyh" }, { providerLabel: hostingProviderLabel })
18660
+ }) : null]
18469
18661
  }),
18470
18662
  /* @__PURE__ */ jsxDEV$1("div", {
18471
18663
  class: "settings-root-signout",
@@ -18633,7 +18825,7 @@ var ICONS = {
18633
18825
  book: `<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M12 7v14"/><path d="M3 18V5a2 2 0 0 1 2-2h6v18H5a2 2 0 0 1-2-2Z"/><path d="M21 18V5a2 2 0 0 0-2-2h-7v18h7a2 2 0 0 0 2-2Z"/></svg>`,
18634
18826
  trash: `<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M3 6h18"/><path d="M19 6v14c0 1-1 2-2 2H7c-1 0-2-1-2-2V6"/><path d="M8 6V4c0-1 1-2 2-2h4c1 0 2 1 2 2v2"/></svg>`
18635
18827
  };
18636
- function AccountMenuContent({ sitePathPrefix = "", demoMode = false, hostedControlPlaneAccountUrl, hostedControlPlaneProviderLabel, hostedControlPlaneSiteDeleteUrl }) {
18828
+ function AccountMenuContent({ sitePathPrefix = "", demoMode = false, hostedControlPlaneAccountUrl, hostedControlPlaneProviderLabel }) {
18637
18829
  const { i18n } = useLingui();
18638
18830
  const isHosted = Boolean(hostedControlPlaneAccountUrl);
18639
18831
  const providerLabel = coalesceDisplayText(hostedControlPlaneProviderLabel, hostedControlPlaneAccountUrl ? extractDomain(hostedControlPlaneAccountUrl) : void 0) ?? i18n._({ id: "8BfEpW" });
@@ -18711,17 +18903,6 @@ function AccountMenuContent({ sitePathPrefix = "", demoMode = false, hostedContr
18711
18903
  description: i18n._({ id: "IF9tPu" })
18712
18904
  })]
18713
18905
  }),
18714
- !demoMode && isHosted && hostedControlPlaneSiteDeleteUrl ? /* @__PURE__ */ jsxDEV$1(SettingsDirectorySection, {
18715
- title: i18n._({ id: "ZQKLI1" }),
18716
- tone: "danger",
18717
- children: /* @__PURE__ */ jsxDEV$1(SettingsDirectoryLink, {
18718
- href: hostedControlPlaneSiteDeleteUrl,
18719
- icon: ICONS.trash,
18720
- tone: "danger",
18721
- name: i18n._({ id: "69OXZB" }),
18722
- description: i18n._({ id: "7z05Pf" }, { providerLabel })
18723
- })
18724
- }) : null,
18725
18906
  !demoMode && !isHosted && /* @__PURE__ */ jsxDEV$1(SettingsDirectorySection, {
18726
18907
  title: i18n._({ id: "ZQKLI1" }),
18727
18908
  tone: "danger",
@@ -21013,7 +21194,7 @@ function parseConfigInt(value, fallback) {
21013
21194
  s3PublicUrl,
21014
21195
  localPublicUrl,
21015
21196
  imageTransformUrl,
21016
- uploadMaxFileSize: parseInt(getEnvString(env, "UPLOAD_MAX_FILE_SIZE_MB") ?? "500", 10) || 500,
21197
+ uploadMaxFileSize: parseInt(getEnvString(env, "UPLOAD_MAX_FILE_SIZE_MB") ?? "1024", 10) || 1024,
21017
21198
  summaryMaxParagraphs: parseInt(getEnvString(env, "SUMMARY_MAX_PARAGRAPHS") ?? "5", 10) || 5,
21018
21199
  summaryMaxChars: parseInt(getEnvString(env, "SUMMARY_MAX_CHARS") ?? "500", 10) || 500,
21019
21200
  slugIdLength: parseInt(getEnvString(env, "SLUG_ID_LENGTH") ?? "5", 10) || 5,
@@ -21180,8 +21361,8 @@ async function syncHostedControlPlaneSiteAvatar(input) {
21180
21361
  return;
21181
21362
  }
21182
21363
  await markSyncPending(settings);
21183
- const { createGitHubSyncService } = await import("./github-sync-BeDecPen.js");
21184
- const { getGitHubAppConfig } = await import("./env-CoSe-1y4.js").then((n) => n.t);
21364
+ const { createGitHubSyncService } = await import("./github-sync-_kPWM4m9.js");
21365
+ const { getGitHubAppConfig } = await import("./env-OHRKGcMj.js").then((n) => n.t);
21185
21366
  const run = runBackgroundSync(settings, createGitHubSyncService(c.var.services, c.var.currentSite.id, await buildSyncSiteConfig(c), {
21186
21367
  storage: c.var.storage,
21187
21368
  githubApp: getGitHubAppConfig(c.env)
@@ -21389,6 +21570,8 @@ function demoRestrictionResponse(c, message) {
21389
21570
  }
21390
21571
  settingsRoutes.get("/", async (c) => {
21391
21572
  const navData = await getNavigationData(c);
21573
+ const hostedControlPlaneSiteSettingsUrl = getHostedControlPlaneSiteSettingsUrl(c.env, c.var.currentSite.id);
21574
+ const hostedControlPlaneProviderLabel = getHostedControlPlaneProviderLabel(c.env);
21392
21575
  return renderPublicPage(c, {
21393
21576
  title: buildPageTitle("Settings", navData.siteName),
21394
21577
  navData,
@@ -21396,7 +21579,9 @@ settingsRoutes.get("/", async (c) => {
21396
21579
  class: "py-6",
21397
21580
  children: /* @__PURE__ */ jsxDEV$1(SettingsRootContent, {
21398
21581
  sitePathPrefix: c.var.appConfig.sitePathPrefix,
21399
- demoMode: c.var.appConfig.demoMode
21582
+ demoMode: c.var.appConfig.demoMode,
21583
+ hostedControlPlaneSiteSettingsUrl,
21584
+ hostedControlPlaneProviderLabel
21400
21585
  })
21401
21586
  })
21402
21587
  });
@@ -21776,7 +21961,6 @@ settingsRoutes.get("/account", async (c) => {
21776
21961
  const navData = await getNavigationData(c);
21777
21962
  const hostedControlPlaneAccountUrl = getHostedControlPlaneAccountUrl(c.env);
21778
21963
  const hostedControlPlaneProviderLabel = getHostedControlPlaneProviderLabel(c.env);
21779
- const hostedControlPlaneSiteDeleteUrl = getHostedControlPlaneSiteDeleteUrl(c.env, c.var.currentSite.id);
21780
21964
  return renderPublicPage(c, {
21781
21965
  title: buildPageTitle("Account", navData.siteName),
21782
21966
  navData,
@@ -21788,8 +21972,7 @@ settingsRoutes.get("/account", async (c) => {
21788
21972
  sitePathPrefix: c.var.appConfig.sitePathPrefix,
21789
21973
  demoMode: c.var.appConfig.demoMode,
21790
21974
  hostedControlPlaneAccountUrl,
21791
- hostedControlPlaneProviderLabel,
21792
- hostedControlPlaneSiteDeleteUrl
21975
+ hostedControlPlaneProviderLabel
21793
21976
  })] })
21794
21977
  });
21795
21978
  });
@@ -21949,7 +22132,7 @@ settingsRoutes.post("/github-sync/connect", async (c) => {
21949
22132
  if (getGitHubAppConfig(c.env)) return dsToast("This deployment uses GitHub App authentication. Use Install GitHub App instead.", "error");
21950
22133
  const body = await c.req.json();
21951
22134
  if (!body.token?.trim() || !body.repo?.trim()) return dsToast("Token and repository are required.", "error");
21952
- const { parseRepoSlug, createGitHubClient } = await import("./github-api-UD4u_7fa.js").then((n) => n.n);
22135
+ const { parseRepoSlug, createGitHubClient } = await import("./github-api-BgSiE71w.js").then((n) => n.n);
21953
22136
  const parsed = parseRepoSlug(body.repo);
21954
22137
  if (!parsed) return dsToast("Invalid repository format. Use owner/repo.", "error");
21955
22138
  const client = createGitHubClient(body.token);
@@ -21968,7 +22151,7 @@ settingsRoutes.post("/github-sync/connect", async (c) => {
21968
22151
  await c.var.services.settings.set("GITHUB_SYNC_AUTH_MODE", "pat");
21969
22152
  await c.var.services.settings.set("GITHUB_SYNC_APP_INSTALLATION_ID", "");
21970
22153
  await c.var.services.settings.set("GITHUB_SYNC_ENABLED", "true");
21971
- const { createGitHubSyncService } = await import("./github-sync-BeDecPen.js");
22154
+ const { createGitHubSyncService } = await import("./github-sync-_kPWM4m9.js");
21972
22155
  const syncService = createGitHubSyncService(c.var.services, c.var.currentSite.id, await buildSyncSiteConfig(c), {
21973
22156
  storage: c.var.storage,
21974
22157
  githubApp: getGitHubAppConfig(c.env)
@@ -21987,7 +22170,7 @@ settingsRoutes.post("/github-sync/connect", async (c) => {
21987
22170
  return dsRedirect(publicPath(c, "/settings/github-sync"));
21988
22171
  });
21989
22172
  settingsRoutes.post("/github-sync/push", async (c) => {
21990
- const { createGitHubSyncService } = await import("./github-sync-BeDecPen.js");
22173
+ const { createGitHubSyncService } = await import("./github-sync-_kPWM4m9.js");
21991
22174
  const syncService = createGitHubSyncService(c.var.services, c.var.currentSite.id, await buildSyncSiteConfig(c), {
21992
22175
  storage: c.var.storage,
21993
22176
  githubApp: getGitHubAppConfig(c.env)
@@ -22007,7 +22190,7 @@ settingsRoutes.post("/github-sync/push", async (c) => {
22007
22190
  });
22008
22191
  });
22009
22192
  settingsRoutes.post("/github-sync/disconnect", async (c) => {
22010
- const { createGitHubSyncService } = await import("./github-sync-BeDecPen.js");
22193
+ const { createGitHubSyncService } = await import("./github-sync-_kPWM4m9.js");
22011
22194
  await createGitHubSyncService(c.var.services, c.var.currentSite.id, await buildSyncSiteConfig(c), { githubApp: getGitHubAppConfig(c.env) }).teardownWebhook();
22012
22195
  return dsRedirect(publicPath(c, "/settings/github-sync"));
22013
22196
  });
@@ -22195,10 +22378,10 @@ function buildRepoPickerLabels(c) {
22195
22378
  const repo = String(body.repo ?? "").trim();
22196
22379
  const confirmForeign = body.confirmForeign === true || body.confirmForeign === "true";
22197
22380
  if (!installationId || !repo) return wantsJson ? c.json({ error: "Missing installationId or repo." }, 400) : c.text("Missing installationId or repo.", 400);
22198
- const { parseRepoSlug, createGitHubClient } = await import("./github-api-UD4u_7fa.js").then((n) => n.n);
22381
+ const { parseRepoSlug, createGitHubClient } = await import("./github-api-BgSiE71w.js").then((n) => n.n);
22199
22382
  const parsed = parseRepoSlug(repo);
22200
22383
  if (!parsed) return wantsJson ? c.json({ error: "Invalid repository format." }, 400) : c.text("Invalid repository format.", 400);
22201
- const { classifyRepoForSync } = await import("./github-sync-BeDecPen.js");
22384
+ const { classifyRepoForSync } = await import("./github-sync-_kPWM4m9.js");
22202
22385
  const ghClient = createGitHubClient(() => getInstallationTokenFromApp(app, installationId));
22203
22386
  let classification;
22204
22387
  try {
@@ -22228,7 +22411,7 @@ function buildRepoPickerLabels(c) {
22228
22411
  await c.var.services.settings.set("GITHUB_SYNC_REPO", repo);
22229
22412
  await c.var.services.settings.set("GITHUB_SYNC_TOKEN", "");
22230
22413
  await c.var.services.settings.set("GITHUB_SYNC_ENABLED", "true");
22231
- const { createGitHubSyncService } = await import("./github-sync-BeDecPen.js");
22414
+ const { createGitHubSyncService } = await import("./github-sync-_kPWM4m9.js");
22232
22415
  const syncService = createGitHubSyncService(c.var.services, c.var.currentSite.id, await buildSyncSiteConfig(c), {
22233
22416
  storage: c.var.storage,
22234
22417
  githubApp: app
@@ -22267,7 +22450,7 @@ function requireGitHubApp(c) {
22267
22450
  * build a client for classification without adding the helper to the
22268
22451
  * top-level imports (the module already lazy-imports github-api).
22269
22452
  */ async function getInstallationTokenFromApp(app, installationId) {
22270
- const { getInstallationToken } = await import("./github-app-DeX6Td1O.js").then((n) => n.i);
22453
+ const { getInstallationToken } = await import("./github-app-BbklkFmU.js").then((n) => n.i);
22271
22454
  return getInstallationToken(app, installationId);
22272
22455
  }
22273
22456
  /** List GitHub App installations authorized for this site. */ settingsRoutes.get("/github-sync/app/installations", async (c) => {
@@ -22341,10 +22524,10 @@ function requireGitHubApp(c) {
22341
22524
  const installationId = String(body.installationId ?? "").trim();
22342
22525
  const repo = String(body.repo ?? "").trim();
22343
22526
  if (!installationId || !repo) return c.json({ error: "Missing installationId or repo." }, 400);
22344
- const { parseRepoSlug, createGitHubClient } = await import("./github-api-UD4u_7fa.js").then((n) => n.n);
22527
+ const { parseRepoSlug, createGitHubClient } = await import("./github-api-BgSiE71w.js").then((n) => n.n);
22345
22528
  const parsed = parseRepoSlug(repo);
22346
22529
  if (!parsed) return c.json({ error: "Invalid repository format." }, 400);
22347
- const { classifyRepoForSync } = await import("./github-sync-BeDecPen.js");
22530
+ const { classifyRepoForSync } = await import("./github-sync-_kPWM4m9.js");
22348
22531
  const client = createGitHubClient(() => getInstallationTokenFromApp(app, installationId));
22349
22532
  try {
22350
22533
  const classification = await classifyRepoForSync(client, parsed.owner, parsed.repo, c.var.currentSite.id);
@@ -22825,7 +23008,7 @@ function NewCustomUrlContent({ sitePathPrefix = "" }) {
22825
23008
  type: "text",
22826
23009
  "data-bind": "archiveQuery",
22827
23010
  class: "input",
22828
- placeholder: "format=note&hasTitle=0&visibility=public&view=list"
23011
+ placeholder: "format=note&title=none&visibility=public&view=list"
22829
23012
  }),
22830
23013
  /* @__PURE__ */ jsxDEV$1("p", {
22831
23014
  class: "text-xs text-muted-foreground mt-1",
@@ -26005,6 +26188,7 @@ var CreateManagedSiteSchema = z.object({
26005
26188
  idempotencyKey: z.string().trim().min(1).max(128).optional()
26006
26189
  });
26007
26190
  var SitePostCountsSchema = z.object({ siteIds: z.array(z.string().trim().min(1)).max(200) });
26191
+ var CleanupSiteUploadsSchema = z.object({ limit: z.number().int().positive().max(500).optional() });
26008
26192
  var ManagedSiteDomainSchema = z.object({
26009
26193
  host: z.string().trim().toLowerCase().min(1).max(255).regex(/^(?=.{1,255}$)(?:[a-z0-9](?:[a-z0-9-]{0,61}[a-z0-9])?\.)+[a-z0-9](?:[a-z0-9-]{0,61}[a-z0-9])?$/, "Domain host must be a valid hostname."),
26010
26194
  makePrimary: z.boolean().optional()
@@ -26049,6 +26233,18 @@ internalSitesRoutes.get("/:siteId/media-usage", requireInternalAdminApi(), async
26049
26233
  const usage = await c.var.services.siteAdmin.getManagedSiteMediaUsage(c.req.param("siteId"));
26050
26234
  return c.json(usage);
26051
26235
  });
26236
+ internalSitesRoutes.post("/:siteId/uploads/cleanup", requireInternalAdminApi(), async (c) => {
26237
+ assertHostBasedMode(c.env);
26238
+ const storage = c.var.storage;
26239
+ if (!storage) return c.json({ error: "File storage isn't set up. Check your server config." }, 500);
26240
+ const body = parseValidated(CleanupSiteUploadsSchema, (c.req.header("Content-Type") || "").includes("application/json") ? await c.req.json().catch(() => ({})) : {});
26241
+ const result = await c.var.servicesForSite(c.req.param("siteId")).uploads.cleanupExpired({
26242
+ storage,
26243
+ storageDriver: getConfiguredStorageDriver(c.env),
26244
+ limit: body.limit
26245
+ });
26246
+ return c.json(result);
26247
+ });
26052
26248
  internalSitesRoutes.get("/:siteId/export", requireInternalAdminApi(), async (c) => {
26053
26249
  assertHostBasedMode(c.env);
26054
26250
  const archive = await c.var.services.siteAdmin.exportManagedSite(c.req.param("siteId"), {
@@ -26665,7 +26861,7 @@ function escapeLinkUrl(url) {
26665
26861
  * the media flow needs are env-driven anyway, so read them straight from the
26666
26862
  * bindings without rebuilding the full appConfig.
26667
26863
  */ function uploadConfigFromEnv(env) {
26668
- const maxFileSizeMB = parseInt(getEnvString(env, "UPLOAD_MAX_FILE_SIZE_MB") ?? "500", 10) || 500;
26864
+ const maxFileSizeMB = parseInt(getEnvString(env, "UPLOAD_MAX_FILE_SIZE_MB") ?? "1024", 10) || 1024;
26669
26865
  return {
26670
26866
  storageDriver: getConfiguredStorageDriver(env),
26671
26867
  maxFileSizeMB
@@ -27277,26 +27473,30 @@ publicPostsApiRoutes.get("/:slug", async (c) => {
27277
27473
  //#region src/routes/api/public/archive.ts
27278
27474
  var publicArchiveApiRoutes = new Hono();
27279
27475
  var MediaKindSchema = z.enum(MEDIA_KINDS);
27280
- var INVALID_MEDIA_KIND_MESSAGE = "Invalid media kind. Allowed: " + MEDIA_KINDS.join(", ");
27476
+ var INVALID_MEDIA_MESSAGE = "Invalid media value. Allowed: any, none, or kinds: " + MEDIA_KINDS.join(", ");
27281
27477
  var BoolFlagSchema = z.enum(["0", "1"]).transform((value) => value === "1");
27478
+ var PresenceSchema = z.enum(["any", "none"]).transform((value) => value === "any");
27282
27479
  var ListPublicArchiveQuerySchema = z.object({
27283
27480
  format: FormatSchema.optional(),
27284
27481
  collection: z.string().optional(),
27285
27482
  year: z.coerce.number().int().min(1971).optional(),
27286
27483
  media: z.string().optional().transform((value, ctx) => {
27287
27484
  if (!value) return void 0;
27485
+ if (value === "any" || value === "none") return value;
27288
27486
  const parts = value.split(",").map((part) => part.trim()).filter((part) => part.length > 0);
27289
27487
  if (parts.length === 0) return void 0;
27290
27488
  const result = z.array(MediaKindSchema).safeParse(parts);
27291
27489
  if (!result.success) {
27292
27490
  ctx.addIssue({
27293
27491
  code: z.ZodIssueCode.custom,
27294
- message: INVALID_MEDIA_KIND_MESSAGE
27492
+ message: INVALID_MEDIA_MESSAGE
27295
27493
  });
27296
27494
  return z.NEVER;
27297
27495
  }
27298
27496
  return result.data;
27299
27497
  }),
27498
+ title: PresenceSchema.optional(),
27499
+ replies: PresenceSchema.optional(),
27300
27500
  hasMedia: BoolFlagSchema.optional(),
27301
27501
  hasTitle: BoolFlagSchema.optional(),
27302
27502
  cursor: z.string().optional(),
@@ -27304,7 +27504,9 @@ var ListPublicArchiveQuerySchema = z.object({
27304
27504
  content: z.enum(["markdown"]).optional()
27305
27505
  });
27306
27506
  publicArchiveApiRoutes.get("/", async (c) => {
27307
- const { format, collection, year, media, hasMedia, hasTitle, cursor, limit, content } = parseValidated(ListPublicArchiveQuerySchema, c.req.query());
27507
+ const { format, collection, year, media, title, replies, hasMedia, hasTitle, cursor, limit, content } = parseValidated(ListPublicArchiveQuerySchema, c.req.query());
27508
+ const mediaKinds = Array.isArray(media) ? media : void 0;
27509
+ const mediaPresence = media === "any" ? true : media === "none" ? false : void 0;
27308
27510
  let collectionIds;
27309
27511
  if (collection) {
27310
27512
  const slugExpression = collection.replace(/,/g, "+");
@@ -27328,9 +27530,10 @@ publicArchiveApiRoutes.get("/", async (c) => {
27328
27530
  excludeReplies: true,
27329
27531
  publishedAfter,
27330
27532
  publishedBefore,
27331
- mediaKinds: media,
27332
- hasMedia,
27333
- hasTitle
27533
+ mediaKinds,
27534
+ hasMedia: mediaPresence ?? hasMedia,
27535
+ hasTitle: title ?? hasTitle,
27536
+ hasReplies: replies
27334
27537
  });
27335
27538
  const postIds = posts.map((post) => post.id);
27336
27539
  const [mediaMap, collectionsMap] = await Promise.all([c.var.services.media.getByPostIds(postIds), c.var.services.collections.getCollectionsByPostIds(postIds)]);
@@ -29759,7 +29962,7 @@ function createPostService(db, config, siteId, paths, databaseSchema = sqliteSch
29759
29962
  }
29760
29963
  const effectiveVisibilityExpr = sql`coalesce(
29761
29964
  ${posts.visibility},
29762
- (SELECT root.visibility FROM post AS root WHERE root.id = ${posts.threadId})
29965
+ (SELECT root.visibility FROM post AS root WHERE root.id = ${posts.threadId} AND root.site_id = ${siteId})
29763
29966
  )`;
29764
29967
  /** Check if a slug is available (not used by posts or path_registry) */ async function isSlugAvailable(slug) {
29765
29968
  return resolvedPaths.isPathAvailable(slug);
@@ -29873,6 +30076,17 @@ function createPostService(db, config, siteId, paths, databaseSchema = sqliteSch
29873
30076
  if (filters.hasTitle !== void 0) if (filters.hasTitle) conditions.push(sql`${posts.title} IS NOT NULL AND ${posts.title} != ''`);
29874
30077
  else conditions.push(sql`(${posts.title} IS NULL OR ${posts.title} = '')`);
29875
30078
  if (filters.hasRating !== void 0) conditions.push(filters.hasRating ? isNotNull(posts.rating) : isNull(posts.rating));
30079
+ if (filters.hasReplies !== void 0) {
30080
+ const repliesExist = sql`EXISTS (
30081
+ SELECT 1
30082
+ FROM post AS reply
30083
+ WHERE reply.site_id = ${siteId}
30084
+ AND reply.thread_id = ${posts.id}
30085
+ AND reply.reply_to_id IS NOT NULL
30086
+ AND reply.status = 'published'
30087
+ )`;
30088
+ conditions.push(filters.hasReplies ? repliesExist : sql`NOT ${repliesExist}`);
30089
+ }
29876
30090
  return conditions;
29877
30091
  }
29878
30092
  async function getLastLivePostIdInThread(threadId) {
@@ -30086,7 +30300,7 @@ function createPostService(db, config, siteId, paths, databaseSchema = sqliteSch
30086
30300
  return result;
30087
30301
  }
30088
30302
  function buildThreadRootPageConditions(options) {
30089
- const conditions = [];
30303
+ const conditions = [eq(posts.siteId, siteId)];
30090
30304
  const status = options?.status;
30091
30305
  if (status) conditions.push(eq(posts.status, status));
30092
30306
  if (options?.excludePrivate) conditions.push(sql`${effectiveVisibilityExpr} != 'private'`);
@@ -31948,7 +32162,7 @@ function createSearchService(rawQuery, siteId, databaseDialect = "sqlite") {
31948
32162
  snippet(post_fts, 1, char(2), char(3), '...', 32) AS snippet
31949
32163
  FROM post_fts
31950
32164
  JOIN post ON post.rowid = post_fts.rowid
31951
- JOIN post AS root_post ON root_post.id = post.thread_id
32165
+ JOIN post AS root_post ON root_post.id = post.thread_id AND root_post.site_id = post.site_id
31952
32166
  JOIN path_registry
31953
32167
  ON path_registry.post_id = post.id
31954
32168
  AND path_registry.site_id = post.site_id
@@ -32012,7 +32226,7 @@ function createSearchService(rawQuery, siteId, databaseDialect = "sqlite") {
32012
32226
  ) AS snippet
32013
32227
  FROM post
32014
32228
  CROSS JOIN search_query
32015
- JOIN post AS root_post ON root_post.id = post.thread_id
32229
+ JOIN post AS root_post ON root_post.id = post.thread_id AND root_post.site_id = post.site_id
32016
32230
  JOIN path_registry
32017
32231
  ON path_registry.post_id = post.id
32018
32232
  AND path_registry.site_id = post.site_id
@@ -32048,7 +32262,7 @@ function createSearchService(rawQuery, siteId, databaseDialect = "sqlite") {
32048
32262
  ) AS rank,
32049
32263
  NULL AS snippet
32050
32264
  FROM post
32051
- JOIN post AS root_post ON root_post.id = post.thread_id
32265
+ JOIN post AS root_post ON root_post.id = post.thread_id AND root_post.site_id = post.site_id
32052
32266
  JOIN path_registry
32053
32267
  ON path_registry.post_id = post.id
32054
32268
  AND path_registry.site_id = post.site_id
@@ -32071,7 +32285,7 @@ function createSearchService(rawQuery, siteId, databaseDialect = "sqlite") {
32071
32285
  0 AS rank,
32072
32286
  NULL AS snippet
32073
32287
  FROM post
32074
- JOIN post AS root_post ON root_post.id = post.thread_id
32288
+ JOIN post AS root_post ON root_post.id = post.thread_id AND root_post.site_id = post.site_id
32075
32289
  JOIN path_registry
32076
32290
  ON path_registry.post_id = post.id
32077
32291
  AND path_registry.site_id = post.site_id
@@ -32789,7 +33003,7 @@ function createSiteAdminService(db, databaseSchema = sqliteSchemaBundle, databas
32789
33003
  const themeCss = buildThemeStyle(activeTheme, appConfig.themeMode, fontOverrides);
32790
33004
  const navItemList = await navItems.list();
32791
33005
  const appleTouchKey = allSettings[SETTINGS_KEYS.SITE_FAVICON_APPLE_TOUCH];
32792
- const { createExportService } = await import("./export-DLukCOO3.js").then((n) => n.n);
33006
+ const { createExportService } = await import("./export-Be082J0n.js").then((n) => n.n);
32793
33007
  const exportService = createExportService({
32794
33008
  collections,
32795
33009
  media: mediaService,
@@ -33017,6 +33231,7 @@ var RELAY_MULTIPART_THRESHOLD = 95 * 1024 * 1024;
33017
33231
  var RELAY_MULTIPART_PART_SIZE = 50 * 1024 * 1024;
33018
33232
  var IMMUTABLE_CACHE_CONTROL = "public, max-age=31536000, immutable";
33019
33233
  var DEFAULT_EXPIRED_UPLOAD_CLEANUP_LIMIT = 20;
33234
+ var ORPHAN_MEDIA_GRACE_SECONDS = 10080 * 60;
33020
33235
  var CLEANUPABLE_UPLOAD_SESSION_STATES = [
33021
33236
  "pending",
33022
33237
  "uploaded",
@@ -33311,9 +33526,15 @@ function createUploadSessionService(db, siteId, media, databaseSchema = sqliteSc
33311
33526
  await db.delete(uploadSessions).where(and(eq(uploadSessions.siteId, siteId), eq(uploadSessions.id, session.id)));
33312
33527
  deletedSessions += 1;
33313
33528
  }
33529
+ const orphanIds = await media.listOrphanedMediaIds({
33530
+ before: now() - ORPHAN_MEDIA_GRACE_SECONDS,
33531
+ limit
33532
+ });
33533
+ if (orphanIds.length > 0) await media.deleteByIds(orphanIds, deps.storage);
33314
33534
  return {
33315
33535
  abortedMultipartUploads,
33316
- deletedSessions
33536
+ deletedSessions,
33537
+ deletedOrphanMedia: orphanIds.length
33317
33538
  };
33318
33539
  }
33319
33540
  };