create-nextblock 0.2.77 → 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 +191 -151
  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 -116
  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,23 @@
1
+ import type { ComponentType } from "react";
2
+
3
+ type PublicBlockRendererLoader = () => Promise<{
4
+ default: ComponentType<any>;
5
+ }>;
6
+
7
+ const ecommerceBlockRendererLoaders: Record<string, PublicBlockRendererLoader> = {
8
+ product_grid: () => import("./renderers/ProductGridBlockRenderer"),
9
+ featured_product: () => import("./renderers/FeaturedProductBlockRenderer"),
10
+ cart: () => import("./renderers/CartBlockRenderer"),
11
+ checkout: () => import("./renderers/CheckoutBlockRenderer"),
12
+ product_details: () => import("./renderers/ProductDetailsBlockRenderer"),
13
+ };
14
+
15
+ export function loadEcommerceBlockRenderer(blockType: string) {
16
+ const loader = ecommerceBlockRendererLoaders[blockType];
17
+
18
+ if (!loader) {
19
+ return Promise.reject(new Error(`Unsupported ecommerce block type: ${blockType}`));
20
+ }
21
+
22
+ return loader();
23
+ }
@@ -0,0 +1,25 @@
1
+ import type { ComponentType } from "react";
2
+ import type { BlockType } from "../../lib/blocks/blockRegistry";
3
+
4
+ type PublicBlockRendererLoader = () => Promise<{
5
+ default: ComponentType<any>;
6
+ }>;
7
+
8
+ const publicBlockRendererLoaders: Partial<
9
+ Record<BlockType | "hero", PublicBlockRendererLoader>
10
+ > = {
11
+ text: () => import("./renderers/TextBlockRenderer"),
12
+ heading: () => import("./renderers/HeadingBlockRenderer"),
13
+ image: () => import("./renderers/ImageBlockRenderer"),
14
+ button: () => import("./renderers/ButtonBlockRenderer"),
15
+ posts_grid: () => import("./renderers/PostsGridBlockRenderer"),
16
+ video_embed: () => import("./renderers/VideoEmbedBlockRenderer"),
17
+ section: () => import("./renderers/SectionBlockRenderer"),
18
+ hero: () => import("./renderers/SectionBlockRenderer"),
19
+ form: () => import("./renderers/FormBlockRenderer"),
20
+ testimonial: () => import("./renderers/TestimonialBlockRenderer"),
21
+ };
22
+
23
+ export function getPublicBlockRendererLoader(blockType: string) {
24
+ return publicBlockRendererLoaders[blockType as BlockType | "hero"];
25
+ }
@@ -1,84 +1,92 @@
1
- import React from "react";
2
- import Link from "next/link";
3
- import { Button } from "@nextblock-cms/ui";
4
- import { cn } from "@nextblock-cms/utils";
5
-
6
- export type ButtonBlockContent = {
7
- text?: string;
8
- url?: string;
9
- variant?: 'default' | 'outline' | 'secondary' | 'ghost' | 'link';
10
- size?: 'default' | 'sm' | 'lg' | 'full';
11
- position?: 'left' | 'center' | 'right';
12
- };
13
-
14
- interface ButtonBlockRendererProps {
15
- content: ButtonBlockContent;
16
- languageId: number; // This prop seems unused
17
- }
18
-
19
- const ButtonBlockRenderer: React.FC<ButtonBlockRendererProps> = ({
20
- content,
21
- }) => {
22
- const isExternal =
23
- content.url?.startsWith("http") ||
24
- content.url?.startsWith("mailto:") ||
25
- content.url?.startsWith("tel:");
26
- const isAnchor = content.url?.startsWith("#");
27
-
28
- const buttonText = content.text || "Button";
29
- const buttonVariant = content.variant || "default";
30
- const buttonSize = content.size || "default";
31
- const buttonPosition = content.position || "left";
32
-
33
- const alignmentClasses = {
34
- left: "justify-start text-left",
35
- center: "justify-center text-center",
36
- right: "justify-end text-right",
37
- };
38
-
39
- return (
40
- <div className={cn("my-6 flex w-full", alignmentClasses[buttonPosition])}>
41
- {/* Case 1: Internal link (not external, not anchor, has URL) */}
42
- {!isExternal && !isAnchor && !!content.url ? (
43
- <Button
44
- asChild
45
- variant={buttonVariant}
46
- size={buttonSize}
47
- className={cn(content.variant === 'outline' && "text-foreground")}
48
- >
49
- <Link href={content.url}>
50
- {buttonText}
51
- </Link>
52
- </Button>
53
- ) : /* Case 2: External or Anchor link (has URL) */
54
- (isExternal || isAnchor) && !!content.url ? (
55
- <Button
56
- asChild
57
- variant={buttonVariant}
58
- size={buttonSize}
59
- className={cn(content.variant === 'outline' && "text-foreground")}
60
- >
61
- <a
62
- href={content.url}
63
- target={isExternal ? "_blank" : undefined}
64
- rel={isExternal ? "noopener noreferrer" : undefined}
65
- >
66
- {buttonText}
67
- </a>
68
- </Button>
69
- ) : (
70
- /* Case 3: No URL or other edge cases - render a plain or disabled button */
71
- <Button
72
- variant={buttonVariant}
73
- size={buttonSize}
74
- disabled={!content.url}
75
- className={cn(content.variant === 'outline' && "text-foreground")}
76
- >
77
- {buttonText}
78
- </Button>
79
- )}
80
- </div>
81
- );
82
- };
83
-
84
- export default ButtonBlockRenderer;
1
+ "use client";
2
+ import React from "react";
3
+ import Link from "next/link";
4
+ import { buttonVariants } from "@nextblock-cms/ui";
5
+ import { cn } from "@nextblock-cms/utils";
6
+ import type { VisualEditAttributes } from "../../../lib/visual-editing/types";
7
+
8
+ export type ButtonBlockContent = {
9
+ text?: string;
10
+ url?: string;
11
+ variant?: 'default' | 'outline' | 'secondary' | 'ghost' | 'link';
12
+ size?: 'default' | 'sm' | 'lg' | 'full';
13
+ position?: 'left' | 'center' | 'right';
14
+ };
15
+
16
+ interface ButtonBlockRendererProps {
17
+ content: ButtonBlockContent;
18
+ languageId: number; // This prop seems unused
19
+ visualEditAttributes?: VisualEditAttributes;
20
+ }
21
+
22
+ const ButtonBlockRenderer: React.FC<ButtonBlockRendererProps> = ({
23
+ content,
24
+ visualEditAttributes,
25
+ }) => {
26
+ const isExternal =
27
+ content.url?.startsWith("http") ||
28
+ content.url?.startsWith("mailto:") ||
29
+ content.url?.startsWith("tel:");
30
+ const isAnchor = content.url?.startsWith("#");
31
+
32
+ const buttonText = content.text || "Button";
33
+ // Map variant name if needed or pass directly if it matches
34
+ // The migration might have uppercase 'DEFAULT' or 'OUTLINE', ensure lowercase
35
+ const variantRaw = content.variant?.toLowerCase() || "default";
36
+ // Cast to specific allowed variants for TS safety if strictly typed, else as any
37
+ const variant = (variantRaw === 'default' || variantRaw === 'outline' || variantRaw === 'secondary' || variantRaw === 'ghost' || variantRaw === 'link' || variantRaw === 'destructive')
38
+ ? variantRaw
39
+ : 'default';
40
+
41
+ const sizeRaw = content.size?.toLowerCase() || "default";
42
+ const size = (sizeRaw === 'default' || sizeRaw === 'sm' || sizeRaw === 'lg' || sizeRaw === 'icon')
43
+ ? sizeRaw
44
+ : 'default';
45
+
46
+ const buttonPosition = content.position || "left";
47
+
48
+ const alignmentClasses = {
49
+ left: "justify-start text-left",
50
+ center: "justify-center text-center",
51
+ right: "justify-end text-right",
52
+ };
53
+
54
+ // We use buttonVariants directly to avoid 'asChild' composition issues with Radix Slot
55
+ // This ensures we just render a clean Link or Anchor with the correct classes.
56
+ const classes = cn(
57
+ buttonVariants({ variant, size }),
58
+ variant === 'outline' && "text-foreground", // Ensure outline text is visible
59
+ "no-underline" // Force no-underline for links
60
+ );
61
+
62
+ return (
63
+ <div
64
+ className={cn("my-6 flex w-full", alignmentClasses[buttonPosition])}
65
+ {...visualEditAttributes}
66
+ >
67
+ {/* Case 1: Internal link */}
68
+ {!isExternal && !isAnchor && !!content.url ? (
69
+ <Link href={content.url} className={classes}>
70
+ {buttonText}
71
+ </Link>
72
+ ) : /* Case 2: External or Anchor link */
73
+ (isExternal || isAnchor) && !!content.url ? (
74
+ <a
75
+ href={content.url}
76
+ target={isExternal ? "_blank" : undefined}
77
+ rel={isExternal ? "noopener noreferrer" : undefined}
78
+ className={classes}
79
+ >
80
+ {buttonText}
81
+ </a>
82
+ ) : (
83
+ /* Case 3: No URL - render a disabled fake button */
84
+ <button disabled className={cn(classes, "opacity-50 cursor-not-allowed")}>
85
+ {buttonText}
86
+ </button>
87
+ )}
88
+ </div>
89
+ );
90
+ };
91
+
92
+ export default ButtonBlockRenderer;
@@ -0,0 +1,17 @@
1
+ 'use client';
2
+
3
+ import React from 'react';
4
+ import { Cart } from '@nextblock-cms/ecommerce/components/Cart';
5
+ import type { VisualEditAttributes } from '../../../lib/visual-editing/types';
6
+
7
+ interface CartBlockRendererProps {
8
+ visualEditAttributes?: VisualEditAttributes;
9
+ }
10
+
11
+ export default function CartBlockRenderer({ visualEditAttributes }: CartBlockRendererProps) {
12
+ return (
13
+ <div {...visualEditAttributes}>
14
+ <Cart />
15
+ </div>
16
+ );
17
+ }
@@ -0,0 +1,19 @@
1
+ 'use client';
2
+
3
+ import React from 'react';
4
+ import { Checkout } from '@nextblock-cms/ecommerce/components/Checkout';
5
+ import type { VisualEditAttributes } from '../../../lib/visual-editing/types';
6
+
7
+ interface CheckoutBlockRendererProps {
8
+ visualEditAttributes?: VisualEditAttributes;
9
+ }
10
+
11
+ export default function CheckoutBlockRenderer({
12
+ visualEditAttributes,
13
+ }: CheckoutBlockRendererProps) {
14
+ return (
15
+ <div {...visualEditAttributes}>
16
+ <Checkout />
17
+ </div>
18
+ );
19
+ }
@@ -1,23 +1,256 @@
1
- "use client";
2
-
3
1
  import React from "react";
4
- import parse, { HTMLReactParserOptions, Element } from 'html-react-parser';
2
+ import Image from "next/image";
3
+ import parse, {
4
+ HTMLReactParserOptions,
5
+ Element,
6
+ attributesToProps,
7
+ } from 'html-react-parser';
5
8
  import AlertWidgetRenderer from "./inline/AlertWidgetRenderer";
6
9
  import CtaWidgetRenderer from "./inline/CtaWidgetRenderer";
7
10
  import type { TextBlockContent } from "./TextBlockRenderer";
11
+ import type { VisualEditAttributes } from "../../../lib/visual-editing/types";
12
+ import { SimpleTiptapRenderer } from "@nextblock-cms/ecommerce/components/SimpleTiptapRenderer";
8
13
 
9
14
  interface ClientTextBlockRendererProps {
10
15
  content: TextBlockContent;
11
16
  languageId: number;
17
+ visualEditAttributes?: VisualEditAttributes;
18
+ renderContext?: 'prose' | 'section';
19
+ }
20
+
21
+ function normalizeHtmlEncodingArtifacts(html: string): string {
22
+ return html
23
+ .replaceAll("✓", "&#10003;")
24
+ .replaceAll("✔", "&#10003;");
25
+ }
26
+
27
+ function normalizeImageAttributes(attribs: Record<string, string>) {
28
+ if (attribs['fetchpriority']) {
29
+ attribs.fetchPriority = attribs['fetchpriority'];
30
+ delete attribs['fetchpriority'];
31
+ }
32
+
33
+ if (!attribs.loading) {
34
+ attribs.loading = 'lazy';
35
+ }
36
+
37
+ if (!attribs.decoding) {
38
+ attribs.decoding = 'async';
39
+ }
40
+
41
+ if (attribs.loading === 'lazy' && attribs.fetchPriority === 'high') {
42
+ delete attribs.fetchPriority;
43
+ }
44
+ }
45
+
46
+ type CmsImageMetadata = {
47
+ src: `/images/${string}`;
48
+ width: number;
49
+ height: number;
50
+ sizes: string;
51
+ priority?: boolean;
52
+ };
53
+
54
+ type HtmlImageProps = {
55
+ alt?: unknown;
56
+ className?: string;
57
+ id?: string;
58
+ sizes?: unknown;
59
+ style?: React.CSSProperties;
60
+ title?: string;
61
+ };
62
+
63
+ const inlineImageSizes =
64
+ '(max-width: 768px) calc(100vw - 2rem), (max-width: 1280px) 42vw, 512px';
65
+ const nbCoverImageSizes =
66
+ '(max-width: 768px) calc(100vw - 2rem), (max-width: 1280px) 38vw, 438px';
67
+ const contentImageSizes =
68
+ '(max-width: 768px) calc(100vw - 2rem), (max-width: 1280px) 75vw, 768px';
69
+
70
+ const knownCmsImages: Record<string, CmsImageMetadata> = {
71
+ 'images/NBcover.webp': {
72
+ src: '/images/NBcover.webp',
73
+ width: 1024,
74
+ height: 572,
75
+ sizes: nbCoverImageSizes,
76
+ priority: true,
77
+ },
78
+ 'images/cap.webp': {
79
+ src: '/images/cap.webp',
80
+ width: 960,
81
+ height: 960,
82
+ sizes: inlineImageSizes,
83
+ },
84
+ 'images/commerce-plan.webp': {
85
+ src: '/images/commerce-plan.webp',
86
+ width: 1024,
87
+ height: 559,
88
+ sizes: contentImageSizes,
89
+ },
90
+ 'images/commerce-square.webp': {
91
+ src: '/images/commerce-square.webp',
92
+ width: 2048,
93
+ height: 2048,
94
+ sizes: inlineImageSizes,
95
+ },
96
+ 'images/cortex-ai-square.webp': {
97
+ src: '/images/cortex-ai-square.webp',
98
+ width: 2048,
99
+ height: 2048,
100
+ sizes: inlineImageSizes,
101
+ },
102
+ 'images/commerce-wide.webp': {
103
+ src: '/images/commerce-wide.webp',
104
+ width: 1024,
105
+ height: 434,
106
+ sizes: contentImageSizes,
107
+ },
108
+ 'images/developer.webp': {
109
+ src: '/images/developer.webp',
110
+ width: 1024,
111
+ height: 1024,
112
+ sizes: '(max-width: 768px) calc(100vw - 2rem), 400px',
113
+ },
114
+ 'images/extensibility.webp': {
115
+ src: '/images/extensibility.webp',
116
+ width: 1024,
117
+ height: 559,
118
+ sizes: contentImageSizes,
119
+ },
120
+ 'images/goals.webp': {
121
+ src: '/images/goals.webp',
122
+ width: 1008,
123
+ height: 1024,
124
+ sizes: inlineImageSizes,
125
+ },
126
+ 'images/included.webp': {
127
+ src: '/images/included.webp',
128
+ width: 1024,
129
+ height: 559,
130
+ sizes: contentImageSizes,
131
+ },
132
+ 'images/metadata_image.webp': {
133
+ src: '/images/metadata_image.webp',
134
+ width: 1200,
135
+ height: 634,
136
+ sizes: contentImageSizes,
137
+ },
138
+ 'images/nextblock-logo-small.webp': {
139
+ src: '/images/nextblock-logo-small.webp',
140
+ width: 1162,
141
+ height: 1164,
142
+ sizes: '128px',
143
+ },
144
+ 'images/nx-graph.webp': {
145
+ src: '/images/nx-graph.webp',
146
+ width: 541,
147
+ height: 670,
148
+ sizes: contentImageSizes,
149
+ },
150
+ 'images/pants.webp': {
151
+ src: '/images/pants.webp',
152
+ width: 740,
153
+ height: 717,
154
+ sizes: inlineImageSizes,
155
+ },
156
+ 'images/programmer-upscaled.webp': {
157
+ src: '/images/programmer-upscaled.webp',
158
+ width: 8192,
159
+ height: 2632,
160
+ sizes: '(max-width: 768px) calc(100vw - 2rem), (max-width: 1280px) 75vw, 1200px',
161
+ },
162
+ 'images/t-shirt.webp': {
163
+ src: '/images/t-shirt.webp',
164
+ width: 740,
165
+ height: 717,
166
+ sizes: inlineImageSizes,
167
+ },
168
+ };
169
+
170
+ function normalizeImagePath(src: string) {
171
+ try {
172
+ return new URL(src, 'https://nextblock.local').pathname.replace(/^\/+/, '');
173
+ } catch {
174
+ return src.replace(/^\/+/, '');
175
+ }
12
176
  }
13
177
 
14
- const ClientTextBlockRenderer: React.FC<ClientTextBlockRendererProps> = ({ content, languageId }) => {
178
+ function getKnownCmsImage(src: string | undefined) {
179
+ if (!src) return null;
180
+
181
+ const imagePath = normalizeImagePath(src);
182
+ const directMatch = knownCmsImages[imagePath];
183
+
184
+ if (directMatch) {
185
+ return directMatch;
186
+ }
187
+
188
+ const entry = Object.entries(knownCmsImages).find(([knownPath]) =>
189
+ imagePath.endsWith(knownPath)
190
+ );
191
+
192
+ return entry?.[1] ?? null;
193
+ }
194
+
195
+ function renderOptimizedCmsImage(attribs: Record<string, string>) {
196
+ const image = getKnownCmsImage(attribs.src);
197
+
198
+ if (!image) {
199
+ normalizeImageAttributes(attribs);
200
+ return undefined;
201
+ }
202
+
203
+ const props = attributesToProps(attribs) as HtmlImageProps;
204
+ const alt = typeof props.alt === 'string' ? props.alt : '';
205
+ const sizes = typeof props.sizes === 'string' ? props.sizes : image.sizes;
206
+ const priority = image.priority === true;
207
+
208
+ return (
209
+ <Image
210
+ src={image.src}
211
+ alt={alt}
212
+ width={image.width}
213
+ height={image.height}
214
+ sizes={sizes}
215
+ quality={60}
216
+ priority={priority}
217
+ fetchPriority={priority ? 'high' : undefined}
218
+ loading={priority ? undefined : 'lazy'}
219
+ decoding="async"
220
+ className={props.className}
221
+ style={props.style}
222
+ title={props.title}
223
+ id={props.id}
224
+ />
225
+ );
226
+ }
227
+
228
+ const ClientTextBlockRenderer: React.FC<ClientTextBlockRendererProps> = ({
229
+ content,
230
+ languageId,
231
+ visualEditAttributes,
232
+ renderContext = 'prose',
233
+ }) => {
15
234
  void languageId;
235
+ const normalizedHtml = normalizeHtmlEncodingArtifacts(content.html_content || "");
236
+ const wrapperClassName =
237
+ renderContext === 'section'
238
+ ? 'w-full min-w-0'
239
+ : 'my-4 prose dark:prose-invert container mx-auto';
16
240
  const options: HTMLReactParserOptions = {
17
241
  replace: (domNode) => {
18
242
  if (domNode instanceof Element && domNode.attribs) {
19
- if (domNode.attribs['fetchpriority']) {
20
- domNode.attribs['fetchPriority'] = domNode.attribs['fetchpriority'];
243
+ // Clean up event handlers (like onclick) to prevent React warnings and XSS
244
+ for (const key of Object.keys(domNode.attribs)) {
245
+ if (key.toLowerCase().startsWith('on')) {
246
+ delete domNode.attribs[key];
247
+ }
248
+ }
249
+
250
+ if (domNode.name === 'img') {
251
+ return renderOptimizedCmsImage(domNode.attribs);
252
+ } else if (domNode.attribs['fetchpriority']) {
253
+ domNode.attribs.fetchPriority = domNode.attribs['fetchpriority'];
21
254
  delete domNode.attribs['fetchpriority'];
22
255
  }
23
256
 
@@ -64,9 +297,30 @@ const ClientTextBlockRenderer: React.FC<ClientTextBlockRendererProps> = ({ conte
64
297
  },
65
298
  };
66
299
 
300
+ const isJson = normalizedHtml.trim().startsWith('{') || normalizedHtml.trim().startsWith('[');
301
+
302
+ if (isJson) {
303
+ try {
304
+ JSON.parse(normalizedHtml);
305
+ return (
306
+ <div
307
+ className={wrapperClassName}
308
+ {...visualEditAttributes}
309
+ >
310
+ <SimpleTiptapRenderer content={normalizedHtml} />
311
+ </div>
312
+ );
313
+ } catch {
314
+ // Fallback to parse as HTML
315
+ }
316
+ }
317
+
67
318
  return (
68
- <div className="my-4 prose dark:prose-invert container mx-auto">
69
- {parse(content.html_content || "", options)}
319
+ <div
320
+ className={wrapperClassName}
321
+ {...visualEditAttributes}
322
+ >
323
+ {parse(normalizedHtml, options)}
70
324
  </div>
71
325
  );
72
326
  };
@@ -0,0 +1,22 @@
1
+ import React from 'react';
2
+ import { FeaturedProductBlock } from '../../../lib/blocks/FeaturedProductBlock';
3
+ import type { FeaturedProductBlockContent } from '../../../lib/blocks/ecommerce-block-schemas';
4
+ import type { VisualEditAttributes } from '../../../lib/visual-editing/types';
5
+
6
+ interface FeaturedProductBlockRendererProps {
7
+ content: FeaturedProductBlockContent;
8
+ languageId: number;
9
+ visualEditAttributes?: VisualEditAttributes;
10
+ }
11
+
12
+ export default function FeaturedProductBlockRenderer({
13
+ content,
14
+ visualEditAttributes,
15
+ }: FeaturedProductBlockRendererProps) {
16
+
17
+ return (
18
+ <div {...visualEditAttributes}>
19
+ <FeaturedProductBlock content={content} />
20
+ </div>
21
+ );
22
+ }