create-nextblock 0.2.78 → 0.8.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (413) hide show
  1. package/bin/create-nextblock.js +740 -459
  2. package/package.json +1 -2
  3. package/scripts/sync-template.js +18 -1
  4. package/templates/nextblock-template/.browserslistrc +11 -0
  5. package/templates/nextblock-template/.swcrc +30 -30
  6. package/templates/nextblock-template/README.md +23 -114
  7. package/templates/nextblock-template/app/(auth-pages)/post-sign-in/page.tsx +27 -28
  8. package/templates/nextblock-template/app/(auth-pages)/sign-in/page.tsx +50 -25
  9. package/templates/nextblock-template/app/(auth-pages)/sign-up/page.tsx +111 -56
  10. package/templates/nextblock-template/app/(auth-pages)/two-factor/actions.ts +91 -0
  11. package/templates/nextblock-template/app/(auth-pages)/two-factor/components/TwoFactorForm.tsx +118 -0
  12. package/templates/nextblock-template/app/(auth-pages)/two-factor/page.tsx +51 -0
  13. package/templates/nextblock-template/app/.well-known/ucp/route.ts +16 -0
  14. package/templates/nextblock-template/app/[slug]/PageClientContent.tsx +48 -28
  15. package/templates/nextblock-template/app/[slug]/page.tsx +63 -6
  16. package/templates/nextblock-template/app/[slug]/page.utils.ts +374 -157
  17. package/templates/nextblock-template/app/[slug]/pageClientActions.ts +7 -0
  18. package/templates/nextblock-template/app/actions/consent.ts +57 -0
  19. package/templates/nextblock-template/app/actions/formActions.ts +130 -11
  20. package/templates/nextblock-template/app/actions/languageActions.ts +31 -30
  21. package/templates/nextblock-template/app/actions/package-actions.ts +183 -0
  22. package/templates/nextblock-template/app/actions/postActions.ts +146 -48
  23. package/templates/nextblock-template/app/actions/twoFactorEmail.ts +21 -0
  24. package/templates/nextblock-template/app/actions/visualEditingActions.test.ts +179 -0
  25. package/templates/nextblock-template/app/actions/visualEditingActions.ts +345 -0
  26. package/templates/nextblock-template/app/actions.ts +67 -12
  27. package/templates/nextblock-template/app/api/ai/cortex/build-widget/route.ts +153 -0
  28. package/templates/nextblock-template/app/api/ai/generate-blocks/route.ts +96 -0
  29. package/templates/nextblock-template/app/api/ai/global-agent/route.ts +965 -0
  30. package/templates/nextblock-template/app/api/checkout/freemius/sync/route.ts +29 -0
  31. package/templates/nextblock-template/app/api/checkout/route.ts +146 -0
  32. package/templates/nextblock-template/app/api/cms/full-backup/export/route.ts +33 -0
  33. package/templates/nextblock-template/app/api/cms/full-backup/restore/route.ts +63 -0
  34. package/templates/nextblock-template/app/api/cron/reset-sandbox/route.ts +3413 -17
  35. package/templates/nextblock-template/app/api/cron/reset-sandbox/sandboxResetSql.ts +7830 -0
  36. package/templates/nextblock-template/app/api/cron/sync-currencies/route.ts +35 -0
  37. package/templates/nextblock-template/app/api/custom-blocks/db-relations/route.ts +92 -0
  38. package/templates/nextblock-template/app/api/custom-blocks/editor-definitions/route.ts +43 -0
  39. package/templates/nextblock-template/app/api/draft/disable/route.ts +25 -0
  40. package/templates/nextblock-template/app/api/draft/route.ts +93 -0
  41. package/templates/nextblock-template/app/api/draft/start/route.ts +77 -0
  42. package/templates/nextblock-template/app/api/media/library/route.ts +65 -0
  43. package/templates/nextblock-template/app/api/media/r2-presigned/route.ts +53 -0
  44. package/templates/nextblock-template/app/api/media/record/route.ts +160 -0
  45. package/templates/nextblock-template/app/api/search/route.ts +43 -0
  46. package/templates/nextblock-template/app/api/visual-editing/block-draft/route.ts +47 -0
  47. package/templates/nextblock-template/app/api/visual-editing/product-draft/route.ts +47 -0
  48. package/templates/nextblock-template/app/api/webhooks/freemius/route.ts +34 -0
  49. package/templates/nextblock-template/app/api/webhooks/stripe/route.ts +27 -0
  50. package/templates/nextblock-template/app/article/[slug]/PostClientContent.tsx +392 -128
  51. package/templates/nextblock-template/app/article/[slug]/page.tsx +179 -127
  52. package/templates/nextblock-template/app/article/[slug]/page.utils.ts +262 -77
  53. package/templates/nextblock-template/app/auth/callback/route.ts +31 -58
  54. package/templates/nextblock-template/app/cart/page.tsx +7 -0
  55. package/templates/nextblock-template/app/checkout/UcpCartHydrator.tsx +20 -0
  56. package/templates/nextblock-template/app/checkout/page.tsx +52 -0
  57. package/templates/nextblock-template/app/checkout/success/actions.ts +136 -0
  58. package/templates/nextblock-template/app/checkout/success/page.tsx +186 -0
  59. package/templates/nextblock-template/app/cms/CmsClientLayout.tsx +163 -33
  60. package/templates/nextblock-template/app/cms/blocks/actions.ts +424 -235
  61. package/templates/nextblock-template/app/cms/blocks/components/BackgroundSelector.tsx +212 -151
  62. package/templates/nextblock-template/app/cms/blocks/components/BlockEditorArea.tsx +41 -20
  63. package/templates/nextblock-template/app/cms/blocks/components/BlockEditorModal.tsx +152 -19
  64. package/templates/nextblock-template/app/cms/blocks/components/BlockTypeCard.tsx +25 -17
  65. package/templates/nextblock-template/app/cms/blocks/components/BlockTypeSelector.tsx +200 -18
  66. package/templates/nextblock-template/app/cms/blocks/components/ColumnEditor.tsx +33 -16
  67. package/templates/nextblock-template/app/cms/blocks/components/CustomBlockEditorPreview.tsx +160 -0
  68. package/templates/nextblock-template/app/cms/blocks/components/EditableBlock.tsx +37 -18
  69. package/templates/nextblock-template/app/cms/blocks/components/MediaLibraryModal.tsx +149 -67
  70. package/templates/nextblock-template/app/cms/blocks/components/SectionConfigPanel.tsx +108 -31
  71. package/templates/nextblock-template/app/cms/blocks/editors/DynamicCustomBlockEditor.tsx +167 -0
  72. package/templates/nextblock-template/app/cms/blocks/editors/FeaturedProductBlockEditor.tsx +31 -0
  73. package/templates/nextblock-template/app/cms/blocks/editors/FormBlockEditor.tsx +2 -2
  74. package/templates/nextblock-template/app/cms/blocks/editors/HeadingBlockEditor.tsx +1 -1
  75. package/templates/nextblock-template/app/cms/blocks/editors/ImageBlockEditor.tsx +29 -29
  76. package/templates/nextblock-template/app/cms/blocks/editors/PostsGridBlockEditor.tsx +14 -18
  77. package/templates/nextblock-template/app/cms/blocks/editors/ProductGridBlockEditor.tsx +41 -0
  78. package/templates/nextblock-template/app/cms/blocks/editors/SectionBlockEditor.tsx +318 -118
  79. package/templates/nextblock-template/app/cms/blocks/editors/TextBlockEditor.tsx +98 -21
  80. package/templates/nextblock-template/app/cms/blocks/editors/VideoEmbedBlockEditor.tsx +1 -1
  81. package/templates/nextblock-template/app/cms/components/ContentLanguageSwitcher.tsx +27 -9
  82. package/templates/nextblock-template/app/cms/components/CopyContentFromLanguage.tsx +1 -1
  83. package/templates/nextblock-template/app/cms/components/CortexAiActiveContext.tsx +23 -0
  84. package/templates/nextblock-template/app/cms/components/CortexAiPageContext.tsx +58 -0
  85. package/templates/nextblock-template/app/cms/components/CortexGlobalAgentChat.tsx +1507 -0
  86. package/templates/nextblock-template/app/cms/components/DraftStatusActions.tsx +145 -0
  87. package/templates/nextblock-template/app/cms/components/FeatureImageField.tsx +244 -0
  88. package/templates/nextblock-template/app/cms/components/FeedbackModal.tsx +38 -24
  89. package/templates/nextblock-template/app/cms/coupons/[id]/edit/page.tsx +16 -0
  90. package/templates/nextblock-template/app/cms/coupons/page.tsx +16 -0
  91. package/templates/nextblock-template/app/cms/custom-blocks/[id]/edit/page.tsx +66 -0
  92. package/templates/nextblock-template/app/cms/custom-blocks/actions.ts +519 -0
  93. package/templates/nextblock-template/app/cms/custom-blocks/components/BlockComposer.tsx +1522 -0
  94. package/templates/nextblock-template/app/cms/custom-blocks/components/BlocksLibraryTransferControls.tsx +256 -0
  95. package/templates/nextblock-template/app/cms/custom-blocks/components/DBRelationSelect.tsx +384 -0
  96. package/templates/nextblock-template/app/cms/custom-blocks/components/ImageR2Picker.tsx +221 -0
  97. package/templates/nextblock-template/app/cms/custom-blocks/new/page.tsx +12 -0
  98. package/templates/nextblock-template/app/cms/custom-blocks/page.tsx +438 -0
  99. package/templates/nextblock-template/app/cms/dashboard/actions.ts +228 -98
  100. package/templates/nextblock-template/app/cms/dashboard/components/DashboardComponents.tsx +200 -0
  101. package/templates/nextblock-template/app/cms/dashboard/page.tsx +182 -154
  102. package/templates/nextblock-template/app/cms/import-export/ContentTransferControls.tsx +391 -0
  103. package/templates/nextblock-template/app/cms/import-export/actions.ts +226 -0
  104. package/templates/nextblock-template/app/cms/layout.tsx +29 -10
  105. package/templates/nextblock-template/app/cms/media/UploadFolderContext.tsx +22 -22
  106. package/templates/nextblock-template/app/cms/media/actions.ts +45 -124
  107. package/templates/nextblock-template/app/cms/media/components/DeleteMediaButtonClient.tsx +1 -1
  108. package/templates/nextblock-template/app/cms/media/components/MediaEditForm.tsx +26 -26
  109. package/templates/nextblock-template/app/cms/media/components/MediaGridClient.tsx +69 -64
  110. package/templates/nextblock-template/app/cms/media/components/MediaPickerDialog.tsx +227 -158
  111. package/templates/nextblock-template/app/cms/media/components/MediaUploadForm.tsx +101 -89
  112. package/templates/nextblock-template/app/cms/media/page.tsx +1 -1
  113. package/templates/nextblock-template/app/cms/navigation/components/NavigationItemForm.tsx +2 -2
  114. package/templates/nextblock-template/app/cms/orders/[id]/MarkPaidButton.tsx +44 -0
  115. package/templates/nextblock-template/app/cms/orders/[id]/page.tsx +16 -0
  116. package/templates/nextblock-template/app/cms/orders/actions.ts +201 -0
  117. package/templates/nextblock-template/app/cms/orders/page.tsx +20 -0
  118. package/templates/nextblock-template/app/cms/orders/types.ts +20 -0
  119. package/templates/nextblock-template/app/cms/pages/[id]/edit/EditPageClient.tsx +156 -121
  120. package/templates/nextblock-template/app/cms/pages/[id]/edit/page.tsx +79 -26
  121. package/templates/nextblock-template/app/cms/pages/actions.ts +54 -38
  122. package/templates/nextblock-template/app/cms/pages/components/DeletePageButtonClient.tsx +1 -1
  123. package/templates/nextblock-template/app/cms/pages/components/PageForm.tsx +267 -116
  124. package/templates/nextblock-template/app/cms/pages/page.tsx +25 -18
  125. package/templates/nextblock-template/app/cms/payments/page.tsx +16 -0
  126. package/templates/nextblock-template/app/cms/posts/[id]/edit/page.tsx +132 -90
  127. package/templates/nextblock-template/app/cms/posts/actions.ts +71 -72
  128. package/templates/nextblock-template/app/cms/posts/components/DeletePostButtonClient.tsx +1 -1
  129. package/templates/nextblock-template/app/cms/posts/components/PostForm.tsx +256 -245
  130. package/templates/nextblock-template/app/cms/posts/new/page.tsx +1 -1
  131. package/templates/nextblock-template/app/cms/posts/page.tsx +20 -13
  132. package/templates/nextblock-template/app/cms/products/ClientNotionEditor.tsx +16 -0
  133. package/templates/nextblock-template/app/cms/products/ProductFormClientShell.tsx +56 -0
  134. package/templates/nextblock-template/app/cms/products/[id]/edit/page.tsx +292 -0
  135. package/templates/nextblock-template/app/cms/products/attributes/page.tsx +12 -0
  136. package/templates/nextblock-template/app/cms/products/categories/page.tsx +12 -0
  137. package/templates/nextblock-template/app/cms/products/inventory/page.tsx +13 -0
  138. package/templates/nextblock-template/app/cms/products/new/page.tsx +143 -0
  139. package/templates/nextblock-template/app/cms/products/page.tsx +42 -0
  140. package/templates/nextblock-template/app/cms/products/productFormData.ts +133 -0
  141. package/templates/nextblock-template/app/cms/products/settings/page.tsx +5 -0
  142. package/templates/nextblock-template/app/cms/promotions/PromotionsWorkspace.tsx +456 -0
  143. package/templates/nextblock-template/app/cms/promotions/actions.ts +115 -0
  144. package/templates/nextblock-template/app/cms/promotions/page.tsx +31 -0
  145. package/templates/nextblock-template/app/cms/revisions/RevisionHistoryButton.tsx +2 -2
  146. package/templates/nextblock-template/app/cms/revisions/actions.ts +285 -285
  147. package/templates/nextblock-template/app/cms/revisions/service.ts +19 -16
  148. package/templates/nextblock-template/app/cms/revisions/utils.ts +8 -3
  149. package/templates/nextblock-template/app/cms/settings/backup-restore/BackupRestoreWorkspace.tsx +1004 -0
  150. package/templates/nextblock-template/app/cms/settings/backup-restore/page.tsx +29 -0
  151. package/templates/nextblock-template/app/cms/settings/bot-protection/actions.ts +93 -0
  152. package/templates/nextblock-template/app/cms/settings/bot-protection/components/BotProtectionForm.tsx +129 -0
  153. package/templates/nextblock-template/app/cms/settings/bot-protection/page.tsx +24 -0
  154. package/templates/nextblock-template/app/cms/settings/copyright/actions.ts +1 -1
  155. package/templates/nextblock-template/app/cms/settings/copyright/components/CopyrightForm.tsx +2 -2
  156. package/templates/nextblock-template/app/cms/settings/copyright/page.tsx +1 -1
  157. package/templates/nextblock-template/app/cms/settings/cortex-ai/SandboxCortexAiSettingsClient.tsx +496 -0
  158. package/templates/nextblock-template/app/cms/settings/cortex-ai/StoredCortexAiSettingsClient.tsx +410 -0
  159. package/templates/nextblock-template/app/cms/settings/cortex-ai/actions.ts +248 -0
  160. package/templates/nextblock-template/app/cms/settings/cortex-ai/page.tsx +80 -0
  161. package/templates/nextblock-template/app/cms/settings/currencies/actions.ts +331 -0
  162. package/templates/nextblock-template/app/cms/settings/currencies/page.tsx +494 -0
  163. package/templates/nextblock-template/app/cms/settings/extra-translations/ExtraTranslationsWorkspace.tsx +767 -0
  164. package/templates/nextblock-template/app/cms/settings/extra-translations/actions.ts +203 -44
  165. package/templates/nextblock-template/app/cms/settings/extra-translations/page.tsx +93 -242
  166. package/templates/nextblock-template/app/cms/settings/global-css/actions.ts +65 -0
  167. package/templates/nextblock-template/app/cms/settings/global-css/components/GlobalCssForm.tsx +46 -0
  168. package/templates/nextblock-template/app/cms/settings/global-css/page.tsx +24 -0
  169. package/templates/nextblock-template/app/cms/settings/languages/components/DeleteLanguageButton.tsx +1 -1
  170. package/templates/nextblock-template/app/cms/settings/languages/components/LanguageForm.tsx +2 -2
  171. package/templates/nextblock-template/app/cms/settings/languages/page.tsx +1 -1
  172. package/templates/nextblock-template/app/cms/settings/logos/[id]/edit/page.tsx +7 -7
  173. package/templates/nextblock-template/app/cms/settings/logos/actions.ts +82 -6
  174. package/templates/nextblock-template/app/cms/settings/logos/components/BrandingSettingsForm.tsx +339 -0
  175. package/templates/nextblock-template/app/cms/settings/logos/components/DeleteLogoButton.tsx +21 -18
  176. package/templates/nextblock-template/app/cms/settings/logos/components/LogoForm.tsx +20 -16
  177. package/templates/nextblock-template/app/cms/settings/logos/components/SiteSeoSettingsForm.tsx +133 -0
  178. package/templates/nextblock-template/app/cms/settings/logos/new/page.tsx +8 -8
  179. package/templates/nextblock-template/app/cms/settings/logos/page.tsx +120 -82
  180. package/templates/nextblock-template/app/cms/settings/logos/types.ts +8 -8
  181. package/templates/nextblock-template/app/cms/settings/packages/activation-form.tsx +84 -0
  182. package/templates/nextblock-template/app/cms/settings/packages/package-card.tsx +122 -0
  183. package/templates/nextblock-template/app/cms/settings/packages/page.tsx +49 -0
  184. package/templates/nextblock-template/app/cms/settings/privacy/actions.ts +53 -0
  185. package/templates/nextblock-template/app/cms/settings/privacy/components/PrivacyForm.tsx +196 -0
  186. package/templates/nextblock-template/app/cms/settings/privacy/page.tsx +26 -0
  187. package/templates/nextblock-template/app/cms/settings/security/actions.ts +251 -0
  188. package/templates/nextblock-template/app/cms/settings/security/components/SecurityPanel.tsx +453 -0
  189. package/templates/nextblock-template/app/cms/settings/security/page.tsx +13 -0
  190. package/templates/nextblock-template/app/cms/settings/taxes/page.tsx +21 -0
  191. package/templates/nextblock-template/app/cms/shipping/page.tsx +20 -0
  192. package/templates/nextblock-template/app/cms/users/[id]/edit/page.tsx +28 -23
  193. package/templates/nextblock-template/app/cms/users/actions.ts +105 -40
  194. package/templates/nextblock-template/app/cms/users/components/DeleteUserButton.tsx +1 -1
  195. package/templates/nextblock-template/app/cms/users/components/UserForm.tsx +65 -152
  196. package/templates/nextblock-template/app/cms/users/page.tsx +15 -10
  197. package/templates/nextblock-template/app/globals.css +9 -0
  198. package/templates/nextblock-template/app/layout.tsx +372 -120
  199. package/templates/nextblock-template/app/lib/seo.test.ts +52 -0
  200. package/templates/nextblock-template/app/lib/seo.ts +279 -0
  201. package/templates/nextblock-template/app/lib/site-settings.ts +87 -0
  202. package/templates/nextblock-template/app/lib/sitemap-utils.ts +224 -39
  203. package/templates/nextblock-template/app/lib/ucp/protocol.ts +190 -0
  204. package/templates/nextblock-template/app/lib/ucp/server.test.ts +56 -0
  205. package/templates/nextblock-template/app/lib/ucp/server.ts +1914 -0
  206. package/templates/nextblock-template/app/page.tsx +165 -73
  207. package/templates/nextblock-template/app/product/[slug]/page.tsx +433 -0
  208. package/templates/nextblock-template/app/profile/ProfileAccountSidebar.tsx +73 -0
  209. package/templates/nextblock-template/app/profile/ProfilePageHeader.tsx +16 -0
  210. package/templates/nextblock-template/app/profile/ProfilePageMissingState.tsx +9 -0
  211. package/templates/nextblock-template/app/profile/account-data.ts +37 -0
  212. package/templates/nextblock-template/app/profile/account-links.ts +22 -0
  213. package/templates/nextblock-template/app/profile/account-types.ts +11 -0
  214. package/templates/nextblock-template/app/profile/orders/CustomerOrdersPageClient.tsx +124 -0
  215. package/templates/nextblock-template/app/profile/orders/[id]/CustomerOrderDetailPageClient.tsx +79 -0
  216. package/templates/nextblock-template/app/profile/orders/[id]/page.tsx +32 -0
  217. package/templates/nextblock-template/app/profile/orders/page.tsx +19 -0
  218. package/templates/nextblock-template/app/profile/page.tsx +51 -0
  219. package/templates/nextblock-template/app/profile/password/PasswordSettingsPageClient.tsx +128 -0
  220. package/templates/nextblock-template/app/profile/password/actions.ts +59 -0
  221. package/templates/nextblock-template/app/profile/password/page.tsx +27 -0
  222. package/templates/nextblock-template/app/providers.tsx +55 -17
  223. package/templates/nextblock-template/app/robots.txt/route.ts +11 -1
  224. package/templates/nextblock-template/app/sitemap.ts +128 -0
  225. package/templates/nextblock-template/app/ucp/v1/carts/[id]/cancel/route.ts +38 -0
  226. package/templates/nextblock-template/app/ucp/v1/carts/[id]/route.ts +68 -0
  227. package/templates/nextblock-template/app/ucp/v1/carts/route.ts +35 -0
  228. package/templates/nextblock-template/app/ucp/v1/catalog/lookup/route.ts +35 -0
  229. package/templates/nextblock-template/app/ucp/v1/catalog/product/route.ts +35 -0
  230. package/templates/nextblock-template/app/ucp/v1/catalog/search/route.ts +34 -0
  231. package/templates/nextblock-template/components/AppShell.tsx +154 -0
  232. package/templates/nextblock-template/components/BlockRenderer.tsx +210 -64
  233. package/templates/nextblock-template/components/CartDrawerLoader.tsx +7 -0
  234. package/templates/nextblock-template/components/CartTranslator.tsx +210 -0
  235. package/templates/nextblock-template/components/CurrentContentSetter.tsx +25 -0
  236. package/templates/nextblock-template/components/DeferredCartDrawer.tsx +23 -0
  237. package/templates/nextblock-template/components/DeferredCartTranslator.tsx +51 -0
  238. package/templates/nextblock-template/components/DeferredGlobalSearch.tsx +68 -0
  239. package/templates/nextblock-template/components/DeferredGoogleTagManager.tsx +70 -0
  240. package/templates/nextblock-template/components/DeferredSpeedInsights.tsx +69 -0
  241. package/templates/nextblock-template/components/FeatureImageHero.tsx +47 -0
  242. package/templates/nextblock-template/components/GitHubLoginButton.tsx +36 -0
  243. package/templates/nextblock-template/components/GlobalSearch.tsx +557 -0
  244. package/templates/nextblock-template/components/Header.tsx +49 -41
  245. package/templates/nextblock-template/components/LanguageSwitcher.tsx +55 -32
  246. package/templates/nextblock-template/components/ResponsiveNav.tsx +138 -43
  247. package/templates/nextblock-template/components/blocks/PostCardSkeleton.tsx +12 -8
  248. package/templates/nextblock-template/components/blocks/PostsGridBlock.tsx +12 -55
  249. package/templates/nextblock-template/components/blocks/PostsGridClient.tsx +42 -37
  250. package/templates/nextblock-template/components/blocks/TestimonialBlock.tsx +6 -2
  251. package/templates/nextblock-template/components/blocks/ecommerceRendererLoaders.ts +23 -0
  252. package/templates/nextblock-template/components/blocks/publicRendererLoaders.ts +25 -0
  253. package/templates/nextblock-template/components/blocks/renderers/ButtonBlockRenderer.tsx +92 -84
  254. package/templates/nextblock-template/components/blocks/renderers/CartBlockRenderer.tsx +17 -0
  255. package/templates/nextblock-template/components/blocks/renderers/CheckoutBlockRenderer.tsx +19 -0
  256. package/templates/nextblock-template/components/blocks/renderers/ClientTextBlockRenderer.tsx +262 -8
  257. package/templates/nextblock-template/components/blocks/renderers/FeaturedProductBlockRenderer.tsx +22 -0
  258. package/templates/nextblock-template/components/blocks/renderers/FormBlockRenderer.tsx +320 -37
  259. package/templates/nextblock-template/components/blocks/renderers/HeadingBlockRenderer.tsx +11 -8
  260. package/templates/nextblock-template/components/blocks/renderers/ImageBlockRenderer.tsx +12 -3
  261. package/templates/nextblock-template/components/blocks/renderers/PostsGridBlockRenderer.tsx +18 -13
  262. package/templates/nextblock-template/components/blocks/renderers/ProductDetailsBlockRenderer.tsx +90 -0
  263. package/templates/nextblock-template/components/blocks/renderers/ProductGridBlockRenderer.tsx +31 -0
  264. package/templates/nextblock-template/components/blocks/renderers/SectionBlockRenderer.tsx +424 -55
  265. package/templates/nextblock-template/components/blocks/renderers/SectionSlider.tsx +137 -0
  266. package/templates/nextblock-template/components/blocks/renderers/TestimonialBlockRenderer.tsx +57 -0
  267. package/templates/nextblock-template/components/blocks/renderers/TextBlockRenderer.tsx +37 -22
  268. package/templates/nextblock-template/components/blocks/renderers/VideoEmbedBlockRenderer.tsx +23 -15
  269. package/templates/nextblock-template/components/blocks/renderers/inline/AlertWidgetRenderer.tsx +1 -3
  270. package/templates/nextblock-template/components/blocks/renderers/inline/CtaWidgetRenderer.tsx +1 -3
  271. package/templates/nextblock-template/components/blocks/types.ts +7 -6
  272. package/templates/nextblock-template/components/env-var-warning.tsx +3 -3
  273. package/templates/nextblock-template/components/form-message.tsx +32 -26
  274. package/templates/nextblock-template/components/header-auth.tsx +69 -17
  275. package/templates/nextblock-template/components/privacy/ConsentBanner.tsx +127 -0
  276. package/templates/nextblock-template/components/privacy/ConsentGatedAnalytics.tsx +59 -0
  277. package/templates/nextblock-template/components/renderers/CachedDynamicLayoutEngine.tsx +28 -0
  278. package/templates/nextblock-template/components/renderers/DynamicLayoutEngine.test.tsx +166 -0
  279. package/templates/nextblock-template/components/renderers/DynamicLayoutEngine.tsx +464 -0
  280. package/templates/nextblock-template/components/theme-switcher.tsx +8 -8
  281. package/templates/nextblock-template/components/visual-editing/DeferredVisualEditing.tsx +21 -0
  282. package/templates/nextblock-template/components/visual-editing/NextblockVisualEditing.tsx +1172 -0
  283. package/templates/nextblock-template/context/AuthContext.tsx +23 -90
  284. package/templates/nextblock-template/context/CurrentContentContext.tsx +10 -4
  285. package/templates/nextblock-template/context/LanguageContext.tsx +16 -16
  286. package/templates/nextblock-template/context/language-rest-client.ts +31 -0
  287. package/templates/nextblock-template/docs/01-PROJECT-OVERVIEW.md +94 -0
  288. package/templates/nextblock-template/docs/02-ECOMMERCE-CAPABILITIES.md +364 -0
  289. package/templates/nextblock-template/docs/03-CMS-AND-EDITOR.md +202 -0
  290. package/templates/nextblock-template/docs/04-DATABASE-AND-AUTH.md +252 -0
  291. package/templates/nextblock-template/docs/05-DEVELOPER-GUIDE.md +238 -0
  292. package/templates/nextblock-template/docs/06-CLI-AND-SCAFFOLDING.md +125 -0
  293. package/templates/nextblock-template/docs/07-BLOCK-SDK-AND-EXTENSIBILITY.md +146 -0
  294. package/templates/nextblock-template/docs/08-NEXTBLOCK-CORTEX-AI-ARCHITECTURE.md +1319 -0
  295. package/templates/nextblock-template/docs/09-LIVE-DRAFT-MODE.md +104 -0
  296. package/templates/nextblock-template/docs/10-CUSTOM-BLOCKS.md +222 -0
  297. package/templates/nextblock-template/docs/README.md +34 -0
  298. package/templates/nextblock-template/docs/TECHNICAL_SPECIFICATION.md +12507 -0
  299. package/templates/nextblock-template/hooks/use-hotkeys.ts +21 -14
  300. package/templates/nextblock-template/hooks/useGlobalSearch.ts +101 -0
  301. package/templates/nextblock-template/index.d.ts +2 -0
  302. package/templates/nextblock-template/lib/ai-block-generation.ts +339 -0
  303. package/templates/nextblock-template/lib/ai-client.ts +247 -0
  304. package/templates/nextblock-template/lib/ai-config.ts +81 -0
  305. package/templates/nextblock-template/lib/ai-cortex-widget-builder.ts +125 -0
  306. package/templates/nextblock-template/lib/ai-global-agent-custom-block-tools.ts +363 -0
  307. package/templates/nextblock-template/lib/ai-global-agent-db-tools.test.ts +405 -0
  308. package/templates/nextblock-template/lib/ai-global-agent-db-tools.ts +1228 -0
  309. package/templates/nextblock-template/lib/ai-global-agent-ecommerce.ts +5 -0
  310. package/templates/nextblock-template/lib/ai-global-agent-tools-stats.test.ts +223 -0
  311. package/templates/nextblock-template/lib/ai-global-agent-tools.test.ts +2183 -0
  312. package/templates/nextblock-template/lib/ai-global-agent-tools.ts +4807 -0
  313. package/templates/nextblock-template/lib/ai-key-crypto.test.ts +70 -0
  314. package/templates/nextblock-template/lib/ai-key-crypto.ts +132 -0
  315. package/templates/nextblock-template/lib/ai-model-catalog.test.ts +49 -0
  316. package/templates/nextblock-template/lib/ai-model-catalog.ts +41 -0
  317. package/templates/nextblock-template/lib/ai-model-registry.test.ts +231 -0
  318. package/templates/nextblock-template/lib/ai-model-registry.ts +522 -0
  319. package/templates/nextblock-template/lib/auth/cookies.ts +47 -0
  320. package/templates/nextblock-template/lib/auth/crypto.ts +42 -0
  321. package/templates/nextblock-template/lib/auth/trustedDevices.ts +92 -0
  322. package/templates/nextblock-template/lib/auth/twoFactor.ts +167 -0
  323. package/templates/nextblock-template/lib/auth-redirects.ts +46 -0
  324. package/templates/nextblock-template/lib/blocks/FeaturedProductBlock.tsx +94 -0
  325. package/templates/nextblock-template/lib/blocks/ProductGridBlock.tsx +137 -0
  326. package/templates/nextblock-template/lib/blocks/README.md +13 -670
  327. package/templates/nextblock-template/lib/blocks/blockRegistry.ts +138 -56
  328. package/templates/nextblock-template/lib/blocks/blockTypes.ts +18 -0
  329. package/templates/nextblock-template/lib/blocks/ecommerce-block-schemas.ts +31 -0
  330. package/templates/nextblock-template/lib/cms-transfer/csv.test.ts +77 -0
  331. package/templates/nextblock-template/lib/cms-transfer/csv.ts +399 -0
  332. package/templates/nextblock-template/lib/cms-transfer/server.ts +2243 -0
  333. package/templates/nextblock-template/lib/cms-transfer/types.ts +145 -0
  334. package/templates/nextblock-template/lib/cortex-widget-registry.test.ts +199 -0
  335. package/templates/nextblock-template/lib/cortex-widget-registry.ts +88 -0
  336. package/templates/nextblock-template/lib/cortex-widget-schema.test.tsx +237 -0
  337. package/templates/nextblock-template/lib/cortex-widget-schema.ts +393 -0
  338. package/templates/nextblock-template/lib/custom-block-definitions.ts +87 -0
  339. package/templates/nextblock-template/lib/custom-block-r2-upload-shared.ts +178 -0
  340. package/templates/nextblock-template/lib/custom-block-r2-upload.test.ts +140 -0
  341. package/templates/nextblock-template/lib/custom-block-r2-upload.ts +68 -0
  342. package/templates/nextblock-template/lib/custom-block-relation-registry.ts +256 -0
  343. package/templates/nextblock-template/lib/custom-block-relations.test.ts +227 -0
  344. package/templates/nextblock-template/lib/custom-block-relations.ts +279 -0
  345. package/templates/nextblock-template/lib/custom-block-safelist.ts +14 -0
  346. package/templates/nextblock-template/lib/editor/dynamic-extension-core.test.ts +172 -0
  347. package/templates/nextblock-template/lib/editor/dynamic-extension-core.ts +213 -0
  348. package/templates/nextblock-template/lib/editor/dynamic-extension-loader.ts +22 -0
  349. package/templates/nextblock-template/lib/editor/dynamic-extensions.tsx +193 -0
  350. package/templates/nextblock-template/lib/full-backup/manifest.test.ts +121 -0
  351. package/templates/nextblock-template/lib/full-backup/manifest.ts +206 -0
  352. package/templates/nextblock-template/lib/full-backup/server.ts +743 -0
  353. package/templates/nextblock-template/lib/media/resolveMediaUrl.ts +45 -0
  354. package/templates/nextblock-template/lib/posts/readTime.ts +60 -0
  355. package/templates/nextblock-template/lib/privacy/consent-client.ts +57 -0
  356. package/templates/nextblock-template/lib/privacy/settings.ts +103 -0
  357. package/templates/nextblock-template/lib/privacy/types.ts +67 -0
  358. package/templates/nextblock-template/lib/promotions/server.test.ts +74 -0
  359. package/templates/nextblock-template/lib/promotions/server.ts +741 -0
  360. package/templates/nextblock-template/lib/resolve-block-relations.test.ts +142 -0
  361. package/templates/nextblock-template/lib/resolve-block-relations.ts +255 -0
  362. package/templates/nextblock-template/lib/search/server.ts +585 -0
  363. package/templates/nextblock-template/lib/search/types.ts +27 -0
  364. package/templates/nextblock-template/lib/visual-editing/draft-content.test.ts +105 -0
  365. package/templates/nextblock-template/lib/visual-editing/draft-content.ts +380 -0
  366. package/templates/nextblock-template/lib/visual-editing/draft-route.test.ts +42 -0
  367. package/templates/nextblock-template/lib/visual-editing/draft-route.ts +82 -0
  368. package/templates/nextblock-template/lib/visual-editing/edit-info.test.ts +143 -0
  369. package/templates/nextblock-template/lib/visual-editing/edit-info.ts +94 -0
  370. package/templates/nextblock-template/lib/visual-editing/mutations.ts +190 -0
  371. package/templates/nextblock-template/lib/visual-editing/product-drafts.test.ts +81 -0
  372. package/templates/nextblock-template/lib/visual-editing/product-drafts.ts +511 -0
  373. package/templates/nextblock-template/lib/visual-editing/types.ts +122 -0
  374. package/templates/nextblock-template/lib/zod-config.ts +5 -0
  375. package/templates/nextblock-template/next.config.js +190 -66
  376. package/templates/nextblock-template/package.json +34 -30
  377. package/templates/nextblock-template/proxy.ts +435 -253
  378. package/templates/nextblock-template/public/images/NBcover.webp +0 -0
  379. package/templates/nextblock-template/public/images/cap.webp +0 -0
  380. package/templates/nextblock-template/public/images/commerce-plan.webp +0 -0
  381. package/templates/nextblock-template/public/images/commerce-square.webp +0 -0
  382. package/templates/nextblock-template/public/images/commerce-wide.webp +0 -0
  383. package/templates/nextblock-template/public/images/cortex-ai-square.webp +0 -0
  384. package/templates/nextblock-template/public/images/cortex-ai.webp +0 -0
  385. package/templates/nextblock-template/public/images/extensibility.webp +0 -0
  386. package/templates/nextblock-template/public/images/goals.webp +0 -0
  387. package/templates/nextblock-template/public/images/included.webp +0 -0
  388. package/templates/nextblock-template/public/images/nx-graph.webp +0 -0
  389. package/templates/nextblock-template/public/images/pants.webp +0 -0
  390. package/templates/nextblock-template/public/images/t-shirt.webp +0 -0
  391. package/templates/nextblock-template/scripts/validate-editor-block-schema.ts +112 -0
  392. package/templates/nextblock-template/scripts/verify-cortex-ai-build-widget.tsx +100 -0
  393. package/templates/nextblock-template/scripts/verify-cortex-ai-generate-blocks.ts +62 -0
  394. package/templates/nextblock-template/scripts/verify-cortex-ai-global-tools.ts +537 -0
  395. package/templates/nextblock-template/scripts/verify-cortex-ai-routing.ts +58 -0
  396. package/templates/nextblock-template/scripts/verify-custom-block-definitions.ts +188 -0
  397. package/templates/nextblock-template/scripts/verify-dynamic-custom-block-extensions.ts +123 -0
  398. package/templates/nextblock-template/scripts/verify-dynamic-layout-engine.tsx +133 -0
  399. package/templates/nextblock-template/scripts/verify-milestone-2-custom-blocks.ts +65 -0
  400. package/templates/nextblock-template/tailwind.config.js +1 -0
  401. package/templates/nextblock-template/tools/configure-supabase-auth.js +282 -0
  402. package/templates/nextblock-template/tools/deploy-supabase.js +69 -71
  403. package/templates/nextblock-template/tsconfig.json +52 -66
  404. package/templates/nextblock-template/tsconfig.tsbuildinfo +1 -1
  405. package/templates/nextblock-template/types/jsdom.d.ts +6 -0
  406. package/templates/nextblock-template/app/force-styles.tsx +0 -31
  407. package/templates/nextblock-template/app/sitemap.xml/route.ts +0 -63
  408. package/templates/nextblock-template/components/blocks/renderers/HeroBlockRenderer.tsx +0 -273
  409. package/templates/nextblock-template/docs/How to Create a Custom Block.md +0 -149
  410. package/templates/nextblock-template/docs/cms-application-overview.md +0 -56
  411. package/templates/nextblock-template/docs/cms-architecture-overview.md +0 -73
  412. package/templates/nextblock-template/docs/files-structure.md +0 -426
  413. package/templates/nextblock-template/docs/tiptap-bundle-optimization-summary.md +0 -174
@@ -2,28 +2,31 @@
2
2
 
3
3
  import { DropdownMenuItem } from "@nextblock-cms/ui";
4
4
  import { Trash2 } from "lucide-react";
5
- import { deleteLogo } from "../actions";
6
- import { useTransition, useState } from 'react';
7
- import { ConfirmationModal } from "@/app/cms/components/ConfirmationModal";
8
- import { toast } from 'react-hot-toast';
5
+ import { deleteLogo } from "../actions";
6
+ import { useTransition, useState } from 'react';
7
+ import { ConfirmationModal } from '../../../components/ConfirmationModal';
8
+ import { toast } from 'react-hot-toast';
9
+ import { useRouter } from 'next/navigation';
9
10
 
10
11
  interface DeleteLogoButtonProps {
11
12
  logoId: string;
12
13
  }
13
14
 
14
- export default function DeleteLogoButton({ logoId }: DeleteLogoButtonProps) {
15
- const [isPending, startTransition] = useTransition();
16
- const [isModalOpen, setIsModalOpen] = useState(false);
15
+ export default function DeleteLogoButton({ logoId }: DeleteLogoButtonProps) {
16
+ const router = useRouter();
17
+ const [isPending, startTransition] = useTransition();
18
+ const [isModalOpen, setIsModalOpen] = useState(false);
17
19
 
18
20
  const handleDeleteConfirm = () => {
19
21
  startTransition(async () => {
20
22
  const result = await deleteLogo(logoId);
21
23
  if (result?.error) {
22
24
  toast.error(`Error: ${result.error}`);
23
- } else {
24
- toast.success("Logo deleted successfully");
25
- }
26
- setIsModalOpen(false);
25
+ } else {
26
+ toast.success("Logo deleted successfully");
27
+ router.refresh();
28
+ }
29
+ setIsModalOpen(false);
27
30
  });
28
31
  };
29
32
 
@@ -38,13 +41,13 @@ export default function DeleteLogoButton({ logoId }: DeleteLogoButtonProps) {
38
41
  <Trash2 className="mr-2 h-4 w-4" />
39
42
  {isPending ? "Deleting..." : "Delete"}
40
43
  </DropdownMenuItem>
41
- <ConfirmationModal
42
- isOpen={isModalOpen}
43
- onClose={() => setIsModalOpen(false)}
44
- onConfirm={handleDeleteConfirm}
45
- title="Delete Logo?"
46
- description="This will permanently delete the logo. This action cannot be undone."
47
- />
44
+ <ConfirmationModal
45
+ isOpen={isModalOpen}
46
+ onClose={() => setIsModalOpen(false)}
47
+ onConfirm={handleDeleteConfirm}
48
+ title="Delete Brand Logo?"
49
+ description="This will permanently delete the logo. This action cannot be undone."
50
+ />
48
51
  </>
49
52
  );
50
53
  }
@@ -8,11 +8,15 @@ import { Label } from '@nextblock-cms/ui'
8
8
  import { Button } from '@nextblock-cms/ui'
9
9
  import { Alert, AlertDescription, Spinner } from '@nextblock-cms/ui'
10
10
  import type { Database } from '@nextblock-cms/db'
11
- import { ImageIcon, X as XIcon } from 'lucide-react'
12
- import MediaPickerDialog from '@/app/cms/media/components/MediaPickerDialog'
13
- import { useHotkeys } from '@/hooks/use-hotkeys'
14
- type Media = Database['public']['Tables']['media']['Row'];
15
- const R2_BASE_URL = process.env.NEXT_PUBLIC_R2_BASE_URL || ''
11
+ import { ImageIcon, X as XIcon } from 'lucide-react'
12
+ import MediaPickerDialog from '../../../media/components/MediaPickerDialog'
13
+ import { useHotkeys } from '../../../../../hooks/use-hotkeys';
14
+ import { resolveMediaUrl } from '../../../../../lib/media/resolveMediaUrl';
15
+ type Media = Database['public']['Tables']['media']['Row'];
16
+
17
+ function resolveLogoSrc(objectKey?: string | null) {
18
+ return resolveMediaUrl(objectKey)
19
+ }
16
20
 
17
21
  interface LogoDetails {
18
22
  id?: string
@@ -102,8 +106,8 @@ export default function LogoForm({ logo, action }: LogoFormProps) {
102
106
 
103
107
  return (
104
108
  <div className="space-y-6">
105
- <div>
106
- <Label htmlFor="name">Logo Name</Label>
109
+ <div>
110
+ <Label htmlFor="name">Logo Name</Label>
107
111
  <Input
108
112
  id="name"
109
113
  name="name"
@@ -126,11 +130,11 @@ export default function LogoForm({ logo, action }: LogoFormProps) {
126
130
  className="relative group inline-block"
127
131
  style={{ maxWidth: logoDetails.width, maxHeight: 200 }}
128
132
  >
129
- <Image
130
- src={`${R2_BASE_URL}/${logoDetails.object_key}`}
131
- alt={logoDetails.name || 'Selected logo'}
132
- width={logoDetails.width}
133
- height={logoDetails.height}
133
+ <Image
134
+ src={resolveLogoSrc(logoDetails.object_key) || ''}
135
+ alt={logoDetails.name || 'Selected logo'}
136
+ width={logoDetails.width}
137
+ height={logoDetails.height}
134
138
  className="rounded-md object-contain"
135
139
  style={{ maxHeight: '200px' }}
136
140
  placeholder={logoDetails.blur_data_url ? 'blur' : 'empty'}
@@ -170,10 +174,10 @@ export default function LogoForm({ logo, action }: LogoFormProps) {
170
174
  <>
171
175
  <Spinner className="mr-2 h-4 w-4" /> Saving...
172
176
  </>
173
- ) : (
174
- `${logo ? 'Update' : 'Create'} Logo`
175
- )}
176
- </Button>
177
+ ) : (
178
+ `${logo ? 'Update' : 'Create'} Logo`
179
+ )}
180
+ </Button>
177
181
  {formError && (
178
182
  <Alert variant="destructive" className="py-2 px-4 w-auto inline-flex items-center">
179
183
  <AlertDescription>{formError}</AlertDescription>
@@ -0,0 +1,133 @@
1
+ 'use client'
2
+
3
+ import { useState, useTransition } from 'react'
4
+ import { Button, Input, Label, Textarea } from '@nextblock-cms/ui'
5
+
6
+ import { saveSiteSeoSettings, type SiteSeoSettings } from '../actions'
7
+
8
+ interface SiteSeoSettingsFormProps {
9
+ initialSettings: SiteSeoSettings
10
+ }
11
+
12
+ const TITLE_RECOMMENDED_MAX = 60
13
+ const DESCRIPTION_RECOMMENDED_MAX = 160
14
+
15
+ export default function SiteSeoSettingsForm({ initialSettings }: SiteSeoSettingsFormProps) {
16
+ const [isPending, startTransition] = useTransition()
17
+ const [message, setMessage] = useState<{ type: 'success' | 'error'; text: string } | null>(null)
18
+ const [settings, setSettings] = useState<SiteSeoSettings>(() => ({
19
+ siteTitle: initialSettings.siteTitle ?? '',
20
+ siteDescription: initialSettings.siteDescription ?? '',
21
+ siteKeywords: initialSettings.siteKeywords ?? '',
22
+ }))
23
+
24
+ const handleSave = () => {
25
+ setMessage(null)
26
+ startTransition(async () => {
27
+ const result = await saveSiteSeoSettings(settings)
28
+
29
+ if (result?.error) {
30
+ setMessage({ type: 'error', text: result.error })
31
+ return
32
+ }
33
+
34
+ setMessage({ type: 'success', text: 'Site SEO settings updated successfully.' })
35
+ })
36
+ }
37
+
38
+ return (
39
+ <div className="space-y-6 rounded-xl border bg-background p-5">
40
+ <div className="space-y-1">
41
+ <h2 className="text-xl font-semibold">Site identity &amp; SEO</h2>
42
+ <p className="text-sm text-muted-foreground">
43
+ Used for the browser tab, search results, and social link previews (Open Graph / Twitter).
44
+ The site title is also appended to every page title and shown next to the logo.
45
+ </p>
46
+ </div>
47
+
48
+ <div className="space-y-2">
49
+ <div className="flex items-center justify-between">
50
+ <Label htmlFor="seo-site-title">Site title</Label>
51
+ <span
52
+ className={`text-xs ${
53
+ settings.siteTitle.length > TITLE_RECOMMENDED_MAX
54
+ ? 'text-amber-600'
55
+ : 'text-muted-foreground'
56
+ }`}
57
+ >
58
+ {settings.siteTitle.length}/{TITLE_RECOMMENDED_MAX}
59
+ </span>
60
+ </div>
61
+ <Input
62
+ id="seo-site-title"
63
+ placeholder="NextBlock™ CMS"
64
+ value={settings.siteTitle}
65
+ onChange={(event) =>
66
+ setSettings((current) => ({ ...current, siteTitle: event.target.value }))
67
+ }
68
+ />
69
+ <p className="text-xs text-muted-foreground">
70
+ Example result: <span className="font-medium">Home | {settings.siteTitle || 'NextBlock™ CMS'}</span>
71
+ </p>
72
+ </div>
73
+
74
+ <div className="space-y-2">
75
+ <div className="flex items-center justify-between">
76
+ <Label htmlFor="seo-site-description">Default meta description</Label>
77
+ <span
78
+ className={`text-xs ${
79
+ settings.siteDescription.length > DESCRIPTION_RECOMMENDED_MAX
80
+ ? 'text-amber-600'
81
+ : 'text-muted-foreground'
82
+ }`}
83
+ >
84
+ {settings.siteDescription.length}/{DESCRIPTION_RECOMMENDED_MAX}
85
+ </span>
86
+ </div>
87
+ <Textarea
88
+ id="seo-site-description"
89
+ rows={3}
90
+ placeholder="A short, compelling summary of your site for search engines and social cards."
91
+ value={settings.siteDescription}
92
+ onChange={(event) =>
93
+ setSettings((current) => ({ ...current, siteDescription: event.target.value }))
94
+ }
95
+ />
96
+ <p className="text-xs text-muted-foreground">
97
+ Shown when a page has no description of its own. Aim for ~150–160 characters.
98
+ </p>
99
+ </div>
100
+
101
+ <div className="space-y-2">
102
+ <Label htmlFor="seo-site-keywords">Default keywords</Label>
103
+ <Input
104
+ id="seo-site-keywords"
105
+ placeholder="NextBlock, CMS, Next.js, Supabase"
106
+ value={settings.siteKeywords}
107
+ onChange={(event) =>
108
+ setSettings((current) => ({ ...current, siteKeywords: event.target.value }))
109
+ }
110
+ />
111
+ <p className="text-xs text-muted-foreground">Comma-separated. Used as the default meta keywords.</p>
112
+ </div>
113
+
114
+ {message ? (
115
+ <div
116
+ className={`rounded-lg border px-4 py-3 text-sm ${
117
+ message.type === 'success'
118
+ ? 'border-emerald-200 bg-emerald-50 text-emerald-700'
119
+ : 'border-red-200 bg-red-50 text-red-700'
120
+ }`}
121
+ >
122
+ {message.text}
123
+ </div>
124
+ ) : null}
125
+
126
+ <div className="flex justify-end">
127
+ <Button type="button" onClick={handleSave} disabled={isPending}>
128
+ {isPending ? 'Saving...' : 'Save SEO Settings'}
129
+ </Button>
130
+ </div>
131
+ </div>
132
+ )
133
+ }
@@ -1,11 +1,11 @@
1
1
  import LogoForm from '../components/LogoForm'
2
2
  import { createLogo } from '../actions'
3
3
 
4
- export default function NewLogoPage() {
5
- return (
6
- <div>
7
- <h1 className="text-2xl font-semibold mb-6">Create New Logo</h1>
8
- <LogoForm action={createLogo} />
9
- </div>
10
- )
11
- }
4
+ export default function NewLogoPage() {
5
+ return (
6
+ <div>
7
+ <h1 className="text-2xl font-semibold mb-6">Add Brand Logo</h1>
8
+ <LogoForm action={createLogo} />
9
+ </div>
10
+ )
11
+ }
@@ -1,6 +1,6 @@
1
- import React from 'react'
2
1
  import Link from 'next/link'
3
2
  import { Button } from '@nextblock-cms/ui'
3
+ import { getInvoiceBrandingData } from '@nextblock-cms/ecommerce/server'
4
4
  import {
5
5
  Table,
6
6
  TableBody,
@@ -17,99 +17,137 @@ import {
17
17
  DropdownMenuTrigger,
18
18
  DropdownMenuSeparator,
19
19
  } from '@nextblock-cms/ui'
20
- import { getLogos } from './actions'
21
- import MediaImage from '@/app/cms/media/components/MediaImage'
20
+ import { getLogos, getSiteSeoSettings } from './actions'
21
+ import BrandingSettingsForm from './components/BrandingSettingsForm'
22
+ import SiteSeoSettingsForm from './components/SiteSeoSettingsForm'
23
+ import MediaImage from '../../media/components/MediaImage'
22
24
  import DeleteLogoButton from './components/DeleteLogoButton'
25
+ import { resolveMediaUrl } from '../../../../lib/media/resolveMediaUrl'
23
26
 
24
- const R2_BASE_URL = process.env.NEXT_PUBLIC_R2_BASE_URL || ''
27
+ function resolveLogoSrc(objectKey?: string | null) {
28
+ return resolveMediaUrl(objectKey)
29
+ }
25
30
 
26
31
  export default async function CmsLogosListPage() {
27
- const logos = await getLogos()
32
+ const [logos, branding, seoSettings] = await Promise.all([
33
+ getLogos(),
34
+ getInvoiceBrandingData(),
35
+ getSiteSeoSettings(),
36
+ ])
28
37
 
29
38
  return (
30
- <div className="w-full">
31
- <div className="flex justify-between items-center mb-6">
32
- <h1 className="text-2xl font-semibold">Manage Logos</h1>
33
- <Button variant="default" asChild>
34
- <Link href="/cms/settings/logos/new">
35
- <PlusCircle className="mr-2 h-4 w-4" /> New Logo
36
- </Link>
37
- </Button>
39
+ <div className="w-full space-y-8">
40
+ <div className="mb-6">
41
+ <div>
42
+ <h1 className="text-2xl font-semibold">Branding</h1>
43
+ <p className="text-sm text-muted-foreground mt-1">
44
+ Manage the site title &amp; SEO, the active website logo, and the seller details printed on invoices.
45
+ </p>
46
+ </div>
38
47
  </div>
39
48
 
40
- {logos.length === 0 ? (
41
- <div className="text-center py-10 border rounded-lg">
42
- <ImageIcon className="mx-auto h-12 w-12 text-muted-foreground" />
43
- <h3 className="mt-2 text-sm font-medium">No logos found.</h3>
44
- <p className="mt-1 text-sm text-muted-foreground">
45
- Get started by creating a new logo.
46
- </p>
47
- <div className="mt-6">
48
- <Button asChild>
49
- <Link href="/cms/settings/logos/new">
50
- <PlusCircle className="mr-2 h-4 w-4" /> Create Logo
51
- </Link>
52
- </Button>
49
+ <SiteSeoSettingsForm initialSettings={seoSettings} />
50
+
51
+ <BrandingSettingsForm initialSettings={branding.settings} />
52
+
53
+ <section className="space-y-4">
54
+ <div className="flex items-center justify-between gap-4">
55
+ <div>
56
+ <h2 className="text-xl font-semibold">Logos</h2>
57
+ <p className="text-sm text-muted-foreground">
58
+ The most recently created logo is used on the storefront and invoices.
59
+ </p>
53
60
  </div>
61
+ <Button variant="default" asChild>
62
+ <Link href="/cms/settings/logos/new">
63
+ <PlusCircle className="mr-2 h-4 w-4" /> New Logo
64
+ </Link>
65
+ </Button>
54
66
  </div>
55
- ) : (
56
- <div className="rounded-lg border overflow-hidden">
57
- <Table>
58
- <TableHeader>
59
- <TableRow>
60
- <TableHead className="w-[80px]">Image</TableHead>
61
- <TableHead>Name</TableHead>
62
- <TableHead className="hidden md:table-cell">Created At</TableHead>
63
- <TableHead className="text-right">Actions</TableHead>
64
- </TableRow>
65
- </TableHeader>
66
- <TableBody>
67
- {logos.map(logo => (
68
- <TableRow key={logo.id}>
69
- <TableCell>
67
+
68
+ {logos.length === 0 ? (
69
+ <div className="text-center py-10 border rounded-lg">
70
+ <ImageIcon className="mx-auto h-12 w-12 text-muted-foreground" />
71
+ <h3 className="mt-2 text-sm font-medium">No logos found.</h3>
72
+ <p className="mt-1 text-sm text-muted-foreground">
73
+ Get started by creating a new logo.
74
+ </p>
75
+ <div className="mt-6">
76
+ <Button asChild>
77
+ <Link href="/cms/settings/logos/new">
78
+ <PlusCircle className="mr-2 h-4 w-4" /> Create Logo
79
+ </Link>
80
+ </Button>
81
+ </div>
82
+ </div>
83
+ ) : (
84
+ <div className="rounded-lg border overflow-hidden">
85
+ <Table>
86
+ <TableHeader>
87
+ <TableRow>
88
+ <TableHead className="w-[80px]">Image</TableHead>
89
+ <TableHead>Name</TableHead>
90
+ <TableHead className="hidden md:table-cell">Created At</TableHead>
91
+ <TableHead className="text-right">Actions</TableHead>
92
+ </TableRow>
93
+ </TableHeader>
94
+ <TableBody>
95
+ {logos.map((logo, index) => (
96
+ <TableRow key={logo.id}>
97
+ <TableCell>
70
98
  {logo.media ? (
71
99
  <MediaImage
72
- src={`${R2_BASE_URL}/${logo.media.object_key}`}
73
- alt={logo.media.alt_text || logo.name}
74
- width={logo.media.width || 100}
75
- height={logo.media.height || 100}
76
- className="max-w-16 max-h-16 object-contain"
77
- />
78
- ) : (
79
- <div className="w-16 h-16 bg-muted rounded-sm flex items-center justify-center">
80
- <ImageIcon className="h-6 w-6 text-muted-foreground" />
100
+ src={resolveLogoSrc(logo.media.object_key) || ''}
101
+ alt={logo.media.alt_text || logo.name}
102
+ width={logo.media.width || 100}
103
+ height={logo.media.height || 100}
104
+ className="max-w-16 max-h-16 object-contain"
105
+ />
106
+ ) : (
107
+ <div className="w-16 h-16 bg-muted rounded-sm flex items-center justify-center">
108
+ <ImageIcon className="h-6 w-6 text-muted-foreground" />
109
+ </div>
110
+ )}
111
+ </TableCell>
112
+ <TableCell className="font-medium">
113
+ <div className="flex items-center gap-2">
114
+ <span>{logo.name}</span>
115
+ {index === 0 ? (
116
+ <span className="rounded-full bg-primary/10 px-2 py-0.5 text-[10px] font-semibold uppercase tracking-wide text-primary">
117
+ Active
118
+ </span>
119
+ ) : null}
81
120
  </div>
82
- )}
83
- </TableCell>
84
- <TableCell className="font-medium">{logo.name}</TableCell>
85
- <TableCell className="hidden md:table-cell">
86
- {new Date(logo.created_at).toLocaleDateString()}
87
- </TableCell>
88
- <TableCell className="text-right">
89
- <DropdownMenu>
90
- <DropdownMenuTrigger asChild>
91
- <Button variant="ghost" size="icon">
92
- <MoreHorizontal className="h-4 w-4" />
93
- </Button>
94
- </DropdownMenuTrigger>
95
- <DropdownMenuContent align="end">
96
- <DropdownMenuItem asChild>
97
- <Link href={`/cms/settings/logos/${logo.id}/edit`}>
98
- <Edit3 className="mr-2 h-4 w-4" />
99
- Edit
100
- </Link>
101
- </DropdownMenuItem>
102
- <DropdownMenuSeparator />
103
- <DeleteLogoButton logoId={logo.id} />
104
- </DropdownMenuContent>
105
- </DropdownMenu>
106
- </TableCell>
107
- </TableRow>
108
- ))}
109
- </TableBody>
110
- </Table>
111
- </div>
112
- )}
121
+ </TableCell>
122
+ <TableCell className="hidden md:table-cell">
123
+ {new Date(logo.created_at).toLocaleDateString()}
124
+ </TableCell>
125
+ <TableCell className="text-right">
126
+ <DropdownMenu>
127
+ <DropdownMenuTrigger asChild>
128
+ <Button id={`logo-trigger-${logo.id}`} variant="ghost" size="icon">
129
+ <MoreHorizontal className="h-4 w-4" />
130
+ </Button>
131
+ </DropdownMenuTrigger>
132
+ <DropdownMenuContent align="end">
133
+ <DropdownMenuItem asChild>
134
+ <Link href={`/cms/settings/logos/${logo.id}/edit`}>
135
+ <Edit3 className="mr-2 h-4 w-4" />
136
+ Edit
137
+ </Link>
138
+ </DropdownMenuItem>
139
+ <DropdownMenuSeparator />
140
+ <DeleteLogoButton logoId={logo.id} />
141
+ </DropdownMenuContent>
142
+ </DropdownMenu>
143
+ </TableCell>
144
+ </TableRow>
145
+ ))}
146
+ </TableBody>
147
+ </Table>
148
+ </div>
149
+ )}
150
+ </section>
113
151
  </div>
114
152
  )
115
- }
153
+ }
@@ -1,8 +1,8 @@
1
- import type { Database } from '@nextblock-cms/db';
2
-
3
- export type Logo = Database['public']['Tables']['logos']['Row'] & {
4
- site_title?: string | null;
5
- media: (Database['public']['Tables']['media']['Row'] & {
6
- alt_text: string | null;
7
- }) | null;
8
- };
1
+ import type { Database } from '@nextblock-cms/db';
2
+
3
+ export type Logo = Database['public']['Tables']['logos']['Row'] & {
4
+ site_title?: string | null;
5
+ media: (Database['public']['Tables']['media']['Row'] & {
6
+ alt_text: string | null;
7
+ }) | null;
8
+ };
@@ -0,0 +1,84 @@
1
+ 'use client';
2
+
3
+ import { useState } from 'react';
4
+ import { activatePackage } from '../../../actions/package-actions';
5
+ import { toast } from 'sonner';
6
+ import { Button } from '@nextblock-cms/ui';
7
+ import { Input } from '@nextblock-cms/ui';
8
+ import { Loader2, FlaskConical } from 'lucide-react';
9
+
10
+ const isSandbox = process.env.NEXT_PUBLIC_IS_SANDBOX === 'true';
11
+
12
+ export function ActivationForm() {
13
+ const [key, setKey] = useState('');
14
+ const [loading, setLoading] = useState(false);
15
+ const [errorMsg, setErrorMsg] = useState<string | null>(null);
16
+
17
+ const handleActivate = async () => {
18
+ if (!key) return;
19
+ setLoading(true);
20
+ setErrorMsg(null);
21
+ try {
22
+ const res = await activatePackage(key);
23
+ if (res?.error) {
24
+ toast.error(res.error);
25
+ setErrorMsg(res.error);
26
+ } else {
27
+ toast.success(`Package "${res?.package}" activated successfully!`);
28
+ setKey('');
29
+ setErrorMsg(null);
30
+ }
31
+ } catch {
32
+ toast.error('Activation failed. Please try again.');
33
+ setErrorMsg('Activation failed. Please try again.');
34
+ } finally {
35
+ setLoading(false);
36
+ }
37
+ };
38
+
39
+ if (isSandbox) {
40
+ return (
41
+ <div className="mt-8 p-6 border rounded-lg bg-amber-50 dark:bg-amber-950/20 border-amber-200 dark:border-amber-800">
42
+ <div className="flex items-center gap-2 mb-2">
43
+ <FlaskConical className="h-5 w-5 text-amber-600 dark:text-amber-400" />
44
+ <h3 className="text-lg font-medium text-amber-800 dark:text-amber-300">Sandbox Environment</h3>
45
+ </div>
46
+ <p className="text-sm text-amber-700 dark:text-amber-400">
47
+ License activation is disabled in this sandbox demo. To purchase a real license for
48
+ your self-hosted instance, visit{' '}
49
+ <a href="https://nextblock.ca" target="_blank" rel="noopener noreferrer" className="underline font-semibold">
50
+ nextblock.ca
51
+ </a>.
52
+ </p>
53
+ </div>
54
+ );
55
+ }
56
+
57
+ return (
58
+ <div className="mt-8 p-6 border rounded-lg bg-card">
59
+ <h3 className="text-lg font-medium mb-4">Activate a Package</h3>
60
+ <div className="flex items-start gap-4">
61
+ <div className="flex flex-col gap-2 flex-1 max-w-md">
62
+ <Input
63
+ placeholder="Enter your Freemius License Key"
64
+ value={key}
65
+ onChange={(e) => {
66
+ setKey(e.target.value);
67
+ setErrorMsg(null);
68
+ }}
69
+ />
70
+ {errorMsg && (
71
+ <p className="text-sm font-medium text-destructive">{errorMsg}</p>
72
+ )}
73
+ </div>
74
+ <Button onClick={handleActivate} disabled={loading || !key}>
75
+ {loading && <Loader2 className="mr-2 h-4 w-4 animate-spin" />}
76
+ Activate License
77
+ </Button>
78
+ </div>
79
+ <p className="text-sm text-muted-foreground mt-2">
80
+ Enter the license key you received from Freemius to unlock the package features.
81
+ </p>
82
+ </div>
83
+ );
84
+ }