create-nextblock 0.2.78 → 0.8.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (413) hide show
  1. package/bin/create-nextblock.js +793 -472
  2. package/package.json +1 -2
  3. package/scripts/sync-template.js +18 -1
  4. package/templates/nextblock-template/.browserslistrc +11 -0
  5. package/templates/nextblock-template/.swcrc +30 -30
  6. package/templates/nextblock-template/README.md +23 -114
  7. package/templates/nextblock-template/app/(auth-pages)/post-sign-in/page.tsx +27 -28
  8. package/templates/nextblock-template/app/(auth-pages)/sign-in/page.tsx +50 -25
  9. package/templates/nextblock-template/app/(auth-pages)/sign-up/page.tsx +111 -56
  10. package/templates/nextblock-template/app/(auth-pages)/two-factor/actions.ts +91 -0
  11. package/templates/nextblock-template/app/(auth-pages)/two-factor/components/TwoFactorForm.tsx +118 -0
  12. package/templates/nextblock-template/app/(auth-pages)/two-factor/page.tsx +51 -0
  13. package/templates/nextblock-template/app/.well-known/ucp/route.ts +16 -0
  14. package/templates/nextblock-template/app/[slug]/PageClientContent.tsx +48 -28
  15. package/templates/nextblock-template/app/[slug]/page.tsx +63 -6
  16. package/templates/nextblock-template/app/[slug]/page.utils.ts +374 -157
  17. package/templates/nextblock-template/app/[slug]/pageClientActions.ts +7 -0
  18. package/templates/nextblock-template/app/actions/consent.ts +57 -0
  19. package/templates/nextblock-template/app/actions/formActions.ts +130 -11
  20. package/templates/nextblock-template/app/actions/languageActions.ts +31 -30
  21. package/templates/nextblock-template/app/actions/package-actions.ts +183 -0
  22. package/templates/nextblock-template/app/actions/postActions.ts +146 -48
  23. package/templates/nextblock-template/app/actions/twoFactorEmail.ts +21 -0
  24. package/templates/nextblock-template/app/actions/visualEditingActions.test.ts +179 -0
  25. package/templates/nextblock-template/app/actions/visualEditingActions.ts +345 -0
  26. package/templates/nextblock-template/app/actions.ts +67 -12
  27. package/templates/nextblock-template/app/api/ai/cortex/build-widget/route.ts +153 -0
  28. package/templates/nextblock-template/app/api/ai/generate-blocks/route.ts +96 -0
  29. package/templates/nextblock-template/app/api/ai/global-agent/route.ts +965 -0
  30. package/templates/nextblock-template/app/api/checkout/freemius/sync/route.ts +29 -0
  31. package/templates/nextblock-template/app/api/checkout/route.ts +146 -0
  32. package/templates/nextblock-template/app/api/cms/full-backup/export/route.ts +33 -0
  33. package/templates/nextblock-template/app/api/cms/full-backup/restore/route.ts +63 -0
  34. package/templates/nextblock-template/app/api/cron/reset-sandbox/route.ts +3413 -17
  35. package/templates/nextblock-template/app/api/cron/reset-sandbox/sandboxResetSql.ts +7830 -0
  36. package/templates/nextblock-template/app/api/cron/sync-currencies/route.ts +35 -0
  37. package/templates/nextblock-template/app/api/custom-blocks/db-relations/route.ts +92 -0
  38. package/templates/nextblock-template/app/api/custom-blocks/editor-definitions/route.ts +43 -0
  39. package/templates/nextblock-template/app/api/draft/disable/route.ts +25 -0
  40. package/templates/nextblock-template/app/api/draft/route.ts +93 -0
  41. package/templates/nextblock-template/app/api/draft/start/route.ts +77 -0
  42. package/templates/nextblock-template/app/api/media/library/route.ts +65 -0
  43. package/templates/nextblock-template/app/api/media/r2-presigned/route.ts +53 -0
  44. package/templates/nextblock-template/app/api/media/record/route.ts +160 -0
  45. package/templates/nextblock-template/app/api/search/route.ts +43 -0
  46. package/templates/nextblock-template/app/api/visual-editing/block-draft/route.ts +47 -0
  47. package/templates/nextblock-template/app/api/visual-editing/product-draft/route.ts +47 -0
  48. package/templates/nextblock-template/app/api/webhooks/freemius/route.ts +34 -0
  49. package/templates/nextblock-template/app/api/webhooks/stripe/route.ts +27 -0
  50. package/templates/nextblock-template/app/article/[slug]/PostClientContent.tsx +392 -128
  51. package/templates/nextblock-template/app/article/[slug]/page.tsx +179 -127
  52. package/templates/nextblock-template/app/article/[slug]/page.utils.ts +262 -77
  53. package/templates/nextblock-template/app/auth/callback/route.ts +31 -58
  54. package/templates/nextblock-template/app/cart/page.tsx +7 -0
  55. package/templates/nextblock-template/app/checkout/UcpCartHydrator.tsx +20 -0
  56. package/templates/nextblock-template/app/checkout/page.tsx +52 -0
  57. package/templates/nextblock-template/app/checkout/success/actions.ts +136 -0
  58. package/templates/nextblock-template/app/checkout/success/page.tsx +186 -0
  59. package/templates/nextblock-template/app/cms/CmsClientLayout.tsx +163 -33
  60. package/templates/nextblock-template/app/cms/blocks/actions.ts +424 -235
  61. package/templates/nextblock-template/app/cms/blocks/components/BackgroundSelector.tsx +212 -151
  62. package/templates/nextblock-template/app/cms/blocks/components/BlockEditorArea.tsx +41 -20
  63. package/templates/nextblock-template/app/cms/blocks/components/BlockEditorModal.tsx +152 -19
  64. package/templates/nextblock-template/app/cms/blocks/components/BlockTypeCard.tsx +25 -17
  65. package/templates/nextblock-template/app/cms/blocks/components/BlockTypeSelector.tsx +200 -18
  66. package/templates/nextblock-template/app/cms/blocks/components/ColumnEditor.tsx +33 -16
  67. package/templates/nextblock-template/app/cms/blocks/components/CustomBlockEditorPreview.tsx +160 -0
  68. package/templates/nextblock-template/app/cms/blocks/components/EditableBlock.tsx +37 -18
  69. package/templates/nextblock-template/app/cms/blocks/components/MediaLibraryModal.tsx +149 -67
  70. package/templates/nextblock-template/app/cms/blocks/components/SectionConfigPanel.tsx +108 -31
  71. package/templates/nextblock-template/app/cms/blocks/editors/DynamicCustomBlockEditor.tsx +167 -0
  72. package/templates/nextblock-template/app/cms/blocks/editors/FeaturedProductBlockEditor.tsx +31 -0
  73. package/templates/nextblock-template/app/cms/blocks/editors/FormBlockEditor.tsx +2 -2
  74. package/templates/nextblock-template/app/cms/blocks/editors/HeadingBlockEditor.tsx +1 -1
  75. package/templates/nextblock-template/app/cms/blocks/editors/ImageBlockEditor.tsx +29 -29
  76. package/templates/nextblock-template/app/cms/blocks/editors/PostsGridBlockEditor.tsx +14 -18
  77. package/templates/nextblock-template/app/cms/blocks/editors/ProductGridBlockEditor.tsx +41 -0
  78. package/templates/nextblock-template/app/cms/blocks/editors/SectionBlockEditor.tsx +318 -118
  79. package/templates/nextblock-template/app/cms/blocks/editors/TextBlockEditor.tsx +98 -21
  80. package/templates/nextblock-template/app/cms/blocks/editors/VideoEmbedBlockEditor.tsx +1 -1
  81. package/templates/nextblock-template/app/cms/components/ContentLanguageSwitcher.tsx +27 -9
  82. package/templates/nextblock-template/app/cms/components/CopyContentFromLanguage.tsx +1 -1
  83. package/templates/nextblock-template/app/cms/components/CortexAiActiveContext.tsx +23 -0
  84. package/templates/nextblock-template/app/cms/components/CortexAiPageContext.tsx +58 -0
  85. package/templates/nextblock-template/app/cms/components/CortexGlobalAgentChat.tsx +1507 -0
  86. package/templates/nextblock-template/app/cms/components/DraftStatusActions.tsx +145 -0
  87. package/templates/nextblock-template/app/cms/components/FeatureImageField.tsx +244 -0
  88. package/templates/nextblock-template/app/cms/components/FeedbackModal.tsx +38 -24
  89. package/templates/nextblock-template/app/cms/coupons/[id]/edit/page.tsx +16 -0
  90. package/templates/nextblock-template/app/cms/coupons/page.tsx +16 -0
  91. package/templates/nextblock-template/app/cms/custom-blocks/[id]/edit/page.tsx +66 -0
  92. package/templates/nextblock-template/app/cms/custom-blocks/actions.ts +519 -0
  93. package/templates/nextblock-template/app/cms/custom-blocks/components/BlockComposer.tsx +1522 -0
  94. package/templates/nextblock-template/app/cms/custom-blocks/components/BlocksLibraryTransferControls.tsx +256 -0
  95. package/templates/nextblock-template/app/cms/custom-blocks/components/DBRelationSelect.tsx +384 -0
  96. package/templates/nextblock-template/app/cms/custom-blocks/components/ImageR2Picker.tsx +221 -0
  97. package/templates/nextblock-template/app/cms/custom-blocks/new/page.tsx +12 -0
  98. package/templates/nextblock-template/app/cms/custom-blocks/page.tsx +438 -0
  99. package/templates/nextblock-template/app/cms/dashboard/actions.ts +228 -98
  100. package/templates/nextblock-template/app/cms/dashboard/components/DashboardComponents.tsx +200 -0
  101. package/templates/nextblock-template/app/cms/dashboard/page.tsx +182 -154
  102. package/templates/nextblock-template/app/cms/import-export/ContentTransferControls.tsx +391 -0
  103. package/templates/nextblock-template/app/cms/import-export/actions.ts +226 -0
  104. package/templates/nextblock-template/app/cms/layout.tsx +29 -10
  105. package/templates/nextblock-template/app/cms/media/UploadFolderContext.tsx +22 -22
  106. package/templates/nextblock-template/app/cms/media/actions.ts +45 -124
  107. package/templates/nextblock-template/app/cms/media/components/DeleteMediaButtonClient.tsx +1 -1
  108. package/templates/nextblock-template/app/cms/media/components/MediaEditForm.tsx +26 -26
  109. package/templates/nextblock-template/app/cms/media/components/MediaGridClient.tsx +69 -64
  110. package/templates/nextblock-template/app/cms/media/components/MediaPickerDialog.tsx +227 -158
  111. package/templates/nextblock-template/app/cms/media/components/MediaUploadForm.tsx +101 -89
  112. package/templates/nextblock-template/app/cms/media/page.tsx +1 -1
  113. package/templates/nextblock-template/app/cms/navigation/components/NavigationItemForm.tsx +2 -2
  114. package/templates/nextblock-template/app/cms/orders/[id]/MarkPaidButton.tsx +44 -0
  115. package/templates/nextblock-template/app/cms/orders/[id]/page.tsx +16 -0
  116. package/templates/nextblock-template/app/cms/orders/actions.ts +201 -0
  117. package/templates/nextblock-template/app/cms/orders/page.tsx +20 -0
  118. package/templates/nextblock-template/app/cms/orders/types.ts +20 -0
  119. package/templates/nextblock-template/app/cms/pages/[id]/edit/EditPageClient.tsx +156 -121
  120. package/templates/nextblock-template/app/cms/pages/[id]/edit/page.tsx +79 -26
  121. package/templates/nextblock-template/app/cms/pages/actions.ts +54 -38
  122. package/templates/nextblock-template/app/cms/pages/components/DeletePageButtonClient.tsx +1 -1
  123. package/templates/nextblock-template/app/cms/pages/components/PageForm.tsx +267 -116
  124. package/templates/nextblock-template/app/cms/pages/page.tsx +25 -18
  125. package/templates/nextblock-template/app/cms/payments/page.tsx +16 -0
  126. package/templates/nextblock-template/app/cms/posts/[id]/edit/page.tsx +132 -90
  127. package/templates/nextblock-template/app/cms/posts/actions.ts +71 -72
  128. package/templates/nextblock-template/app/cms/posts/components/DeletePostButtonClient.tsx +1 -1
  129. package/templates/nextblock-template/app/cms/posts/components/PostForm.tsx +256 -245
  130. package/templates/nextblock-template/app/cms/posts/new/page.tsx +1 -1
  131. package/templates/nextblock-template/app/cms/posts/page.tsx +20 -13
  132. package/templates/nextblock-template/app/cms/products/ClientNotionEditor.tsx +16 -0
  133. package/templates/nextblock-template/app/cms/products/ProductFormClientShell.tsx +56 -0
  134. package/templates/nextblock-template/app/cms/products/[id]/edit/page.tsx +292 -0
  135. package/templates/nextblock-template/app/cms/products/attributes/page.tsx +12 -0
  136. package/templates/nextblock-template/app/cms/products/categories/page.tsx +12 -0
  137. package/templates/nextblock-template/app/cms/products/inventory/page.tsx +13 -0
  138. package/templates/nextblock-template/app/cms/products/new/page.tsx +143 -0
  139. package/templates/nextblock-template/app/cms/products/page.tsx +42 -0
  140. package/templates/nextblock-template/app/cms/products/productFormData.ts +133 -0
  141. package/templates/nextblock-template/app/cms/products/settings/page.tsx +5 -0
  142. package/templates/nextblock-template/app/cms/promotions/PromotionsWorkspace.tsx +456 -0
  143. package/templates/nextblock-template/app/cms/promotions/actions.ts +115 -0
  144. package/templates/nextblock-template/app/cms/promotions/page.tsx +31 -0
  145. package/templates/nextblock-template/app/cms/revisions/RevisionHistoryButton.tsx +2 -2
  146. package/templates/nextblock-template/app/cms/revisions/actions.ts +285 -285
  147. package/templates/nextblock-template/app/cms/revisions/service.ts +19 -16
  148. package/templates/nextblock-template/app/cms/revisions/utils.ts +8 -3
  149. package/templates/nextblock-template/app/cms/settings/backup-restore/BackupRestoreWorkspace.tsx +1004 -0
  150. package/templates/nextblock-template/app/cms/settings/backup-restore/page.tsx +29 -0
  151. package/templates/nextblock-template/app/cms/settings/bot-protection/actions.ts +93 -0
  152. package/templates/nextblock-template/app/cms/settings/bot-protection/components/BotProtectionForm.tsx +129 -0
  153. package/templates/nextblock-template/app/cms/settings/bot-protection/page.tsx +24 -0
  154. package/templates/nextblock-template/app/cms/settings/copyright/actions.ts +1 -1
  155. package/templates/nextblock-template/app/cms/settings/copyright/components/CopyrightForm.tsx +2 -2
  156. package/templates/nextblock-template/app/cms/settings/copyright/page.tsx +1 -1
  157. package/templates/nextblock-template/app/cms/settings/cortex-ai/SandboxCortexAiSettingsClient.tsx +496 -0
  158. package/templates/nextblock-template/app/cms/settings/cortex-ai/StoredCortexAiSettingsClient.tsx +410 -0
  159. package/templates/nextblock-template/app/cms/settings/cortex-ai/actions.ts +248 -0
  160. package/templates/nextblock-template/app/cms/settings/cortex-ai/page.tsx +80 -0
  161. package/templates/nextblock-template/app/cms/settings/currencies/actions.ts +331 -0
  162. package/templates/nextblock-template/app/cms/settings/currencies/page.tsx +494 -0
  163. package/templates/nextblock-template/app/cms/settings/extra-translations/ExtraTranslationsWorkspace.tsx +767 -0
  164. package/templates/nextblock-template/app/cms/settings/extra-translations/actions.ts +203 -44
  165. package/templates/nextblock-template/app/cms/settings/extra-translations/page.tsx +93 -242
  166. package/templates/nextblock-template/app/cms/settings/global-css/actions.ts +65 -0
  167. package/templates/nextblock-template/app/cms/settings/global-css/components/GlobalCssForm.tsx +46 -0
  168. package/templates/nextblock-template/app/cms/settings/global-css/page.tsx +24 -0
  169. package/templates/nextblock-template/app/cms/settings/languages/components/DeleteLanguageButton.tsx +1 -1
  170. package/templates/nextblock-template/app/cms/settings/languages/components/LanguageForm.tsx +2 -2
  171. package/templates/nextblock-template/app/cms/settings/languages/page.tsx +1 -1
  172. package/templates/nextblock-template/app/cms/settings/logos/[id]/edit/page.tsx +7 -7
  173. package/templates/nextblock-template/app/cms/settings/logos/actions.ts +82 -6
  174. package/templates/nextblock-template/app/cms/settings/logos/components/BrandingSettingsForm.tsx +339 -0
  175. package/templates/nextblock-template/app/cms/settings/logos/components/DeleteLogoButton.tsx +21 -18
  176. package/templates/nextblock-template/app/cms/settings/logos/components/LogoForm.tsx +20 -16
  177. package/templates/nextblock-template/app/cms/settings/logos/components/SiteSeoSettingsForm.tsx +133 -0
  178. package/templates/nextblock-template/app/cms/settings/logos/new/page.tsx +8 -8
  179. package/templates/nextblock-template/app/cms/settings/logos/page.tsx +120 -82
  180. package/templates/nextblock-template/app/cms/settings/logos/types.ts +8 -8
  181. package/templates/nextblock-template/app/cms/settings/packages/activation-form.tsx +84 -0
  182. package/templates/nextblock-template/app/cms/settings/packages/package-card.tsx +122 -0
  183. package/templates/nextblock-template/app/cms/settings/packages/page.tsx +49 -0
  184. package/templates/nextblock-template/app/cms/settings/privacy/actions.ts +53 -0
  185. package/templates/nextblock-template/app/cms/settings/privacy/components/PrivacyForm.tsx +196 -0
  186. package/templates/nextblock-template/app/cms/settings/privacy/page.tsx +26 -0
  187. package/templates/nextblock-template/app/cms/settings/security/actions.ts +251 -0
  188. package/templates/nextblock-template/app/cms/settings/security/components/SecurityPanel.tsx +453 -0
  189. package/templates/nextblock-template/app/cms/settings/security/page.tsx +13 -0
  190. package/templates/nextblock-template/app/cms/settings/taxes/page.tsx +21 -0
  191. package/templates/nextblock-template/app/cms/shipping/page.tsx +20 -0
  192. package/templates/nextblock-template/app/cms/users/[id]/edit/page.tsx +28 -23
  193. package/templates/nextblock-template/app/cms/users/actions.ts +105 -40
  194. package/templates/nextblock-template/app/cms/users/components/DeleteUserButton.tsx +1 -1
  195. package/templates/nextblock-template/app/cms/users/components/UserForm.tsx +65 -152
  196. package/templates/nextblock-template/app/cms/users/page.tsx +15 -10
  197. package/templates/nextblock-template/app/globals.css +9 -0
  198. package/templates/nextblock-template/app/layout.tsx +372 -120
  199. package/templates/nextblock-template/app/lib/seo.test.ts +52 -0
  200. package/templates/nextblock-template/app/lib/seo.ts +279 -0
  201. package/templates/nextblock-template/app/lib/site-settings.ts +87 -0
  202. package/templates/nextblock-template/app/lib/sitemap-utils.ts +224 -39
  203. package/templates/nextblock-template/app/lib/ucp/protocol.ts +190 -0
  204. package/templates/nextblock-template/app/lib/ucp/server.test.ts +56 -0
  205. package/templates/nextblock-template/app/lib/ucp/server.ts +1914 -0
  206. package/templates/nextblock-template/app/page.tsx +165 -73
  207. package/templates/nextblock-template/app/product/[slug]/page.tsx +433 -0
  208. package/templates/nextblock-template/app/profile/ProfileAccountSidebar.tsx +73 -0
  209. package/templates/nextblock-template/app/profile/ProfilePageHeader.tsx +16 -0
  210. package/templates/nextblock-template/app/profile/ProfilePageMissingState.tsx +9 -0
  211. package/templates/nextblock-template/app/profile/account-data.ts +37 -0
  212. package/templates/nextblock-template/app/profile/account-links.ts +22 -0
  213. package/templates/nextblock-template/app/profile/account-types.ts +11 -0
  214. package/templates/nextblock-template/app/profile/orders/CustomerOrdersPageClient.tsx +124 -0
  215. package/templates/nextblock-template/app/profile/orders/[id]/CustomerOrderDetailPageClient.tsx +79 -0
  216. package/templates/nextblock-template/app/profile/orders/[id]/page.tsx +32 -0
  217. package/templates/nextblock-template/app/profile/orders/page.tsx +19 -0
  218. package/templates/nextblock-template/app/profile/page.tsx +51 -0
  219. package/templates/nextblock-template/app/profile/password/PasswordSettingsPageClient.tsx +128 -0
  220. package/templates/nextblock-template/app/profile/password/actions.ts +59 -0
  221. package/templates/nextblock-template/app/profile/password/page.tsx +27 -0
  222. package/templates/nextblock-template/app/providers.tsx +55 -17
  223. package/templates/nextblock-template/app/robots.txt/route.ts +11 -1
  224. package/templates/nextblock-template/app/sitemap.ts +128 -0
  225. package/templates/nextblock-template/app/ucp/v1/carts/[id]/cancel/route.ts +38 -0
  226. package/templates/nextblock-template/app/ucp/v1/carts/[id]/route.ts +68 -0
  227. package/templates/nextblock-template/app/ucp/v1/carts/route.ts +35 -0
  228. package/templates/nextblock-template/app/ucp/v1/catalog/lookup/route.ts +35 -0
  229. package/templates/nextblock-template/app/ucp/v1/catalog/product/route.ts +35 -0
  230. package/templates/nextblock-template/app/ucp/v1/catalog/search/route.ts +34 -0
  231. package/templates/nextblock-template/components/AppShell.tsx +154 -0
  232. package/templates/nextblock-template/components/BlockRenderer.tsx +210 -64
  233. package/templates/nextblock-template/components/CartDrawerLoader.tsx +7 -0
  234. package/templates/nextblock-template/components/CartTranslator.tsx +210 -0
  235. package/templates/nextblock-template/components/CurrentContentSetter.tsx +25 -0
  236. package/templates/nextblock-template/components/DeferredCartDrawer.tsx +23 -0
  237. package/templates/nextblock-template/components/DeferredCartTranslator.tsx +51 -0
  238. package/templates/nextblock-template/components/DeferredGlobalSearch.tsx +68 -0
  239. package/templates/nextblock-template/components/DeferredGoogleTagManager.tsx +70 -0
  240. package/templates/nextblock-template/components/DeferredSpeedInsights.tsx +69 -0
  241. package/templates/nextblock-template/components/FeatureImageHero.tsx +47 -0
  242. package/templates/nextblock-template/components/GitHubLoginButton.tsx +36 -0
  243. package/templates/nextblock-template/components/GlobalSearch.tsx +557 -0
  244. package/templates/nextblock-template/components/Header.tsx +49 -41
  245. package/templates/nextblock-template/components/LanguageSwitcher.tsx +55 -32
  246. package/templates/nextblock-template/components/ResponsiveNav.tsx +138 -43
  247. package/templates/nextblock-template/components/blocks/PostCardSkeleton.tsx +12 -8
  248. package/templates/nextblock-template/components/blocks/PostsGridBlock.tsx +12 -55
  249. package/templates/nextblock-template/components/blocks/PostsGridClient.tsx +42 -37
  250. package/templates/nextblock-template/components/blocks/TestimonialBlock.tsx +6 -2
  251. package/templates/nextblock-template/components/blocks/ecommerceRendererLoaders.ts +23 -0
  252. package/templates/nextblock-template/components/blocks/publicRendererLoaders.ts +25 -0
  253. package/templates/nextblock-template/components/blocks/renderers/ButtonBlockRenderer.tsx +92 -84
  254. package/templates/nextblock-template/components/blocks/renderers/CartBlockRenderer.tsx +17 -0
  255. package/templates/nextblock-template/components/blocks/renderers/CheckoutBlockRenderer.tsx +19 -0
  256. package/templates/nextblock-template/components/blocks/renderers/ClientTextBlockRenderer.tsx +262 -8
  257. package/templates/nextblock-template/components/blocks/renderers/FeaturedProductBlockRenderer.tsx +22 -0
  258. package/templates/nextblock-template/components/blocks/renderers/FormBlockRenderer.tsx +320 -37
  259. package/templates/nextblock-template/components/blocks/renderers/HeadingBlockRenderer.tsx +11 -8
  260. package/templates/nextblock-template/components/blocks/renderers/ImageBlockRenderer.tsx +12 -3
  261. package/templates/nextblock-template/components/blocks/renderers/PostsGridBlockRenderer.tsx +18 -13
  262. package/templates/nextblock-template/components/blocks/renderers/ProductDetailsBlockRenderer.tsx +90 -0
  263. package/templates/nextblock-template/components/blocks/renderers/ProductGridBlockRenderer.tsx +31 -0
  264. package/templates/nextblock-template/components/blocks/renderers/SectionBlockRenderer.tsx +424 -55
  265. package/templates/nextblock-template/components/blocks/renderers/SectionSlider.tsx +137 -0
  266. package/templates/nextblock-template/components/blocks/renderers/TestimonialBlockRenderer.tsx +57 -0
  267. package/templates/nextblock-template/components/blocks/renderers/TextBlockRenderer.tsx +37 -22
  268. package/templates/nextblock-template/components/blocks/renderers/VideoEmbedBlockRenderer.tsx +23 -15
  269. package/templates/nextblock-template/components/blocks/renderers/inline/AlertWidgetRenderer.tsx +1 -3
  270. package/templates/nextblock-template/components/blocks/renderers/inline/CtaWidgetRenderer.tsx +1 -3
  271. package/templates/nextblock-template/components/blocks/types.ts +7 -6
  272. package/templates/nextblock-template/components/env-var-warning.tsx +3 -3
  273. package/templates/nextblock-template/components/form-message.tsx +32 -26
  274. package/templates/nextblock-template/components/header-auth.tsx +69 -17
  275. package/templates/nextblock-template/components/privacy/ConsentBanner.tsx +127 -0
  276. package/templates/nextblock-template/components/privacy/ConsentGatedAnalytics.tsx +59 -0
  277. package/templates/nextblock-template/components/renderers/CachedDynamicLayoutEngine.tsx +28 -0
  278. package/templates/nextblock-template/components/renderers/DynamicLayoutEngine.test.tsx +166 -0
  279. package/templates/nextblock-template/components/renderers/DynamicLayoutEngine.tsx +464 -0
  280. package/templates/nextblock-template/components/theme-switcher.tsx +8 -8
  281. package/templates/nextblock-template/components/visual-editing/DeferredVisualEditing.tsx +21 -0
  282. package/templates/nextblock-template/components/visual-editing/NextblockVisualEditing.tsx +1172 -0
  283. package/templates/nextblock-template/context/AuthContext.tsx +23 -90
  284. package/templates/nextblock-template/context/CurrentContentContext.tsx +10 -4
  285. package/templates/nextblock-template/context/LanguageContext.tsx +16 -16
  286. package/templates/nextblock-template/context/language-rest-client.ts +31 -0
  287. package/templates/nextblock-template/docs/01-PROJECT-OVERVIEW.md +94 -0
  288. package/templates/nextblock-template/docs/02-ECOMMERCE-CAPABILITIES.md +364 -0
  289. package/templates/nextblock-template/docs/03-CMS-AND-EDITOR.md +202 -0
  290. package/templates/nextblock-template/docs/04-DATABASE-AND-AUTH.md +252 -0
  291. package/templates/nextblock-template/docs/05-DEVELOPER-GUIDE.md +238 -0
  292. package/templates/nextblock-template/docs/06-CLI-AND-SCAFFOLDING.md +125 -0
  293. package/templates/nextblock-template/docs/07-BLOCK-SDK-AND-EXTENSIBILITY.md +146 -0
  294. package/templates/nextblock-template/docs/08-NEXTBLOCK-CORTEX-AI-ARCHITECTURE.md +1319 -0
  295. package/templates/nextblock-template/docs/09-LIVE-DRAFT-MODE.md +104 -0
  296. package/templates/nextblock-template/docs/10-CUSTOM-BLOCKS.md +222 -0
  297. package/templates/nextblock-template/docs/README.md +34 -0
  298. package/templates/nextblock-template/docs/TECHNICAL_SPECIFICATION.md +12507 -0
  299. package/templates/nextblock-template/hooks/use-hotkeys.ts +21 -14
  300. package/templates/nextblock-template/hooks/useGlobalSearch.ts +101 -0
  301. package/templates/nextblock-template/index.d.ts +2 -0
  302. package/templates/nextblock-template/lib/ai-block-generation.ts +339 -0
  303. package/templates/nextblock-template/lib/ai-client.ts +247 -0
  304. package/templates/nextblock-template/lib/ai-config.ts +81 -0
  305. package/templates/nextblock-template/lib/ai-cortex-widget-builder.ts +125 -0
  306. package/templates/nextblock-template/lib/ai-global-agent-custom-block-tools.ts +363 -0
  307. package/templates/nextblock-template/lib/ai-global-agent-db-tools.test.ts +405 -0
  308. package/templates/nextblock-template/lib/ai-global-agent-db-tools.ts +1228 -0
  309. package/templates/nextblock-template/lib/ai-global-agent-ecommerce.ts +5 -0
  310. package/templates/nextblock-template/lib/ai-global-agent-tools-stats.test.ts +223 -0
  311. package/templates/nextblock-template/lib/ai-global-agent-tools.test.ts +2183 -0
  312. package/templates/nextblock-template/lib/ai-global-agent-tools.ts +4807 -0
  313. package/templates/nextblock-template/lib/ai-key-crypto.test.ts +70 -0
  314. package/templates/nextblock-template/lib/ai-key-crypto.ts +132 -0
  315. package/templates/nextblock-template/lib/ai-model-catalog.test.ts +49 -0
  316. package/templates/nextblock-template/lib/ai-model-catalog.ts +41 -0
  317. package/templates/nextblock-template/lib/ai-model-registry.test.ts +231 -0
  318. package/templates/nextblock-template/lib/ai-model-registry.ts +522 -0
  319. package/templates/nextblock-template/lib/auth/cookies.ts +47 -0
  320. package/templates/nextblock-template/lib/auth/crypto.ts +42 -0
  321. package/templates/nextblock-template/lib/auth/trustedDevices.ts +92 -0
  322. package/templates/nextblock-template/lib/auth/twoFactor.ts +167 -0
  323. package/templates/nextblock-template/lib/auth-redirects.ts +46 -0
  324. package/templates/nextblock-template/lib/blocks/FeaturedProductBlock.tsx +94 -0
  325. package/templates/nextblock-template/lib/blocks/ProductGridBlock.tsx +137 -0
  326. package/templates/nextblock-template/lib/blocks/README.md +13 -670
  327. package/templates/nextblock-template/lib/blocks/blockRegistry.ts +138 -56
  328. package/templates/nextblock-template/lib/blocks/blockTypes.ts +18 -0
  329. package/templates/nextblock-template/lib/blocks/ecommerce-block-schemas.ts +31 -0
  330. package/templates/nextblock-template/lib/cms-transfer/csv.test.ts +77 -0
  331. package/templates/nextblock-template/lib/cms-transfer/csv.ts +399 -0
  332. package/templates/nextblock-template/lib/cms-transfer/server.ts +2243 -0
  333. package/templates/nextblock-template/lib/cms-transfer/types.ts +145 -0
  334. package/templates/nextblock-template/lib/cortex-widget-registry.test.ts +199 -0
  335. package/templates/nextblock-template/lib/cortex-widget-registry.ts +88 -0
  336. package/templates/nextblock-template/lib/cortex-widget-schema.test.tsx +237 -0
  337. package/templates/nextblock-template/lib/cortex-widget-schema.ts +393 -0
  338. package/templates/nextblock-template/lib/custom-block-definitions.ts +87 -0
  339. package/templates/nextblock-template/lib/custom-block-r2-upload-shared.ts +178 -0
  340. package/templates/nextblock-template/lib/custom-block-r2-upload.test.ts +140 -0
  341. package/templates/nextblock-template/lib/custom-block-r2-upload.ts +68 -0
  342. package/templates/nextblock-template/lib/custom-block-relation-registry.ts +256 -0
  343. package/templates/nextblock-template/lib/custom-block-relations.test.ts +227 -0
  344. package/templates/nextblock-template/lib/custom-block-relations.ts +279 -0
  345. package/templates/nextblock-template/lib/custom-block-safelist.ts +14 -0
  346. package/templates/nextblock-template/lib/editor/dynamic-extension-core.test.ts +172 -0
  347. package/templates/nextblock-template/lib/editor/dynamic-extension-core.ts +213 -0
  348. package/templates/nextblock-template/lib/editor/dynamic-extension-loader.ts +22 -0
  349. package/templates/nextblock-template/lib/editor/dynamic-extensions.tsx +193 -0
  350. package/templates/nextblock-template/lib/full-backup/manifest.test.ts +121 -0
  351. package/templates/nextblock-template/lib/full-backup/manifest.ts +206 -0
  352. package/templates/nextblock-template/lib/full-backup/server.ts +743 -0
  353. package/templates/nextblock-template/lib/media/resolveMediaUrl.ts +45 -0
  354. package/templates/nextblock-template/lib/posts/readTime.ts +60 -0
  355. package/templates/nextblock-template/lib/privacy/consent-client.ts +57 -0
  356. package/templates/nextblock-template/lib/privacy/settings.ts +103 -0
  357. package/templates/nextblock-template/lib/privacy/types.ts +67 -0
  358. package/templates/nextblock-template/lib/promotions/server.test.ts +74 -0
  359. package/templates/nextblock-template/lib/promotions/server.ts +741 -0
  360. package/templates/nextblock-template/lib/resolve-block-relations.test.ts +142 -0
  361. package/templates/nextblock-template/lib/resolve-block-relations.ts +255 -0
  362. package/templates/nextblock-template/lib/search/server.ts +585 -0
  363. package/templates/nextblock-template/lib/search/types.ts +27 -0
  364. package/templates/nextblock-template/lib/visual-editing/draft-content.test.ts +105 -0
  365. package/templates/nextblock-template/lib/visual-editing/draft-content.ts +380 -0
  366. package/templates/nextblock-template/lib/visual-editing/draft-route.test.ts +42 -0
  367. package/templates/nextblock-template/lib/visual-editing/draft-route.ts +82 -0
  368. package/templates/nextblock-template/lib/visual-editing/edit-info.test.ts +143 -0
  369. package/templates/nextblock-template/lib/visual-editing/edit-info.ts +94 -0
  370. package/templates/nextblock-template/lib/visual-editing/mutations.ts +190 -0
  371. package/templates/nextblock-template/lib/visual-editing/product-drafts.test.ts +81 -0
  372. package/templates/nextblock-template/lib/visual-editing/product-drafts.ts +511 -0
  373. package/templates/nextblock-template/lib/visual-editing/types.ts +122 -0
  374. package/templates/nextblock-template/lib/zod-config.ts +5 -0
  375. package/templates/nextblock-template/next.config.js +190 -66
  376. package/templates/nextblock-template/package.json +34 -30
  377. package/templates/nextblock-template/proxy.ts +435 -253
  378. package/templates/nextblock-template/public/images/NBcover.webp +0 -0
  379. package/templates/nextblock-template/public/images/cap.webp +0 -0
  380. package/templates/nextblock-template/public/images/commerce-plan.webp +0 -0
  381. package/templates/nextblock-template/public/images/commerce-square.webp +0 -0
  382. package/templates/nextblock-template/public/images/commerce-wide.webp +0 -0
  383. package/templates/nextblock-template/public/images/cortex-ai-square.webp +0 -0
  384. package/templates/nextblock-template/public/images/cortex-ai.webp +0 -0
  385. package/templates/nextblock-template/public/images/extensibility.webp +0 -0
  386. package/templates/nextblock-template/public/images/goals.webp +0 -0
  387. package/templates/nextblock-template/public/images/included.webp +0 -0
  388. package/templates/nextblock-template/public/images/nx-graph.webp +0 -0
  389. package/templates/nextblock-template/public/images/pants.webp +0 -0
  390. package/templates/nextblock-template/public/images/t-shirt.webp +0 -0
  391. package/templates/nextblock-template/scripts/validate-editor-block-schema.ts +112 -0
  392. package/templates/nextblock-template/scripts/verify-cortex-ai-build-widget.tsx +100 -0
  393. package/templates/nextblock-template/scripts/verify-cortex-ai-generate-blocks.ts +62 -0
  394. package/templates/nextblock-template/scripts/verify-cortex-ai-global-tools.ts +537 -0
  395. package/templates/nextblock-template/scripts/verify-cortex-ai-routing.ts +58 -0
  396. package/templates/nextblock-template/scripts/verify-custom-block-definitions.ts +188 -0
  397. package/templates/nextblock-template/scripts/verify-dynamic-custom-block-extensions.ts +123 -0
  398. package/templates/nextblock-template/scripts/verify-dynamic-layout-engine.tsx +133 -0
  399. package/templates/nextblock-template/scripts/verify-milestone-2-custom-blocks.ts +65 -0
  400. package/templates/nextblock-template/tailwind.config.js +1 -0
  401. package/templates/nextblock-template/tools/configure-supabase-auth.js +282 -0
  402. package/templates/nextblock-template/tools/deploy-supabase.js +69 -71
  403. package/templates/nextblock-template/tsconfig.json +52 -66
  404. package/templates/nextblock-template/tsconfig.tsbuildinfo +1 -1
  405. package/templates/nextblock-template/types/jsdom.d.ts +6 -0
  406. package/templates/nextblock-template/app/force-styles.tsx +0 -31
  407. package/templates/nextblock-template/app/sitemap.xml/route.ts +0 -63
  408. package/templates/nextblock-template/components/blocks/renderers/HeroBlockRenderer.tsx +0 -273
  409. package/templates/nextblock-template/docs/How to Create a Custom Block.md +0 -149
  410. package/templates/nextblock-template/docs/cms-application-overview.md +0 -56
  411. package/templates/nextblock-template/docs/cms-architecture-overview.md +0 -73
  412. package/templates/nextblock-template/docs/files-structure.md +0 -426
  413. package/templates/nextblock-template/docs/tiptap-bundle-optimization-summary.md +0 -174
@@ -0,0 +1,494 @@
1
+ import { createClient } from '@nextblock-cms/db/server';
2
+ import {
3
+ describeCurrencyRoundingRule,
4
+ normalizeCurrencyRecord,
5
+ type CurrencyRecord,
6
+ } from '@nextblock-cms/ecommerce/server';
7
+ import { formatPrice, minorUnitAmountToMajor } from '@nextblock-cms/utils';
8
+ import { Badge, Button, Input, Label } from '@nextblock-cms/ui';
9
+
10
+ import {
11
+ deleteCurrencyAction,
12
+ syncCurrencyRatesAction,
13
+ upsertCurrencyAction,
14
+ } from './actions';
15
+
16
+ const ROUNDING_MODE_OPTIONS = [
17
+ { value: 'none', label: 'Exact conversion' },
18
+ { value: 'nearest', label: 'Round to nearest step' },
19
+ { value: 'up', label: 'Always round up' },
20
+ { value: 'down', label: 'Always round down' },
21
+ { value: 'charm', label: 'Charm ending' },
22
+ ] as const;
23
+
24
+ type CurrencySettingsRow = CurrencyRecord & {
25
+ created_at: string;
26
+ id: string;
27
+ updated_at: string;
28
+ };
29
+
30
+ function formatExchangeRate(exchangeRate: number) {
31
+ return Number(exchangeRate).toFixed(6);
32
+ }
33
+
34
+ function formatMinorUnitInput(amount: number | null | undefined, currencyCode: string) {
35
+ if (typeof amount !== 'number') {
36
+ return '';
37
+ }
38
+
39
+ return String(minorUnitAmountToMajor(amount, currencyCode));
40
+ }
41
+
42
+ function buildRateStatusText(currency: CurrencyRecord) {
43
+ if (currency.is_default) {
44
+ return 'Base currency. Rate is locked to 1.000000 and live syncing stays off.';
45
+ }
46
+
47
+ if (currency.exchange_rate_updated_at) {
48
+ const updatedAt = new Date(currency.exchange_rate_updated_at).toLocaleString();
49
+ const source = currency.exchange_rate_source || 'manual';
50
+ return `Last rate update ${updatedAt} via ${source}.`;
51
+ }
52
+
53
+ if (currency.exchange_rate_source) {
54
+ return `Rate source: ${currency.exchange_rate_source}.`;
55
+ }
56
+
57
+ return 'No live FX sync has been run yet.';
58
+ }
59
+
60
+ function buildProductPricingStatusText(currency: CurrencyRecord) {
61
+ if (currency.is_default) {
62
+ return 'Base product prices are entered directly in this currency.';
63
+ }
64
+
65
+ if (currency.auto_sync_product_prices) {
66
+ return 'Product and variant prices are derived automatically from the base currency using the FX rate and rounding rule below.';
67
+ }
68
+
69
+ return 'Merchants can set explicit product and variant prices for this currency.';
70
+ }
71
+
72
+ async function getCurrencies() {
73
+ const supabase = createClient();
74
+ const { data, error } = await supabase
75
+ .from('currencies')
76
+ .select(
77
+ 'id, code, symbol, exchange_rate, is_default, is_active, auto_update_exchange_rate, auto_sync_product_prices, exchange_rate_source, exchange_rate_updated_at, rounding_mode, rounding_increment, rounding_charm_amount, created_at, updated_at'
78
+ )
79
+ .order('is_default', { ascending: false })
80
+ .order('code', { ascending: true });
81
+
82
+ if (error) {
83
+ throw new Error(error.message);
84
+ }
85
+
86
+ return (data || []).map(
87
+ (currency) =>
88
+ ({
89
+ ...currency,
90
+ ...normalizeCurrencyRecord(currency),
91
+ }) as CurrencySettingsRow
92
+ );
93
+ }
94
+
95
+ export default async function CmsCurrenciesPage({
96
+ searchParams,
97
+ }: {
98
+ searchParams?: Promise<{ error?: string; success?: string }>;
99
+ }) {
100
+ const [currencies, resolvedSearchParams] = await Promise.all([
101
+ getCurrencies(),
102
+ searchParams,
103
+ ]);
104
+ const successMessage = resolvedSearchParams?.success || null;
105
+ const errorMessage = resolvedSearchParams?.error || null;
106
+
107
+ return (
108
+ <div className="space-y-8">
109
+ <div className="space-y-2">
110
+ <h1 className="text-2xl font-semibold">Currency Settings</h1>
111
+ <p className="max-w-3xl text-sm text-muted-foreground">
112
+ Manage shopper-facing currencies, store base pricing, live FX syncing, and the
113
+ rounding rules that keep converted prices looking intentional instead of raw.
114
+ You can also decide whether product prices in each currency are entered manually
115
+ or derived automatically from the store default currency.
116
+ </p>
117
+ </div>
118
+
119
+ {successMessage ? (
120
+ <div className="rounded-lg border border-emerald-200 bg-emerald-50 px-4 py-3 text-sm text-emerald-700">
121
+ {successMessage}
122
+ </div>
123
+ ) : null}
124
+
125
+ {errorMessage ? (
126
+ <div className="rounded-lg border border-red-200 bg-red-50 px-4 py-3 text-sm text-red-700">
127
+ {errorMessage}
128
+ </div>
129
+ ) : null}
130
+
131
+ <section className="space-y-4 rounded-lg border bg-card p-6 shadow-sm">
132
+ <div className="flex flex-wrap items-start justify-between gap-4">
133
+ <div className="space-y-1">
134
+ <h2 className="text-lg font-semibold">Live FX Sync</h2>
135
+ <p className="max-w-3xl text-sm text-muted-foreground">
136
+ Rates can be refreshed manually here and are also ready for a daily cron sync.
137
+ Only currencies with auto-update enabled are refreshed, while the default
138
+ currency stays fixed at 1.
139
+ </p>
140
+ </div>
141
+ <form action={syncCurrencyRatesAction}>
142
+ <Button type="submit">Sync Live Rates</Button>
143
+ </form>
144
+ </div>
145
+ <div className="rounded-lg border border-dashed bg-muted/20 p-4 text-sm text-muted-foreground">
146
+ Use rounding steps like <strong>0.05</strong> for nickel rounding or charm endings
147
+ like <strong>0.90</strong> to auto-fill prices such as <strong>29.90</strong> and{' '}
148
+ <strong>39.00</strong>.
149
+ </div>
150
+ <div className="rounded-lg border border-dashed bg-muted/20 p-4 text-sm text-muted-foreground">
151
+ Store-managed product pricing disables manual product and variant inputs for that
152
+ currency and derives them from the store default price instead.
153
+ </div>
154
+ </section>
155
+
156
+ <section className="space-y-4 rounded-lg border bg-card p-6 shadow-sm">
157
+ <div className="space-y-1">
158
+ <h2 className="text-lg font-semibold">Add Currency</h2>
159
+ <p className="text-sm text-muted-foreground">
160
+ Add a new ISO 4217 currency, decide whether it should auto-sync, and define the
161
+ storefront rounding behavior merchants want shoppers to see.
162
+ </p>
163
+ </div>
164
+
165
+ <form action={upsertCurrencyAction} className="grid gap-4 md:grid-cols-2 xl:grid-cols-4">
166
+ <div className="space-y-2">
167
+ <Label htmlFor="new-code">Code</Label>
168
+ <Input id="new-code" name="code" placeholder="USD" maxLength={3} required />
169
+ </div>
170
+ <div className="space-y-2">
171
+ <Label htmlFor="new-symbol">Symbol</Label>
172
+ <Input id="new-symbol" name="symbol" placeholder="$" required />
173
+ </div>
174
+ <div className="space-y-2">
175
+ <Label htmlFor="new-rate">Exchange Rate</Label>
176
+ <Input
177
+ id="new-rate"
178
+ name="exchange_rate"
179
+ type="number"
180
+ min="0.000001"
181
+ step="0.000001"
182
+ defaultValue="1"
183
+ required
184
+ />
185
+ </div>
186
+ <div className="space-y-2">
187
+ <Label htmlFor="new-rounding-mode">Rounding Mode</Label>
188
+ <select
189
+ id="new-rounding-mode"
190
+ name="rounding_mode"
191
+ defaultValue="none"
192
+ className="flex h-10 w-full rounded-md border border-input bg-background px-3 py-2 text-sm"
193
+ >
194
+ {ROUNDING_MODE_OPTIONS.map((option) => (
195
+ <option key={option.value} value={option.value}>
196
+ {option.label}
197
+ </option>
198
+ ))}
199
+ </select>
200
+ </div>
201
+ <div className="space-y-2">
202
+ <Label htmlFor="new-rounding-increment">Rounding Step</Label>
203
+ <Input
204
+ id="new-rounding-increment"
205
+ name="rounding_increment"
206
+ type="number"
207
+ min="0.01"
208
+ step="0.01"
209
+ defaultValue="0.01"
210
+ required
211
+ />
212
+ </div>
213
+ <div className="space-y-2">
214
+ <Label htmlFor="new-rounding-charm">Charm Ending</Label>
215
+ <Input
216
+ id="new-rounding-charm"
217
+ name="rounding_charm_amount"
218
+ type="number"
219
+ min="0"
220
+ step="0.01"
221
+ placeholder="0.90"
222
+ />
223
+ </div>
224
+ <div className="flex flex-col justify-end gap-3 rounded-lg border bg-muted/20 px-4 py-3 text-sm">
225
+ <label className="flex items-center gap-2">
226
+ <input type="checkbox" name="is_active" defaultChecked className="h-4 w-4" />
227
+ Active on storefront
228
+ </label>
229
+ <label className="flex items-center gap-2">
230
+ <input
231
+ type="checkbox"
232
+ name="auto_update_exchange_rate"
233
+ defaultChecked
234
+ className="h-4 w-4"
235
+ />
236
+ Auto-sync live FX
237
+ </label>
238
+ <label className="flex items-center gap-2">
239
+ <input
240
+ type="checkbox"
241
+ name="auto_sync_product_prices"
242
+ defaultChecked
243
+ className="h-4 w-4"
244
+ />
245
+ Auto-sync product prices
246
+ </label>
247
+ <label className="flex items-center gap-2">
248
+ <input type="checkbox" name="is_default" className="h-4 w-4" />
249
+ Make default currency
250
+ </label>
251
+ </div>
252
+ <div className="flex items-end justify-end">
253
+ <Button type="submit">Save Currency</Button>
254
+ </div>
255
+ </form>
256
+
257
+ <p className="text-xs text-muted-foreground">
258
+ For zero-decimal currencies like JPY, use whole-number rounding steps such as{' '}
259
+ <strong>1</strong> or <strong>10</strong>.
260
+ </p>
261
+ </section>
262
+
263
+ <section className="space-y-4">
264
+ {currencies.map((currency) => {
265
+ const roundingRule = describeCurrencyRoundingRule(currency);
266
+
267
+ return (
268
+ <div key={currency.code} className="space-y-4 rounded-lg border bg-card p-6 shadow-sm">
269
+ <div className="flex flex-wrap items-start justify-between gap-4">
270
+ <div className="space-y-2">
271
+ <div className="flex flex-wrap items-center gap-2">
272
+ <h2 className="text-lg font-semibold">{currency.code}</h2>
273
+ <Badge variant="outline">{currency.symbol}</Badge>
274
+ {currency.is_default ? <Badge>Default</Badge> : null}
275
+ {currency.is_active ? (
276
+ <Badge variant="secondary">Active</Badge>
277
+ ) : (
278
+ <Badge variant="secondary">Inactive</Badge>
279
+ )}
280
+ {currency.auto_update_exchange_rate && !currency.is_default ? (
281
+ <Badge variant="outline">Auto FX</Badge>
282
+ ) : null}
283
+ {currency.auto_sync_product_prices && !currency.is_default ? (
284
+ <Badge variant="outline">Auto Prices</Badge>
285
+ ) : null}
286
+ </div>
287
+ <p className="max-w-3xl text-sm text-muted-foreground">
288
+ {buildRateStatusText(currency)}
289
+ </p>
290
+ <p className="max-w-3xl text-sm text-muted-foreground">
291
+ {buildProductPricingStatusText(currency)}
292
+ </p>
293
+ </div>
294
+ <div className="space-y-1 text-right text-xs text-muted-foreground">
295
+ <div>Exchange rate: {formatExchangeRate(currency.exchange_rate)}</div>
296
+ <div>{roundingRule}</div>
297
+ <div>Updated {new Date(currency.updated_at).toLocaleString()}</div>
298
+ </div>
299
+ </div>
300
+
301
+ <div className="grid gap-4 rounded-lg border bg-muted/20 p-4 md:grid-cols-3 xl:grid-cols-4">
302
+ <div className="rounded-lg border bg-background p-3">
303
+ <div className="text-xs uppercase tracking-wide text-muted-foreground">
304
+ Rounding Step
305
+ </div>
306
+ <div className="mt-1 font-medium">
307
+ {formatPrice(currency.rounding_increment ?? 1, currency.code)}
308
+ </div>
309
+ </div>
310
+ <div className="rounded-lg border bg-background p-3">
311
+ <div className="text-xs uppercase tracking-wide text-muted-foreground">
312
+ Charm Ending
313
+ </div>
314
+ <div className="mt-1 font-medium">
315
+ {typeof currency.rounding_charm_amount === 'number'
316
+ ? formatPrice(currency.rounding_charm_amount, currency.code)
317
+ : 'Not set'}
318
+ </div>
319
+ </div>
320
+ <div className="rounded-lg border bg-background p-3">
321
+ <div className="text-xs uppercase tracking-wide text-muted-foreground">
322
+ Rate Source
323
+ </div>
324
+ <div className="mt-1 font-medium">
325
+ {currency.exchange_rate_source || 'Manual'}
326
+ </div>
327
+ </div>
328
+ <div className="rounded-lg border bg-background p-3">
329
+ <div className="text-xs uppercase tracking-wide text-muted-foreground">
330
+ Product Pricing
331
+ </div>
332
+ <div className="mt-1 font-medium">
333
+ {currency.is_default
334
+ ? 'Base prices'
335
+ : currency.auto_sync_product_prices
336
+ ? 'Store-managed'
337
+ : 'Manual'}
338
+ </div>
339
+ </div>
340
+ <div className="rounded-lg border bg-background p-3">
341
+ <div className="text-xs uppercase tracking-wide text-muted-foreground">
342
+ Last FX Sync
343
+ </div>
344
+ <div className="mt-1 font-medium">
345
+ {currency.exchange_rate_updated_at
346
+ ? new Date(currency.exchange_rate_updated_at).toLocaleString()
347
+ : 'Never'}
348
+ </div>
349
+ </div>
350
+ </div>
351
+
352
+ <div className="grid gap-4 md:grid-cols-[1fr_auto]">
353
+ <form action={upsertCurrencyAction} className="grid gap-4 md:grid-cols-2 xl:grid-cols-4">
354
+ <input type="hidden" name="id" value={currency.id} />
355
+ <div className="space-y-2">
356
+ <Label htmlFor={`code-${currency.id}`}>Code</Label>
357
+ <Input
358
+ id={`code-${currency.id}`}
359
+ name="code"
360
+ defaultValue={currency.code}
361
+ maxLength={3}
362
+ required
363
+ />
364
+ </div>
365
+ <div className="space-y-2">
366
+ <Label htmlFor={`symbol-${currency.id}`}>Symbol</Label>
367
+ <Input
368
+ id={`symbol-${currency.id}`}
369
+ name="symbol"
370
+ defaultValue={currency.symbol}
371
+ required
372
+ />
373
+ </div>
374
+ <div className="space-y-2">
375
+ <Label htmlFor={`rate-${currency.id}`}>Exchange Rate</Label>
376
+ <Input
377
+ id={`rate-${currency.id}`}
378
+ name="exchange_rate"
379
+ type="number"
380
+ min="0.000001"
381
+ step="0.000001"
382
+ defaultValue={formatExchangeRate(currency.exchange_rate)}
383
+ required
384
+ />
385
+ </div>
386
+ <div className="space-y-2">
387
+ <Label htmlFor={`rounding-mode-${currency.id}`}>Rounding Mode</Label>
388
+ <select
389
+ id={`rounding-mode-${currency.id}`}
390
+ name="rounding_mode"
391
+ defaultValue={currency.rounding_mode || 'none'}
392
+ className="flex h-10 w-full rounded-md border border-input bg-background px-3 py-2 text-sm"
393
+ >
394
+ {ROUNDING_MODE_OPTIONS.map((option) => (
395
+ <option key={option.value} value={option.value}>
396
+ {option.label}
397
+ </option>
398
+ ))}
399
+ </select>
400
+ </div>
401
+ <div className="space-y-2">
402
+ <Label htmlFor={`rounding-increment-${currency.id}`}>Rounding Step</Label>
403
+ <Input
404
+ id={`rounding-increment-${currency.id}`}
405
+ name="rounding_increment"
406
+ type="number"
407
+ min="0.01"
408
+ step="0.01"
409
+ defaultValue={formatMinorUnitInput(
410
+ currency.rounding_increment,
411
+ currency.code
412
+ )}
413
+ required
414
+ />
415
+ </div>
416
+ <div className="space-y-2">
417
+ <Label htmlFor={`rounding-charm-${currency.id}`}>Charm Ending</Label>
418
+ <Input
419
+ id={`rounding-charm-${currency.id}`}
420
+ name="rounding_charm_amount"
421
+ type="number"
422
+ min="0"
423
+ step="0.01"
424
+ defaultValue={formatMinorUnitInput(
425
+ currency.rounding_charm_amount,
426
+ currency.code
427
+ )}
428
+ placeholder="0.90"
429
+ />
430
+ </div>
431
+ <div className="flex flex-col justify-end gap-3 rounded-lg border bg-muted/20 px-4 py-3 text-sm">
432
+ <label className="flex items-center gap-2">
433
+ <input
434
+ type="checkbox"
435
+ name="is_active"
436
+ defaultChecked={currency.is_active}
437
+ className="h-4 w-4"
438
+ />
439
+ Active on storefront
440
+ </label>
441
+ <label className="flex items-center gap-2">
442
+ <input
443
+ type="checkbox"
444
+ name="auto_update_exchange_rate"
445
+ defaultChecked={
446
+ currency.is_default
447
+ ? false
448
+ : currency.auto_update_exchange_rate !== false
449
+ }
450
+ className="h-4 w-4"
451
+ />
452
+ Auto-sync live FX
453
+ </label>
454
+ <label className="flex items-center gap-2">
455
+ <input
456
+ type="checkbox"
457
+ name="auto_sync_product_prices"
458
+ defaultChecked={
459
+ currency.is_default ? false : currency.auto_sync_product_prices === true
460
+ }
461
+ disabled={currency.is_default}
462
+ className="h-4 w-4"
463
+ />
464
+ Auto-sync product prices
465
+ </label>
466
+ <label className="flex items-center gap-2">
467
+ <input
468
+ type="checkbox"
469
+ name="is_default"
470
+ defaultChecked={currency.is_default}
471
+ className="h-4 w-4"
472
+ />
473
+ Default currency
474
+ </label>
475
+ </div>
476
+ <div className="flex items-end justify-end">
477
+ <Button type="submit">Update</Button>
478
+ </div>
479
+ </form>
480
+
481
+ <form action={deleteCurrencyAction} className="flex items-end">
482
+ <input type="hidden" name="id" value={currency.id} />
483
+ <Button type="submit" variant="outline" disabled={currency.is_default}>
484
+ Delete
485
+ </Button>
486
+ </form>
487
+ </div>
488
+ </div>
489
+ );
490
+ })}
491
+ </section>
492
+ </div>
493
+ );
494
+ }