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
@@ -1,195 +1,264 @@
1
- "use client";
1
+ "use client";
2
2
 
3
- import React, { useCallback, useEffect, useMemo, useState } from "react";
3
+ import React, { useCallback, useEffect, useMemo, useRef, useState } from "react";
4
4
  import type { Database } from "@nextblock-cms/db";
5
- import { createClient as createBrowserClient } from "@nextblock-cms/db";
6
5
  import {
7
6
  Dialog,
8
7
  DialogContent,
9
- DialogHeader,
10
- DialogTitle,
11
- DialogTrigger,
12
- DialogFooter,
13
- DialogClose,
14
- Button,
15
- Input,
16
- Separator,
8
+ DialogHeader,
9
+ DialogTitle,
10
+ DialogTrigger,
11
+ DialogFooter,
12
+ DialogClose,
13
+ Button,
14
+ Input,
15
+ Separator,
17
16
  } from "@nextblock-cms/ui";
18
17
  import { Search, CheckCircle } from "lucide-react";
19
18
  import Image from "next/image";
20
19
  import MediaUploadForm from "./MediaUploadForm";
20
+ import { resolveMediaUrl } from "../../../../lib/media/resolveMediaUrl";
21
21
 
22
22
  type Media = Database["public"]["Tables"]["media"]["Row"];
23
23
 
24
- const R2_BASE_URL = process.env.NEXT_PUBLIC_R2_BASE_URL || "";
24
+ const MEDIA_REQUEST_TIMEOUT_MS = 8000;
25
+ const MEDIA_LIBRARY_LIMIT = 50;
25
26
 
26
- interface MediaPickerDialogProps {
27
- triggerLabel?: string;
28
- triggerVariant?: "default" | "outline" | "secondary" | "destructive" | "ghost";
29
- onSelect: (media: Media) => void;
30
- accept?: (m: Media) => boolean; // filter, e.g. only images
31
- title?: string;
32
- open?: boolean; onOpenChange?: (open: boolean) => void; hideTrigger?: boolean;
33
- defaultFolder?: string; // optional folder to pre-populate upload
27
+ function resolveMediaPreviewPath(media: Media) {
28
+ return media.file_path || media.object_key || null;
34
29
  }
35
30
 
36
- export default function MediaPickerDialog({
37
- triggerLabel = "Select from Library",
38
- triggerVariant = "outline",
39
- onSelect,
40
- accept,
41
- title = "Select or Upload Media",
42
- open,
43
- onOpenChange,
44
- hideTrigger,
45
- defaultFolder,
46
- }: MediaPickerDialogProps) {
47
- const [internalOpen, setInternalOpen] = useState(false);
48
- const isControlled = typeof open === "boolean";
49
- const isOpen = isControlled ? (open as boolean) : internalOpen;
50
- const setIsOpen = (v: boolean) => {
51
- if (!isControlled) setInternalOpen(v);
52
- onOpenChange?.(v);
53
- };
31
+ function resolveMediaPreviewSrc(path: string) {
32
+ return resolveMediaUrl(path);
33
+ }
34
+
35
+ interface MediaPickerDialogProps {
36
+ triggerLabel?: string;
37
+ triggerVariant?: "default" | "outline" | "secondary" | "destructive" | "ghost";
38
+ onSelect?: (media: Media) => void;
39
+ accept?: (m: Media) => boolean; // filter, e.g. only images
40
+ title?: string;
41
+ open?: boolean; onOpenChange?: (open: boolean) => void; hideTrigger?: boolean;
42
+ defaultFolder?: string; // optional folder to pre-populate upload
43
+ children?: React.ReactNode;
44
+ }
45
+
46
+ export default function MediaPickerDialog({
47
+ triggerLabel = "Select from Library",
48
+ triggerVariant = "outline",
49
+ onSelect,
50
+ accept,
51
+ title = "Select or Upload Media",
52
+ open,
53
+ onOpenChange,
54
+ hideTrigger,
55
+ defaultFolder,
56
+ children,
57
+ }: MediaPickerDialogProps) {
58
+ const [internalOpen, setInternalOpen] = useState(false);
59
+ const isControlled = typeof open === "boolean";
60
+ const isOpen = isControlled ? (open as boolean) : internalOpen;
61
+ const setIsOpen = (v: boolean) => {
62
+ if (!isControlled) setInternalOpen(v);
63
+ onOpenChange?.(v);
64
+ };
54
65
  const [searchTerm, setSearchTerm] = useState("");
55
66
  const [isLoading, setIsLoading] = useState(false);
67
+ const [loadError, setLoadError] = useState<string | null>(null);
56
68
  const [items, setItems] = useState<Media[]>([]);
57
- const supabase = useMemo(() => createBrowserClient(), []);
69
+ const requestIdRef = useRef(0);
58
70
 
59
71
  const fetchLibrary = useCallback(async () => {
72
+ const requestId = ++requestIdRef.current;
60
73
  setIsLoading(true);
74
+ setLoadError(null);
75
+ let timeoutId: ReturnType<typeof setTimeout> | null = null;
76
+
61
77
  try {
62
- let query = supabase
63
- .from("media")
64
- .select("*")
65
- .order("created_at", { ascending: false })
66
- .limit(50);
67
- if (searchTerm) {
68
- query = query.ilike("file_name", `%${searchTerm}%`);
78
+ const controller = new AbortController();
79
+ timeoutId = setTimeout(
80
+ () => controller.abort(new Error("Media library request timed out.")),
81
+ MEDIA_REQUEST_TIMEOUT_MS
82
+ );
83
+
84
+ const params = new URLSearchParams({
85
+ limit: MEDIA_LIBRARY_LIMIT.toString(),
86
+ });
87
+
88
+ if (searchTerm.trim()) {
89
+ params.set("q", searchTerm.trim());
69
90
  }
70
- const { data, error } = await query;
71
- if (error) {
72
- console.error("Error fetching media library:", error);
73
- setItems([]);
91
+
92
+ const response = await fetch(`/api/media/library?${params.toString()}`, {
93
+ method: "GET",
94
+ cache: "no-store",
95
+ signal: controller.signal,
96
+ });
97
+ clearTimeout(timeoutId);
98
+
99
+ const payload = (await response.json()) as {
100
+ items?: Media[];
101
+ error?: string;
102
+ };
103
+
104
+ if (requestId !== requestIdRef.current) {
105
+ return;
106
+ }
107
+
108
+ if (!response.ok) {
109
+ console.error("Error fetching media library:", payload.error);
110
+ setLoadError(payload.error || "We couldn't load the media library. Please retry.");
74
111
  } else {
75
- setItems(data || []);
112
+ setItems(payload.items || []);
113
+ }
114
+ } catch (error: any) {
115
+ if (requestId !== requestIdRef.current) {
116
+ return;
117
+ }
118
+
119
+ console.error("Error fetching media library:", error);
120
+
121
+ if (error?.name === "AbortError" || error?.message === "Media library request timed out.") {
122
+ setLoadError("The media library took too long to respond. Please retry.");
123
+ } else {
124
+ setLoadError("We couldn't load the media library. Please retry.");
76
125
  }
77
126
  } finally {
78
- setIsLoading(false);
127
+ if (timeoutId) {
128
+ clearTimeout(timeoutId);
129
+ }
130
+ if (requestId === requestIdRef.current) {
131
+ setIsLoading(false);
132
+ }
79
133
  }
80
- }, [supabase, searchTerm]);
81
-
82
- useEffect(() => {
83
- if (isOpen) fetchLibrary();
84
- }, [isOpen, fetchLibrary]);
85
-
86
- const filtered = useMemo(() => {
87
- return accept ? items.filter(accept) : items;
88
- }, [items, accept]);
89
-
90
- const handleSelect = (media: Media) => {
91
- onSelect(media);
92
- setIsOpen(false);
93
- };
94
-
95
- return (
96
- <Dialog open={isOpen} onOpenChange={setIsOpen}>
97
- {!hideTrigger && (
98
- <DialogTrigger asChild>
99
- <Button type="button" variant={triggerVariant} size="sm">
100
- {triggerLabel}
101
- </Button>
102
- </DialogTrigger>
103
- )}
104
- <DialogContent className="sm:max-w-[650px] md:max-w-[800px] lg:max-w-[1000px] max-h-[90vh] flex flex-col">
105
- <DialogHeader>
106
- <DialogTitle>{title}</DialogTitle>
107
- </DialogHeader>
108
-
109
- <div className="p-1">
110
- <MediaUploadForm
111
- returnJustData={true}
112
- defaultFolder={defaultFolder}
113
- onUploadSuccess={(newMedia) => {
114
- setItems((prev) => [newMedia, ...prev.filter((m) => m.id !== newMedia.id)]);
115
- handleSelect(newMedia);
116
- }}
117
- />
118
- </div>
119
-
120
- <Separator className="my-4" />
121
-
122
- <div className="flex flex-col flex-grow overflow-hidden">
123
- <h3 className="text-lg font-medium mb-3 text-center">Or Select from Library</h3>
124
- <div className="relative mb-2">
125
- <Input
126
- type="search"
127
- placeholder="Search library..."
128
- value={searchTerm}
129
- onChange={(e) => setSearchTerm(e.target.value)}
130
- className="pl-10"
131
- />
132
- <Search className="absolute left-3 top-1/2 -translate-y-1/2 h-4 w-4 text-muted-foreground" />
133
- </div>
134
- {isLoading ? (
134
+ }, [searchTerm]);
135
+
136
+ useEffect(() => {
137
+ if (isOpen) fetchLibrary();
138
+ }, [isOpen, fetchLibrary]);
139
+
140
+ const filtered = useMemo(() => {
141
+ return accept ? items.filter(accept) : items;
142
+ }, [items, accept]);
143
+
144
+ const handleSelect = (media: Media) => {
145
+ if (onSelect) {
146
+ onSelect(media);
147
+ }
148
+ setIsOpen(false);
149
+ };
150
+
151
+ return (
152
+ <Dialog open={isOpen} onOpenChange={setIsOpen}>
153
+ {!hideTrigger && (
154
+ <DialogTrigger asChild>
155
+ {children || (
156
+ <Button type="button" variant={triggerVariant} size="sm">
157
+ {triggerLabel}
158
+ </Button>
159
+ )}
160
+ </DialogTrigger>
161
+ )}
162
+ <DialogContent className="sm:max-w-[650px] md:max-w-[800px] lg:max-w-[1000px] max-h-[90vh] flex flex-col">
163
+ <DialogHeader>
164
+ <DialogTitle>{title}</DialogTitle>
165
+ </DialogHeader>
166
+
167
+ <div className="p-1">
168
+ <MediaUploadForm
169
+ returnJustData={true}
170
+ defaultFolder={defaultFolder}
171
+ onUploadSuccess={(newMedia) => {
172
+ setItems((prev) => [newMedia, ...prev.filter((m) => m.id !== newMedia.id)]);
173
+ handleSelect(newMedia);
174
+ }}
175
+ />
176
+ </div>
177
+
178
+ <Separator className="my-4" />
179
+
180
+ <div className="flex flex-col flex-grow overflow-hidden">
181
+ <h3 className="text-lg font-medium mb-3 text-center">Or Select from Library</h3>
182
+ <div className="relative mb-2">
183
+ <Input
184
+ type="search"
185
+ placeholder="Search library..."
186
+ value={searchTerm}
187
+ onChange={(e) => setSearchTerm(e.target.value)}
188
+ className="pl-10"
189
+ />
190
+ <Search className="absolute left-3 top-1/2 -translate-y-1/2 h-4 w-4 text-muted-foreground" />
191
+ </div>
192
+ {isLoading && filtered.length === 0 ? (
135
193
  <div className="flex-grow flex items-center justify-center">
136
194
  <p>Loading media...</p>
137
195
  </div>
196
+ ) : loadError && filtered.length === 0 ? (
197
+ <div className="flex-grow flex flex-col items-center justify-center gap-3 text-center">
198
+ <p className="max-w-sm text-sm text-muted-foreground">{loadError}</p>
199
+ <Button type="button" variant="outline" size="sm" onClick={() => void fetchLibrary()}>
200
+ Retry
201
+ </Button>
202
+ </div>
138
203
  ) : filtered.length === 0 ? (
139
204
  <div className="flex-grow flex items-center justify-center">
140
205
  <p>No media found.</p>
141
- </div>
142
- ) : (
206
+ </div>
207
+ ) : (
143
208
  <div className="flex flex-wrap gap-3 overflow-y-auto min-h-0 pr-2 pb-2">
144
- {filtered.map((media) => (
145
- <button
146
- key={media.id}
147
- type="button"
148
- className="relative aspect-square border rounded-md overflow-hidden group focus:outline-none focus:ring-2 focus:ring-primary min-w-0 w-1/3 sm:w-1/4 md:w-1/5 lg:w-1/6"
149
- onClick={() => handleSelect(media)}
150
- >
151
- {media.file_type?.startsWith("image/") && typeof media.width === "number" && typeof media.height === "number" && media.width > 0 && media.height > 0 ? (
152
- <>
153
- <Image
154
- src={`${R2_BASE_URL}/${media.object_key}`}
155
- alt={media.description || media.file_name || "Media library image"}
156
- width={media.width}
157
- height={media.height}
158
- className="absolute inset-0 w-full h-full object-cover"
159
- placeholder={media.blur_data_url ? "blur" : "empty"}
160
- blurDataURL={media.blur_data_url || undefined}
161
- sizes="(max-width: 639px) 33vw, (max-width: 767px) 25vw, (max-width: 1023px) 20vw, 17vw"
162
- />
163
- <div className="absolute inset-0 bg-black/40 opacity-0 group-hover:opacity-100 group-focus:opacity-100 transition-opacity flex items-center justify-center">
164
- <CheckCircle className="h-8 w-8 text-white" />
209
+ {filtered.map((media: Media) => {
210
+ const previewPath = resolveMediaPreviewPath(media);
211
+ const previewSrc = previewPath ? resolveMediaPreviewSrc(previewPath) : null;
212
+
213
+ return (
214
+ <button
215
+ key={media.id}
216
+ type="button"
217
+ className="relative aspect-square border rounded-md overflow-hidden group focus:outline-none focus:ring-2 focus:ring-primary min-w-0 w-1/3 sm:w-1/4 md:w-1/5 lg:w-1/6"
218
+ onClick={() => handleSelect(media)}
219
+ >
220
+ {media.file_type?.startsWith("image/") && previewSrc ? (
221
+ <>
222
+ <Image
223
+ src={previewSrc}
224
+ alt={media.description || media.file_name || "Media library image"}
225
+ fill
226
+ className="absolute inset-0 w-full h-full object-cover"
227
+ placeholder={media.blur_data_url ? "blur" : "empty"}
228
+ blurDataURL={media.blur_data_url || undefined}
229
+ sizes="(max-width: 639px) 33vw, (max-width: 767px) 25vw, (max-width: 1023px) 20vw, 17vw"
230
+ />
231
+ <div className="absolute inset-0 bg-black/40 opacity-0 group-hover:opacity-100 group-focus:opacity-100 transition-opacity flex items-center justify-center">
232
+ <CheckCircle className="h-8 w-8 text-white" />
233
+ </div>
234
+ </>
235
+ ) : (
236
+ <div className="absolute inset-0 flex items-center justify-center text-xs text-muted-foreground p-1 text-center">
237
+ Preview unavailable
165
238
  </div>
166
- </>
167
- ) : (
168
- <div className="absolute inset-0 flex items-center justify-center text-xs text-muted-foreground p-1 text-center">
169
- Preview unavailable
170
- </div>
171
- )}
172
- <p className="absolute bottom-0 left-0 right-0 bg-black/50 text-white text-xs p-1 truncate text-center">
173
- {media.file_name}
174
- </p>
175
- </button>
176
- ))}
239
+ )}
240
+ <p className="absolute bottom-0 left-0 right-0 bg-black/50 text-white text-xs p-1 truncate text-center">
241
+ {media.file_name}
242
+ </p>
243
+ </button>
244
+ );
245
+ })}
177
246
  </div>
178
247
  )}
179
- </div>
180
- <DialogFooter className="mt-auto pt-4">
181
- <DialogClose asChild>
182
- <Button type="button" variant="outline">
183
- Close
184
- </Button>
185
- </DialogClose>
186
- </DialogFooter>
187
- </DialogContent>
188
- </Dialog>
189
- );
190
- }
191
-
192
-
193
-
194
-
195
-
248
+ </div>
249
+ <DialogFooter className="mt-auto pt-4">
250
+ <DialogClose asChild>
251
+ <Button type="button" variant="outline">
252
+ Close
253
+ </Button>
254
+ </DialogClose>
255
+ </DialogFooter>
256
+ </DialogContent>
257
+ </Dialog>
258
+ );
259
+ }
260
+
261
+
262
+
263
+
264
+
@@ -1,18 +1,18 @@
1
1
  // app/cms/media/components/MediaUploadForm.tsx
2
- "use client";
3
-
4
- import React, { useState, useRef, useTransition, useEffect } from "react";
5
- import Image from "next/image";
6
- import { Button } from "@nextblock-cms/ui";
7
- import { Spinner, Alert, AlertDescription } from "@nextblock-cms/ui";
8
- import { Input } from "@nextblock-cms/ui";
9
- import { Label } from "@nextblock-cms/ui";
10
- import { Progress } from "@nextblock-cms/ui"; // Assuming you have this shadcn/ui component
11
- import { UploadCloud, XCircle, CheckCircle2 } from "lucide-react";
12
- import { recordMediaUpload } from "../actions"; // Server action
13
- import type { Database } from "@nextblock-cms/db"; // Import Media type
14
-
15
- type Media = Database['public']['Tables']['media']['Row'];
2
+ "use client";
3
+
4
+ import React, { useState, useRef, useTransition, useEffect } from "react";
5
+ import Image from "next/image";
6
+ import { useRouter } from "next/navigation";
7
+ import { Button } from "@nextblock-cms/ui";
8
+ import { Spinner, Alert, AlertDescription } from "@nextblock-cms/ui";
9
+ import { Input } from "@nextblock-cms/ui";
10
+ import { Label } from "@nextblock-cms/ui";
11
+ import { Progress } from "@nextblock-cms/ui"; // Assuming you have this shadcn/ui component
12
+ import { UploadCloud, XCircle, CheckCircle2 } from "lucide-react";
13
+ import type { Database } from "@nextblock-cms/db"; // Import Media type
14
+
15
+ type Media = Database['public']['Tables']['media']['Row'];
16
16
 
17
17
 
18
18
  interface MediaUploadFormProps {
@@ -25,9 +25,10 @@ interface MediaUploadFormProps {
25
25
 
26
26
  import { useUploadFolder } from "../UploadFolderContext";
27
27
 
28
- export default function MediaUploadForm({ onUploadSuccess, returnJustData, defaultFolder }: MediaUploadFormProps) {
29
- const [isPending, startTransition] = useTransition();
30
- const [file, setFile] = useState<File | null>(null);
28
+ export default function MediaUploadForm({ onUploadSuccess, returnJustData, defaultFolder }: MediaUploadFormProps) {
29
+ const router = useRouter();
30
+ const [isPending, startTransition] = useTransition();
31
+ const [file, setFile] = useState<File | null>(null);
31
32
  const [previewUrl, setPreviewUrl] = useState<string | null>(null); // For image preview
32
33
  const [imageDimensions, setImageDimensions] = useState<{ width: number; height: number } | null>(null); // For image dimensions
33
34
  const [uploadProgress, setUploadProgress] = useState(0);
@@ -227,75 +228,86 @@ export default function MediaUploadForm({ onUploadSuccess, returnJustData, defau
227
228
  }
228
229
 
229
230
  // 4. Record media in Supabase with variant info
230
- const finalMediaPayload = {
231
- ...mediaDataPayload,
232
- r2OriginalKey: objectKey,
233
- r2Variants: processData.processedVariants || [],
234
- originalImageDetails: processData.originalImage,
235
- blurDataURL: processData.blurDataURL || null,
236
- };
237
-
238
- const recordResult = await recordMediaUpload(finalMediaPayload, returnJustData);
239
-
240
- const handleSuccess = (newMedia?: Media) => {
241
- setUploadStatus("success");
242
- if (returnJustData && newMedia) {
243
- onUploadSuccess?.(newMedia);
244
- }
245
- // Reset form state
246
- resetFileSelection();
247
- if (processingStatus !== "processed_error") {
248
- setProcessingStatus("idle");
249
- }
250
- };
251
-
252
- if (returnJustData) {
253
- if (recordResult && 'success' in recordResult && recordResult.success && recordResult.data) {
254
- handleSuccess(recordResult.data);
255
- } else {
256
- throw new Error((recordResult && 'error' in recordResult && recordResult.error) || "Media record action failed.");
257
- }
258
- } else {
259
- handleSuccess();
260
- }
261
-
262
- } catch (err: unknown) {
263
- const isRedirect = (err instanceof Error && err.message === 'NEXT_REDIRECT') || (typeof (err as any)?.digest === 'string' && (err as any).digest.startsWith('NEXT_REDIRECT'));
264
-
265
- if (isRedirect && !returnJustData) {
266
- setUploadStatus("success");
267
- resetFileSelection();
268
- } else {
269
- console.error("Upload process error:", err);
270
- setUploadStatus("error");
271
- setErrorMessage(err instanceof Error ? err.message : "An unknown error occurred during upload.");
272
- setUploadProgress(0);
273
- }
274
- }
275
- });
276
- };
277
-
278
- const handleFormSubmit = (event: React.FormEvent<HTMLFormElement>) => {
279
- event.preventDefault();
280
- performUpload();
281
- };
282
-
283
- return (
284
- <div className="p-6 border rounded-lg shadow-sm bg-card mb-6">
285
- <form onSubmit={handleFormSubmit} className="space-y-4">
286
- <div>
287
- <Label htmlFor="media-file" className="text-base font-medium">Upload New Media</Label>
288
- <div className="mt-2 grid grid-cols-1 sm:grid-cols-2 gap-3">
231
+ const finalMediaPayload = {
232
+ ...mediaDataPayload,
233
+ r2OriginalKey: objectKey,
234
+ r2Variants: processData.processedVariants || [],
235
+ originalImageDetails: processData.originalImage,
236
+ blurDataUrl: processData.blurDataURL || null,
237
+ };
238
+
239
+ const recordResponse = await fetch('/api/media/record', {
240
+ method: 'POST',
241
+ headers: {
242
+ 'Content-Type': 'application/json',
243
+ },
244
+ body: JSON.stringify(finalMediaPayload),
245
+ });
246
+
247
+ const recordResult = await recordResponse.json();
248
+
249
+ if (!recordResponse.ok) {
250
+ throw new Error(recordResult?.error || 'Media record action failed.');
251
+ }
252
+
253
+ const handleSuccess = (newMedia?: Media) => {
254
+ setUploadStatus("success");
255
+ if (newMedia) onUploadSuccess?.(newMedia);
256
+ // Reset form state
257
+ resetFileSelection();
258
+ if (processingStatus !== "processed_error") {
259
+ setProcessingStatus("idle");
260
+ }
261
+ if (!returnJustData) {
262
+ router.refresh();
263
+ }
264
+ };
265
+
266
+ if (recordResult?.success && recordResult.data) {
267
+ handleSuccess(recordResult.data);
268
+ } else {
269
+ throw new Error(recordResult?.error || "Media record action failed.");
270
+ }
271
+
272
+ } catch (err: unknown) {
273
+ console.error("Upload process error:", err);
274
+ setUploadStatus("error");
275
+ setErrorMessage(err instanceof Error ? err.message : "An unknown error occurred during upload.");
276
+ setUploadProgress(0);
277
+ }
278
+ });
279
+ };
280
+
281
+ const handleFolderKeyDown = (event: React.KeyboardEvent<HTMLInputElement>) => {
282
+ if (event.key !== "Enter") {
283
+ return;
284
+ }
285
+
286
+ event.preventDefault();
287
+ event.stopPropagation();
288
+
289
+ if (file) {
290
+ void performUpload();
291
+ }
292
+ };
293
+
294
+ return (
295
+ <div className="p-6 border rounded-lg shadow-sm bg-card mb-6">
296
+ <div role="group" aria-label="Upload new media" className="space-y-4">
297
+ <div>
298
+ <Label htmlFor="media-file" className="text-base font-medium">Upload New Media</Label>
299
+ <div className="mt-2 grid grid-cols-1 sm:grid-cols-2 gap-3">
289
300
  <div>
290
301
  <Label htmlFor="media-folder" className="text-sm">Folder (e.g., uploads/images/)</Label>
291
302
  <Input
292
- id="media-folder"
293
- placeholder="uploads/"
294
- value={folder}
295
- onChange={(e) => setFolder(e.target.value)}
296
- />
297
- </div>
298
- </div>
303
+ id="media-folder"
304
+ placeholder="uploads/"
305
+ value={folder}
306
+ onChange={(e) => setFolder(e.target.value)}
307
+ onKeyDown={handleFolderKeyDown}
308
+ />
309
+ </div>
310
+ </div>
299
311
  <div className="mt-2 flex items-center justify-center w-full">
300
312
  <label
301
313
  htmlFor="media-file-input"
@@ -365,9 +377,9 @@ export default function MediaUploadForm({ onUploadSuccess, returnJustData, defau
365
377
  </>
366
378
  ) : (
367
379
  "Upload File"
368
- )}
369
- </Button>
370
- </form>
371
- </div>
372
- );
373
- }
380
+ )}
381
+ </Button>
382
+ </div>
383
+ </div>
384
+ );
385
+ }
@@ -1,4 +1,4 @@
1
- // app/cms/media/page.tsx
1
+ // app/cms/media/page.tsx
2
2
  import React from 'react';
3
3
  import { createClient } from "@nextblock-cms/db/server";
4
4
  // import Link from "next/link"; // Unused, MediaGridClient handles item links