@jant/core 0.3.36 → 0.3.38

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 (271) hide show
  1. package/bin/commands/export.js +1 -1
  2. package/bin/commands/import-site.js +529 -0
  3. package/bin/commands/reset-password.js +3 -2
  4. package/dist/client/assets/heic-to-DIRPI3VF.js +1 -0
  5. package/dist/client/assets/url-FWFqPJPb.js +1 -0
  6. package/dist/client/client.css +1 -1
  7. package/dist/client/client.js +4012 -3276
  8. package/dist/index.js +10285 -5809
  9. package/package.json +11 -3
  10. package/src/__tests__/helpers/app.ts +9 -9
  11. package/src/__tests__/helpers/db.ts +91 -93
  12. package/src/app.tsx +157 -27
  13. package/src/auth.ts +20 -2
  14. package/src/client/archive-nav.js +187 -0
  15. package/src/client/audio-player.ts +478 -0
  16. package/src/client/audio-processor.ts +84 -0
  17. package/src/client/avatar-upload.ts +3 -2
  18. package/src/client/components/__tests__/jant-compose-dialog.test.ts +645 -49
  19. package/src/client/components/__tests__/jant-compose-editor.test.ts +208 -16
  20. package/src/client/components/__tests__/jant-post-form.test.ts +19 -9
  21. package/src/client/components/__tests__/jant-settings-avatar.test.ts +2 -2
  22. package/src/client/components/__tests__/jant-settings-general.test.ts +3 -3
  23. package/src/client/components/collection-sidebar-types.ts +7 -9
  24. package/src/client/components/compose-types.ts +101 -4
  25. package/src/client/components/jant-collection-form.ts +43 -7
  26. package/src/client/components/jant-collection-sidebar.ts +88 -84
  27. package/src/client/components/jant-compose-dialog.ts +1655 -219
  28. package/src/client/components/jant-compose-editor.ts +732 -168
  29. package/src/client/components/jant-compose-fullscreen.ts +23 -78
  30. package/src/client/components/jant-media-lightbox.ts +2 -0
  31. package/src/client/components/jant-nav-manager.ts +24 -284
  32. package/src/client/components/jant-post-form.ts +89 -9
  33. package/src/client/components/jant-post-menu.ts +1019 -0
  34. package/src/client/components/jant-settings-avatar.ts +3 -3
  35. package/src/client/components/jant-settings-general.ts +38 -4
  36. package/src/client/components/jant-text-preview.ts +232 -0
  37. package/src/client/components/nav-manager-types.ts +4 -19
  38. package/src/client/components/post-form-template.ts +107 -12
  39. package/src/client/components/post-form-types.ts +11 -4
  40. package/src/client/compose-bridge.ts +410 -109
  41. package/src/client/image-processor.ts +26 -8
  42. package/src/client/media-metadata.ts +247 -0
  43. package/src/client/multipart-upload.ts +160 -0
  44. package/src/client/post-form-bridge.ts +52 -1
  45. package/src/client/settings-bridge.ts +0 -12
  46. package/src/client/thread-context.ts +140 -0
  47. package/src/client/tiptap/create-editor.ts +46 -0
  48. package/src/client/tiptap/extensions.ts +5 -0
  49. package/src/client/tiptap/image-node.ts +2 -8
  50. package/src/client/tiptap/paste-image.ts +2 -13
  51. package/src/client/tiptap/slash-commands.ts +173 -63
  52. package/src/client/toast.ts +101 -3
  53. package/src/client/types/sortablejs.d.ts +15 -0
  54. package/src/client/upload-with-metadata.ts +54 -0
  55. package/src/client/video-processor.ts +207 -0
  56. package/src/client.ts +5 -2
  57. package/src/db/__tests__/migrations.test.ts +118 -0
  58. package/src/db/index.ts +52 -0
  59. package/src/db/migrations/0000_baseline.sql +269 -0
  60. package/src/db/migrations/0001_fts_setup.sql +31 -0
  61. package/src/db/migrations/meta/0000_snapshot.json +703 -119
  62. package/src/db/migrations/meta/0001_snapshot.json +1337 -0
  63. package/src/db/migrations/meta/_journal.json +4 -39
  64. package/src/db/schema.ts +409 -145
  65. package/src/i18n/__tests__/detect.test.ts +115 -0
  66. package/src/i18n/context.tsx +2 -2
  67. package/src/i18n/detect.ts +85 -1
  68. package/src/i18n/i18n.ts +1 -1
  69. package/src/i18n/index.ts +2 -1
  70. package/src/i18n/locales/en.po +487 -1217
  71. package/src/i18n/locales/en.ts +1 -1
  72. package/src/i18n/locales/zh-Hans.po +613 -996
  73. package/src/i18n/locales/zh-Hans.ts +1 -1
  74. package/src/i18n/locales/zh-Hant.po +624 -1007
  75. package/src/i18n/locales/zh-Hant.ts +1 -1
  76. package/src/i18n/middleware.ts +6 -0
  77. package/src/index.ts +5 -7
  78. package/src/lib/__tests__/blurhash-placeholder.test.ts +75 -0
  79. package/src/lib/__tests__/constants.test.ts +0 -1
  80. package/src/lib/__tests__/markdown-to-tiptap.test.ts +358 -0
  81. package/src/lib/__tests__/nanoid.test.ts +26 -0
  82. package/src/lib/__tests__/schemas.test.ts +181 -63
  83. package/src/lib/__tests__/slug.test.ts +126 -0
  84. package/src/lib/__tests__/sse.test.ts +6 -6
  85. package/src/lib/__tests__/summary.test.ts +264 -0
  86. package/src/lib/__tests__/theme.test.ts +1 -1
  87. package/src/lib/__tests__/timeline.test.ts +33 -30
  88. package/src/lib/__tests__/tiptap-to-markdown.test.ts +346 -0
  89. package/src/lib/__tests__/view.test.ts +141 -66
  90. package/src/lib/blurhash-placeholder.ts +102 -0
  91. package/src/lib/constants.ts +3 -1
  92. package/src/lib/emoji-catalog.ts +885 -68
  93. package/src/lib/errors.ts +11 -8
  94. package/src/lib/feed.ts +78 -32
  95. package/src/lib/html.ts +2 -1
  96. package/src/lib/icon-catalog.ts +5033 -1
  97. package/src/lib/icons.ts +3 -2
  98. package/src/lib/index.ts +0 -1
  99. package/src/lib/markdown-to-tiptap.ts +286 -0
  100. package/src/lib/media-helpers.ts +12 -3
  101. package/src/lib/nanoid.ts +29 -0
  102. package/src/lib/navigation.ts +1 -1
  103. package/src/lib/render.tsx +20 -2
  104. package/src/lib/resolve-config.ts +6 -2
  105. package/src/lib/schemas.ts +224 -55
  106. package/src/lib/search-snippet.ts +34 -0
  107. package/src/lib/slug.ts +96 -0
  108. package/src/lib/sse.ts +6 -6
  109. package/src/lib/storage.ts +115 -7
  110. package/src/lib/summary.ts +66 -0
  111. package/src/lib/theme.ts +11 -8
  112. package/src/lib/timeline.ts +74 -34
  113. package/src/lib/tiptap-render.ts +5 -10
  114. package/src/lib/tiptap-to-markdown.ts +305 -0
  115. package/src/lib/upload.ts +190 -29
  116. package/src/lib/url.ts +31 -0
  117. package/src/lib/view.ts +204 -37
  118. package/src/middleware/__tests__/auth.test.ts +191 -11
  119. package/src/middleware/__tests__/onboarding.test.ts +12 -10
  120. package/src/middleware/auth.ts +63 -9
  121. package/src/middleware/onboarding.ts +1 -1
  122. package/src/middleware/secure-headers.ts +40 -0
  123. package/src/preset.css +45 -2
  124. package/src/routes/__tests__/compose.test.ts +17 -24
  125. package/src/routes/api/__tests__/collections.test.ts +109 -61
  126. package/src/routes/api/__tests__/nav-items.test.ts +46 -29
  127. package/src/routes/api/__tests__/posts.test.ts +132 -68
  128. package/src/routes/api/__tests__/search.test.ts +15 -2
  129. package/src/routes/api/__tests__/upload-multipart.test.ts +534 -0
  130. package/src/routes/api/collections.ts +51 -42
  131. package/src/routes/api/custom-urls.ts +80 -0
  132. package/src/routes/api/export.ts +31 -0
  133. package/src/routes/api/nav-items.ts +23 -19
  134. package/src/routes/api/posts.ts +43 -39
  135. package/src/routes/api/search.ts +3 -4
  136. package/src/routes/api/upload-multipart.ts +245 -0
  137. package/src/routes/api/upload.ts +85 -19
  138. package/src/routes/auth/__tests__/setup.test.ts +20 -60
  139. package/src/routes/auth/setup.tsx +26 -33
  140. package/src/routes/auth/signin.tsx +3 -7
  141. package/src/routes/compose.tsx +10 -55
  142. package/src/routes/dash/__tests__/settings-avatar.test.ts +1 -1
  143. package/src/routes/dash/custom-urls.tsx +414 -0
  144. package/src/routes/dash/settings.tsx +304 -232
  145. package/src/routes/feed/__tests__/rss.test.ts +27 -28
  146. package/src/routes/feed/rss.ts +6 -4
  147. package/src/routes/feed/sitemap.ts +2 -12
  148. package/src/routes/pages/__tests__/collections.test.ts +5 -6
  149. package/src/routes/pages/__tests__/featured.test.ts +41 -22
  150. package/src/routes/pages/archive.tsx +175 -39
  151. package/src/routes/pages/collection.tsx +22 -10
  152. package/src/routes/pages/collections.tsx +3 -3
  153. package/src/routes/pages/featured.tsx +28 -4
  154. package/src/routes/pages/home.tsx +16 -15
  155. package/src/routes/pages/latest.tsx +1 -11
  156. package/src/routes/pages/new.tsx +39 -0
  157. package/src/routes/pages/page.tsx +95 -49
  158. package/src/routes/pages/search.tsx +1 -1
  159. package/src/services/__tests__/api-token.test.ts +135 -0
  160. package/src/services/__tests__/collection.test.ts +275 -227
  161. package/src/services/__tests__/custom-url.test.ts +213 -0
  162. package/src/services/__tests__/media.test.ts +162 -22
  163. package/src/services/__tests__/navigation.test.ts +109 -68
  164. package/src/services/__tests__/post-timeline.test.ts +205 -32
  165. package/src/services/__tests__/post.test.ts +713 -234
  166. package/src/services/__tests__/search.test.ts +67 -10
  167. package/src/services/api-token.ts +166 -0
  168. package/src/services/auth.ts +17 -2
  169. package/src/services/collection.ts +397 -131
  170. package/src/services/custom-url.ts +188 -0
  171. package/src/services/export.ts +802 -0
  172. package/src/services/index.ts +26 -19
  173. package/src/services/media.ts +100 -22
  174. package/src/services/navigation.ts +158 -47
  175. package/src/services/path.ts +339 -0
  176. package/src/services/post.ts +687 -154
  177. package/src/services/search.ts +160 -75
  178. package/src/styles/components.css +58 -7
  179. package/src/styles/tokens.css +84 -6
  180. package/src/styles/ui.css +2964 -457
  181. package/src/types/bindings.ts +4 -1
  182. package/src/types/config.ts +12 -4
  183. package/src/types/constants.ts +15 -3
  184. package/src/types/entities.ts +74 -35
  185. package/src/types/operations.ts +11 -24
  186. package/src/types/props.ts +51 -16
  187. package/src/types/views.ts +45 -22
  188. package/src/ui/color-themes.ts +133 -23
  189. package/src/ui/compose/ComposeDialog.tsx +239 -17
  190. package/src/ui/compose/ComposePrompt.tsx +1 -1
  191. package/src/ui/dash/CrudPageHeader.tsx +1 -1
  192. package/src/ui/dash/ListItemRow.tsx +1 -1
  193. package/src/ui/dash/StatusBadge.tsx +3 -1
  194. package/src/ui/dash/appearance/AdvancedContent.tsx +22 -1
  195. package/src/ui/dash/appearance/ColorThemeContent.tsx +22 -2
  196. package/src/ui/dash/appearance/FontThemeContent.tsx +1 -1
  197. package/src/ui/dash/appearance/NavigationContent.tsx +5 -45
  198. package/src/ui/dash/index.ts +0 -3
  199. package/src/ui/dash/settings/AccountContent.tsx +3 -57
  200. package/src/ui/dash/settings/AccountMenuContent.tsx +147 -0
  201. package/src/ui/dash/settings/ApiTokensContent.tsx +232 -0
  202. package/src/ui/dash/settings/AvatarContent.tsx +8 -0
  203. package/src/ui/dash/settings/SessionsContent.tsx +159 -0
  204. package/src/ui/dash/settings/SettingsRootContent.tsx +45 -15
  205. package/src/ui/feed/LinkCard.tsx +89 -40
  206. package/src/ui/feed/NoteCard.tsx +39 -25
  207. package/src/ui/feed/PostStatusBadges.tsx +67 -0
  208. package/src/ui/feed/QuoteCard.tsx +38 -23
  209. package/src/ui/feed/ThreadPreview.tsx +90 -26
  210. package/src/ui/feed/TimelineFeed.tsx +3 -2
  211. package/src/ui/feed/TimelineItem.tsx +15 -6
  212. package/src/ui/feed/__tests__/thread-preview.test.ts +107 -0
  213. package/src/ui/feed/thread-preview-state.ts +61 -0
  214. package/src/ui/font-themes.ts +2 -2
  215. package/src/ui/layouts/BaseLayout.tsx +2 -2
  216. package/src/ui/layouts/SiteLayout.tsx +105 -92
  217. package/src/ui/pages/ArchivePage.tsx +923 -98
  218. package/src/ui/pages/ComposePage.tsx +54 -0
  219. package/src/ui/pages/PostPage.tsx +30 -45
  220. package/src/ui/pages/SearchPage.tsx +181 -37
  221. package/src/ui/shared/AdminBreadcrumb.tsx +29 -0
  222. package/src/ui/shared/CollectionsSidebar.tsx +47 -37
  223. package/src/ui/shared/MediaGallery.tsx +445 -149
  224. package/src/ui/shared/PostFooter.tsx +204 -0
  225. package/src/ui/shared/StarRating.tsx +27 -0
  226. package/src/ui/shared/__tests__/format-chars.test.ts +35 -0
  227. package/src/ui/shared/index.ts +0 -1
  228. package/dist/client/assets/url-8Dj-5CLW.js +0 -1
  229. package/src/client/media-upload.ts +0 -161
  230. package/src/client/page-slug-bridge.ts +0 -42
  231. package/src/db/migrations/0000_square_wallflower.sql +0 -118
  232. package/src/db/migrations/0001_add_search_fts.sql +0 -34
  233. package/src/db/migrations/0002_add_media_attachments.sql +0 -3
  234. package/src/db/migrations/0003_add_navigation_links.sql +0 -8
  235. package/src/db/migrations/0004_add_storage_provider.sql +0 -3
  236. package/src/db/migrations/0005_v2_schema_migration.sql +0 -268
  237. package/src/db/migrations/0006_rename_slug_to_path.sql +0 -5
  238. package/src/db/migrations/0007_post_collections_m2m.sql +0 -94
  239. package/src/db/migrations/0008_add_collection_dividers.sql +0 -8
  240. package/src/db/migrations/0009_drop_collection_show_divider.sql +0 -2
  241. package/src/db/migrations/0010_add_performance_indexes.sql +0 -16
  242. package/src/db/migrations/0011_add_path_registry.sql +0 -23
  243. package/src/db/migrations/0012_add_tiptap_columns.sql +0 -2
  244. package/src/db/migrations/0013_replace_featured_with_visibility.sql +0 -8
  245. package/src/db/migrations/meta/0003_snapshot.json +0 -821
  246. package/src/lib/__tests__/sqid.test.ts +0 -65
  247. package/src/lib/sqid.ts +0 -79
  248. package/src/routes/api/__tests__/pages.test.ts +0 -218
  249. package/src/routes/api/pages.ts +0 -73
  250. package/src/routes/dash/__tests__/pages.test.ts +0 -226
  251. package/src/routes/dash/index.tsx +0 -109
  252. package/src/routes/dash/media.tsx +0 -135
  253. package/src/routes/dash/pages.tsx +0 -245
  254. package/src/routes/dash/posts.tsx +0 -338
  255. package/src/routes/dash/redirects.tsx +0 -263
  256. package/src/routes/pages/post.tsx +0 -59
  257. package/src/services/__tests__/page.test.ts +0 -298
  258. package/src/services/__tests__/path-registry.test.ts +0 -165
  259. package/src/services/__tests__/redirect.test.ts +0 -159
  260. package/src/services/page.ts +0 -216
  261. package/src/services/path-registry.ts +0 -160
  262. package/src/services/redirect.ts +0 -97
  263. package/src/ui/dash/PageForm.tsx +0 -187
  264. package/src/ui/dash/PostList.tsx +0 -95
  265. package/src/ui/dash/media/MediaListContent.tsx +0 -206
  266. package/src/ui/dash/media/ViewMediaContent.tsx +0 -208
  267. package/src/ui/dash/pages/PagesContent.tsx +0 -75
  268. package/src/ui/dash/posts/PostForm.tsx +0 -260
  269. package/src/ui/layouts/DashLayout.tsx +0 -247
  270. package/src/ui/pages/SinglePage.tsx +0 -23
  271. package/src/ui/shared/ThreadView.tsx +0 -136
@@ -0,0 +1,204 @@
1
+ /**
2
+ * Post Footer
3
+ *
4
+ * Shared footer for all post cards (feed + detail page).
5
+ * Shows timestamp, collection tags, reply button, and menu trigger.
6
+ */
7
+
8
+ import type { FC } from "hono/jsx";
9
+ import { useLingui } from "@lingui/react/macro";
10
+ import type {
11
+ PostView,
12
+ CollectionTagView,
13
+ PostFooterDisplayOptions,
14
+ } from "../../types.js";
15
+ import { sanitizeUrl } from "../../lib/url.js";
16
+
17
+ interface PostFooterProps {
18
+ post: PostView;
19
+ /** Detail page variant: border-top, shows permalink */
20
+ detail?: boolean;
21
+ display?: PostFooterDisplayOptions;
22
+ }
23
+
24
+ const CollectionTags: FC<{ collections: CollectionTagView[] }> = ({
25
+ collections,
26
+ }) => {
27
+ if (collections.length === 0) return null;
28
+
29
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion -- length checked above
30
+ const first = collections[0]!;
31
+ const rest = collections.slice(1);
32
+
33
+ return (
34
+ <span class="post-collection-tags">
35
+ <span class="post-collection-sep" aria-hidden="true">
36
+ &middot;
37
+ </span>
38
+ <a href={`/c/${first.slug}`} class="post-collection-tag">
39
+ {first.iconHtml && (
40
+ <span
41
+ class="post-collection-icon"
42
+ dangerouslySetInnerHTML={{ __html: first.iconHtml }}
43
+ />
44
+ )}
45
+ {first.title}
46
+ </a>
47
+ {rest.length > 0 && (
48
+ <span class="post-collection-more-wrap">
49
+ <button
50
+ type="button"
51
+ class="post-collection-more"
52
+ data-collection-popover-trigger
53
+ >
54
+ and {rest.length} more
55
+ </button>
56
+ <div class="post-collection-popover" data-collection-popover>
57
+ {collections.map((c) => (
58
+ <a
59
+ key={c.slug}
60
+ href={`/c/${c.slug}`}
61
+ class="post-collection-popover-item"
62
+ >
63
+ {c.iconHtml && (
64
+ <span
65
+ class="post-collection-icon"
66
+ dangerouslySetInnerHTML={{ __html: c.iconHtml }}
67
+ />
68
+ )}
69
+ {c.title}
70
+ </a>
71
+ ))}
72
+ </div>
73
+ </span>
74
+ )}
75
+ </span>
76
+ );
77
+ };
78
+
79
+ export const PostFooter: FC<PostFooterProps> = ({ post, detail, display }) => {
80
+ const { t } = useLingui();
81
+ const safeExternalUrl =
82
+ post.format === "link" && post.url ? sanitizeUrl(post.url) : "";
83
+
84
+ return (
85
+ <footer
86
+ class={`post-menu-footer${detail ? " post-footer-detail" : ""}`}
87
+ data-post-meta
88
+ >
89
+ <div class="post-footer-meta">
90
+ {detail ? (
91
+ <time
92
+ class="dt-published"
93
+ datetime={post.publishedAt}
94
+ title={`${post.publishedAtFormatted} ${post.publishedAtTime} UTC`}
95
+ >
96
+ {post.publishedAtFormatted}
97
+ </time>
98
+ ) : (
99
+ <a
100
+ href={post.permalink}
101
+ class="u-url text-xs text-muted-foreground hover:underline"
102
+ >
103
+ <time
104
+ class="dt-published"
105
+ datetime={post.publishedAt}
106
+ title={`${post.publishedAtFormatted} ${post.publishedAtTime} UTC`}
107
+ >
108
+ {post.publishedAtFormatted}
109
+ </time>
110
+ </a>
111
+ )}
112
+ {safeExternalUrl && (
113
+ <a
114
+ href={safeExternalUrl}
115
+ class="post-footer-external-link"
116
+ target="_blank"
117
+ rel="noopener noreferrer"
118
+ aria-label="Open external link"
119
+ >
120
+ <svg
121
+ xmlns="http://www.w3.org/2000/svg"
122
+ viewBox="0 0 24 24"
123
+ fill="none"
124
+ stroke="currentColor"
125
+ stroke-width="2"
126
+ stroke-linecap="round"
127
+ stroke-linejoin="round"
128
+ >
129
+ <path d="M7 17 17 7" />
130
+ <path d="M9 7h8v8" />
131
+ </svg>
132
+ </a>
133
+ )}
134
+ {detail && (
135
+ <a href={post.permalink} class="u-url ml-4">
136
+ {t({
137
+ message: "Permalink",
138
+ comment: "@context: Link to permanent URL of post",
139
+ })}
140
+ </a>
141
+ )}
142
+ {post.threadRootPermalink && (
143
+ <span class="post-collection-sep" aria-hidden="true">
144
+ &middot;
145
+ </span>
146
+ )}
147
+ {post.threadRootPermalink && (
148
+ <a
149
+ href={post.threadRootPermalink}
150
+ class="text-xs text-muted-foreground hover:underline"
151
+ >
152
+ In thread &rarr;
153
+ </a>
154
+ )}
155
+ <CollectionTags collections={post.collections} />
156
+ </div>
157
+ {!display?.hideActions && (
158
+ <div class="post-menu-actions">
159
+ {post.isLastInThread && (
160
+ <button
161
+ type="button"
162
+ class="reply-trigger"
163
+ aria-label="Reply"
164
+ data-reply-trigger
165
+ >
166
+ <svg
167
+ xmlns="http://www.w3.org/2000/svg"
168
+ width="14"
169
+ height="14"
170
+ viewBox="0 0 24 24"
171
+ fill="none"
172
+ stroke="currentColor"
173
+ stroke-width="2"
174
+ stroke-linecap="round"
175
+ stroke-linejoin="round"
176
+ >
177
+ <polyline points="9 17 4 12 9 7" />
178
+ <path d="M20 18v-2a4 4 0 0 0-4-4H4" />
179
+ </svg>
180
+ </button>
181
+ )}
182
+ <button
183
+ type="button"
184
+ class="post-menu-trigger"
185
+ aria-label="More actions"
186
+ data-post-menu-trigger
187
+ >
188
+ <svg
189
+ xmlns="http://www.w3.org/2000/svg"
190
+ width="16"
191
+ height="16"
192
+ viewBox="0 0 24 24"
193
+ fill="currentColor"
194
+ >
195
+ <circle cx="5" cy="12" r="2" />
196
+ <circle cx="12" cy="12" r="2" />
197
+ <circle cx="19" cy="12" r="2" />
198
+ </svg>
199
+ </button>
200
+ </div>
201
+ )}
202
+ </footer>
203
+ );
204
+ };
@@ -0,0 +1,27 @@
1
+ /**
2
+ * Star Rating (read-only)
3
+ *
4
+ * Displays a 1-5 star rating inline. Returns null when no rating is set.
5
+ */
6
+
7
+ import type { FC } from "hono/jsx";
8
+
9
+ interface StarRatingProps {
10
+ rating?: number;
11
+ }
12
+
13
+ export const StarRating: FC<StarRatingProps> = ({ rating }) => {
14
+ if (!rating || rating <= 0) return null;
15
+
16
+ const stars = [1, 2, 3, 4, 5];
17
+
18
+ return (
19
+ <div class="post-rating" aria-label={`${rating} out of 5`}>
20
+ {stars.map((n) => (
21
+ <span class={n <= rating ? "post-star-filled" : "post-star-empty"}>
22
+
23
+ </span>
24
+ ))}
25
+ </div>
26
+ );
27
+ };
@@ -0,0 +1,35 @@
1
+ import { describe, it, expect } from "vitest";
2
+ import { formatChars } from "../MediaGallery.js";
3
+
4
+ describe("formatChars", () => {
5
+ it("shows raw count below 1000", () => {
6
+ expect(formatChars(0)).toBe("0 chars");
7
+ expect(formatChars(1)).toBe("1 chars");
8
+ expect(formatChars(500)).toBe("500 chars");
9
+ expect(formatChars(999)).toBe("999 chars");
10
+ });
11
+
12
+ it("formats thousands without trailing .0", () => {
13
+ expect(formatChars(1000)).toBe("1k chars");
14
+ expect(formatChars(4000)).toBe("4k chars");
15
+ expect(formatChars(4023)).toBe("4k chars");
16
+ expect(formatChars(10000)).toBe("10k chars");
17
+ expect(formatChars(100000)).toBe("100k chars");
18
+ });
19
+
20
+ it("keeps meaningful decimal in thousands", () => {
21
+ expect(formatChars(1500)).toBe("1.5k chars");
22
+ expect(formatChars(4500)).toBe("4.5k chars");
23
+ expect(formatChars(12300)).toBe("12.3k chars");
24
+ });
25
+
26
+ it("formats millions without trailing .0", () => {
27
+ expect(formatChars(1000000)).toBe("1M chars");
28
+ expect(formatChars(2000000)).toBe("2M chars");
29
+ });
30
+
31
+ it("keeps meaningful decimal in millions", () => {
32
+ expect(formatChars(1500000)).toBe("1.5M chars");
33
+ expect(formatChars(2300000)).toBe("2.3M chars");
34
+ });
35
+ });
@@ -9,4 +9,3 @@ export {
9
9
  type PagePaginationProps,
10
10
  } from "./Pagination.js";
11
11
  export { getPageNumbers } from "../../lib/pagination.js";
12
- export { ThreadView, type ThreadViewProps } from "./ThreadView.js";