@jant/core 0.3.24 → 0.3.26

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (277) hide show
  1. package/dist/app.js +101 -571
  2. package/dist/client.js +1 -0
  3. package/dist/db/schema.js +1 -1
  4. package/dist/i18n/locales/en.js +1 -1
  5. package/dist/i18n/locales/zh-Hans.js +1 -1
  6. package/dist/i18n/locales/zh-Hant.js +1 -1
  7. package/dist/index.js +3 -9
  8. package/dist/lib/avatar-upload.js +134 -0
  9. package/dist/lib/config.js +39 -0
  10. package/dist/lib/constants.js +10 -9
  11. package/dist/lib/favicon.js +102 -0
  12. package/dist/lib/image.js +13 -17
  13. package/dist/lib/media-helpers.js +2 -2
  14. package/dist/lib/nav-reorder.js +1 -1
  15. package/dist/lib/navigation.js +48 -3
  16. package/dist/lib/pagination.js +44 -0
  17. package/dist/lib/render.js +16 -11
  18. package/dist/lib/schemas.js +34 -3
  19. package/dist/lib/theme.js +4 -4
  20. package/dist/lib/timeline.js +24 -48
  21. package/dist/lib/timezones.js +388 -0
  22. package/dist/lib/view.js +3 -3
  23. package/dist/routes/api/collections.js +124 -0
  24. package/dist/routes/api/nav-items.js +104 -0
  25. package/dist/routes/api/pages.js +91 -0
  26. package/dist/routes/api/posts.js +3 -3
  27. package/dist/routes/api/search.js +2 -2
  28. package/dist/routes/api/settings.js +68 -0
  29. package/dist/routes/api/upload.js +3 -3
  30. package/dist/routes/auth/reset.js +221 -0
  31. package/dist/routes/auth/setup.js +194 -0
  32. package/dist/routes/auth/signin.js +176 -0
  33. package/dist/routes/compose.js +48 -0
  34. package/dist/routes/dash/collections.js +24 -416
  35. package/dist/routes/dash/index.js +1 -1
  36. package/dist/routes/dash/media.js +13 -393
  37. package/dist/routes/dash/pages.js +112 -86
  38. package/dist/routes/dash/posts.js +3 -5
  39. package/dist/routes/dash/redirects.js +20 -14
  40. package/dist/routes/dash/settings.js +213 -518
  41. package/dist/routes/feed/rss.js +4 -3
  42. package/dist/routes/feed/sitemap.js +5 -3
  43. package/dist/routes/pages/archive.js +3 -6
  44. package/dist/routes/pages/collection.js +3 -6
  45. package/dist/routes/pages/collections.js +28 -0
  46. package/dist/routes/pages/featured.js +36 -0
  47. package/dist/routes/pages/home.js +33 -49
  48. package/dist/routes/pages/latest.js +45 -0
  49. package/dist/routes/pages/page.js +29 -32
  50. package/dist/routes/pages/post.js +3 -6
  51. package/dist/routes/pages/search.js +3 -6
  52. package/dist/services/page.js +5 -1
  53. package/dist/services/post.js +45 -31
  54. package/dist/services/search.js +1 -1
  55. package/dist/types/bindings.js +3 -0
  56. package/dist/types/config.js +147 -0
  57. package/dist/types/constants.js +27 -0
  58. package/dist/types/entities.js +3 -0
  59. package/dist/types/operations.js +3 -0
  60. package/dist/types/props.js +3 -0
  61. package/dist/types/views.js +5 -0
  62. package/dist/types.js +8 -111
  63. package/dist/{theme → ui}/color-themes.js +33 -33
  64. package/dist/ui/compose/ComposeDialog.js +467 -0
  65. package/dist/ui/compose/ComposePrompt.js +55 -0
  66. package/dist/{theme/components/TypeBadge.js → ui/dash/FormatBadge.js} +1 -2
  67. package/dist/{theme/components → ui/dash}/PageForm.js +21 -15
  68. package/dist/{theme/components → ui/dash}/PostForm.js +22 -43
  69. package/dist/{theme/components → ui/dash}/PostList.js +6 -6
  70. package/dist/{theme/components/VisibilityBadge.js → ui/dash/StatusBadge.js} +1 -2
  71. package/dist/ui/dash/collections/CollectionForm.js +152 -0
  72. package/dist/ui/dash/collections/CollectionsListContent.js +68 -0
  73. package/dist/ui/dash/collections/ViewCollectionContent.js +96 -0
  74. package/dist/{theme/components → ui/dash}/index.js +3 -6
  75. package/dist/ui/dash/media/MediaListContent.js +166 -0
  76. package/dist/ui/dash/media/ViewMediaContent.js +212 -0
  77. package/dist/ui/dash/pages/LinkFormContent.js +130 -0
  78. package/dist/ui/dash/pages/UnifiedPagesContent.js +193 -0
  79. package/dist/ui/dash/settings/AccountContent.js +209 -0
  80. package/dist/ui/dash/settings/AppearanceContent.js +259 -0
  81. package/dist/ui/dash/settings/GeneralContent.js +536 -0
  82. package/dist/ui/dash/settings/SettingsNav.js +41 -0
  83. package/dist/{themes/threads/timeline → ui/feed}/LinkCard.js +6 -2
  84. package/dist/{themes/threads/timeline → ui/feed}/NoteCard.js +11 -6
  85. package/dist/{themes/threads/timeline → ui/feed}/QuoteCard.js +10 -6
  86. package/dist/{themes/threads/timeline → ui/feed}/ThreadPreview.js +7 -9
  87. package/dist/ui/feed/TimelineFeed.js +41 -0
  88. package/dist/ui/feed/TimelineItem.js +27 -0
  89. package/dist/ui/font-themes.js +36 -0
  90. package/dist/{theme → ui}/layouts/BaseLayout.js +34 -2
  91. package/dist/{theme → ui}/layouts/DashLayout.js +0 -8
  92. package/dist/ui/layouts/SiteLayout.js +169 -0
  93. package/dist/{themes/threads → ui}/pages/ArchivePage.js +16 -14
  94. package/dist/{themes/threads → ui}/pages/CollectionPage.js +6 -1
  95. package/dist/ui/pages/CollectionsPage.js +76 -0
  96. package/dist/ui/pages/FeaturedPage.js +24 -0
  97. package/dist/ui/pages/HomePage.js +24 -0
  98. package/dist/{themes/threads → ui}/pages/PostPage.js +13 -8
  99. package/dist/{themes/threads → ui}/pages/SearchPage.js +9 -7
  100. package/dist/{themes/threads → ui}/pages/SinglePage.js +3 -2
  101. package/dist/{theme/components → ui/shared}/MediaGallery.js +1 -1
  102. package/dist/{theme/components → ui/shared}/Pagination.js +41 -2
  103. package/dist/{theme/components → ui/shared}/ThreadView.js +2 -2
  104. package/dist/ui/shared/index.js +5 -0
  105. package/package.json +1 -9
  106. package/src/__tests__/helpers/db.ts +3 -0
  107. package/src/app.tsx +131 -561
  108. package/src/client.ts +1 -0
  109. package/src/db/migrations/0006_rename_slug_to_path.sql +5 -0
  110. package/src/db/migrations/meta/_journal.json +7 -0
  111. package/src/db/schema.ts +1 -1
  112. package/src/i18n/locales/en.po +477 -261
  113. package/src/i18n/locales/en.ts +1 -1
  114. package/src/i18n/locales/zh-Hans.po +477 -261
  115. package/src/i18n/locales/zh-Hans.ts +1 -1
  116. package/src/i18n/locales/zh-Hant.po +477 -261
  117. package/src/i18n/locales/zh-Hant.ts +1 -1
  118. package/src/index.ts +7 -36
  119. package/src/lib/__tests__/config.test.ts +192 -0
  120. package/src/lib/__tests__/favicon.test.ts +151 -0
  121. package/src/lib/__tests__/image.test.ts +2 -6
  122. package/src/lib/__tests__/schemas.test.ts +60 -19
  123. package/src/lib/__tests__/timeline.test.ts +45 -81
  124. package/src/lib/__tests__/timezones.test.ts +61 -0
  125. package/src/lib/__tests__/view.test.ts +15 -9
  126. package/src/lib/avatar-upload.ts +165 -0
  127. package/src/lib/config.ts +47 -0
  128. package/src/lib/constants.ts +19 -10
  129. package/src/lib/favicon.ts +115 -0
  130. package/src/lib/image.ts +13 -21
  131. package/src/lib/media-helpers.ts +2 -2
  132. package/src/lib/nav-reorder.ts +1 -1
  133. package/src/lib/navigation.ts +73 -4
  134. package/src/lib/pagination.ts +50 -0
  135. package/src/lib/render.tsx +22 -15
  136. package/src/lib/schemas.ts +47 -6
  137. package/src/lib/theme.ts +5 -5
  138. package/src/lib/timeline.ts +28 -57
  139. package/src/lib/timezones.ts +325 -0
  140. package/src/lib/view.ts +3 -3
  141. package/src/preset.css +2 -1
  142. package/src/routes/__tests__/compose.test.ts +199 -0
  143. package/src/routes/api/__tests__/collections.test.ts +249 -0
  144. package/src/routes/api/__tests__/nav-items.test.ts +222 -0
  145. package/src/routes/api/__tests__/pages.test.ts +218 -0
  146. package/src/routes/api/__tests__/settings.test.ts +132 -0
  147. package/src/routes/api/collections.ts +143 -0
  148. package/src/routes/api/nav-items.ts +115 -0
  149. package/src/routes/api/pages.ts +101 -0
  150. package/src/routes/api/posts.ts +3 -3
  151. package/src/routes/api/search.ts +2 -2
  152. package/src/routes/api/settings.ts +91 -0
  153. package/src/routes/api/upload.ts +2 -3
  154. package/src/routes/auth/reset.tsx +239 -0
  155. package/src/routes/auth/setup.tsx +189 -0
  156. package/src/routes/auth/signin.tsx +163 -0
  157. package/src/routes/compose.ts +63 -0
  158. package/src/routes/dash/__tests__/pages.test.ts +225 -0
  159. package/src/routes/dash/__tests__/settings-avatar.test.ts +89 -0
  160. package/src/routes/dash/collections.tsx +18 -367
  161. package/src/routes/dash/index.tsx +1 -1
  162. package/src/routes/dash/media.tsx +13 -415
  163. package/src/routes/dash/pages.tsx +131 -98
  164. package/src/routes/dash/posts.tsx +3 -7
  165. package/src/routes/dash/redirects.tsx +22 -16
  166. package/src/routes/dash/settings.tsx +265 -478
  167. package/src/routes/feed/__tests__/rss.test.ts +141 -0
  168. package/src/routes/feed/rss.ts +5 -3
  169. package/src/routes/feed/sitemap.ts +5 -3
  170. package/src/routes/pages/__tests__/collections.test.ts +94 -0
  171. package/src/routes/pages/__tests__/featured.test.ts +94 -0
  172. package/src/routes/pages/archive.tsx +2 -6
  173. package/src/routes/pages/collection.tsx +2 -6
  174. package/src/routes/pages/collections.tsx +36 -0
  175. package/src/routes/pages/featured.tsx +44 -0
  176. package/src/routes/pages/home.tsx +30 -53
  177. package/src/routes/pages/latest.tsx +59 -0
  178. package/src/routes/pages/page.tsx +28 -30
  179. package/src/routes/pages/post.tsx +2 -5
  180. package/src/routes/pages/search.tsx +2 -6
  181. package/src/services/__tests__/page.test.ts +106 -0
  182. package/src/services/__tests__/post.test.ts +114 -15
  183. package/src/services/page.ts +13 -1
  184. package/src/services/post.ts +58 -40
  185. package/src/services/search.ts +2 -2
  186. package/src/styles/components.css +0 -65
  187. package/src/styles/tokens.css +47 -0
  188. package/src/styles/ui.css +475 -0
  189. package/src/types/bindings.ts +30 -0
  190. package/src/types/config.ts +183 -0
  191. package/src/types/constants.ts +26 -0
  192. package/src/types/entities.ts +109 -0
  193. package/src/types/operations.ts +88 -0
  194. package/src/types/props.ts +115 -0
  195. package/src/types/views.ts +172 -0
  196. package/src/types.ts +8 -774
  197. package/src/ui/__tests__/font-themes.test.ts +34 -0
  198. package/src/{theme → ui}/color-themes.ts +34 -34
  199. package/src/ui/compose/ComposeDialog.tsx +414 -0
  200. package/src/ui/compose/ComposePrompt.tsx +55 -0
  201. package/src/{theme/components/TypeBadge.tsx → ui/dash/FormatBadge.tsx} +2 -3
  202. package/src/{theme/components → ui/dash}/PageForm.tsx +25 -19
  203. package/src/{theme/components → ui/dash}/PostForm.tsx +26 -45
  204. package/src/{theme/components → ui/dash}/PostList.tsx +7 -7
  205. package/src/{theme/components/VisibilityBadge.tsx → ui/dash/StatusBadge.tsx} +2 -3
  206. package/src/ui/dash/collections/CollectionForm.tsx +153 -0
  207. package/src/ui/dash/collections/CollectionsListContent.tsx +85 -0
  208. package/src/ui/dash/collections/ViewCollectionContent.tsx +92 -0
  209. package/src/ui/dash/index.ts +10 -0
  210. package/src/ui/dash/media/MediaListContent.tsx +201 -0
  211. package/src/ui/dash/media/ViewMediaContent.tsx +208 -0
  212. package/src/ui/dash/pages/LinkFormContent.tsx +119 -0
  213. package/src/ui/dash/pages/UnifiedPagesContent.tsx +203 -0
  214. package/src/ui/dash/settings/AccountContent.tsx +176 -0
  215. package/src/ui/dash/settings/AppearanceContent.tsx +254 -0
  216. package/src/ui/dash/settings/GeneralContent.tsx +533 -0
  217. package/src/ui/dash/settings/SettingsNav.tsx +56 -0
  218. package/src/{themes/threads/timeline → ui/feed}/LinkCard.tsx +9 -4
  219. package/src/{themes/threads/timeline → ui/feed}/NoteCard.tsx +13 -8
  220. package/src/{themes/threads/timeline → ui/feed}/QuoteCard.tsx +13 -8
  221. package/src/{themes/threads/timeline → ui/feed}/ThreadPreview.tsx +7 -8
  222. package/src/ui/feed/TimelineFeed.tsx +49 -0
  223. package/src/ui/feed/TimelineItem.tsx +45 -0
  224. package/src/ui/font-themes.ts +54 -0
  225. package/src/{theme → ui}/layouts/BaseLayout.tsx +28 -1
  226. package/src/{theme → ui}/layouts/DashLayout.tsx +0 -10
  227. package/src/ui/layouts/SiteLayout.tsx +164 -0
  228. package/src/{themes/threads → ui}/pages/ArchivePage.tsx +22 -17
  229. package/src/{themes/threads → ui}/pages/CollectionPage.tsx +14 -5
  230. package/src/ui/pages/CollectionsPage.tsx +73 -0
  231. package/src/ui/pages/FeaturedPage.tsx +31 -0
  232. package/src/{themes/threads → ui}/pages/HomePage.tsx +11 -15
  233. package/src/{themes/threads → ui}/pages/PostPage.tsx +23 -14
  234. package/src/{themes/threads → ui}/pages/SearchPage.tsx +13 -11
  235. package/src/{themes/threads → ui}/pages/SinglePage.tsx +4 -4
  236. package/src/{theme/components → ui/shared}/MediaGallery.tsx +1 -1
  237. package/src/{theme/components → ui/shared}/Pagination.tsx +67 -4
  238. package/src/{theme/components → ui/shared}/ThreadView.tsx +2 -2
  239. package/src/ui/shared/__tests__/pagination.test.ts +46 -0
  240. package/src/ui/shared/index.ts +12 -0
  241. package/bin/jant.js +0 -185
  242. package/dist/lib/theme-components.js +0 -46
  243. package/dist/routes/dash/navigation.js +0 -289
  244. package/dist/theme/index.js +0 -18
  245. package/dist/theme/layouts/index.js +0 -2
  246. package/dist/themes/threads/ThreadsSiteLayout.js +0 -172
  247. package/dist/themes/threads/index.js +0 -81
  248. package/dist/themes/threads/pages/HomePage.js +0 -25
  249. package/dist/themes/threads/timeline/TimelineFeed.js +0 -58
  250. package/dist/themes/threads/timeline/TimelineItem.js +0 -36
  251. package/dist/themes/threads/timeline/TimelineLoadMore.js +0 -23
  252. package/dist/themes/threads/timeline/groupByDate.js +0 -22
  253. package/dist/themes/threads/timeline/timelineMore.js +0 -107
  254. package/src/lib/__tests__/theme-components.test.ts +0 -105
  255. package/src/lib/theme-components.ts +0 -65
  256. package/src/routes/dash/navigation.tsx +0 -317
  257. package/src/theme/components/index.ts +0 -23
  258. package/src/theme/index.ts +0 -22
  259. package/src/theme/layouts/index.ts +0 -7
  260. package/src/themes/threads/ThreadsSiteLayout.tsx +0 -194
  261. package/src/themes/threads/index.ts +0 -100
  262. package/src/themes/threads/style.css +0 -336
  263. package/src/themes/threads/timeline/TimelineFeed.tsx +0 -62
  264. package/src/themes/threads/timeline/TimelineItem.tsx +0 -67
  265. package/src/themes/threads/timeline/TimelineLoadMore.tsx +0 -35
  266. package/src/themes/threads/timeline/groupByDate.ts +0 -30
  267. package/src/themes/threads/timeline/timelineMore.tsx +0 -130
  268. /package/dist/{theme/components → ui/dash}/ActionButtons.js +0 -0
  269. /package/dist/{theme/components → ui/dash}/CrudPageHeader.js +0 -0
  270. /package/dist/{theme/components → ui/dash}/DangerZone.js +0 -0
  271. /package/dist/{theme/components → ui/dash}/ListItemRow.js +0 -0
  272. /package/dist/{theme/components → ui/shared}/EmptyState.js +0 -0
  273. /package/src/{theme/components → ui/dash}/ActionButtons.tsx +0 -0
  274. /package/src/{theme/components → ui/dash}/CrudPageHeader.tsx +0 -0
  275. /package/src/{theme/components → ui/dash}/DangerZone.tsx +0 -0
  276. /package/src/{theme/components → ui/dash}/ListItemRow.tsx +0 -0
  277. /package/src/{theme/components → ui/shared}/EmptyState.tsx +0 -0
@@ -13,7 +13,6 @@ export const PostForm = ({ post, action, mediaAttachments, r2PublicUrl, imageTra
13
13
  body: post?.body ?? "",
14
14
  url: post?.url ?? "",
15
15
  quoteText: post?.quoteText ?? "",
16
- slug: post?.slug ?? "",
17
16
  status: post?.status ?? "published",
18
17
  featured: post?.featured === 1,
19
18
  pinned: post?.pinned === 1,
@@ -170,7 +169,7 @@ export const PostForm = ({ post, action, mediaAttachments, r2PublicUrl, imageTra
170
169
  class: "grid grid-cols-4 sm:grid-cols-6 gap-2 mb-2",
171
170
  children: mediaAttachments.map((m)=>{
172
171
  const pUrl = getPublicUrlForProvider(m.provider, r2PublicUrl, s3PublicUrl);
173
- const mUrl = getMediaUrl(m.id, m.storageKey, pUrl);
172
+ const mUrl = getMediaUrl(m.storageKey, pUrl);
174
173
  const thumbUrl = getImageUrl(mUrl, imageTransformUrl, {
175
174
  width: 150,
176
175
  quality: 80,
@@ -309,56 +308,36 @@ export const PostForm = ({ post, action, mediaAttachments, r2PublicUrl, imageTra
309
308
  })
310
309
  ]
311
310
  }),
312
- /*#__PURE__*/ _jsxs("div", {
313
- class: "field",
314
- children: [
315
- /*#__PURE__*/ _jsx("label", {
316
- class: "label",
317
- children: $__i18n._({
318
- id: "quFPTj",
319
- message: "Custom Slug (optional)"
320
- })
321
- }),
322
- /*#__PURE__*/ _jsx("input", {
323
- type: "text",
324
- "data-bind": "slug",
325
- class: "input",
326
- placeholder: "my-custom-url",
327
- pattern: "[a-z0-9-]*"
328
- }),
329
- /*#__PURE__*/ _jsx("p", {
330
- class: "text-xs text-muted-foreground mt-1",
331
- children: $__i18n._({
332
- id: "iBc+/N",
333
- message: "Custom URL path. Leave empty to use default /p/ID format."
334
- })
335
- })
336
- ]
337
- }),
338
311
  /*#__PURE__*/ _jsxs("div", {
339
312
  class: "flex gap-2",
340
313
  children: [
341
314
  /*#__PURE__*/ _jsxs("button", {
342
315
  type: "submit",
343
316
  class: "btn",
344
- "data-attr-disabled": "$_loading",
317
+ "data-attr:disabled": "$_loading",
345
318
  children: [
346
- /*#__PURE__*/ _jsx("span", {
347
- "data-show": "!$_loading",
348
- children: isEdit ? $__i18n._({
349
- id: "EkH9pt",
350
- message: "Update"
351
- }) : $__i18n._({
352
- id: "EEYbdt",
353
- message: "Publish"
354
- })
355
- }),
356
- /*#__PURE__*/ _jsx("span", {
319
+ /*#__PURE__*/ _jsx("svg", {
357
320
  "data-show": "$_loading",
358
- children: $__i18n._({
359
- id: "k1ifdL",
360
- message: "Processing..."
321
+ style: "display:none",
322
+ class: "animate-spin size-4",
323
+ xmlns: "http://www.w3.org/2000/svg",
324
+ viewBox: "0 0 24 24",
325
+ fill: "none",
326
+ stroke: "currentColor",
327
+ "stroke-width": "2",
328
+ "stroke-linecap": "round",
329
+ "stroke-linejoin": "round",
330
+ role: "status",
331
+ children: /*#__PURE__*/ _jsx("path", {
332
+ d: "M21 12a9 9 0 1 1-6.219-8.56"
361
333
  })
334
+ }),
335
+ isEdit ? $__i18n._({
336
+ id: "EkH9pt",
337
+ message: "Update"
338
+ }) : $__i18n._({
339
+ id: "EEYbdt",
340
+ message: "Publish"
362
341
  })
363
342
  ]
364
343
  }),
@@ -4,9 +4,9 @@
4
4
  import { useLingui as $_useLingui } from "@jant/core/i18n";
5
5
  import * as sqid from "../../lib/sqid.js";
6
6
  import * as time from "../../lib/time.js";
7
- import { VisibilityBadge } from "./VisibilityBadge.js";
8
- import { TypeBadge } from "./TypeBadge.js";
9
- import { EmptyState } from "./EmptyState.js";
7
+ import { StatusBadge } from "./StatusBadge.js";
8
+ import { FormatBadge } from "./FormatBadge.js";
9
+ import { EmptyState } from "../shared/EmptyState.js";
10
10
  import { ListItemRow } from "./ListItemRow.js";
11
11
  import { ActionButtons } from "./ActionButtons.js";
12
12
  export const PostList = ({ posts })=>{
@@ -27,7 +27,7 @@ export const PostList = ({ posts })=>{
27
27
  return /*#__PURE__*/ _jsx("div", {
28
28
  class: "flex flex-col divide-y",
29
29
  children: posts.map((post)=>{
30
- const permalink = post.slug ? `/${post.slug}` : `/p/${sqid.encode(post.id)}`;
30
+ const permalink = post.path ? `/${post.path}` : `/p/${sqid.encode(post.id)}`;
31
31
  return /*#__PURE__*/ _jsxs(ListItemRow, {
32
32
  actions: /*#__PURE__*/ _jsx(ActionButtons, {
33
33
  editHref: `/dash/posts/${sqid.encode(post.id)}/edit`,
@@ -50,10 +50,10 @@ export const PostList = ({ posts })=>{
50
50
  /*#__PURE__*/ _jsxs("div", {
51
51
  class: "flex items-center gap-2 mb-1",
52
52
  children: [
53
- /*#__PURE__*/ _jsx(TypeBadge, {
53
+ /*#__PURE__*/ _jsx(FormatBadge, {
54
54
  type: post.format
55
55
  }),
56
- /*#__PURE__*/ _jsx(VisibilityBadge, {
56
+ /*#__PURE__*/ _jsx(StatusBadge, {
57
57
  status: post.status,
58
58
  featured: post.featured === 1,
59
59
  pinned: post.pinned === 1
@@ -2,10 +2,9 @@
2
2
  * Status Badge Component
3
3
  *
4
4
  * Displays badges for post status, featured, and pinned state.
5
- * Named VisibilityBadge for backward compatibility with theme overrides.
6
5
  */ import { jsx as _jsx, jsxs as _jsxs } from "hono/jsx/jsx-runtime";
7
6
  import { useLingui as $_useLingui } from "@jant/core/i18n";
8
- export const VisibilityBadge = ({ status, featured, pinned })=>{
7
+ export const StatusBadge = ({ status, featured, pinned })=>{
9
8
  const { i18n: $__i18n, _: $__ } = $_useLingui();
10
9
  const statusVariants = {
11
10
  published: "badge-secondary",
@@ -0,0 +1,152 @@
1
+ /**
2
+ * Shared collection form (new + edit)
3
+ */ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "hono/jsx/jsx-runtime";
4
+ import { useLingui as $_useLingui } from "@jant/core/i18n";
5
+ export function CollectionForm({ collection, isEdit }) {
6
+ const { i18n: $__i18n, _: $__ } = $_useLingui();
7
+ const signals = JSON.stringify({
8
+ title: collection?.title ?? "",
9
+ slug: collection?.slug ?? "",
10
+ description: collection?.description ?? ""
11
+ }).replace(/</g, "\\u003c");
12
+ const action = isEdit ? `/dash/collections/${collection?.id}` : "/dash/collections";
13
+ const heading = isEdit ? $__i18n._({
14
+ id: "/0D1Xp",
15
+ message: "Edit Collection"
16
+ }) : $__i18n._({
17
+ id: "vzU4k9",
18
+ message: "New Collection"
19
+ });
20
+ const submitLabel = isEdit ? $__i18n._({
21
+ id: "7Mk+/h",
22
+ message: "Update Collection"
23
+ }) : $__i18n._({
24
+ id: "Pbm2/N",
25
+ message: "Create Collection"
26
+ });
27
+ const cancelHref = isEdit ? `/dash/collections/${collection?.id}` : "/dash/collections";
28
+ return /*#__PURE__*/ _jsxs(_Fragment, {
29
+ children: [
30
+ /*#__PURE__*/ _jsx("h1", {
31
+ class: "text-2xl font-semibold mb-6",
32
+ children: heading
33
+ }),
34
+ /*#__PURE__*/ _jsxs("form", {
35
+ "data-signals": signals,
36
+ "data-on:submit__prevent": `@post('${action}')`,
37
+ "data-indicator": "_loading",
38
+ class: "flex flex-col gap-4 max-w-lg",
39
+ children: [
40
+ /*#__PURE__*/ _jsxs("div", {
41
+ class: "field",
42
+ children: [
43
+ /*#__PURE__*/ _jsx("label", {
44
+ class: "label",
45
+ children: $__i18n._({
46
+ id: "MHrjPM",
47
+ message: "Title"
48
+ })
49
+ }),
50
+ /*#__PURE__*/ _jsx("input", {
51
+ type: "text",
52
+ "data-bind": "title",
53
+ class: "input",
54
+ required: true,
55
+ placeholder: isEdit ? undefined : $__i18n._({
56
+ id: "fttd2R",
57
+ message: "My Collection"
58
+ })
59
+ })
60
+ ]
61
+ }),
62
+ /*#__PURE__*/ _jsxs("div", {
63
+ class: "field",
64
+ children: [
65
+ /*#__PURE__*/ _jsx("label", {
66
+ class: "label",
67
+ children: $__i18n._({
68
+ id: "L85WcV",
69
+ message: "Slug"
70
+ })
71
+ }),
72
+ /*#__PURE__*/ _jsx("input", {
73
+ type: "text",
74
+ "data-bind": "slug",
75
+ class: "input",
76
+ required: true,
77
+ pattern: "[a-z0-9-]+",
78
+ placeholder: isEdit ? undefined : "my-collection"
79
+ }),
80
+ !isEdit && /*#__PURE__*/ _jsx("p", {
81
+ class: "text-xs text-muted-foreground mt-1",
82
+ children: $__i18n._({
83
+ id: "1CU1Td",
84
+ message: "URL-safe identifier (lowercase, numbers, hyphens)"
85
+ })
86
+ })
87
+ ]
88
+ }),
89
+ /*#__PURE__*/ _jsxs("div", {
90
+ class: "field",
91
+ children: [
92
+ /*#__PURE__*/ _jsx("label", {
93
+ class: "label",
94
+ children: $__i18n._({
95
+ id: "Fxf4jq",
96
+ message: "Description (optional)"
97
+ })
98
+ }),
99
+ /*#__PURE__*/ _jsx("textarea", {
100
+ "data-bind": "description",
101
+ class: "textarea",
102
+ rows: 3,
103
+ placeholder: isEdit ? undefined : $__i18n._({
104
+ id: "AyHO4m",
105
+ message: "What's this collection about?"
106
+ }),
107
+ children: collection?.description ?? ""
108
+ })
109
+ ]
110
+ }),
111
+ /*#__PURE__*/ _jsxs("div", {
112
+ class: "flex gap-2",
113
+ children: [
114
+ /*#__PURE__*/ _jsxs("button", {
115
+ type: "submit",
116
+ class: "btn",
117
+ "data-attr:disabled": "$_loading",
118
+ children: [
119
+ /*#__PURE__*/ _jsx("svg", {
120
+ "data-show": "$_loading",
121
+ style: "display:none",
122
+ class: "animate-spin size-4",
123
+ xmlns: "http://www.w3.org/2000/svg",
124
+ viewBox: "0 0 24 24",
125
+ fill: "none",
126
+ stroke: "currentColor",
127
+ "stroke-width": "2",
128
+ "stroke-linecap": "round",
129
+ "stroke-linejoin": "round",
130
+ role: "status",
131
+ children: /*#__PURE__*/ _jsx("path", {
132
+ d: "M21 12a9 9 0 1 1-6.219-8.56"
133
+ })
134
+ }),
135
+ submitLabel
136
+ ]
137
+ }),
138
+ /*#__PURE__*/ _jsx("a", {
139
+ href: cancelHref,
140
+ class: "btn-outline",
141
+ children: $__i18n._({
142
+ id: "dEgA5A",
143
+ message: "Cancel"
144
+ })
145
+ })
146
+ ]
147
+ })
148
+ ]
149
+ })
150
+ ]
151
+ });
152
+ }
@@ -0,0 +1,68 @@
1
+ /**
2
+ * Collections list view
3
+ */ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "hono/jsx/jsx-runtime";
4
+ import { useLingui as $_useLingui } from "@jant/core/i18n";
5
+ import { EmptyState, ListItemRow, ActionButtons, CrudPageHeader } from "../index.js";
6
+ export function CollectionsListContent({ collections }) {
7
+ const { i18n: $__i18n, _: $__ } = $_useLingui();
8
+ return /*#__PURE__*/ _jsxs(_Fragment, {
9
+ children: [
10
+ /*#__PURE__*/ _jsx(CrudPageHeader, {
11
+ title: $__i18n._({
12
+ id: "DoJzLz",
13
+ message: "Collections"
14
+ }),
15
+ ctaLabel: $__i18n._({
16
+ id: "vzU4k9",
17
+ message: "New Collection"
18
+ }),
19
+ ctaHref: "/dash/collections/new"
20
+ }),
21
+ collections.length === 0 ? /*#__PURE__*/ _jsx(EmptyState, {
22
+ message: $__i18n._({
23
+ id: "+MACwa",
24
+ message: "No collections yet."
25
+ }),
26
+ ctaText: $__i18n._({
27
+ id: "vzU4k9",
28
+ message: "New Collection"
29
+ }),
30
+ ctaHref: "/dash/collections/new"
31
+ }) : /*#__PURE__*/ _jsx("div", {
32
+ class: "flex flex-col divide-y",
33
+ children: collections.map((col)=>/*#__PURE__*/ _jsxs(ListItemRow, {
34
+ actions: /*#__PURE__*/ _jsx(ActionButtons, {
35
+ editHref: `/dash/collections/${col.id}/edit`,
36
+ editLabel: $__i18n._({
37
+ id: "ePK91l",
38
+ message: "Edit"
39
+ }),
40
+ viewHref: `/c/${col.slug}`,
41
+ viewLabel: $__i18n._({
42
+ id: "jpctdh",
43
+ message: "View"
44
+ })
45
+ }),
46
+ children: [
47
+ /*#__PURE__*/ _jsx("a", {
48
+ href: `/dash/collections/${col.id}`,
49
+ class: "font-medium hover:underline",
50
+ children: col.title
51
+ }),
52
+ /*#__PURE__*/ _jsxs("p", {
53
+ class: "text-sm text-muted-foreground",
54
+ children: [
55
+ "/",
56
+ col.slug
57
+ ]
58
+ }),
59
+ col.description && /*#__PURE__*/ _jsx("p", {
60
+ class: "text-sm text-muted-foreground mt-1",
61
+ children: col.description
62
+ })
63
+ ]
64
+ }, col.id))
65
+ })
66
+ ]
67
+ });
68
+ }
@@ -0,0 +1,96 @@
1
+ /**
2
+ * Single collection detail view
3
+ */ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "hono/jsx/jsx-runtime";
4
+ import { useLingui as $_useLingui } from "@jant/core/i18n";
5
+ import { ActionButtons } from "../index.js";
6
+ import { encode } from "../../../lib/sqid.js";
7
+ export function ViewCollectionContent({ collection, posts }) {
8
+ const { i18n: $__i18n, _: $__ } = $_useLingui();
9
+ const postsHeader = $__i18n._({
10
+ id: "pRhYH2",
11
+ message: "Posts in Collection ({count})"
12
+ });
13
+ return /*#__PURE__*/ _jsxs(_Fragment, {
14
+ children: [
15
+ /*#__PURE__*/ _jsxs("div", {
16
+ class: "flex items-center justify-between mb-6",
17
+ children: [
18
+ /*#__PURE__*/ _jsxs("div", {
19
+ children: [
20
+ /*#__PURE__*/ _jsx("h1", {
21
+ class: "text-2xl font-semibold",
22
+ children: collection.title
23
+ }),
24
+ /*#__PURE__*/ _jsxs("p", {
25
+ class: "text-sm text-muted-foreground",
26
+ children: [
27
+ "/",
28
+ collection.slug
29
+ ]
30
+ })
31
+ ]
32
+ }),
33
+ /*#__PURE__*/ _jsx(ActionButtons, {
34
+ editHref: `/dash/collections/${collection.id}/edit`,
35
+ editLabel: $__i18n._({
36
+ id: "ePK91l",
37
+ message: "Edit"
38
+ }),
39
+ viewHref: `/c/${collection.slug}`,
40
+ viewLabel: $__i18n._({
41
+ id: "jpctdh",
42
+ message: "View"
43
+ })
44
+ })
45
+ ]
46
+ }),
47
+ collection.description && /*#__PURE__*/ _jsx("p", {
48
+ class: "text-muted-foreground mb-6",
49
+ children: collection.description
50
+ }),
51
+ /*#__PURE__*/ _jsxs("div", {
52
+ class: "card",
53
+ children: [
54
+ /*#__PURE__*/ _jsx("header", {
55
+ children: /*#__PURE__*/ _jsx("h2", {
56
+ children: postsHeader
57
+ })
58
+ }),
59
+ /*#__PURE__*/ _jsx("section", {
60
+ children: posts.length === 0 ? /*#__PURE__*/ _jsx("p", {
61
+ class: "text-muted-foreground",
62
+ children: $__i18n._({
63
+ id: "J4FNfC",
64
+ message: "No posts in this collection."
65
+ })
66
+ }) : /*#__PURE__*/ _jsx("div", {
67
+ class: "flex flex-col divide-y",
68
+ children: posts.map((post)=>/*#__PURE__*/ _jsx("div", {
69
+ class: "py-3 flex items-center gap-4",
70
+ children: /*#__PURE__*/ _jsx("div", {
71
+ class: "flex-1 min-w-0",
72
+ children: /*#__PURE__*/ _jsx("a", {
73
+ href: `/dash/posts/${encode(post.id)}`,
74
+ class: "font-medium hover:underline",
75
+ children: post.title || post.excerpt?.slice(0, 50) || `Post #${post.id}`
76
+ })
77
+ })
78
+ }, post.id))
79
+ })
80
+ })
81
+ ]
82
+ }),
83
+ /*#__PURE__*/ _jsx("div", {
84
+ class: "mt-6",
85
+ children: /*#__PURE__*/ _jsx("a", {
86
+ href: "/dash/collections",
87
+ class: "text-sm hover:underline",
88
+ children: $__i18n._({
89
+ id: "e6Jr7Q",
90
+ message: "← Back to Collections"
91
+ })
92
+ })
93
+ })
94
+ ]
95
+ });
96
+ }
@@ -1,13 +1,10 @@
1
1
  export { ActionButtons } from "./ActionButtons.js";
2
2
  export { CrudPageHeader } from "./CrudPageHeader.js";
3
3
  export { DangerZone } from "./DangerZone.js";
4
- export { EmptyState } from "./EmptyState.js";
4
+ export { EmptyState } from "../shared/EmptyState.js";
5
+ export { FormatBadge } from "./FormatBadge.js";
5
6
  export { ListItemRow } from "./ListItemRow.js";
6
- export { MediaGallery } from "./MediaGallery.js";
7
7
  export { PageForm } from "./PageForm.js";
8
- export { Pagination, LoadMore, PagePagination } from "./Pagination.js";
9
8
  export { PostForm } from "./PostForm.js";
10
9
  export { PostList } from "./PostList.js";
11
- export { ThreadView } from "./ThreadView.js";
12
- export { TypeBadge } from "./TypeBadge.js";
13
- export { VisibilityBadge } from "./VisibilityBadge.js";
10
+ export { StatusBadge } from "./StatusBadge.js";
@@ -0,0 +1,166 @@
1
+ /**
2
+ * Media grid list with upload UI
3
+ */ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "hono/jsx/jsx-runtime";
4
+ import { useLingui as $_useLingui } from "@jant/core/i18n";
5
+ import { EmptyState } from "../index.js";
6
+ import { getMediaUrl, getImageUrl, getPublicUrlForProvider } from "../../../lib/image.js";
7
+ function formatSize(bytes) {
8
+ if (bytes < 1024) return `${bytes} B`;
9
+ if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`;
10
+ return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;
11
+ }
12
+ function MediaCard({ media, r2PublicUrl, imageTransformUrl, s3PublicUrl }) {
13
+ const publicUrl = getPublicUrlForProvider(media.provider, r2PublicUrl, s3PublicUrl);
14
+ const fullUrl = getMediaUrl(media.storageKey, publicUrl);
15
+ const thumbnailUrl = getImageUrl(fullUrl, imageTransformUrl, {
16
+ width: 300,
17
+ quality: 80,
18
+ format: "auto",
19
+ fit: "cover"
20
+ });
21
+ const isImage = media.mimeType.startsWith("image/");
22
+ return /*#__PURE__*/ _jsxs("div", {
23
+ class: "group relative",
24
+ "data-media-id": media.id,
25
+ children: [
26
+ isImage ? /*#__PURE__*/ _jsx("button", {
27
+ type: "button",
28
+ class: "block w-full aspect-square bg-muted rounded-lg overflow-hidden border hover:border-primary cursor-pointer",
29
+ onclick: `document.getElementById('lightbox-img').src = '${fullUrl}'; document.getElementById('lightbox').showModal()`,
30
+ children: /*#__PURE__*/ _jsx("img", {
31
+ src: thumbnailUrl,
32
+ alt: media.alt || media.originalName,
33
+ class: "w-full h-full object-cover",
34
+ loading: "lazy"
35
+ })
36
+ }) : /*#__PURE__*/ _jsx("a", {
37
+ href: `/dash/media/${media.id}`,
38
+ class: "block aspect-square bg-muted rounded-lg overflow-hidden border hover:border-primary",
39
+ children: /*#__PURE__*/ _jsx("div", {
40
+ class: "w-full h-full flex items-center justify-center text-muted-foreground",
41
+ children: /*#__PURE__*/ _jsx("span", {
42
+ class: "text-xs",
43
+ children: media.mimeType
44
+ })
45
+ })
46
+ }),
47
+ /*#__PURE__*/ _jsx("a", {
48
+ href: `/dash/media/${media.id}`,
49
+ class: "block mt-2 text-xs truncate hover:underline",
50
+ title: media.originalName,
51
+ children: media.originalName
52
+ }),
53
+ /*#__PURE__*/ _jsx("div", {
54
+ class: "text-xs text-muted-foreground",
55
+ children: formatSize(media.size)
56
+ })
57
+ ]
58
+ });
59
+ }
60
+ export function MediaListContent({ mediaList, r2PublicUrl, imageTransformUrl, s3PublicUrl }) {
61
+ const { i18n: $__i18n, _: $__ } = $_useLingui();
62
+ const processingText = $__i18n._({
63
+ id: "k1ifdL",
64
+ message: "Processing..."
65
+ });
66
+ const uploadingText = $__i18n._({
67
+ id: "GxkJXS",
68
+ message: "Uploading..."
69
+ });
70
+ const uploadText = $__i18n._({
71
+ id: "ONWvwQ",
72
+ message: "Upload"
73
+ });
74
+ const errorText = $__i18n._({
75
+ id: "pZq3aX",
76
+ message: "Upload failed. Please try again."
77
+ });
78
+ return /*#__PURE__*/ _jsxs(_Fragment, {
79
+ children: [
80
+ /*#__PURE__*/ _jsx("form", {
81
+ id: "upload-form",
82
+ class: "hidden",
83
+ enctype: "multipart/form-data",
84
+ "data-on:submit__prevent": "@post('/api/upload', {contentType: 'form'})",
85
+ children: /*#__PURE__*/ _jsx("input", {
86
+ id: "upload-file-input",
87
+ type: "file",
88
+ name: "file"
89
+ })
90
+ }),
91
+ /*#__PURE__*/ _jsxs("div", {
92
+ class: "flex items-center justify-between mb-6",
93
+ children: [
94
+ /*#__PURE__*/ _jsx("h1", {
95
+ class: "text-2xl font-semibold",
96
+ children: $__i18n._({
97
+ id: "xYilR2",
98
+ message: "Media"
99
+ })
100
+ }),
101
+ /*#__PURE__*/ _jsxs("label", {
102
+ class: "btn cursor-pointer",
103
+ children: [
104
+ /*#__PURE__*/ _jsx("span", {
105
+ children: uploadText
106
+ }),
107
+ /*#__PURE__*/ _jsx("input", {
108
+ type: "file",
109
+ class: "hidden",
110
+ accept: "image/*",
111
+ "data-media-upload": true,
112
+ "data-text-processing": processingText,
113
+ "data-text-uploading": uploadingText,
114
+ "data-text-error": errorText
115
+ })
116
+ ]
117
+ })
118
+ ]
119
+ }),
120
+ /*#__PURE__*/ _jsx("div", {
121
+ class: "card mb-6",
122
+ children: /*#__PURE__*/ _jsx("section", {
123
+ class: "text-sm text-muted-foreground",
124
+ children: /*#__PURE__*/ _jsx("p", {
125
+ children: $__i18n._({
126
+ id: "x+doid",
127
+ message: "Images are automatically optimized: resized to max 1920px, converted to WebP, and metadata stripped."
128
+ })
129
+ })
130
+ })
131
+ }),
132
+ /*#__PURE__*/ _jsx("div", {
133
+ id: "media-content",
134
+ children: mediaList.length === 0 ? /*#__PURE__*/ _jsx("div", {
135
+ id: "empty-state",
136
+ children: /*#__PURE__*/ _jsx(EmptyState, {
137
+ message: $__i18n._({
138
+ id: "ST+lN2",
139
+ message: "No media uploaded yet."
140
+ })
141
+ })
142
+ }) : /*#__PURE__*/ _jsx("div", {
143
+ id: "media-grid",
144
+ class: "grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 lg:grid-cols-5 gap-4",
145
+ children: mediaList.map((m)=>/*#__PURE__*/ _jsx(MediaCard, {
146
+ media: m,
147
+ r2PublicUrl: r2PublicUrl,
148
+ imageTransformUrl: imageTransformUrl,
149
+ s3PublicUrl: s3PublicUrl
150
+ }, m.id))
151
+ })
152
+ }),
153
+ /*#__PURE__*/ _jsx("dialog", {
154
+ id: "lightbox",
155
+ class: "p-0 m-auto bg-transparent backdrop:bg-black/80",
156
+ onclick: "event.target === this && this.close()",
157
+ children: /*#__PURE__*/ _jsx("img", {
158
+ id: "lightbox-img",
159
+ src: "",
160
+ alt: "",
161
+ class: "max-w-[90vw] max-h-[90vh] object-contain rounded-lg"
162
+ })
163
+ })
164
+ ]
165
+ });
166
+ }