convex-cms 0.0.5-alpha.0 → 0.0.5-alpha.2

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 (323) hide show
  1. package/README.md +95 -144
  2. package/admin/README.md +99 -0
  3. package/admin/src/components/AdminLayout.tsx +22 -0
  4. package/admin/src/components/BreakingChangesWarningDialog.tsx +81 -0
  5. package/admin/src/components/BulkActionBar.tsx +190 -0
  6. package/admin/src/components/BulkOperationModal.tsx +177 -0
  7. package/admin/src/components/ContentEntryEditor.tsx +1104 -0
  8. package/admin/src/components/ContentTypeFormModal.tsx +1012 -0
  9. package/admin/src/components/ErrorBoundary.tsx +83 -0
  10. package/admin/src/components/ErrorState.tsx +147 -0
  11. package/admin/src/components/Header.tsx +294 -0
  12. package/admin/src/components/RouteGuard.tsx +264 -0
  13. package/admin/src/components/Sidebar.tsx +90 -0
  14. package/admin/src/components/TaxonomyEditor.tsx +348 -0
  15. package/admin/src/components/TermTree.tsx +533 -0
  16. package/admin/src/components/UploadDropzone.tsx +383 -0
  17. package/admin/src/components/VersionCompare.tsx +250 -0
  18. package/admin/src/components/VersionHistory.tsx +279 -0
  19. package/admin/src/components/VersionRollbackModal.tsx +79 -0
  20. package/admin/src/components/cmsds/CmsButton.tsx +101 -0
  21. package/admin/src/components/cmsds/CmsDialog.tsx +139 -0
  22. package/admin/src/components/cmsds/CmsDropdown.tsx +62 -0
  23. package/admin/src/components/cmsds/CmsEmptyState.tsx +54 -0
  24. package/admin/src/components/cmsds/CmsField.tsx +47 -0
  25. package/admin/src/components/cmsds/CmsPageHeader.tsx +35 -0
  26. package/admin/src/components/cmsds/CmsStatusBadge.tsx +153 -0
  27. package/admin/src/components/cmsds/CmsSurface.tsx +52 -0
  28. package/admin/src/components/cmsds/CmsTable.tsx +164 -0
  29. package/admin/src/components/cmsds/CmsToolbar.tsx +58 -0
  30. package/admin/src/components/cmsds/index.ts +10 -0
  31. package/admin/src/components/fields/BooleanField.tsx +74 -0
  32. package/admin/src/components/fields/CategoryField.tsx +394 -0
  33. package/admin/src/components/fields/DateField.tsx +173 -0
  34. package/admin/src/components/fields/DefaultFieldRenderer.tsx +74 -0
  35. package/admin/src/components/fields/FieldRenderer.tsx +180 -0
  36. package/admin/src/components/fields/FieldWrapper.tsx +57 -0
  37. package/admin/src/components/fields/JsonField.tsx +172 -0
  38. package/admin/src/components/fields/MediaField.tsx +367 -0
  39. package/admin/src/components/fields/MultiSelectField.tsx +118 -0
  40. package/admin/src/components/fields/NumberField.tsx +77 -0
  41. package/admin/src/components/fields/ReferenceField.tsx +386 -0
  42. package/admin/src/components/fields/RichTextField.tsx +171 -0
  43. package/admin/src/components/fields/SelectField.tsx +62 -0
  44. package/admin/src/components/fields/TagField.tsx +325 -0
  45. package/admin/src/components/fields/TextAreaField.tsx +68 -0
  46. package/admin/src/components/fields/TextField.tsx +56 -0
  47. package/admin/src/components/fields/index.ts +54 -0
  48. package/admin/src/components/fields/registry.ts +64 -0
  49. package/admin/src/components/fields/types.ts +217 -0
  50. package/admin/src/components/filters/TaxonomyFilter.tsx +254 -0
  51. package/admin/src/components/filters/index.ts +1 -0
  52. package/admin/src/components/index.ts +8 -0
  53. package/admin/src/components/media/MediaAssetActions.tsx +115 -0
  54. package/admin/src/components/media/MediaAssetEditDialog.tsx +217 -0
  55. package/admin/src/components/media/MediaBulkActionBar.tsx +51 -0
  56. package/admin/src/components/media/MediaFolderActions.tsx +69 -0
  57. package/admin/src/components/media/MediaFolderEditDialog.tsx +126 -0
  58. package/admin/src/components/media/MediaMoveModal.tsx +179 -0
  59. package/admin/src/components/media/MediaPreviewModal.tsx +370 -0
  60. package/admin/src/components/media/MediaTaxonomyPicker.tsx +304 -0
  61. package/admin/src/components/media/MediaTrashBulkActionBar.tsx +59 -0
  62. package/admin/src/components/ui/accordion.tsx +64 -0
  63. package/admin/src/components/ui/alert-dialog.tsx +155 -0
  64. package/admin/src/components/ui/alert.tsx +66 -0
  65. package/admin/src/components/ui/avatar.tsx +53 -0
  66. package/admin/src/components/ui/badge.tsx +46 -0
  67. package/admin/src/components/ui/breadcrumb.tsx +109 -0
  68. package/admin/src/components/ui/button.tsx +62 -0
  69. package/admin/src/components/ui/calendar.tsx +220 -0
  70. package/admin/src/components/ui/card.tsx +92 -0
  71. package/admin/src/components/ui/checkbox.tsx +30 -0
  72. package/admin/src/components/ui/command.tsx +182 -0
  73. package/admin/src/components/ui/dialog.tsx +143 -0
  74. package/admin/src/components/ui/dropdown-menu.tsx +257 -0
  75. package/admin/src/components/ui/form.tsx +167 -0
  76. package/admin/src/components/ui/input.tsx +21 -0
  77. package/admin/src/components/ui/label.tsx +24 -0
  78. package/admin/src/components/ui/popover.tsx +46 -0
  79. package/admin/src/components/ui/scroll-area.tsx +56 -0
  80. package/admin/src/components/ui/select.tsx +190 -0
  81. package/admin/src/components/ui/separator.tsx +26 -0
  82. package/admin/src/components/ui/sheet.tsx +137 -0
  83. package/admin/src/components/ui/sidebar.tsx +724 -0
  84. package/admin/src/components/ui/skeleton.tsx +13 -0
  85. package/admin/src/components/ui/sonner.tsx +38 -0
  86. package/admin/src/components/ui/switch.tsx +31 -0
  87. package/admin/src/components/ui/table.tsx +114 -0
  88. package/admin/src/components/ui/tabs.tsx +66 -0
  89. package/admin/src/components/ui/textarea.tsx +18 -0
  90. package/admin/src/components/ui/tooltip.tsx +61 -0
  91. package/admin/src/contexts/AdminConfigContext.tsx +30 -0
  92. package/admin/src/contexts/AuthContext.tsx +330 -0
  93. package/admin/src/contexts/BreadcrumbContext.tsx +49 -0
  94. package/admin/src/contexts/SettingsConfigContext.tsx +57 -0
  95. package/admin/src/contexts/ThemeContext.tsx +91 -0
  96. package/admin/src/contexts/index.ts +20 -0
  97. package/admin/src/embed/components/EmbedHeader.tsx +103 -0
  98. package/admin/src/embed/components/EmbedLayout.tsx +29 -0
  99. package/admin/src/embed/components/EmbedSidebar.tsx +119 -0
  100. package/admin/src/embed/components/index.ts +3 -0
  101. package/admin/src/embed/contexts/ApiContext.tsx +32 -0
  102. package/admin/src/embed/index.tsx +184 -0
  103. package/admin/src/embed/navigation.tsx +202 -0
  104. package/admin/src/embed/pages/Content.tsx +19 -0
  105. package/admin/src/embed/pages/ContentTypes.tsx +19 -0
  106. package/admin/src/embed/pages/Dashboard.tsx +19 -0
  107. package/admin/src/embed/pages/Media.tsx +19 -0
  108. package/admin/src/embed/pages/Settings.tsx +22 -0
  109. package/admin/src/embed/pages/Taxonomies.tsx +22 -0
  110. package/admin/src/embed/pages/Trash.tsx +22 -0
  111. package/admin/src/embed/pages/index.ts +7 -0
  112. package/admin/src/embed/types.ts +24 -0
  113. package/admin/src/hooks/index.ts +2 -0
  114. package/admin/src/hooks/use-mobile.ts +19 -0
  115. package/admin/src/hooks/useBreadcrumbLabel.ts +15 -0
  116. package/admin/src/hooks/usePermissions.ts +211 -0
  117. package/admin/src/lib/admin-config.ts +111 -0
  118. package/admin/src/lib/cn.ts +6 -0
  119. package/admin/src/lib/config.server.ts +56 -0
  120. package/admin/src/lib/convex.ts +26 -0
  121. package/admin/src/lib/embed-adapter.ts +80 -0
  122. package/admin/src/lib/icons.tsx +96 -0
  123. package/admin/src/lib/loadAdminConfig.ts +92 -0
  124. package/admin/src/lib/motion.ts +29 -0
  125. package/admin/src/lib/navigation.ts +43 -0
  126. package/admin/src/lib/tanstack-adapter.ts +82 -0
  127. package/admin/src/pages/ContentPage.tsx +337 -0
  128. package/admin/src/pages/ContentTypesPage.tsx +457 -0
  129. package/admin/src/pages/DashboardPage.tsx +163 -0
  130. package/admin/src/pages/MediaPage.tsx +34 -0
  131. package/admin/src/pages/SettingsPage.tsx +486 -0
  132. package/admin/src/pages/TaxonomiesPage.tsx +289 -0
  133. package/admin/src/pages/TrashPage.tsx +421 -0
  134. package/admin/src/pages/index.ts +14 -0
  135. package/admin/src/routeTree.gen.ts +262 -0
  136. package/admin/src/router.tsx +22 -0
  137. package/admin/src/routes/__root.tsx +250 -0
  138. package/admin/src/routes/content-types.tsx +20 -0
  139. package/admin/src/routes/content.tsx +20 -0
  140. package/admin/src/routes/entries/$entryId.tsx +107 -0
  141. package/admin/src/routes/entries/new.$contentTypeId.tsx +69 -0
  142. package/admin/src/routes/entries/type/$contentTypeId.tsx +503 -0
  143. package/admin/src/routes/index.tsx +20 -0
  144. package/admin/src/routes/media.tsx +1095 -0
  145. package/admin/src/routes/settings.tsx +20 -0
  146. package/admin/src/routes/taxonomies.tsx +20 -0
  147. package/admin/src/routes/trash.tsx +20 -0
  148. package/admin/src/styles/globals.css +69 -0
  149. package/admin/src/styles/tailwind-config.css +74 -0
  150. package/admin/src/styles/theme.css +73 -0
  151. package/admin/src/types/index.ts +221 -0
  152. package/admin/src/utils/errorParsing.ts +163 -0
  153. package/admin/src/utils/index.ts +5 -0
  154. package/admin/src/vite-env.d.ts +14 -0
  155. package/admin/tailwind.preset.cjs +102 -0
  156. package/admin-dist/nitro.json +1 -1
  157. package/admin-dist/public/assets/{CmsEmptyState-CiMQwSQV.js → CmsEmptyState-CkqBIab3.js} +1 -1
  158. package/admin-dist/public/assets/{CmsPageHeader-ohOq0luT.js → CmsPageHeader-CUtl5MMG.js} +1 -1
  159. package/admin-dist/public/assets/{CmsStatusBadge-BdNf0V9v.js → CmsStatusBadge-CUYFgEe-.js} +1 -1
  160. package/admin-dist/public/assets/{CmsSurface-CWup6Jh7.js → CmsSurface-CsJfAVa3.js} +1 -1
  161. package/admin-dist/public/assets/{CmsToolbar-cEBlCHa3.js → CmsToolbar-CnfbcxeP.js} +1 -1
  162. package/admin-dist/public/assets/{ContentEntryEditor-BY5ypfUs.js → ContentEntryEditor-BU220CCy.js} +1 -1
  163. package/admin-dist/public/assets/TaxonomyFilter-CWCxC5HZ.js +1 -0
  164. package/admin-dist/public/assets/_contentTypeId-DK8cskRt.js +1 -0
  165. package/admin-dist/public/assets/{_entryId-BpSmrfAm.js → _entryId-CuVMExbb.js} +1 -1
  166. package/admin-dist/public/assets/{alert-Bf2l8kxw.js → alert-CF1BSzGR.js} +1 -1
  167. package/admin-dist/public/assets/{badge-qPrc4AUM.js → badge-CmuOIVKp.js} +1 -1
  168. package/admin-dist/public/assets/{circle-check-big-Dgozy3vV.js → circle-check-big-BKDVG6DU.js} +1 -1
  169. package/admin-dist/public/assets/{command-QOmNhlb0.js → command-XJxnF2Sd.js} +1 -1
  170. package/admin-dist/public/assets/content-QBUxdxbS.js +1 -0
  171. package/admin-dist/public/assets/content-types-CrNEm8Hf.js +2 -0
  172. package/admin-dist/public/assets/globals-B7Wsfh_v.css +1 -0
  173. package/admin-dist/public/assets/index-C7xOwudI.js +1 -0
  174. package/admin-dist/public/assets/{label-DCsUdvFh.js → label-CHCnXeBk.js} +1 -1
  175. package/admin-dist/public/assets/{link-2-Czw1N61H.js → link-2-Bb34judH.js} +1 -1
  176. package/admin-dist/public/assets/{list-DtCsXj8-.js → list-9Pzt48ld.js} +1 -1
  177. package/admin-dist/public/assets/{main-CXgkZMhe.js → main-CjQ2VI9L.js} +3 -3
  178. package/admin-dist/public/assets/media-Dc5PWt2Q.js +1 -0
  179. package/admin-dist/public/assets/{new._contentTypeId-CoTDxKzf.js → new._contentTypeId-C_I4YxIa.js} +1 -1
  180. package/admin-dist/public/assets/{plus-xCFJK0RC.js → plus-Ceef7DHk.js} +1 -1
  181. package/admin-dist/public/assets/{rotate-ccw-DIqK63wY.js → rotate-ccw-7k7-4VUq.js} +1 -1
  182. package/admin-dist/public/assets/{scroll-area-B-yrE66a.js → scroll-area-CC6wujnp.js} +1 -1
  183. package/admin-dist/public/assets/{search-CbCbboeU.js → search-DwoUV2pv.js} +1 -1
  184. package/admin-dist/public/assets/{select-Co3TZFJb.js → select-hOZTp8aC.js} +1 -1
  185. package/admin-dist/public/assets/{settings-BspTTv_o.js → settings-t2PbCZh4.js} +1 -1
  186. package/admin-dist/public/assets/{switch-CfavASmR.js → switch-jX2pDaNU.js} +1 -1
  187. package/admin-dist/public/assets/{tabs-CN5s5u2W.js → tabs-q4EbZk7c.js} +1 -1
  188. package/admin-dist/public/assets/{tanstack-adapter-npeE3RdY.js → tanstack-adapter-B-Glm4kH.js} +1 -1
  189. package/admin-dist/public/assets/taxonomies-kyk5P4ZW.js +1 -0
  190. package/admin-dist/public/assets/{textarea-BJ0XFZpT.js → textarea-B6SfBmr0.js} +1 -1
  191. package/admin-dist/public/assets/trash-BOCnIznD.js +1 -0
  192. package/admin-dist/public/assets/{triangle-alert-BZRcqsUg.js → triangle-alert-CXFIO_Gu.js} +1 -1
  193. package/admin-dist/public/assets/{useBreadcrumbLabel-DwZlwvFF.js → useBreadcrumbLabel-_6qBagc3.js} +1 -1
  194. package/admin-dist/public/assets/{usePermissions-C1JQhfqb.js → usePermissions-M1ijZ7a6.js} +1 -1
  195. package/admin-dist/server/_ssr/{CmsButton-B45JAKR1.mjs → CmsButton-DOiTVKQq.mjs} +1 -1
  196. package/admin-dist/server/_ssr/{CmsEmptyState-D_BQFAVR.mjs → CmsEmptyState-fbnGt3LD.mjs} +2 -2
  197. package/admin-dist/server/_ssr/{CmsPageHeader-CrUZA59A.mjs → CmsPageHeader-DHRrdOZa.mjs} +1 -1
  198. package/admin-dist/server/_ssr/{CmsStatusBadge-B-sj6yaj.mjs → CmsStatusBadge-s7obWbKZ.mjs} +2 -2
  199. package/admin-dist/server/_ssr/{CmsSurface-DKJZhpjk.mjs → CmsSurface-rFoYjb62.mjs} +1 -1
  200. package/admin-dist/server/_ssr/{CmsToolbar-ByaW5iXf.mjs → CmsToolbar-zTE45z2q.mjs} +2 -2
  201. package/admin-dist/server/_ssr/{ContentEntryEditor-D3_Jb1dq.mjs → ContentEntryEditor-BLoEjT_m.mjs} +12 -12
  202. package/admin-dist/server/_ssr/{TaxonomyFilter-BRJkuCtA.mjs → TaxonomyFilter-XAtaJC2z.mjs} +5 -5
  203. package/admin-dist/server/_ssr/{_contentTypeId-B9kA6CaM.mjs → _contentTypeId-Csl4822C.mjs} +13 -13
  204. package/admin-dist/server/_ssr/{_entryId-BddcMkZN.mjs → _entryId-D8alLFBx.mjs} +15 -15
  205. package/admin-dist/server/_ssr/_tanstack-start-manifest_v-BffZedId.mjs +4 -0
  206. package/admin-dist/server/_ssr/{command-CGtVr8Gb.mjs → command-C0Di14--.mjs} +1 -1
  207. package/admin-dist/server/_ssr/{content-D1tbeOd0.mjs → content-CT-FPsmV.mjs} +12 -55
  208. package/admin-dist/server/_ssr/{content-types-BZqY_BER.mjs → content-types-C8cBFdzE.mjs} +15 -46
  209. package/admin-dist/server/_ssr/{index-BIdq4xaC.mjs → index-BJtcrEc-.mjs} +5 -5
  210. package/admin-dist/server/_ssr/index.mjs +2 -2
  211. package/admin-dist/server/_ssr/{label-T-QNKAr6.mjs → label-qn2Afwl4.mjs} +1 -1
  212. package/admin-dist/server/_ssr/{media-C-xqjBrl.mjs → media-qv5IAsMZ.mjs} +14 -14
  213. package/admin-dist/server/_ssr/{new._contentTypeId-DWic9cRq.mjs → new._contentTypeId-DdGyrhqs.mjs} +13 -13
  214. package/admin-dist/server/_ssr/{router-D1BMAMJT.mjs → router-nSVkxb6Y.mjs} +11 -11
  215. package/admin-dist/server/_ssr/{scroll-area-C0pic_WA.mjs → scroll-area-BCinP455.mjs} +1 -1
  216. package/admin-dist/server/_ssr/{select-CqmuN2F6.mjs → select-BKQlQScw.mjs} +1 -1
  217. package/admin-dist/server/_ssr/{settings-CAkncGGV.mjs → settings-BCr2KQlk.mjs} +55 -40
  218. package/admin-dist/server/_ssr/{switch-CgmuJkT9.mjs → switch-BaOi42fE.mjs} +1 -1
  219. package/admin-dist/server/_ssr/{tabs-CnMj0aRy.mjs → tabs-DYXEi9kq.mjs} +2 -2
  220. package/admin-dist/server/_ssr/{tanstack-adapter-BXZrMauE.mjs → tanstack-adapter-Bsz8kha-.mjs} +1 -1
  221. package/admin-dist/server/_ssr/{taxonomies-thl3BfVm.mjs → taxonomies-CueMHTbE.mjs} +30 -19
  222. package/admin-dist/server/_ssr/{textarea-4K5OJgeh.mjs → textarea-CI0Jqx2x.mjs} +1 -1
  223. package/admin-dist/server/_ssr/{trash-B40Gx5zP.mjs → trash-DE6W8GoX.mjs} +20 -17
  224. package/admin-dist/server/_ssr/{useBreadcrumbLabel-rn-fL4zV.mjs → useBreadcrumbLabel-B5Yi72lM.mjs} +1 -1
  225. package/admin-dist/server/_ssr/{usePermissions-CKeM6_Vw.mjs → usePermissions-C3nZ-Izm.mjs} +1 -1
  226. package/admin-dist/server/index.mjs +183 -190
  227. package/dist/client/admin/bulk.d.ts +79 -0
  228. package/dist/client/admin/bulk.d.ts.map +1 -0
  229. package/dist/client/admin/bulk.js +72 -0
  230. package/dist/client/admin/bulk.js.map +1 -0
  231. package/dist/client/admin/contentLock.d.ts +118 -0
  232. package/dist/client/admin/contentLock.d.ts.map +1 -0
  233. package/dist/client/admin/contentLock.js +81 -0
  234. package/dist/client/admin/contentLock.js.map +1 -0
  235. package/dist/client/{adminApi.d.ts → admin/contentTypes.d.ts} +39 -1134
  236. package/dist/client/admin/contentTypes.d.ts.map +1 -0
  237. package/dist/client/admin/contentTypes.js +122 -0
  238. package/dist/client/admin/contentTypes.js.map +1 -0
  239. package/dist/client/admin/dashboard.d.ts +16 -0
  240. package/dist/client/admin/dashboard.d.ts.map +1 -0
  241. package/dist/client/admin/dashboard.js +33 -0
  242. package/dist/client/admin/dashboard.js.map +1 -0
  243. package/dist/client/admin/entries.d.ts +358 -0
  244. package/dist/client/admin/entries.d.ts.map +1 -0
  245. package/dist/client/admin/entries.js +220 -0
  246. package/dist/client/admin/entries.js.map +1 -0
  247. package/dist/client/admin/index.d.ts +6568 -0
  248. package/dist/client/admin/index.d.ts.map +1 -0
  249. package/dist/client/admin/index.js +305 -0
  250. package/dist/client/admin/index.js.map +1 -0
  251. package/dist/client/admin/media.d.ts +1038 -0
  252. package/dist/client/admin/media.d.ts.map +1 -0
  253. package/dist/client/admin/media.js +489 -0
  254. package/dist/client/admin/media.js.map +1 -0
  255. package/dist/client/admin/taxonomies.d.ts +339 -0
  256. package/dist/client/admin/taxonomies.d.ts.map +1 -0
  257. package/dist/client/admin/taxonomies.js +364 -0
  258. package/dist/client/admin/taxonomies.js.map +1 -0
  259. package/dist/client/admin/trash.d.ts +91 -0
  260. package/dist/client/admin/trash.d.ts.map +1 -0
  261. package/dist/client/admin/trash.js +71 -0
  262. package/dist/client/admin/trash.js.map +1 -0
  263. package/dist/client/admin/types.d.ts +320 -0
  264. package/dist/client/admin/types.d.ts.map +1 -0
  265. package/dist/client/admin/types.js +7 -0
  266. package/dist/client/admin/types.js.map +1 -0
  267. package/dist/client/admin/validators.d.ts +3886 -0
  268. package/dist/client/admin/validators.d.ts.map +1 -0
  269. package/dist/client/admin/validators.js +322 -0
  270. package/dist/client/admin/validators.js.map +1 -0
  271. package/dist/client/admin/versions.d.ts +106 -0
  272. package/dist/client/admin/versions.d.ts.map +1 -0
  273. package/dist/client/admin/versions.js +57 -0
  274. package/dist/client/admin/versions.js.map +1 -0
  275. package/dist/client/adminApiTypes.d.ts +27 -0
  276. package/dist/client/adminApiTypes.d.ts.map +1 -0
  277. package/dist/client/adminApiTypes.js +12 -0
  278. package/dist/client/adminApiTypes.js.map +1 -0
  279. package/dist/client/{admin-config.d.ts → adminConfig.d.ts} +2 -2
  280. package/dist/client/adminConfig.d.ts.map +1 -0
  281. package/dist/client/{admin-config.js → adminConfig.js} +1 -1
  282. package/dist/client/adminConfig.js.map +1 -0
  283. package/dist/client/agentTools.d.ts +4 -4
  284. package/dist/client/index.d.ts +2 -2
  285. package/dist/client/index.d.ts.map +1 -1
  286. package/dist/client/index.js +15 -2
  287. package/dist/client/index.js.map +1 -1
  288. package/dist/component/contentEntries.d.ts +4 -4
  289. package/dist/component/contentEntryMutations.d.ts +46 -0
  290. package/dist/component/contentEntryMutations.d.ts.map +1 -1
  291. package/dist/component/contentEntryMutations.js +1 -1
  292. package/dist/component/contentEntryMutations.js.map +1 -1
  293. package/dist/component/contentTypeMigration.d.ts +1 -1
  294. package/dist/component/contentTypeMutations.d.ts +22 -0
  295. package/dist/component/contentTypeMutations.d.ts.map +1 -1
  296. package/dist/component/contentTypeMutations.js +1 -1
  297. package/dist/component/contentTypeMutations.js.map +1 -1
  298. package/dist/component/mediaAssetMutations.d.ts +47 -0
  299. package/dist/component/mediaAssetMutations.d.ts.map +1 -1
  300. package/dist/component/mediaAssetMutations.js +1 -1
  301. package/dist/component/mediaAssetMutations.js.map +1 -1
  302. package/dist/component/schema.d.ts +9 -0
  303. package/dist/component/schema.d.ts.map +1 -1
  304. package/dist/component/schema.js +1 -1
  305. package/dist/component/schema.js.map +1 -1
  306. package/package.json +85 -3
  307. package/admin-dist/public/assets/ErrorState-C4nJ-ml4.js +0 -1
  308. package/admin-dist/public/assets/TaxonomyFilter-BgE_SR_O.js +0 -1
  309. package/admin-dist/public/assets/_contentTypeId-DtZectcC.js +0 -1
  310. package/admin-dist/public/assets/content-OEBGlxg1.js +0 -1
  311. package/admin-dist/public/assets/content-types-CjQliqVV.js +0 -2
  312. package/admin-dist/public/assets/globals-hAmgC66w.css +0 -1
  313. package/admin-dist/public/assets/index-BH_ECMhv.js +0 -1
  314. package/admin-dist/public/assets/media-DTJ3-ViE.js +0 -1
  315. package/admin-dist/public/assets/taxonomies-CgG46fIF.js +0 -1
  316. package/admin-dist/public/assets/trash-B3daldm5.js +0 -1
  317. package/admin-dist/server/_ssr/ErrorState-cI-bKLez.mjs +0 -89
  318. package/admin-dist/server/_ssr/_tanstack-start-manifest_v-Dd7AmelK.mjs +0 -4
  319. package/dist/client/admin-config.d.ts.map +0 -1
  320. package/dist/client/admin-config.js.map +0 -1
  321. package/dist/client/adminApi.d.ts.map +0 -1
  322. package/dist/client/adminApi.js +0 -736
  323. package/dist/client/adminApi.js.map +0 -1
@@ -0,0 +1,367 @@
1
+ import { useState, useCallback } from 'react'
2
+ import { useQuery } from 'convex/react'
3
+ import { api } from '../../../convex/_generated/api'
4
+ // IDs are strings when crossing component boundaries
5
+ import { FieldWrapper } from './FieldWrapper'
6
+ import type { BaseFieldProps } from './types'
7
+ import { UploadDropzone, type UploadedFile } from '../UploadDropzone'
8
+ import {
9
+ Dialog,
10
+ DialogContent,
11
+ DialogHeader,
12
+ DialogTitle,
13
+ DialogFooter,
14
+ } from '~/components/ui/dialog'
15
+ import { Tabs, TabsContent, TabsList, TabsTrigger } from '~/components/ui/tabs'
16
+ import { Input } from '~/components/ui/input'
17
+ import {
18
+ Select,
19
+ SelectContent,
20
+ SelectItem,
21
+ SelectTrigger,
22
+ SelectValue,
23
+ } from '~/components/ui/select'
24
+ import { CmsButton } from '~/components/cmsds/CmsButton'
25
+ import { CmsEmptyState } from '~/components/cmsds/CmsEmptyState'
26
+ import { cn } from '~/lib/cn'
27
+ import {
28
+ Image,
29
+ Video,
30
+ Music,
31
+ FileText,
32
+ File,
33
+ Pencil,
34
+ X,
35
+ Upload,
36
+ FolderOpen,
37
+ Search,
38
+ Check,
39
+ } from 'lucide-react'
40
+
41
+ export interface MediaFieldProps extends BaseFieldProps<string | null> {
42
+ placeholder?: string
43
+ }
44
+
45
+ function getMediaTypeIcon(type: string, className = 'size-6') {
46
+ const iconProps = { className }
47
+ switch (type) {
48
+ case 'image':
49
+ return <Image {...iconProps} />
50
+ case 'video':
51
+ return <Video {...iconProps} />
52
+ case 'audio':
53
+ return <Music {...iconProps} />
54
+ case 'document':
55
+ return <FileText {...iconProps} />
56
+ default:
57
+ return <File {...iconProps} />
58
+ }
59
+ }
60
+
61
+ type MediaType = 'image' | 'video' | 'audio' | 'document' | 'other'
62
+
63
+ function getMediaTypeFromMimeType(mimeType?: string): MediaType {
64
+ if (!mimeType) return 'other'
65
+ if (mimeType.startsWith('image/')) return 'image'
66
+ if (mimeType.startsWith('video/')) return 'video'
67
+ if (mimeType.startsWith('audio/')) return 'audio'
68
+ if (
69
+ mimeType === 'application/pdf' ||
70
+ mimeType.includes('document') ||
71
+ mimeType.includes('sheet') ||
72
+ mimeType.includes('presentation') ||
73
+ mimeType.startsWith('text/')
74
+ ) {
75
+ return 'document'
76
+ }
77
+ return 'other'
78
+ }
79
+
80
+ function formatFileSize(bytes: number): string {
81
+ if (bytes === 0) return '0 B'
82
+ const k = 1024
83
+ const sizes = ['B', 'KB', 'MB', 'GB']
84
+ const i = Math.floor(Math.log(bytes) / Math.log(k))
85
+ return `${parseFloat((bytes / Math.pow(k, i)).toFixed(1))} ${sizes[i]}`
86
+ }
87
+
88
+ export function MediaField({
89
+ field,
90
+ value,
91
+ onChange,
92
+ error,
93
+ disabled = false,
94
+ readOnly = false,
95
+ className = '',
96
+ id,
97
+ placeholder = 'Select media...',
98
+ }: MediaFieldProps) {
99
+ const fieldId = id || `field-${field.name}`
100
+ const [showPicker, setShowPicker] = useState(false)
101
+ const [activeTab, setActiveTab] = useState<string>('browse')
102
+ const [searchQuery, setSearchQuery] = useState('')
103
+ const [typeFilter, setTypeFilter] = useState<string>('')
104
+
105
+ const allowedMimeTypes = field.options?.allowedMimeTypes ?? []
106
+
107
+ const selectedAsset = useQuery(
108
+ api.media.getAsset,
109
+ value ? { id: value } : 'skip'
110
+ )
111
+
112
+ const assetsResult = useQuery(
113
+ api.media.listAssets,
114
+ showPicker
115
+ ? {
116
+ type: typeFilter
117
+ ? (typeFilter as 'image' | 'video' | 'audio' | 'document' | 'other')
118
+ : undefined,
119
+ search: searchQuery || undefined,
120
+ paginationOpts: { numItems: 50, cursor: null },
121
+ }
122
+ : 'skip'
123
+ )
124
+
125
+ const handleSelect = useCallback(
126
+ (assetId: string) => {
127
+ onChange(assetId)
128
+ setShowPicker(false)
129
+ },
130
+ [onChange]
131
+ )
132
+
133
+ const handleClear = useCallback(() => {
134
+ onChange(null)
135
+ }, [onChange])
136
+
137
+ const handleUploadComplete = useCallback((_results: UploadedFile[]) => {
138
+ setActiveTab('browse')
139
+ }, [])
140
+
141
+ const filteredAssets = assetsResult?.page?.filter((asset) => {
142
+ if (allowedMimeTypes.length === 0) return true
143
+ return allowedMimeTypes.some((pattern) => {
144
+ if (pattern.endsWith('/*')) {
145
+ const prefix = pattern.slice(0, -1)
146
+ return asset.mimeType?.startsWith(prefix)
147
+ }
148
+ return asset.mimeType === pattern
149
+ })
150
+ })
151
+
152
+ return (
153
+ <FieldWrapper field={field} error={error} className={className} id={fieldId}>
154
+ <div className="space-y-2">
155
+ {value && selectedAsset ? (
156
+ (() => {
157
+ const selectedMediaType = getMediaTypeFromMimeType(selectedAsset.mimeType)
158
+ return (
159
+ <div className="flex items-center gap-3 rounded-lg border border-border bg-card p-3">
160
+ <div className="flex size-12 shrink-0 items-center justify-center overflow-hidden rounded-md bg-muted">
161
+ {selectedMediaType === 'image' && selectedAsset.url ? (
162
+ <img
163
+ src={selectedAsset.url}
164
+ alt={selectedAsset.title || selectedAsset.name}
165
+ className="size-full object-cover"
166
+ />
167
+ ) : (
168
+ <div className="text-muted-foreground">
169
+ {getMediaTypeIcon(selectedMediaType)}
170
+ </div>
171
+ )}
172
+ </div>
173
+ <div className="min-w-0 flex-1">
174
+ <p
175
+ className="truncate text-sm font-medium text-foreground"
176
+ title={selectedAsset.name}
177
+ >
178
+ {selectedAsset.name}
179
+ </p>
180
+ <div className="flex items-center gap-2 text-xs text-muted-foreground">
181
+ <span className="capitalize">{selectedMediaType}</span>
182
+ <span>•</span>
183
+ <span>{formatFileSize(selectedAsset.size ?? 0)}</span>
184
+ </div>
185
+ </div>
186
+ {!disabled && !readOnly && (
187
+ <div className="flex shrink-0 items-center gap-1">
188
+ <CmsButton
189
+ variant="ghost"
190
+ size="icon-sm"
191
+ onClick={() => setShowPicker(true)}
192
+ title="Change media"
193
+ >
194
+ <Pencil className="size-4" />
195
+ </CmsButton>
196
+ <CmsButton
197
+ variant="ghost"
198
+ size="icon-sm"
199
+ onClick={handleClear}
200
+ title="Remove media"
201
+ >
202
+ <X className="size-4" />
203
+ </CmsButton>
204
+ </div>
205
+ )}
206
+ </div>
207
+ )
208
+ })()
209
+ ) : (
210
+ <button
211
+ type="button"
212
+ className={cn(
213
+ 'flex w-full flex-col items-center justify-center gap-2 rounded-lg border-2 border-dashed border-border bg-muted/30 p-6 text-center transition-colors',
214
+ 'hover:border-primary/50 hover:bg-muted/50',
215
+ 'focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring',
216
+ disabled && 'cursor-not-allowed opacity-50'
217
+ )}
218
+ onClick={() => setShowPicker(true)}
219
+ disabled={disabled || readOnly}
220
+ >
221
+ <div className="flex size-10 items-center justify-center rounded-full bg-muted">
222
+ <Image className="size-5 text-muted-foreground" />
223
+ </div>
224
+ <span className="text-sm font-medium text-foreground">{placeholder}</span>
225
+ <span className="text-xs text-muted-foreground">
226
+ Click to browse or upload media
227
+ </span>
228
+ </button>
229
+ )}
230
+ </div>
231
+
232
+ <Dialog open={showPicker} onOpenChange={setShowPicker}>
233
+ <DialogContent className="max-w-2xl">
234
+ <DialogHeader>
235
+ <DialogTitle>Select Media</DialogTitle>
236
+ </DialogHeader>
237
+
238
+ <Tabs value={activeTab} onValueChange={setActiveTab}>
239
+ <TabsList className="w-full">
240
+ <TabsTrigger value="browse" className="flex-1 gap-2">
241
+ <FolderOpen className="size-4" />
242
+ Browse Library
243
+ </TabsTrigger>
244
+ <TabsTrigger value="upload" className="flex-1 gap-2">
245
+ <Upload className="size-4" />
246
+ Upload New
247
+ </TabsTrigger>
248
+ </TabsList>
249
+
250
+ <TabsContent value="browse" className="mt-4 space-y-4">
251
+ <div className="flex gap-2">
252
+ <div className="relative flex-1">
253
+ <Search className="absolute left-3 top-1/2 size-4 -translate-y-1/2 text-muted-foreground" />
254
+ <Input
255
+ type="search"
256
+ placeholder="Search files..."
257
+ value={searchQuery}
258
+ onChange={(e) => setSearchQuery(e.target.value)}
259
+ className="pl-9"
260
+ />
261
+ </div>
262
+ <Select
263
+ value={typeFilter || 'all'}
264
+ onValueChange={(v) => setTypeFilter(v === 'all' ? '' : v)}
265
+ >
266
+ <SelectTrigger className="w-[140px]">
267
+ <SelectValue placeholder="All Types" />
268
+ </SelectTrigger>
269
+ <SelectContent>
270
+ <SelectItem value="all">All Types</SelectItem>
271
+ <SelectItem value="image">Images</SelectItem>
272
+ <SelectItem value="video">Videos</SelectItem>
273
+ <SelectItem value="audio">Audio</SelectItem>
274
+ <SelectItem value="document">Documents</SelectItem>
275
+ <SelectItem value="other">Other</SelectItem>
276
+ </SelectContent>
277
+ </Select>
278
+ </div>
279
+
280
+ {assetsResult === undefined ? (
281
+ <div className="flex flex-col items-center justify-center py-12">
282
+ <div className="size-6 animate-spin rounded-full border-2 border-muted border-t-primary" />
283
+ <p className="mt-2 text-sm text-muted-foreground">Loading media...</p>
284
+ </div>
285
+ ) : filteredAssets && filteredAssets.length > 0 ? (
286
+ <div className="grid max-h-[300px] grid-cols-4 gap-2 overflow-y-auto">
287
+ {filteredAssets.map((asset) => {
288
+ const assetMediaType = getMediaTypeFromMimeType(asset.mimeType)
289
+ return (
290
+ <button
291
+ key={asset._id}
292
+ type="button"
293
+ className={cn(
294
+ 'group relative flex flex-col overflow-hidden rounded-lg border bg-card transition-all',
295
+ 'hover:border-primary hover:shadow-sm',
296
+ value === asset._id && 'border-primary ring-2 ring-primary/20'
297
+ )}
298
+ onClick={() => handleSelect(asset._id)}
299
+ >
300
+ <div className="aspect-square bg-muted">
301
+ {assetMediaType === 'image' && asset.url ? (
302
+ <img
303
+ src={asset.url}
304
+ alt={asset.title || asset.name}
305
+ className="size-full object-cover"
306
+ />
307
+ ) : (
308
+ <div className="flex size-full items-center justify-center text-muted-foreground">
309
+ {getMediaTypeIcon(assetMediaType)}
310
+ </div>
311
+ )}
312
+ </div>
313
+ <div className="p-1.5">
314
+ <p
315
+ className="truncate text-xs font-medium"
316
+ title={asset.name}
317
+ >
318
+ {asset.name}
319
+ </p>
320
+ <p className="text-[10px] text-muted-foreground">
321
+ {formatFileSize(asset.size ?? 0)}
322
+ </p>
323
+ </div>
324
+ {value === asset._id && (
325
+ <div className="absolute right-1 top-1 flex size-5 items-center justify-center rounded-full bg-primary text-primary-foreground">
326
+ <Check className="size-3" />
327
+ </div>
328
+ )}
329
+ </button>
330
+ )
331
+ })}
332
+ </div>
333
+ ) : (
334
+ <CmsEmptyState
335
+ icon={<Image className="size-6" />}
336
+ title="No media found"
337
+ description="Upload some media to get started"
338
+ action={{
339
+ label: 'Upload Media',
340
+ onClick: () => setActiveTab('upload'),
341
+ }}
342
+ />
343
+ )}
344
+ </TabsContent>
345
+
346
+ <TabsContent value="upload" className="mt-4">
347
+ <UploadDropzone
348
+ generateUploadUrl={api.media.generateUploadUrl}
349
+ createAsset={api.media.createAsset}
350
+ onUploadComplete={handleUploadComplete}
351
+ allowedMimeTypes={allowedMimeTypes}
352
+ maxFileSize={field.options?.maxFileSize}
353
+ maxConcurrentUploads={3}
354
+ />
355
+ </TabsContent>
356
+ </Tabs>
357
+
358
+ <DialogFooter>
359
+ <CmsButton variant="secondary" onClick={() => setShowPicker(false)}>
360
+ Cancel
361
+ </CmsButton>
362
+ </DialogFooter>
363
+ </DialogContent>
364
+ </Dialog>
365
+ </FieldWrapper>
366
+ )
367
+ }
@@ -0,0 +1,118 @@
1
+ import { useId } from 'react'
2
+ import { FieldWrapper } from './FieldWrapper'
3
+ import type { BaseFieldProps } from './types'
4
+ import { Checkbox } from '~/components/ui/checkbox'
5
+ import { Button } from '~/components/ui/button'
6
+ import { Label } from '~/components/ui/label'
7
+ import { cn } from '~/lib/cn'
8
+
9
+ export interface MultiSelectFieldProps extends BaseFieldProps<string[]> {
10
+ placeholder?: string
11
+ }
12
+
13
+ export function MultiSelectField({
14
+ field,
15
+ value,
16
+ onChange,
17
+ error,
18
+ disabled = false,
19
+ readOnly = false,
20
+ className = '',
21
+ id,
22
+ }: MultiSelectFieldProps) {
23
+ const generatedId = useId()
24
+ const fieldId = id ?? `field-${field.name}-${generatedId}`
25
+ const options = field.options?.options ?? []
26
+ const selectedValues = value ?? []
27
+
28
+ const handleChange = (optionValue: string, checked: boolean) => {
29
+ if (checked) {
30
+ onChange([...selectedValues, optionValue])
31
+ } else {
32
+ onChange(selectedValues.filter((v) => v !== optionValue))
33
+ }
34
+ }
35
+
36
+ const handleSelectAll = () => {
37
+ const allValues = options.map((o) => o.value)
38
+ onChange(allValues)
39
+ }
40
+
41
+ const handleClearAll = () => {
42
+ onChange([])
43
+ }
44
+
45
+ return (
46
+ <FieldWrapper field={field} error={error} className={className} id={fieldId}>
47
+ <div
48
+ className={cn(
49
+ 'rounded-md border border-input p-3',
50
+ error && 'border-destructive'
51
+ )}
52
+ role="group"
53
+ aria-labelledby={`${fieldId}-label`}
54
+ >
55
+ {options.length > 3 && (
56
+ <div className="mb-3 flex gap-2">
57
+ <Button
58
+ type="button"
59
+ variant="outline"
60
+ size="sm"
61
+ onClick={handleSelectAll}
62
+ disabled={disabled || readOnly}
63
+ >
64
+ Select all
65
+ </Button>
66
+ <Button
67
+ type="button"
68
+ variant="outline"
69
+ size="sm"
70
+ onClick={handleClearAll}
71
+ disabled={disabled || readOnly || selectedValues.length === 0}
72
+ >
73
+ Clear
74
+ </Button>
75
+ </div>
76
+ )}
77
+
78
+ <div className="space-y-2">
79
+ {options.map((option) => {
80
+ const optionId = `${fieldId}-${option.value}`
81
+ const isChecked = selectedValues.includes(option.value)
82
+
83
+ return (
84
+ <div
85
+ key={option.value}
86
+ className={cn(
87
+ 'flex items-center space-x-2 rounded-md p-2',
88
+ isChecked && 'bg-accent'
89
+ )}
90
+ >
91
+ <Checkbox
92
+ id={optionId}
93
+ checked={isChecked}
94
+ onCheckedChange={(checked) =>
95
+ handleChange(option.value, checked === true)
96
+ }
97
+ disabled={disabled || readOnly}
98
+ />
99
+ <Label
100
+ htmlFor={optionId}
101
+ className="cursor-pointer text-sm font-normal"
102
+ >
103
+ {option.label}
104
+ </Label>
105
+ </div>
106
+ )
107
+ })}
108
+ </div>
109
+
110
+ {selectedValues.length > 0 && (
111
+ <div className="mt-2 text-xs text-muted-foreground">
112
+ {selectedValues.length} of {options.length} selected
113
+ </div>
114
+ )}
115
+ </div>
116
+ </FieldWrapper>
117
+ )
118
+ }
@@ -0,0 +1,77 @@
1
+ import { useId, type ChangeEvent } from 'react'
2
+ import { FieldWrapper } from './FieldWrapper'
3
+ import type { NumberFieldProps } from './types'
4
+ import { Input } from '~/components/ui/input'
5
+ import { cn } from '~/lib/cn'
6
+
7
+ export function NumberField({
8
+ field,
9
+ value,
10
+ onChange,
11
+ error,
12
+ disabled = false,
13
+ readOnly = false,
14
+ className = '',
15
+ id: providedId,
16
+ placeholder,
17
+ }: NumberFieldProps) {
18
+ const generatedId = useId()
19
+ const id = providedId ?? `field-${field.name}-${generatedId}`
20
+
21
+ const handleChange = (e: ChangeEvent<HTMLInputElement>) => {
22
+ const inputValue = e.target.value
23
+
24
+ if (inputValue === '') {
25
+ onChange(null)
26
+ return
27
+ }
28
+
29
+ const numValue = parseFloat(inputValue)
30
+
31
+ if (!isNaN(numValue)) {
32
+ const { precision } = field.options ?? {}
33
+ if (precision !== undefined && precision >= 0) {
34
+ const factor = Math.pow(10, precision)
35
+ onChange(Math.round(numValue * factor) / factor)
36
+ } else {
37
+ onChange(numValue)
38
+ }
39
+ }
40
+ }
41
+
42
+ const { min, max, step, precision } = field.options ?? {}
43
+ const computedStep = step ?? (precision !== undefined ? Math.pow(10, -precision) : 'any')
44
+
45
+ return (
46
+ <FieldWrapper field={field} error={error} className={className} id={id}>
47
+ <Input
48
+ type="number"
49
+ id={id}
50
+ name={field.name}
51
+ value={value ?? ''}
52
+ onChange={handleChange}
53
+ disabled={disabled}
54
+ readOnly={readOnly}
55
+ required={field.required}
56
+ min={min}
57
+ max={max}
58
+ step={computedStep}
59
+ placeholder={placeholder}
60
+ className={cn(error && 'border-destructive focus-visible:ring-destructive')}
61
+ aria-invalid={!!error}
62
+ aria-describedby={
63
+ error ? `${id}-error` : field.description ? `${id}-description` : undefined
64
+ }
65
+ />
66
+ {(min !== undefined || max !== undefined) && (
67
+ <span className="mt-1 block text-xs text-muted-foreground">
68
+ {min !== undefined && max !== undefined
69
+ ? `Range: ${min} - ${max}`
70
+ : min !== undefined
71
+ ? `Min: ${min}`
72
+ : `Max: ${max}`}
73
+ </span>
74
+ )}
75
+ </FieldWrapper>
76
+ )
77
+ }