create-nextblock 0.2.77 → 0.8.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (413) hide show
  1. package/bin/create-nextblock.js +740 -459
  2. package/package.json +1 -2
  3. package/scripts/sync-template.js +18 -1
  4. package/templates/nextblock-template/.browserslistrc +11 -0
  5. package/templates/nextblock-template/.swcrc +30 -30
  6. package/templates/nextblock-template/README.md +23 -114
  7. package/templates/nextblock-template/app/(auth-pages)/post-sign-in/page.tsx +27 -28
  8. package/templates/nextblock-template/app/(auth-pages)/sign-in/page.tsx +50 -25
  9. package/templates/nextblock-template/app/(auth-pages)/sign-up/page.tsx +111 -56
  10. package/templates/nextblock-template/app/(auth-pages)/two-factor/actions.ts +91 -0
  11. package/templates/nextblock-template/app/(auth-pages)/two-factor/components/TwoFactorForm.tsx +118 -0
  12. package/templates/nextblock-template/app/(auth-pages)/two-factor/page.tsx +51 -0
  13. package/templates/nextblock-template/app/.well-known/ucp/route.ts +16 -0
  14. package/templates/nextblock-template/app/[slug]/PageClientContent.tsx +48 -28
  15. package/templates/nextblock-template/app/[slug]/page.tsx +63 -6
  16. package/templates/nextblock-template/app/[slug]/page.utils.ts +374 -157
  17. package/templates/nextblock-template/app/[slug]/pageClientActions.ts +7 -0
  18. package/templates/nextblock-template/app/actions/consent.ts +57 -0
  19. package/templates/nextblock-template/app/actions/formActions.ts +130 -11
  20. package/templates/nextblock-template/app/actions/languageActions.ts +31 -30
  21. package/templates/nextblock-template/app/actions/package-actions.ts +183 -0
  22. package/templates/nextblock-template/app/actions/postActions.ts +146 -48
  23. package/templates/nextblock-template/app/actions/twoFactorEmail.ts +21 -0
  24. package/templates/nextblock-template/app/actions/visualEditingActions.test.ts +179 -0
  25. package/templates/nextblock-template/app/actions/visualEditingActions.ts +345 -0
  26. package/templates/nextblock-template/app/actions.ts +67 -12
  27. package/templates/nextblock-template/app/api/ai/cortex/build-widget/route.ts +153 -0
  28. package/templates/nextblock-template/app/api/ai/generate-blocks/route.ts +96 -0
  29. package/templates/nextblock-template/app/api/ai/global-agent/route.ts +965 -0
  30. package/templates/nextblock-template/app/api/checkout/freemius/sync/route.ts +29 -0
  31. package/templates/nextblock-template/app/api/checkout/route.ts +146 -0
  32. package/templates/nextblock-template/app/api/cms/full-backup/export/route.ts +33 -0
  33. package/templates/nextblock-template/app/api/cms/full-backup/restore/route.ts +63 -0
  34. package/templates/nextblock-template/app/api/cron/reset-sandbox/route.ts +3413 -17
  35. package/templates/nextblock-template/app/api/cron/reset-sandbox/sandboxResetSql.ts +7830 -0
  36. package/templates/nextblock-template/app/api/cron/sync-currencies/route.ts +35 -0
  37. package/templates/nextblock-template/app/api/custom-blocks/db-relations/route.ts +92 -0
  38. package/templates/nextblock-template/app/api/custom-blocks/editor-definitions/route.ts +43 -0
  39. package/templates/nextblock-template/app/api/draft/disable/route.ts +25 -0
  40. package/templates/nextblock-template/app/api/draft/route.ts +93 -0
  41. package/templates/nextblock-template/app/api/draft/start/route.ts +77 -0
  42. package/templates/nextblock-template/app/api/media/library/route.ts +65 -0
  43. package/templates/nextblock-template/app/api/media/r2-presigned/route.ts +53 -0
  44. package/templates/nextblock-template/app/api/media/record/route.ts +160 -0
  45. package/templates/nextblock-template/app/api/search/route.ts +43 -0
  46. package/templates/nextblock-template/app/api/visual-editing/block-draft/route.ts +47 -0
  47. package/templates/nextblock-template/app/api/visual-editing/product-draft/route.ts +47 -0
  48. package/templates/nextblock-template/app/api/webhooks/freemius/route.ts +34 -0
  49. package/templates/nextblock-template/app/api/webhooks/stripe/route.ts +27 -0
  50. package/templates/nextblock-template/app/article/[slug]/PostClientContent.tsx +392 -128
  51. package/templates/nextblock-template/app/article/[slug]/page.tsx +179 -127
  52. package/templates/nextblock-template/app/article/[slug]/page.utils.ts +262 -77
  53. package/templates/nextblock-template/app/auth/callback/route.ts +31 -58
  54. package/templates/nextblock-template/app/cart/page.tsx +7 -0
  55. package/templates/nextblock-template/app/checkout/UcpCartHydrator.tsx +20 -0
  56. package/templates/nextblock-template/app/checkout/page.tsx +52 -0
  57. package/templates/nextblock-template/app/checkout/success/actions.ts +136 -0
  58. package/templates/nextblock-template/app/checkout/success/page.tsx +186 -0
  59. package/templates/nextblock-template/app/cms/CmsClientLayout.tsx +163 -33
  60. package/templates/nextblock-template/app/cms/blocks/actions.ts +424 -235
  61. package/templates/nextblock-template/app/cms/blocks/components/BackgroundSelector.tsx +212 -151
  62. package/templates/nextblock-template/app/cms/blocks/components/BlockEditorArea.tsx +41 -20
  63. package/templates/nextblock-template/app/cms/blocks/components/BlockEditorModal.tsx +152 -19
  64. package/templates/nextblock-template/app/cms/blocks/components/BlockTypeCard.tsx +25 -17
  65. package/templates/nextblock-template/app/cms/blocks/components/BlockTypeSelector.tsx +200 -18
  66. package/templates/nextblock-template/app/cms/blocks/components/ColumnEditor.tsx +33 -16
  67. package/templates/nextblock-template/app/cms/blocks/components/CustomBlockEditorPreview.tsx +160 -0
  68. package/templates/nextblock-template/app/cms/blocks/components/EditableBlock.tsx +37 -18
  69. package/templates/nextblock-template/app/cms/blocks/components/MediaLibraryModal.tsx +149 -67
  70. package/templates/nextblock-template/app/cms/blocks/components/SectionConfigPanel.tsx +108 -31
  71. package/templates/nextblock-template/app/cms/blocks/editors/DynamicCustomBlockEditor.tsx +167 -0
  72. package/templates/nextblock-template/app/cms/blocks/editors/FeaturedProductBlockEditor.tsx +31 -0
  73. package/templates/nextblock-template/app/cms/blocks/editors/FormBlockEditor.tsx +2 -2
  74. package/templates/nextblock-template/app/cms/blocks/editors/HeadingBlockEditor.tsx +1 -1
  75. package/templates/nextblock-template/app/cms/blocks/editors/ImageBlockEditor.tsx +29 -29
  76. package/templates/nextblock-template/app/cms/blocks/editors/PostsGridBlockEditor.tsx +14 -18
  77. package/templates/nextblock-template/app/cms/blocks/editors/ProductGridBlockEditor.tsx +41 -0
  78. package/templates/nextblock-template/app/cms/blocks/editors/SectionBlockEditor.tsx +318 -118
  79. package/templates/nextblock-template/app/cms/blocks/editors/TextBlockEditor.tsx +98 -21
  80. package/templates/nextblock-template/app/cms/blocks/editors/VideoEmbedBlockEditor.tsx +1 -1
  81. package/templates/nextblock-template/app/cms/components/ContentLanguageSwitcher.tsx +27 -9
  82. package/templates/nextblock-template/app/cms/components/CopyContentFromLanguage.tsx +1 -1
  83. package/templates/nextblock-template/app/cms/components/CortexAiActiveContext.tsx +23 -0
  84. package/templates/nextblock-template/app/cms/components/CortexAiPageContext.tsx +58 -0
  85. package/templates/nextblock-template/app/cms/components/CortexGlobalAgentChat.tsx +1507 -0
  86. package/templates/nextblock-template/app/cms/components/DraftStatusActions.tsx +145 -0
  87. package/templates/nextblock-template/app/cms/components/FeatureImageField.tsx +244 -0
  88. package/templates/nextblock-template/app/cms/components/FeedbackModal.tsx +38 -24
  89. package/templates/nextblock-template/app/cms/coupons/[id]/edit/page.tsx +16 -0
  90. package/templates/nextblock-template/app/cms/coupons/page.tsx +16 -0
  91. package/templates/nextblock-template/app/cms/custom-blocks/[id]/edit/page.tsx +66 -0
  92. package/templates/nextblock-template/app/cms/custom-blocks/actions.ts +519 -0
  93. package/templates/nextblock-template/app/cms/custom-blocks/components/BlockComposer.tsx +1522 -0
  94. package/templates/nextblock-template/app/cms/custom-blocks/components/BlocksLibraryTransferControls.tsx +256 -0
  95. package/templates/nextblock-template/app/cms/custom-blocks/components/DBRelationSelect.tsx +384 -0
  96. package/templates/nextblock-template/app/cms/custom-blocks/components/ImageR2Picker.tsx +221 -0
  97. package/templates/nextblock-template/app/cms/custom-blocks/new/page.tsx +12 -0
  98. package/templates/nextblock-template/app/cms/custom-blocks/page.tsx +438 -0
  99. package/templates/nextblock-template/app/cms/dashboard/actions.ts +228 -98
  100. package/templates/nextblock-template/app/cms/dashboard/components/DashboardComponents.tsx +200 -0
  101. package/templates/nextblock-template/app/cms/dashboard/page.tsx +191 -151
  102. package/templates/nextblock-template/app/cms/import-export/ContentTransferControls.tsx +391 -0
  103. package/templates/nextblock-template/app/cms/import-export/actions.ts +226 -0
  104. package/templates/nextblock-template/app/cms/layout.tsx +29 -10
  105. package/templates/nextblock-template/app/cms/media/UploadFolderContext.tsx +22 -22
  106. package/templates/nextblock-template/app/cms/media/actions.ts +45 -124
  107. package/templates/nextblock-template/app/cms/media/components/DeleteMediaButtonClient.tsx +1 -1
  108. package/templates/nextblock-template/app/cms/media/components/MediaEditForm.tsx +26 -26
  109. package/templates/nextblock-template/app/cms/media/components/MediaGridClient.tsx +69 -64
  110. package/templates/nextblock-template/app/cms/media/components/MediaPickerDialog.tsx +227 -158
  111. package/templates/nextblock-template/app/cms/media/components/MediaUploadForm.tsx +101 -89
  112. package/templates/nextblock-template/app/cms/media/page.tsx +1 -1
  113. package/templates/nextblock-template/app/cms/navigation/components/NavigationItemForm.tsx +2 -2
  114. package/templates/nextblock-template/app/cms/orders/[id]/MarkPaidButton.tsx +44 -0
  115. package/templates/nextblock-template/app/cms/orders/[id]/page.tsx +16 -0
  116. package/templates/nextblock-template/app/cms/orders/actions.ts +201 -0
  117. package/templates/nextblock-template/app/cms/orders/page.tsx +20 -0
  118. package/templates/nextblock-template/app/cms/orders/types.ts +20 -0
  119. package/templates/nextblock-template/app/cms/pages/[id]/edit/EditPageClient.tsx +156 -121
  120. package/templates/nextblock-template/app/cms/pages/[id]/edit/page.tsx +79 -26
  121. package/templates/nextblock-template/app/cms/pages/actions.ts +54 -38
  122. package/templates/nextblock-template/app/cms/pages/components/DeletePageButtonClient.tsx +1 -1
  123. package/templates/nextblock-template/app/cms/pages/components/PageForm.tsx +267 -116
  124. package/templates/nextblock-template/app/cms/pages/page.tsx +25 -18
  125. package/templates/nextblock-template/app/cms/payments/page.tsx +16 -0
  126. package/templates/nextblock-template/app/cms/posts/[id]/edit/page.tsx +132 -90
  127. package/templates/nextblock-template/app/cms/posts/actions.ts +71 -72
  128. package/templates/nextblock-template/app/cms/posts/components/DeletePostButtonClient.tsx +1 -1
  129. package/templates/nextblock-template/app/cms/posts/components/PostForm.tsx +256 -245
  130. package/templates/nextblock-template/app/cms/posts/new/page.tsx +1 -1
  131. package/templates/nextblock-template/app/cms/posts/page.tsx +20 -13
  132. package/templates/nextblock-template/app/cms/products/ClientNotionEditor.tsx +16 -0
  133. package/templates/nextblock-template/app/cms/products/ProductFormClientShell.tsx +56 -0
  134. package/templates/nextblock-template/app/cms/products/[id]/edit/page.tsx +292 -0
  135. package/templates/nextblock-template/app/cms/products/attributes/page.tsx +12 -0
  136. package/templates/nextblock-template/app/cms/products/categories/page.tsx +12 -0
  137. package/templates/nextblock-template/app/cms/products/inventory/page.tsx +13 -0
  138. package/templates/nextblock-template/app/cms/products/new/page.tsx +143 -0
  139. package/templates/nextblock-template/app/cms/products/page.tsx +42 -0
  140. package/templates/nextblock-template/app/cms/products/productFormData.ts +133 -0
  141. package/templates/nextblock-template/app/cms/products/settings/page.tsx +5 -0
  142. package/templates/nextblock-template/app/cms/promotions/PromotionsWorkspace.tsx +456 -0
  143. package/templates/nextblock-template/app/cms/promotions/actions.ts +115 -0
  144. package/templates/nextblock-template/app/cms/promotions/page.tsx +31 -0
  145. package/templates/nextblock-template/app/cms/revisions/RevisionHistoryButton.tsx +2 -2
  146. package/templates/nextblock-template/app/cms/revisions/actions.ts +285 -285
  147. package/templates/nextblock-template/app/cms/revisions/service.ts +19 -16
  148. package/templates/nextblock-template/app/cms/revisions/utils.ts +8 -3
  149. package/templates/nextblock-template/app/cms/settings/backup-restore/BackupRestoreWorkspace.tsx +1004 -0
  150. package/templates/nextblock-template/app/cms/settings/backup-restore/page.tsx +29 -0
  151. package/templates/nextblock-template/app/cms/settings/bot-protection/actions.ts +93 -0
  152. package/templates/nextblock-template/app/cms/settings/bot-protection/components/BotProtectionForm.tsx +129 -0
  153. package/templates/nextblock-template/app/cms/settings/bot-protection/page.tsx +24 -0
  154. package/templates/nextblock-template/app/cms/settings/copyright/actions.ts +1 -1
  155. package/templates/nextblock-template/app/cms/settings/copyright/components/CopyrightForm.tsx +2 -2
  156. package/templates/nextblock-template/app/cms/settings/copyright/page.tsx +1 -1
  157. package/templates/nextblock-template/app/cms/settings/cortex-ai/SandboxCortexAiSettingsClient.tsx +496 -0
  158. package/templates/nextblock-template/app/cms/settings/cortex-ai/StoredCortexAiSettingsClient.tsx +410 -0
  159. package/templates/nextblock-template/app/cms/settings/cortex-ai/actions.ts +248 -0
  160. package/templates/nextblock-template/app/cms/settings/cortex-ai/page.tsx +80 -0
  161. package/templates/nextblock-template/app/cms/settings/currencies/actions.ts +331 -0
  162. package/templates/nextblock-template/app/cms/settings/currencies/page.tsx +494 -0
  163. package/templates/nextblock-template/app/cms/settings/extra-translations/ExtraTranslationsWorkspace.tsx +767 -0
  164. package/templates/nextblock-template/app/cms/settings/extra-translations/actions.ts +203 -44
  165. package/templates/nextblock-template/app/cms/settings/extra-translations/page.tsx +93 -242
  166. package/templates/nextblock-template/app/cms/settings/global-css/actions.ts +65 -0
  167. package/templates/nextblock-template/app/cms/settings/global-css/components/GlobalCssForm.tsx +46 -0
  168. package/templates/nextblock-template/app/cms/settings/global-css/page.tsx +24 -0
  169. package/templates/nextblock-template/app/cms/settings/languages/components/DeleteLanguageButton.tsx +1 -1
  170. package/templates/nextblock-template/app/cms/settings/languages/components/LanguageForm.tsx +2 -2
  171. package/templates/nextblock-template/app/cms/settings/languages/page.tsx +1 -1
  172. package/templates/nextblock-template/app/cms/settings/logos/[id]/edit/page.tsx +7 -7
  173. package/templates/nextblock-template/app/cms/settings/logos/actions.ts +82 -6
  174. package/templates/nextblock-template/app/cms/settings/logos/components/BrandingSettingsForm.tsx +339 -0
  175. package/templates/nextblock-template/app/cms/settings/logos/components/DeleteLogoButton.tsx +21 -18
  176. package/templates/nextblock-template/app/cms/settings/logos/components/LogoForm.tsx +20 -16
  177. package/templates/nextblock-template/app/cms/settings/logos/components/SiteSeoSettingsForm.tsx +133 -0
  178. package/templates/nextblock-template/app/cms/settings/logos/new/page.tsx +8 -8
  179. package/templates/nextblock-template/app/cms/settings/logos/page.tsx +120 -82
  180. package/templates/nextblock-template/app/cms/settings/logos/types.ts +8 -8
  181. package/templates/nextblock-template/app/cms/settings/packages/activation-form.tsx +84 -0
  182. package/templates/nextblock-template/app/cms/settings/packages/package-card.tsx +122 -0
  183. package/templates/nextblock-template/app/cms/settings/packages/page.tsx +49 -0
  184. package/templates/nextblock-template/app/cms/settings/privacy/actions.ts +53 -0
  185. package/templates/nextblock-template/app/cms/settings/privacy/components/PrivacyForm.tsx +196 -0
  186. package/templates/nextblock-template/app/cms/settings/privacy/page.tsx +26 -0
  187. package/templates/nextblock-template/app/cms/settings/security/actions.ts +251 -0
  188. package/templates/nextblock-template/app/cms/settings/security/components/SecurityPanel.tsx +453 -0
  189. package/templates/nextblock-template/app/cms/settings/security/page.tsx +13 -0
  190. package/templates/nextblock-template/app/cms/settings/taxes/page.tsx +21 -0
  191. package/templates/nextblock-template/app/cms/shipping/page.tsx +20 -0
  192. package/templates/nextblock-template/app/cms/users/[id]/edit/page.tsx +28 -23
  193. package/templates/nextblock-template/app/cms/users/actions.ts +105 -40
  194. package/templates/nextblock-template/app/cms/users/components/DeleteUserButton.tsx +1 -1
  195. package/templates/nextblock-template/app/cms/users/components/UserForm.tsx +65 -152
  196. package/templates/nextblock-template/app/cms/users/page.tsx +15 -10
  197. package/templates/nextblock-template/app/globals.css +9 -0
  198. package/templates/nextblock-template/app/layout.tsx +372 -116
  199. package/templates/nextblock-template/app/lib/seo.test.ts +52 -0
  200. package/templates/nextblock-template/app/lib/seo.ts +279 -0
  201. package/templates/nextblock-template/app/lib/site-settings.ts +87 -0
  202. package/templates/nextblock-template/app/lib/sitemap-utils.ts +224 -39
  203. package/templates/nextblock-template/app/lib/ucp/protocol.ts +190 -0
  204. package/templates/nextblock-template/app/lib/ucp/server.test.ts +56 -0
  205. package/templates/nextblock-template/app/lib/ucp/server.ts +1914 -0
  206. package/templates/nextblock-template/app/page.tsx +165 -73
  207. package/templates/nextblock-template/app/product/[slug]/page.tsx +433 -0
  208. package/templates/nextblock-template/app/profile/ProfileAccountSidebar.tsx +73 -0
  209. package/templates/nextblock-template/app/profile/ProfilePageHeader.tsx +16 -0
  210. package/templates/nextblock-template/app/profile/ProfilePageMissingState.tsx +9 -0
  211. package/templates/nextblock-template/app/profile/account-data.ts +37 -0
  212. package/templates/nextblock-template/app/profile/account-links.ts +22 -0
  213. package/templates/nextblock-template/app/profile/account-types.ts +11 -0
  214. package/templates/nextblock-template/app/profile/orders/CustomerOrdersPageClient.tsx +124 -0
  215. package/templates/nextblock-template/app/profile/orders/[id]/CustomerOrderDetailPageClient.tsx +79 -0
  216. package/templates/nextblock-template/app/profile/orders/[id]/page.tsx +32 -0
  217. package/templates/nextblock-template/app/profile/orders/page.tsx +19 -0
  218. package/templates/nextblock-template/app/profile/page.tsx +51 -0
  219. package/templates/nextblock-template/app/profile/password/PasswordSettingsPageClient.tsx +128 -0
  220. package/templates/nextblock-template/app/profile/password/actions.ts +59 -0
  221. package/templates/nextblock-template/app/profile/password/page.tsx +27 -0
  222. package/templates/nextblock-template/app/providers.tsx +55 -17
  223. package/templates/nextblock-template/app/robots.txt/route.ts +11 -1
  224. package/templates/nextblock-template/app/sitemap.ts +128 -0
  225. package/templates/nextblock-template/app/ucp/v1/carts/[id]/cancel/route.ts +38 -0
  226. package/templates/nextblock-template/app/ucp/v1/carts/[id]/route.ts +68 -0
  227. package/templates/nextblock-template/app/ucp/v1/carts/route.ts +35 -0
  228. package/templates/nextblock-template/app/ucp/v1/catalog/lookup/route.ts +35 -0
  229. package/templates/nextblock-template/app/ucp/v1/catalog/product/route.ts +35 -0
  230. package/templates/nextblock-template/app/ucp/v1/catalog/search/route.ts +34 -0
  231. package/templates/nextblock-template/components/AppShell.tsx +154 -0
  232. package/templates/nextblock-template/components/BlockRenderer.tsx +210 -64
  233. package/templates/nextblock-template/components/CartDrawerLoader.tsx +7 -0
  234. package/templates/nextblock-template/components/CartTranslator.tsx +210 -0
  235. package/templates/nextblock-template/components/CurrentContentSetter.tsx +25 -0
  236. package/templates/nextblock-template/components/DeferredCartDrawer.tsx +23 -0
  237. package/templates/nextblock-template/components/DeferredCartTranslator.tsx +51 -0
  238. package/templates/nextblock-template/components/DeferredGlobalSearch.tsx +68 -0
  239. package/templates/nextblock-template/components/DeferredGoogleTagManager.tsx +70 -0
  240. package/templates/nextblock-template/components/DeferredSpeedInsights.tsx +69 -0
  241. package/templates/nextblock-template/components/FeatureImageHero.tsx +47 -0
  242. package/templates/nextblock-template/components/GitHubLoginButton.tsx +36 -0
  243. package/templates/nextblock-template/components/GlobalSearch.tsx +557 -0
  244. package/templates/nextblock-template/components/Header.tsx +49 -41
  245. package/templates/nextblock-template/components/LanguageSwitcher.tsx +55 -32
  246. package/templates/nextblock-template/components/ResponsiveNav.tsx +138 -43
  247. package/templates/nextblock-template/components/blocks/PostCardSkeleton.tsx +12 -8
  248. package/templates/nextblock-template/components/blocks/PostsGridBlock.tsx +12 -55
  249. package/templates/nextblock-template/components/blocks/PostsGridClient.tsx +42 -37
  250. package/templates/nextblock-template/components/blocks/TestimonialBlock.tsx +6 -2
  251. package/templates/nextblock-template/components/blocks/ecommerceRendererLoaders.ts +23 -0
  252. package/templates/nextblock-template/components/blocks/publicRendererLoaders.ts +25 -0
  253. package/templates/nextblock-template/components/blocks/renderers/ButtonBlockRenderer.tsx +92 -84
  254. package/templates/nextblock-template/components/blocks/renderers/CartBlockRenderer.tsx +17 -0
  255. package/templates/nextblock-template/components/blocks/renderers/CheckoutBlockRenderer.tsx +19 -0
  256. package/templates/nextblock-template/components/blocks/renderers/ClientTextBlockRenderer.tsx +262 -8
  257. package/templates/nextblock-template/components/blocks/renderers/FeaturedProductBlockRenderer.tsx +22 -0
  258. package/templates/nextblock-template/components/blocks/renderers/FormBlockRenderer.tsx +320 -37
  259. package/templates/nextblock-template/components/blocks/renderers/HeadingBlockRenderer.tsx +11 -8
  260. package/templates/nextblock-template/components/blocks/renderers/ImageBlockRenderer.tsx +12 -3
  261. package/templates/nextblock-template/components/blocks/renderers/PostsGridBlockRenderer.tsx +18 -13
  262. package/templates/nextblock-template/components/blocks/renderers/ProductDetailsBlockRenderer.tsx +90 -0
  263. package/templates/nextblock-template/components/blocks/renderers/ProductGridBlockRenderer.tsx +31 -0
  264. package/templates/nextblock-template/components/blocks/renderers/SectionBlockRenderer.tsx +424 -55
  265. package/templates/nextblock-template/components/blocks/renderers/SectionSlider.tsx +137 -0
  266. package/templates/nextblock-template/components/blocks/renderers/TestimonialBlockRenderer.tsx +57 -0
  267. package/templates/nextblock-template/components/blocks/renderers/TextBlockRenderer.tsx +37 -22
  268. package/templates/nextblock-template/components/blocks/renderers/VideoEmbedBlockRenderer.tsx +23 -15
  269. package/templates/nextblock-template/components/blocks/renderers/inline/AlertWidgetRenderer.tsx +1 -3
  270. package/templates/nextblock-template/components/blocks/renderers/inline/CtaWidgetRenderer.tsx +1 -3
  271. package/templates/nextblock-template/components/blocks/types.ts +7 -6
  272. package/templates/nextblock-template/components/env-var-warning.tsx +3 -3
  273. package/templates/nextblock-template/components/form-message.tsx +32 -26
  274. package/templates/nextblock-template/components/header-auth.tsx +69 -17
  275. package/templates/nextblock-template/components/privacy/ConsentBanner.tsx +127 -0
  276. package/templates/nextblock-template/components/privacy/ConsentGatedAnalytics.tsx +59 -0
  277. package/templates/nextblock-template/components/renderers/CachedDynamicLayoutEngine.tsx +28 -0
  278. package/templates/nextblock-template/components/renderers/DynamicLayoutEngine.test.tsx +166 -0
  279. package/templates/nextblock-template/components/renderers/DynamicLayoutEngine.tsx +464 -0
  280. package/templates/nextblock-template/components/theme-switcher.tsx +8 -8
  281. package/templates/nextblock-template/components/visual-editing/DeferredVisualEditing.tsx +21 -0
  282. package/templates/nextblock-template/components/visual-editing/NextblockVisualEditing.tsx +1172 -0
  283. package/templates/nextblock-template/context/AuthContext.tsx +23 -90
  284. package/templates/nextblock-template/context/CurrentContentContext.tsx +10 -4
  285. package/templates/nextblock-template/context/LanguageContext.tsx +16 -16
  286. package/templates/nextblock-template/context/language-rest-client.ts +31 -0
  287. package/templates/nextblock-template/docs/01-PROJECT-OVERVIEW.md +94 -0
  288. package/templates/nextblock-template/docs/02-ECOMMERCE-CAPABILITIES.md +364 -0
  289. package/templates/nextblock-template/docs/03-CMS-AND-EDITOR.md +202 -0
  290. package/templates/nextblock-template/docs/04-DATABASE-AND-AUTH.md +252 -0
  291. package/templates/nextblock-template/docs/05-DEVELOPER-GUIDE.md +238 -0
  292. package/templates/nextblock-template/docs/06-CLI-AND-SCAFFOLDING.md +125 -0
  293. package/templates/nextblock-template/docs/07-BLOCK-SDK-AND-EXTENSIBILITY.md +146 -0
  294. package/templates/nextblock-template/docs/08-NEXTBLOCK-CORTEX-AI-ARCHITECTURE.md +1319 -0
  295. package/templates/nextblock-template/docs/09-LIVE-DRAFT-MODE.md +104 -0
  296. package/templates/nextblock-template/docs/10-CUSTOM-BLOCKS.md +222 -0
  297. package/templates/nextblock-template/docs/README.md +34 -0
  298. package/templates/nextblock-template/docs/TECHNICAL_SPECIFICATION.md +12507 -0
  299. package/templates/nextblock-template/hooks/use-hotkeys.ts +21 -14
  300. package/templates/nextblock-template/hooks/useGlobalSearch.ts +101 -0
  301. package/templates/nextblock-template/index.d.ts +2 -0
  302. package/templates/nextblock-template/lib/ai-block-generation.ts +339 -0
  303. package/templates/nextblock-template/lib/ai-client.ts +247 -0
  304. package/templates/nextblock-template/lib/ai-config.ts +81 -0
  305. package/templates/nextblock-template/lib/ai-cortex-widget-builder.ts +125 -0
  306. package/templates/nextblock-template/lib/ai-global-agent-custom-block-tools.ts +363 -0
  307. package/templates/nextblock-template/lib/ai-global-agent-db-tools.test.ts +405 -0
  308. package/templates/nextblock-template/lib/ai-global-agent-db-tools.ts +1228 -0
  309. package/templates/nextblock-template/lib/ai-global-agent-ecommerce.ts +5 -0
  310. package/templates/nextblock-template/lib/ai-global-agent-tools-stats.test.ts +223 -0
  311. package/templates/nextblock-template/lib/ai-global-agent-tools.test.ts +2183 -0
  312. package/templates/nextblock-template/lib/ai-global-agent-tools.ts +4807 -0
  313. package/templates/nextblock-template/lib/ai-key-crypto.test.ts +70 -0
  314. package/templates/nextblock-template/lib/ai-key-crypto.ts +132 -0
  315. package/templates/nextblock-template/lib/ai-model-catalog.test.ts +49 -0
  316. package/templates/nextblock-template/lib/ai-model-catalog.ts +41 -0
  317. package/templates/nextblock-template/lib/ai-model-registry.test.ts +231 -0
  318. package/templates/nextblock-template/lib/ai-model-registry.ts +522 -0
  319. package/templates/nextblock-template/lib/auth/cookies.ts +47 -0
  320. package/templates/nextblock-template/lib/auth/crypto.ts +42 -0
  321. package/templates/nextblock-template/lib/auth/trustedDevices.ts +92 -0
  322. package/templates/nextblock-template/lib/auth/twoFactor.ts +167 -0
  323. package/templates/nextblock-template/lib/auth-redirects.ts +46 -0
  324. package/templates/nextblock-template/lib/blocks/FeaturedProductBlock.tsx +94 -0
  325. package/templates/nextblock-template/lib/blocks/ProductGridBlock.tsx +137 -0
  326. package/templates/nextblock-template/lib/blocks/README.md +13 -670
  327. package/templates/nextblock-template/lib/blocks/blockRegistry.ts +138 -56
  328. package/templates/nextblock-template/lib/blocks/blockTypes.ts +18 -0
  329. package/templates/nextblock-template/lib/blocks/ecommerce-block-schemas.ts +31 -0
  330. package/templates/nextblock-template/lib/cms-transfer/csv.test.ts +77 -0
  331. package/templates/nextblock-template/lib/cms-transfer/csv.ts +399 -0
  332. package/templates/nextblock-template/lib/cms-transfer/server.ts +2243 -0
  333. package/templates/nextblock-template/lib/cms-transfer/types.ts +145 -0
  334. package/templates/nextblock-template/lib/cortex-widget-registry.test.ts +199 -0
  335. package/templates/nextblock-template/lib/cortex-widget-registry.ts +88 -0
  336. package/templates/nextblock-template/lib/cortex-widget-schema.test.tsx +237 -0
  337. package/templates/nextblock-template/lib/cortex-widget-schema.ts +393 -0
  338. package/templates/nextblock-template/lib/custom-block-definitions.ts +87 -0
  339. package/templates/nextblock-template/lib/custom-block-r2-upload-shared.ts +178 -0
  340. package/templates/nextblock-template/lib/custom-block-r2-upload.test.ts +140 -0
  341. package/templates/nextblock-template/lib/custom-block-r2-upload.ts +68 -0
  342. package/templates/nextblock-template/lib/custom-block-relation-registry.ts +256 -0
  343. package/templates/nextblock-template/lib/custom-block-relations.test.ts +227 -0
  344. package/templates/nextblock-template/lib/custom-block-relations.ts +279 -0
  345. package/templates/nextblock-template/lib/custom-block-safelist.ts +14 -0
  346. package/templates/nextblock-template/lib/editor/dynamic-extension-core.test.ts +172 -0
  347. package/templates/nextblock-template/lib/editor/dynamic-extension-core.ts +213 -0
  348. package/templates/nextblock-template/lib/editor/dynamic-extension-loader.ts +22 -0
  349. package/templates/nextblock-template/lib/editor/dynamic-extensions.tsx +193 -0
  350. package/templates/nextblock-template/lib/full-backup/manifest.test.ts +121 -0
  351. package/templates/nextblock-template/lib/full-backup/manifest.ts +206 -0
  352. package/templates/nextblock-template/lib/full-backup/server.ts +743 -0
  353. package/templates/nextblock-template/lib/media/resolveMediaUrl.ts +45 -0
  354. package/templates/nextblock-template/lib/posts/readTime.ts +60 -0
  355. package/templates/nextblock-template/lib/privacy/consent-client.ts +57 -0
  356. package/templates/nextblock-template/lib/privacy/settings.ts +103 -0
  357. package/templates/nextblock-template/lib/privacy/types.ts +67 -0
  358. package/templates/nextblock-template/lib/promotions/server.test.ts +74 -0
  359. package/templates/nextblock-template/lib/promotions/server.ts +741 -0
  360. package/templates/nextblock-template/lib/resolve-block-relations.test.ts +142 -0
  361. package/templates/nextblock-template/lib/resolve-block-relations.ts +255 -0
  362. package/templates/nextblock-template/lib/search/server.ts +585 -0
  363. package/templates/nextblock-template/lib/search/types.ts +27 -0
  364. package/templates/nextblock-template/lib/visual-editing/draft-content.test.ts +105 -0
  365. package/templates/nextblock-template/lib/visual-editing/draft-content.ts +380 -0
  366. package/templates/nextblock-template/lib/visual-editing/draft-route.test.ts +42 -0
  367. package/templates/nextblock-template/lib/visual-editing/draft-route.ts +82 -0
  368. package/templates/nextblock-template/lib/visual-editing/edit-info.test.ts +143 -0
  369. package/templates/nextblock-template/lib/visual-editing/edit-info.ts +94 -0
  370. package/templates/nextblock-template/lib/visual-editing/mutations.ts +190 -0
  371. package/templates/nextblock-template/lib/visual-editing/product-drafts.test.ts +81 -0
  372. package/templates/nextblock-template/lib/visual-editing/product-drafts.ts +511 -0
  373. package/templates/nextblock-template/lib/visual-editing/types.ts +122 -0
  374. package/templates/nextblock-template/lib/zod-config.ts +5 -0
  375. package/templates/nextblock-template/next.config.js +190 -66
  376. package/templates/nextblock-template/package.json +34 -30
  377. package/templates/nextblock-template/proxy.ts +435 -253
  378. package/templates/nextblock-template/public/images/NBcover.webp +0 -0
  379. package/templates/nextblock-template/public/images/cap.webp +0 -0
  380. package/templates/nextblock-template/public/images/commerce-plan.webp +0 -0
  381. package/templates/nextblock-template/public/images/commerce-square.webp +0 -0
  382. package/templates/nextblock-template/public/images/commerce-wide.webp +0 -0
  383. package/templates/nextblock-template/public/images/cortex-ai-square.webp +0 -0
  384. package/templates/nextblock-template/public/images/cortex-ai.webp +0 -0
  385. package/templates/nextblock-template/public/images/extensibility.webp +0 -0
  386. package/templates/nextblock-template/public/images/goals.webp +0 -0
  387. package/templates/nextblock-template/public/images/included.webp +0 -0
  388. package/templates/nextblock-template/public/images/nx-graph.webp +0 -0
  389. package/templates/nextblock-template/public/images/pants.webp +0 -0
  390. package/templates/nextblock-template/public/images/t-shirt.webp +0 -0
  391. package/templates/nextblock-template/scripts/validate-editor-block-schema.ts +112 -0
  392. package/templates/nextblock-template/scripts/verify-cortex-ai-build-widget.tsx +100 -0
  393. package/templates/nextblock-template/scripts/verify-cortex-ai-generate-blocks.ts +62 -0
  394. package/templates/nextblock-template/scripts/verify-cortex-ai-global-tools.ts +537 -0
  395. package/templates/nextblock-template/scripts/verify-cortex-ai-routing.ts +58 -0
  396. package/templates/nextblock-template/scripts/verify-custom-block-definitions.ts +188 -0
  397. package/templates/nextblock-template/scripts/verify-dynamic-custom-block-extensions.ts +123 -0
  398. package/templates/nextblock-template/scripts/verify-dynamic-layout-engine.tsx +133 -0
  399. package/templates/nextblock-template/scripts/verify-milestone-2-custom-blocks.ts +65 -0
  400. package/templates/nextblock-template/tailwind.config.js +1 -0
  401. package/templates/nextblock-template/tools/configure-supabase-auth.js +282 -0
  402. package/templates/nextblock-template/tools/deploy-supabase.js +69 -71
  403. package/templates/nextblock-template/tsconfig.json +52 -66
  404. package/templates/nextblock-template/tsconfig.tsbuildinfo +1 -1
  405. package/templates/nextblock-template/types/jsdom.d.ts +6 -0
  406. package/templates/nextblock-template/app/force-styles.tsx +0 -31
  407. package/templates/nextblock-template/app/sitemap.xml/route.ts +0 -63
  408. package/templates/nextblock-template/components/blocks/renderers/HeroBlockRenderer.tsx +0 -273
  409. package/templates/nextblock-template/docs/How to Create a Custom Block.md +0 -149
  410. package/templates/nextblock-template/docs/cms-application-overview.md +0 -56
  411. package/templates/nextblock-template/docs/cms-architecture-overview.md +0 -73
  412. package/templates/nextblock-template/docs/files-structure.md +0 -426
  413. package/templates/nextblock-template/docs/tiptap-bundle-optimization-summary.md +0 -174
@@ -0,0 +1,741 @@
1
+ import "server-only";
2
+
3
+ import { getServiceRoleSupabaseClient } from "@nextblock-cms/db/server";
4
+ import { syncProductSaleCouponToFreemius } from "@nextblock-cms/ecommerce/server";
5
+
6
+ import {
7
+ majorToMinor,
8
+ minorToMajor,
9
+ parseCsv,
10
+ priceMapMajorToMinor,
11
+ priceMapMinorToMajor,
12
+ salePriceMapMajorToMinor,
13
+ salePriceMapMinorToMajor,
14
+ stringifyCsv,
15
+ toJsonCell,
16
+ } from "../cms-transfer/csv";
17
+
18
+ export type PromotionKind = "sale" | "price_change";
19
+
20
+ export interface PromotionImportMessage {
21
+ row: number;
22
+ type: "error" | "warning";
23
+ message: string;
24
+ }
25
+
26
+ export interface PromotionImportPreviewItem {
27
+ row: number;
28
+ action: "update" | "skip";
29
+ identifier: string;
30
+ matched: number;
31
+ }
32
+
33
+ export interface PromotionImportSummary {
34
+ success: boolean;
35
+ totalRows: number;
36
+ applied: number;
37
+ skipped: number;
38
+ errors: PromotionImportMessage[];
39
+ warnings: PromotionImportMessage[];
40
+ preview: PromotionImportPreviewItem[];
41
+ }
42
+
43
+ export interface PromotionTransferResult {
44
+ success: boolean;
45
+ error?: string;
46
+ fileName?: string;
47
+ mimeType?: string;
48
+ content?: string;
49
+ }
50
+
51
+ // Canonical column keys used internally. The CSV is intentionally minimal:
52
+ // - `sku` is the key, looked up in BOTH the products and product_variants
53
+ // tables (one column covers products and variations).
54
+ // - `id` is optional (a convenience pointer; the SKU is what matters).
55
+ // - the price column accepts a single number ("14.99") OR a multi-currency
56
+ // JSON object ('{"USD":14.99,"EUR":13.5}').
57
+ const SALE_COLUMNS = ["id", "sku", "sale_price", "sale_start_at", "sale_end_at"] as const;
58
+ const PRICE_CHANGE_COLUMNS = ["id", "sku", "new_price", "effective_at"] as const;
59
+ const OPTIONAL_COLUMNS = new Set<string>(["id"]);
60
+
61
+ function toDisplayHeader(column: string) {
62
+ return OPTIONAL_COLUMNS.has(column) ? `${column} (optional)` : column;
63
+ }
64
+
65
+ function getColumns(kind: PromotionKind): readonly string[] {
66
+ return kind === "sale" ? SALE_COLUMNS : PRICE_CHANGE_COLUMNS;
67
+ }
68
+
69
+ function getDisplayHeaders(kind: PromotionKind): string[] {
70
+ return getColumns(kind).map(toDisplayHeader);
71
+ }
72
+
73
+ // Exposed display headers (template + export column order).
74
+ export const SALE_CSV_HEADERS = getDisplayHeaders("sale");
75
+ export const PRICE_CHANGE_CSV_HEADERS = getDisplayHeaders("price_change");
76
+
77
+ type ProductRow = {
78
+ id: string;
79
+ sku: string | null;
80
+ price: number | null;
81
+ payment_provider: string | null;
82
+ };
83
+
84
+ type VariantRow = {
85
+ id: string;
86
+ sku: string | null;
87
+ price: number | null;
88
+ product_id: string;
89
+ };
90
+
91
+ type SaleUpdate = {
92
+ sale_price: number | null;
93
+ sale_prices: Record<string, number | null> | null;
94
+ sale_start_at: string | null;
95
+ sale_end_at: string | null;
96
+ };
97
+
98
+ type PriceChangeUpdate = {
99
+ scheduled_price: number | null;
100
+ scheduled_prices: Record<string, number> | null;
101
+ scheduled_price_at: string | null;
102
+ };
103
+
104
+ type ResolvedOperation = {
105
+ rowNumber: number;
106
+ identifier: string;
107
+ productIds: string[];
108
+ variantIds: string[];
109
+ freemiusProductIds: string[];
110
+ saleUpdate?: SaleUpdate;
111
+ priceChangeUpdate?: PriceChangeUpdate;
112
+ };
113
+
114
+ function addMessage(
115
+ list: PromotionImportMessage[],
116
+ row: number,
117
+ type: "error" | "warning",
118
+ message: string
119
+ ) {
120
+ list.push({ row, type, message });
121
+ }
122
+
123
+ // Map a canonical-keyed object to display headers (e.g. id -> "id (optional)").
124
+ function toDisplayRow(
125
+ kind: PromotionKind,
126
+ canonical: Record<string, string>
127
+ ): Record<string, string> {
128
+ const out: Record<string, string> = {};
129
+ for (const column of getColumns(kind)) {
130
+ out[toDisplayHeader(column)] = canonical[column] ?? "";
131
+ }
132
+ return out;
133
+ }
134
+
135
+ // Read a parsed CSV row back to canonical keys: strip an optional "(optional)"
136
+ // hint from headers and lowercase, so the importer is forgiving of header
137
+ // casing and the optional annotation.
138
+ function normalizeRowKeys(row: Record<string, string>): Record<string, string> {
139
+ const normalized: Record<string, string> = {};
140
+ for (const [key, value] of Object.entries(row)) {
141
+ const canonical = key
142
+ .replace(/\s*\(optional\)\s*$/i, "")
143
+ .trim()
144
+ .toLowerCase();
145
+ normalized[canonical] = value;
146
+ }
147
+ return normalized;
148
+ }
149
+
150
+ const DATE_ONLY = /^\d{4}-\d{2}-\d{2}$/;
151
+
152
+ // Normalize a CSV date cell to an ISO-8601 UTC string.
153
+ // - date-only "YYYY-MM-DD": inclusive — start-of-day for starts, end-of-day
154
+ // (23:59:59) for ends.
155
+ // - with a time: starts keep :00 seconds; ends are forced to :59 so the
156
+ // selected minute is inclusive.
157
+ // Bare values are interpreted in the server's local time (matching the form's
158
+ // datetime-local pickers).
159
+ export function normalizeDateInput(
160
+ value: string | undefined,
161
+ position: "start" | "end"
162
+ ): { value: string | null; valid: boolean } {
163
+ const trimmed = (value || "").trim();
164
+ if (!trimmed) {
165
+ return { value: null, valid: true };
166
+ }
167
+
168
+ let date: Date;
169
+ if (DATE_ONLY.test(trimmed)) {
170
+ const [year, month, day] = trimmed.split("-").map(Number);
171
+ date =
172
+ position === "end"
173
+ ? new Date(year, month - 1, day, 23, 59, 59, 0)
174
+ : new Date(year, month - 1, day, 0, 0, 0, 0);
175
+ } else {
176
+ const parsed = Date.parse(trimmed);
177
+ if (Number.isNaN(parsed)) {
178
+ return { value: null, valid: false };
179
+ }
180
+ date = new Date(parsed);
181
+ date.setSeconds(position === "end" ? 59 : 0, 0);
182
+ }
183
+
184
+ if (Number.isNaN(date.getTime())) {
185
+ return { value: null, valid: false };
186
+ }
187
+ return { value: date.toISOString(), valid: true };
188
+ }
189
+
190
+ // Parse a price cell that may be a single number or a multi-currency JSON map.
191
+ // Amounts are in major units.
192
+ export function parsePriceCell(raw: string | undefined): {
193
+ scalar: number | null;
194
+ map: Record<string, number | null> | null;
195
+ error: string | null;
196
+ } {
197
+ const trimmed = (raw || "").trim();
198
+ if (!trimmed) {
199
+ return { scalar: null, map: null, error: null };
200
+ }
201
+
202
+ if (trimmed.startsWith("{")) {
203
+ let parsed: unknown;
204
+ try {
205
+ parsed = JSON.parse(trimmed);
206
+ } catch {
207
+ return { scalar: null, map: null, error: "is not valid JSON." };
208
+ }
209
+ if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) {
210
+ return { scalar: null, map: null, error: "JSON must be an object of currency codes." };
211
+ }
212
+ const map: Record<string, number | null> = {};
213
+ for (const [code, amount] of Object.entries(parsed as Record<string, unknown>)) {
214
+ if (amount === null || amount === "") {
215
+ map[code.toUpperCase()] = null;
216
+ continue;
217
+ }
218
+ const num = Number(amount);
219
+ if (!Number.isFinite(num) || num < 0) {
220
+ return { scalar: null, map: null, error: "JSON values must be non-negative numbers." };
221
+ }
222
+ map[code.toUpperCase()] = num;
223
+ }
224
+ return { scalar: null, map, error: null };
225
+ }
226
+
227
+ const num = Number(trimmed);
228
+ if (!Number.isFinite(num) || num < 0) {
229
+ return { scalar: null, map: null, error: "must be a non-negative number or a JSON object." };
230
+ }
231
+ return { scalar: num, map: null, error: null };
232
+ }
233
+
234
+ // Example date in the simple `YYYY-MM-DDTHH:mm` shape (no seconds / ms / Z) so
235
+ // the template doesn't confuse users with precision they don't need to type.
236
+ function exampleDateTime(offsetDays = 0) {
237
+ return new Date(Date.now() + offsetDays * 24 * 60 * 60 * 1000)
238
+ .toISOString()
239
+ .slice(0, 16);
240
+ }
241
+
242
+ export function getPromotionsTemplateCsv(kind: PromotionKind) {
243
+ const canonical: Record<string, string> = {};
244
+ for (const column of getColumns(kind)) {
245
+ canonical[column] = "";
246
+ }
247
+ canonical.sku = "EXAMPLE-SKU";
248
+
249
+ if (kind === "sale") {
250
+ // A single number or a JSON map are both accepted in the price column.
251
+ canonical.sale_price = "14.99";
252
+ canonical.sale_start_at = exampleDateTime(0);
253
+ canonical.sale_end_at = exampleDateTime(7);
254
+ } else {
255
+ canonical.new_price = "24.99";
256
+ canonical.effective_at = exampleDateTime(0);
257
+ }
258
+
259
+ return stringifyCsv([toDisplayRow(kind, canonical)], getDisplayHeaders(kind));
260
+ }
261
+
262
+ export async function exportPromotionsCsv(kind: PromotionKind) {
263
+ const client = getServiceRoleSupabaseClient();
264
+ const canonicalRows: Array<Record<string, string>> = [];
265
+
266
+ const salePriceCell = (
267
+ scalarMinor: number | null | undefined,
268
+ map: unknown
269
+ ) => {
270
+ if (map && typeof map === "object" && Object.keys(map as object).length > 0) {
271
+ return toJsonCell(salePriceMapMinorToMajor(map));
272
+ }
273
+ return typeof scalarMinor === "number" ? String(minorToMajor(scalarMinor) ?? "") : "";
274
+ };
275
+ const regularPriceCell = (
276
+ scalarMinor: number | null | undefined,
277
+ map: unknown
278
+ ) => {
279
+ // Prefer the simple scalar; only emit a JSON map for genuinely multi-currency
280
+ // prices (more than one currency configured).
281
+ if (map && typeof map === "object" && Object.keys(map as object).length > 1) {
282
+ return toJsonCell(priceMapMinorToMajor(map));
283
+ }
284
+ return typeof scalarMinor === "number" ? String(minorToMajor(scalarMinor) ?? "") : "";
285
+ };
286
+
287
+ if (kind === "sale") {
288
+ const { data: products } = await (client as any)
289
+ .from("products")
290
+ .select("id, sku, sale_price, sale_prices, sale_start_at, sale_end_at")
291
+ .or("sale_price.not.is.null,sale_start_at.not.is.null,sale_end_at.not.is.null")
292
+ .order("sku", { ascending: true });
293
+ for (const product of products || []) {
294
+ canonicalRows.push({
295
+ id: product.id ?? "",
296
+ sku: product.sku ?? "",
297
+ sale_price: salePriceCell(product.sale_price, product.sale_prices),
298
+ sale_start_at: product.sale_start_at ?? "",
299
+ sale_end_at: product.sale_end_at ?? "",
300
+ });
301
+ }
302
+
303
+ const { data: variants } = await (client as any)
304
+ .from("product_variants")
305
+ .select("sku, sale_price, sale_prices, sale_start_at, sale_end_at")
306
+ .or("sale_price.not.is.null,sale_start_at.not.is.null,sale_end_at.not.is.null")
307
+ .order("sku", { ascending: true });
308
+ for (const variant of variants || []) {
309
+ canonicalRows.push({
310
+ id: "",
311
+ sku: variant.sku ?? "",
312
+ sale_price: salePriceCell(variant.sale_price, variant.sale_prices),
313
+ sale_start_at: variant.sale_start_at ?? "",
314
+ sale_end_at: variant.sale_end_at ?? "",
315
+ });
316
+ }
317
+ } else {
318
+ // Price changes export a worksheet of every physical (Stripe) SKU with its
319
+ // CURRENT regular price, deduped by SKU. The user only edits `new_price` and
320
+ // `effective_at`, then re-imports. (Freemius/digital prices are owned by
321
+ // Freemius, so they're excluded.)
322
+ const { data: products } = await (client as any)
323
+ .from("products")
324
+ .select("sku, price, prices, product_variants(sku, price, prices)")
325
+ .eq("product_type", "physical")
326
+ .order("sku", { ascending: true });
327
+
328
+ const seen = new Set<string>();
329
+ for (const product of products || []) {
330
+ const variants = product.product_variants || [];
331
+ if (variants.length > 0) {
332
+ // Variant products: list each variant SKU + its own price.
333
+ for (const variant of variants) {
334
+ const sku = (variant.sku || "").trim();
335
+ if (!sku || seen.has(sku)) {
336
+ continue;
337
+ }
338
+ seen.add(sku);
339
+ canonicalRows.push({
340
+ id: "",
341
+ sku,
342
+ new_price: regularPriceCell(variant.price, variant.prices),
343
+ effective_at: "",
344
+ });
345
+ }
346
+ } else {
347
+ const sku = (product.sku || "").trim();
348
+ if (!sku || seen.has(sku)) {
349
+ continue;
350
+ }
351
+ seen.add(sku);
352
+ canonicalRows.push({
353
+ id: "",
354
+ sku,
355
+ new_price: regularPriceCell(product.price, product.prices),
356
+ effective_at: "",
357
+ });
358
+ }
359
+ }
360
+
361
+ canonicalRows.sort((left, right) => (left.sku || "").localeCompare(right.sku || ""));
362
+ }
363
+
364
+ const source =
365
+ canonicalRows.length > 0
366
+ ? canonicalRows
367
+ : [Object.fromEntries(getColumns(kind).map((column) => [column, ""]))];
368
+ return stringifyCsv(
369
+ source.map((row) => toDisplayRow(kind, row)),
370
+ getDisplayHeaders(kind)
371
+ );
372
+ }
373
+
374
+ async function buildPlan(kind: PromotionKind, csvContent: string) {
375
+ const client = getServiceRoleSupabaseClient();
376
+ const parsed = parseCsv(csvContent);
377
+ const rows = parsed.rows.map(normalizeRowKeys);
378
+ const errors: PromotionImportMessage[] = parsed.errors.map((error) => ({
379
+ row: error.row,
380
+ type: "error" as const,
381
+ message: error.message,
382
+ }));
383
+ const warnings: PromotionImportMessage[] = [];
384
+ const preview: PromotionImportPreviewItem[] = [];
385
+ const operations: ResolvedOperation[] = [];
386
+
387
+ // Default currency code, used to derive the scalar (default-currency) price
388
+ // when a JSON price map is supplied.
389
+ let defaultCurrencyCode = "USD";
390
+ {
391
+ const { data } = await (client as any)
392
+ .from("currencies")
393
+ .select("code")
394
+ .eq("is_default", true)
395
+ .maybeSingle();
396
+ if (data?.code) {
397
+ defaultCurrencyCode = String(data.code).toUpperCase();
398
+ }
399
+ }
400
+
401
+ // Collect SKUs (the key) and ids. An id resolves to its SKU so a single SKU
402
+ // lookup covers it too.
403
+ const idSet = new Set<string>();
404
+ const skuSet = new Set<string>();
405
+ rows.forEach((row) => {
406
+ const id = (row.id || "").trim();
407
+ const sku = (row.sku || "").trim();
408
+ if (sku) {
409
+ skuSet.add(sku);
410
+ } else if (id) {
411
+ idSet.add(id);
412
+ }
413
+ });
414
+
415
+ const productsById = new Map<string, ProductRow>();
416
+ if (idSet.size > 0) {
417
+ const { data } = await (client as any)
418
+ .from("products")
419
+ .select("id, sku, price, payment_provider")
420
+ .in("id", [...idSet]);
421
+ for (const product of (data || []) as ProductRow[]) {
422
+ productsById.set(product.id, product);
423
+ const resolvedSku = (product.sku || "").trim();
424
+ if (resolvedSku) {
425
+ skuSet.add(resolvedSku);
426
+ }
427
+ }
428
+ }
429
+
430
+ const productsBySku = new Map<string, ProductRow[]>();
431
+ const variantsBySku = new Map<string, VariantRow[]>();
432
+ if (skuSet.size > 0) {
433
+ const skus = [...skuSet];
434
+ const [{ data: productData }, { data: variantData }] = await Promise.all([
435
+ (client as any)
436
+ .from("products")
437
+ .select("id, sku, price, payment_provider")
438
+ .in("sku", skus),
439
+ (client as any)
440
+ .from("product_variants")
441
+ .select("id, sku, price, product_id")
442
+ .in("sku", skus),
443
+ ]);
444
+ for (const product of (productData || []) as ProductRow[]) {
445
+ const key = (product.sku || "").trim();
446
+ const list = productsBySku.get(key) ?? [];
447
+ list.push(product);
448
+ productsBySku.set(key, list);
449
+ }
450
+ for (const variant of (variantData || []) as VariantRow[]) {
451
+ const key = (variant.sku || "").trim();
452
+ const list = variantsBySku.get(key) ?? [];
453
+ list.push(variant);
454
+ variantsBySku.set(key, list);
455
+ }
456
+ }
457
+
458
+ // Resolve parent payment_provider for matched variants (Freemius handling).
459
+ const variantParentIds = new Set<string>();
460
+ variantsBySku.forEach((list) => list.forEach((variant) => variantParentIds.add(variant.product_id)));
461
+ const variantParentProviders = new Map<string, string | null>();
462
+ if (variantParentIds.size > 0) {
463
+ const { data } = await (client as any)
464
+ .from("products")
465
+ .select("id, payment_provider")
466
+ .in("id", [...variantParentIds]);
467
+ for (const product of data || []) {
468
+ variantParentProviders.set(product.id, product.payment_provider ?? null);
469
+ }
470
+ }
471
+
472
+ rows.forEach((row, index) => {
473
+ const rowNumber = index + 2; // header is row 1
474
+ const id = (row.id || "").trim();
475
+ let sku = (row.sku || "").trim();
476
+ let identifier: string;
477
+
478
+ if (!sku && id) {
479
+ const product = productsById.get(id);
480
+ if (!product) {
481
+ addMessage(errors, rowNumber, "error", `No product found with id "${id}".`);
482
+ preview.push({ row: rowNumber, action: "skip", identifier: `id ${id}`, matched: 0 });
483
+ return;
484
+ }
485
+ sku = (product.sku || "").trim();
486
+ identifier = `id ${id} (sku ${sku})`;
487
+ } else if (sku) {
488
+ identifier = `sku ${sku}`;
489
+ } else {
490
+ addMessage(errors, rowNumber, "error", "Provide a SKU (or id) to target.");
491
+ preview.push({ row: rowNumber, action: "skip", identifier: "(none)", matched: 0 });
492
+ return;
493
+ }
494
+
495
+ // One SKU resolves against BOTH tables.
496
+ const productMatches = productsBySku.get(sku) ?? [];
497
+ const variantMatches = variantsBySku.get(sku) ?? [];
498
+ if (productMatches.length === 0 && variantMatches.length === 0) {
499
+ addMessage(errors, rowNumber, "error", `No product or variant found with SKU "${sku}".`);
500
+ preview.push({ row: rowNumber, action: "skip", identifier, matched: 0 });
501
+ return;
502
+ }
503
+
504
+ const productIds = productMatches.map((product) => product.id);
505
+ const variantIds = variantMatches.map((variant) => variant.id);
506
+ const freemiusProductIds = new Set<string>();
507
+ productMatches.forEach((product) => {
508
+ if (product.payment_provider === "freemius") {
509
+ freemiusProductIds.add(product.id);
510
+ }
511
+ });
512
+ variantMatches.forEach((variant) => {
513
+ if (variantParentProviders.get(variant.product_id) === "freemius") {
514
+ freemiusProductIds.add(variant.product_id);
515
+ }
516
+ });
517
+
518
+ const matchedCount = productIds.length + variantIds.length;
519
+
520
+ if (kind === "sale") {
521
+ const start = normalizeDateInput(row.sale_start_at, "start");
522
+ const end = normalizeDateInput(row.sale_end_at, "end");
523
+ if (!start.valid) {
524
+ addMessage(errors, rowNumber, "error", "sale_start_at is not a valid date.");
525
+ }
526
+ if (!end.valid) {
527
+ addMessage(errors, rowNumber, "error", "sale_end_at is not a valid date.");
528
+ }
529
+ if (start.value && end.value && Date.parse(start.value) >= Date.parse(end.value)) {
530
+ addMessage(errors, rowNumber, "error", "sale_end_at must be after sale_start_at.");
531
+ }
532
+
533
+ const price = parsePriceCell(row.sale_price);
534
+ if (price.error) {
535
+ addMessage(errors, rowNumber, "error", `sale_price ${price.error}`);
536
+ }
537
+
538
+ let saleScalarMinor: number | null = null;
539
+ let salePricesMinor: Record<string, number | null> | null = null;
540
+ if (price.map) {
541
+ salePricesMinor = salePriceMapMajorToMinor(price.map);
542
+ const def = price.map[defaultCurrencyCode];
543
+ saleScalarMinor = typeof def === "number" ? majorToMinor(def) : null;
544
+ } else if (price.scalar !== null) {
545
+ saleScalarMinor = majorToMinor(price.scalar);
546
+ }
547
+
548
+ if (saleScalarMinor !== null) {
549
+ const threshold = saleScalarMinor;
550
+ const exceeds = [...productMatches, ...variantMatches].some(
551
+ (record) => typeof record.price === "number" && threshold > record.price
552
+ );
553
+ if (exceeds) {
554
+ addMessage(
555
+ warnings,
556
+ rowNumber,
557
+ "warning",
558
+ "sale price exceeds the regular price for at least one match; that sale is ignored at checkout until corrected."
559
+ );
560
+ }
561
+ }
562
+
563
+ const saleConfigured =
564
+ saleScalarMinor !== null ||
565
+ (salePricesMinor !== null && Object.keys(salePricesMinor).length > 0);
566
+
567
+ const saleUpdate: SaleUpdate = saleConfigured
568
+ ? {
569
+ sale_price: saleScalarMinor,
570
+ sale_prices: salePricesMinor,
571
+ sale_start_at: start.value,
572
+ sale_end_at: end.value,
573
+ }
574
+ : {
575
+ // Empty price clears the scheduled sale entirely.
576
+ sale_price: null,
577
+ sale_prices: null,
578
+ sale_start_at: null,
579
+ sale_end_at: null,
580
+ };
581
+
582
+ operations.push({
583
+ rowNumber,
584
+ identifier,
585
+ productIds,
586
+ variantIds,
587
+ freemiusProductIds: [...freemiusProductIds],
588
+ saleUpdate,
589
+ });
590
+ } else {
591
+ const effective = normalizeDateInput(row.effective_at, "start");
592
+ if (!effective.valid) {
593
+ addMessage(errors, rowNumber, "error", "effective_at is not a valid date.");
594
+ }
595
+ if (!effective.value) {
596
+ addMessage(errors, rowNumber, "error", "effective_at is required for a price change.");
597
+ }
598
+
599
+ const price = parsePriceCell(row.new_price);
600
+ if (price.error) {
601
+ addMessage(errors, rowNumber, "error", `new_price ${price.error}`);
602
+ }
603
+ if (price.scalar === null && !price.map) {
604
+ addMessage(errors, rowNumber, "error", "new_price is required.");
605
+ }
606
+
607
+ let scheduledScalarMinor: number | null = null;
608
+ let scheduledPricesMinor: Record<string, number> | null = null;
609
+ if (price.map) {
610
+ scheduledPricesMinor = priceMapMajorToMinor(price.map);
611
+ const def = price.map[defaultCurrencyCode];
612
+ scheduledScalarMinor = typeof def === "number" ? majorToMinor(def) : null;
613
+ } else if (price.scalar !== null) {
614
+ scheduledScalarMinor = majorToMinor(price.scalar);
615
+ }
616
+
617
+ // Freemius products own their regular price; skip them with a warning.
618
+ const applicableProductIds = productIds.filter((pid) => !freemiusProductIds.has(pid));
619
+ if (applicableProductIds.length < productIds.length) {
620
+ addMessage(
621
+ warnings,
622
+ rowNumber,
623
+ "warning",
624
+ "Scheduled price changes do not apply to Freemius products; those are skipped."
625
+ );
626
+ }
627
+
628
+ const priceChangeUpdate: PriceChangeUpdate = {
629
+ scheduled_price: scheduledScalarMinor,
630
+ scheduled_prices: scheduledPricesMinor,
631
+ scheduled_price_at: effective.value,
632
+ };
633
+
634
+ operations.push({
635
+ rowNumber,
636
+ identifier,
637
+ productIds: applicableProductIds,
638
+ variantIds,
639
+ freemiusProductIds: [...freemiusProductIds],
640
+ priceChangeUpdate,
641
+ });
642
+ }
643
+
644
+ preview.push({ row: rowNumber, action: "update", identifier, matched: matchedCount });
645
+ });
646
+
647
+ return { client, errors, warnings, preview, operations, totalRows: rows.length };
648
+ }
649
+
650
+ function summarize(
651
+ plan: Awaited<ReturnType<typeof buildPlan>>,
652
+ applied: number,
653
+ skipped: number
654
+ ): PromotionImportSummary {
655
+ return {
656
+ success: plan.errors.length === 0,
657
+ totalRows: plan.totalRows,
658
+ applied,
659
+ skipped,
660
+ errors: plan.errors,
661
+ warnings: plan.warnings,
662
+ preview: plan.preview,
663
+ };
664
+ }
665
+
666
+ export async function dryRunPromotionsImport(
667
+ kind: PromotionKind,
668
+ csvContent: string
669
+ ): Promise<PromotionImportSummary> {
670
+ const plan = await buildPlan(kind, csvContent);
671
+ const skipped = plan.preview.filter((item) => item.action === "skip").length;
672
+ return summarize(plan, 0, skipped);
673
+ }
674
+
675
+ export async function applyPromotionsImport(
676
+ kind: PromotionKind,
677
+ csvContent: string
678
+ ): Promise<PromotionImportSummary> {
679
+ const plan = await buildPlan(kind, csvContent);
680
+
681
+ if (plan.errors.length > 0) {
682
+ const skipped = plan.preview.filter((item) => item.action === "skip").length;
683
+ return summarize(plan, 0, skipped);
684
+ }
685
+
686
+ const client = plan.client;
687
+ let applied = 0;
688
+ let skipped = plan.preview.filter((item) => item.action === "skip").length;
689
+ const freemiusProductIdsToSync = new Set<string>();
690
+
691
+ for (const operation of plan.operations) {
692
+ const update = kind === "sale" ? operation.saleUpdate : operation.priceChangeUpdate;
693
+ if (!update) {
694
+ continue;
695
+ }
696
+ try {
697
+ const now = new Date().toISOString();
698
+ let touched = false;
699
+ if (operation.productIds.length > 0) {
700
+ await (client as any)
701
+ .from("products")
702
+ .update({ ...update, updated_at: now })
703
+ .in("id", operation.productIds);
704
+ touched = true;
705
+ }
706
+ if (operation.variantIds.length > 0) {
707
+ await (client as any)
708
+ .from("product_variants")
709
+ .update({ ...update, updated_at: now })
710
+ .in("id", operation.variantIds);
711
+ touched = true;
712
+ }
713
+ if (touched) {
714
+ applied += 1;
715
+ if (kind === "sale") {
716
+ operation.freemiusProductIds.forEach((pid) => freemiusProductIdsToSync.add(pid));
717
+ }
718
+ } else {
719
+ skipped += 1;
720
+ }
721
+ } catch (error) {
722
+ addMessage(
723
+ plan.errors,
724
+ operation.rowNumber,
725
+ "error",
726
+ error instanceof Error ? error.message : "Failed to apply row."
727
+ );
728
+ }
729
+ }
730
+
731
+ // Reconcile Freemius sale coupons for affected Freemius products.
732
+ for (const productId of freemiusProductIdsToSync) {
733
+ try {
734
+ await syncProductSaleCouponToFreemius({ productId, client: client as any });
735
+ } catch (error) {
736
+ console.error("Failed to sync Freemius sale coupon during promotions import:", error);
737
+ }
738
+ }
739
+
740
+ return summarize(plan, applied, skipped);
741
+ }