convex-cms 0.0.5-alpha.0 → 0.0.5-alpha.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +95 -144
- package/admin/README.md +99 -0
- package/admin/src/components/AdminLayout.tsx +22 -0
- package/admin/src/components/BreakingChangesWarningDialog.tsx +81 -0
- package/admin/src/components/BulkActionBar.tsx +190 -0
- package/admin/src/components/BulkOperationModal.tsx +177 -0
- package/admin/src/components/ContentEntryEditor.tsx +1104 -0
- package/admin/src/components/ContentTypeFormModal.tsx +1012 -0
- package/admin/src/components/ErrorBoundary.tsx +83 -0
- package/admin/src/components/ErrorState.tsx +147 -0
- package/admin/src/components/Header.tsx +294 -0
- package/admin/src/components/RouteGuard.tsx +264 -0
- package/admin/src/components/Sidebar.tsx +90 -0
- package/admin/src/components/TaxonomyEditor.tsx +348 -0
- package/admin/src/components/TermTree.tsx +533 -0
- package/admin/src/components/UploadDropzone.tsx +383 -0
- package/admin/src/components/VersionCompare.tsx +250 -0
- package/admin/src/components/VersionHistory.tsx +279 -0
- package/admin/src/components/VersionRollbackModal.tsx +79 -0
- package/admin/src/components/cmsds/CmsButton.tsx +101 -0
- package/admin/src/components/cmsds/CmsDialog.tsx +139 -0
- package/admin/src/components/cmsds/CmsDropdown.tsx +62 -0
- package/admin/src/components/cmsds/CmsEmptyState.tsx +54 -0
- package/admin/src/components/cmsds/CmsField.tsx +47 -0
- package/admin/src/components/cmsds/CmsPageHeader.tsx +35 -0
- package/admin/src/components/cmsds/CmsStatusBadge.tsx +153 -0
- package/admin/src/components/cmsds/CmsSurface.tsx +52 -0
- package/admin/src/components/cmsds/CmsTable.tsx +164 -0
- package/admin/src/components/cmsds/CmsToolbar.tsx +58 -0
- package/admin/src/components/cmsds/index.ts +10 -0
- package/admin/src/components/fields/BooleanField.tsx +74 -0
- package/admin/src/components/fields/CategoryField.tsx +394 -0
- package/admin/src/components/fields/DateField.tsx +173 -0
- package/admin/src/components/fields/DefaultFieldRenderer.tsx +74 -0
- package/admin/src/components/fields/FieldRenderer.tsx +180 -0
- package/admin/src/components/fields/FieldWrapper.tsx +57 -0
- package/admin/src/components/fields/JsonField.tsx +172 -0
- package/admin/src/components/fields/MediaField.tsx +367 -0
- package/admin/src/components/fields/MultiSelectField.tsx +118 -0
- package/admin/src/components/fields/NumberField.tsx +77 -0
- package/admin/src/components/fields/ReferenceField.tsx +386 -0
- package/admin/src/components/fields/RichTextField.tsx +171 -0
- package/admin/src/components/fields/SelectField.tsx +62 -0
- package/admin/src/components/fields/TagField.tsx +325 -0
- package/admin/src/components/fields/TextAreaField.tsx +68 -0
- package/admin/src/components/fields/TextField.tsx +56 -0
- package/admin/src/components/fields/index.ts +54 -0
- package/admin/src/components/fields/registry.ts +64 -0
- package/admin/src/components/fields/types.ts +217 -0
- package/admin/src/components/filters/TaxonomyFilter.tsx +254 -0
- package/admin/src/components/filters/index.ts +1 -0
- package/admin/src/components/index.ts +8 -0
- package/admin/src/components/media/MediaAssetActions.tsx +115 -0
- package/admin/src/components/media/MediaAssetEditDialog.tsx +217 -0
- package/admin/src/components/media/MediaBulkActionBar.tsx +51 -0
- package/admin/src/components/media/MediaFolderActions.tsx +69 -0
- package/admin/src/components/media/MediaFolderEditDialog.tsx +126 -0
- package/admin/src/components/media/MediaMoveModal.tsx +179 -0
- package/admin/src/components/media/MediaPreviewModal.tsx +370 -0
- package/admin/src/components/media/MediaTaxonomyPicker.tsx +304 -0
- package/admin/src/components/media/MediaTrashBulkActionBar.tsx +59 -0
- package/admin/src/components/ui/accordion.tsx +64 -0
- package/admin/src/components/ui/alert-dialog.tsx +155 -0
- package/admin/src/components/ui/alert.tsx +66 -0
- package/admin/src/components/ui/avatar.tsx +53 -0
- package/admin/src/components/ui/badge.tsx +46 -0
- package/admin/src/components/ui/breadcrumb.tsx +109 -0
- package/admin/src/components/ui/button.tsx +62 -0
- package/admin/src/components/ui/calendar.tsx +220 -0
- package/admin/src/components/ui/card.tsx +92 -0
- package/admin/src/components/ui/checkbox.tsx +30 -0
- package/admin/src/components/ui/command.tsx +182 -0
- package/admin/src/components/ui/dialog.tsx +143 -0
- package/admin/src/components/ui/dropdown-menu.tsx +257 -0
- package/admin/src/components/ui/form.tsx +167 -0
- package/admin/src/components/ui/input.tsx +21 -0
- package/admin/src/components/ui/label.tsx +24 -0
- package/admin/src/components/ui/popover.tsx +46 -0
- package/admin/src/components/ui/scroll-area.tsx +56 -0
- package/admin/src/components/ui/select.tsx +190 -0
- package/admin/src/components/ui/separator.tsx +26 -0
- package/admin/src/components/ui/sheet.tsx +137 -0
- package/admin/src/components/ui/sidebar.tsx +724 -0
- package/admin/src/components/ui/skeleton.tsx +13 -0
- package/admin/src/components/ui/sonner.tsx +38 -0
- package/admin/src/components/ui/switch.tsx +31 -0
- package/admin/src/components/ui/table.tsx +114 -0
- package/admin/src/components/ui/tabs.tsx +66 -0
- package/admin/src/components/ui/textarea.tsx +18 -0
- package/admin/src/components/ui/tooltip.tsx +61 -0
- package/admin/src/contexts/AdminConfigContext.tsx +30 -0
- package/admin/src/contexts/AuthContext.tsx +330 -0
- package/admin/src/contexts/BreadcrumbContext.tsx +49 -0
- package/admin/src/contexts/SettingsConfigContext.tsx +57 -0
- package/admin/src/contexts/ThemeContext.tsx +91 -0
- package/admin/src/contexts/index.ts +20 -0
- package/admin/src/embed/components/EmbedHeader.tsx +103 -0
- package/admin/src/embed/components/EmbedLayout.tsx +29 -0
- package/admin/src/embed/components/EmbedSidebar.tsx +119 -0
- package/admin/src/embed/components/index.ts +3 -0
- package/admin/src/embed/contexts/ApiContext.tsx +32 -0
- package/admin/src/embed/index.tsx +184 -0
- package/admin/src/embed/navigation.tsx +202 -0
- package/admin/src/embed/pages/Content.tsx +19 -0
- package/admin/src/embed/pages/ContentTypes.tsx +19 -0
- package/admin/src/embed/pages/Dashboard.tsx +19 -0
- package/admin/src/embed/pages/Media.tsx +19 -0
- package/admin/src/embed/pages/Settings.tsx +22 -0
- package/admin/src/embed/pages/Taxonomies.tsx +22 -0
- package/admin/src/embed/pages/Trash.tsx +22 -0
- package/admin/src/embed/pages/index.ts +7 -0
- package/admin/src/embed/types.ts +24 -0
- package/admin/src/hooks/index.ts +2 -0
- package/admin/src/hooks/use-mobile.ts +19 -0
- package/admin/src/hooks/useBreadcrumbLabel.ts +15 -0
- package/admin/src/hooks/usePermissions.ts +211 -0
- package/admin/src/lib/admin-config.ts +111 -0
- package/admin/src/lib/cn.ts +6 -0
- package/admin/src/lib/config.server.ts +56 -0
- package/admin/src/lib/convex.ts +26 -0
- package/admin/src/lib/embed-adapter.ts +80 -0
- package/admin/src/lib/icons.tsx +96 -0
- package/admin/src/lib/loadAdminConfig.ts +92 -0
- package/admin/src/lib/motion.ts +29 -0
- package/admin/src/lib/navigation.ts +43 -0
- package/admin/src/lib/tanstack-adapter.ts +82 -0
- package/admin/src/pages/ContentPage.tsx +337 -0
- package/admin/src/pages/ContentTypesPage.tsx +457 -0
- package/admin/src/pages/DashboardPage.tsx +163 -0
- package/admin/src/pages/MediaPage.tsx +34 -0
- package/admin/src/pages/SettingsPage.tsx +486 -0
- package/admin/src/pages/TaxonomiesPage.tsx +289 -0
- package/admin/src/pages/TrashPage.tsx +421 -0
- package/admin/src/pages/index.ts +14 -0
- package/admin/src/routeTree.gen.ts +262 -0
- package/admin/src/router.tsx +22 -0
- package/admin/src/routes/__root.tsx +250 -0
- package/admin/src/routes/content-types.tsx +20 -0
- package/admin/src/routes/content.tsx +20 -0
- package/admin/src/routes/entries/$entryId.tsx +107 -0
- package/admin/src/routes/entries/new.$contentTypeId.tsx +69 -0
- package/admin/src/routes/entries/type/$contentTypeId.tsx +503 -0
- package/admin/src/routes/index.tsx +20 -0
- package/admin/src/routes/media.tsx +1095 -0
- package/admin/src/routes/settings.tsx +20 -0
- package/admin/src/routes/taxonomies.tsx +20 -0
- package/admin/src/routes/trash.tsx +20 -0
- package/admin/src/styles/globals.css +69 -0
- package/admin/src/styles/tailwind-config.css +74 -0
- package/admin/src/styles/theme.css +73 -0
- package/admin/src/types/index.ts +221 -0
- package/admin/src/utils/errorParsing.ts +163 -0
- package/admin/src/utils/index.ts +5 -0
- package/admin/src/vite-env.d.ts +14 -0
- package/admin/tailwind.preset.cjs +102 -0
- package/admin-dist/nitro.json +1 -1
- package/admin-dist/public/assets/{CmsEmptyState-CiMQwSQV.js → CmsEmptyState-CkqBIab3.js} +1 -1
- package/admin-dist/public/assets/{CmsPageHeader-ohOq0luT.js → CmsPageHeader-CUtl5MMG.js} +1 -1
- package/admin-dist/public/assets/{CmsStatusBadge-BdNf0V9v.js → CmsStatusBadge-CUYFgEe-.js} +1 -1
- package/admin-dist/public/assets/{CmsSurface-CWup6Jh7.js → CmsSurface-CsJfAVa3.js} +1 -1
- package/admin-dist/public/assets/{CmsToolbar-cEBlCHa3.js → CmsToolbar-CnfbcxeP.js} +1 -1
- package/admin-dist/public/assets/{ContentEntryEditor-BY5ypfUs.js → ContentEntryEditor-BU220CCy.js} +1 -1
- package/admin-dist/public/assets/TaxonomyFilter-CWCxC5HZ.js +1 -0
- package/admin-dist/public/assets/_contentTypeId-DK8cskRt.js +1 -0
- package/admin-dist/public/assets/{_entryId-BpSmrfAm.js → _entryId-CuVMExbb.js} +1 -1
- package/admin-dist/public/assets/{alert-Bf2l8kxw.js → alert-CF1BSzGR.js} +1 -1
- package/admin-dist/public/assets/{badge-qPrc4AUM.js → badge-CmuOIVKp.js} +1 -1
- package/admin-dist/public/assets/{circle-check-big-Dgozy3vV.js → circle-check-big-BKDVG6DU.js} +1 -1
- package/admin-dist/public/assets/{command-QOmNhlb0.js → command-XJxnF2Sd.js} +1 -1
- package/admin-dist/public/assets/content-QBUxdxbS.js +1 -0
- package/admin-dist/public/assets/content-types-CrNEm8Hf.js +2 -0
- package/admin-dist/public/assets/globals-B7Wsfh_v.css +1 -0
- package/admin-dist/public/assets/index-C7xOwudI.js +1 -0
- package/admin-dist/public/assets/{label-DCsUdvFh.js → label-CHCnXeBk.js} +1 -1
- package/admin-dist/public/assets/{link-2-Czw1N61H.js → link-2-Bb34judH.js} +1 -1
- package/admin-dist/public/assets/{list-DtCsXj8-.js → list-9Pzt48ld.js} +1 -1
- package/admin-dist/public/assets/{main-CXgkZMhe.js → main-CjQ2VI9L.js} +3 -3
- package/admin-dist/public/assets/media-Dc5PWt2Q.js +1 -0
- package/admin-dist/public/assets/{new._contentTypeId-CoTDxKzf.js → new._contentTypeId-C_I4YxIa.js} +1 -1
- package/admin-dist/public/assets/{plus-xCFJK0RC.js → plus-Ceef7DHk.js} +1 -1
- package/admin-dist/public/assets/{rotate-ccw-DIqK63wY.js → rotate-ccw-7k7-4VUq.js} +1 -1
- package/admin-dist/public/assets/{scroll-area-B-yrE66a.js → scroll-area-CC6wujnp.js} +1 -1
- package/admin-dist/public/assets/{search-CbCbboeU.js → search-DwoUV2pv.js} +1 -1
- package/admin-dist/public/assets/{select-Co3TZFJb.js → select-hOZTp8aC.js} +1 -1
- package/admin-dist/public/assets/{settings-BspTTv_o.js → settings-t2PbCZh4.js} +1 -1
- package/admin-dist/public/assets/{switch-CfavASmR.js → switch-jX2pDaNU.js} +1 -1
- package/admin-dist/public/assets/{tabs-CN5s5u2W.js → tabs-q4EbZk7c.js} +1 -1
- package/admin-dist/public/assets/{tanstack-adapter-npeE3RdY.js → tanstack-adapter-B-Glm4kH.js} +1 -1
- package/admin-dist/public/assets/taxonomies-kyk5P4ZW.js +1 -0
- package/admin-dist/public/assets/{textarea-BJ0XFZpT.js → textarea-B6SfBmr0.js} +1 -1
- package/admin-dist/public/assets/trash-BOCnIznD.js +1 -0
- package/admin-dist/public/assets/{triangle-alert-BZRcqsUg.js → triangle-alert-CXFIO_Gu.js} +1 -1
- package/admin-dist/public/assets/{useBreadcrumbLabel-DwZlwvFF.js → useBreadcrumbLabel-_6qBagc3.js} +1 -1
- package/admin-dist/public/assets/{usePermissions-C1JQhfqb.js → usePermissions-M1ijZ7a6.js} +1 -1
- package/admin-dist/server/_ssr/{CmsButton-B45JAKR1.mjs → CmsButton-DOiTVKQq.mjs} +1 -1
- package/admin-dist/server/_ssr/{CmsEmptyState-D_BQFAVR.mjs → CmsEmptyState-fbnGt3LD.mjs} +2 -2
- package/admin-dist/server/_ssr/{CmsPageHeader-CrUZA59A.mjs → CmsPageHeader-DHRrdOZa.mjs} +1 -1
- package/admin-dist/server/_ssr/{CmsStatusBadge-B-sj6yaj.mjs → CmsStatusBadge-s7obWbKZ.mjs} +2 -2
- package/admin-dist/server/_ssr/{CmsSurface-DKJZhpjk.mjs → CmsSurface-rFoYjb62.mjs} +1 -1
- package/admin-dist/server/_ssr/{CmsToolbar-ByaW5iXf.mjs → CmsToolbar-zTE45z2q.mjs} +2 -2
- package/admin-dist/server/_ssr/{ContentEntryEditor-D3_Jb1dq.mjs → ContentEntryEditor-BLoEjT_m.mjs} +12 -12
- package/admin-dist/server/_ssr/{TaxonomyFilter-BRJkuCtA.mjs → TaxonomyFilter-XAtaJC2z.mjs} +5 -5
- package/admin-dist/server/_ssr/{_contentTypeId-B9kA6CaM.mjs → _contentTypeId-Csl4822C.mjs} +13 -13
- package/admin-dist/server/_ssr/{_entryId-BddcMkZN.mjs → _entryId-D8alLFBx.mjs} +15 -15
- package/admin-dist/server/_ssr/_tanstack-start-manifest_v-BffZedId.mjs +4 -0
- package/admin-dist/server/_ssr/{command-CGtVr8Gb.mjs → command-C0Di14--.mjs} +1 -1
- package/admin-dist/server/_ssr/{content-D1tbeOd0.mjs → content-CT-FPsmV.mjs} +12 -55
- package/admin-dist/server/_ssr/{content-types-BZqY_BER.mjs → content-types-C8cBFdzE.mjs} +15 -46
- package/admin-dist/server/_ssr/{index-BIdq4xaC.mjs → index-BJtcrEc-.mjs} +5 -5
- package/admin-dist/server/_ssr/index.mjs +2 -2
- package/admin-dist/server/_ssr/{label-T-QNKAr6.mjs → label-qn2Afwl4.mjs} +1 -1
- package/admin-dist/server/_ssr/{media-C-xqjBrl.mjs → media-qv5IAsMZ.mjs} +14 -14
- package/admin-dist/server/_ssr/{new._contentTypeId-DWic9cRq.mjs → new._contentTypeId-DdGyrhqs.mjs} +13 -13
- package/admin-dist/server/_ssr/{router-D1BMAMJT.mjs → router-nSVkxb6Y.mjs} +11 -11
- package/admin-dist/server/_ssr/{scroll-area-C0pic_WA.mjs → scroll-area-BCinP455.mjs} +1 -1
- package/admin-dist/server/_ssr/{select-CqmuN2F6.mjs → select-BKQlQScw.mjs} +1 -1
- package/admin-dist/server/_ssr/{settings-CAkncGGV.mjs → settings-BCr2KQlk.mjs} +55 -40
- package/admin-dist/server/_ssr/{switch-CgmuJkT9.mjs → switch-BaOi42fE.mjs} +1 -1
- package/admin-dist/server/_ssr/{tabs-CnMj0aRy.mjs → tabs-DYXEi9kq.mjs} +2 -2
- package/admin-dist/server/_ssr/{tanstack-adapter-BXZrMauE.mjs → tanstack-adapter-Bsz8kha-.mjs} +1 -1
- package/admin-dist/server/_ssr/{taxonomies-thl3BfVm.mjs → taxonomies-CueMHTbE.mjs} +30 -19
- package/admin-dist/server/_ssr/{textarea-4K5OJgeh.mjs → textarea-CI0Jqx2x.mjs} +1 -1
- package/admin-dist/server/_ssr/{trash-B40Gx5zP.mjs → trash-DE6W8GoX.mjs} +20 -17
- package/admin-dist/server/_ssr/{useBreadcrumbLabel-rn-fL4zV.mjs → useBreadcrumbLabel-B5Yi72lM.mjs} +1 -1
- package/admin-dist/server/_ssr/{usePermissions-CKeM6_Vw.mjs → usePermissions-C3nZ-Izm.mjs} +1 -1
- package/admin-dist/server/index.mjs +187 -194
- package/dist/client/admin/bulk.d.ts +79 -0
- package/dist/client/admin/bulk.d.ts.map +1 -0
- package/dist/client/admin/bulk.js +72 -0
- package/dist/client/admin/bulk.js.map +1 -0
- package/dist/client/admin/contentLock.d.ts +118 -0
- package/dist/client/admin/contentLock.d.ts.map +1 -0
- package/dist/client/admin/contentLock.js +81 -0
- package/dist/client/admin/contentLock.js.map +1 -0
- package/dist/client/{adminApi.d.ts → admin/contentTypes.d.ts} +39 -1134
- package/dist/client/admin/contentTypes.d.ts.map +1 -0
- package/dist/client/admin/contentTypes.js +122 -0
- package/dist/client/admin/contentTypes.js.map +1 -0
- package/dist/client/admin/dashboard.d.ts +16 -0
- package/dist/client/admin/dashboard.d.ts.map +1 -0
- package/dist/client/admin/dashboard.js +33 -0
- package/dist/client/admin/dashboard.js.map +1 -0
- package/dist/client/admin/entries.d.ts +358 -0
- package/dist/client/admin/entries.d.ts.map +1 -0
- package/dist/client/admin/entries.js +220 -0
- package/dist/client/admin/entries.js.map +1 -0
- package/dist/client/admin/index.d.ts +6568 -0
- package/dist/client/admin/index.d.ts.map +1 -0
- package/dist/client/admin/index.js +305 -0
- package/dist/client/admin/index.js.map +1 -0
- package/dist/client/admin/media.d.ts +1038 -0
- package/dist/client/admin/media.d.ts.map +1 -0
- package/dist/client/admin/media.js +489 -0
- package/dist/client/admin/media.js.map +1 -0
- package/dist/client/admin/taxonomies.d.ts +339 -0
- package/dist/client/admin/taxonomies.d.ts.map +1 -0
- package/dist/client/admin/taxonomies.js +364 -0
- package/dist/client/admin/taxonomies.js.map +1 -0
- package/dist/client/admin/trash.d.ts +91 -0
- package/dist/client/admin/trash.d.ts.map +1 -0
- package/dist/client/admin/trash.js +71 -0
- package/dist/client/admin/trash.js.map +1 -0
- package/dist/client/admin/types.d.ts +320 -0
- package/dist/client/admin/types.d.ts.map +1 -0
- package/dist/client/admin/types.js +7 -0
- package/dist/client/admin/types.js.map +1 -0
- package/dist/client/admin/validators.d.ts +3886 -0
- package/dist/client/admin/validators.d.ts.map +1 -0
- package/dist/client/admin/validators.js +322 -0
- package/dist/client/admin/validators.js.map +1 -0
- package/dist/client/admin/versions.d.ts +106 -0
- package/dist/client/admin/versions.d.ts.map +1 -0
- package/dist/client/admin/versions.js +57 -0
- package/dist/client/admin/versions.js.map +1 -0
- package/dist/client/adminApiTypes.d.ts +27 -0
- package/dist/client/adminApiTypes.d.ts.map +1 -0
- package/dist/client/adminApiTypes.js +12 -0
- package/dist/client/adminApiTypes.js.map +1 -0
- package/dist/client/{admin-config.d.ts → adminConfig.d.ts} +2 -2
- package/dist/client/adminConfig.d.ts.map +1 -0
- package/dist/client/{admin-config.js → adminConfig.js} +1 -1
- package/dist/client/adminConfig.js.map +1 -0
- package/dist/client/agentTools.d.ts +4 -4
- package/dist/client/index.d.ts +2 -2
- package/dist/client/index.d.ts.map +1 -1
- package/dist/client/index.js +15 -2
- package/dist/client/index.js.map +1 -1
- package/dist/component/contentEntries.d.ts +4 -4
- package/dist/component/contentEntryMutations.d.ts +46 -0
- package/dist/component/contentEntryMutations.d.ts.map +1 -1
- package/dist/component/contentEntryMutations.js +1 -1
- package/dist/component/contentEntryMutations.js.map +1 -1
- package/dist/component/contentTypeMigration.d.ts +1 -1
- package/dist/component/contentTypeMutations.d.ts +22 -0
- package/dist/component/contentTypeMutations.d.ts.map +1 -1
- package/dist/component/contentTypeMutations.js +1 -1
- package/dist/component/contentTypeMutations.js.map +1 -1
- package/dist/component/mediaAssetMutations.d.ts +47 -0
- package/dist/component/mediaAssetMutations.d.ts.map +1 -1
- package/dist/component/mediaAssetMutations.js +1 -1
- package/dist/component/mediaAssetMutations.js.map +1 -1
- package/dist/component/schema.d.ts +9 -0
- package/dist/component/schema.d.ts.map +1 -1
- package/dist/component/schema.js +1 -1
- package/dist/component/schema.js.map +1 -1
- package/package.json +85 -3
- package/admin-dist/public/assets/ErrorState-C4nJ-ml4.js +0 -1
- package/admin-dist/public/assets/TaxonomyFilter-BgE_SR_O.js +0 -1
- package/admin-dist/public/assets/_contentTypeId-DtZectcC.js +0 -1
- package/admin-dist/public/assets/content-OEBGlxg1.js +0 -1
- package/admin-dist/public/assets/content-types-CjQliqVV.js +0 -2
- package/admin-dist/public/assets/globals-hAmgC66w.css +0 -1
- package/admin-dist/public/assets/index-BH_ECMhv.js +0 -1
- package/admin-dist/public/assets/media-DTJ3-ViE.js +0 -1
- package/admin-dist/public/assets/taxonomies-CgG46fIF.js +0 -1
- package/admin-dist/public/assets/trash-B3daldm5.js +0 -1
- package/admin-dist/server/_ssr/ErrorState-cI-bKLez.mjs +0 -89
- package/admin-dist/server/_ssr/_tanstack-start-manifest_v-Dd7AmelK.mjs +0 -4
- package/dist/client/admin-config.d.ts.map +0 -1
- package/dist/client/admin-config.js.map +0 -1
- package/dist/client/adminApi.d.ts.map +0 -1
- package/dist/client/adminApi.js +0 -736
- package/dist/client/adminApi.js.map +0 -1
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { cn } from '~/lib/cn'
|
|
3
|
+
import { CmsButton, type CmsButtonProps } from './CmsButton'
|
|
4
|
+
|
|
5
|
+
export interface CmsEmptyStateProps extends React.HTMLAttributes<HTMLDivElement> {
|
|
6
|
+
icon?: React.ReactNode
|
|
7
|
+
title: string
|
|
8
|
+
description?: string
|
|
9
|
+
action?: {
|
|
10
|
+
label: string
|
|
11
|
+
onClick: () => void
|
|
12
|
+
variant?: CmsButtonProps['variant']
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
export function CmsEmptyState({
|
|
17
|
+
icon,
|
|
18
|
+
title,
|
|
19
|
+
description,
|
|
20
|
+
action,
|
|
21
|
+
className,
|
|
22
|
+
...props
|
|
23
|
+
}: CmsEmptyStateProps) {
|
|
24
|
+
return (
|
|
25
|
+
<div
|
|
26
|
+
className={cn(
|
|
27
|
+
'flex flex-col items-center justify-center py-12 text-center',
|
|
28
|
+
className
|
|
29
|
+
)}
|
|
30
|
+
{...props}
|
|
31
|
+
>
|
|
32
|
+
{icon && (
|
|
33
|
+
<div className="mb-4 flex size-12 items-center justify-center rounded-full bg-muted text-muted-foreground">
|
|
34
|
+
{icon}
|
|
35
|
+
</div>
|
|
36
|
+
)}
|
|
37
|
+
<h3 className="text-base font-semibold text-foreground">{title}</h3>
|
|
38
|
+
{description && (
|
|
39
|
+
<p className="mt-1 max-w-sm text-sm text-muted-foreground">
|
|
40
|
+
{description}
|
|
41
|
+
</p>
|
|
42
|
+
)}
|
|
43
|
+
{action && (
|
|
44
|
+
<CmsButton
|
|
45
|
+
variant={action.variant ?? 'primary'}
|
|
46
|
+
onClick={action.onClick}
|
|
47
|
+
className="mt-4"
|
|
48
|
+
>
|
|
49
|
+
{action.label}
|
|
50
|
+
</CmsButton>
|
|
51
|
+
)}
|
|
52
|
+
</div>
|
|
53
|
+
)
|
|
54
|
+
}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { Label } from '~/components/ui/label'
|
|
3
|
+
import { cn } from '~/lib/cn'
|
|
4
|
+
|
|
5
|
+
export interface CmsFieldProps {
|
|
6
|
+
label?: string
|
|
7
|
+
description?: string
|
|
8
|
+
error?: string
|
|
9
|
+
required?: boolean
|
|
10
|
+
htmlFor?: string
|
|
11
|
+
className?: string
|
|
12
|
+
children: React.ReactNode
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export function CmsField({
|
|
16
|
+
label,
|
|
17
|
+
description,
|
|
18
|
+
error,
|
|
19
|
+
required,
|
|
20
|
+
htmlFor,
|
|
21
|
+
className,
|
|
22
|
+
children,
|
|
23
|
+
}: CmsFieldProps) {
|
|
24
|
+
return (
|
|
25
|
+
<div className={cn('space-y-2', className)}>
|
|
26
|
+
{label && (
|
|
27
|
+
<Label
|
|
28
|
+
htmlFor={htmlFor}
|
|
29
|
+
className={cn(
|
|
30
|
+
'text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70',
|
|
31
|
+
error && 'text-destructive'
|
|
32
|
+
)}
|
|
33
|
+
>
|
|
34
|
+
{label}
|
|
35
|
+
{required && <span className="ml-1 text-destructive">*</span>}
|
|
36
|
+
</Label>
|
|
37
|
+
)}
|
|
38
|
+
{children}
|
|
39
|
+
{description && !error && (
|
|
40
|
+
<p className="text-[13px] text-muted-foreground">{description}</p>
|
|
41
|
+
)}
|
|
42
|
+
{error && (
|
|
43
|
+
<p className="text-[13px] font-medium text-destructive">{error}</p>
|
|
44
|
+
)}
|
|
45
|
+
</div>
|
|
46
|
+
)
|
|
47
|
+
}
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { cn } from '~/lib/cn'
|
|
3
|
+
|
|
4
|
+
export interface CmsPageHeaderProps extends React.HTMLAttributes<HTMLDivElement> {
|
|
5
|
+
title: string
|
|
6
|
+
description?: string
|
|
7
|
+
actions?: React.ReactNode
|
|
8
|
+
breadcrumbs?: React.ReactNode
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
export function CmsPageHeader({
|
|
12
|
+
title,
|
|
13
|
+
description,
|
|
14
|
+
actions,
|
|
15
|
+
breadcrumbs,
|
|
16
|
+
className,
|
|
17
|
+
...props
|
|
18
|
+
}: CmsPageHeaderProps) {
|
|
19
|
+
return (
|
|
20
|
+
<div className={cn('mb-6', className)} {...props}>
|
|
21
|
+
{breadcrumbs && <div className="mb-2">{breadcrumbs}</div>}
|
|
22
|
+
<div className="flex items-start justify-between gap-4">
|
|
23
|
+
<div className="space-y-1">
|
|
24
|
+
<h1 className="text-2xl font-semibold tracking-tight text-foreground">
|
|
25
|
+
{title}
|
|
26
|
+
</h1>
|
|
27
|
+
{description && (
|
|
28
|
+
<p className="text-sm text-muted-foreground">{description}</p>
|
|
29
|
+
)}
|
|
30
|
+
</div>
|
|
31
|
+
{actions && <div className="flex items-center gap-2">{actions}</div>}
|
|
32
|
+
</div>
|
|
33
|
+
</div>
|
|
34
|
+
)
|
|
35
|
+
}
|
|
@@ -0,0 +1,153 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
import { Badge } from "~/components/ui/badge";
|
|
3
|
+
import { cn } from "~/lib/cn";
|
|
4
|
+
|
|
5
|
+
export type ContentStatus = "draft" | "published" | "scheduled" | "archived";
|
|
6
|
+
|
|
7
|
+
export type WorkflowStateColor =
|
|
8
|
+
| "gray"
|
|
9
|
+
| "yellow"
|
|
10
|
+
| "blue"
|
|
11
|
+
| "green"
|
|
12
|
+
| "red"
|
|
13
|
+
| "purple"
|
|
14
|
+
| "orange";
|
|
15
|
+
|
|
16
|
+
export interface CustomStatusConfig {
|
|
17
|
+
name: string;
|
|
18
|
+
displayName: string;
|
|
19
|
+
color: WorkflowStateColor;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export interface CmsStatusBadgeProps
|
|
23
|
+
extends Omit<React.ComponentProps<typeof Badge>, "variant"> {
|
|
24
|
+
status: ContentStatus | string;
|
|
25
|
+
customConfig?: CustomStatusConfig;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
const statusConfig: Record<
|
|
29
|
+
ContentStatus,
|
|
30
|
+
{ label: string; className: string; icon: React.ReactNode }
|
|
31
|
+
> = {
|
|
32
|
+
draft: {
|
|
33
|
+
label: "Draft",
|
|
34
|
+
className: "status-draft",
|
|
35
|
+
icon: (
|
|
36
|
+
<svg className="size-3" fill="currentColor" viewBox="0 0 8 8">
|
|
37
|
+
<circle cx="4" cy="4" r="3" />
|
|
38
|
+
</svg>
|
|
39
|
+
),
|
|
40
|
+
},
|
|
41
|
+
published: {
|
|
42
|
+
label: "Published",
|
|
43
|
+
className: "status-published",
|
|
44
|
+
icon: (
|
|
45
|
+
<svg
|
|
46
|
+
className="size-3"
|
|
47
|
+
fill="none"
|
|
48
|
+
stroke="currentColor"
|
|
49
|
+
strokeWidth="2"
|
|
50
|
+
viewBox="0 0 24 24"
|
|
51
|
+
>
|
|
52
|
+
<path strokeLinecap="round" strokeLinejoin="round" d="M5 13l4 4L19 7" />
|
|
53
|
+
</svg>
|
|
54
|
+
),
|
|
55
|
+
},
|
|
56
|
+
scheduled: {
|
|
57
|
+
label: "Scheduled",
|
|
58
|
+
className: "status-scheduled",
|
|
59
|
+
icon: (
|
|
60
|
+
<svg
|
|
61
|
+
className="size-3"
|
|
62
|
+
fill="none"
|
|
63
|
+
stroke="currentColor"
|
|
64
|
+
strokeWidth="2"
|
|
65
|
+
viewBox="0 0 24 24"
|
|
66
|
+
>
|
|
67
|
+
<circle cx="12" cy="12" r="10" />
|
|
68
|
+
<path strokeLinecap="round" d="M12 6v6l4 2" />
|
|
69
|
+
</svg>
|
|
70
|
+
),
|
|
71
|
+
},
|
|
72
|
+
archived: {
|
|
73
|
+
label: "Archived",
|
|
74
|
+
className: "status-archived",
|
|
75
|
+
icon: (
|
|
76
|
+
<svg
|
|
77
|
+
className="size-3"
|
|
78
|
+
fill="none"
|
|
79
|
+
stroke="currentColor"
|
|
80
|
+
strokeWidth="2"
|
|
81
|
+
viewBox="0 0 24 24"
|
|
82
|
+
>
|
|
83
|
+
<path
|
|
84
|
+
strokeLinecap="round"
|
|
85
|
+
strokeLinejoin="round"
|
|
86
|
+
d="M5 8h14M5 8a2 2 0 110-4h14a2 2 0 110 4M5 8v10a2 2 0 002 2h10a2 2 0 002-2V8m-9 4h4"
|
|
87
|
+
/>
|
|
88
|
+
</svg>
|
|
89
|
+
),
|
|
90
|
+
},
|
|
91
|
+
};
|
|
92
|
+
|
|
93
|
+
const colorToClassName: Record<WorkflowStateColor, string> = {
|
|
94
|
+
gray: "bg-muted text-muted-foreground",
|
|
95
|
+
yellow: "bg-yellow-100 text-yellow-800 dark:bg-yellow-900/30 dark:text-yellow-400",
|
|
96
|
+
blue: "bg-blue-100 text-blue-800 dark:bg-blue-900/30 dark:text-blue-400",
|
|
97
|
+
green: "bg-green-100 text-green-800 dark:bg-green-900/30 dark:text-green-400",
|
|
98
|
+
red: "bg-red-100 text-red-800 dark:bg-red-900/30 dark:text-red-400",
|
|
99
|
+
purple: "bg-purple-100 text-purple-800 dark:bg-purple-900/30 dark:text-purple-400",
|
|
100
|
+
orange: "bg-orange-100 text-orange-800 dark:bg-orange-900/30 dark:text-orange-400",
|
|
101
|
+
};
|
|
102
|
+
|
|
103
|
+
function getDefaultIcon() {
|
|
104
|
+
return (
|
|
105
|
+
<svg className="size-3" fill="currentColor" viewBox="0 0 8 8">
|
|
106
|
+
<circle cx="4" cy="4" r="3" />
|
|
107
|
+
</svg>
|
|
108
|
+
);
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
export function CmsStatusBadge({ status, customConfig, className, ...props }: CmsStatusBadgeProps) {
|
|
112
|
+
if (customConfig) {
|
|
113
|
+
return (
|
|
114
|
+
<Badge
|
|
115
|
+
variant="secondary"
|
|
116
|
+
className={cn(
|
|
117
|
+
"gap-1.5 px-2 py-0.5 text-xs font-medium",
|
|
118
|
+
colorToClassName[customConfig.color],
|
|
119
|
+
className
|
|
120
|
+
)}
|
|
121
|
+
{...props}
|
|
122
|
+
>
|
|
123
|
+
{getDefaultIcon()}
|
|
124
|
+
{customConfig.displayName}
|
|
125
|
+
</Badge>
|
|
126
|
+
);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
const config = statusConfig[status as ContentStatus];
|
|
130
|
+
if (!config) {
|
|
131
|
+
return (
|
|
132
|
+
<Badge
|
|
133
|
+
variant="secondary"
|
|
134
|
+
className={cn("gap-1.5 px-2 py-0.5 text-xs font-medium", colorToClassName.gray, className)}
|
|
135
|
+
{...props}
|
|
136
|
+
>
|
|
137
|
+
{getDefaultIcon()}
|
|
138
|
+
{status}
|
|
139
|
+
</Badge>
|
|
140
|
+
);
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
return (
|
|
144
|
+
<Badge
|
|
145
|
+
variant="secondary"
|
|
146
|
+
className={cn("gap-1.5 px-2 py-0.5 text-xs font-medium", config.className, className)}
|
|
147
|
+
{...props}
|
|
148
|
+
>
|
|
149
|
+
{config.icon}
|
|
150
|
+
{config.label}
|
|
151
|
+
</Badge>
|
|
152
|
+
);
|
|
153
|
+
}
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { cn } from '~/lib/cn'
|
|
3
|
+
|
|
4
|
+
export interface CmsSurfaceProps extends React.HTMLAttributes<HTMLDivElement> {
|
|
5
|
+
elevation?: 'base' | 'elevated' | 'floating'
|
|
6
|
+
padding?: 'none' | 'sm' | 'md' | 'lg'
|
|
7
|
+
rounded?: 'none' | 'sm' | 'md' | 'lg'
|
|
8
|
+
asChild?: boolean
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
const elevationClasses = {
|
|
12
|
+
base: 'surface-base',
|
|
13
|
+
elevated: 'surface-elevated',
|
|
14
|
+
floating: 'surface-floating',
|
|
15
|
+
} as const
|
|
16
|
+
|
|
17
|
+
const paddingClasses = {
|
|
18
|
+
none: '',
|
|
19
|
+
sm: 'p-3',
|
|
20
|
+
md: 'p-4',
|
|
21
|
+
lg: 'p-6',
|
|
22
|
+
} as const
|
|
23
|
+
|
|
24
|
+
const roundedClasses = {
|
|
25
|
+
none: 'rounded-none',
|
|
26
|
+
sm: 'rounded-sm',
|
|
27
|
+
md: 'rounded-md',
|
|
28
|
+
lg: 'rounded-lg',
|
|
29
|
+
} as const
|
|
30
|
+
|
|
31
|
+
export function CmsSurface({
|
|
32
|
+
elevation = 'base',
|
|
33
|
+
padding = 'md',
|
|
34
|
+
rounded = 'lg',
|
|
35
|
+
className,
|
|
36
|
+
children,
|
|
37
|
+
...props
|
|
38
|
+
}: CmsSurfaceProps) {
|
|
39
|
+
return (
|
|
40
|
+
<div
|
|
41
|
+
className={cn(
|
|
42
|
+
elevationClasses[elevation],
|
|
43
|
+
paddingClasses[padding],
|
|
44
|
+
roundedClasses[rounded],
|
|
45
|
+
className
|
|
46
|
+
)}
|
|
47
|
+
{...props}
|
|
48
|
+
>
|
|
49
|
+
{children}
|
|
50
|
+
</div>
|
|
51
|
+
)
|
|
52
|
+
}
|
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import {
|
|
3
|
+
Table,
|
|
4
|
+
TableBody,
|
|
5
|
+
TableCell,
|
|
6
|
+
TableHead,
|
|
7
|
+
TableHeader,
|
|
8
|
+
TableRow,
|
|
9
|
+
} from '~/components/ui/table'
|
|
10
|
+
import { Checkbox } from '~/components/ui/checkbox'
|
|
11
|
+
import { cn } from '~/lib/cn'
|
|
12
|
+
import { ChevronDown, ChevronUp, ChevronsUpDown } from 'lucide-react'
|
|
13
|
+
|
|
14
|
+
export interface CmsTableColumn<T> {
|
|
15
|
+
key: string
|
|
16
|
+
header: string
|
|
17
|
+
cell: (row: T) => React.ReactNode
|
|
18
|
+
sortable?: boolean
|
|
19
|
+
className?: string
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export interface CmsTableProps<T> {
|
|
23
|
+
columns: CmsTableColumn<T>[]
|
|
24
|
+
data: T[]
|
|
25
|
+
getRowId: (row: T) => string
|
|
26
|
+
selectable?: boolean
|
|
27
|
+
selectedIds?: Set<string>
|
|
28
|
+
onSelectionChange?: (ids: Set<string>) => void
|
|
29
|
+
sortColumn?: string
|
|
30
|
+
sortDirection?: 'asc' | 'desc'
|
|
31
|
+
onSort?: (column: string) => void
|
|
32
|
+
onRowClick?: (row: T) => void
|
|
33
|
+
emptyMessage?: string
|
|
34
|
+
className?: string
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export function CmsTable<T>({
|
|
38
|
+
columns,
|
|
39
|
+
data,
|
|
40
|
+
getRowId,
|
|
41
|
+
selectable,
|
|
42
|
+
selectedIds = new Set(),
|
|
43
|
+
onSelectionChange,
|
|
44
|
+
sortColumn,
|
|
45
|
+
sortDirection,
|
|
46
|
+
onSort,
|
|
47
|
+
onRowClick,
|
|
48
|
+
emptyMessage = 'No items found',
|
|
49
|
+
className,
|
|
50
|
+
}: CmsTableProps<T>) {
|
|
51
|
+
const allSelected = data.length > 0 && data.every((row) => selectedIds.has(getRowId(row)))
|
|
52
|
+
const someSelected = data.some((row) => selectedIds.has(getRowId(row)))
|
|
53
|
+
|
|
54
|
+
const handleSelectAll = (checked: boolean) => {
|
|
55
|
+
if (!onSelectionChange) return
|
|
56
|
+
if (checked) {
|
|
57
|
+
onSelectionChange(new Set(data.map(getRowId)))
|
|
58
|
+
} else {
|
|
59
|
+
onSelectionChange(new Set())
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
const handleSelectRow = (rowId: string, checked: boolean) => {
|
|
64
|
+
if (!onSelectionChange) return
|
|
65
|
+
const newSelection = new Set(selectedIds)
|
|
66
|
+
if (checked) {
|
|
67
|
+
newSelection.add(rowId)
|
|
68
|
+
} else {
|
|
69
|
+
newSelection.delete(rowId)
|
|
70
|
+
}
|
|
71
|
+
onSelectionChange(newSelection)
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
const renderSortIcon = (column: CmsTableColumn<T>) => {
|
|
75
|
+
if (!column.sortable) return null
|
|
76
|
+
if (sortColumn !== column.key) {
|
|
77
|
+
return <ChevronsUpDown className="ml-1 inline size-3.5 text-muted-foreground" />
|
|
78
|
+
}
|
|
79
|
+
return sortDirection === 'asc' ? (
|
|
80
|
+
<ChevronUp className="ml-1 inline size-3.5" />
|
|
81
|
+
) : (
|
|
82
|
+
<ChevronDown className="ml-1 inline size-3.5" />
|
|
83
|
+
)
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
return (
|
|
87
|
+
<div className={cn('rounded-lg border', className)}>
|
|
88
|
+
<Table>
|
|
89
|
+
<TableHeader>
|
|
90
|
+
<TableRow className="hover:bg-transparent">
|
|
91
|
+
{selectable && (
|
|
92
|
+
<TableHead className="w-12">
|
|
93
|
+
<Checkbox
|
|
94
|
+
checked={allSelected ? true : someSelected ? 'indeterminate' : false}
|
|
95
|
+
onCheckedChange={handleSelectAll}
|
|
96
|
+
aria-label="Select all"
|
|
97
|
+
/>
|
|
98
|
+
</TableHead>
|
|
99
|
+
)}
|
|
100
|
+
{columns.map((column) => (
|
|
101
|
+
<TableHead
|
|
102
|
+
key={column.key}
|
|
103
|
+
className={cn(
|
|
104
|
+
column.sortable && 'cursor-pointer select-none',
|
|
105
|
+
column.className
|
|
106
|
+
)}
|
|
107
|
+
onClick={() => column.sortable && onSort?.(column.key)}
|
|
108
|
+
>
|
|
109
|
+
{column.header}
|
|
110
|
+
{renderSortIcon(column)}
|
|
111
|
+
</TableHead>
|
|
112
|
+
))}
|
|
113
|
+
</TableRow>
|
|
114
|
+
</TableHeader>
|
|
115
|
+
<TableBody>
|
|
116
|
+
{data.length === 0 ? (
|
|
117
|
+
<TableRow>
|
|
118
|
+
<TableCell
|
|
119
|
+
colSpan={columns.length + (selectable ? 1 : 0)}
|
|
120
|
+
className="h-24 text-center text-muted-foreground"
|
|
121
|
+
>
|
|
122
|
+
{emptyMessage}
|
|
123
|
+
</TableCell>
|
|
124
|
+
</TableRow>
|
|
125
|
+
) : (
|
|
126
|
+
data.map((row) => {
|
|
127
|
+
const rowId = getRowId(row)
|
|
128
|
+
const isSelected = selectedIds.has(rowId)
|
|
129
|
+
|
|
130
|
+
return (
|
|
131
|
+
<TableRow
|
|
132
|
+
key={rowId}
|
|
133
|
+
data-state={isSelected && 'selected'}
|
|
134
|
+
className={cn(onRowClick && 'cursor-pointer')}
|
|
135
|
+
onClick={() => onRowClick?.(row)}
|
|
136
|
+
>
|
|
137
|
+
{selectable && (
|
|
138
|
+
<TableCell
|
|
139
|
+
onClick={(e) => e.stopPropagation()}
|
|
140
|
+
className="w-12"
|
|
141
|
+
>
|
|
142
|
+
<Checkbox
|
|
143
|
+
checked={isSelected}
|
|
144
|
+
onCheckedChange={(checked) =>
|
|
145
|
+
handleSelectRow(rowId, checked === true)
|
|
146
|
+
}
|
|
147
|
+
aria-label={`Select row ${rowId}`}
|
|
148
|
+
/>
|
|
149
|
+
</TableCell>
|
|
150
|
+
)}
|
|
151
|
+
{columns.map((column) => (
|
|
152
|
+
<TableCell key={column.key} className={column.className}>
|
|
153
|
+
{column.cell(row)}
|
|
154
|
+
</TableCell>
|
|
155
|
+
))}
|
|
156
|
+
</TableRow>
|
|
157
|
+
)
|
|
158
|
+
})
|
|
159
|
+
)}
|
|
160
|
+
</TableBody>
|
|
161
|
+
</Table>
|
|
162
|
+
</div>
|
|
163
|
+
)
|
|
164
|
+
}
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { cn } from '~/lib/cn'
|
|
3
|
+
import { Input } from '~/components/ui/input'
|
|
4
|
+
import { Search } from 'lucide-react'
|
|
5
|
+
|
|
6
|
+
export interface CmsToolbarProps extends React.HTMLAttributes<HTMLDivElement> {
|
|
7
|
+
left?: React.ReactNode
|
|
8
|
+
right?: React.ReactNode
|
|
9
|
+
search?: {
|
|
10
|
+
value: string
|
|
11
|
+
onChange: (value: string) => void
|
|
12
|
+
placeholder?: string
|
|
13
|
+
}
|
|
14
|
+
filters?: React.ReactNode
|
|
15
|
+
actions?: React.ReactNode
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
export function CmsToolbar({
|
|
19
|
+
left,
|
|
20
|
+
right,
|
|
21
|
+
search,
|
|
22
|
+
filters,
|
|
23
|
+
actions,
|
|
24
|
+
className,
|
|
25
|
+
children,
|
|
26
|
+
...props
|
|
27
|
+
}: CmsToolbarProps) {
|
|
28
|
+
return (
|
|
29
|
+
<div
|
|
30
|
+
className={cn(
|
|
31
|
+
'flex flex-wrap items-center justify-between gap-3 pb-4',
|
|
32
|
+
className
|
|
33
|
+
)}
|
|
34
|
+
{...props}
|
|
35
|
+
>
|
|
36
|
+
<div className="flex flex-1 flex-wrap items-center gap-2">
|
|
37
|
+
{search && (
|
|
38
|
+
<div className="relative w-full max-w-xs">
|
|
39
|
+
<Search className="absolute left-3 top-1/2 size-4 -translate-y-1/2 text-muted-foreground" />
|
|
40
|
+
<Input
|
|
41
|
+
type="search"
|
|
42
|
+
placeholder={search.placeholder ?? 'Search...'}
|
|
43
|
+
value={search.value}
|
|
44
|
+
onChange={(e) => search.onChange(e.target.value)}
|
|
45
|
+
className="pl-9"
|
|
46
|
+
/>
|
|
47
|
+
</div>
|
|
48
|
+
)}
|
|
49
|
+
{filters}
|
|
50
|
+
{left}
|
|
51
|
+
{children}
|
|
52
|
+
</div>
|
|
53
|
+
{(right || actions) && (
|
|
54
|
+
<div className="flex items-center gap-2">{right ?? actions}</div>
|
|
55
|
+
)}
|
|
56
|
+
</div>
|
|
57
|
+
)
|
|
58
|
+
}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export { CmsButton, type CmsButtonProps } from './CmsButton'
|
|
2
|
+
export { CmsSurface, type CmsSurfaceProps } from './CmsSurface'
|
|
3
|
+
export { CmsStatusBadge, type CmsStatusBadgeProps, type ContentStatus } from './CmsStatusBadge'
|
|
4
|
+
export { CmsField, type CmsFieldProps } from './CmsField'
|
|
5
|
+
export { CmsEmptyState, type CmsEmptyStateProps } from './CmsEmptyState'
|
|
6
|
+
export { CmsPageHeader, type CmsPageHeaderProps } from './CmsPageHeader'
|
|
7
|
+
export { CmsToolbar, type CmsToolbarProps } from './CmsToolbar'
|
|
8
|
+
export { CmsDropdown, type CmsDropdownProps, type CmsDropdownAction } from './CmsDropdown'
|
|
9
|
+
export { CmsTable, type CmsTableProps, type CmsTableColumn } from './CmsTable'
|
|
10
|
+
export { CmsDialog, CmsConfirmDialog, type CmsDialogProps, type CmsConfirmDialogProps } from './CmsDialog'
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import { useId } from 'react'
|
|
2
|
+
import type { BooleanFieldProps } from './types'
|
|
3
|
+
import { Switch } from '~/components/ui/switch'
|
|
4
|
+
import { Label } from '~/components/ui/label'
|
|
5
|
+
import { cn } from '~/lib/cn'
|
|
6
|
+
|
|
7
|
+
export function BooleanField({
|
|
8
|
+
field,
|
|
9
|
+
value,
|
|
10
|
+
onChange,
|
|
11
|
+
error,
|
|
12
|
+
disabled = false,
|
|
13
|
+
readOnly = false,
|
|
14
|
+
className = '',
|
|
15
|
+
id: providedId,
|
|
16
|
+
trueLabel = 'Yes',
|
|
17
|
+
falseLabel = 'No',
|
|
18
|
+
}: BooleanFieldProps) {
|
|
19
|
+
const generatedId = useId()
|
|
20
|
+
const id = providedId ?? `field-${field.name}-${generatedId}`
|
|
21
|
+
|
|
22
|
+
const handleChange = (checked: boolean) => {
|
|
23
|
+
if (!readOnly) {
|
|
24
|
+
onChange(checked)
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
const hasError = !!error
|
|
29
|
+
|
|
30
|
+
return (
|
|
31
|
+
<div className={cn('space-y-2', className)}>
|
|
32
|
+
<div className="flex items-center justify-between">
|
|
33
|
+
<Label
|
|
34
|
+
htmlFor={id}
|
|
35
|
+
className={cn(
|
|
36
|
+
'text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70',
|
|
37
|
+
hasError && 'text-destructive'
|
|
38
|
+
)}
|
|
39
|
+
>
|
|
40
|
+
{field.label}
|
|
41
|
+
{field.required && <span className="ml-1 text-destructive">*</span>}
|
|
42
|
+
</Label>
|
|
43
|
+
|
|
44
|
+
<div className="flex items-center gap-2">
|
|
45
|
+
<Switch
|
|
46
|
+
id={id}
|
|
47
|
+
checked={value ?? false}
|
|
48
|
+
onCheckedChange={handleChange}
|
|
49
|
+
disabled={disabled || readOnly}
|
|
50
|
+
aria-invalid={hasError}
|
|
51
|
+
aria-describedby={
|
|
52
|
+
error ? `${id}-error` : field.description ? `${id}-description` : undefined
|
|
53
|
+
}
|
|
54
|
+
/>
|
|
55
|
+
<span className="text-sm text-muted-foreground">
|
|
56
|
+
{value ? trueLabel : falseLabel}
|
|
57
|
+
</span>
|
|
58
|
+
</div>
|
|
59
|
+
</div>
|
|
60
|
+
|
|
61
|
+
{field.description && !hasError && (
|
|
62
|
+
<p id={`${id}-description`} className="text-[13px] text-muted-foreground">
|
|
63
|
+
{field.description}
|
|
64
|
+
</p>
|
|
65
|
+
)}
|
|
66
|
+
|
|
67
|
+
{hasError && (
|
|
68
|
+
<p id={`${id}-error`} className="text-[13px] font-medium text-destructive" role="alert">
|
|
69
|
+
{error.message}
|
|
70
|
+
</p>
|
|
71
|
+
)}
|
|
72
|
+
</div>
|
|
73
|
+
)
|
|
74
|
+
}
|