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,160 @@
1
+ import { sql } from 'drizzle-orm'
2
+ import {
3
+ boolean,
4
+ index,
5
+ integer,
6
+ jsonb,
7
+ pgTable,
8
+ text,
9
+ timestamp,
10
+ uniqueIndex,
11
+ uuid,
12
+ varchar
13
+ } from 'drizzle-orm/pg-core'
14
+
15
+ // ============================================================================
16
+ // Better Auth tables
17
+ // ============================================================================
18
+
19
+ export const user = pgTable('user', {
20
+ id: text('id').primaryKey(),
21
+ name: text('name').notNull(),
22
+ email: text('email').notNull().unique(),
23
+ emailVerified: boolean('email_verified').notNull().default(false),
24
+ image: text('image'),
25
+ role: text('role').notNull().default('member'),
26
+ createdAt: timestamp('created_at').notNull().defaultNow(),
27
+ updatedAt: timestamp('updated_at').notNull().defaultNow()
28
+ })
29
+
30
+ export const session = pgTable('session', {
31
+ id: text('id').primaryKey(),
32
+ expiresAt: timestamp('expires_at').notNull(),
33
+ token: text('token').notNull().unique(),
34
+ createdAt: timestamp('created_at').notNull().defaultNow(),
35
+ updatedAt: timestamp('updated_at').notNull().defaultNow(),
36
+ ipAddress: text('ip_address'),
37
+ userAgent: text('user_agent'),
38
+ userId: text('user_id')
39
+ .notNull()
40
+ .references(() => user.id, { onDelete: 'cascade' })
41
+ })
42
+
43
+ export const account = pgTable('account', {
44
+ id: text('id').primaryKey(),
45
+ accountId: text('account_id').notNull(),
46
+ providerId: text('provider_id').notNull(),
47
+ userId: text('user_id')
48
+ .notNull()
49
+ .references(() => user.id, { onDelete: 'cascade' }),
50
+ accessToken: text('access_token'),
51
+ refreshToken: text('refresh_token'),
52
+ idToken: text('id_token'),
53
+ accessTokenExpiresAt: timestamp('access_token_expires_at'),
54
+ refreshTokenExpiresAt: timestamp('refresh_token_expires_at'),
55
+ scope: text('scope'),
56
+ password: text('password'),
57
+ createdAt: timestamp('created_at').notNull().defaultNow(),
58
+ updatedAt: timestamp('updated_at').notNull().defaultNow()
59
+ })
60
+
61
+ export const verification = pgTable('verification', {
62
+ id: text('id').primaryKey(),
63
+ identifier: text('identifier').notNull(),
64
+ value: text('value').notNull(),
65
+ expiresAt: timestamp('expires_at').notNull(),
66
+ createdAt: timestamp('created_at').defaultNow(),
67
+ updatedAt: timestamp('updated_at').defaultNow()
68
+ })
69
+
70
+ // ============================================================================
71
+ // Form Settings (shared table for all form configurations)
72
+ // ============================================================================
73
+
74
+ export const formSettings = pgTable(
75
+ 'FormSettings',
76
+ {
77
+ id: uuid().defaultRandom().primaryKey().notNull(),
78
+ formName: varchar('formName', { length: 100 }).notNull(),
79
+ webhookUrl: text('webhookUrl'),
80
+ webhookEnabled: boolean('webhookEnabled').default(false).notNull(),
81
+ notificationEmails: text('notificationEmails'),
82
+ createdAt: timestamp('createdAt', { precision: 3, mode: 'string' })
83
+ .default(sql`CURRENT_TIMESTAMP`)
84
+ .notNull(),
85
+ updatedAt: timestamp('updatedAt', { precision: 3, mode: 'string' })
86
+ .default(sql`CURRENT_TIMESTAMP`)
87
+ .notNull()
88
+ },
89
+ (table) => [
90
+ uniqueIndex('FormSettings_formName_key').using(
91
+ 'btree',
92
+ table.formName.asc().nullsLast().op('text_ops')
93
+ )
94
+ ]
95
+ )
96
+
97
+ // ============================================================================
98
+ // Entity Versions
99
+ // ============================================================================
100
+
101
+ export const entityVersions = pgTable(
102
+ 'EntityVersions',
103
+ {
104
+ id: uuid().defaultRandom().primaryKey().notNull(),
105
+ entityType: varchar('entity_type', { length: 100 }).notNull(),
106
+ entityId: uuid('entity_id').notNull(),
107
+ version: integer().notNull(),
108
+ data: jsonb().notNull(),
109
+ createdAt: timestamp('created_at', { precision: 3, mode: 'string' })
110
+ .default(sql`CURRENT_TIMESTAMP`)
111
+ .notNull(),
112
+ createdBy: text('created_by').references(() => user.id, {
113
+ onDelete: 'set null'
114
+ })
115
+ },
116
+ (table) => [
117
+ uniqueIndex('EntityVersions_entityType_entityId_version_key').using(
118
+ 'btree',
119
+ table.entityType.asc().nullsLast().op('text_ops'),
120
+ table.entityId.asc().nullsLast().op('uuid_ops'),
121
+ table.version.asc().nullsLast()
122
+ ),
123
+ index('EntityVersions_entityType_entityId_createdAt_idx').using(
124
+ 'btree',
125
+ table.entityType.asc().nullsLast().op('text_ops'),
126
+ table.entityId.asc().nullsLast().op('uuid_ops'),
127
+ table.createdAt.desc().nullsLast().op('timestamp_ops')
128
+ )
129
+ ]
130
+ )
131
+
132
+ // ============================================================================
133
+ // Media Library
134
+ // ============================================================================
135
+
136
+ export const adminMedia = pgTable('AdminMedia', {
137
+ id: text('id')
138
+ .primaryKey()
139
+ .$defaultFn(() => crypto.randomUUID()),
140
+ url: text('url').notNull(),
141
+ filename: varchar('filename', { length: 500 }).notNull(),
142
+ contentType: varchar('content_type', { length: 255 }).notNull(),
143
+ size: integer('size').notNull(),
144
+ width: integer('width'),
145
+ height: integer('height'),
146
+ alt: text('alt'),
147
+ tags: text('tags').array(),
148
+ createdAt: timestamp('created_at', { precision: 3, mode: 'string' })
149
+ .default(sql`CURRENT_TIMESTAMP`)
150
+ .notNull(),
151
+ updatedAt: timestamp('updated_at', { precision: 3, mode: 'string' })
152
+ .default(sql`CURRENT_TIMESTAMP`)
153
+ .notNull(),
154
+ createdBy: text('created_by').references(() => user.id, {
155
+ onDelete: 'set null'
156
+ }),
157
+ updatedBy: text('updated_by').references(() => user.id, {
158
+ onDelete: 'set null'
159
+ })
160
+ })
@@ -0,0 +1 @@
1
+ export * from './core'
@@ -0,0 +1,19 @@
1
+ import './register'
2
+ import './register.local'
3
+
4
+ export {
5
+ getLifecycleHookSummary,
6
+ getRegisteredHooks,
7
+ registerHook
8
+ } from './registry'
9
+ export { runAfterHooks, runBeforeHooks } from './runner'
10
+ export type {
11
+ AfterLifecycleHookEvent,
12
+ AfterLifecycleHookHandler,
13
+ BeforeLifecycleHookEvent,
14
+ BeforeLifecycleHookHandler,
15
+ LifecycleHookAbort,
16
+ LifecycleHookContext,
17
+ LifecycleHookEvent,
18
+ LifecycleHookHandler
19
+ } from './types'
@@ -0,0 +1,2 @@
1
+ // Register project-specific lifecycle hooks here.
2
+ // This file is committed so teams can share hooks alongside generated admin code.
@@ -0,0 +1,9 @@
1
+ // This file is managed by BetterStart and may be overwritten by CLI updates.
2
+ // Add project-specific lifecycle hook registrations in register.local.ts.
3
+ //
4
+ // Example:
5
+ // import { registerHook } from './registry'
6
+ //
7
+ // registerHook('posts', 'afterCreate', async ({ user, entityId, data }) => {
8
+ // console.log('Post created', { userId: user.id, entityId, data })
9
+ // })
@@ -0,0 +1,55 @@
1
+ import type { LifecycleHookEvent, LifecycleHookHandler } from './types'
2
+
3
+ type HookBuckets = Map<LifecycleHookEvent, LifecycleHookHandler<LifecycleHookEvent>[]>
4
+
5
+ const lifecycleHooks = new Map<string, HookBuckets>()
6
+
7
+ export function registerHook<Event extends LifecycleHookEvent>(
8
+ entityType: string,
9
+ event: Event,
10
+ handler: LifecycleHookHandler<Event>
11
+ ): () => void {
12
+ const normalizedEntityType = entityType.trim()
13
+ if (!normalizedEntityType || normalizedEntityType === '*') {
14
+ throw new Error('Lifecycle hooks must be registered for a specific entity type')
15
+ }
16
+
17
+ const buckets = lifecycleHooks.get(normalizedEntityType) ?? new Map()
18
+ const handlers = buckets.get(event) ?? []
19
+ handlers.push(handler as LifecycleHookHandler<LifecycleHookEvent>)
20
+ buckets.set(event, handlers)
21
+ lifecycleHooks.set(normalizedEntityType, buckets)
22
+
23
+ return () => {
24
+ const currentHandlers = buckets.get(event)
25
+ if (!currentHandlers) return
26
+
27
+ const index = currentHandlers.indexOf(handler as LifecycleHookHandler<LifecycleHookEvent>)
28
+ if (index >= 0) {
29
+ currentHandlers.splice(index, 1)
30
+ }
31
+ }
32
+ }
33
+
34
+ export function getRegisteredHooks<Event extends LifecycleHookEvent>(
35
+ entityType: string,
36
+ event: Event
37
+ ): LifecycleHookHandler<Event>[] {
38
+ return [...((lifecycleHooks.get(entityType)?.get(event) ?? []) as LifecycleHookHandler<Event>[])]
39
+ }
40
+
41
+ export function getLifecycleHookSummary(): Record<
42
+ string,
43
+ Partial<Record<LifecycleHookEvent, number>>
44
+ > {
45
+ const summary: Record<string, Partial<Record<LifecycleHookEvent, number>>> = {}
46
+
47
+ for (const [entityType, events] of lifecycleHooks.entries()) {
48
+ summary[entityType] = {}
49
+ for (const [event, handlers] of events.entries()) {
50
+ summary[entityType][event] = handlers.length
51
+ }
52
+ }
53
+
54
+ return summary
55
+ }
@@ -0,0 +1,51 @@
1
+ import { getRegisteredHooks } from './registry'
2
+ import type {
3
+ AfterLifecycleHookEvent,
4
+ BeforeLifecycleHookEvent,
5
+ LifecycleHookAbort,
6
+ LifecycleHookContext
7
+ } from './types'
8
+
9
+ function isLifecycleHookAbort(value: unknown): value is LifecycleHookAbort {
10
+ return (
11
+ typeof value === 'object' &&
12
+ value !== null &&
13
+ 'abort' in value &&
14
+ (value as { abort: unknown }).abort === true
15
+ )
16
+ }
17
+
18
+ function createAbortError(event: BeforeLifecycleHookEvent, context: LifecycleHookContext): Error {
19
+ return new Error(`Lifecycle hook ${event} aborted ${context.entityType}`)
20
+ }
21
+
22
+ export async function runBeforeHooks(
23
+ event: BeforeLifecycleHookEvent,
24
+ context: LifecycleHookContext
25
+ ): Promise<void> {
26
+ const handlers = getRegisteredHooks(context.entityType, event)
27
+
28
+ for (const handler of handlers) {
29
+ const result = await handler(context)
30
+ if (isLifecycleHookAbort(result)) {
31
+ throw result.reason ? new Error(result.reason) : createAbortError(event, context)
32
+ }
33
+ }
34
+ }
35
+
36
+ export function runAfterHooks(event: AfterLifecycleHookEvent, context: LifecycleHookContext): void {
37
+ const handlers = getRegisteredHooks(context.entityType, event)
38
+ if (handlers.length === 0) return
39
+
40
+ void Promise.allSettled(handlers.map((handler) => handler(context))).then((results) => {
41
+ for (const result of results) {
42
+ if (result.status === 'rejected') {
43
+ console.error('Lifecycle hook failed:', {
44
+ entityType: context.entityType,
45
+ event,
46
+ error: result.reason
47
+ })
48
+ }
49
+ }
50
+ })
51
+ }
@@ -0,0 +1,39 @@
1
+ import type { User } from '@admin/auth'
2
+
3
+ export type LifecycleHookEvent =
4
+ | 'beforeCreate'
5
+ | 'afterCreate'
6
+ | 'beforeUpdate'
7
+ | 'afterUpdate'
8
+ | 'beforeDelete'
9
+ | 'afterDelete'
10
+ | 'beforePublish'
11
+ | 'afterPublish'
12
+ | 'beforeUnpublish'
13
+ | 'afterUnpublish'
14
+
15
+ export type BeforeLifecycleHookEvent = Extract<LifecycleHookEvent, `before${string}`>
16
+ export type AfterLifecycleHookEvent = Extract<LifecycleHookEvent, `after${string}`>
17
+
18
+ export interface LifecycleHookContext {
19
+ user: User
20
+ entityType: string
21
+ timestamp: string
22
+ entityId?: string | undefined
23
+ input?: Record<string, unknown> | undefined
24
+ data?: Record<string, unknown> | undefined
25
+ }
26
+
27
+ export interface LifecycleHookAbort {
28
+ abort: true
29
+ reason?: string
30
+ }
31
+
32
+ export type BeforeLifecycleHookHandler = (
33
+ context: LifecycleHookContext
34
+ ) => undefined | LifecycleHookAbort | Promise<undefined | LifecycleHookAbort>
35
+
36
+ export type AfterLifecycleHookHandler = (context: LifecycleHookContext) => void | Promise<void>
37
+
38
+ export type LifecycleHookHandler<Event extends LifecycleHookEvent> =
39
+ Event extends BeforeLifecycleHookEvent ? BeforeLifecycleHookHandler : AfterLifecycleHookHandler
@@ -0,0 +1,11 @@
1
+ import { ContentSkeleton } from '@admin/components/layouts/content-skeleton'
2
+ import { Suspense } from 'react'
3
+ import { AccountShellRsc } from './account-shell-rsc'
4
+
5
+ export default function AccountLayout({ children }: { children: React.ReactNode }) {
6
+ return (
7
+ <Suspense fallback={<ContentSkeleton />}>
8
+ <AccountShellRsc>{children}</AccountShellRsc>
9
+ </Suspense>
10
+ )
11
+ }
@@ -0,0 +1,30 @@
1
+ import { getSession } from '@admin/auth/middleware'
2
+ import { AdminSidebar } from '@admin/components/layouts/admin-sidebar'
3
+ import { SidebarProvider } from '@admin/components/ui/sidebar'
4
+ import { admin } from '@admin/utils/app/admin'
5
+ import { canAccessAdmin } from '@admin/utils/auth/roles'
6
+ import { redirect } from 'next/navigation'
7
+ import { MinimalAccountShell } from './minimal-account-shell'
8
+
9
+ export async function AccountShellRsc({ children }: { children: React.ReactNode }) {
10
+ const session = await getSession({ disableCookieCache: true })
11
+
12
+ if (!session?.user) {
13
+ redirect('/admin/login')
14
+ }
15
+
16
+ if (!canAccessAdmin(session.user.role)) {
17
+ return (
18
+ <MinimalAccountShell name={session.user.name ?? admin.name}>{children}</MinimalAccountShell>
19
+ )
20
+ }
21
+
22
+ return (
23
+ <SidebarProvider className="min-h-svh">
24
+ <AdminSidebar />
25
+ <div className="relative flex min-w-0 flex-1 flex-col [--admin-shell-content-left:0px] md:[--admin-shell-content-left:var(--sidebar-width)] md:peer-data-[state=collapsed]:[--admin-shell-content-left:var(--sidebar-width-icon)]">
26
+ {children}
27
+ </div>
28
+ </SidebarProvider>
29
+ )
30
+ }
@@ -0,0 +1,24 @@
1
+ import '@admin/admin-globals.css'
2
+ import { AdminProviders } from '@admin/components/layouts/admin-providers'
3
+ import { cn } from '@admin/utils/shared/cn'
4
+ import { GeistMono } from 'geist/font/mono'
5
+ import { GeistSans } from 'geist/font/sans'
6
+ import { NuqsAdapter } from 'nuqs/adapters/next/app'
7
+
8
+ export default function AdminLayout({ children }: { children: React.ReactNode }) {
9
+ return (
10
+ <NuqsAdapter>
11
+ <AdminProviders>
12
+ <div
13
+ className={cn(
14
+ 'admin-root min-h-screen antialiased w-screen mt-px',
15
+ GeistSans.variable,
16
+ GeistMono.variable
17
+ )}
18
+ >
19
+ {children}
20
+ </div>
21
+ </AdminProviders>
22
+ </NuqsAdapter>
23
+ )
24
+ }
@@ -0,0 +1,6 @@
1
+ import { requireRole, UserRole } from '@admin/auth/middleware'
2
+
3
+ export async function AuthGateRsc({ children }: { children: React.ReactNode }) {
4
+ await requireRole([UserRole.ADMIN, UserRole.EDITOR])
5
+ return <>{children}</>
6
+ }
@@ -0,0 +1,18 @@
1
+ import { AdminSidebar } from '@admin/components/layouts/admin-sidebar'
2
+ import { ContentSkeleton } from '@admin/components/layouts/content-skeleton'
3
+ import { SidebarProvider } from '@admin/components/ui/sidebar'
4
+ import { Suspense } from 'react'
5
+ import { AuthGateRsc } from './auth-gate-rsc'
6
+
7
+ export default function AdminAuthLayout({ children }: { children: React.ReactNode }) {
8
+ return (
9
+ <SidebarProvider className="min-h-svh">
10
+ <AdminSidebar />
11
+ <div className="relative flex min-w-0 flex-1 flex-col [--admin-shell-content-left:0px] md:[--admin-shell-content-left:var(--sidebar-width)] md:peer-data-[state=collapsed]:[--admin-shell-content-left:var(--sidebar-width-icon)]">
12
+ <Suspense fallback={<ContentSkeleton />}>
13
+ <AuthGateRsc>{children}</AuthGateRsc>
14
+ </Suspense>
15
+ </div>
16
+ </SidebarProvider>
17
+ )
18
+ }
@@ -0,0 +1,121 @@
1
+ import { PageHeader } from '@admin/components/shared/page-header'
2
+ import {
3
+ Accordion,
4
+ AccordionContent,
5
+ AccordionItem,
6
+ AccordionTrigger
7
+ } from '@admin/components/ui/accordion'
8
+ import { Badge } from '@admin/components/ui/badge'
9
+ import { Button } from '@admin/components/ui/button'
10
+ import { Card, CardContent, CardHeader, CardTitle } from '@admin/components/ui/card'
11
+ import { Blocks, CircleGauge, FileQuestion, Globe, PlugZap, ShieldEllipsis } from 'lucide-react'
12
+ import React from 'react'
13
+
14
+ const envVariables = [
15
+ {
16
+ label: 'Database',
17
+ name: 'DATABASE_URL',
18
+ value: process.env.DATABASE_URL
19
+ },
20
+ {
21
+ label: 'Storage (Cloudflare R2)',
22
+ name: 'BETTERSTART_R2_BUCKET_NAME',
23
+ value: process.env.BETTERSTART_R2_BUCKET_NAME
24
+ },
25
+ {
26
+ label: 'Email (Resend)',
27
+ name: 'BETTERSTART_RESEND_API_KEY',
28
+ value: process.env.BETTERSTART_RESEND_API_KEY
29
+ }
30
+ ]
31
+
32
+ export default function DashboardPage() {
33
+ return (
34
+ <React.Fragment>
35
+ <PageHeader title="Dashboard" hasBackButton={false} />
36
+
37
+ <div className="p-6 space-y-6 w-full">
38
+ <Card>
39
+ <CardHeader>
40
+ <div className="flex items-center gap-2">
41
+ <CircleGauge className="size-4 text-muted-foreground" />
42
+ <CardTitle className="font-medium">Setup your dashboard</CardTitle>
43
+ </div>
44
+ </CardHeader>
45
+ <CardContent className="p-10!">
46
+ <div className="grid grid-cols-2 gap-10">
47
+ <div className="flex flex-col w-full gap-1">
48
+ <p className="text-base font-medium">Hey, let's get started</p>
49
+ <p className="text-muted-foreground">
50
+ Lorem ipsum dolor sit amet consectetur adipisicing elit. Quisquam, quos.
51
+ </p>
52
+ </div>
53
+ <div className="flex flex-col w-full">
54
+ <Accordion type="single" collapsible defaultValue="item-1">
55
+ <AccordionItem value="item-1">
56
+ <AccordionTrigger>
57
+ <div className="flex items-center gap-3">
58
+ <ShieldEllipsis className="size-4 text-muted-foreground" />
59
+ <span>Setup ENV Variables</span>
60
+ </div>
61
+ </AccordionTrigger>
62
+ <AccordionContent className="pl-7 h-auto">
63
+ <div className="flex flex-col w-full gap-1">
64
+ {envVariables.map((env) => (
65
+ <div
66
+ className="flex flex-col w-full bg-card border shadow-card border-border rounded-lg pl-4 pr-2 py-2"
67
+ key={env.name}
68
+ >
69
+ <div className="flex items-center justify-between">
70
+ <div className="flex items-center gap-2">
71
+ <span className="">{env.label}</span>
72
+ <Badge variant={env.value ? 'success' : 'destructive'}>
73
+ {env.value ? 'Connected' : 'Not configured'}
74
+ </Badge>
75
+ </div>
76
+
77
+ <Button variant="ghost" className="size-8">
78
+ <FileQuestion className="size-4" />
79
+ </Button>
80
+ </div>
81
+ </div>
82
+ ))}
83
+ </div>
84
+ </AccordionContent>
85
+ </AccordionItem>
86
+ <AccordionItem value="item-2">
87
+ <AccordionTrigger>
88
+ <div className="flex items-center gap-3">
89
+ <Blocks className="size-4 text-muted-foreground" />
90
+ <span>Install Pre-built Modules</span>
91
+ </div>
92
+ </AccordionTrigger>
93
+ <AccordionContent className="pl-7">Lorem</AccordionContent>
94
+ </AccordionItem>
95
+ <AccordionItem value="item-3">
96
+ <AccordionTrigger>
97
+ <div className="flex items-center gap-3">
98
+ <PlugZap className="size-4 text-muted-foreground" />
99
+ <span>Setup Provider Integrations</span>
100
+ </div>
101
+ </AccordionTrigger>
102
+ <AccordionContent className="pl-7">Lorem</AccordionContent>
103
+ </AccordionItem>
104
+ <AccordionItem value="item-4">
105
+ <AccordionTrigger>
106
+ <div className="flex items-center gap-3">
107
+ <Globe className="size-4 text-muted-foreground" />
108
+ <span>Deploy</span>
109
+ </div>
110
+ </AccordionTrigger>
111
+ <AccordionContent className="pl-7">Lorem</AccordionContent>
112
+ </AccordionItem>
113
+ </Accordion>
114
+ </div>
115
+ </div>
116
+ </CardContent>
117
+ </Card>
118
+ </div>
119
+ </React.Fragment>
120
+ )
121
+ }