create-nextblock 0.2.78 → 0.8.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (413) hide show
  1. package/bin/create-nextblock.js +740 -459
  2. package/package.json +1 -2
  3. package/scripts/sync-template.js +18 -1
  4. package/templates/nextblock-template/.browserslistrc +11 -0
  5. package/templates/nextblock-template/.swcrc +30 -30
  6. package/templates/nextblock-template/README.md +23 -114
  7. package/templates/nextblock-template/app/(auth-pages)/post-sign-in/page.tsx +27 -28
  8. package/templates/nextblock-template/app/(auth-pages)/sign-in/page.tsx +50 -25
  9. package/templates/nextblock-template/app/(auth-pages)/sign-up/page.tsx +111 -56
  10. package/templates/nextblock-template/app/(auth-pages)/two-factor/actions.ts +91 -0
  11. package/templates/nextblock-template/app/(auth-pages)/two-factor/components/TwoFactorForm.tsx +118 -0
  12. package/templates/nextblock-template/app/(auth-pages)/two-factor/page.tsx +51 -0
  13. package/templates/nextblock-template/app/.well-known/ucp/route.ts +16 -0
  14. package/templates/nextblock-template/app/[slug]/PageClientContent.tsx +48 -28
  15. package/templates/nextblock-template/app/[slug]/page.tsx +63 -6
  16. package/templates/nextblock-template/app/[slug]/page.utils.ts +374 -157
  17. package/templates/nextblock-template/app/[slug]/pageClientActions.ts +7 -0
  18. package/templates/nextblock-template/app/actions/consent.ts +57 -0
  19. package/templates/nextblock-template/app/actions/formActions.ts +130 -11
  20. package/templates/nextblock-template/app/actions/languageActions.ts +31 -30
  21. package/templates/nextblock-template/app/actions/package-actions.ts +183 -0
  22. package/templates/nextblock-template/app/actions/postActions.ts +146 -48
  23. package/templates/nextblock-template/app/actions/twoFactorEmail.ts +21 -0
  24. package/templates/nextblock-template/app/actions/visualEditingActions.test.ts +179 -0
  25. package/templates/nextblock-template/app/actions/visualEditingActions.ts +345 -0
  26. package/templates/nextblock-template/app/actions.ts +67 -12
  27. package/templates/nextblock-template/app/api/ai/cortex/build-widget/route.ts +153 -0
  28. package/templates/nextblock-template/app/api/ai/generate-blocks/route.ts +96 -0
  29. package/templates/nextblock-template/app/api/ai/global-agent/route.ts +965 -0
  30. package/templates/nextblock-template/app/api/checkout/freemius/sync/route.ts +29 -0
  31. package/templates/nextblock-template/app/api/checkout/route.ts +146 -0
  32. package/templates/nextblock-template/app/api/cms/full-backup/export/route.ts +33 -0
  33. package/templates/nextblock-template/app/api/cms/full-backup/restore/route.ts +63 -0
  34. package/templates/nextblock-template/app/api/cron/reset-sandbox/route.ts +3413 -17
  35. package/templates/nextblock-template/app/api/cron/reset-sandbox/sandboxResetSql.ts +7830 -0
  36. package/templates/nextblock-template/app/api/cron/sync-currencies/route.ts +35 -0
  37. package/templates/nextblock-template/app/api/custom-blocks/db-relations/route.ts +92 -0
  38. package/templates/nextblock-template/app/api/custom-blocks/editor-definitions/route.ts +43 -0
  39. package/templates/nextblock-template/app/api/draft/disable/route.ts +25 -0
  40. package/templates/nextblock-template/app/api/draft/route.ts +93 -0
  41. package/templates/nextblock-template/app/api/draft/start/route.ts +77 -0
  42. package/templates/nextblock-template/app/api/media/library/route.ts +65 -0
  43. package/templates/nextblock-template/app/api/media/r2-presigned/route.ts +53 -0
  44. package/templates/nextblock-template/app/api/media/record/route.ts +160 -0
  45. package/templates/nextblock-template/app/api/search/route.ts +43 -0
  46. package/templates/nextblock-template/app/api/visual-editing/block-draft/route.ts +47 -0
  47. package/templates/nextblock-template/app/api/visual-editing/product-draft/route.ts +47 -0
  48. package/templates/nextblock-template/app/api/webhooks/freemius/route.ts +34 -0
  49. package/templates/nextblock-template/app/api/webhooks/stripe/route.ts +27 -0
  50. package/templates/nextblock-template/app/article/[slug]/PostClientContent.tsx +392 -128
  51. package/templates/nextblock-template/app/article/[slug]/page.tsx +179 -127
  52. package/templates/nextblock-template/app/article/[slug]/page.utils.ts +262 -77
  53. package/templates/nextblock-template/app/auth/callback/route.ts +31 -58
  54. package/templates/nextblock-template/app/cart/page.tsx +7 -0
  55. package/templates/nextblock-template/app/checkout/UcpCartHydrator.tsx +20 -0
  56. package/templates/nextblock-template/app/checkout/page.tsx +52 -0
  57. package/templates/nextblock-template/app/checkout/success/actions.ts +136 -0
  58. package/templates/nextblock-template/app/checkout/success/page.tsx +186 -0
  59. package/templates/nextblock-template/app/cms/CmsClientLayout.tsx +163 -33
  60. package/templates/nextblock-template/app/cms/blocks/actions.ts +424 -235
  61. package/templates/nextblock-template/app/cms/blocks/components/BackgroundSelector.tsx +212 -151
  62. package/templates/nextblock-template/app/cms/blocks/components/BlockEditorArea.tsx +41 -20
  63. package/templates/nextblock-template/app/cms/blocks/components/BlockEditorModal.tsx +152 -19
  64. package/templates/nextblock-template/app/cms/blocks/components/BlockTypeCard.tsx +25 -17
  65. package/templates/nextblock-template/app/cms/blocks/components/BlockTypeSelector.tsx +200 -18
  66. package/templates/nextblock-template/app/cms/blocks/components/ColumnEditor.tsx +33 -16
  67. package/templates/nextblock-template/app/cms/blocks/components/CustomBlockEditorPreview.tsx +160 -0
  68. package/templates/nextblock-template/app/cms/blocks/components/EditableBlock.tsx +37 -18
  69. package/templates/nextblock-template/app/cms/blocks/components/MediaLibraryModal.tsx +149 -67
  70. package/templates/nextblock-template/app/cms/blocks/components/SectionConfigPanel.tsx +108 -31
  71. package/templates/nextblock-template/app/cms/blocks/editors/DynamicCustomBlockEditor.tsx +167 -0
  72. package/templates/nextblock-template/app/cms/blocks/editors/FeaturedProductBlockEditor.tsx +31 -0
  73. package/templates/nextblock-template/app/cms/blocks/editors/FormBlockEditor.tsx +2 -2
  74. package/templates/nextblock-template/app/cms/blocks/editors/HeadingBlockEditor.tsx +1 -1
  75. package/templates/nextblock-template/app/cms/blocks/editors/ImageBlockEditor.tsx +29 -29
  76. package/templates/nextblock-template/app/cms/blocks/editors/PostsGridBlockEditor.tsx +14 -18
  77. package/templates/nextblock-template/app/cms/blocks/editors/ProductGridBlockEditor.tsx +41 -0
  78. package/templates/nextblock-template/app/cms/blocks/editors/SectionBlockEditor.tsx +318 -118
  79. package/templates/nextblock-template/app/cms/blocks/editors/TextBlockEditor.tsx +98 -21
  80. package/templates/nextblock-template/app/cms/blocks/editors/VideoEmbedBlockEditor.tsx +1 -1
  81. package/templates/nextblock-template/app/cms/components/ContentLanguageSwitcher.tsx +27 -9
  82. package/templates/nextblock-template/app/cms/components/CopyContentFromLanguage.tsx +1 -1
  83. package/templates/nextblock-template/app/cms/components/CortexAiActiveContext.tsx +23 -0
  84. package/templates/nextblock-template/app/cms/components/CortexAiPageContext.tsx +58 -0
  85. package/templates/nextblock-template/app/cms/components/CortexGlobalAgentChat.tsx +1507 -0
  86. package/templates/nextblock-template/app/cms/components/DraftStatusActions.tsx +145 -0
  87. package/templates/nextblock-template/app/cms/components/FeatureImageField.tsx +244 -0
  88. package/templates/nextblock-template/app/cms/components/FeedbackModal.tsx +38 -24
  89. package/templates/nextblock-template/app/cms/coupons/[id]/edit/page.tsx +16 -0
  90. package/templates/nextblock-template/app/cms/coupons/page.tsx +16 -0
  91. package/templates/nextblock-template/app/cms/custom-blocks/[id]/edit/page.tsx +66 -0
  92. package/templates/nextblock-template/app/cms/custom-blocks/actions.ts +519 -0
  93. package/templates/nextblock-template/app/cms/custom-blocks/components/BlockComposer.tsx +1522 -0
  94. package/templates/nextblock-template/app/cms/custom-blocks/components/BlocksLibraryTransferControls.tsx +256 -0
  95. package/templates/nextblock-template/app/cms/custom-blocks/components/DBRelationSelect.tsx +384 -0
  96. package/templates/nextblock-template/app/cms/custom-blocks/components/ImageR2Picker.tsx +221 -0
  97. package/templates/nextblock-template/app/cms/custom-blocks/new/page.tsx +12 -0
  98. package/templates/nextblock-template/app/cms/custom-blocks/page.tsx +438 -0
  99. package/templates/nextblock-template/app/cms/dashboard/actions.ts +228 -98
  100. package/templates/nextblock-template/app/cms/dashboard/components/DashboardComponents.tsx +200 -0
  101. package/templates/nextblock-template/app/cms/dashboard/page.tsx +182 -154
  102. package/templates/nextblock-template/app/cms/import-export/ContentTransferControls.tsx +391 -0
  103. package/templates/nextblock-template/app/cms/import-export/actions.ts +226 -0
  104. package/templates/nextblock-template/app/cms/layout.tsx +29 -10
  105. package/templates/nextblock-template/app/cms/media/UploadFolderContext.tsx +22 -22
  106. package/templates/nextblock-template/app/cms/media/actions.ts +45 -124
  107. package/templates/nextblock-template/app/cms/media/components/DeleteMediaButtonClient.tsx +1 -1
  108. package/templates/nextblock-template/app/cms/media/components/MediaEditForm.tsx +26 -26
  109. package/templates/nextblock-template/app/cms/media/components/MediaGridClient.tsx +69 -64
  110. package/templates/nextblock-template/app/cms/media/components/MediaPickerDialog.tsx +227 -158
  111. package/templates/nextblock-template/app/cms/media/components/MediaUploadForm.tsx +101 -89
  112. package/templates/nextblock-template/app/cms/media/page.tsx +1 -1
  113. package/templates/nextblock-template/app/cms/navigation/components/NavigationItemForm.tsx +2 -2
  114. package/templates/nextblock-template/app/cms/orders/[id]/MarkPaidButton.tsx +44 -0
  115. package/templates/nextblock-template/app/cms/orders/[id]/page.tsx +16 -0
  116. package/templates/nextblock-template/app/cms/orders/actions.ts +201 -0
  117. package/templates/nextblock-template/app/cms/orders/page.tsx +20 -0
  118. package/templates/nextblock-template/app/cms/orders/types.ts +20 -0
  119. package/templates/nextblock-template/app/cms/pages/[id]/edit/EditPageClient.tsx +156 -121
  120. package/templates/nextblock-template/app/cms/pages/[id]/edit/page.tsx +79 -26
  121. package/templates/nextblock-template/app/cms/pages/actions.ts +54 -38
  122. package/templates/nextblock-template/app/cms/pages/components/DeletePageButtonClient.tsx +1 -1
  123. package/templates/nextblock-template/app/cms/pages/components/PageForm.tsx +267 -116
  124. package/templates/nextblock-template/app/cms/pages/page.tsx +25 -18
  125. package/templates/nextblock-template/app/cms/payments/page.tsx +16 -0
  126. package/templates/nextblock-template/app/cms/posts/[id]/edit/page.tsx +132 -90
  127. package/templates/nextblock-template/app/cms/posts/actions.ts +71 -72
  128. package/templates/nextblock-template/app/cms/posts/components/DeletePostButtonClient.tsx +1 -1
  129. package/templates/nextblock-template/app/cms/posts/components/PostForm.tsx +256 -245
  130. package/templates/nextblock-template/app/cms/posts/new/page.tsx +1 -1
  131. package/templates/nextblock-template/app/cms/posts/page.tsx +20 -13
  132. package/templates/nextblock-template/app/cms/products/ClientNotionEditor.tsx +16 -0
  133. package/templates/nextblock-template/app/cms/products/ProductFormClientShell.tsx +56 -0
  134. package/templates/nextblock-template/app/cms/products/[id]/edit/page.tsx +292 -0
  135. package/templates/nextblock-template/app/cms/products/attributes/page.tsx +12 -0
  136. package/templates/nextblock-template/app/cms/products/categories/page.tsx +12 -0
  137. package/templates/nextblock-template/app/cms/products/inventory/page.tsx +13 -0
  138. package/templates/nextblock-template/app/cms/products/new/page.tsx +143 -0
  139. package/templates/nextblock-template/app/cms/products/page.tsx +42 -0
  140. package/templates/nextblock-template/app/cms/products/productFormData.ts +133 -0
  141. package/templates/nextblock-template/app/cms/products/settings/page.tsx +5 -0
  142. package/templates/nextblock-template/app/cms/promotions/PromotionsWorkspace.tsx +456 -0
  143. package/templates/nextblock-template/app/cms/promotions/actions.ts +115 -0
  144. package/templates/nextblock-template/app/cms/promotions/page.tsx +31 -0
  145. package/templates/nextblock-template/app/cms/revisions/RevisionHistoryButton.tsx +2 -2
  146. package/templates/nextblock-template/app/cms/revisions/actions.ts +285 -285
  147. package/templates/nextblock-template/app/cms/revisions/service.ts +19 -16
  148. package/templates/nextblock-template/app/cms/revisions/utils.ts +8 -3
  149. package/templates/nextblock-template/app/cms/settings/backup-restore/BackupRestoreWorkspace.tsx +1004 -0
  150. package/templates/nextblock-template/app/cms/settings/backup-restore/page.tsx +29 -0
  151. package/templates/nextblock-template/app/cms/settings/bot-protection/actions.ts +93 -0
  152. package/templates/nextblock-template/app/cms/settings/bot-protection/components/BotProtectionForm.tsx +129 -0
  153. package/templates/nextblock-template/app/cms/settings/bot-protection/page.tsx +24 -0
  154. package/templates/nextblock-template/app/cms/settings/copyright/actions.ts +1 -1
  155. package/templates/nextblock-template/app/cms/settings/copyright/components/CopyrightForm.tsx +2 -2
  156. package/templates/nextblock-template/app/cms/settings/copyright/page.tsx +1 -1
  157. package/templates/nextblock-template/app/cms/settings/cortex-ai/SandboxCortexAiSettingsClient.tsx +496 -0
  158. package/templates/nextblock-template/app/cms/settings/cortex-ai/StoredCortexAiSettingsClient.tsx +410 -0
  159. package/templates/nextblock-template/app/cms/settings/cortex-ai/actions.ts +248 -0
  160. package/templates/nextblock-template/app/cms/settings/cortex-ai/page.tsx +80 -0
  161. package/templates/nextblock-template/app/cms/settings/currencies/actions.ts +331 -0
  162. package/templates/nextblock-template/app/cms/settings/currencies/page.tsx +494 -0
  163. package/templates/nextblock-template/app/cms/settings/extra-translations/ExtraTranslationsWorkspace.tsx +767 -0
  164. package/templates/nextblock-template/app/cms/settings/extra-translations/actions.ts +203 -44
  165. package/templates/nextblock-template/app/cms/settings/extra-translations/page.tsx +93 -242
  166. package/templates/nextblock-template/app/cms/settings/global-css/actions.ts +65 -0
  167. package/templates/nextblock-template/app/cms/settings/global-css/components/GlobalCssForm.tsx +46 -0
  168. package/templates/nextblock-template/app/cms/settings/global-css/page.tsx +24 -0
  169. package/templates/nextblock-template/app/cms/settings/languages/components/DeleteLanguageButton.tsx +1 -1
  170. package/templates/nextblock-template/app/cms/settings/languages/components/LanguageForm.tsx +2 -2
  171. package/templates/nextblock-template/app/cms/settings/languages/page.tsx +1 -1
  172. package/templates/nextblock-template/app/cms/settings/logos/[id]/edit/page.tsx +7 -7
  173. package/templates/nextblock-template/app/cms/settings/logos/actions.ts +82 -6
  174. package/templates/nextblock-template/app/cms/settings/logos/components/BrandingSettingsForm.tsx +339 -0
  175. package/templates/nextblock-template/app/cms/settings/logos/components/DeleteLogoButton.tsx +21 -18
  176. package/templates/nextblock-template/app/cms/settings/logos/components/LogoForm.tsx +20 -16
  177. package/templates/nextblock-template/app/cms/settings/logos/components/SiteSeoSettingsForm.tsx +133 -0
  178. package/templates/nextblock-template/app/cms/settings/logos/new/page.tsx +8 -8
  179. package/templates/nextblock-template/app/cms/settings/logos/page.tsx +120 -82
  180. package/templates/nextblock-template/app/cms/settings/logos/types.ts +8 -8
  181. package/templates/nextblock-template/app/cms/settings/packages/activation-form.tsx +84 -0
  182. package/templates/nextblock-template/app/cms/settings/packages/package-card.tsx +122 -0
  183. package/templates/nextblock-template/app/cms/settings/packages/page.tsx +49 -0
  184. package/templates/nextblock-template/app/cms/settings/privacy/actions.ts +53 -0
  185. package/templates/nextblock-template/app/cms/settings/privacy/components/PrivacyForm.tsx +196 -0
  186. package/templates/nextblock-template/app/cms/settings/privacy/page.tsx +26 -0
  187. package/templates/nextblock-template/app/cms/settings/security/actions.ts +251 -0
  188. package/templates/nextblock-template/app/cms/settings/security/components/SecurityPanel.tsx +453 -0
  189. package/templates/nextblock-template/app/cms/settings/security/page.tsx +13 -0
  190. package/templates/nextblock-template/app/cms/settings/taxes/page.tsx +21 -0
  191. package/templates/nextblock-template/app/cms/shipping/page.tsx +20 -0
  192. package/templates/nextblock-template/app/cms/users/[id]/edit/page.tsx +28 -23
  193. package/templates/nextblock-template/app/cms/users/actions.ts +105 -40
  194. package/templates/nextblock-template/app/cms/users/components/DeleteUserButton.tsx +1 -1
  195. package/templates/nextblock-template/app/cms/users/components/UserForm.tsx +65 -152
  196. package/templates/nextblock-template/app/cms/users/page.tsx +15 -10
  197. package/templates/nextblock-template/app/globals.css +9 -0
  198. package/templates/nextblock-template/app/layout.tsx +372 -120
  199. package/templates/nextblock-template/app/lib/seo.test.ts +52 -0
  200. package/templates/nextblock-template/app/lib/seo.ts +279 -0
  201. package/templates/nextblock-template/app/lib/site-settings.ts +87 -0
  202. package/templates/nextblock-template/app/lib/sitemap-utils.ts +224 -39
  203. package/templates/nextblock-template/app/lib/ucp/protocol.ts +190 -0
  204. package/templates/nextblock-template/app/lib/ucp/server.test.ts +56 -0
  205. package/templates/nextblock-template/app/lib/ucp/server.ts +1914 -0
  206. package/templates/nextblock-template/app/page.tsx +165 -73
  207. package/templates/nextblock-template/app/product/[slug]/page.tsx +433 -0
  208. package/templates/nextblock-template/app/profile/ProfileAccountSidebar.tsx +73 -0
  209. package/templates/nextblock-template/app/profile/ProfilePageHeader.tsx +16 -0
  210. package/templates/nextblock-template/app/profile/ProfilePageMissingState.tsx +9 -0
  211. package/templates/nextblock-template/app/profile/account-data.ts +37 -0
  212. package/templates/nextblock-template/app/profile/account-links.ts +22 -0
  213. package/templates/nextblock-template/app/profile/account-types.ts +11 -0
  214. package/templates/nextblock-template/app/profile/orders/CustomerOrdersPageClient.tsx +124 -0
  215. package/templates/nextblock-template/app/profile/orders/[id]/CustomerOrderDetailPageClient.tsx +79 -0
  216. package/templates/nextblock-template/app/profile/orders/[id]/page.tsx +32 -0
  217. package/templates/nextblock-template/app/profile/orders/page.tsx +19 -0
  218. package/templates/nextblock-template/app/profile/page.tsx +51 -0
  219. package/templates/nextblock-template/app/profile/password/PasswordSettingsPageClient.tsx +128 -0
  220. package/templates/nextblock-template/app/profile/password/actions.ts +59 -0
  221. package/templates/nextblock-template/app/profile/password/page.tsx +27 -0
  222. package/templates/nextblock-template/app/providers.tsx +55 -17
  223. package/templates/nextblock-template/app/robots.txt/route.ts +11 -1
  224. package/templates/nextblock-template/app/sitemap.ts +128 -0
  225. package/templates/nextblock-template/app/ucp/v1/carts/[id]/cancel/route.ts +38 -0
  226. package/templates/nextblock-template/app/ucp/v1/carts/[id]/route.ts +68 -0
  227. package/templates/nextblock-template/app/ucp/v1/carts/route.ts +35 -0
  228. package/templates/nextblock-template/app/ucp/v1/catalog/lookup/route.ts +35 -0
  229. package/templates/nextblock-template/app/ucp/v1/catalog/product/route.ts +35 -0
  230. package/templates/nextblock-template/app/ucp/v1/catalog/search/route.ts +34 -0
  231. package/templates/nextblock-template/components/AppShell.tsx +154 -0
  232. package/templates/nextblock-template/components/BlockRenderer.tsx +210 -64
  233. package/templates/nextblock-template/components/CartDrawerLoader.tsx +7 -0
  234. package/templates/nextblock-template/components/CartTranslator.tsx +210 -0
  235. package/templates/nextblock-template/components/CurrentContentSetter.tsx +25 -0
  236. package/templates/nextblock-template/components/DeferredCartDrawer.tsx +23 -0
  237. package/templates/nextblock-template/components/DeferredCartTranslator.tsx +51 -0
  238. package/templates/nextblock-template/components/DeferredGlobalSearch.tsx +68 -0
  239. package/templates/nextblock-template/components/DeferredGoogleTagManager.tsx +70 -0
  240. package/templates/nextblock-template/components/DeferredSpeedInsights.tsx +69 -0
  241. package/templates/nextblock-template/components/FeatureImageHero.tsx +47 -0
  242. package/templates/nextblock-template/components/GitHubLoginButton.tsx +36 -0
  243. package/templates/nextblock-template/components/GlobalSearch.tsx +557 -0
  244. package/templates/nextblock-template/components/Header.tsx +49 -41
  245. package/templates/nextblock-template/components/LanguageSwitcher.tsx +55 -32
  246. package/templates/nextblock-template/components/ResponsiveNav.tsx +138 -43
  247. package/templates/nextblock-template/components/blocks/PostCardSkeleton.tsx +12 -8
  248. package/templates/nextblock-template/components/blocks/PostsGridBlock.tsx +12 -55
  249. package/templates/nextblock-template/components/blocks/PostsGridClient.tsx +42 -37
  250. package/templates/nextblock-template/components/blocks/TestimonialBlock.tsx +6 -2
  251. package/templates/nextblock-template/components/blocks/ecommerceRendererLoaders.ts +23 -0
  252. package/templates/nextblock-template/components/blocks/publicRendererLoaders.ts +25 -0
  253. package/templates/nextblock-template/components/blocks/renderers/ButtonBlockRenderer.tsx +92 -84
  254. package/templates/nextblock-template/components/blocks/renderers/CartBlockRenderer.tsx +17 -0
  255. package/templates/nextblock-template/components/blocks/renderers/CheckoutBlockRenderer.tsx +19 -0
  256. package/templates/nextblock-template/components/blocks/renderers/ClientTextBlockRenderer.tsx +262 -8
  257. package/templates/nextblock-template/components/blocks/renderers/FeaturedProductBlockRenderer.tsx +22 -0
  258. package/templates/nextblock-template/components/blocks/renderers/FormBlockRenderer.tsx +320 -37
  259. package/templates/nextblock-template/components/blocks/renderers/HeadingBlockRenderer.tsx +11 -8
  260. package/templates/nextblock-template/components/blocks/renderers/ImageBlockRenderer.tsx +12 -3
  261. package/templates/nextblock-template/components/blocks/renderers/PostsGridBlockRenderer.tsx +18 -13
  262. package/templates/nextblock-template/components/blocks/renderers/ProductDetailsBlockRenderer.tsx +90 -0
  263. package/templates/nextblock-template/components/blocks/renderers/ProductGridBlockRenderer.tsx +31 -0
  264. package/templates/nextblock-template/components/blocks/renderers/SectionBlockRenderer.tsx +424 -55
  265. package/templates/nextblock-template/components/blocks/renderers/SectionSlider.tsx +137 -0
  266. package/templates/nextblock-template/components/blocks/renderers/TestimonialBlockRenderer.tsx +57 -0
  267. package/templates/nextblock-template/components/blocks/renderers/TextBlockRenderer.tsx +37 -22
  268. package/templates/nextblock-template/components/blocks/renderers/VideoEmbedBlockRenderer.tsx +23 -15
  269. package/templates/nextblock-template/components/blocks/renderers/inline/AlertWidgetRenderer.tsx +1 -3
  270. package/templates/nextblock-template/components/blocks/renderers/inline/CtaWidgetRenderer.tsx +1 -3
  271. package/templates/nextblock-template/components/blocks/types.ts +7 -6
  272. package/templates/nextblock-template/components/env-var-warning.tsx +3 -3
  273. package/templates/nextblock-template/components/form-message.tsx +32 -26
  274. package/templates/nextblock-template/components/header-auth.tsx +69 -17
  275. package/templates/nextblock-template/components/privacy/ConsentBanner.tsx +127 -0
  276. package/templates/nextblock-template/components/privacy/ConsentGatedAnalytics.tsx +59 -0
  277. package/templates/nextblock-template/components/renderers/CachedDynamicLayoutEngine.tsx +28 -0
  278. package/templates/nextblock-template/components/renderers/DynamicLayoutEngine.test.tsx +166 -0
  279. package/templates/nextblock-template/components/renderers/DynamicLayoutEngine.tsx +464 -0
  280. package/templates/nextblock-template/components/theme-switcher.tsx +8 -8
  281. package/templates/nextblock-template/components/visual-editing/DeferredVisualEditing.tsx +21 -0
  282. package/templates/nextblock-template/components/visual-editing/NextblockVisualEditing.tsx +1172 -0
  283. package/templates/nextblock-template/context/AuthContext.tsx +23 -90
  284. package/templates/nextblock-template/context/CurrentContentContext.tsx +10 -4
  285. package/templates/nextblock-template/context/LanguageContext.tsx +16 -16
  286. package/templates/nextblock-template/context/language-rest-client.ts +31 -0
  287. package/templates/nextblock-template/docs/01-PROJECT-OVERVIEW.md +94 -0
  288. package/templates/nextblock-template/docs/02-ECOMMERCE-CAPABILITIES.md +364 -0
  289. package/templates/nextblock-template/docs/03-CMS-AND-EDITOR.md +202 -0
  290. package/templates/nextblock-template/docs/04-DATABASE-AND-AUTH.md +252 -0
  291. package/templates/nextblock-template/docs/05-DEVELOPER-GUIDE.md +238 -0
  292. package/templates/nextblock-template/docs/06-CLI-AND-SCAFFOLDING.md +125 -0
  293. package/templates/nextblock-template/docs/07-BLOCK-SDK-AND-EXTENSIBILITY.md +146 -0
  294. package/templates/nextblock-template/docs/08-NEXTBLOCK-CORTEX-AI-ARCHITECTURE.md +1319 -0
  295. package/templates/nextblock-template/docs/09-LIVE-DRAFT-MODE.md +104 -0
  296. package/templates/nextblock-template/docs/10-CUSTOM-BLOCKS.md +222 -0
  297. package/templates/nextblock-template/docs/README.md +34 -0
  298. package/templates/nextblock-template/docs/TECHNICAL_SPECIFICATION.md +12507 -0
  299. package/templates/nextblock-template/hooks/use-hotkeys.ts +21 -14
  300. package/templates/nextblock-template/hooks/useGlobalSearch.ts +101 -0
  301. package/templates/nextblock-template/index.d.ts +2 -0
  302. package/templates/nextblock-template/lib/ai-block-generation.ts +339 -0
  303. package/templates/nextblock-template/lib/ai-client.ts +247 -0
  304. package/templates/nextblock-template/lib/ai-config.ts +81 -0
  305. package/templates/nextblock-template/lib/ai-cortex-widget-builder.ts +125 -0
  306. package/templates/nextblock-template/lib/ai-global-agent-custom-block-tools.ts +363 -0
  307. package/templates/nextblock-template/lib/ai-global-agent-db-tools.test.ts +405 -0
  308. package/templates/nextblock-template/lib/ai-global-agent-db-tools.ts +1228 -0
  309. package/templates/nextblock-template/lib/ai-global-agent-ecommerce.ts +5 -0
  310. package/templates/nextblock-template/lib/ai-global-agent-tools-stats.test.ts +223 -0
  311. package/templates/nextblock-template/lib/ai-global-agent-tools.test.ts +2183 -0
  312. package/templates/nextblock-template/lib/ai-global-agent-tools.ts +4807 -0
  313. package/templates/nextblock-template/lib/ai-key-crypto.test.ts +70 -0
  314. package/templates/nextblock-template/lib/ai-key-crypto.ts +132 -0
  315. package/templates/nextblock-template/lib/ai-model-catalog.test.ts +49 -0
  316. package/templates/nextblock-template/lib/ai-model-catalog.ts +41 -0
  317. package/templates/nextblock-template/lib/ai-model-registry.test.ts +231 -0
  318. package/templates/nextblock-template/lib/ai-model-registry.ts +522 -0
  319. package/templates/nextblock-template/lib/auth/cookies.ts +47 -0
  320. package/templates/nextblock-template/lib/auth/crypto.ts +42 -0
  321. package/templates/nextblock-template/lib/auth/trustedDevices.ts +92 -0
  322. package/templates/nextblock-template/lib/auth/twoFactor.ts +167 -0
  323. package/templates/nextblock-template/lib/auth-redirects.ts +46 -0
  324. package/templates/nextblock-template/lib/blocks/FeaturedProductBlock.tsx +94 -0
  325. package/templates/nextblock-template/lib/blocks/ProductGridBlock.tsx +137 -0
  326. package/templates/nextblock-template/lib/blocks/README.md +13 -670
  327. package/templates/nextblock-template/lib/blocks/blockRegistry.ts +138 -56
  328. package/templates/nextblock-template/lib/blocks/blockTypes.ts +18 -0
  329. package/templates/nextblock-template/lib/blocks/ecommerce-block-schemas.ts +31 -0
  330. package/templates/nextblock-template/lib/cms-transfer/csv.test.ts +77 -0
  331. package/templates/nextblock-template/lib/cms-transfer/csv.ts +399 -0
  332. package/templates/nextblock-template/lib/cms-transfer/server.ts +2243 -0
  333. package/templates/nextblock-template/lib/cms-transfer/types.ts +145 -0
  334. package/templates/nextblock-template/lib/cortex-widget-registry.test.ts +199 -0
  335. package/templates/nextblock-template/lib/cortex-widget-registry.ts +88 -0
  336. package/templates/nextblock-template/lib/cortex-widget-schema.test.tsx +237 -0
  337. package/templates/nextblock-template/lib/cortex-widget-schema.ts +393 -0
  338. package/templates/nextblock-template/lib/custom-block-definitions.ts +87 -0
  339. package/templates/nextblock-template/lib/custom-block-r2-upload-shared.ts +178 -0
  340. package/templates/nextblock-template/lib/custom-block-r2-upload.test.ts +140 -0
  341. package/templates/nextblock-template/lib/custom-block-r2-upload.ts +68 -0
  342. package/templates/nextblock-template/lib/custom-block-relation-registry.ts +256 -0
  343. package/templates/nextblock-template/lib/custom-block-relations.test.ts +227 -0
  344. package/templates/nextblock-template/lib/custom-block-relations.ts +279 -0
  345. package/templates/nextblock-template/lib/custom-block-safelist.ts +14 -0
  346. package/templates/nextblock-template/lib/editor/dynamic-extension-core.test.ts +172 -0
  347. package/templates/nextblock-template/lib/editor/dynamic-extension-core.ts +213 -0
  348. package/templates/nextblock-template/lib/editor/dynamic-extension-loader.ts +22 -0
  349. package/templates/nextblock-template/lib/editor/dynamic-extensions.tsx +193 -0
  350. package/templates/nextblock-template/lib/full-backup/manifest.test.ts +121 -0
  351. package/templates/nextblock-template/lib/full-backup/manifest.ts +206 -0
  352. package/templates/nextblock-template/lib/full-backup/server.ts +743 -0
  353. package/templates/nextblock-template/lib/media/resolveMediaUrl.ts +45 -0
  354. package/templates/nextblock-template/lib/posts/readTime.ts +60 -0
  355. package/templates/nextblock-template/lib/privacy/consent-client.ts +57 -0
  356. package/templates/nextblock-template/lib/privacy/settings.ts +103 -0
  357. package/templates/nextblock-template/lib/privacy/types.ts +67 -0
  358. package/templates/nextblock-template/lib/promotions/server.test.ts +74 -0
  359. package/templates/nextblock-template/lib/promotions/server.ts +741 -0
  360. package/templates/nextblock-template/lib/resolve-block-relations.test.ts +142 -0
  361. package/templates/nextblock-template/lib/resolve-block-relations.ts +255 -0
  362. package/templates/nextblock-template/lib/search/server.ts +585 -0
  363. package/templates/nextblock-template/lib/search/types.ts +27 -0
  364. package/templates/nextblock-template/lib/visual-editing/draft-content.test.ts +105 -0
  365. package/templates/nextblock-template/lib/visual-editing/draft-content.ts +380 -0
  366. package/templates/nextblock-template/lib/visual-editing/draft-route.test.ts +42 -0
  367. package/templates/nextblock-template/lib/visual-editing/draft-route.ts +82 -0
  368. package/templates/nextblock-template/lib/visual-editing/edit-info.test.ts +143 -0
  369. package/templates/nextblock-template/lib/visual-editing/edit-info.ts +94 -0
  370. package/templates/nextblock-template/lib/visual-editing/mutations.ts +190 -0
  371. package/templates/nextblock-template/lib/visual-editing/product-drafts.test.ts +81 -0
  372. package/templates/nextblock-template/lib/visual-editing/product-drafts.ts +511 -0
  373. package/templates/nextblock-template/lib/visual-editing/types.ts +122 -0
  374. package/templates/nextblock-template/lib/zod-config.ts +5 -0
  375. package/templates/nextblock-template/next.config.js +190 -66
  376. package/templates/nextblock-template/package.json +34 -30
  377. package/templates/nextblock-template/proxy.ts +435 -253
  378. package/templates/nextblock-template/public/images/NBcover.webp +0 -0
  379. package/templates/nextblock-template/public/images/cap.webp +0 -0
  380. package/templates/nextblock-template/public/images/commerce-plan.webp +0 -0
  381. package/templates/nextblock-template/public/images/commerce-square.webp +0 -0
  382. package/templates/nextblock-template/public/images/commerce-wide.webp +0 -0
  383. package/templates/nextblock-template/public/images/cortex-ai-square.webp +0 -0
  384. package/templates/nextblock-template/public/images/cortex-ai.webp +0 -0
  385. package/templates/nextblock-template/public/images/extensibility.webp +0 -0
  386. package/templates/nextblock-template/public/images/goals.webp +0 -0
  387. package/templates/nextblock-template/public/images/included.webp +0 -0
  388. package/templates/nextblock-template/public/images/nx-graph.webp +0 -0
  389. package/templates/nextblock-template/public/images/pants.webp +0 -0
  390. package/templates/nextblock-template/public/images/t-shirt.webp +0 -0
  391. package/templates/nextblock-template/scripts/validate-editor-block-schema.ts +112 -0
  392. package/templates/nextblock-template/scripts/verify-cortex-ai-build-widget.tsx +100 -0
  393. package/templates/nextblock-template/scripts/verify-cortex-ai-generate-blocks.ts +62 -0
  394. package/templates/nextblock-template/scripts/verify-cortex-ai-global-tools.ts +537 -0
  395. package/templates/nextblock-template/scripts/verify-cortex-ai-routing.ts +58 -0
  396. package/templates/nextblock-template/scripts/verify-custom-block-definitions.ts +188 -0
  397. package/templates/nextblock-template/scripts/verify-dynamic-custom-block-extensions.ts +123 -0
  398. package/templates/nextblock-template/scripts/verify-dynamic-layout-engine.tsx +133 -0
  399. package/templates/nextblock-template/scripts/verify-milestone-2-custom-blocks.ts +65 -0
  400. package/templates/nextblock-template/tailwind.config.js +1 -0
  401. package/templates/nextblock-template/tools/configure-supabase-auth.js +282 -0
  402. package/templates/nextblock-template/tools/deploy-supabase.js +69 -71
  403. package/templates/nextblock-template/tsconfig.json +52 -66
  404. package/templates/nextblock-template/tsconfig.tsbuildinfo +1 -1
  405. package/templates/nextblock-template/types/jsdom.d.ts +6 -0
  406. package/templates/nextblock-template/app/force-styles.tsx +0 -31
  407. package/templates/nextblock-template/app/sitemap.xml/route.ts +0 -63
  408. package/templates/nextblock-template/components/blocks/renderers/HeroBlockRenderer.tsx +0 -273
  409. package/templates/nextblock-template/docs/How to Create a Custom Block.md +0 -149
  410. package/templates/nextblock-template/docs/cms-application-overview.md +0 -56
  411. package/templates/nextblock-template/docs/cms-architecture-overview.md +0 -73
  412. package/templates/nextblock-template/docs/files-structure.md +0 -426
  413. package/templates/nextblock-template/docs/tiptap-bundle-optimization-summary.md +0 -174
@@ -12,130 +12,12 @@ type Media = Database['public']['Tables']['media']['Row'];
12
12
 
13
13
  // --- recordMediaUpload and updateMediaItem functions to be updated similarly ---
14
14
 
15
- // Define the structure for a single variant, mirroring what /api/process-image returns
16
- interface ImageVariant {
17
- objectKey: string;
18
- url: string;
19
- width: number;
20
- height: number;
21
- fileType: string;
22
- sizeBytes: number;
23
- variantLabel: string;
24
- }
25
-
26
- export async function recordMediaUpload(payload: {
27
- fileName: string; // Original filename, can still be useful
28
- // objectKey: string; // This might now be derived from the primary variant
29
- // fileType: string; // This will come from the primary variant
30
- // sizeBytes: number; // This will come from the primary variant
31
- description?: string;
32
- // width?: number; // This will come from the primary variant
33
- // height?: number; // This will come from the primary variant
34
- r2OriginalKey: string; // Key of the initially uploaded file in R2
35
- r2Variants: ImageVariant[]; // Array of processed variants
36
- originalImageDetails: ImageVariant; // Details of the original uploaded image from process-image
37
- blurDataUrl?: string; // Optional, if generated and passed from client
38
- }, returnJustData?: boolean): Promise<{ success: true; data: Media } | { error: string } | void> {
39
- const supabase = createClient();
40
- const { data: { user }, error: authError } = await supabase.auth.getUser();
41
-
42
- if (authError || !user) {
43
- if (returnJustData) return { error: "User not authenticated for media record." };
44
- return encodedRedirect("error", "/cms/media", "User not authenticated for media record.");
45
- }
46
-
47
- const { data: profile } = await supabase
48
- .from("profiles")
49
- .select("role")
50
- .eq("id", user.id)
51
- .single();
52
- if (!profile || !["ADMIN", "WRITER"].includes(profile.role)) {
53
- if (returnJustData) return { error: "Forbidden: Insufficient permissions to record media." };
54
- return encodedRedirect("error", "/cms/media", "Forbidden: Insufficient permissions to record media.");
55
- }
56
-
57
- // Determine the primary variant to use for the main table columns
58
- // This logic can be adjusted. Prioritize 'original_avif', then 'xlarge_avif', then the first variant.
59
- let primaryVariant =
60
- payload.r2Variants.find(v => v.variantLabel === 'original_avif') ||
61
- payload.r2Variants.find(v => v.variantLabel === 'xlarge_avif') ||
62
- payload.r2Variants[0] || // Fallback to the first variant if specific ones aren't found
63
- payload.originalImageDetails; // Fallback to original uploaded details if no variants
64
-
65
- if (!primaryVariant) {
66
- // This case should ideally not happen if originalImageDetails is always present
67
- // but as a safeguard:
68
- primaryVariant = payload.originalImageDetails || {
69
- objectKey: payload.r2OriginalKey,
70
- url: `YOUR_R2_PUBLIC_BASE_URL/${payload.r2OriginalKey}`, // Construct URL if needed
71
- width: 0, // Or fetch if necessary, though client sends initial dimensions
72
- height: 0,
73
- fileType: 'application/octet-stream', // A generic fallback
74
- sizeBytes: 0,
75
- variantLabel: 'fallback_original'
76
- };
77
- }
78
-
79
- // Construct the full list of variants to store, including the original uploaded file details
80
- const allVariantsToStore = [
81
- ...(payload.originalImageDetails && payload.originalImageDetails.objectKey !== primaryVariant.objectKey ? [payload.originalImageDetails] : []),
82
- ...payload.r2Variants,
83
- ].filter((variant, index, self) =>
84
- index === self.findIndex((v) => v.objectKey === variant.objectKey)
85
- ); // Ensure unique variants by objectKey
86
-
87
- // If no description provided for images, derive one from the filename (un-slug)
88
- const deriveAltFromFilename = (name: string) => {
89
- const lastDot = name.lastIndexOf('.');
90
- const base = lastDot > 0 ? name.substring(0, lastDot) : name;
91
- const spaced = base.replace(/[-+_\\]+/g, ' ').replace(/\s+/g, ' ').trim();
92
- return spaced.replace(/\b\w+/g, (w) => w.charAt(0).toUpperCase() + w.slice(1));
93
- };
94
-
95
- const computedDescription = payload.description
96
- ?? ((primaryVariant.fileType?.startsWith('image/') || payload.originalImageDetails?.fileType?.startsWith('image/'))
97
- ? deriveAltFromFilename(payload.fileName)
98
- : null);
99
-
100
- const mediaData: Omit<Media, 'id' | 'created_at' | 'updated_at'> & { uploader_id: string } = {
101
- uploader_id: user.id,
102
- file_name: payload.fileName, // Keep original file name for reference
103
- object_key: primaryVariant.objectKey, // Key of the primary display version
104
- file_path: primaryVariant.objectKey,
105
- folder: (() => {
106
- const match = primaryVariant.objectKey.match(/^(.*\/)?.*$/);
107
- const path = match && match[1] ? match[1] : null;
108
- return path;
109
- })(),
110
- // file_url is removed as it's not in the Media type; URLs are in variants
111
- file_type: primaryVariant.fileType,
112
- size_bytes: primaryVariant.sizeBytes,
113
- description: computedDescription,
114
- width: primaryVariant.width,
115
- height: primaryVariant.height,
116
- variants: allVariantsToStore as any, // Store all variants including the original
117
- blur_data_url: payload.blurDataUrl || null, // Store if provided
118
- // Ensure all other required fields for 'Media' type are present or nullable
119
- };
120
15
 
121
- const { data: newMedia, error } = await supabase
122
- .from("media")
123
- .insert(mediaData)
124
- .select()
125
- .single();
126
16
 
127
- if (error) {
128
- console.error("Error recording media upload:", error);
129
- if (returnJustData) return { error: `Failed to record media: ${error.message}` };
130
- return encodedRedirect("error", "/cms/media", `Failed to record media: ${error.message}`);
131
- }
17
+ import { recordMediaUpload as sharedRecordMediaUpload } from "@nextblock-cms/db";
132
18
 
133
- revalidatePath("/cms/media");
134
- if (returnJustData) {
135
- return { success: true, data: newMedia as Media };
136
- } else {
137
- encodedRedirect("success", "/cms/media", "Media recorded successfully.");
138
- }
19
+ export async function recordMediaUpload(payload: any, returnJustData?: boolean) {
20
+ return sharedRecordMediaUpload(payload, returnJustData);
139
21
  }
140
22
 
141
23
 
@@ -213,10 +95,26 @@ export async function deleteMediaItem(mediaId: string, objectKey: string) {
213
95
  return encodedRedirect("error", "/cms/media", "R2 client is not configured for deletion.");
214
96
  }
215
97
 
98
+ // Fetch media to get variants
99
+ const { data: mediaToDelete } = await supabase
100
+ .from('media')
101
+ .select('variants')
102
+ .eq('id', mediaId)
103
+ .single();
104
+
105
+ const keysToDelete = [objectKey];
106
+ if (mediaToDelete?.variants && Array.isArray(mediaToDelete.variants)) {
107
+ mediaToDelete.variants.forEach((v: any) => {
108
+ if (v.objectKey) keysToDelete.push(v.objectKey);
109
+ });
110
+ }
111
+
216
112
  try {
217
- const deleteCommand = new DeleteObjectCommand({
113
+ const deleteCommand = new DeleteObjectsCommand({
218
114
  Bucket: R2_BUCKET_NAME,
219
- Key: objectKey,
115
+ Delete: {
116
+ Objects: keysToDelete.map(key => ({ Key: key })),
117
+ },
220
118
  });
221
119
  await s3Client.send(deleteCommand);
222
120
  } catch (r2Error: unknown) {
@@ -265,7 +163,30 @@ export async function deleteMultipleMediaItems(items: Array<{ id: string; object
265
163
  return { error: "R2 client is not configured for deletion." };
266
164
  }
267
165
 
268
- const r2ObjectsToDelete = items.map(item => ({ Key: item.objectKey }));
166
+ // Fetch variants for all items
167
+ const { data: mediaItems } = await supabase
168
+ .from('media')
169
+ .select('id, object_key, variants')
170
+ .in('id', items.map(i => i.id));
171
+
172
+ const allKeysToDelete: string[] = [];
173
+
174
+ // If we found the items in DB, use their variants data
175
+ if (mediaItems) {
176
+ mediaItems.forEach(item => {
177
+ allKeysToDelete.push(item.object_key);
178
+ if (item.variants && Array.isArray(item.variants)) {
179
+ item.variants.forEach((v: any) => {
180
+ if (v.objectKey) allKeysToDelete.push(v.objectKey);
181
+ });
182
+ }
183
+ });
184
+ } else {
185
+ // Fallback to just the provided keys if DB fetch fails (unlikely)
186
+ items.forEach(item => allKeysToDelete.push(item.objectKey));
187
+ }
188
+
189
+ const r2ObjectsToDelete = allKeysToDelete.map(key => ({ Key: key }));
269
190
  const itemIdsToDelete = items.map(item => item.id);
270
191
  let r2DeletionError = null;
271
192
  let dbDeletionError = null;
@@ -6,7 +6,7 @@ import { DropdownMenuItem } from "@nextblock-cms/ui";
6
6
  import { Trash2 } from "lucide-react";
7
7
  import type { Database } from "@nextblock-cms/db";
8
8
  import { deleteMediaItem } from "../actions";
9
- import { ConfirmationModal } from '@/app/cms/components/ConfirmationModal';
9
+ import { ConfirmationModal } from '../../components/ConfirmationModal';
10
10
 
11
11
  type Media = Database['public']['Tables']['media']['Row'];
12
12
 
@@ -9,22 +9,21 @@ import { Spinner, Alert, AlertDescription } from '@nextblock-cms/ui';
9
9
  import { Input } from '@nextblock-cms/ui';
10
10
  import { Label } from '@nextblock-cms/ui';
11
11
  import { Textarea } from '@nextblock-cms/ui';
12
- import type { Database } from '@nextblock-cms/db';
13
- import { useAuth } from '@/context/AuthContext';
14
- import { useHotkeys } from '@/hooks/use-hotkeys';
12
+ import type { Database } from '@nextblock-cms/db';
13
+ import { useAuth } from '../../../../context/AuthContext';
14
+ import { useHotkeys } from '../../../../hooks/use-hotkeys';
15
+ import { resolveMediaUrl } from '../../../../lib/media/resolveMediaUrl';
15
16
 
16
17
  type Media = Database['public']['Tables']['media']['Row'];
17
18
  import { FileText } from 'lucide-react';
18
19
 
19
- interface MediaEditFormProps {
20
- mediaItem: Media;
21
- // The formAction will be updateMediaItem bound with the mediaItem.id
22
- formAction: (formData: FormData) => Promise<{ error?: string; success?: boolean; media?: Media } | void>;
23
- }
24
-
25
- const R2_BASE_URL = process.env.NEXT_PUBLIC_R2_BASE_URL || "";
26
-
27
- export default function MediaEditForm({ mediaItem, formAction }: MediaEditFormProps) {
20
+ interface MediaEditFormProps {
21
+ mediaItem: Media;
22
+ // The formAction will be updateMediaItem bound with the mediaItem.id
23
+ formAction: (formData: FormData) => Promise<{ error?: string; success?: boolean; media?: Media } | void>;
24
+ }
25
+
26
+ export default function MediaEditForm({ mediaItem, formAction }: MediaEditFormProps) {
28
27
  const router = useRouter();
29
28
  const searchParams = useSearchParams();
30
29
  const [isPending, startTransition] = useTransition();
@@ -76,19 +75,20 @@ export default function MediaEditForm({ mediaItem, formAction }: MediaEditFormPr
76
75
  return <div>Access Denied. You do not have permission to edit media.</div>;
77
76
  }
78
77
 
79
- const formRef = React.useRef<HTMLFormElement>(null);
80
- useHotkeys('ctrl+s', () => formRef.current?.requestSubmit());
81
-
82
- return (
83
- <div className="grid grid-cols-1 md:grid-cols-3 gap-6">
84
- <div className="md:col-span-1 space-y-4">
85
- <h2 className="text-lg font-semibold">Media Preview</h2>
86
- {mediaItem.file_type?.startsWith("image/") ? (
87
- <Image
88
- src={`${R2_BASE_URL}/${mediaItem.object_key}`}
89
- alt={description || fileName}
90
- width={400}
91
- height={400}
78
+ const formRef = React.useRef<HTMLFormElement>(null);
79
+ useHotkeys('ctrl+s', () => formRef.current?.requestSubmit());
80
+ const previewUrl = resolveMediaUrl(mediaItem.file_path || mediaItem.object_key);
81
+
82
+ return (
83
+ <div className="grid grid-cols-1 md:grid-cols-3 gap-6">
84
+ <div className="md:col-span-1 space-y-4">
85
+ <h2 className="text-lg font-semibold">Media Preview</h2>
86
+ {mediaItem.file_type?.startsWith("image/") && previewUrl ? (
87
+ <Image
88
+ src={previewUrl}
89
+ alt={description || fileName}
90
+ width={400}
91
+ height={400}
92
92
  className="rounded-lg border object-contain aspect-square w-full max-w-sm mx-auto"
93
93
  />
94
94
  ) : (
@@ -159,4 +159,4 @@ export default function MediaEditForm({ mediaItem, formAction }: MediaEditFormPr
159
159
  </form>
160
160
  </div>
161
161
  );
162
- }
162
+ }
@@ -17,9 +17,10 @@ import {
17
17
  DropdownMenuTrigger,
18
18
  DropdownMenuSeparator,
19
19
  } from "@nextblock-cms/ui";
20
- import MediaImage from "./MediaImage";
21
- import DeleteMediaButtonClient from "./DeleteMediaButtonClient"; // For single item deletion
22
- import { deleteMultipleMediaItems, moveSingleMediaItem } from "../actions"; // Server actions for bulk ops
20
+ import MediaImage from "./MediaImage";
21
+ import DeleteMediaButtonClient from "./DeleteMediaButtonClient"; // For single item deletion
22
+ import { deleteMultipleMediaItems, moveSingleMediaItem } from "../actions"; // Server actions for bulk ops
23
+ import { resolveMediaUrl } from "../../../../lib/media/resolveMediaUrl";
23
24
 
24
25
  interface MediaGridClientProps {
25
26
  initialMediaItems: Media[];
@@ -209,67 +210,71 @@ export default function MediaGridClient({ initialMediaItems, r2BaseUrl }: MediaG
209
210
  </p>
210
211
  </div>
211
212
  ) : (
212
- <div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 lg:grid-cols-5 xl:grid-cols-6 gap-4 mt-6">
213
- {mediaItems.map((item) => (
214
- <div
215
- key={item.id}
216
- className={`group relative border rounded-lg overflow-hidden shadow-sm aspect-square bg-muted/20 transition-all
217
- ${isSelected(item.id) ? "ring-2 ring-primary ring-offset-2" : ""}`}
218
- >
219
- <div className="absolute top-2 left-2 z-10">
220
- <Checkbox
221
- id={`select-${item.id}`}
222
- checked={isSelected(item.id)}
223
- onCheckedChange={(checked) => {
224
- handleSelectionChange(item.id, item.object_key, !!checked);
225
- }}
226
- aria-label={`Select ${item.file_name}`}
227
- className="bg-white/70 hover:bg-white border-slate-400 data-[state=checked]:bg-primary data-[state=checked]:text-primary-foreground"
228
- />
229
- </div>
230
-
231
- {item.file_type?.startsWith("image/") ? (
232
- <MediaImage
233
- src={`${r2BaseUrl}/${item.object_key}`}
234
- alt={item.description || item.file_name}
235
- width={item.width || 500} // Default width if null
236
- height={item.height || 500} // Default height if null
237
- blurDataURL={item.blur_data_url || undefined}
238
- className="h-full w-full object-contain transition-transform group-hover:scale-105"
239
- />
240
- ) : (
241
- <div className="h-full w-full bg-muted flex flex-col items-center justify-center p-2">
242
- <FileText className="h-12 w-12 text-muted-foreground mb-2" />
243
- <p className="text-xs text-center text-muted-foreground truncate w-full" title={item.file_name}>
244
- {item.file_name}
245
- </p>
246
- </div>
247
- )}
248
- <div className="absolute inset-0 bg-gradient-to-t from-black/50 via-black/20 to-transparent opacity-0 group-hover:opacity-100 transition-opacity flex flex-col justify-between p-2">
249
- <div className="text-xs text-black truncate pt-1 ps-5" title={item.file_name}>{item.file_name}</div>
250
- <div className="self-end">
251
- <DropdownMenu>
252
- <DropdownMenuTrigger asChild>
253
- <Button variant="secondary" size="icon" className="text-white bg-black/40 hover:bg-black/60 h-7 w-7 rounded-full">
254
- <MoreHorizontal className="h-4 w-4" />
255
- </Button>
256
- </DropdownMenuTrigger>
257
- <DropdownMenuContent align="end">
258
- <DropdownMenuItem asChild>
259
- <Link href={`/cms/media/${item.id}/edit`} className="flex items-center cursor-pointer">
260
- <Edit3 className="mr-2 h-4 w-4" /> Edit Details
261
- </Link>
262
- </DropdownMenuItem>
263
- <DropdownMenuSeparator />
264
- <DeleteMediaButtonClient mediaItem={item} />
265
- </DropdownMenuContent>
266
- </DropdownMenu>
267
- </div>
268
- </div>
269
- </div>
270
- ))}
271
- </div>
272
- )}
213
+ <div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 lg:grid-cols-5 xl:grid-cols-6 gap-4 mt-6">
214
+ {mediaItems.map((item) => {
215
+ const imageUrl = resolveMediaUrl(item.file_path || item.object_key, r2BaseUrl);
216
+
217
+ return (
218
+ <div
219
+ key={item.id}
220
+ className={`group relative border rounded-lg overflow-hidden shadow-sm aspect-square bg-muted/20 transition-all
221
+ ${isSelected(item.id) ? "ring-2 ring-primary ring-offset-2" : ""}`}
222
+ >
223
+ <div className="absolute top-2 left-2 z-10">
224
+ <Checkbox
225
+ id={`select-${item.id}`}
226
+ checked={isSelected(item.id)}
227
+ onCheckedChange={(checked) => {
228
+ handleSelectionChange(item.id, item.object_key, !!checked);
229
+ }}
230
+ aria-label={`Select ${item.file_name}`}
231
+ className="bg-white/70 hover:bg-white border-slate-400 data-[state=checked]:bg-primary data-[state=checked]:text-primary-foreground"
232
+ />
233
+ </div>
234
+
235
+ {item.file_type?.startsWith("image/") && imageUrl ? (
236
+ <MediaImage
237
+ src={imageUrl}
238
+ alt={item.description || item.file_name}
239
+ width={item.width || 500} // Default width if null
240
+ height={item.height || 500} // Default height if null
241
+ blurDataURL={item.blur_data_url || undefined}
242
+ className="h-full w-full object-contain transition-transform group-hover:scale-105"
243
+ />
244
+ ) : (
245
+ <div className="h-full w-full bg-muted flex flex-col items-center justify-center p-2">
246
+ <FileText className="h-12 w-12 text-muted-foreground mb-2" />
247
+ <p className="text-xs text-center text-muted-foreground truncate w-full" title={item.file_name}>
248
+ {item.file_name}
249
+ </p>
250
+ </div>
251
+ )}
252
+ <div className="absolute inset-0 bg-gradient-to-t from-black/50 via-black/20 to-transparent opacity-0 group-hover:opacity-100 transition-opacity flex flex-col justify-between p-2">
253
+ <div className="text-xs text-black truncate pt-1 ps-5" title={item.file_name}>{item.file_name}</div>
254
+ <div className="self-end">
255
+ <DropdownMenu>
256
+ <DropdownMenuTrigger asChild>
257
+ <Button variant="secondary" size="icon" className="text-white bg-black/40 hover:bg-black/60 h-7 w-7 rounded-full">
258
+ <MoreHorizontal className="h-4 w-4" />
259
+ </Button>
260
+ </DropdownMenuTrigger>
261
+ <DropdownMenuContent align="end">
262
+ <DropdownMenuItem asChild>
263
+ <Link href={`/cms/media/${item.id}/edit`} className="flex items-center cursor-pointer">
264
+ <Edit3 className="mr-2 h-4 w-4" /> Edit Details
265
+ </Link>
266
+ </DropdownMenuItem>
267
+ <DropdownMenuSeparator />
268
+ <DeleteMediaButtonClient mediaItem={item} />
269
+ </DropdownMenuContent>
270
+ </DropdownMenu>
271
+ </div>
272
+ </div>
273
+ </div>
274
+ );
275
+ })}
276
+ </div>
277
+ )}
273
278
  </div>
274
279
  );
275
280
  }