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,116 @@
1
+ 'use client'
2
+
3
+ import { Button } from '@admin/components/ui/button'
4
+ import { Input } from '@admin/components/ui/input'
5
+ import { Label } from '@admin/components/ui/label'
6
+ import { Popover, PopoverContent, PopoverTrigger } from '@admin/components/ui/popover'
7
+ import { normalizeCount } from '@admin/utils/editor/table-input'
8
+ import { insertContentEditorTable } from '@admin/utils/editor/tiptap'
9
+ import type { Editor } from '@tiptap/core'
10
+ import { TableIcon } from 'lucide-react'
11
+ import * as React from 'react'
12
+
13
+ const DEFAULT_TABLE_ROWS = 3
14
+ const DEFAULT_TABLE_COLS = 3
15
+ const MIN_TABLE_ROWS = 2
16
+ const MIN_TABLE_COLS = 1
17
+
18
+ interface TableButtonProps {
19
+ disabled?: boolean
20
+ editor?: Editor | null
21
+ onInsertTable?: (size: { cols: number; rows: number }) => void
22
+ }
23
+
24
+ export function TableButton({ disabled, editor, onInsertTable }: TableButtonProps) {
25
+ const [isOpen, setIsOpen] = React.useState(false)
26
+ const [rows, setRows] = React.useState(String(DEFAULT_TABLE_ROWS))
27
+ const [cols, setCols] = React.useState(String(DEFAULT_TABLE_COLS))
28
+ const rowsId = React.useId()
29
+ const colsId = React.useId()
30
+ const canInsert = Boolean(onInsertTable || editor?.isEditable)
31
+
32
+ const closePopover = React.useCallback(() => {
33
+ setIsOpen(false)
34
+ window.setTimeout(() => setIsOpen(false), 0)
35
+ }, [])
36
+
37
+ const insertTable = React.useCallback(() => {
38
+ if (disabled || !canInsert) return
39
+
40
+ const rowCount = normalizeCount(rows, DEFAULT_TABLE_ROWS, MIN_TABLE_ROWS)
41
+ const colCount = normalizeCount(cols, DEFAULT_TABLE_COLS, MIN_TABLE_COLS)
42
+
43
+ if (onInsertTable) {
44
+ onInsertTable({ cols: colCount, rows: rowCount })
45
+ closePopover()
46
+ return
47
+ }
48
+
49
+ if (!editor) return
50
+
51
+ insertContentEditorTable(editor, rowCount, colCount)
52
+ closePopover()
53
+ }, [canInsert, closePopover, cols, disabled, editor, onInsertTable, rows])
54
+
55
+ const handleKeyDown = React.useCallback(
56
+ (event: React.KeyboardEvent<HTMLDivElement>) => {
57
+ if (event.key !== 'Enter') return
58
+
59
+ event.preventDefault()
60
+ event.stopPropagation()
61
+ insertTable()
62
+ },
63
+ [insertTable]
64
+ )
65
+
66
+ return (
67
+ <Popover open={isOpen} onOpenChange={setIsOpen}>
68
+ <PopoverTrigger asChild>
69
+ <Button
70
+ type="button"
71
+ variant="ghost"
72
+ size="sm"
73
+ disabled={disabled || !canInsert}
74
+ aria-label="Insert table"
75
+ title="Table"
76
+ onPointerDown={(event) => event.preventDefault()}
77
+ onClick={() => setIsOpen((current) => !current)}
78
+ >
79
+ <TableIcon className="tiptap-button-icon" />
80
+ <span className="tiptap-button-text">Table</span>
81
+ </Button>
82
+ </PopoverTrigger>
83
+ <PopoverContent align="start" className="w-56 p-3">
84
+ <div className="flex flex-col gap-3" onKeyDownCapture={handleKeyDown}>
85
+ <div className="grid grid-cols-2 gap-2">
86
+ <div className="flex flex-col gap-1.5">
87
+ <Label htmlFor={rowsId}>Rows</Label>
88
+ <Input
89
+ id={rowsId}
90
+ inputMode="numeric"
91
+ min={MIN_TABLE_ROWS}
92
+ onChange={(event) => setRows(event.target.value)}
93
+ type="number"
94
+ value={rows}
95
+ />
96
+ </div>
97
+ <div className="flex flex-col gap-1.5">
98
+ <Label htmlFor={colsId}>Cols</Label>
99
+ <Input
100
+ id={colsId}
101
+ inputMode="numeric"
102
+ min={MIN_TABLE_COLS}
103
+ onChange={(event) => setCols(event.target.value)}
104
+ type="number"
105
+ value={cols}
106
+ />
107
+ </div>
108
+ </div>
109
+ <Button type="button" size="sm" disabled={disabled || !canInsert} onClick={insertTable}>
110
+ Enter
111
+ </Button>
112
+ </div>
113
+ </PopoverContent>
114
+ </Popover>
115
+ )
116
+ }
@@ -0,0 +1,138 @@
1
+ import { getSelectedNodesOfType, updateNodesAttr } from '@admin/utils/editor/tiptap'
2
+ import type { NodeWithPos } from '@tiptap/core'
3
+ import { Extension } from '@tiptap/core'
4
+ import type { EditorState, Transaction } from '@tiptap/pm/state'
5
+
6
+ declare module '@tiptap/core' {
7
+ interface Commands<ReturnType> {
8
+ nodeBackground: {
9
+ setNodeBackgroundColor: (backgroundColor: string) => ReturnType
10
+ unsetNodeBackgroundColor: () => ReturnType
11
+ toggleNodeBackgroundColor: (backgroundColor: string) => ReturnType
12
+ }
13
+ }
14
+ }
15
+
16
+ export interface NodeBackgroundOptions {
17
+ /**
18
+ * Node types that should support background colors
19
+ * @default ["paragraph", "heading", "blockquote", "taskList", "bulletList", "orderedList", "tableCell", "tableHeader"]
20
+ */
21
+ types: string[]
22
+ /**
23
+ * Use inline style instead of data attribute
24
+ * @default true
25
+ */
26
+ useStyle?: boolean
27
+ }
28
+
29
+ /**
30
+ * Determines the target color for toggle operations
31
+ */
32
+ function getToggleColor(targets: NodeWithPos[], inputColor: string): string | null {
33
+ if (targets.length === 0) return null
34
+
35
+ for (const target of targets) {
36
+ const currentColor = target.node.attrs?.backgroundColor ?? null
37
+ if (currentColor !== inputColor) {
38
+ return inputColor
39
+ }
40
+ }
41
+
42
+ return null
43
+ }
44
+
45
+ export const NodeBackground = Extension.create<NodeBackgroundOptions>({
46
+ name: 'nodeBackground',
47
+
48
+ addOptions() {
49
+ return {
50
+ types: [
51
+ 'paragraph',
52
+ 'heading',
53
+ 'blockquote',
54
+ 'taskList',
55
+ 'bulletList',
56
+ 'orderedList',
57
+ 'tableCell',
58
+ 'tableHeader'
59
+ ],
60
+ useStyle: true
61
+ }
62
+ },
63
+
64
+ addGlobalAttributes() {
65
+ return [
66
+ {
67
+ types: this.options.types,
68
+ attributes: {
69
+ backgroundColor: {
70
+ default: null as string | null,
71
+
72
+ parseHTML: (element: HTMLElement) => {
73
+ const styleColor = element.style?.backgroundColor
74
+ if (styleColor) return styleColor
75
+
76
+ const dataColor = element.getAttribute('data-background-color')
77
+ return dataColor || null
78
+ },
79
+
80
+ renderHTML: (attributes) => {
81
+ const color = attributes.backgroundColor as string | null
82
+ if (!color) return {}
83
+
84
+ if (this.options.useStyle) {
85
+ return {
86
+ style: `background-color: ${color}`
87
+ }
88
+ } else {
89
+ return {
90
+ 'data-background-color': color
91
+ }
92
+ }
93
+ }
94
+ }
95
+ }
96
+ }
97
+ ]
98
+ },
99
+
100
+ addCommands() {
101
+ /**
102
+ * Generic command executor for background color operations
103
+ */
104
+ const executeBackgroundCommand = (
105
+ getTargetColor: (targets: NodeWithPos[], inputColor?: string) => string | null
106
+ ) => {
107
+ return (inputColor?: string) =>
108
+ ({ state, tr }: { state: EditorState; tr: Transaction }) => {
109
+ const targets = getSelectedNodesOfType(state.selection, this.options.types)
110
+
111
+ if (targets.length === 0) return false
112
+
113
+ const targetColor = getTargetColor(targets, inputColor)
114
+
115
+ return updateNodesAttr(tr, targets, 'backgroundColor', targetColor)
116
+ }
117
+ }
118
+
119
+ return {
120
+ /**
121
+ * Set background color to specific value
122
+ */
123
+ setNodeBackgroundColor: executeBackgroundCommand((_, inputColor) => inputColor || null),
124
+
125
+ /**
126
+ * Remove background color
127
+ */
128
+ unsetNodeBackgroundColor: executeBackgroundCommand(() => null),
129
+
130
+ /**
131
+ * Toggle background color (set if different/missing, unset if all have it)
132
+ */
133
+ toggleNodeBackgroundColor: executeBackgroundCommand((targets, inputColor) =>
134
+ getToggleColor(targets, inputColor || '')
135
+ )
136
+ }
137
+ }
138
+ })
@@ -0,0 +1,10 @@
1
+ import TiptapHorizontalRule from '@tiptap/extension-horizontal-rule'
2
+ import { mergeAttributes } from '@tiptap/react'
3
+
4
+ export const HorizontalRule = TiptapHorizontalRule.extend({
5
+ renderHTML() {
6
+ return ['div', mergeAttributes(this.options.HTMLAttributes, { 'data-type': this.name }), ['hr']]
7
+ }
8
+ })
9
+
10
+ export default HorizontalRule
@@ -0,0 +1 @@
1
+ export * from './media-gallery-placeholder-node-extension'
@@ -0,0 +1,117 @@
1
+ import type { NodeType } from '@tiptap/pm/model'
2
+ import { mergeAttributes, Node, ReactNodeViewRenderer } from '@tiptap/react'
3
+ import { MediaGalleryPlaceholderNode as MediaGalleryPlaceholderNodeComponent } from './media-gallery-placeholder-node'
4
+
5
+ export interface MediaGalleryPlaceholderNodeOptions {
6
+ /**
7
+ * The type of the node.
8
+ * @default 'image'
9
+ */
10
+ type?: string | NodeType | undefined
11
+ /**
12
+ * Acceptable media types for the gallery dialog.
13
+ * @default 'image/*'
14
+ */
15
+ accept?: string
16
+ /**
17
+ * HTML attributes to add to the image element.
18
+ * @default {}
19
+ * @example { class: 'foo' }
20
+ */
21
+ HTMLAttributes: Record<string, unknown>
22
+ }
23
+
24
+ declare module '@tiptap/react' {
25
+ interface Commands<ReturnType> {
26
+ mediaGalleryPlaceholder: {
27
+ setMediaGalleryPlaceholderNode: (options?: MediaGalleryPlaceholderNodeOptions) => ReturnType
28
+ }
29
+ }
30
+ }
31
+
32
+ /**
33
+ * A Tiptap node extension that creates a media gallery placeholder.
34
+ * @see registry/tiptap-node/media-gallery-placeholder-node/media-gallery-placeholder-node
35
+ */
36
+ export const MediaGalleryPlaceholderNode = Node.create<MediaGalleryPlaceholderNodeOptions>({
37
+ name: 'mediaGalleryPlaceholder',
38
+
39
+ group: 'block',
40
+
41
+ draggable: true,
42
+
43
+ selectable: true,
44
+
45
+ atom: true,
46
+
47
+ addOptions() {
48
+ return {
49
+ type: 'image',
50
+ accept: 'image/*',
51
+ HTMLAttributes: {}
52
+ }
53
+ },
54
+
55
+ addAttributes() {
56
+ return {
57
+ accept: {
58
+ default: this.options.accept
59
+ }
60
+ }
61
+ },
62
+
63
+ parseHTML() {
64
+ return [{ tag: 'div[data-type="media-gallery-placeholder"]' }]
65
+ },
66
+
67
+ renderHTML({ HTMLAttributes }) {
68
+ return ['div', mergeAttributes({ 'data-type': 'media-gallery-placeholder' }, HTMLAttributes)]
69
+ },
70
+
71
+ addNodeView() {
72
+ return ReactNodeViewRenderer(MediaGalleryPlaceholderNodeComponent)
73
+ },
74
+
75
+ addCommands() {
76
+ return {
77
+ setMediaGalleryPlaceholderNode:
78
+ (options) =>
79
+ ({ commands }) => {
80
+ return commands.insertContent({
81
+ type: this.name,
82
+ attrs: options
83
+ })
84
+ }
85
+ }
86
+ },
87
+
88
+ /**
89
+ * Adds Enter key handler to reopen the media gallery when the placeholder is selected.
90
+ */
91
+ addKeyboardShortcuts() {
92
+ return {
93
+ Enter: ({ editor }) => {
94
+ const { selection } = editor.state
95
+ const { nodeAfter } = selection.$from
96
+
97
+ if (
98
+ nodeAfter &&
99
+ nodeAfter.type.name === 'mediaGalleryPlaceholder' &&
100
+ editor.isActive('mediaGalleryPlaceholder')
101
+ ) {
102
+ const nodeEl = editor.view.nodeDOM(selection.$from.pos)
103
+ if (nodeEl && nodeEl instanceof HTMLElement) {
104
+ const trigger = nodeEl.querySelector('button')
105
+ if (trigger && trigger instanceof HTMLElement) {
106
+ trigger.click()
107
+ return true
108
+ }
109
+ }
110
+ }
111
+ return false
112
+ }
113
+ }
114
+ }
115
+ })
116
+
117
+ export default MediaGalleryPlaceholderNode
@@ -0,0 +1,63 @@
1
+ 'use client'
2
+
3
+ import { ContentEditorMediaGalleryBlock } from '@admin/components/custom/content-editor/media-gallery-block'
4
+ import { getMediaLabel } from '@admin/utils/editor/source-media'
5
+ import { focusNextNode, isValidPosition } from '@admin/utils/editor/tiptap'
6
+ import type { NodeViewProps } from '@tiptap/react'
7
+ import { NodeViewWrapper } from '@tiptap/react'
8
+ import * as React from 'react'
9
+
10
+ export const MediaGalleryPlaceholderNode: React.FC<NodeViewProps> = (props) => {
11
+ const { accept } = props.node.attrs
12
+ const { editor, node } = props
13
+ const extension = props.extension
14
+
15
+ const removePlaceholder = React.useCallback(() => {
16
+ const pos = props.getPos()
17
+ if (!isValidPosition(pos)) return
18
+
19
+ editor
20
+ .chain()
21
+ .focus()
22
+ .deleteRange({ from: pos, to: pos + node.nodeSize })
23
+ .run()
24
+ focusNextNode(editor)
25
+ }, [editor, node.nodeSize, props])
26
+
27
+ const handleMediaSelected = React.useCallback(
28
+ (url: string) => {
29
+ const pos = props.getPos()
30
+ if (!isValidPosition(pos) || !url) return
31
+ const label = getMediaLabel(url)
32
+
33
+ editor
34
+ .chain()
35
+ .focus()
36
+ .deleteRange({ from: pos, to: pos + node.nodeSize })
37
+ .insertContentAt(pos, {
38
+ type: extension.options.type,
39
+ attrs: {
40
+ ...extension.options,
41
+ src: url,
42
+ alt: label,
43
+ title: label
44
+ }
45
+ })
46
+ .run()
47
+
48
+ focusNextNode(editor)
49
+ },
50
+ [editor, extension.options, node.nodeSize, props]
51
+ )
52
+
53
+ return (
54
+ <NodeViewWrapper className="block" tabIndex={0}>
55
+ <ContentEditorMediaGalleryBlock
56
+ accept={accept}
57
+ disabled={!editor.isEditable}
58
+ onMediaSelected={handleMediaSelected}
59
+ onRemove={removePlaceholder}
60
+ />
61
+ </NodeViewWrapper>
62
+ )
63
+ }
@@ -0,0 +1 @@
1
+ export { RemovableImage } from './removable-image-node-extension'
@@ -0,0 +1,11 @@
1
+ import { Image } from '@tiptap/extension-image'
2
+ import { ReactNodeViewRenderer } from '@tiptap/react'
3
+ import { RemovableImageNode } from './removable-image-node'
4
+
5
+ export const RemovableImage = Image.extend({
6
+ addNodeView() {
7
+ return ReactNodeViewRenderer(RemovableImageNode)
8
+ }
9
+ })
10
+
11
+ export default RemovableImage
@@ -0,0 +1,168 @@
1
+ 'use client'
2
+
3
+ import { MediaGalleryDialog } from '@admin/components/shared/media/media-gallery-dialog'
4
+ import { Button } from '@admin/components/ui/button'
5
+ import type { AdminMedia } from '@admin/types'
6
+ import { getNumberAttr, getStringAttr } from '@admin/utils/editor/node-attrs'
7
+ import { getMediaLabel } from '@admin/utils/editor/source-media'
8
+ import { focusNextNode, isValidPosition } from '@admin/utils/editor/tiptap'
9
+ import { cn } from '@admin/utils/shared/cn'
10
+ import type { NodeViewProps } from '@tiptap/react'
11
+ import { NodeViewWrapper } from '@tiptap/react'
12
+ import NextImage from 'next/image'
13
+ import * as React from 'react'
14
+
15
+ export const RemovableImageNode: React.FC<NodeViewProps> = (props) => {
16
+ const { editor, node } = props
17
+ const src = getStringAttr(node.attrs.src)
18
+ const alt = getStringAttr(node.attrs.alt)
19
+ const title = getStringAttr(node.attrs.title)
20
+ const width = getNumberAttr(node.attrs.width)
21
+ const height = getNumberAttr(node.attrs.height)
22
+ const aspectRatio = width && height ? `${width} / ${height}` : '16 / 9'
23
+ const [isDialogOpen, setIsDialogOpen] = React.useState(false)
24
+ const [isControlsVisible, setIsControlsVisible] = React.useState(false)
25
+
26
+ const handleChangeClick = React.useCallback((event: React.MouseEvent<HTMLButtonElement>) => {
27
+ event.preventDefault()
28
+ event.stopPropagation()
29
+ setIsDialogOpen(true)
30
+ }, [])
31
+
32
+ const handleMediaSelected = React.useCallback(
33
+ (selected: AdminMedia | AdminMedia[]) => {
34
+ const item = Array.isArray(selected) ? selected[0] : selected
35
+ const pos = props.getPos()
36
+ if (!item?.url || !isValidPosition(pos)) return
37
+
38
+ const label = item.alt || getMediaLabel(item.url)
39
+ editor.view.dispatch(
40
+ editor.state.tr.setNodeMarkup(pos, undefined, {
41
+ ...node.attrs,
42
+ src: item.url,
43
+ alt: label,
44
+ title: item.filename || label,
45
+ width: item.width ?? null,
46
+ height: item.height ?? null
47
+ })
48
+ )
49
+ editor.commands.focus()
50
+ },
51
+ [editor, node.attrs, props]
52
+ )
53
+
54
+ const handleImageLoad = React.useCallback(
55
+ (event: React.SyntheticEvent<HTMLImageElement>) => {
56
+ if (width && height) return
57
+ const image = event.currentTarget
58
+ if (!image.naturalWidth || !image.naturalHeight) return
59
+ const pos = props.getPos()
60
+ if (!isValidPosition(pos)) return
61
+ const tr = editor.state.tr
62
+ .setNodeMarkup(pos, undefined, {
63
+ ...node.attrs,
64
+ width: image.naturalWidth,
65
+ height: image.naturalHeight
66
+ })
67
+ .setMeta('addToHistory', false)
68
+ editor.view.dispatch(tr)
69
+ },
70
+ [editor, height, node.attrs, props, width]
71
+ )
72
+
73
+ const handleRemove = React.useCallback(
74
+ (event: React.MouseEvent<HTMLButtonElement>) => {
75
+ event.preventDefault()
76
+ event.stopPropagation()
77
+
78
+ const pos = props.getPos()
79
+ if (!isValidPosition(pos)) return
80
+
81
+ editor
82
+ .chain()
83
+ .focus()
84
+ .deleteRange({ from: pos, to: pos + node.nodeSize })
85
+ .run()
86
+ focusNextNode(editor)
87
+ },
88
+ [editor, node.nodeSize, props]
89
+ )
90
+
91
+ return (
92
+ <NodeViewWrapper
93
+ className="not-prose relative my-4 block max-w-full rounded-md"
94
+ contentEditable={false}
95
+ data-content-editor-image-node=""
96
+ onMouseEnter={() => setIsControlsVisible(true)}
97
+ onMouseMove={() => setIsControlsVisible(true)}
98
+ onMouseOver={() => setIsControlsVisible(true)}
99
+ onMouseLeave={() => setIsControlsVisible(false)}
100
+ onFocusCapture={() => setIsControlsVisible(true)}
101
+ onBlurCapture={(event: React.FocusEvent<HTMLDivElement>) => {
102
+ if (!event.currentTarget.contains(event.relatedTarget as Node | null)) {
103
+ setIsControlsVisible(false)
104
+ }
105
+ }}
106
+ tabIndex={0}
107
+ style={{ width: width ? `${width}px` : '100%' }}
108
+ >
109
+ <div
110
+ className="relative max-w-full overflow-hidden rounded-md bg-muted/20"
111
+ style={{ aspectRatio }}
112
+ >
113
+ {src && (
114
+ <NextImage
115
+ src={src}
116
+ alt={alt}
117
+ title={title || undefined}
118
+ fill
119
+ sizes="(min-width: 1024px) 768px, 100vw"
120
+ unoptimized
121
+ className="!m-0 !h-full !w-full !max-w-full object-contain"
122
+ onLoad={handleImageLoad}
123
+ />
124
+ )}
125
+ {editor.isEditable && (
126
+ <div
127
+ data-content-editor-image-controls-overlay=""
128
+ className={cn(
129
+ 'pointer-events-none rounded-md absolute inset-0 flex items-center justify-center gap-1 bg-black/10 backdrop-blur-xs transition-opacity duration-200 ease-out',
130
+ {
131
+ 'opacity-100': isControlsVisible,
132
+ 'opacity-0': !isControlsVisible
133
+ }
134
+ )}
135
+ >
136
+ <Button
137
+ type="button"
138
+ variant="outline"
139
+ onMouseDown={(event) => event.preventDefault()}
140
+ onClick={handleChangeClick}
141
+ className="pointer-events-auto border-primary/10"
142
+ >
143
+ Change
144
+ </Button>
145
+ <Button
146
+ type="button"
147
+ variant="default"
148
+ onMouseDown={(event) => event.preventDefault()}
149
+ onClick={handleRemove}
150
+ className="pointer-events-auto"
151
+ >
152
+ Remove
153
+ </Button>
154
+ </div>
155
+ )}
156
+ </div>
157
+ {editor.isEditable && (
158
+ <MediaGalleryDialog
159
+ open={isDialogOpen}
160
+ onOpenChange={setIsDialogOpen}
161
+ onSelect={handleMediaSelected}
162
+ mode="single"
163
+ accept="image/*"
164
+ />
165
+ )}
166
+ </NodeViewWrapper>
167
+ )
168
+ }