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,16 @@
1
+ 'use client';
2
+
3
+ import dynamic from 'next/dynamic';
4
+
5
+ export interface EditorProps {
6
+ initialContent?: any;
7
+ onUpdate?: (content: any) => void;
8
+ showAiPrompt?: boolean;
9
+ }
10
+
11
+ export const ClientNotionEditor = dynamic<EditorProps>(
12
+ () => import('@nextblock-cms/editor').then((mod) => mod.NotionEditor as any),
13
+ { ssr: false }
14
+ );
15
+
16
+ export default ClientNotionEditor;
@@ -0,0 +1,56 @@
1
+ 'use client';
2
+
3
+ import React from 'react';
4
+ import { ProductForm } from '@nextblock-cms/ecommerce';
5
+ import MediaPickerDialog from '../media/components/MediaPickerDialog';
6
+ type ProductFormClientShellProps = React.ComponentProps<typeof ProductForm>;
7
+
8
+ const productFormSkeletonRows = ['details', 'description', 'media', 'inventory'];
9
+
10
+ export default function ProductFormClientShell(props: ProductFormClientShellProps) {
11
+ const [isMounted, setIsMounted] = React.useState(false);
12
+
13
+ React.useEffect(() => {
14
+ setIsMounted(true);
15
+ }, []);
16
+
17
+ if (!isMounted) {
18
+ return <ProductFormShellSkeleton />;
19
+ }
20
+
21
+ return (
22
+ <ProductForm
23
+ {...props}
24
+ mediaPickerNode={
25
+ <MediaPickerDialog
26
+ triggerLabel="+ Add Image"
27
+ triggerVariant="outline"
28
+ defaultFolder="uploads/products/"
29
+ />
30
+ }
31
+ />
32
+ );
33
+ }
34
+
35
+ function ProductFormShellSkeleton() {
36
+ return (
37
+ <div className="space-y-8" aria-hidden="true">
38
+ <div className="space-y-3">
39
+ <div className="h-8 w-56 rounded-md bg-muted" />
40
+ <div className="h-4 w-80 max-w-full rounded-md bg-muted/70" />
41
+ </div>
42
+
43
+ <div className="grid gap-4 md:grid-cols-2">
44
+ <div className="h-24 rounded-lg border bg-muted/30" />
45
+ <div className="h-24 rounded-lg border bg-muted/30" />
46
+ </div>
47
+
48
+ {productFormSkeletonRows.map((row) => (
49
+ <div key={row} className="rounded-lg border p-4">
50
+ <div className="mb-4 h-5 w-40 rounded-md bg-muted" />
51
+ <div className="h-32 rounded-md bg-muted/30" />
52
+ </div>
53
+ ))}
54
+ </div>
55
+ );
56
+ }
@@ -0,0 +1,292 @@
1
+ import { verifyPackageOnline, getActiveLanguagesServerSide } from '@nextblock-cms/db/server';
2
+ import { redirect, notFound } from 'next/navigation';
3
+ import Link from 'next/link';
4
+ import { ArrowLeft, ChevronDown, ExternalLink } from 'lucide-react';
5
+ import {
6
+ Badge,
7
+ Button,
8
+ DropdownMenu,
9
+ DropdownMenuContent,
10
+ DropdownMenuItem,
11
+ DropdownMenuLabel,
12
+ DropdownMenuSeparator,
13
+ DropdownMenuTrigger,
14
+ } from '@nextblock-cms/ui';
15
+ import ProductFormClientShell from '../../ProductFormClientShell';
16
+ import DraftStatusActions from '../../../components/DraftStatusActions';
17
+ import {
18
+ getCmsProduct,
19
+ getEnabledPaymentProviders,
20
+ getGlobalProductAttributes,
21
+ getProductTranslations,
22
+ getStoreConfigStatus,
23
+ normalizeCurrencyRecord,
24
+ updateProductAction,
25
+ getCategoriesWithCount,
26
+ getProductCategories,
27
+ } from '@nextblock-cms/ecommerce/server';
28
+ import { createClient } from '@nextblock-cms/db/server';
29
+ import {
30
+ buildGlobalAttributesForForm,
31
+ buildProductFormInitialData,
32
+ } from '../../productFormData';
33
+ import { CortexAiPageContextRegistrar } from '../../../components/CortexAiPageContext';
34
+ import BlockEditorArea from '../../../blocks/components/BlockEditorArea';
35
+
36
+ export default async function EditProductPage({
37
+ params,
38
+ searchParams,
39
+ }: {
40
+ params: Promise<{ id: string }>;
41
+ searchParams: Promise<{ missing_lang_id?: string }>;
42
+ }) {
43
+ const [
44
+ { id },
45
+ { missing_lang_id },
46
+ isOnline,
47
+ languages,
48
+ enabledProviders,
49
+ configStatus,
50
+ ] =
51
+ await Promise.all([
52
+ params,
53
+ searchParams,
54
+ verifyPackageOnline('ecommerce'),
55
+ getActiveLanguagesServerSide(),
56
+ getEnabledPaymentProviders(),
57
+ getStoreConfigStatus(),
58
+ ]);
59
+
60
+ if (!isOnline) {
61
+ redirect('/cms/settings/packages');
62
+ }
63
+
64
+ const product = await getCmsProduct(id);
65
+
66
+ if (!product) {
67
+ notFound();
68
+ }
69
+
70
+ const [globalAttributesRaw, translations, allCategories, assignedCategories] = await Promise.all([
71
+ getGlobalProductAttributes(),
72
+ product.translation_group_id ? getProductTranslations(product.translation_group_id) : Promise.resolve([]),
73
+ getCategoriesWithCount(),
74
+ getProductCategories(product.id),
75
+ ]);
76
+ const supabase = createClient();
77
+ const { data: currenciesResult } = await supabase
78
+ .from('currencies')
79
+ .select(
80
+ 'code, symbol, exchange_rate, is_default, is_active, auto_sync_product_prices, auto_update_exchange_rate, exchange_rate_source, exchange_rate_updated_at, rounding_mode, rounding_increment, rounding_charm_amount'
81
+ )
82
+ .eq('is_active', true)
83
+ .order('code', { ascending: true });
84
+ const currencies = (currenciesResult ?? []).map((currency) =>
85
+ normalizeCurrencyRecord(currency)
86
+ );
87
+
88
+ const missingLanguageId = missing_lang_id ? parseInt(missing_lang_id, 10) : null;
89
+ const missingLanguage =
90
+ missingLanguageId && Number.isFinite(missingLanguageId)
91
+ ? languages.find((language) => language.id === missingLanguageId)
92
+ : null;
93
+ const translationByLanguageId = new Map(
94
+ translations.map((translation: any) => [translation.language_id, translation])
95
+ );
96
+ const existingLanguages = languages.filter(
97
+ (language) => language.id === product.language_id || translationByLanguageId.has(language.id)
98
+ );
99
+ const missingLanguages = languages.filter(
100
+ (language) => language.id !== product.language_id && !translationByLanguageId.has(language.id)
101
+ );
102
+ const primaryCreateLanguage =
103
+ missingLanguage && missingLanguages.some((language) => language.id === missingLanguage.id)
104
+ ? missingLanguage
105
+ : missingLanguages.length === 1
106
+ ? missingLanguages[0]
107
+ : null;
108
+ const additionalCreateLanguages = missingLanguages.filter(
109
+ (language) => language.id !== primaryCreateLanguage?.id
110
+ );
111
+ const buildTranslationCreateHref = (languageId: number) =>
112
+ `/cms/products/new?from_group=${product.translation_group_id}&target_lang_id=${languageId}`;
113
+ const globalAttributes = buildGlobalAttributesForForm(globalAttributesRaw || []);
114
+
115
+ const { data: draftData } = await supabase
116
+ .from('product_drafts')
117
+ .select('*')
118
+ .eq('product_id', product.id)
119
+ .maybeSingle();
120
+
121
+ const hasDraft = draftData !== null;
122
+ let normalizedInitialData = {
123
+ ...buildProductFormInitialData(product, languages),
124
+ category_ids: assignedCategories.map((c: any) => c.id),
125
+ };
126
+ if (draftData && draftData.meta && typeof draftData.meta === 'object') {
127
+ const meta = draftData.meta as any;
128
+
129
+ // Drafts store product_media as { media_id } only, so the gallery loses the
130
+ // file_path/object_key needed to render thumbnails. Re-hydrate from the
131
+ // media table so the main image (and the rest of the gallery) still loads.
132
+ let hydratedProductMedia = meta.product_media;
133
+ if (Array.isArray(meta.product_media) && meta.product_media.length > 0) {
134
+ const draftMediaIds = meta.product_media
135
+ .map((pm: any) => pm?.media_id)
136
+ .filter(Boolean);
137
+ if (draftMediaIds.length > 0) {
138
+ const { data: mediaRows } = await supabase
139
+ .from('media')
140
+ .select('id, file_path, object_key, file_name, description')
141
+ .in('id', draftMediaIds);
142
+ const mediaById = new Map((mediaRows || []).map((m: any) => [m.id, m]));
143
+ hydratedProductMedia = meta.product_media.map((pm: any, index: number) => {
144
+ const media = mediaById.get(pm?.media_id);
145
+ return {
146
+ media_id: pm?.media_id,
147
+ sort_order: pm?.sort_order ?? index,
148
+ media: media
149
+ ? {
150
+ file_path: media.file_path,
151
+ object_key: media.object_key,
152
+ alt_text: media.description || media.file_name || '',
153
+ }
154
+ : pm?.media ?? null,
155
+ };
156
+ });
157
+ }
158
+ }
159
+
160
+ normalizedInitialData = {
161
+ ...meta,
162
+ id: product.id,
163
+ product_media: hydratedProductMedia,
164
+ category_ids: meta.category_ids ?? assignedCategories.map((c: any) => c.id),
165
+ };
166
+ }
167
+
168
+ let descriptionBlocks: any[] = [];
169
+ if (draftData && draftData.blocks && Array.isArray(draftData.blocks)) {
170
+ descriptionBlocks = draftData.blocks;
171
+ } else {
172
+ const { data: liveBlocks } = await supabase
173
+ .from('blocks')
174
+ .select('*')
175
+ .eq('product_id', product.id)
176
+ .order('order', { ascending: true });
177
+ descriptionBlocks = liveBlocks || [];
178
+ }
179
+
180
+ return (
181
+ <div className="space-y-8 w-full max-w-[1400px] mx-auto px-6 py-8">
182
+ <DraftStatusActions parentId={product.id} parentType="product" hasDraft={hasDraft} />
183
+ <CortexAiPageContextRegistrar
184
+ context={{
185
+ contentType: 'product',
186
+ entityId: product.id,
187
+ languageId: product.language_id,
188
+ slug: product.slug,
189
+ title: product.title,
190
+ translationGroupId: product.translation_group_id,
191
+ }}
192
+ />
193
+ <div className="flex justify-between items-center flex-wrap gap-4 w-full">
194
+ <div className="flex items-center gap-3">
195
+ <Button variant="outline" size="icon" aria-label="Back to products" asChild>
196
+ <Link href="/cms/products">
197
+ <ArrowLeft className="h-4 w-4" />
198
+ </Link>
199
+ </Button>
200
+ <div>
201
+ <h1 className="text-2xl font-bold">Edit Product</h1>
202
+ <p className="text-sm text-muted-foreground truncate max-w-md" title={product.title}>
203
+ {product.title}
204
+ </p>
205
+ </div>
206
+ </div>
207
+
208
+ <div className="flex items-center gap-3 flex-wrap">
209
+ {existingLanguages.map((language) => {
210
+ const version = translationByLanguageId.get(language.id);
211
+ const isCurrent = language.id === product.language_id;
212
+ const href = version
213
+ ? `/cms/products/${version.id}/edit`
214
+ : `/cms/products/${product.id}/edit`;
215
+
216
+ return (
217
+ <Button key={language.id} asChild variant={isCurrent ? 'default' : 'outline'} size="sm">
218
+ <Link href={href}>
219
+ {language.name} ({language.code.toUpperCase()})
220
+ </Link>
221
+ </Button>
222
+ );
223
+ })}
224
+
225
+ {primaryCreateLanguage && product.translation_group_id ? (
226
+ <Button asChild variant="secondary" size="sm">
227
+ <Link href={buildTranslationCreateHref(primaryCreateLanguage.id)}>
228
+ Create {primaryCreateLanguage.name} Translation
229
+ </Link>
230
+ </Button>
231
+ ) : null}
232
+
233
+ {additionalCreateLanguages.length > 0 && product.translation_group_id ? (
234
+ <DropdownMenu>
235
+ <DropdownMenuTrigger asChild>
236
+ <Button id={`create-translation-trigger-${product.id}`} variant="outline" size="sm">
237
+ Create Translation
238
+ <Badge variant="secondary" className="ml-2 px-1.5 py-0 text-[10px]">
239
+ {additionalCreateLanguages.length}
240
+ </Badge>
241
+ <ChevronDown className="ml-2 h-4 w-4" />
242
+ </Button>
243
+ </DropdownMenuTrigger>
244
+ <DropdownMenuContent align="end" className="w-64">
245
+ <DropdownMenuLabel>Missing Languages</DropdownMenuLabel>
246
+ <DropdownMenuSeparator />
247
+ {additionalCreateLanguages.map((language) => (
248
+ <DropdownMenuItem key={language.id} asChild>
249
+ <Link href={buildTranslationCreateHref(language.id)}>
250
+ Create {language.name}
251
+ </Link>
252
+ </DropdownMenuItem>
253
+ ))}
254
+ </DropdownMenuContent>
255
+ </DropdownMenu>
256
+ ) : null}
257
+
258
+ {product.slug && product.status === 'active' && (
259
+ <Button variant="outline" asChild>
260
+ <Link href={`/product/${product.slug}`} target="_blank">
261
+ <ExternalLink className="w-4 h-4 mr-2" />
262
+ View
263
+ </Link>
264
+ </Button>
265
+ )}
266
+ </div>
267
+ </div>
268
+
269
+ <ProductFormClientShell
270
+ initialData={normalizedInitialData}
271
+ isEdit
272
+ availableLanguagesProp={languages}
273
+ globalAttributesProp={globalAttributes}
274
+ currenciesProp={currencies}
275
+ enabledProviders={enabledProviders}
276
+ configStatus={configStatus}
277
+ updateAction={updateProductAction.bind(null, product.id)}
278
+ availableCategoriesProp={allCategories}
279
+ />
280
+
281
+ <div className="border-t pt-8">
282
+ <h2 className="text-xl font-bold mb-4">Product Description Blocks</h2>
283
+ <BlockEditorArea
284
+ parentId={product.id}
285
+ parentType="product"
286
+ initialBlocks={descriptionBlocks}
287
+ languageId={product.language_id}
288
+ />
289
+ </div>
290
+ </div>
291
+ );
292
+ }
@@ -0,0 +1,12 @@
1
+ import { redirect } from 'next/navigation';
2
+ import { verifyPackageOnline } from '@nextblock-cms/db/server';
3
+ import { AttributeManagementPage as AttributeManagementPageUI } from '@nextblock-cms/ecommerce/server';
4
+
5
+ export default async function AttributeManagementPage() {
6
+ const isOnline = await verifyPackageOnline('ecommerce');
7
+ if (!isOnline) {
8
+ redirect('/cms/settings/packages');
9
+ }
10
+
11
+ return <AttributeManagementPageUI />;
12
+ }
@@ -0,0 +1,12 @@
1
+ import { redirect } from 'next/navigation';
2
+ import { verifyPackageOnline } from '@nextblock-cms/db/server';
3
+ import { CategoryManagementPage as CategoryManagementPageUI } from '@nextblock-cms/ecommerce/server';
4
+
5
+ export default async function CategoryManagementPage() {
6
+ const isOnline = await verifyPackageOnline('ecommerce');
7
+ if (!isOnline) {
8
+ redirect('/cms/settings/packages');
9
+ }
10
+
11
+ return <CategoryManagementPageUI />;
12
+ }
@@ -0,0 +1,13 @@
1
+ import { verifyPackageOnline } from '@nextblock-cms/db/server';
2
+ import { InventoryPage as InventoryPageUI } from '@nextblock-cms/ecommerce/server';
3
+ import { redirect } from 'next/navigation';
4
+
5
+ export default async function InventoryPage() {
6
+ const isOnline = await verifyPackageOnline('ecommerce');
7
+
8
+ if (!isOnline) {
9
+ redirect('/cms/settings/packages');
10
+ }
11
+
12
+ return <InventoryPageUI />;
13
+ }
@@ -0,0 +1,143 @@
1
+ import { verifyPackageOnline } from '@nextblock-cms/db/server';
2
+ import { redirect } from 'next/navigation';
3
+ import { getActiveLanguagesServerSide } from '@nextblock-cms/db/server';
4
+ import { createClient } from '@nextblock-cms/db/server';
5
+ import Link from 'next/link';
6
+ import {
7
+ createProductAction,
8
+ getGlobalProductAttributes,
9
+ getEnabledPaymentProviders,
10
+ getProduct,
11
+ getStoreConfigStatus,
12
+ normalizeCurrencyRecord,
13
+ getCategoriesWithCount,
14
+ } from '@nextblock-cms/ecommerce/server';
15
+ import { ArrowLeft } from 'lucide-react';
16
+ import { Badge, Button } from '@nextblock-cms/ui';
17
+ import ProductFormClientShell from '../ProductFormClientShell';
18
+ import {
19
+ buildGlobalAttributesForForm,
20
+ buildProductFormInitialData,
21
+ buildTranslationSourceInitialData,
22
+ } from '../productFormData';
23
+
24
+ export default async function NewProductPage({
25
+ searchParams
26
+ }: {
27
+ searchParams: Promise<{ from_group?: string; target_lang_id?: string }>
28
+ }) {
29
+ const [
30
+ isOnline,
31
+ languages,
32
+ enabledProviders,
33
+ configStatus,
34
+ globalAttributesRaw,
35
+ currenciesResult,
36
+ categories,
37
+ { from_group, target_lang_id },
38
+ ] = await Promise.all([
39
+ verifyPackageOnline('ecommerce'),
40
+ getActiveLanguagesServerSide(),
41
+ getEnabledPaymentProviders(),
42
+ getStoreConfigStatus(),
43
+ getGlobalProductAttributes(),
44
+ createClient()
45
+ .from('currencies')
46
+ .select(
47
+ 'code, symbol, exchange_rate, is_default, is_active, auto_sync_product_prices, auto_update_exchange_rate, exchange_rate_source, exchange_rate_updated_at, rounding_mode, rounding_increment, rounding_charm_amount'
48
+ )
49
+ .eq('is_active', true)
50
+ .order('code', { ascending: true }),
51
+ getCategoriesWithCount(),
52
+ searchParams,
53
+ ]);
54
+
55
+ if (!isOnline) {
56
+ redirect('/cms/settings/packages');
57
+ }
58
+
59
+ let initialData = null;
60
+ if (from_group) {
61
+ try {
62
+ const supabase = createClient();
63
+ const { data: groupProducts } = await supabase
64
+ .from('products')
65
+ .select('id')
66
+ .eq('translation_group_id', from_group)
67
+ .limit(1);
68
+
69
+ if (groupProducts && groupProducts[0]) {
70
+ const { data: sourceProduct, error: fetchError } = await getProduct(supabase, groupProducts[0].id);
71
+ if (sourceProduct && !fetchError) {
72
+ initialData = buildTranslationSourceInitialData(
73
+ sourceProduct,
74
+ from_group,
75
+ target_lang_id
76
+ );
77
+ }
78
+ }
79
+ } catch (e) {
80
+ console.error('Error pre-filling translation data:', e);
81
+ }
82
+ }
83
+
84
+ const targetLanguageId = target_lang_id ? Number.parseInt(target_lang_id, 10) : undefined;
85
+ const targetLanguage =
86
+ Number.isFinite(targetLanguageId)
87
+ ? languages.find((language) => language.id === targetLanguageId)
88
+ : null;
89
+ const globalAttributes = buildGlobalAttributesForForm(globalAttributesRaw || []);
90
+ const currencies = (currenciesResult.data ?? []).map((currency) =>
91
+ normalizeCurrencyRecord(currency)
92
+ );
93
+ const normalizedInitialData = buildProductFormInitialData(
94
+ initialData,
95
+ languages,
96
+ targetLanguageId
97
+ );
98
+
99
+ return (
100
+ <div className="space-y-8 w-full max-w-[1400px] mx-auto px-6 py-8">
101
+ <div className="flex justify-between items-start flex-wrap gap-4 w-full">
102
+ <div className="flex items-start gap-3">
103
+ <Button variant="outline" size="icon" aria-label="Back to products" asChild>
104
+ <Link href="/cms/products">
105
+ <ArrowLeft className="h-4 w-4" />
106
+ </Link>
107
+ </Button>
108
+ <div className="space-y-2">
109
+ <div className="flex items-center gap-2 flex-wrap">
110
+ <h1 className="text-2xl font-bold">
111
+ {targetLanguage ? `Create ${targetLanguage.name} Translation` : 'Create Product'}
112
+ </h1>
113
+ {targetLanguage ? (
114
+ <Badge variant="secondary">
115
+ {targetLanguage.code.toUpperCase()}
116
+ </Badge>
117
+ ) : null}
118
+ </div>
119
+ <p className="text-sm text-muted-foreground max-w-3xl">
120
+ {targetLanguage
121
+ ? 'We prefilled this draft from the source product so you can localize the content and fine-tune pricing, media, and variations faster.'
122
+ : 'Build a catalog-ready product with structured information, media, pricing, inventory, and optional variations.'}
123
+ </p>
124
+ </div>
125
+ </div>
126
+ </div>
127
+
128
+ <ProductFormClientShell
129
+ availableLanguagesProp={languages}
130
+ globalAttributesProp={globalAttributes}
131
+ currenciesProp={currencies}
132
+ translationGroupId={from_group}
133
+ targetLanguageId={target_lang_id}
134
+ initialData={normalizedInitialData}
135
+ enabledProviders={enabledProviders}
136
+ configStatus={configStatus}
137
+ createAction={createProductAction}
138
+ availableCategoriesProp={categories}
139
+ />
140
+ </div>
141
+ );
142
+ }
143
+
@@ -0,0 +1,42 @@
1
+ import { ProductsPage as ProductsPageUI } from '@nextblock-cms/ecommerce/server';
2
+ import { verifyPackageOnline } from '@nextblock-cms/db/server';
3
+ import { redirect } from 'next/navigation';
4
+
5
+ import { getActiveLanguagesServerSide } from '@nextblock-cms/db/server';
6
+ import LanguageFilterSelect from '../components/LanguageFilterSelect';
7
+ import { ContentTransferControls } from '../import-export/ContentTransferControls';
8
+
9
+ export default async function ProductsPage({ searchParams }: { searchParams: Promise<{ lang?: string }> }) {
10
+ const isOnline = await verifyPackageOnline('ecommerce');
11
+ if (!isOnline) {
12
+ redirect('/cms/settings/packages');
13
+ }
14
+
15
+ const resolvedSearchParams = await searchParams;
16
+ const allLanguages = await getActiveLanguagesServerSide();
17
+ const selectedLangId = resolvedSearchParams?.lang ? parseInt(resolvedSearchParams.lang, 10) : undefined;
18
+ const isValidLangId = selectedLangId
19
+ ? allLanguages.some((language) => language.id === selectedLangId)
20
+ : true;
21
+ const filterLangId = isValidLangId ? selectedLangId : undefined;
22
+
23
+ return (
24
+ <ProductsPageUI
25
+ searchParams={resolvedSearchParams}
26
+ transferControlsNode={
27
+ <ContentTransferControls
28
+ contentType="products"
29
+ label="Products"
30
+ languageId={filterLangId}
31
+ />
32
+ }
33
+ languageFilterNode={
34
+ <LanguageFilterSelect
35
+ allLanguages={allLanguages}
36
+ currentFilterLangId={filterLangId}
37
+ basePath="/cms/products"
38
+ />
39
+ }
40
+ />
41
+ );
42
+ }