create-nextblock 0.2.78 → 0.8.0

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 (413) hide show
  1. package/bin/create-nextblock.js +740 -459
  2. package/package.json +1 -2
  3. package/scripts/sync-template.js +18 -1
  4. package/templates/nextblock-template/.browserslistrc +11 -0
  5. package/templates/nextblock-template/.swcrc +30 -30
  6. package/templates/nextblock-template/README.md +23 -114
  7. package/templates/nextblock-template/app/(auth-pages)/post-sign-in/page.tsx +27 -28
  8. package/templates/nextblock-template/app/(auth-pages)/sign-in/page.tsx +50 -25
  9. package/templates/nextblock-template/app/(auth-pages)/sign-up/page.tsx +111 -56
  10. package/templates/nextblock-template/app/(auth-pages)/two-factor/actions.ts +91 -0
  11. package/templates/nextblock-template/app/(auth-pages)/two-factor/components/TwoFactorForm.tsx +118 -0
  12. package/templates/nextblock-template/app/(auth-pages)/two-factor/page.tsx +51 -0
  13. package/templates/nextblock-template/app/.well-known/ucp/route.ts +16 -0
  14. package/templates/nextblock-template/app/[slug]/PageClientContent.tsx +48 -28
  15. package/templates/nextblock-template/app/[slug]/page.tsx +63 -6
  16. package/templates/nextblock-template/app/[slug]/page.utils.ts +374 -157
  17. package/templates/nextblock-template/app/[slug]/pageClientActions.ts +7 -0
  18. package/templates/nextblock-template/app/actions/consent.ts +57 -0
  19. package/templates/nextblock-template/app/actions/formActions.ts +130 -11
  20. package/templates/nextblock-template/app/actions/languageActions.ts +31 -30
  21. package/templates/nextblock-template/app/actions/package-actions.ts +183 -0
  22. package/templates/nextblock-template/app/actions/postActions.ts +146 -48
  23. package/templates/nextblock-template/app/actions/twoFactorEmail.ts +21 -0
  24. package/templates/nextblock-template/app/actions/visualEditingActions.test.ts +179 -0
  25. package/templates/nextblock-template/app/actions/visualEditingActions.ts +345 -0
  26. package/templates/nextblock-template/app/actions.ts +67 -12
  27. package/templates/nextblock-template/app/api/ai/cortex/build-widget/route.ts +153 -0
  28. package/templates/nextblock-template/app/api/ai/generate-blocks/route.ts +96 -0
  29. package/templates/nextblock-template/app/api/ai/global-agent/route.ts +965 -0
  30. package/templates/nextblock-template/app/api/checkout/freemius/sync/route.ts +29 -0
  31. package/templates/nextblock-template/app/api/checkout/route.ts +146 -0
  32. package/templates/nextblock-template/app/api/cms/full-backup/export/route.ts +33 -0
  33. package/templates/nextblock-template/app/api/cms/full-backup/restore/route.ts +63 -0
  34. package/templates/nextblock-template/app/api/cron/reset-sandbox/route.ts +3413 -17
  35. package/templates/nextblock-template/app/api/cron/reset-sandbox/sandboxResetSql.ts +7830 -0
  36. package/templates/nextblock-template/app/api/cron/sync-currencies/route.ts +35 -0
  37. package/templates/nextblock-template/app/api/custom-blocks/db-relations/route.ts +92 -0
  38. package/templates/nextblock-template/app/api/custom-blocks/editor-definitions/route.ts +43 -0
  39. package/templates/nextblock-template/app/api/draft/disable/route.ts +25 -0
  40. package/templates/nextblock-template/app/api/draft/route.ts +93 -0
  41. package/templates/nextblock-template/app/api/draft/start/route.ts +77 -0
  42. package/templates/nextblock-template/app/api/media/library/route.ts +65 -0
  43. package/templates/nextblock-template/app/api/media/r2-presigned/route.ts +53 -0
  44. package/templates/nextblock-template/app/api/media/record/route.ts +160 -0
  45. package/templates/nextblock-template/app/api/search/route.ts +43 -0
  46. package/templates/nextblock-template/app/api/visual-editing/block-draft/route.ts +47 -0
  47. package/templates/nextblock-template/app/api/visual-editing/product-draft/route.ts +47 -0
  48. package/templates/nextblock-template/app/api/webhooks/freemius/route.ts +34 -0
  49. package/templates/nextblock-template/app/api/webhooks/stripe/route.ts +27 -0
  50. package/templates/nextblock-template/app/article/[slug]/PostClientContent.tsx +392 -128
  51. package/templates/nextblock-template/app/article/[slug]/page.tsx +179 -127
  52. package/templates/nextblock-template/app/article/[slug]/page.utils.ts +262 -77
  53. package/templates/nextblock-template/app/auth/callback/route.ts +31 -58
  54. package/templates/nextblock-template/app/cart/page.tsx +7 -0
  55. package/templates/nextblock-template/app/checkout/UcpCartHydrator.tsx +20 -0
  56. package/templates/nextblock-template/app/checkout/page.tsx +52 -0
  57. package/templates/nextblock-template/app/checkout/success/actions.ts +136 -0
  58. package/templates/nextblock-template/app/checkout/success/page.tsx +186 -0
  59. package/templates/nextblock-template/app/cms/CmsClientLayout.tsx +163 -33
  60. package/templates/nextblock-template/app/cms/blocks/actions.ts +424 -235
  61. package/templates/nextblock-template/app/cms/blocks/components/BackgroundSelector.tsx +212 -151
  62. package/templates/nextblock-template/app/cms/blocks/components/BlockEditorArea.tsx +41 -20
  63. package/templates/nextblock-template/app/cms/blocks/components/BlockEditorModal.tsx +152 -19
  64. package/templates/nextblock-template/app/cms/blocks/components/BlockTypeCard.tsx +25 -17
  65. package/templates/nextblock-template/app/cms/blocks/components/BlockTypeSelector.tsx +200 -18
  66. package/templates/nextblock-template/app/cms/blocks/components/ColumnEditor.tsx +33 -16
  67. package/templates/nextblock-template/app/cms/blocks/components/CustomBlockEditorPreview.tsx +160 -0
  68. package/templates/nextblock-template/app/cms/blocks/components/EditableBlock.tsx +37 -18
  69. package/templates/nextblock-template/app/cms/blocks/components/MediaLibraryModal.tsx +149 -67
  70. package/templates/nextblock-template/app/cms/blocks/components/SectionConfigPanel.tsx +108 -31
  71. package/templates/nextblock-template/app/cms/blocks/editors/DynamicCustomBlockEditor.tsx +167 -0
  72. package/templates/nextblock-template/app/cms/blocks/editors/FeaturedProductBlockEditor.tsx +31 -0
  73. package/templates/nextblock-template/app/cms/blocks/editors/FormBlockEditor.tsx +2 -2
  74. package/templates/nextblock-template/app/cms/blocks/editors/HeadingBlockEditor.tsx +1 -1
  75. package/templates/nextblock-template/app/cms/blocks/editors/ImageBlockEditor.tsx +29 -29
  76. package/templates/nextblock-template/app/cms/blocks/editors/PostsGridBlockEditor.tsx +14 -18
  77. package/templates/nextblock-template/app/cms/blocks/editors/ProductGridBlockEditor.tsx +41 -0
  78. package/templates/nextblock-template/app/cms/blocks/editors/SectionBlockEditor.tsx +318 -118
  79. package/templates/nextblock-template/app/cms/blocks/editors/TextBlockEditor.tsx +98 -21
  80. package/templates/nextblock-template/app/cms/blocks/editors/VideoEmbedBlockEditor.tsx +1 -1
  81. package/templates/nextblock-template/app/cms/components/ContentLanguageSwitcher.tsx +27 -9
  82. package/templates/nextblock-template/app/cms/components/CopyContentFromLanguage.tsx +1 -1
  83. package/templates/nextblock-template/app/cms/components/CortexAiActiveContext.tsx +23 -0
  84. package/templates/nextblock-template/app/cms/components/CortexAiPageContext.tsx +58 -0
  85. package/templates/nextblock-template/app/cms/components/CortexGlobalAgentChat.tsx +1507 -0
  86. package/templates/nextblock-template/app/cms/components/DraftStatusActions.tsx +145 -0
  87. package/templates/nextblock-template/app/cms/components/FeatureImageField.tsx +244 -0
  88. package/templates/nextblock-template/app/cms/components/FeedbackModal.tsx +38 -24
  89. package/templates/nextblock-template/app/cms/coupons/[id]/edit/page.tsx +16 -0
  90. package/templates/nextblock-template/app/cms/coupons/page.tsx +16 -0
  91. package/templates/nextblock-template/app/cms/custom-blocks/[id]/edit/page.tsx +66 -0
  92. package/templates/nextblock-template/app/cms/custom-blocks/actions.ts +519 -0
  93. package/templates/nextblock-template/app/cms/custom-blocks/components/BlockComposer.tsx +1522 -0
  94. package/templates/nextblock-template/app/cms/custom-blocks/components/BlocksLibraryTransferControls.tsx +256 -0
  95. package/templates/nextblock-template/app/cms/custom-blocks/components/DBRelationSelect.tsx +384 -0
  96. package/templates/nextblock-template/app/cms/custom-blocks/components/ImageR2Picker.tsx +221 -0
  97. package/templates/nextblock-template/app/cms/custom-blocks/new/page.tsx +12 -0
  98. package/templates/nextblock-template/app/cms/custom-blocks/page.tsx +438 -0
  99. package/templates/nextblock-template/app/cms/dashboard/actions.ts +228 -98
  100. package/templates/nextblock-template/app/cms/dashboard/components/DashboardComponents.tsx +200 -0
  101. package/templates/nextblock-template/app/cms/dashboard/page.tsx +182 -154
  102. package/templates/nextblock-template/app/cms/import-export/ContentTransferControls.tsx +391 -0
  103. package/templates/nextblock-template/app/cms/import-export/actions.ts +226 -0
  104. package/templates/nextblock-template/app/cms/layout.tsx +29 -10
  105. package/templates/nextblock-template/app/cms/media/UploadFolderContext.tsx +22 -22
  106. package/templates/nextblock-template/app/cms/media/actions.ts +45 -124
  107. package/templates/nextblock-template/app/cms/media/components/DeleteMediaButtonClient.tsx +1 -1
  108. package/templates/nextblock-template/app/cms/media/components/MediaEditForm.tsx +26 -26
  109. package/templates/nextblock-template/app/cms/media/components/MediaGridClient.tsx +69 -64
  110. package/templates/nextblock-template/app/cms/media/components/MediaPickerDialog.tsx +227 -158
  111. package/templates/nextblock-template/app/cms/media/components/MediaUploadForm.tsx +101 -89
  112. package/templates/nextblock-template/app/cms/media/page.tsx +1 -1
  113. package/templates/nextblock-template/app/cms/navigation/components/NavigationItemForm.tsx +2 -2
  114. package/templates/nextblock-template/app/cms/orders/[id]/MarkPaidButton.tsx +44 -0
  115. package/templates/nextblock-template/app/cms/orders/[id]/page.tsx +16 -0
  116. package/templates/nextblock-template/app/cms/orders/actions.ts +201 -0
  117. package/templates/nextblock-template/app/cms/orders/page.tsx +20 -0
  118. package/templates/nextblock-template/app/cms/orders/types.ts +20 -0
  119. package/templates/nextblock-template/app/cms/pages/[id]/edit/EditPageClient.tsx +156 -121
  120. package/templates/nextblock-template/app/cms/pages/[id]/edit/page.tsx +79 -26
  121. package/templates/nextblock-template/app/cms/pages/actions.ts +54 -38
  122. package/templates/nextblock-template/app/cms/pages/components/DeletePageButtonClient.tsx +1 -1
  123. package/templates/nextblock-template/app/cms/pages/components/PageForm.tsx +267 -116
  124. package/templates/nextblock-template/app/cms/pages/page.tsx +25 -18
  125. package/templates/nextblock-template/app/cms/payments/page.tsx +16 -0
  126. package/templates/nextblock-template/app/cms/posts/[id]/edit/page.tsx +132 -90
  127. package/templates/nextblock-template/app/cms/posts/actions.ts +71 -72
  128. package/templates/nextblock-template/app/cms/posts/components/DeletePostButtonClient.tsx +1 -1
  129. package/templates/nextblock-template/app/cms/posts/components/PostForm.tsx +256 -245
  130. package/templates/nextblock-template/app/cms/posts/new/page.tsx +1 -1
  131. package/templates/nextblock-template/app/cms/posts/page.tsx +20 -13
  132. package/templates/nextblock-template/app/cms/products/ClientNotionEditor.tsx +16 -0
  133. package/templates/nextblock-template/app/cms/products/ProductFormClientShell.tsx +56 -0
  134. package/templates/nextblock-template/app/cms/products/[id]/edit/page.tsx +292 -0
  135. package/templates/nextblock-template/app/cms/products/attributes/page.tsx +12 -0
  136. package/templates/nextblock-template/app/cms/products/categories/page.tsx +12 -0
  137. package/templates/nextblock-template/app/cms/products/inventory/page.tsx +13 -0
  138. package/templates/nextblock-template/app/cms/products/new/page.tsx +143 -0
  139. package/templates/nextblock-template/app/cms/products/page.tsx +42 -0
  140. package/templates/nextblock-template/app/cms/products/productFormData.ts +133 -0
  141. package/templates/nextblock-template/app/cms/products/settings/page.tsx +5 -0
  142. package/templates/nextblock-template/app/cms/promotions/PromotionsWorkspace.tsx +456 -0
  143. package/templates/nextblock-template/app/cms/promotions/actions.ts +115 -0
  144. package/templates/nextblock-template/app/cms/promotions/page.tsx +31 -0
  145. package/templates/nextblock-template/app/cms/revisions/RevisionHistoryButton.tsx +2 -2
  146. package/templates/nextblock-template/app/cms/revisions/actions.ts +285 -285
  147. package/templates/nextblock-template/app/cms/revisions/service.ts +19 -16
  148. package/templates/nextblock-template/app/cms/revisions/utils.ts +8 -3
  149. package/templates/nextblock-template/app/cms/settings/backup-restore/BackupRestoreWorkspace.tsx +1004 -0
  150. package/templates/nextblock-template/app/cms/settings/backup-restore/page.tsx +29 -0
  151. package/templates/nextblock-template/app/cms/settings/bot-protection/actions.ts +93 -0
  152. package/templates/nextblock-template/app/cms/settings/bot-protection/components/BotProtectionForm.tsx +129 -0
  153. package/templates/nextblock-template/app/cms/settings/bot-protection/page.tsx +24 -0
  154. package/templates/nextblock-template/app/cms/settings/copyright/actions.ts +1 -1
  155. package/templates/nextblock-template/app/cms/settings/copyright/components/CopyrightForm.tsx +2 -2
  156. package/templates/nextblock-template/app/cms/settings/copyright/page.tsx +1 -1
  157. package/templates/nextblock-template/app/cms/settings/cortex-ai/SandboxCortexAiSettingsClient.tsx +496 -0
  158. package/templates/nextblock-template/app/cms/settings/cortex-ai/StoredCortexAiSettingsClient.tsx +410 -0
  159. package/templates/nextblock-template/app/cms/settings/cortex-ai/actions.ts +248 -0
  160. package/templates/nextblock-template/app/cms/settings/cortex-ai/page.tsx +80 -0
  161. package/templates/nextblock-template/app/cms/settings/currencies/actions.ts +331 -0
  162. package/templates/nextblock-template/app/cms/settings/currencies/page.tsx +494 -0
  163. package/templates/nextblock-template/app/cms/settings/extra-translations/ExtraTranslationsWorkspace.tsx +767 -0
  164. package/templates/nextblock-template/app/cms/settings/extra-translations/actions.ts +203 -44
  165. package/templates/nextblock-template/app/cms/settings/extra-translations/page.tsx +93 -242
  166. package/templates/nextblock-template/app/cms/settings/global-css/actions.ts +65 -0
  167. package/templates/nextblock-template/app/cms/settings/global-css/components/GlobalCssForm.tsx +46 -0
  168. package/templates/nextblock-template/app/cms/settings/global-css/page.tsx +24 -0
  169. package/templates/nextblock-template/app/cms/settings/languages/components/DeleteLanguageButton.tsx +1 -1
  170. package/templates/nextblock-template/app/cms/settings/languages/components/LanguageForm.tsx +2 -2
  171. package/templates/nextblock-template/app/cms/settings/languages/page.tsx +1 -1
  172. package/templates/nextblock-template/app/cms/settings/logos/[id]/edit/page.tsx +7 -7
  173. package/templates/nextblock-template/app/cms/settings/logos/actions.ts +82 -6
  174. package/templates/nextblock-template/app/cms/settings/logos/components/BrandingSettingsForm.tsx +339 -0
  175. package/templates/nextblock-template/app/cms/settings/logos/components/DeleteLogoButton.tsx +21 -18
  176. package/templates/nextblock-template/app/cms/settings/logos/components/LogoForm.tsx +20 -16
  177. package/templates/nextblock-template/app/cms/settings/logos/components/SiteSeoSettingsForm.tsx +133 -0
  178. package/templates/nextblock-template/app/cms/settings/logos/new/page.tsx +8 -8
  179. package/templates/nextblock-template/app/cms/settings/logos/page.tsx +120 -82
  180. package/templates/nextblock-template/app/cms/settings/logos/types.ts +8 -8
  181. package/templates/nextblock-template/app/cms/settings/packages/activation-form.tsx +84 -0
  182. package/templates/nextblock-template/app/cms/settings/packages/package-card.tsx +122 -0
  183. package/templates/nextblock-template/app/cms/settings/packages/page.tsx +49 -0
  184. package/templates/nextblock-template/app/cms/settings/privacy/actions.ts +53 -0
  185. package/templates/nextblock-template/app/cms/settings/privacy/components/PrivacyForm.tsx +196 -0
  186. package/templates/nextblock-template/app/cms/settings/privacy/page.tsx +26 -0
  187. package/templates/nextblock-template/app/cms/settings/security/actions.ts +251 -0
  188. package/templates/nextblock-template/app/cms/settings/security/components/SecurityPanel.tsx +453 -0
  189. package/templates/nextblock-template/app/cms/settings/security/page.tsx +13 -0
  190. package/templates/nextblock-template/app/cms/settings/taxes/page.tsx +21 -0
  191. package/templates/nextblock-template/app/cms/shipping/page.tsx +20 -0
  192. package/templates/nextblock-template/app/cms/users/[id]/edit/page.tsx +28 -23
  193. package/templates/nextblock-template/app/cms/users/actions.ts +105 -40
  194. package/templates/nextblock-template/app/cms/users/components/DeleteUserButton.tsx +1 -1
  195. package/templates/nextblock-template/app/cms/users/components/UserForm.tsx +65 -152
  196. package/templates/nextblock-template/app/cms/users/page.tsx +15 -10
  197. package/templates/nextblock-template/app/globals.css +9 -0
  198. package/templates/nextblock-template/app/layout.tsx +372 -120
  199. package/templates/nextblock-template/app/lib/seo.test.ts +52 -0
  200. package/templates/nextblock-template/app/lib/seo.ts +279 -0
  201. package/templates/nextblock-template/app/lib/site-settings.ts +87 -0
  202. package/templates/nextblock-template/app/lib/sitemap-utils.ts +224 -39
  203. package/templates/nextblock-template/app/lib/ucp/protocol.ts +190 -0
  204. package/templates/nextblock-template/app/lib/ucp/server.test.ts +56 -0
  205. package/templates/nextblock-template/app/lib/ucp/server.ts +1914 -0
  206. package/templates/nextblock-template/app/page.tsx +165 -73
  207. package/templates/nextblock-template/app/product/[slug]/page.tsx +433 -0
  208. package/templates/nextblock-template/app/profile/ProfileAccountSidebar.tsx +73 -0
  209. package/templates/nextblock-template/app/profile/ProfilePageHeader.tsx +16 -0
  210. package/templates/nextblock-template/app/profile/ProfilePageMissingState.tsx +9 -0
  211. package/templates/nextblock-template/app/profile/account-data.ts +37 -0
  212. package/templates/nextblock-template/app/profile/account-links.ts +22 -0
  213. package/templates/nextblock-template/app/profile/account-types.ts +11 -0
  214. package/templates/nextblock-template/app/profile/orders/CustomerOrdersPageClient.tsx +124 -0
  215. package/templates/nextblock-template/app/profile/orders/[id]/CustomerOrderDetailPageClient.tsx +79 -0
  216. package/templates/nextblock-template/app/profile/orders/[id]/page.tsx +32 -0
  217. package/templates/nextblock-template/app/profile/orders/page.tsx +19 -0
  218. package/templates/nextblock-template/app/profile/page.tsx +51 -0
  219. package/templates/nextblock-template/app/profile/password/PasswordSettingsPageClient.tsx +128 -0
  220. package/templates/nextblock-template/app/profile/password/actions.ts +59 -0
  221. package/templates/nextblock-template/app/profile/password/page.tsx +27 -0
  222. package/templates/nextblock-template/app/providers.tsx +55 -17
  223. package/templates/nextblock-template/app/robots.txt/route.ts +11 -1
  224. package/templates/nextblock-template/app/sitemap.ts +128 -0
  225. package/templates/nextblock-template/app/ucp/v1/carts/[id]/cancel/route.ts +38 -0
  226. package/templates/nextblock-template/app/ucp/v1/carts/[id]/route.ts +68 -0
  227. package/templates/nextblock-template/app/ucp/v1/carts/route.ts +35 -0
  228. package/templates/nextblock-template/app/ucp/v1/catalog/lookup/route.ts +35 -0
  229. package/templates/nextblock-template/app/ucp/v1/catalog/product/route.ts +35 -0
  230. package/templates/nextblock-template/app/ucp/v1/catalog/search/route.ts +34 -0
  231. package/templates/nextblock-template/components/AppShell.tsx +154 -0
  232. package/templates/nextblock-template/components/BlockRenderer.tsx +210 -64
  233. package/templates/nextblock-template/components/CartDrawerLoader.tsx +7 -0
  234. package/templates/nextblock-template/components/CartTranslator.tsx +210 -0
  235. package/templates/nextblock-template/components/CurrentContentSetter.tsx +25 -0
  236. package/templates/nextblock-template/components/DeferredCartDrawer.tsx +23 -0
  237. package/templates/nextblock-template/components/DeferredCartTranslator.tsx +51 -0
  238. package/templates/nextblock-template/components/DeferredGlobalSearch.tsx +68 -0
  239. package/templates/nextblock-template/components/DeferredGoogleTagManager.tsx +70 -0
  240. package/templates/nextblock-template/components/DeferredSpeedInsights.tsx +69 -0
  241. package/templates/nextblock-template/components/FeatureImageHero.tsx +47 -0
  242. package/templates/nextblock-template/components/GitHubLoginButton.tsx +36 -0
  243. package/templates/nextblock-template/components/GlobalSearch.tsx +557 -0
  244. package/templates/nextblock-template/components/Header.tsx +49 -41
  245. package/templates/nextblock-template/components/LanguageSwitcher.tsx +55 -32
  246. package/templates/nextblock-template/components/ResponsiveNav.tsx +138 -43
  247. package/templates/nextblock-template/components/blocks/PostCardSkeleton.tsx +12 -8
  248. package/templates/nextblock-template/components/blocks/PostsGridBlock.tsx +12 -55
  249. package/templates/nextblock-template/components/blocks/PostsGridClient.tsx +42 -37
  250. package/templates/nextblock-template/components/blocks/TestimonialBlock.tsx +6 -2
  251. package/templates/nextblock-template/components/blocks/ecommerceRendererLoaders.ts +23 -0
  252. package/templates/nextblock-template/components/blocks/publicRendererLoaders.ts +25 -0
  253. package/templates/nextblock-template/components/blocks/renderers/ButtonBlockRenderer.tsx +92 -84
  254. package/templates/nextblock-template/components/blocks/renderers/CartBlockRenderer.tsx +17 -0
  255. package/templates/nextblock-template/components/blocks/renderers/CheckoutBlockRenderer.tsx +19 -0
  256. package/templates/nextblock-template/components/blocks/renderers/ClientTextBlockRenderer.tsx +262 -8
  257. package/templates/nextblock-template/components/blocks/renderers/FeaturedProductBlockRenderer.tsx +22 -0
  258. package/templates/nextblock-template/components/blocks/renderers/FormBlockRenderer.tsx +320 -37
  259. package/templates/nextblock-template/components/blocks/renderers/HeadingBlockRenderer.tsx +11 -8
  260. package/templates/nextblock-template/components/blocks/renderers/ImageBlockRenderer.tsx +12 -3
  261. package/templates/nextblock-template/components/blocks/renderers/PostsGridBlockRenderer.tsx +18 -13
  262. package/templates/nextblock-template/components/blocks/renderers/ProductDetailsBlockRenderer.tsx +90 -0
  263. package/templates/nextblock-template/components/blocks/renderers/ProductGridBlockRenderer.tsx +31 -0
  264. package/templates/nextblock-template/components/blocks/renderers/SectionBlockRenderer.tsx +424 -55
  265. package/templates/nextblock-template/components/blocks/renderers/SectionSlider.tsx +137 -0
  266. package/templates/nextblock-template/components/blocks/renderers/TestimonialBlockRenderer.tsx +57 -0
  267. package/templates/nextblock-template/components/blocks/renderers/TextBlockRenderer.tsx +37 -22
  268. package/templates/nextblock-template/components/blocks/renderers/VideoEmbedBlockRenderer.tsx +23 -15
  269. package/templates/nextblock-template/components/blocks/renderers/inline/AlertWidgetRenderer.tsx +1 -3
  270. package/templates/nextblock-template/components/blocks/renderers/inline/CtaWidgetRenderer.tsx +1 -3
  271. package/templates/nextblock-template/components/blocks/types.ts +7 -6
  272. package/templates/nextblock-template/components/env-var-warning.tsx +3 -3
  273. package/templates/nextblock-template/components/form-message.tsx +32 -26
  274. package/templates/nextblock-template/components/header-auth.tsx +69 -17
  275. package/templates/nextblock-template/components/privacy/ConsentBanner.tsx +127 -0
  276. package/templates/nextblock-template/components/privacy/ConsentGatedAnalytics.tsx +59 -0
  277. package/templates/nextblock-template/components/renderers/CachedDynamicLayoutEngine.tsx +28 -0
  278. package/templates/nextblock-template/components/renderers/DynamicLayoutEngine.test.tsx +166 -0
  279. package/templates/nextblock-template/components/renderers/DynamicLayoutEngine.tsx +464 -0
  280. package/templates/nextblock-template/components/theme-switcher.tsx +8 -8
  281. package/templates/nextblock-template/components/visual-editing/DeferredVisualEditing.tsx +21 -0
  282. package/templates/nextblock-template/components/visual-editing/NextblockVisualEditing.tsx +1172 -0
  283. package/templates/nextblock-template/context/AuthContext.tsx +23 -90
  284. package/templates/nextblock-template/context/CurrentContentContext.tsx +10 -4
  285. package/templates/nextblock-template/context/LanguageContext.tsx +16 -16
  286. package/templates/nextblock-template/context/language-rest-client.ts +31 -0
  287. package/templates/nextblock-template/docs/01-PROJECT-OVERVIEW.md +94 -0
  288. package/templates/nextblock-template/docs/02-ECOMMERCE-CAPABILITIES.md +364 -0
  289. package/templates/nextblock-template/docs/03-CMS-AND-EDITOR.md +202 -0
  290. package/templates/nextblock-template/docs/04-DATABASE-AND-AUTH.md +252 -0
  291. package/templates/nextblock-template/docs/05-DEVELOPER-GUIDE.md +238 -0
  292. package/templates/nextblock-template/docs/06-CLI-AND-SCAFFOLDING.md +125 -0
  293. package/templates/nextblock-template/docs/07-BLOCK-SDK-AND-EXTENSIBILITY.md +146 -0
  294. package/templates/nextblock-template/docs/08-NEXTBLOCK-CORTEX-AI-ARCHITECTURE.md +1319 -0
  295. package/templates/nextblock-template/docs/09-LIVE-DRAFT-MODE.md +104 -0
  296. package/templates/nextblock-template/docs/10-CUSTOM-BLOCKS.md +222 -0
  297. package/templates/nextblock-template/docs/README.md +34 -0
  298. package/templates/nextblock-template/docs/TECHNICAL_SPECIFICATION.md +12507 -0
  299. package/templates/nextblock-template/hooks/use-hotkeys.ts +21 -14
  300. package/templates/nextblock-template/hooks/useGlobalSearch.ts +101 -0
  301. package/templates/nextblock-template/index.d.ts +2 -0
  302. package/templates/nextblock-template/lib/ai-block-generation.ts +339 -0
  303. package/templates/nextblock-template/lib/ai-client.ts +247 -0
  304. package/templates/nextblock-template/lib/ai-config.ts +81 -0
  305. package/templates/nextblock-template/lib/ai-cortex-widget-builder.ts +125 -0
  306. package/templates/nextblock-template/lib/ai-global-agent-custom-block-tools.ts +363 -0
  307. package/templates/nextblock-template/lib/ai-global-agent-db-tools.test.ts +405 -0
  308. package/templates/nextblock-template/lib/ai-global-agent-db-tools.ts +1228 -0
  309. package/templates/nextblock-template/lib/ai-global-agent-ecommerce.ts +5 -0
  310. package/templates/nextblock-template/lib/ai-global-agent-tools-stats.test.ts +223 -0
  311. package/templates/nextblock-template/lib/ai-global-agent-tools.test.ts +2183 -0
  312. package/templates/nextblock-template/lib/ai-global-agent-tools.ts +4807 -0
  313. package/templates/nextblock-template/lib/ai-key-crypto.test.ts +70 -0
  314. package/templates/nextblock-template/lib/ai-key-crypto.ts +132 -0
  315. package/templates/nextblock-template/lib/ai-model-catalog.test.ts +49 -0
  316. package/templates/nextblock-template/lib/ai-model-catalog.ts +41 -0
  317. package/templates/nextblock-template/lib/ai-model-registry.test.ts +231 -0
  318. package/templates/nextblock-template/lib/ai-model-registry.ts +522 -0
  319. package/templates/nextblock-template/lib/auth/cookies.ts +47 -0
  320. package/templates/nextblock-template/lib/auth/crypto.ts +42 -0
  321. package/templates/nextblock-template/lib/auth/trustedDevices.ts +92 -0
  322. package/templates/nextblock-template/lib/auth/twoFactor.ts +167 -0
  323. package/templates/nextblock-template/lib/auth-redirects.ts +46 -0
  324. package/templates/nextblock-template/lib/blocks/FeaturedProductBlock.tsx +94 -0
  325. package/templates/nextblock-template/lib/blocks/ProductGridBlock.tsx +137 -0
  326. package/templates/nextblock-template/lib/blocks/README.md +13 -670
  327. package/templates/nextblock-template/lib/blocks/blockRegistry.ts +138 -56
  328. package/templates/nextblock-template/lib/blocks/blockTypes.ts +18 -0
  329. package/templates/nextblock-template/lib/blocks/ecommerce-block-schemas.ts +31 -0
  330. package/templates/nextblock-template/lib/cms-transfer/csv.test.ts +77 -0
  331. package/templates/nextblock-template/lib/cms-transfer/csv.ts +399 -0
  332. package/templates/nextblock-template/lib/cms-transfer/server.ts +2243 -0
  333. package/templates/nextblock-template/lib/cms-transfer/types.ts +145 -0
  334. package/templates/nextblock-template/lib/cortex-widget-registry.test.ts +199 -0
  335. package/templates/nextblock-template/lib/cortex-widget-registry.ts +88 -0
  336. package/templates/nextblock-template/lib/cortex-widget-schema.test.tsx +237 -0
  337. package/templates/nextblock-template/lib/cortex-widget-schema.ts +393 -0
  338. package/templates/nextblock-template/lib/custom-block-definitions.ts +87 -0
  339. package/templates/nextblock-template/lib/custom-block-r2-upload-shared.ts +178 -0
  340. package/templates/nextblock-template/lib/custom-block-r2-upload.test.ts +140 -0
  341. package/templates/nextblock-template/lib/custom-block-r2-upload.ts +68 -0
  342. package/templates/nextblock-template/lib/custom-block-relation-registry.ts +256 -0
  343. package/templates/nextblock-template/lib/custom-block-relations.test.ts +227 -0
  344. package/templates/nextblock-template/lib/custom-block-relations.ts +279 -0
  345. package/templates/nextblock-template/lib/custom-block-safelist.ts +14 -0
  346. package/templates/nextblock-template/lib/editor/dynamic-extension-core.test.ts +172 -0
  347. package/templates/nextblock-template/lib/editor/dynamic-extension-core.ts +213 -0
  348. package/templates/nextblock-template/lib/editor/dynamic-extension-loader.ts +22 -0
  349. package/templates/nextblock-template/lib/editor/dynamic-extensions.tsx +193 -0
  350. package/templates/nextblock-template/lib/full-backup/manifest.test.ts +121 -0
  351. package/templates/nextblock-template/lib/full-backup/manifest.ts +206 -0
  352. package/templates/nextblock-template/lib/full-backup/server.ts +743 -0
  353. package/templates/nextblock-template/lib/media/resolveMediaUrl.ts +45 -0
  354. package/templates/nextblock-template/lib/posts/readTime.ts +60 -0
  355. package/templates/nextblock-template/lib/privacy/consent-client.ts +57 -0
  356. package/templates/nextblock-template/lib/privacy/settings.ts +103 -0
  357. package/templates/nextblock-template/lib/privacy/types.ts +67 -0
  358. package/templates/nextblock-template/lib/promotions/server.test.ts +74 -0
  359. package/templates/nextblock-template/lib/promotions/server.ts +741 -0
  360. package/templates/nextblock-template/lib/resolve-block-relations.test.ts +142 -0
  361. package/templates/nextblock-template/lib/resolve-block-relations.ts +255 -0
  362. package/templates/nextblock-template/lib/search/server.ts +585 -0
  363. package/templates/nextblock-template/lib/search/types.ts +27 -0
  364. package/templates/nextblock-template/lib/visual-editing/draft-content.test.ts +105 -0
  365. package/templates/nextblock-template/lib/visual-editing/draft-content.ts +380 -0
  366. package/templates/nextblock-template/lib/visual-editing/draft-route.test.ts +42 -0
  367. package/templates/nextblock-template/lib/visual-editing/draft-route.ts +82 -0
  368. package/templates/nextblock-template/lib/visual-editing/edit-info.test.ts +143 -0
  369. package/templates/nextblock-template/lib/visual-editing/edit-info.ts +94 -0
  370. package/templates/nextblock-template/lib/visual-editing/mutations.ts +190 -0
  371. package/templates/nextblock-template/lib/visual-editing/product-drafts.test.ts +81 -0
  372. package/templates/nextblock-template/lib/visual-editing/product-drafts.ts +511 -0
  373. package/templates/nextblock-template/lib/visual-editing/types.ts +122 -0
  374. package/templates/nextblock-template/lib/zod-config.ts +5 -0
  375. package/templates/nextblock-template/next.config.js +190 -66
  376. package/templates/nextblock-template/package.json +34 -30
  377. package/templates/nextblock-template/proxy.ts +435 -253
  378. package/templates/nextblock-template/public/images/NBcover.webp +0 -0
  379. package/templates/nextblock-template/public/images/cap.webp +0 -0
  380. package/templates/nextblock-template/public/images/commerce-plan.webp +0 -0
  381. package/templates/nextblock-template/public/images/commerce-square.webp +0 -0
  382. package/templates/nextblock-template/public/images/commerce-wide.webp +0 -0
  383. package/templates/nextblock-template/public/images/cortex-ai-square.webp +0 -0
  384. package/templates/nextblock-template/public/images/cortex-ai.webp +0 -0
  385. package/templates/nextblock-template/public/images/extensibility.webp +0 -0
  386. package/templates/nextblock-template/public/images/goals.webp +0 -0
  387. package/templates/nextblock-template/public/images/included.webp +0 -0
  388. package/templates/nextblock-template/public/images/nx-graph.webp +0 -0
  389. package/templates/nextblock-template/public/images/pants.webp +0 -0
  390. package/templates/nextblock-template/public/images/t-shirt.webp +0 -0
  391. package/templates/nextblock-template/scripts/validate-editor-block-schema.ts +112 -0
  392. package/templates/nextblock-template/scripts/verify-cortex-ai-build-widget.tsx +100 -0
  393. package/templates/nextblock-template/scripts/verify-cortex-ai-generate-blocks.ts +62 -0
  394. package/templates/nextblock-template/scripts/verify-cortex-ai-global-tools.ts +537 -0
  395. package/templates/nextblock-template/scripts/verify-cortex-ai-routing.ts +58 -0
  396. package/templates/nextblock-template/scripts/verify-custom-block-definitions.ts +188 -0
  397. package/templates/nextblock-template/scripts/verify-dynamic-custom-block-extensions.ts +123 -0
  398. package/templates/nextblock-template/scripts/verify-dynamic-layout-engine.tsx +133 -0
  399. package/templates/nextblock-template/scripts/verify-milestone-2-custom-blocks.ts +65 -0
  400. package/templates/nextblock-template/tailwind.config.js +1 -0
  401. package/templates/nextblock-template/tools/configure-supabase-auth.js +282 -0
  402. package/templates/nextblock-template/tools/deploy-supabase.js +69 -71
  403. package/templates/nextblock-template/tsconfig.json +52 -66
  404. package/templates/nextblock-template/tsconfig.tsbuildinfo +1 -1
  405. package/templates/nextblock-template/types/jsdom.d.ts +6 -0
  406. package/templates/nextblock-template/app/force-styles.tsx +0 -31
  407. package/templates/nextblock-template/app/sitemap.xml/route.ts +0 -63
  408. package/templates/nextblock-template/components/blocks/renderers/HeroBlockRenderer.tsx +0 -273
  409. package/templates/nextblock-template/docs/How to Create a Custom Block.md +0 -149
  410. package/templates/nextblock-template/docs/cms-application-overview.md +0 -56
  411. package/templates/nextblock-template/docs/cms-architecture-overview.md +0 -73
  412. package/templates/nextblock-template/docs/files-structure.md +0 -426
  413. package/templates/nextblock-template/docs/tiptap-bundle-optimization-summary.md +0 -174
@@ -0,0 +1,221 @@
1
+ 'use client';
2
+
3
+ import { useRef, useState } from 'react';
4
+ import { Image as ImageIcon, Loader2, UploadCloud, X } from 'lucide-react';
5
+ import { Button, Input, Label } from '@nextblock-cms/ui';
6
+
7
+ export type ImageR2Value = {
8
+ alt?: string;
9
+ file_name?: string;
10
+ file_type?: string;
11
+ height?: number;
12
+ object_key: string;
13
+ size_bytes?: number;
14
+ url: string;
15
+ width?: number;
16
+ };
17
+
18
+ export type ImageR2PickerProps = {
19
+ accept?: string[];
20
+ disabled?: boolean;
21
+ folder?: string;
22
+ maxBytes?: number;
23
+ onChange: (value: ImageR2Value | null) => void;
24
+ value?: ImageR2Value | null;
25
+ };
26
+
27
+ type PresignedUploadResponse = {
28
+ error?: string;
29
+ headers?: Record<string, string>;
30
+ objectKey?: string;
31
+ publicUrl?: string;
32
+ uploadUrl?: string;
33
+ };
34
+
35
+ const DEFAULT_ACCEPT = ['image/avif', 'image/gif', 'image/jpeg', 'image/png', 'image/webp'];
36
+ const DEFAULT_MAX_BYTES = 10 * 1024 * 1024;
37
+
38
+ function deriveAltText(filename: string) {
39
+ const base = filename.replace(/\.[^.]+$/, '');
40
+ return base
41
+ .replace(/[-_]+/g, ' ')
42
+ .replace(/\s+/g, ' ')
43
+ .trim();
44
+ }
45
+
46
+ function readImageDimensions(file: File) {
47
+ return new Promise<{ height?: number; width?: number }>((resolve) => {
48
+ const objectUrl = URL.createObjectURL(file);
49
+ const image = new window.Image();
50
+
51
+ image.onload = () => {
52
+ URL.revokeObjectURL(objectUrl);
53
+ resolve({ height: image.naturalHeight || undefined, width: image.naturalWidth || undefined });
54
+ };
55
+ image.onerror = () => {
56
+ URL.revokeObjectURL(objectUrl);
57
+ resolve({});
58
+ };
59
+ image.src = objectUrl;
60
+ });
61
+ }
62
+
63
+ export function ImageR2Picker({
64
+ accept = DEFAULT_ACCEPT,
65
+ disabled = false,
66
+ folder = 'custom-blocks',
67
+ maxBytes = DEFAULT_MAX_BYTES,
68
+ onChange,
69
+ value,
70
+ }: ImageR2PickerProps) {
71
+ const inputRef = useRef<HTMLInputElement | null>(null);
72
+ const [error, setError] = useState<string | null>(null);
73
+ const [isUploading, setIsUploading] = useState(false);
74
+
75
+ async function handleFile(file: File | null) {
76
+ if (!file) {
77
+ return;
78
+ }
79
+
80
+ setError(null);
81
+
82
+ if (!accept.includes(file.type)) {
83
+ setError('Select an AVIF, GIF, JPEG, PNG, or WebP image.');
84
+ return;
85
+ }
86
+
87
+ if (file.size > maxBytes) {
88
+ setError(`Images are limited to ${Math.round(maxBytes / (1024 * 1024))} MB.`);
89
+ return;
90
+ }
91
+
92
+ setIsUploading(true);
93
+ try {
94
+ const dimensions = await readImageDimensions(file);
95
+ const presignedResponse = await fetch('/api/media/r2-presigned', {
96
+ body: JSON.stringify({
97
+ contentType: file.type,
98
+ filename: file.name,
99
+ folder,
100
+ size: file.size,
101
+ }),
102
+ headers: { 'Content-Type': 'application/json' },
103
+ method: 'POST',
104
+ });
105
+ const presignedPayload = (await presignedResponse.json()) as PresignedUploadResponse;
106
+
107
+ if (!presignedResponse.ok || !presignedPayload.uploadUrl || !presignedPayload.objectKey) {
108
+ throw new Error(presignedPayload.error || 'Could not prepare the upload.');
109
+ }
110
+
111
+ const uploadResponse = await fetch(presignedPayload.uploadUrl, {
112
+ body: file,
113
+ headers: presignedPayload.headers ?? { 'Content-Type': file.type },
114
+ method: 'PUT',
115
+ });
116
+
117
+ if (!uploadResponse.ok) {
118
+ throw new Error('The image upload did not complete.');
119
+ }
120
+
121
+ onChange({
122
+ alt: value?.alt || deriveAltText(file.name),
123
+ file_name: file.name,
124
+ file_type: file.type,
125
+ height: dimensions.height,
126
+ object_key: presignedPayload.objectKey,
127
+ size_bytes: file.size,
128
+ url: presignedPayload.publicUrl || `/${presignedPayload.objectKey}`,
129
+ width: dimensions.width,
130
+ });
131
+ } catch (uploadError) {
132
+ console.error('[ImageR2Picker] Upload failed:', uploadError);
133
+ setError(uploadError instanceof Error ? uploadError.message : 'The upload failed.');
134
+ } finally {
135
+ setIsUploading(false);
136
+ if (inputRef.current) {
137
+ inputRef.current.value = '';
138
+ }
139
+ }
140
+ }
141
+
142
+ return (
143
+ <div className="space-y-3">
144
+ <input
145
+ ref={inputRef}
146
+ accept={accept.join(',')}
147
+ className="sr-only"
148
+ disabled={disabled || isUploading}
149
+ onChange={(event) => void handleFile(event.target.files?.[0] ?? null)}
150
+ type="file"
151
+ />
152
+
153
+ <div className="flex items-center gap-3">
154
+ <div className="relative flex h-20 w-20 shrink-0 items-center justify-center overflow-hidden rounded-md border bg-muted">
155
+ {value?.url ? (
156
+ // Plain img (not next/image): the source can be any R2/external host
157
+ // and must never crash the editor on an unconfigured next/image host.
158
+ // eslint-disable-next-line @next/next/no-img-element
159
+ <img
160
+ alt={value.alt || value.file_name || 'Uploaded image'}
161
+ className="h-full w-full object-cover"
162
+ src={value.url}
163
+ />
164
+ ) : (
165
+ <ImageIcon className="h-7 w-7 text-muted-foreground" />
166
+ )}
167
+ </div>
168
+
169
+ <div className="min-w-0 flex-1 space-y-2">
170
+ <div className="flex flex-wrap gap-2">
171
+ <Button
172
+ disabled={disabled || isUploading}
173
+ onClick={() => inputRef.current?.click()}
174
+ size="sm"
175
+ type="button"
176
+ variant="outline"
177
+ >
178
+ {isUploading ? (
179
+ <Loader2 className="mr-2 h-4 w-4 animate-spin" />
180
+ ) : (
181
+ <UploadCloud className="mr-2 h-4 w-4" />
182
+ )}
183
+ Upload
184
+ </Button>
185
+ {value ? (
186
+ <Button
187
+ disabled={disabled || isUploading}
188
+ onClick={() => onChange(null)}
189
+ size="sm"
190
+ type="button"
191
+ variant="ghost"
192
+ >
193
+ <X className="mr-2 h-4 w-4" />
194
+ Clear
195
+ </Button>
196
+ ) : null}
197
+ </div>
198
+
199
+ <p className="truncate text-xs text-muted-foreground">
200
+ {value?.object_key || 'No R2 image selected'}
201
+ </p>
202
+ </div>
203
+ </div>
204
+
205
+ {value ? (
206
+ <div className="space-y-1">
207
+ <Label htmlFor="image-r2-alt">Alt text</Label>
208
+ <Input
209
+ disabled={disabled || isUploading}
210
+ id="image-r2-alt"
211
+ onChange={(event) => onChange({ ...value, alt: event.target.value })}
212
+ placeholder="Describe the image"
213
+ value={value.alt ?? ''}
214
+ />
215
+ </div>
216
+ ) : null}
217
+
218
+ {error ? <p className="text-sm text-destructive">{error}</p> : null}
219
+ </div>
220
+ );
221
+ }
@@ -0,0 +1,12 @@
1
+ import React from "react";
2
+ import { BlockComposer } from "../components/BlockComposer";
3
+
4
+ export const dynamic = "force-dynamic";
5
+
6
+ export default function CreateCustomBlockPage() {
7
+ return (
8
+ <div className="w-full">
9
+ <BlockComposer mode="create" />
10
+ </div>
11
+ );
12
+ }
@@ -0,0 +1,438 @@
1
+ "use client";
2
+
3
+ import React, { useState, useEffect, useTransition } from "react";
4
+ import Link from "next/link";
5
+ import { useRouter } from "next/navigation";
6
+ import { toast } from "react-hot-toast";
7
+ import {
8
+ PlusCircle,
9
+ Copy,
10
+ Trash2,
11
+ Edit,
12
+ Search,
13
+ Grid,
14
+ List,
15
+ Boxes,
16
+ Loader2,
17
+ Code,
18
+ Layers,
19
+ Database,
20
+ ImageIcon,
21
+ Type,
22
+ HelpCircle,
23
+ } from "lucide-react";
24
+ import {
25
+ Button,
26
+ Card,
27
+ CardHeader,
28
+ CardTitle,
29
+ CardDescription,
30
+ CardContent,
31
+ CardFooter,
32
+ Badge,
33
+ Input,
34
+ ConfirmationDialog,
35
+ Skeleton,
36
+ } from "@nextblock-cms/ui";
37
+ import {
38
+ listCustomBlockDefinitions,
39
+ deleteCustomBlockDefinition,
40
+ duplicateCustomBlockDefinition,
41
+ } from "./actions";
42
+ import { BlocksLibraryTransferControls } from "./components/BlocksLibraryTransferControls";
43
+ import type { CustomBlockDefinition } from "@nextblock-cms/utils";
44
+
45
+ export default function CustomBlocksListPage() {
46
+ const router = useRouter();
47
+ const [isPending, startTransition] = useTransition();
48
+ const [blocks, setBlocks] = useState<CustomBlockDefinition[]>([]);
49
+ const [loading, setLoading] = useState(true);
50
+ const [searchQuery, setSearchQuery] = useState("");
51
+ const [viewMode, setViewMode] = useState<"grid" | "list">("grid");
52
+ const [deletingBlock, setDeletingBlock] = useState<CustomBlockDefinition | null>(null);
53
+ const [duplicatingId, setDuplicatingId] = useState<string | null>(null);
54
+
55
+ const fetchBlocks = async () => {
56
+ setLoading(true);
57
+ try {
58
+ const res = await listCustomBlockDefinitions();
59
+ if (res.success) {
60
+ setBlocks(res.data);
61
+ } else {
62
+ toast.error(res.error || "Failed to load custom blocks.");
63
+ }
64
+ } catch (err) {
65
+ console.error(err);
66
+ toast.error("An error occurred loading blocks.");
67
+ } finally {
68
+ setLoading(false);
69
+ }
70
+ };
71
+
72
+ useEffect(() => {
73
+ fetchBlocks();
74
+
75
+ // Refetch when Cortex AI (or another surface) reports a custom block change,
76
+ // and when the tab regains focus, so the library stays in sync without a reload.
77
+ const handleDataChanged = () => fetchBlocks();
78
+ window.addEventListener("nextblock:cortex-data-changed", handleDataChanged);
79
+ window.addEventListener("focus", handleDataChanged);
80
+
81
+ return () => {
82
+ window.removeEventListener("nextblock:cortex-data-changed", handleDataChanged);
83
+ window.removeEventListener("focus", handleDataChanged);
84
+ };
85
+ }, []);
86
+
87
+ const handleDuplicate = async (id: string, name: string) => {
88
+ setDuplicatingId(id);
89
+ const toastId = toast.loading(`Duplicating block "${name}"...`);
90
+ try {
91
+ const res = await duplicateCustomBlockDefinition(id);
92
+ if (res.success) {
93
+ toast.success(`Duplicated "${name}" successfully!`, { id: toastId });
94
+ fetchBlocks();
95
+ } else {
96
+ toast.error(res.error || `Failed to duplicate "${name}".`, { id: toastId });
97
+ }
98
+ } catch (err) {
99
+ console.error(err);
100
+ toast.error("An unexpected error occurred during duplication.", { id: toastId });
101
+ } finally {
102
+ setDuplicatingId(null);
103
+ }
104
+ };
105
+
106
+ const handleDelete = async () => {
107
+ if (!deletingBlock) return;
108
+ const blockName = deletingBlock.name;
109
+ const blockId = deletingBlock.id;
110
+ setDeletingBlock(null);
111
+
112
+ const toastId = toast.loading(`Deleting block "${blockName}"...`);
113
+ try {
114
+ const res = await deleteCustomBlockDefinition(blockId);
115
+ if (res.success) {
116
+ toast.success(`Deleted block "${blockName}".`, { id: toastId });
117
+ fetchBlocks();
118
+ } else {
119
+ toast.error(res.error || `Failed to delete block "${blockName}".`, { id: toastId });
120
+ }
121
+ } catch (err) {
122
+ console.error(err);
123
+ toast.error("An unexpected error occurred during deletion.", { id: toastId });
124
+ }
125
+ };
126
+
127
+ const filteredBlocks = blocks.filter(
128
+ (b) =>
129
+ b.name.toLowerCase().includes(searchQuery.toLowerCase()) ||
130
+ b.slug.toLowerCase().includes(searchQuery.toLowerCase()) ||
131
+ (b.description && b.description.toLowerCase().includes(searchQuery.toLowerCase()))
132
+ );
133
+
134
+ const getFieldIcon = (type: string) => {
135
+ switch (type) {
136
+ case "text":
137
+ return <Type className="h-3 w-3 mr-1 text-sky-500" />;
138
+ case "rich-text":
139
+ return <Code className="h-3 w-3 mr-1 text-emerald-500" />;
140
+ case "image_r2":
141
+ return <ImageIcon className="h-3 w-3 mr-1 text-amber-500" />;
142
+ case "db_relation":
143
+ return <Database className="h-3 w-3 mr-1 text-violet-500" />;
144
+ default:
145
+ return <HelpCircle className="h-3 w-3 mr-1 text-slate-500" />;
146
+ }
147
+ };
148
+
149
+ return (
150
+ <div className="w-full space-y-6">
151
+ {/* Header Panel */}
152
+ <div className="flex flex-col sm:flex-row justify-between items-start sm:items-center gap-4 border-b border-slate-200 dark:border-slate-800 pb-5">
153
+ <div>
154
+ <h2 className="text-2xl font-bold tracking-tight text-slate-900 dark:text-slate-50">Blocks Library</h2>
155
+ <p className="text-sm text-muted-foreground mt-1">
156
+ Build and manage custom block modules, layouts, and relations for rich pages.
157
+ </p>
158
+ </div>
159
+ <div className="flex flex-wrap items-center gap-2">
160
+ <BlocksLibraryTransferControls onImported={fetchBlocks} />
161
+ <Button asChild className="shadow-sm hover:scale-[1.01] transition-transform">
162
+ <Link href="/cms/custom-blocks/new">
163
+ <PlusCircle className="mr-2 h-4 w-4" /> Create Custom Block
164
+ </Link>
165
+ </Button>
166
+ </div>
167
+ </div>
168
+
169
+ {/* Toolbar */}
170
+ <div className="flex flex-col sm:flex-row gap-4 items-center justify-between">
171
+ <div className="relative w-full sm:max-w-md">
172
+ <Search className="absolute left-3 top-1/2 -translate-y-1/2 h-4 w-4 text-muted-foreground" />
173
+ <Input
174
+ placeholder="Search custom blocks by name or slug..."
175
+ value={searchQuery}
176
+ onChange={(e) => setSearchQuery(e.target.value)}
177
+ className="pl-9 dark:bg-slate-900/60"
178
+ />
179
+ </div>
180
+ <div className="flex items-center gap-2 border rounded-lg p-1 bg-white dark:bg-slate-900 border-slate-200 dark:border-slate-800 self-end sm:self-auto">
181
+ <Button
182
+ variant={viewMode === "grid" ? "secondary" : "ghost"}
183
+ size="icon"
184
+ onClick={() => setViewMode("grid")}
185
+ className="h-8 w-8"
186
+ title="Grid View"
187
+ >
188
+ <Grid className="h-4 w-4" />
189
+ </Button>
190
+ <Button
191
+ variant={viewMode === "list" ? "secondary" : "ghost"}
192
+ size="icon"
193
+ onClick={() => setViewMode("list")}
194
+ className="h-8 w-8"
195
+ title="List View"
196
+ >
197
+ <List className="h-4 w-4" />
198
+ </Button>
199
+ </div>
200
+ </div>
201
+
202
+ {/* Listing Grid/List */}
203
+ {loading ? (
204
+ <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
205
+ {[1, 2, 3].map((n) => (
206
+ <Card key={n} className="h-[240px] border-slate-200 dark:border-slate-800">
207
+ <CardHeader className="space-y-2">
208
+ <Skeleton className="h-5 w-2/3" />
209
+ <Skeleton className="h-4 w-1/3" />
210
+ </CardHeader>
211
+ <CardContent className="space-y-3">
212
+ <Skeleton className="h-10 w-full" />
213
+ <Skeleton className="h-4 w-1/2" />
214
+ </CardContent>
215
+ </Card>
216
+ ))}
217
+ </div>
218
+ ) : filteredBlocks.length === 0 ? (
219
+ <div className="flex flex-col items-center justify-center text-center py-16 px-4 border border-dashed border-slate-200 dark:border-slate-800 rounded-xl bg-white dark:bg-slate-900/40">
220
+ <div className="h-16 w-16 rounded-full bg-primary/10 flex items-center justify-center text-primary mb-4">
221
+ <Boxes className="h-8 w-8" />
222
+ </div>
223
+ <h3 className="text-lg font-semibold text-slate-900 dark:text-slate-50">No blocks found</h3>
224
+ <p className="text-sm text-muted-foreground mt-2 max-w-sm">
225
+ {searchQuery
226
+ ? "No blocks matched your search criteria. Try modifying your filters."
227
+ : "No custom blocks are registered. Start creating custom layouts or prompt Cortex AI."}
228
+ </p>
229
+ {!searchQuery && (
230
+ <Button asChild className="mt-6">
231
+ <Link href="/cms/custom-blocks/new">
232
+ <PlusCircle className="mr-2 h-4 w-4" /> Add Your First Block
233
+ </Link>
234
+ </Button>
235
+ )}
236
+ </div>
237
+ ) : viewMode === "grid" ? (
238
+ <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
239
+ {filteredBlocks.map((block) => (
240
+ <Card
241
+ key={block.id}
242
+ className="flex flex-col justify-between border-slate-200 dark:border-slate-800/80 bg-white dark:bg-slate-900 hover:shadow-lg hover:border-slate-300 dark:hover:border-slate-700/80 transition-all duration-200 rounded-xl overflow-hidden"
243
+ >
244
+ <CardHeader className="pb-3 border-b border-slate-100 dark:border-slate-800/50">
245
+ <div className="flex justify-between items-start gap-2">
246
+ <div className="space-y-1">
247
+ <CardTitle className="text-lg font-bold text-slate-900 dark:text-slate-50 leading-tight">
248
+ {block.name}
249
+ </CardTitle>
250
+ <code className="text-xs font-mono bg-slate-100 dark:bg-slate-800 text-slate-600 dark:text-slate-300 px-1.5 py-0.5 rounded block w-fit">
251
+ {block.slug}
252
+ </code>
253
+ </div>
254
+ <Badge
255
+ variant={block.is_original ? "outline" : "default"}
256
+ className={
257
+ block.is_original
258
+ ? "bg-slate-50 dark:bg-slate-800 text-slate-600 dark:text-slate-300 border-slate-200 dark:border-slate-700 font-medium shrink-0"
259
+ : "bg-indigo-50 dark:bg-indigo-950/40 text-indigo-700 dark:text-indigo-300 border-indigo-200 dark:border-indigo-800/60 font-medium shrink-0"
260
+ }
261
+ >
262
+ {block.is_original ? "Original" : "Custom"}
263
+ </Badge>
264
+ </div>
265
+ {block.description && (
266
+ <p className="text-xs text-muted-foreground line-clamp-2 mt-2 leading-relaxed">
267
+ {block.description}
268
+ </p>
269
+ )}
270
+ </CardHeader>
271
+
272
+ <CardContent className="py-4 space-y-3 flex-grow">
273
+ <div>
274
+ <h4 className="text-xs font-semibold text-slate-500 dark:text-slate-400 uppercase tracking-wider mb-2">
275
+ Fields Schema ({block.fields.length})
276
+ </h4>
277
+ {block.fields.length === 0 ? (
278
+ <span className="text-xs text-muted-foreground italic">No fields defined</span>
279
+ ) : (
280
+ <div className="flex flex-wrap gap-1.5 max-h-24 overflow-y-auto">
281
+ {block.fields.map((f) => (
282
+ <div
283
+ key={f.key}
284
+ className="inline-flex items-center text-xs px-2 py-1 rounded-md border border-slate-200 dark:border-slate-800 bg-slate-50/50 dark:bg-slate-900/50 text-slate-700 dark:text-slate-300 font-medium"
285
+ title={`${f.label} (${f.type})`}
286
+ >
287
+ {getFieldIcon(f.type)}
288
+ <span className="truncate max-w-[100px]">{f.key}</span>
289
+ </div>
290
+ ))}
291
+ </div>
292
+ )}
293
+ </div>
294
+ </CardContent>
295
+
296
+ <CardFooter className="pt-3 pb-4 px-6 bg-slate-50/50 dark:bg-slate-900/30 border-t border-slate-100 dark:border-slate-800/40 flex justify-between items-center">
297
+ <Button
298
+ variant="ghost"
299
+ size="sm"
300
+ className="text-destructive hover:bg-destructive/10 hover:text-destructive shrink-0 px-2.5"
301
+ onClick={() => setDeletingBlock(block)}
302
+ >
303
+ <Trash2 className="h-4 w-4 mr-1.5" /> Delete
304
+ </Button>
305
+ <div className="flex gap-2 shrink-0">
306
+ <Button
307
+ variant="outline"
308
+ size="sm"
309
+ className="px-2.5 dark:border-slate-700"
310
+ disabled={duplicatingId === block.id}
311
+ onClick={() => handleDuplicate(block.id, block.name)}
312
+ >
313
+ {duplicatingId === block.id ? (
314
+ <Loader2 className="h-4 w-4 animate-spin mr-1.5" />
315
+ ) : (
316
+ <Copy className="h-4 w-4 mr-1.5" />
317
+ )}
318
+ Duplicate
319
+ </Button>
320
+ <Button asChild size="sm" className="px-2.5">
321
+ <Link href={`/cms/custom-blocks/${block.id}/edit`}>
322
+ <Edit className="h-4 w-4 mr-1.5" /> Edit
323
+ </Link>
324
+ </Button>
325
+ </div>
326
+ </CardFooter>
327
+ </Card>
328
+ ))}
329
+ </div>
330
+ ) : (
331
+ /* List View */
332
+ <div className="rounded-xl border border-slate-200 dark:border-slate-800 bg-white dark:bg-slate-900 overflow-hidden shadow-sm">
333
+ <table className="w-full text-left border-collapse">
334
+ <thead>
335
+ <tr className="border-b border-slate-200 dark:border-slate-800 bg-slate-50/50 dark:bg-slate-900/50 text-xs font-semibold text-slate-500 uppercase tracking-wider">
336
+ <th className="px-6 py-4">Block Name / Slug</th>
337
+ <th className="px-6 py-4">Description</th>
338
+ <th className="px-6 py-4">Origin</th>
339
+ <th className="px-6 py-4">Fields count</th>
340
+ <th className="px-6 py-4 text-right">Actions</th>
341
+ </tr>
342
+ </thead>
343
+ <tbody className="divide-y divide-slate-200 dark:divide-slate-800 text-sm">
344
+ {filteredBlocks.map((block) => (
345
+ <tr
346
+ key={block.id}
347
+ className="hover:bg-slate-50/50 dark:hover:bg-slate-900/30 transition-colors"
348
+ >
349
+ <td className="px-6 py-4 font-medium text-slate-900 dark:text-slate-100">
350
+ <div className="flex flex-col gap-1">
351
+ <span className="font-semibold text-slate-900 dark:text-slate-100">
352
+ {block.name}
353
+ </span>
354
+ <code className="text-xs font-mono text-slate-500 dark:text-slate-400">
355
+ {block.slug}
356
+ </code>
357
+ </div>
358
+ </td>
359
+ <td className="px-6 py-4 text-muted-foreground max-w-sm truncate">
360
+ {block.description || <span className="italic text-xs">No description</span>}
361
+ </td>
362
+ <td className="px-6 py-4">
363
+ <Badge
364
+ variant={block.is_original ? "outline" : "default"}
365
+ className={
366
+ block.is_original
367
+ ? "bg-slate-50 dark:bg-slate-800 text-slate-600 dark:text-slate-300 border-slate-200 dark:border-slate-700"
368
+ : "bg-indigo-50 dark:bg-indigo-950/40 text-indigo-700 dark:text-indigo-300 border-indigo-200 dark:border-indigo-800/60"
369
+ }
370
+ >
371
+ {block.is_original ? "Original" : "Custom"}
372
+ </Badge>
373
+ </td>
374
+ <td className="px-6 py-4">
375
+ <Badge variant="secondary" className="font-medium bg-slate-100 dark:bg-slate-800">
376
+ {block.fields.length} fields
377
+ </Badge>
378
+ </td>
379
+ <td className="px-6 py-4 text-right">
380
+ <div className="flex items-center justify-end gap-2">
381
+ <Button
382
+ variant="ghost"
383
+ size="icon"
384
+ disabled={duplicatingId === block.id}
385
+ onClick={() => handleDuplicate(block.id, block.name)}
386
+ title="Duplicate"
387
+ className="h-8 w-8 text-slate-600 dark:text-slate-300"
388
+ >
389
+ {duplicatingId === block.id ? (
390
+ <Loader2 className="h-4 w-4 animate-spin" />
391
+ ) : (
392
+ <Copy className="h-4 w-4" />
393
+ )}
394
+ </Button>
395
+ <Button
396
+ asChild
397
+ variant="ghost"
398
+ size="icon"
399
+ title="Edit"
400
+ className="h-8 w-8 text-slate-600 dark:text-slate-300"
401
+ >
402
+ <Link href={`/cms/custom-blocks/${block.id}/edit`}>
403
+ <Edit className="h-4 w-4" />
404
+ </Link>
405
+ </Button>
406
+ <Button
407
+ variant="ghost"
408
+ size="icon"
409
+ onClick={() => setDeletingBlock(block)}
410
+ title="Delete"
411
+ className="h-8 w-8 text-destructive hover:bg-destructive/10 hover:text-destructive"
412
+ >
413
+ <Trash2 className="h-4 w-4" />
414
+ </Button>
415
+ </div>
416
+ </td>
417
+ </tr>
418
+ ))}
419
+ </tbody>
420
+ </table>
421
+ </div>
422
+ )}
423
+
424
+ {/* Confirmation Dialog for Deleting Block */}
425
+ <ConfirmationDialog
426
+ isOpen={deletingBlock !== null}
427
+ onOpenChange={(open) => {
428
+ if (!open) setDeletingBlock(null);
429
+ }}
430
+ onConfirm={handleDelete}
431
+ title="Delete Block Definition"
432
+ description={`Are you sure you want to delete the block "${deletingBlock?.name}"? All custom instances of this block in pages and posts will lose their visual formatting and render settings.`}
433
+ confirmText="Delete Block"
434
+ isDestructive={true}
435
+ />
436
+ </div>
437
+ );
438
+ }