betterstart-cli 0.0.1

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 (369) hide show
  1. package/README.md +5 -0
  2. package/dist/assets/adapters/next/integrations/mailchimp/actions/mailchimp.ts +38 -0
  3. package/dist/assets/adapters/next/integrations/mailchimp/integration.ts +33 -0
  4. package/dist/assets/adapters/next/integrations/r2/actions/r2.ts +77 -0
  5. package/dist/assets/adapters/next/integrations/r2/integration.ts +39 -0
  6. package/dist/assets/adapters/next/integrations/resend/actions/resend.ts +138 -0
  7. package/dist/assets/adapters/next/integrations/resend/integration.ts +36 -0
  8. package/dist/assets/adapters/next/plugins/blog/plugin.ts +17 -0
  9. package/dist/assets/adapters/next/plugins/blog/schemas/posts.json +169 -0
  10. package/dist/assets/adapters/next/templates/init/admin-globals.css +677 -0
  11. package/dist/assets/adapters/next/templates/init/api/auth-route.ts +3 -0
  12. package/dist/assets/adapters/next/templates/init/api/upload-route.ts +132 -0
  13. package/dist/assets/adapters/next/templates/init/components/layouts/admin-header.tsx +32 -0
  14. package/dist/assets/adapters/next/templates/init/components/layouts/admin-nav-link.tsx +23 -0
  15. package/dist/assets/adapters/next/templates/init/components/layouts/admin-providers.tsx +33 -0
  16. package/dist/assets/adapters/next/templates/init/components/layouts/admin-sidebar-branding-rsc.tsx +8 -0
  17. package/dist/assets/adapters/next/templates/init/components/layouts/admin-sidebar-branding-skeleton.tsx +10 -0
  18. package/dist/assets/adapters/next/templates/init/components/layouts/admin-sidebar-nav-link-skeleton.tsx +11 -0
  19. package/dist/assets/adapters/next/templates/init/components/layouts/admin-sidebar-nav-link.tsx +12 -0
  20. package/dist/assets/adapters/next/templates/init/components/layouts/admin-sidebar-user-menu-skeleton.tsx +15 -0
  21. package/dist/assets/adapters/next/templates/init/components/layouts/admin-sidebar-user-menu.tsx +90 -0
  22. package/dist/assets/adapters/next/templates/init/components/layouts/admin-sidebar.tsx +78 -0
  23. package/dist/assets/adapters/next/templates/init/components/layouts/admin-sign-out.tsx +44 -0
  24. package/dist/assets/adapters/next/templates/init/components/layouts/content-skeleton.tsx +44 -0
  25. package/dist/assets/adapters/next/templates/init/components/layouts/sidebar-branding.tsx +41 -0
  26. package/dist/assets/adapters/next/templates/init/components/shared/data-table/data-table-pagination.tsx +139 -0
  27. package/dist/assets/adapters/next/templates/init/components/shared/data-table/data-table.tsx +236 -0
  28. package/dist/assets/adapters/next/templates/init/components/shared/delete-dialog.tsx +67 -0
  29. package/dist/assets/adapters/next/templates/init/components/shared/dev-mode/copyable-code-block.tsx +104 -0
  30. package/dist/assets/adapters/next/templates/init/components/shared/dev-mode/dev-mode-code-mirror.tsx +68 -0
  31. package/dist/assets/adapters/next/templates/init/components/shared/dev-mode/dev-mode-types.ts +75 -0
  32. package/dist/assets/adapters/next/templates/init/components/shared/dev-mode/lifecycle-hooks-tab.tsx +111 -0
  33. package/dist/assets/adapters/next/templates/init/components/shared/dev-mode/plain-code-fallback.tsx +11 -0
  34. package/dist/assets/adapters/next/templates/init/components/shared/dev-mode/snippets-tab.tsx +125 -0
  35. package/dist/assets/adapters/next/templates/init/components/shared/dev-mode-integrate.tsx +108 -0
  36. package/dist/assets/adapters/next/templates/init/components/shared/entity-filters-bar.tsx +184 -0
  37. package/dist/assets/adapters/next/templates/init/components/shared/entity-metadata.tsx +93 -0
  38. package/dist/assets/adapters/next/templates/init/components/shared/entity-versions/entity-version-item.tsx +55 -0
  39. package/dist/assets/adapters/next/templates/init/components/shared/entity-versions/entity-version-restore-dialog.tsx +80 -0
  40. package/dist/assets/adapters/next/templates/init/components/shared/entity-versions/entity-versions-button.tsx +74 -0
  41. package/dist/assets/adapters/next/templates/init/components/shared/entity-versions/entity-versions-current-row.tsx +48 -0
  42. package/dist/assets/adapters/next/templates/init/components/shared/entity-versions/entity-versions-drawer.tsx +79 -0
  43. package/dist/assets/adapters/next/templates/init/components/shared/media/edit-media-dialog-content.tsx +222 -0
  44. package/dist/assets/adapters/next/templates/init/components/shared/media/edit-media-dialog.tsx +56 -0
  45. package/dist/assets/adapters/next/templates/init/components/shared/media/media-delete-dialog.tsx +83 -0
  46. package/dist/assets/adapters/next/templates/init/components/shared/media/media-delete-drawer.tsx +148 -0
  47. package/dist/assets/adapters/next/templates/init/components/shared/media/media-empty-state.tsx +45 -0
  48. package/dist/assets/adapters/next/templates/init/components/shared/media/media-filters-bar.tsx +129 -0
  49. package/dist/assets/adapters/next/templates/init/components/shared/media/media-gallery-dialog.tsx +182 -0
  50. package/dist/assets/adapters/next/templates/init/components/shared/media/media-grid-item.tsx +56 -0
  51. package/dist/assets/adapters/next/templates/init/components/shared/media/media-grid-pagination.tsx +114 -0
  52. package/dist/assets/adapters/next/templates/init/components/shared/media/media-grid.tsx +44 -0
  53. package/dist/assets/adapters/next/templates/init/components/shared/media/media-preview.tsx +69 -0
  54. package/dist/assets/adapters/next/templates/init/components/shared/media/media-url-importer.tsx +139 -0
  55. package/dist/assets/adapters/next/templates/init/components/shared/page-header.tsx +46 -0
  56. package/dist/assets/adapters/next/templates/init/components/shared/search-input.tsx +88 -0
  57. package/dist/assets/adapters/next/templates/init/components/shared/sort-indicator.tsx +24 -0
  58. package/dist/assets/adapters/next/templates/init/components/shared/sort-order-dialog.tsx +242 -0
  59. package/dist/assets/adapters/next/templates/init/components/shared/sort-order-drag-overlay-item.tsx +15 -0
  60. package/dist/assets/adapters/next/templates/init/components/shared/sort-order-item.tsx +32 -0
  61. package/dist/assets/adapters/next/templates/init/components/shared/sort-order-types.ts +9 -0
  62. package/dist/assets/adapters/next/templates/init/data/navigation.ts +43 -0
  63. package/dist/assets/adapters/next/templates/init/drizzle.config.ts +36 -0
  64. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-blockquote.ts +251 -0
  65. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-code-block.ts +258 -0
  66. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-color-highlight.ts +347 -0
  67. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-content-editor-media-insertion.ts +59 -0
  68. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-content-editor-mobile-toolbar.ts +17 -0
  69. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-content-editor-slash-menu.ts +116 -0
  70. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-content-editor-source-mode.tsx +638 -0
  71. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-content-editor-table-add-controls.ts +174 -0
  72. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-content-editor.ts +288 -0
  73. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-heading-dropdown-menu.ts +127 -0
  74. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-heading.ts +269 -0
  75. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-is-breakpoint.ts +32 -0
  76. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-link-popover.ts +278 -0
  77. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-list-dropdown-menu.ts +199 -0
  78. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-list.ts +290 -0
  79. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-mark.ts +199 -0
  80. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-menu-navigation.ts +221 -0
  81. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-tiptap-editor.ts +46 -0
  82. package/dist/assets/adapters/next/templates/init/hooks/content-editor/use-undo-redo.ts +169 -0
  83. package/dist/assets/adapters/next/templates/init/hooks/use-admin-theme.tsx +74 -0
  84. package/dist/assets/adapters/next/templates/init/hooks/use-copy-to-clipboard.ts +48 -0
  85. package/dist/assets/adapters/next/templates/init/hooks/use-dev-mode-integration.ts +43 -0
  86. package/dist/assets/adapters/next/templates/init/hooks/use-entity-versions.ts +32 -0
  87. package/dist/assets/adapters/next/templates/init/hooks/use-failed-image-url.ts +21 -0
  88. package/dist/assets/adapters/next/templates/init/hooks/use-local-storage.ts +46 -0
  89. package/dist/assets/adapters/next/templates/init/hooks/use-media-filters-bar.ts +72 -0
  90. package/dist/assets/adapters/next/templates/init/hooks/use-media.ts +102 -0
  91. package/dist/assets/adapters/next/templates/init/hooks/use-mobile.ts +19 -0
  92. package/dist/assets/adapters/next/templates/init/hooks/use-page-boundary-blur-visibility.ts +59 -0
  93. package/dist/assets/adapters/next/templates/init/hooks/use-page-scroll-threshold.ts +27 -0
  94. package/dist/assets/adapters/next/templates/init/hooks/use-table-utils.ts +315 -0
  95. package/dist/assets/adapters/next/templates/init/hooks/use-upload.ts +321 -0
  96. package/dist/assets/adapters/next/templates/init/hooks/use-users.ts +12 -0
  97. package/dist/assets/adapters/next/templates/init/lib/actions/auth/auth.ts +58 -0
  98. package/dist/assets/adapters/next/templates/init/lib/actions/auth/client.ts +12 -0
  99. package/dist/assets/adapters/next/templates/init/lib/actions/auth/middleware.ts +46 -0
  100. package/dist/assets/adapters/next/templates/init/lib/actions/email/form-delivery.ts +72 -0
  101. package/dist/assets/adapters/next/templates/init/lib/actions/email/index.ts +4 -0
  102. package/dist/assets/adapters/next/templates/init/lib/actions/email/is-email-delivery-configured.ts +5 -0
  103. package/dist/assets/adapters/next/templates/init/lib/actions/email/none.ts +24 -0
  104. package/dist/assets/adapters/next/templates/init/lib/actions/email/provider.ts +6 -0
  105. package/dist/assets/adapters/next/templates/init/lib/actions/email/send-email.ts +6 -0
  106. package/dist/assets/adapters/next/templates/init/lib/actions/email/send-password-reset-email.ts +10 -0
  107. package/dist/assets/adapters/next/templates/init/lib/actions/email/types.ts +25 -0
  108. package/dist/assets/adapters/next/templates/init/lib/actions/entity-versions/get-entity-versions.ts +48 -0
  109. package/dist/assets/adapters/next/templates/init/lib/actions/entity-versions/index.ts +10 -0
  110. package/dist/assets/adapters/next/templates/init/lib/actions/entity-versions/internal-create-entity-version.ts +43 -0
  111. package/dist/assets/adapters/next/templates/init/lib/actions/entity-versions/internal-delete-entity-versions.ts +19 -0
  112. package/dist/assets/adapters/next/templates/init/lib/actions/entity-versions/types.ts +26 -0
  113. package/dist/assets/adapters/next/templates/init/lib/actions/forms/get-all-form-settings.ts +21 -0
  114. package/dist/assets/adapters/next/templates/init/lib/actions/forms/get-form-settings.ts +27 -0
  115. package/dist/assets/adapters/next/templates/init/lib/actions/forms/index.ts +9 -0
  116. package/dist/assets/adapters/next/templates/init/lib/actions/forms/test-form-webhook.ts +40 -0
  117. package/dist/assets/adapters/next/templates/init/lib/actions/forms/types.ts +26 -0
  118. package/dist/assets/adapters/next/templates/init/lib/actions/forms/upsert-form-settings.ts +40 -0
  119. package/dist/assets/adapters/next/templates/init/lib/actions/media/create-media.ts +39 -0
  120. package/dist/assets/adapters/next/templates/init/lib/actions/media/delete-media-bulk.ts +29 -0
  121. package/dist/assets/adapters/next/templates/init/lib/actions/media/delete-media.ts +22 -0
  122. package/dist/assets/adapters/next/templates/init/lib/actions/media/get-media-by-id.ts +18 -0
  123. package/dist/assets/adapters/next/templates/init/lib/actions/media/get-media-by-ids.ts +25 -0
  124. package/dist/assets/adapters/next/templates/init/lib/actions/media/get-media.ts +71 -0
  125. package/dist/assets/adapters/next/templates/init/lib/actions/media/index.ts +14 -0
  126. package/dist/assets/adapters/next/templates/init/lib/actions/media/types.ts +31 -0
  127. package/dist/assets/adapters/next/templates/init/lib/actions/media/update-media.ts +35 -0
  128. package/dist/assets/adapters/next/templates/init/lib/actions/profile/index.ts +4 -0
  129. package/dist/assets/adapters/next/templates/init/lib/actions/profile/invalidate-users-cache.ts +14 -0
  130. package/dist/assets/adapters/next/templates/init/lib/actions/profile/is-email-configured.ts +7 -0
  131. package/dist/assets/adapters/next/templates/init/lib/actions/profile/types.ts +4 -0
  132. package/dist/assets/adapters/next/templates/init/lib/actions/profile/update-email.ts +57 -0
  133. package/dist/assets/adapters/next/templates/init/lib/actions/storage/index.ts +2 -0
  134. package/dist/assets/adapters/next/templates/init/lib/actions/storage/local.ts +31 -0
  135. package/dist/assets/adapters/next/templates/init/lib/actions/storage/provider.ts +6 -0
  136. package/dist/assets/adapters/next/templates/init/lib/actions/storage/save-upload.ts +11 -0
  137. package/dist/assets/adapters/next/templates/init/lib/actions/storage/types.ts +18 -0
  138. package/dist/assets/adapters/next/templates/init/lib/actions/upload/index.ts +9 -0
  139. package/dist/assets/adapters/next/templates/init/lib/actions/upload/types.ts +17 -0
  140. package/dist/assets/adapters/next/templates/init/lib/actions/upload/upload-file.ts +11 -0
  141. package/dist/assets/adapters/next/templates/init/lib/actions/upload/upload-files.ts +92 -0
  142. package/dist/assets/adapters/next/templates/init/lib/actions/upload/upload-image-from-url.ts +22 -0
  143. package/dist/assets/adapters/next/templates/init/lib/actions/upload/upload-media-from-url.ts +133 -0
  144. package/dist/assets/adapters/next/templates/init/lib/actions/users/create-user.ts +55 -0
  145. package/dist/assets/adapters/next/templates/init/lib/actions/users/delete-user.ts +24 -0
  146. package/dist/assets/adapters/next/templates/init/lib/actions/users/get-users.ts +49 -0
  147. package/dist/assets/adapters/next/templates/init/lib/actions/users/index.ts +12 -0
  148. package/dist/assets/adapters/next/templates/init/lib/actions/users/types.ts +43 -0
  149. package/dist/assets/adapters/next/templates/init/lib/actions/users/update-user-role.ts +28 -0
  150. package/dist/assets/adapters/next/templates/init/lib/db/client.ts +13 -0
  151. package/dist/assets/adapters/next/templates/init/lib/db/core/schema.ts +160 -0
  152. package/dist/assets/adapters/next/templates/init/lib/db/schema.ts +1 -0
  153. package/dist/assets/adapters/next/templates/init/lib/lifecycle-hooks/index.ts +19 -0
  154. package/dist/assets/adapters/next/templates/init/lib/lifecycle-hooks/register.local.ts +2 -0
  155. package/dist/assets/adapters/next/templates/init/lib/lifecycle-hooks/register.ts +9 -0
  156. package/dist/assets/adapters/next/templates/init/lib/lifecycle-hooks/registry.ts +55 -0
  157. package/dist/assets/adapters/next/templates/init/lib/lifecycle-hooks/runner.ts +51 -0
  158. package/dist/assets/adapters/next/templates/init/lib/lifecycle-hooks/types.ts +39 -0
  159. package/dist/assets/adapters/next/templates/init/pages/account-layout.tsx +11 -0
  160. package/dist/assets/adapters/next/templates/init/pages/account-shell-rsc.tsx +30 -0
  161. package/dist/assets/adapters/next/templates/init/pages/admin-layout.tsx +24 -0
  162. package/dist/assets/adapters/next/templates/init/pages/auth-gate-rsc.tsx +6 -0
  163. package/dist/assets/adapters/next/templates/init/pages/authenticated-layout.tsx +18 -0
  164. package/dist/assets/adapters/next/templates/init/pages/dashboard-page.tsx +121 -0
  165. package/dist/assets/adapters/next/templates/init/pages/forgot-password-form.tsx +124 -0
  166. package/dist/assets/adapters/next/templates/init/pages/forgot-password-page-skeleton.tsx +24 -0
  167. package/dist/assets/adapters/next/templates/init/pages/forgot-password-page.tsx +21 -0
  168. package/dist/assets/adapters/next/templates/init/pages/login-form.tsx +131 -0
  169. package/dist/assets/adapters/next/templates/init/pages/login-page-rsc.tsx +14 -0
  170. package/dist/assets/adapters/next/templates/init/pages/login-page-skeleton.tsx +26 -0
  171. package/dist/assets/adapters/next/templates/init/pages/login-page.tsx +21 -0
  172. package/dist/assets/adapters/next/templates/init/pages/media/media-page-content.tsx +273 -0
  173. package/dist/assets/adapters/next/templates/init/pages/media/media-page-skeleton.tsx +7 -0
  174. package/dist/assets/adapters/next/templates/init/pages/media/media-page.tsx +11 -0
  175. package/dist/assets/adapters/next/templates/init/pages/minimal-account-shell.tsx +25 -0
  176. package/dist/assets/adapters/next/templates/init/pages/profile/profile-form.tsx +281 -0
  177. package/dist/assets/adapters/next/templates/init/pages/profile/profile-page.tsx +31 -0
  178. package/dist/assets/adapters/next/templates/init/pages/reset-password-form.tsx +161 -0
  179. package/dist/assets/adapters/next/templates/init/pages/reset-password-page-skeleton.tsx +26 -0
  180. package/dist/assets/adapters/next/templates/init/pages/reset-password-page.tsx +21 -0
  181. package/dist/assets/adapters/next/templates/init/pages/users/columns.tsx +170 -0
  182. package/dist/assets/adapters/next/templates/init/pages/users/create-user-dialog.tsx +221 -0
  183. package/dist/assets/adapters/next/templates/init/pages/users/delete-user-dialog.tsx +172 -0
  184. package/dist/assets/adapters/next/templates/init/pages/users/edit-role-dialog.tsx +91 -0
  185. package/dist/assets/adapters/next/templates/init/pages/users/users-page-content.tsx +25 -0
  186. package/dist/assets/adapters/next/templates/init/pages/users/users-page-skeleton.tsx +7 -0
  187. package/dist/assets/adapters/next/templates/init/pages/users/users-page.tsx +11 -0
  188. package/dist/assets/adapters/next/templates/init/pages/users/users-table.tsx +221 -0
  189. package/dist/assets/adapters/next/templates/init/types/auth.ts +71 -0
  190. package/dist/assets/adapters/next/templates/init/types/index.ts +108 -0
  191. package/dist/assets/adapters/next/templates/init/types/navigation.ts +11 -0
  192. package/dist/assets/adapters/next/templates/init/types/table-meta.ts +14 -0
  193. package/dist/assets/adapters/next/templates/init/utils/auth/roles.ts +17 -0
  194. package/dist/assets/adapters/next/templates/init/utils/date/date.ts +90 -0
  195. package/dist/assets/adapters/next/templates/init/utils/dev-mode/code-block-height.ts +9 -0
  196. package/dist/assets/adapters/next/templates/init/utils/editor/content-editor-rich-extensions.ts +824 -0
  197. package/dist/assets/adapters/next/templates/init/utils/editor/content-editor.ts +316 -0
  198. package/dist/assets/adapters/next/templates/init/utils/editor/editor-view.ts +19 -0
  199. package/dist/assets/adapters/next/templates/init/utils/editor/markdown.ts +542 -0
  200. package/dist/assets/adapters/next/templates/init/utils/editor/node-attrs.ts +25 -0
  201. package/dist/assets/adapters/next/templates/init/utils/editor/slash-commands.ts +148 -0
  202. package/dist/assets/adapters/next/templates/init/utils/editor/source-media.ts +11 -0
  203. package/dist/assets/adapters/next/templates/init/utils/editor/table-add-controls.ts +91 -0
  204. package/dist/assets/adapters/next/templates/init/utils/editor/table-bubble.ts +172 -0
  205. package/dist/assets/adapters/next/templates/init/utils/editor/table-input.ts +5 -0
  206. package/dist/assets/adapters/next/templates/init/utils/editor/task-item.ts +19 -0
  207. package/dist/assets/adapters/next/templates/init/utils/editor/tiptap.ts +991 -0
  208. package/dist/assets/adapters/next/templates/init/utils/email/form-delivery.ts +104 -0
  209. package/dist/assets/adapters/next/templates/init/utils/media/fallback.ts +37 -0
  210. package/dist/assets/adapters/next/templates/init/utils/media/media.ts +91 -0
  211. package/dist/assets/adapters/next/templates/init/utils/media/query.ts +96 -0
  212. package/dist/assets/adapters/next/templates/init/utils/navigation/order.ts +6 -0
  213. package/dist/assets/adapters/next/templates/init/utils/navigation/sidebar.ts +26 -0
  214. package/dist/assets/adapters/next/templates/init/utils/page/boundary.ts +32 -0
  215. package/dist/assets/adapters/next/templates/init/utils/seo/seo.ts +90 -0
  216. package/dist/assets/adapters/next/templates/init/utils/shared/cn.ts +6 -0
  217. package/dist/assets/adapters/next/templates/init/utils/storage/local.ts +9 -0
  218. package/dist/assets/adapters/next/templates/init/utils/table/table.ts +10 -0
  219. package/dist/assets/adapters/next/templates/init/utils/text/text.ts +4 -0
  220. package/dist/assets/adapters/next/templates/init/utils/theme/system.ts +6 -0
  221. package/dist/assets/adapters/next/templates/init/utils/upload/remote.ts +55 -0
  222. package/dist/assets/adapters/next/templates/init/utils/upload/upload.ts +26 -0
  223. package/dist/assets/adapters/next/templates/init/utils/user/user.ts +11 -0
  224. package/dist/assets/adapters/next/templates/init/utils/validation/validation.ts +114 -0
  225. package/dist/assets/adapters/next/templates/init/utils/webhook/webhook.ts +28 -0
  226. package/dist/assets/shared-assets/react-admin/custom/content-editor/editor-toolbar.tsx +25 -0
  227. package/dist/assets/shared-assets/react-admin/custom/content-editor/horizontal-rule-button.tsx +22 -0
  228. package/dist/assets/shared-assets/react-admin/custom/content-editor/index.tsx +142 -0
  229. package/dist/assets/shared-assets/react-admin/custom/content-editor/main-toolbar-content.tsx +118 -0
  230. package/dist/assets/shared-assets/react-admin/custom/content-editor/math-bubble-menu.tsx +80 -0
  231. package/dist/assets/shared-assets/react-admin/custom/content-editor/math-button.tsx +22 -0
  232. package/dist/assets/shared-assets/react-admin/custom/content-editor/math-editor-controls.tsx +117 -0
  233. package/dist/assets/shared-assets/react-admin/custom/content-editor/math-popover-button.tsx +59 -0
  234. package/dist/assets/shared-assets/react-admin/custom/content-editor/math-popover.tsx +6 -0
  235. package/dist/assets/shared-assets/react-admin/custom/content-editor/media-gallery-block.tsx +31 -0
  236. package/dist/assets/shared-assets/react-admin/custom/content-editor/mobile-toolbar-content.tsx +56 -0
  237. package/dist/assets/shared-assets/react-admin/custom/content-editor/mode-toggle-button.tsx +29 -0
  238. package/dist/assets/shared-assets/react-admin/custom/content-editor/remove-table-part-icon.tsx +17 -0
  239. package/dist/assets/shared-assets/react-admin/custom/content-editor/selection-bubble-menu.tsx +105 -0
  240. package/dist/assets/shared-assets/react-admin/custom/content-editor/slash-command-menu.tsx +65 -0
  241. package/dist/assets/shared-assets/react-admin/custom/content-editor/source-mode-dropdown-button.tsx +52 -0
  242. package/dist/assets/shared-assets/react-admin/custom/content-editor/source-mode.tsx +360 -0
  243. package/dist/assets/shared-assets/react-admin/custom/content-editor/table-add-controls.tsx +46 -0
  244. package/dist/assets/shared-assets/react-admin/custom/content-editor/table-bubble-menu.tsx +290 -0
  245. package/dist/assets/shared-assets/react-admin/custom/content-editor/table-button.tsx +116 -0
  246. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-extension/node-background-extension.ts +138 -0
  247. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-node/horizontal-rule-node/horizontal-rule-node-extension.ts +10 -0
  248. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-node/media-gallery-placeholder-node/index.tsx +1 -0
  249. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-node/media-gallery-placeholder-node/media-gallery-placeholder-node-extension.ts +117 -0
  250. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-node/media-gallery-placeholder-node/media-gallery-placeholder-node.tsx +63 -0
  251. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-node/removable-image-node/index.tsx +1 -0
  252. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-node/removable-image-node/removable-image-node-extension.ts +11 -0
  253. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-node/removable-image-node/removable-image-node.tsx +168 -0
  254. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-node/task-item-node/task-item-node-extension.tsx +142 -0
  255. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/blockquote-button/blockquote-button.tsx +114 -0
  256. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/blockquote-button/index.tsx +1 -0
  257. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/code-block-button/code-block-button.tsx +112 -0
  258. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/code-block-button/index.tsx +1 -0
  259. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/color-highlight-button/color-highlight-button.tsx +185 -0
  260. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/color-highlight-button/index.tsx +1 -0
  261. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/color-highlight-popover/color-highlight-popover-button.tsx +40 -0
  262. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/color-highlight-popover/color-highlight-popover-content.tsx +130 -0
  263. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/color-highlight-popover/color-highlight-popover.tsx +98 -0
  264. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/color-highlight-popover/highlight-color-button.tsx +75 -0
  265. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/color-highlight-popover/highlight-colors.ts +24 -0
  266. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/color-highlight-popover/index.tsx +1 -0
  267. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/color-highlight-popover/source-color-highlight-popover.tsx +65 -0
  268. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/control-options.ts +27 -0
  269. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/heading-dropdown-menu/heading-dropdown-menu-item.tsx +35 -0
  270. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/heading-dropdown-menu/heading-dropdown-menu.tsx +119 -0
  271. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/heading-dropdown-menu/index.tsx +1 -0
  272. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/link-popover/index.tsx +1 -0
  273. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/link-popover/link-button.tsx +39 -0
  274. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/link-popover/link-content.tsx +13 -0
  275. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/link-popover/link-control-popover.tsx +90 -0
  276. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/link-popover/link-main.tsx +96 -0
  277. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/link-popover/link-popover.tsx +121 -0
  278. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/list-dropdown-menu/index.tsx +1 -0
  279. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/list-dropdown-menu/list-dropdown-menu-item.tsx +44 -0
  280. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/list-dropdown-menu/list-dropdown-menu.tsx +115 -0
  281. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/mark-button/index.tsx +1 -0
  282. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/mark-button/mark-button.tsx +117 -0
  283. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/undo-redo-button/index.tsx +1 -0
  284. package/dist/assets/shared-assets/react-admin/custom/content-editor/tiptap-ui/undo-redo-button/undo-redo-button.tsx +115 -0
  285. package/dist/assets/shared-assets/react-admin/custom/date-picker.tsx +70 -0
  286. package/dist/assets/shared-assets/react-admin/custom/date-range-picker.tsx +216 -0
  287. package/dist/assets/shared-assets/react-admin/custom/dynamic-list-field.tsx +111 -0
  288. package/dist/assets/shared-assets/react-admin/custom/gallery-field.tsx +124 -0
  289. package/dist/assets/shared-assets/react-admin/custom/gallery-thumbnail.tsx +39 -0
  290. package/dist/assets/shared-assets/react-admin/custom/icon-picker.tsx +377 -0
  291. package/dist/assets/shared-assets/react-admin/custom/icons-column-skeleton.tsx +16 -0
  292. package/dist/assets/shared-assets/react-admin/custom/icons-data.ts +8 -0
  293. package/dist/assets/shared-assets/react-admin/custom/logo.tsx +115 -0
  294. package/dist/assets/shared-assets/react-admin/custom/media-gallery-field.tsx +182 -0
  295. package/dist/assets/shared-assets/react-admin/custom/page-skeleton.tsx +11 -0
  296. package/dist/assets/shared-assets/react-admin/custom/placeholder-card.tsx +34 -0
  297. package/dist/assets/shared-assets/react-admin/custom/placeholder.tsx +25 -0
  298. package/dist/assets/shared-assets/react-admin/custom/progressive-blur.tsx +62 -0
  299. package/dist/assets/shared-assets/react-admin/custom/sortable-gallery-item.tsx +68 -0
  300. package/dist/assets/shared-assets/react-admin/custom/upload-dropzone.tsx +107 -0
  301. package/dist/assets/shared-assets/react-admin/dependencies.ts +73 -0
  302. package/dist/assets/shared-assets/react-admin/schema.json +1670 -0
  303. package/dist/assets/shared-assets/react-admin/ui/accordion.tsx +86 -0
  304. package/dist/assets/shared-assets/react-admin/ui/alert-dialog.tsx +178 -0
  305. package/dist/assets/shared-assets/react-admin/ui/alert.tsx +72 -0
  306. package/dist/assets/shared-assets/react-admin/ui/aspect-ratio.tsx +9 -0
  307. package/dist/assets/shared-assets/react-admin/ui/avatar.tsx +95 -0
  308. package/dist/assets/shared-assets/react-admin/ui/badge.tsx +48 -0
  309. package/dist/assets/shared-assets/react-admin/ui/breadcrumb.tsx +99 -0
  310. package/dist/assets/shared-assets/react-admin/ui/button-group.tsx +76 -0
  311. package/dist/assets/shared-assets/react-admin/ui/button.tsx +66 -0
  312. package/dist/assets/shared-assets/react-admin/ui/calendar.tsx +184 -0
  313. package/dist/assets/shared-assets/react-admin/ui/card.tsx +94 -0
  314. package/dist/assets/shared-assets/react-admin/ui/carousel.tsx +239 -0
  315. package/dist/assets/shared-assets/react-admin/ui/chart.tsx +336 -0
  316. package/dist/assets/shared-assets/react-admin/ui/checkbox.tsx +28 -0
  317. package/dist/assets/shared-assets/react-admin/ui/collapsible.tsx +21 -0
  318. package/dist/assets/shared-assets/react-admin/ui/combobox.tsx +272 -0
  319. package/dist/assets/shared-assets/react-admin/ui/command.tsx +180 -0
  320. package/dist/assets/shared-assets/react-admin/ui/context-menu.tsx +243 -0
  321. package/dist/assets/shared-assets/react-admin/ui/dialog.tsx +141 -0
  322. package/dist/assets/shared-assets/react-admin/ui/direction.tsx +20 -0
  323. package/dist/assets/shared-assets/react-admin/ui/drawer.tsx +119 -0
  324. package/dist/assets/shared-assets/react-admin/ui/dropdown-menu.tsx +253 -0
  325. package/dist/assets/shared-assets/react-admin/ui/empty.tsx +93 -0
  326. package/dist/assets/shared-assets/react-admin/ui/field.tsx +234 -0
  327. package/dist/assets/shared-assets/react-admin/ui/form.tsx +172 -0
  328. package/dist/assets/shared-assets/react-admin/ui/hover-card.tsx +37 -0
  329. package/dist/assets/shared-assets/react-admin/ui/input-group.tsx +134 -0
  330. package/dist/assets/shared-assets/react-admin/ui/input-otp.tsx +85 -0
  331. package/dist/assets/shared-assets/react-admin/ui/input.tsx +18 -0
  332. package/dist/assets/shared-assets/react-admin/ui/item.tsx +180 -0
  333. package/dist/assets/shared-assets/react-admin/ui/kbd.tsx +26 -0
  334. package/dist/assets/shared-assets/react-admin/ui/label.tsx +20 -0
  335. package/dist/assets/shared-assets/react-admin/ui/menubar.tsx +259 -0
  336. package/dist/assets/shared-assets/react-admin/ui/native-select.tsx +54 -0
  337. package/dist/assets/shared-assets/react-admin/ui/navigation-menu.tsx +159 -0
  338. package/dist/assets/shared-assets/react-admin/ui/pagination.tsx +111 -0
  339. package/dist/assets/shared-assets/react-admin/ui/popover.tsx +75 -0
  340. package/dist/assets/shared-assets/react-admin/ui/progress.tsx +30 -0
  341. package/dist/assets/shared-assets/react-admin/ui/radio-group.tsx +43 -0
  342. package/dist/assets/shared-assets/react-admin/ui/resizable.tsx +41 -0
  343. package/dist/assets/shared-assets/react-admin/ui/scroll-area.tsx +54 -0
  344. package/dist/assets/shared-assets/react-admin/ui/select.tsx +183 -0
  345. package/dist/assets/shared-assets/react-admin/ui/separator.tsx +27 -0
  346. package/dist/assets/shared-assets/react-admin/ui/sheet.tsx +129 -0
  347. package/dist/assets/shared-assets/react-admin/ui/sidebar.tsx +688 -0
  348. package/dist/assets/shared-assets/react-admin/ui/skeleton.tsx +13 -0
  349. package/dist/assets/shared-assets/react-admin/ui/slider.tsx +53 -0
  350. package/dist/assets/shared-assets/react-admin/ui/sonner.tsx +45 -0
  351. package/dist/assets/shared-assets/react-admin/ui/spinner.tsx +15 -0
  352. package/dist/assets/shared-assets/react-admin/ui/switch.tsx +32 -0
  353. package/dist/assets/shared-assets/react-admin/ui/table.tsx +101 -0
  354. package/dist/assets/shared-assets/react-admin/ui/tabs.tsx +79 -0
  355. package/dist/assets/shared-assets/react-admin/ui/textarea.tsx +17 -0
  356. package/dist/assets/shared-assets/react-admin/ui/toggle-group.tsx +85 -0
  357. package/dist/assets/shared-assets/react-admin/ui/toggle.tsx +45 -0
  358. package/dist/assets/shared-assets/react-admin/ui/tooltip.tsx +51 -0
  359. package/dist/chunk-MUZQCVQA.js +306 -0
  360. package/dist/chunk-MUZQCVQA.js.map +1 -0
  361. package/dist/cli.d.ts +2 -0
  362. package/dist/cli.js +23437 -0
  363. package/dist/cli.js.map +1 -0
  364. package/dist/index.d.ts +90 -0
  365. package/dist/index.js +8 -0
  366. package/dist/index.js.map +1 -0
  367. package/dist/template-reader-YKWE2C7O.js +13 -0
  368. package/dist/template-reader-YKWE2C7O.js.map +1 -0
  369. package/package.json +74 -0
@@ -0,0 +1,43 @@
1
+ import type db from '@admin/db'
2
+ import { entityVersions } from '@admin/db/schema'
3
+ import { and, desc, eq, sql } from 'drizzle-orm'
4
+ import type { EntityVersionData } from './types'
5
+
6
+ type EntityVersionClient = Parameters<Parameters<typeof db.transaction>[0]>[0]
7
+
8
+ interface CreateEntityVersionInput {
9
+ entityType: string
10
+ entityId: string
11
+ data: EntityVersionData
12
+ createdBy?: string | null
13
+ }
14
+
15
+ export async function createEntityVersion(
16
+ client: EntityVersionClient,
17
+ input: CreateEntityVersionInput
18
+ ): Promise<void> {
19
+ await client.execute(
20
+ sql`select pg_advisory_xact_lock(hashtext(${input.entityType}), hashtext(${input.entityId}))`
21
+ )
22
+
23
+ const [latestVersion] = await client
24
+ .select({ version: entityVersions.version })
25
+ .from(entityVersions)
26
+ .where(
27
+ and(
28
+ eq(entityVersions.entityType, input.entityType),
29
+ eq(entityVersions.entityId, input.entityId)
30
+ )
31
+ )
32
+ .orderBy(desc(entityVersions.version))
33
+ .limit(1)
34
+
35
+ await client.insert(entityVersions).values({
36
+ entityType: input.entityType,
37
+ entityId: input.entityId,
38
+ version: (latestVersion?.version ?? 0) + 1,
39
+ data: input.data,
40
+ createdAt: new Date().toISOString(),
41
+ createdBy: input.createdBy ?? null
42
+ })
43
+ }
@@ -0,0 +1,19 @@
1
+ import type db from '@admin/db'
2
+ import { entityVersions } from '@admin/db/schema'
3
+ import { and, eq, inArray } from 'drizzle-orm'
4
+
5
+ type EntityVersionClient = Parameters<Parameters<typeof db.transaction>[0]>[0]
6
+
7
+ export async function deleteEntityVersions(
8
+ client: EntityVersionClient,
9
+ entityType: string,
10
+ entityIds: string[]
11
+ ): Promise<void> {
12
+ if (entityIds.length === 0) return
13
+
14
+ await client
15
+ .delete(entityVersions)
16
+ .where(
17
+ and(eq(entityVersions.entityType, entityType), inArray(entityVersions.entityId, entityIds))
18
+ )
19
+ }
@@ -0,0 +1,26 @@
1
+ export type EntityVersionData = Record<string, unknown>
2
+
3
+ export interface EntityVersionAuthor {
4
+ id: string
5
+ name: string
6
+ email: string
7
+ image?: string | null
8
+ }
9
+
10
+ export interface EntityVersion {
11
+ id: string
12
+ entityType: string
13
+ entityId: string
14
+ version: number
15
+ data: EntityVersionData
16
+ createdAt: string
17
+ createdBy: string | null
18
+ }
19
+
20
+ export interface EntityVersionWithAuthor extends EntityVersion {
21
+ createdByUser: EntityVersionAuthor | null
22
+ }
23
+
24
+ export const entityVersionsCacheTags = {
25
+ for: (entityType: string, entityId: string) => `entityVersions:${entityType}:${entityId}`
26
+ } as const
@@ -0,0 +1,21 @@
1
+ 'use server'
2
+
3
+ import db from '@admin/db'
4
+ import { formSettings } from '@admin/db/schema'
5
+ import { cacheLife, cacheTag } from 'next/cache'
6
+ import type { FormSettingsData } from './types'
7
+ import { formSettingsCacheTags } from './types'
8
+
9
+ export async function getAllFormSettings(): Promise<FormSettingsData[]> {
10
+ 'use cache'
11
+ cacheLife('max')
12
+ cacheTag(formSettingsCacheTags.all)
13
+
14
+ try {
15
+ const settings = await db.select().from(formSettings)
16
+ return settings as FormSettingsData[]
17
+ } catch (error) {
18
+ console.error('Error fetching all form settings:', error)
19
+ return []
20
+ }
21
+ }
@@ -0,0 +1,27 @@
1
+ 'use server'
2
+
3
+ import db from '@admin/db'
4
+ import { formSettings } from '@admin/db/schema'
5
+ import { eq } from 'drizzle-orm'
6
+ import { cacheLife, cacheTag } from 'next/cache'
7
+ import type { FormSettingsData } from './types'
8
+ import { formSettingsCacheTags } from './types'
9
+
10
+ export async function getFormSettings(formName: string): Promise<FormSettingsData | null> {
11
+ 'use cache'
12
+ cacheLife('max')
13
+ cacheTag(formSettingsCacheTags.byName(formName))
14
+ cacheTag(formSettingsCacheTags.all)
15
+
16
+ try {
17
+ const [settings] = await db
18
+ .select()
19
+ .from(formSettings)
20
+ .where(eq(formSettings.formName, formName))
21
+ .limit(1)
22
+ return (settings as FormSettingsData) ?? null
23
+ } catch (error) {
24
+ console.error(`Error fetching form settings for ${formName}:`, error)
25
+ return null
26
+ }
27
+ }
@@ -0,0 +1,9 @@
1
+ export { getAllFormSettings } from './get-all-form-settings'
2
+ export { getFormSettings } from './get-form-settings'
3
+ export { testFormWebhook } from './test-form-webhook'
4
+ export type {
5
+ FormSettingsData,
6
+ FormSettingsResult,
7
+ UpsertFormSettingsInput
8
+ } from './types'
9
+ export { upsertFormSettings } from './upsert-form-settings'
@@ -0,0 +1,40 @@
1
+ 'use server'
2
+
3
+ import { getFormSettings } from './get-form-settings'
4
+
5
+ export async function testFormWebhook(
6
+ formName: string
7
+ ): Promise<{ success: boolean; error?: string }> {
8
+ try {
9
+ const settings = await getFormSettings(formName)
10
+ if (!settings?.webhookUrl) {
11
+ return { success: false, error: 'No webhook URL configured' }
12
+ }
13
+
14
+ const formData = new URLSearchParams()
15
+ formData.append('form_name', formName)
16
+ formData.append('test', 'true')
17
+ formData.append('message', 'This is a test webhook from BetterStart')
18
+ formData.append('timestamp', new Date().toISOString())
19
+
20
+ const response = await fetch(settings.webhookUrl, {
21
+ method: 'POST',
22
+ headers: { 'Content-Type': 'application/x-www-form-urlencoded' },
23
+ body: formData.toString()
24
+ })
25
+
26
+ if (!response.ok) {
27
+ return {
28
+ success: false,
29
+ error: `Webhook returned status ${response.status}`
30
+ }
31
+ }
32
+
33
+ return { success: true }
34
+ } catch (error) {
35
+ return {
36
+ success: false,
37
+ error: error instanceof Error ? error.message : 'Failed to send test webhook'
38
+ }
39
+ }
40
+ }
@@ -0,0 +1,26 @@
1
+ export interface FormSettingsData {
2
+ id: string
3
+ formName: string
4
+ webhookUrl: string | null
5
+ webhookEnabled: boolean
6
+ notificationEmails: string | null
7
+ createdAt: string
8
+ updatedAt: string
9
+ }
10
+
11
+ export interface UpsertFormSettingsInput {
12
+ webhookUrl?: string | null
13
+ webhookEnabled?: boolean
14
+ notificationEmails?: string | null
15
+ }
16
+
17
+ export interface FormSettingsResult {
18
+ success: boolean
19
+ error?: string
20
+ settings?: FormSettingsData
21
+ }
22
+
23
+ export const formSettingsCacheTags = {
24
+ all: 'form-settings:all',
25
+ byName: (formName: string) => `form-settings:name:${formName}`
26
+ } as const
@@ -0,0 +1,40 @@
1
+ 'use server'
2
+
3
+ import db from '@admin/db'
4
+ import { formSettings } from '@admin/db/schema'
5
+ import { updateTag } from 'next/cache'
6
+ import type { FormSettingsData, FormSettingsResult, UpsertFormSettingsInput } from './types'
7
+ import { formSettingsCacheTags } from './types'
8
+
9
+ export async function upsertFormSettings(
10
+ formName: string,
11
+ data: UpsertFormSettingsInput
12
+ ): Promise<FormSettingsResult> {
13
+ try {
14
+ const [settings] = await db
15
+ .insert(formSettings)
16
+ .values({
17
+ formName,
18
+ webhookUrl: data.webhookUrl ?? null,
19
+ webhookEnabled: data.webhookEnabled ?? false,
20
+ notificationEmails: data.notificationEmails ?? null
21
+ })
22
+ .onConflictDoUpdate({
23
+ target: formSettings.formName,
24
+ set: {
25
+ ...data,
26
+ updatedAt: new Date().toISOString()
27
+ }
28
+ })
29
+ .returning()
30
+ updateTag(formSettingsCacheTags.all)
31
+ updateTag(formSettingsCacheTags.byName(formName))
32
+ return { success: true, settings: settings as FormSettingsData }
33
+ } catch (error) {
34
+ console.error(`Error upserting form settings for ${formName}:`, error)
35
+ return {
36
+ success: false,
37
+ error: error instanceof Error ? error.message : 'Failed to save form settings'
38
+ }
39
+ }
40
+ }
@@ -0,0 +1,39 @@
1
+ 'use server'
2
+
3
+ import { getSession } from '@admin/auth/middleware'
4
+ import db from '@admin/db'
5
+ import { adminMedia } from '@admin/db/schema'
6
+ import { updateTag } from 'next/cache'
7
+ import { getMediaById } from './get-media-by-id'
8
+ import { type CreateMediaInput, type CreateMediaResult, mediaCacheTags } from './types'
9
+
10
+ export async function createMedia(data: CreateMediaInput): Promise<CreateMediaResult> {
11
+ try {
12
+ const session = await getSession()
13
+ const actorId = session?.user?.id ?? null
14
+ const now = new Date().toISOString()
15
+ const [row] = await db
16
+ .insert(adminMedia)
17
+ .values({
18
+ ...data,
19
+ createdAt: now,
20
+ updatedAt: now,
21
+ createdBy: actorId,
22
+ updatedBy: actorId
23
+ })
24
+ .returning({ id: adminMedia.id })
25
+ if (row?.id) {
26
+ updateTag(mediaCacheTags.all)
27
+ updateTag(mediaCacheTags.byId(row.id))
28
+ }
29
+ const media = row ? await getMediaById(row.id) : null
30
+ return media
31
+ ? { success: true, media }
32
+ : { success: false, error: 'Failed to create media record' }
33
+ } catch (error) {
34
+ return {
35
+ success: false,
36
+ error: error instanceof Error ? error.message : 'Failed to create media record'
37
+ }
38
+ }
39
+ }
@@ -0,0 +1,29 @@
1
+ 'use server'
2
+
3
+ import db from '@admin/db'
4
+ import { adminMedia } from '@admin/db/schema'
5
+ import { inArray } from 'drizzle-orm'
6
+ import { updateTag } from 'next/cache'
7
+ import type { DeleteMediaBulkResult } from './types'
8
+ import { mediaCacheTags } from './types'
9
+
10
+ export async function deleteMediaBulk(ids: string[]): Promise<DeleteMediaBulkResult> {
11
+ if (ids.length === 0) return { success: true, deletedCount: 0 }
12
+
13
+ try {
14
+ const deletedRows = await db
15
+ .delete(adminMedia)
16
+ .where(inArray(adminMedia.id, ids))
17
+ .returning({ id: adminMedia.id })
18
+ updateTag(mediaCacheTags.all)
19
+ for (const id of ids) {
20
+ updateTag(mediaCacheTags.byId(id))
21
+ }
22
+ return { success: true, deletedCount: deletedRows.length }
23
+ } catch (error) {
24
+ return {
25
+ success: false,
26
+ error: error instanceof Error ? error.message : 'Failed to delete media'
27
+ }
28
+ }
29
+ }
@@ -0,0 +1,22 @@
1
+ 'use server'
2
+
3
+ import db from '@admin/db'
4
+ import { adminMedia } from '@admin/db/schema'
5
+ import { eq } from 'drizzle-orm'
6
+ import { updateTag } from 'next/cache'
7
+ import type { DeleteMediaResult } from './types'
8
+ import { mediaCacheTags } from './types'
9
+
10
+ export async function deleteMedia(id: string): Promise<DeleteMediaResult> {
11
+ try {
12
+ await db.delete(adminMedia).where(eq(adminMedia.id, id))
13
+ updateTag(mediaCacheTags.all)
14
+ updateTag(mediaCacheTags.byId(id))
15
+ return { success: true }
16
+ } catch (error) {
17
+ return {
18
+ success: false,
19
+ error: error instanceof Error ? error.message : 'Failed to delete media'
20
+ }
21
+ }
22
+ }
@@ -0,0 +1,18 @@
1
+ 'use server'
2
+
3
+ import { adminMedia } from '@admin/db/schema'
4
+ import type { AdminMedia } from '@admin/types'
5
+ import { mapMediaRow, selectMedia } from '@admin/utils/media/query'
6
+ import { eq } from 'drizzle-orm'
7
+ import { cacheLife, cacheTag } from 'next/cache'
8
+ import { mediaCacheTags } from './types'
9
+
10
+ export async function getMediaById(id: string): Promise<AdminMedia | null> {
11
+ 'use cache'
12
+ cacheLife('max')
13
+ cacheTag(mediaCacheTags.byId(id))
14
+ cacheTag(mediaCacheTags.all)
15
+
16
+ const [row] = await selectMedia().where(eq(adminMedia.id, id)).limit(1)
17
+ return row ? mapMediaRow(row) : null
18
+ }
@@ -0,0 +1,25 @@
1
+ 'use server'
2
+
3
+ import { adminMedia } from '@admin/db/schema'
4
+ import type { AdminMedia } from '@admin/types'
5
+ import { mapMediaRow, selectMedia } from '@admin/utils/media/query'
6
+ import { inArray } from 'drizzle-orm'
7
+ import { cacheLife, cacheTag } from 'next/cache'
8
+ import { mediaCacheTags } from './types'
9
+
10
+ export async function getMediaByIds(ids: string[]): Promise<(AdminMedia | null)[]> {
11
+ 'use cache'
12
+ cacheLife('max')
13
+ cacheTag(mediaCacheTags.all)
14
+ for (const id of ids) {
15
+ cacheTag(mediaCacheTags.byId(id))
16
+ }
17
+
18
+ if (ids.length === 0) return []
19
+
20
+ const uniqueIds = [...new Set(ids)]
21
+ const rows = await selectMedia().where(inArray(adminMedia.id, uniqueIds))
22
+
23
+ const map = new Map(rows.map((row) => [row.id, mapMediaRow(row)]))
24
+ return ids.map((id) => map.get(id) ?? null)
25
+ }
@@ -0,0 +1,71 @@
1
+ 'use server'
2
+
3
+ import db from '@admin/db'
4
+ import { adminMedia } from '@admin/db/schema'
5
+ import type { MediaListParams, MediaListResult } from '@admin/types'
6
+ import { mapMediaRow, selectMedia } from '@admin/utils/media/query'
7
+ import { and, desc, ilike, sql } from 'drizzle-orm'
8
+ import { cacheLife, cacheTag } from 'next/cache'
9
+ import { mediaCacheTags } from './types'
10
+
11
+ const DEFAULT_MEDIA_PAGE = 1
12
+ const DEFAULT_MEDIA_PAGE_SIZE = 40
13
+
14
+ type GetMediaParams = MediaListParams
15
+ type GetMediaResult = MediaListResult
16
+
17
+ function clampMediaPage(value: number | undefined): number {
18
+ if (typeof value !== 'number' || !Number.isFinite(value)) {
19
+ return DEFAULT_MEDIA_PAGE
20
+ }
21
+
22
+ return Math.max(DEFAULT_MEDIA_PAGE, Math.floor(value))
23
+ }
24
+
25
+ function clampMediaPageSize(value: number | undefined): number {
26
+ if (typeof value !== 'number' || !Number.isFinite(value) || value <= 0) {
27
+ return DEFAULT_MEDIA_PAGE_SIZE
28
+ }
29
+
30
+ return Math.max(1, Math.floor(value))
31
+ }
32
+
33
+ export async function getMedia(params: GetMediaParams = {}) {
34
+ 'use cache'
35
+ cacheLife('max')
36
+ cacheTag(mediaCacheTags.all)
37
+
38
+ const { search, contentType } = params
39
+ const page = clampMediaPage(params.page)
40
+ const pageSize = clampMediaPageSize(params.pageSize)
41
+ const offset = (page - 1) * pageSize
42
+
43
+ const conditions = []
44
+ if (search) {
45
+ conditions.push(ilike(adminMedia.filename, `%${search}%`))
46
+ }
47
+ if (contentType) {
48
+ conditions.push(ilike(adminMedia.contentType, `${contentType}%`))
49
+ }
50
+
51
+ const where = conditions.length > 0 ? and(...conditions) : undefined
52
+ const itemsQuery = selectMedia()
53
+ .where(where)
54
+ .orderBy(desc(adminMedia.createdAt))
55
+ .limit(pageSize)
56
+ .offset(offset)
57
+
58
+ const [items, countResult] = await Promise.all([
59
+ itemsQuery,
60
+ db.select({ count: sql<number>`count(*)` }).from(adminMedia).where(where)
61
+ ])
62
+
63
+ const result: GetMediaResult = {
64
+ media: items.map(mapMediaRow),
65
+ total: Number(countResult[0]?.count ?? 0),
66
+ page,
67
+ pageSize
68
+ }
69
+
70
+ return result
71
+ }
@@ -0,0 +1,14 @@
1
+ export { createMedia } from './create-media'
2
+ export { deleteMedia } from './delete-media'
3
+ export { deleteMediaBulk } from './delete-media-bulk'
4
+ export { getMedia } from './get-media'
5
+ export { getMediaById } from './get-media-by-id'
6
+ export { getMediaByIds } from './get-media-by-ids'
7
+ export type {
8
+ CreateMediaInput,
9
+ CreateMediaResult,
10
+ DeleteMediaBulkResult,
11
+ DeleteMediaResult,
12
+ UpdateMediaResult
13
+ } from './types'
14
+ export { updateMedia } from './update-media'
@@ -0,0 +1,31 @@
1
+ import type { AdminMedia } from '@admin/types'
2
+
3
+ export type CreateMediaInput = Omit<
4
+ AdminMedia,
5
+ 'id' | 'createdAt' | 'updatedAt' | 'createdBy' | 'updatedBy' | 'createdByUser' | 'updatedByUser'
6
+ >
7
+
8
+ export interface CreateMediaResult {
9
+ success: boolean
10
+ error?: string
11
+ media?: AdminMedia
12
+ }
13
+
14
+ export interface UpdateMediaResult {
15
+ success: boolean
16
+ error?: string
17
+ }
18
+
19
+ export interface DeleteMediaResult {
20
+ success: boolean
21
+ error?: string
22
+ }
23
+
24
+ export interface DeleteMediaBulkResult extends DeleteMediaResult {
25
+ deletedCount?: number
26
+ }
27
+
28
+ export const mediaCacheTags = {
29
+ all: 'media:all',
30
+ byId: (id: string) => `media:id:${id}`
31
+ } as const
@@ -0,0 +1,35 @@
1
+ 'use server'
2
+
3
+ import { getSession } from '@admin/auth/middleware'
4
+ import db from '@admin/db'
5
+ import { adminMedia } from '@admin/db/schema'
6
+ import { eq } from 'drizzle-orm'
7
+ import { updateTag } from 'next/cache'
8
+ import type { UpdateMediaResult } from './types'
9
+ import { mediaCacheTags } from './types'
10
+
11
+ export async function updateMedia(
12
+ id: string,
13
+ data: { alt?: string; tags?: string[] }
14
+ ): Promise<UpdateMediaResult> {
15
+ try {
16
+ const session = await getSession()
17
+ const actorId = session?.user?.id ?? null
18
+ await db
19
+ .update(adminMedia)
20
+ .set({
21
+ ...data,
22
+ updatedAt: new Date().toISOString(),
23
+ updatedBy: actorId
24
+ })
25
+ .where(eq(adminMedia.id, id))
26
+ updateTag(mediaCacheTags.all)
27
+ updateTag(mediaCacheTags.byId(id))
28
+ return { success: true }
29
+ } catch (error) {
30
+ return {
31
+ success: false,
32
+ error: error instanceof Error ? error.message : 'Failed to update media'
33
+ }
34
+ }
35
+ }
@@ -0,0 +1,4 @@
1
+ export { invalidateUsersCache } from './invalidate-users-cache'
2
+ export { isEmailConfigured } from './is-email-configured'
3
+ export type { UpdateEmailResult } from './types'
4
+ export { updateEmail } from './update-email'
@@ -0,0 +1,14 @@
1
+ 'use server'
2
+
3
+ import { getSession } from '@admin/auth/middleware'
4
+ import { updateTag } from 'next/cache'
5
+ import { usersCacheTags } from '../users/types'
6
+
7
+ export async function invalidateUsersCache(): Promise<void> {
8
+ const session = await getSession()
9
+ if (!session?.user) {
10
+ return
11
+ }
12
+
13
+ updateTag(usersCacheTags.all)
14
+ }
@@ -0,0 +1,7 @@
1
+ 'use server'
2
+
3
+ import { isEmailDeliveryConfigured } from '@admin/actions/email'
4
+
5
+ export async function isEmailConfigured(): Promise<boolean> {
6
+ return isEmailDeliveryConfigured()
7
+ }
@@ -0,0 +1,4 @@
1
+ export interface UpdateEmailResult {
2
+ success: boolean
3
+ error?: string
4
+ }
@@ -0,0 +1,57 @@
1
+ 'use server'
2
+
3
+ import { auth } from '@admin/auth'
4
+ import { getSession } from '@admin/auth/middleware'
5
+ import db from '@admin/db'
6
+ import { user } from '@admin/db/schema'
7
+ import { eq } from 'drizzle-orm'
8
+ import { headers } from 'next/headers'
9
+ import { invalidateUsersCache } from './invalidate-users-cache'
10
+ import type { UpdateEmailResult } from './types'
11
+
12
+ export async function updateEmail(
13
+ newEmail: string,
14
+ currentPassword: string
15
+ ): Promise<UpdateEmailResult> {
16
+ try {
17
+ const session = await getSession({ disableCookieCache: true })
18
+ if (!session?.user) {
19
+ return { success: false, error: 'Not authenticated' }
20
+ }
21
+
22
+ const verification = await auth.api
23
+ .verifyPassword({
24
+ body: { password: currentPassword },
25
+ headers: await headers()
26
+ })
27
+ .catch(() => null)
28
+
29
+ if (!verification?.status) {
30
+ return { success: false, error: 'Incorrect password' }
31
+ }
32
+
33
+ const existing = await db
34
+ .select({ id: user.id })
35
+ .from(user)
36
+ .where(eq(user.email, newEmail))
37
+ .limit(1)
38
+
39
+ if (existing.length > 0) {
40
+ return { success: false, error: 'Email already in use' }
41
+ }
42
+
43
+ await db
44
+ .update(user)
45
+ .set({ email: newEmail, updatedAt: new Date() })
46
+ .where(eq(user.id, session.user.id))
47
+
48
+ await invalidateUsersCache()
49
+
50
+ return { success: true }
51
+ } catch (error) {
52
+ return {
53
+ success: false,
54
+ error: error instanceof Error ? error.message : 'Failed to update email'
55
+ }
56
+ }
57
+ }
@@ -0,0 +1,2 @@
1
+ export { getStorageProvider } from './provider'
2
+ export { saveUpload } from './save-upload'