create-nextblock 0.2.78 → 0.8.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 (413) hide show
  1. package/bin/create-nextblock.js +793 -472
  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,166 @@
1
+ import React from 'react';
2
+ import { renderToStaticMarkup } from 'react-dom/server';
3
+ import { describe, expect, it, vi } from 'vitest';
4
+ import type { CustomBlockDefinition } from '@nextblock-cms/utils';
5
+
6
+ import { DynamicLayoutEngine } from './DynamicLayoutEngine';
7
+
8
+ const testimonialDefinition: Pick<
9
+ CustomBlockDefinition,
10
+ 'fields' | 'id' | 'layout_schema' | 'name' | 'slug'
11
+ > = {
12
+ fields: [
13
+ { key: 'quote', label: 'Quote', required: false, type: 'rich-text' },
14
+ { key: 'author_name', label: 'Author Name', required: false, type: 'text' },
15
+ { key: 'portrait', label: 'Portrait', required: false, type: 'image_r2' },
16
+ {
17
+ display_column: 'full_name',
18
+ key: 'customer',
19
+ label: 'Customer',
20
+ multiple: false,
21
+ required: false,
22
+ table: 'profiles',
23
+ type: 'db_relation',
24
+ value_column: 'id',
25
+ },
26
+ ],
27
+ id: '33333333-3333-4333-8333-333333333333',
28
+ layout_schema: {
29
+ as: 'article',
30
+ children: [
31
+ {
32
+ as: 'div',
33
+ children: [
34
+ {
35
+ as: 'div',
36
+ children: [
37
+ {
38
+ as: 'figure',
39
+ children: [
40
+ {
41
+ className: 'h-16 w-16 rounded-full object-cover',
42
+ field_key: 'portrait',
43
+ type: 'field_render',
44
+ },
45
+ ],
46
+ className: 'flex justify-center',
47
+ type: 'container',
48
+ },
49
+ {
50
+ as: 'div',
51
+ children: [
52
+ {
53
+ as: 'blockquote',
54
+ className: 'text-lg font-medium',
55
+ field_key: 'quote',
56
+ type: 'field_render',
57
+ },
58
+ {
59
+ as: 'p',
60
+ className: 'text-sm font-semibold',
61
+ field_key: 'author_name',
62
+ type: 'field_render',
63
+ },
64
+ {
65
+ as: 'span',
66
+ className: 'text-xs text-muted-foreground',
67
+ field_key: 'customer',
68
+ type: 'field_render',
69
+ },
70
+ ],
71
+ className: 'flex flex-col gap-2',
72
+ type: 'container',
73
+ },
74
+ ],
75
+ className: 'grid gap-6 md:grid-cols-[auto_1fr]',
76
+ type: 'container',
77
+ },
78
+ ],
79
+ className: 'mx-auto max-w-3xl rounded-lg border p-6',
80
+ type: 'container',
81
+ },
82
+ ],
83
+ className: 'py-10',
84
+ type: 'container',
85
+ },
86
+ name: 'Nested Testimonial Card',
87
+ slug: 'nested-testimonial-card',
88
+ };
89
+
90
+ describe('DynamicLayoutEngine', () => {
91
+ it('renders an intricately nested layout schema into Tailwind structural markup', () => {
92
+ const html = renderToStaticMarkup(
93
+ <DynamicLayoutEngine
94
+ definition={testimonialDefinition}
95
+ data={{
96
+ author_name: 'Ada Lovelace',
97
+ customer: 'profile-1',
98
+ portrait: {
99
+ alt: 'Ada Lovelace portrait',
100
+ height: 128,
101
+ object_key: 'custom-blocks/testimonials/ada.webp',
102
+ url: '/custom-blocks/testimonials/ada.webp',
103
+ width: 128,
104
+ },
105
+ quote: '<p>Computation belongs in the imagination first.</p>',
106
+ resolved_relations: {
107
+ customer: {
108
+ record: {
109
+ full_name: 'Analytical Engine Society',
110
+ id: 'profile-1',
111
+ },
112
+ table: 'profiles',
113
+ value: 'profile-1',
114
+ },
115
+ },
116
+ }}
117
+ />
118
+ );
119
+
120
+ expect(html).toContain('py-10');
121
+ expect(html).toContain('mx-auto max-w-3xl rounded-lg border p-6');
122
+ expect(html).toContain('grid gap-6 md:grid-cols-[auto_1fr]');
123
+ expect(html).toContain('src="/custom-blocks/testimonials/ada.webp"');
124
+ expect(html).toContain('alt="Ada Lovelace portrait"');
125
+ expect(html).toContain('Computation belongs in the imagination first.');
126
+ expect(html).toContain('Ada Lovelace');
127
+ expect(html).toContain('Analytical Engine Society');
128
+ });
129
+
130
+ it('fails softly on corrupted layout nodes', () => {
131
+ const warnSpy = vi.spyOn(console, 'error').mockImplementation(() => undefined);
132
+ const cyclicNode: any = {
133
+ children: [],
134
+ type: 'container',
135
+ };
136
+ cyclicNode.children.push(cyclicNode);
137
+
138
+ const html = renderToStaticMarkup(
139
+ <DynamicLayoutEngine
140
+ fields={testimonialDefinition.fields}
141
+ layoutSchema={cyclicNode}
142
+ maxDepth={8}
143
+ />
144
+ );
145
+
146
+ expect(html).toContain('data-dynamic-layout-warning');
147
+ expect(html).toContain('cycle detected');
148
+ expect(warnSpy).not.toHaveBeenCalled();
149
+ warnSpy.mockRestore();
150
+ });
151
+
152
+ it('warns instead of throwing when a field_render references a missing field', () => {
153
+ const html = renderToStaticMarkup(
154
+ <DynamicLayoutEngine
155
+ fields={testimonialDefinition.fields}
156
+ layoutSchema={{
157
+ field_key: 'missing_field',
158
+ type: 'field_render',
159
+ }}
160
+ />
161
+ );
162
+
163
+ expect(html).toContain('data-dynamic-layout-warning');
164
+ expect(html).toContain('Unknown field');
165
+ });
166
+ });
@@ -0,0 +1,464 @@
1
+ import React from 'react';
2
+ import type { CustomBlockDefinition, CustomBlockField, CustomBlockLayoutNode } from '@nextblock-cms/utils';
3
+
4
+ import { resolveMediaUrl } from '../../lib/media/resolveMediaUrl';
5
+
6
+ export const DYNAMIC_LAYOUT_ENGINE_CACHE_TAG = 'dynamic-layout-engine';
7
+ export const DYNAMIC_LAYOUT_ENGINE_MAX_DEPTH = 64;
8
+
9
+ const CONTAINER_ELEMENTS = new Set([
10
+ 'article',
11
+ 'aside',
12
+ 'blockquote',
13
+ 'div',
14
+ 'figure',
15
+ 'figcaption',
16
+ 'h2',
17
+ 'h3',
18
+ 'p',
19
+ 'section',
20
+ 'span',
21
+ ]);
22
+
23
+ const FIELD_ELEMENTS = new Set([...CONTAINER_ELEMENTS, 'img']);
24
+ const RELATION_DISPLAY_FALLBACK_COLUMNS = [
25
+ 'title',
26
+ 'name',
27
+ 'full_name',
28
+ 'file_name',
29
+ 'slug',
30
+ 'id',
31
+ ];
32
+
33
+ type DynamicLayoutData = Record<string, unknown> & {
34
+ resolved_relations?: Record<string, unknown>;
35
+ };
36
+
37
+ type ResolvedRelationEntry = {
38
+ error?: string;
39
+ record?: Record<string, unknown> | null;
40
+ table?: string;
41
+ value?: string;
42
+ };
43
+
44
+ export type DynamicLayoutEngineProps = {
45
+ cacheTags?: string[];
46
+ className?: string;
47
+ data?: DynamicLayoutData;
48
+ definition?: Pick<CustomBlockDefinition, 'fields' | 'id' | 'layout_schema' | 'name' | 'slug'>;
49
+ fields?: CustomBlockField[];
50
+ layoutSchema?: CustomBlockLayoutNode;
51
+ maxDepth?: number;
52
+ };
53
+
54
+ type RenderContext = {
55
+ data: DynamicLayoutData;
56
+ fieldsByKey: Map<string, CustomBlockField>;
57
+ maxDepth: number;
58
+ path: string;
59
+ visited: WeakSet<object>;
60
+ };
61
+
62
+ export function getDynamicLayoutDefinitionCacheTag(idOrSlug: string) {
63
+ return `${DYNAMIC_LAYOUT_ENGINE_CACHE_TAG}:definition:${idOrSlug}`;
64
+ }
65
+
66
+ function WarningTag({ message }: { message: string }) {
67
+ return (
68
+ <span
69
+ className="inline-flex rounded border border-destructive/30 bg-destructive/10 px-2 py-1 text-xs text-destructive"
70
+ data-dynamic-layout-warning=""
71
+ >
72
+ {message}
73
+ </span>
74
+ );
75
+ }
76
+
77
+ function isRecord(value: unknown): value is Record<string, unknown> {
78
+ return Boolean(value) && typeof value === 'object' && !Array.isArray(value);
79
+ }
80
+
81
+ function isLayoutNode(value: unknown): value is CustomBlockLayoutNode {
82
+ return isRecord(value) && (value.type === 'container' || value.type === 'field_render');
83
+ }
84
+
85
+ function resolveElement(
86
+ requested: unknown,
87
+ fallback: keyof React.JSX.IntrinsicElements,
88
+ allowedElements: Set<string>
89
+ ) {
90
+ return typeof requested === 'string' && allowedElements.has(requested)
91
+ ? (requested as keyof React.JSX.IntrinsicElements)
92
+ : fallback;
93
+ }
94
+
95
+ function stringifyDisplayValue(value: unknown): string {
96
+ if (value === null || value === undefined || value === '') {
97
+ return '';
98
+ }
99
+
100
+ if (typeof value === 'string' || typeof value === 'number' || typeof value === 'boolean') {
101
+ return String(value);
102
+ }
103
+
104
+ if (Array.isArray(value)) {
105
+ return value.map((entry): string => stringifyDisplayValue(entry)).filter(Boolean).join(', ');
106
+ }
107
+
108
+ if (isRecord(value)) {
109
+ for (const column of RELATION_DISPLAY_FALLBACK_COLUMNS) {
110
+ const displayValue = value[column];
111
+ if (displayValue !== null && displayValue !== undefined && displayValue !== '') {
112
+ return String(displayValue);
113
+ }
114
+ }
115
+
116
+ return JSON.stringify(value);
117
+ }
118
+
119
+ return String(value);
120
+ }
121
+
122
+ // Monetary columns are stored as integer minor units (cents) in the database but
123
+ // must be shown to visitors as currency (e.g. 25000 -> $250.00).
124
+ function isPriceColumn(column: string) {
125
+ return column === 'price' || column === 'prices' || column === 'price_adjustment' || /_price$/.test(column) || /_prices$/.test(column);
126
+ }
127
+
128
+ function formatCentsAsCurrency(cents: number) {
129
+ return `$${(cents / 100).toFixed(2)}`;
130
+ }
131
+
132
+ function formatRelationColumnValue(column: string, value: unknown): string {
133
+ if (isPriceColumn(column)) {
134
+ if (typeof value === 'number' && Number.isFinite(value)) {
135
+ return formatCentsAsCurrency(value);
136
+ }
137
+ // Multi-currency maps like { "USD": 25000 } store minor units too.
138
+ if (isRecord(value)) {
139
+ const amounts = Object.values(value).filter(
140
+ (entry): entry is number => typeof entry === 'number' && Number.isFinite(entry)
141
+ );
142
+ if (amounts.length > 0) {
143
+ return formatCentsAsCurrency(amounts[0]);
144
+ }
145
+ }
146
+ }
147
+
148
+ return stringifyDisplayValue(value);
149
+ }
150
+
151
+ function getResolvedRelationLabel(
152
+ field: CustomBlockField,
153
+ data: DynamicLayoutData,
154
+ column?: string
155
+ ) {
156
+ if (field.type !== 'db_relation') {
157
+ return '';
158
+ }
159
+
160
+ const targetColumn = column ?? field.display_column;
161
+ const relation = data.resolved_relations?.[field.key];
162
+ const entries = Array.isArray(relation) ? relation : relation ? [relation] : [];
163
+ const labels = entries
164
+ .map((entry) => {
165
+ if (!isRecord(entry)) {
166
+ return '';
167
+ }
168
+
169
+ const relationEntry = entry as ResolvedRelationEntry;
170
+ if (relationEntry.record) {
171
+ const preferred = relationEntry.record[targetColumn];
172
+ if (preferred !== null && preferred !== undefined && preferred !== '') {
173
+ return formatRelationColumnValue(targetColumn, preferred);
174
+ }
175
+
176
+ return stringifyDisplayValue(relationEntry.record);
177
+ }
178
+
179
+ return relationEntry.value ?? '';
180
+ })
181
+ .filter(Boolean);
182
+
183
+ return labels.join(', ');
184
+ }
185
+
186
+ const UUID_RE = /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i;
187
+
188
+ function looksLikeImageRef(value: unknown): value is string {
189
+ return (
190
+ typeof value === 'string' &&
191
+ value.length > 0 &&
192
+ !UUID_RE.test(value) &&
193
+ (value.startsWith('http') ||
194
+ value.startsWith('/') ||
195
+ value.includes('/') ||
196
+ /\.(avif|gif|jpe?g|png|svg|webp)$/i.test(value))
197
+ );
198
+ }
199
+
200
+ // Pulls a usable image source out of a resolved relation record. Prefers the
201
+ // configured display column when it points at an asset, then falls back to the
202
+ // well-known image-bearing columns and any nested media relation object.
203
+ function extractRelationImageRef(
204
+ record: Record<string, unknown>,
205
+ displayColumn: string
206
+ ): string | null {
207
+ if (looksLikeImageRef(record[displayColumn])) {
208
+ return record[displayColumn] as string;
209
+ }
210
+
211
+ for (const column of ['object_key', 'main_image', 'avatar_url', 'url', 'image_url', 'file_path']) {
212
+ if (looksLikeImageRef(record[column])) {
213
+ return record[column] as string;
214
+ }
215
+ }
216
+
217
+ const nested = record.image;
218
+ if (isRecord(nested)) {
219
+ if (looksLikeImageRef(nested.object_key)) {
220
+ return nested.object_key as string;
221
+ }
222
+ if (looksLikeImageRef(nested.url)) {
223
+ return nested.url as string;
224
+ }
225
+ }
226
+
227
+ return null;
228
+ }
229
+
230
+ function getImageValue(value: unknown) {
231
+ if (typeof value === 'string') {
232
+ const src = resolveMediaUrl(value);
233
+ return src ? { alt: '', src } : null;
234
+ }
235
+
236
+ if (!isRecord(value)) {
237
+ return null;
238
+ }
239
+
240
+ const url = typeof value.url === 'string' ? value.url : null;
241
+ const objectKey = typeof value.object_key === 'string' ? value.object_key : null;
242
+ // Resolve both candidates through resolveMediaUrl: full URLs pass through
243
+ // unchanged while bare object keys (e.g. "images/commerce-square.webp") are
244
+ // mapped to their public/bundled location instead of resolving relative to
245
+ // the current page URL.
246
+ const src = resolveMediaUrl(url) ?? resolveMediaUrl(objectKey);
247
+
248
+ if (!src) {
249
+ return null;
250
+ }
251
+
252
+ return {
253
+ alt: typeof value.alt === 'string' ? value.alt : '',
254
+ height: typeof value.height === 'number' && value.height > 0 ? value.height : undefined,
255
+ src,
256
+ width: typeof value.width === 'number' && value.width > 0 ? value.width : undefined,
257
+ };
258
+ }
259
+
260
+ function renderImageField({
261
+ className,
262
+ field,
263
+ node,
264
+ value,
265
+ }: {
266
+ className?: string;
267
+ field: CustomBlockField;
268
+ node: Extract<CustomBlockLayoutNode, { type: 'field_render' }>;
269
+ value: unknown;
270
+ }) {
271
+ const image = getImageValue(value);
272
+ if (!image) {
273
+ return <WarningTag message={`Missing image field "${field.key}"`} />;
274
+ }
275
+
276
+ const img = React.createElement('img', {
277
+ alt: image.alt || field.label,
278
+ className: resolveElement(node.as, 'img', FIELD_ELEMENTS) === 'img' ? className : undefined,
279
+ height: image.height,
280
+ loading: 'lazy',
281
+ src: image.src,
282
+ width: image.width,
283
+ });
284
+
285
+ const requestedElement = resolveElement(node.as, 'img', FIELD_ELEMENTS);
286
+ if (requestedElement === 'img') {
287
+ return img;
288
+ }
289
+
290
+ return React.createElement(requestedElement, { className }, img);
291
+ }
292
+
293
+ function renderTextField({
294
+ className,
295
+ field,
296
+ node,
297
+ value,
298
+ }: {
299
+ className?: string;
300
+ field: CustomBlockField;
301
+ node: Extract<CustomBlockLayoutNode, { type: 'field_render' }>;
302
+ value: unknown;
303
+ }) {
304
+ const requestedElement = resolveElement(
305
+ node.as,
306
+ field.type === 'rich-text' ? 'div' : 'span',
307
+ FIELD_ELEMENTS
308
+ );
309
+ const element = requestedElement === 'img' ? 'span' : requestedElement;
310
+
311
+ if (field.type === 'rich-text' && typeof value === 'string') {
312
+ return React.createElement(element, {
313
+ className,
314
+ dangerouslySetInnerHTML: { __html: value || node.emptyFallback || '' },
315
+ });
316
+ }
317
+
318
+ const displayValue =
319
+ field.type === 'db_relation' ? stringifyDisplayValue(value) : stringifyDisplayValue(value);
320
+
321
+ return React.createElement(
322
+ element,
323
+ { className },
324
+ displayValue || node.emptyFallback || ''
325
+ );
326
+ }
327
+
328
+ function renderFieldNode(
329
+ node: Extract<CustomBlockLayoutNode, { type: 'field_render' }>,
330
+ context: RenderContext
331
+ ) {
332
+ if (typeof node.field_key !== 'string') {
333
+ return <WarningTag message="Invalid custom block field reference" />;
334
+ }
335
+
336
+ const field = context.fieldsByKey.get(node.field_key);
337
+ if (!field) {
338
+ return <WarningTag message={`Unknown field "${node.field_key}"`} />;
339
+ }
340
+
341
+ // A field_render node bound to a db_relation may pick a specific column of the
342
+ // resolved record (e.g. show a product's price or title), overriding the
343
+ // field's default display_column.
344
+ const relationColumn = typeof node.column === 'string' && node.column ? node.column : undefined;
345
+ const value =
346
+ field.type === 'db_relation'
347
+ ? getResolvedRelationLabel(field, context.data, relationColumn) || context.data[field.key]
348
+ : context.data[field.key];
349
+ const className = typeof node.className === 'string' ? node.className : undefined;
350
+
351
+ // Resolve relation fields rendered as images
352
+ if (field.type === 'db_relation' && node.as === 'img') {
353
+ const relation = context.data.resolved_relations?.[field.key];
354
+ const entry = Array.isArray(relation) ? relation[0] : relation;
355
+ let imageRef: string | null = null;
356
+
357
+ if (isRecord(entry) && isRecord(entry.record)) {
358
+ imageRef = extractRelationImageRef(entry.record, relationColumn ?? field.display_column);
359
+ } else if (looksLikeImageRef(value)) {
360
+ imageRef = value;
361
+ }
362
+
363
+ return renderImageField({
364
+ className,
365
+ field,
366
+ node,
367
+ value: imageRef ? { object_key: imageRef } : null,
368
+ });
369
+ }
370
+
371
+ if (field.type === 'image_r2') {
372
+ return renderImageField({ className, field, node, value });
373
+ }
374
+
375
+ return renderTextField({ className, field, node, value });
376
+ }
377
+
378
+ function renderContainerNode(
379
+ node: Extract<CustomBlockLayoutNode, { type: 'container' }>,
380
+ context: RenderContext,
381
+ depth: number
382
+ ) {
383
+ const Element = resolveElement(node.as, 'div', CONTAINER_ELEMENTS);
384
+ const children = Array.isArray(node.children) ? node.children : [];
385
+ const renderedChildren = children.map((child, index) => (
386
+ <React.Fragment key={`${context.path}.${index}`}>
387
+ {renderDynamicLayoutNode(child, {
388
+ ...context,
389
+ path: `${context.path}.${index}`,
390
+ }, depth + 1)}
391
+ </React.Fragment>
392
+ ));
393
+
394
+ return React.createElement(
395
+ Element,
396
+ { className: typeof node.className === 'string' ? node.className : undefined },
397
+ renderedChildren
398
+ );
399
+ }
400
+
401
+ export function renderDynamicLayoutNode(
402
+ node: unknown,
403
+ context: RenderContext,
404
+ depth = 0
405
+ ): React.ReactNode {
406
+ try {
407
+ if (depth > context.maxDepth) {
408
+ return <WarningTag message="Custom block layout depth limit reached" />;
409
+ }
410
+
411
+ if (!isLayoutNode(node)) {
412
+ return <WarningTag message="Invalid custom block layout node" />;
413
+ }
414
+
415
+ if (isRecord(node)) {
416
+ if (context.visited.has(node)) {
417
+ return <WarningTag message="Custom block layout cycle detected" />;
418
+ }
419
+ context.visited.add(node);
420
+ }
421
+
422
+ if (node.type === 'container') {
423
+ return renderContainerNode(node, context, depth);
424
+ }
425
+
426
+ return renderFieldNode(node, context);
427
+ } catch (error) {
428
+ console.error('[DynamicLayoutEngine] Failed to render layout node:', error);
429
+ return <WarningTag message="Invalid custom block layout" />;
430
+ }
431
+ }
432
+
433
+ export function DynamicLayoutEngine({
434
+ className,
435
+ data,
436
+ definition,
437
+ fields,
438
+ layoutSchema,
439
+ maxDepth = DYNAMIC_LAYOUT_ENGINE_MAX_DEPTH,
440
+ }: DynamicLayoutEngineProps) {
441
+ const resolvedLayoutSchema = layoutSchema ?? definition?.layout_schema;
442
+ const resolvedFields = fields ?? definition?.fields ?? [];
443
+
444
+ if (!resolvedLayoutSchema) {
445
+ return <WarningTag message="Missing custom block layout" />;
446
+ }
447
+
448
+ const fieldsByKey = new Map(resolvedFields.map((field) => [field.key, field]));
449
+ const rendered = renderDynamicLayoutNode(resolvedLayoutSchema, {
450
+ data: data ?? {},
451
+ fieldsByKey,
452
+ maxDepth,
453
+ path: 'root',
454
+ visited: new WeakSet<object>(),
455
+ });
456
+
457
+ if (!className) {
458
+ return <>{rendered}</>;
459
+ }
460
+
461
+ return <div className={className}>{rendered}</div>;
462
+ }
463
+
464
+ export default DynamicLayoutEngine;
@@ -1,13 +1,13 @@
1
1
  "use client";
2
2
 
3
- import { Button } from "@nextblock-cms/ui";
4
- import {
5
- DropdownMenu,
6
- DropdownMenuContent,
7
- DropdownMenuRadioGroup,
8
- DropdownMenuRadioItem,
9
- DropdownMenuTrigger,
10
- } from "@nextblock-cms/ui";
3
+ import { Button } from "@nextblock-cms/ui";
4
+ import {
5
+ DropdownMenu,
6
+ DropdownMenuContent,
7
+ DropdownMenuRadioGroup,
8
+ DropdownMenuRadioItem,
9
+ DropdownMenuTrigger,
10
+ } from "@nextblock-cms/ui";
11
11
  import { Laptop, Moon, Sun, Zap } from "lucide-react";
12
12
  import { useTheme } from "next-themes";
13
13
  import { useEffect, useState } from "react";
@@ -0,0 +1,21 @@
1
+ 'use client';
2
+
3
+ import dynamic from 'next/dynamic';
4
+
5
+ // Defer the visual-editing toolbar (and its heavy editor graph: BlockEditorModal,
6
+ // the per-block editors, and editor.css) out of the public bundle. The layout only
7
+ // renders this in draft/visual-editing mode, but a *static* import would still ship
8
+ // the entire editor graph to every public page. Loading it via next/dynamic with
9
+ // ssr:false keeps it out of the public homepage's render-blocking CSS and JS, and
10
+ // pulls it in client-side only when visual editing is actually active.
11
+ const NextblockVisualEditing = dynamic(
12
+ () =>
13
+ import('./NextblockVisualEditing').then(
14
+ (module) => module.NextblockVisualEditing
15
+ ),
16
+ { ssr: false }
17
+ );
18
+
19
+ export function DeferredVisualEditing() {
20
+ return <NextblockVisualEditing />;
21
+ }