nextworks 0.0.1 → 0.1.0-alpha.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 (277) hide show
  1. package/README.md +209 -30
  2. package/dist/.gitkeep +0 -0
  3. package/dist/cli_manifests/auth_manifest.json +86 -0
  4. package/dist/cli_manifests/blocks_manifest.json +185 -0
  5. package/dist/cli_manifests/data_manifest.json +51 -0
  6. package/dist/cli_manifests/forms_manifest.json +61 -0
  7. package/dist/commands/admin-posts.d.ts +2 -0
  8. package/dist/commands/admin-posts.d.ts.map +1 -0
  9. package/dist/commands/admin-posts.js +15 -0
  10. package/dist/commands/admin-posts.js.map +1 -0
  11. package/dist/commands/admin-users.d.ts +2 -0
  12. package/dist/commands/admin-users.d.ts.map +1 -0
  13. package/dist/commands/admin-users.js +15 -0
  14. package/dist/commands/admin-users.js.map +1 -0
  15. package/dist/commands/auth-core.d.ts +2 -0
  16. package/dist/commands/auth-core.d.ts.map +1 -0
  17. package/dist/commands/auth-core.js +83 -0
  18. package/dist/commands/auth-core.js.map +1 -0
  19. package/dist/commands/auth-forms.d.ts +2 -0
  20. package/dist/commands/auth-forms.d.ts.map +1 -0
  21. package/dist/commands/auth-forms.js +15 -0
  22. package/dist/commands/auth-forms.js.map +1 -0
  23. package/dist/commands/blocks-options.d.ts +7 -0
  24. package/dist/commands/blocks-options.d.ts.map +1 -0
  25. package/dist/commands/blocks-options.js +19 -0
  26. package/dist/commands/blocks-options.js.map +1 -0
  27. package/dist/commands/blocks.d.ts +7 -0
  28. package/dist/commands/blocks.d.ts.map +1 -0
  29. package/dist/commands/blocks.js +140 -0
  30. package/dist/commands/blocks.js.map +1 -0
  31. package/dist/commands/data.d.ts +3 -0
  32. package/dist/commands/data.d.ts.map +1 -0
  33. package/dist/commands/data.js +88 -0
  34. package/dist/commands/data.js.map +1 -0
  35. package/dist/commands/forms.d.ts +6 -0
  36. package/dist/commands/forms.d.ts.map +1 -0
  37. package/dist/commands/forms.js +107 -0
  38. package/dist/commands/forms.js.map +1 -0
  39. package/dist/commands/remove-auth-core.d.ts +2 -0
  40. package/dist/commands/remove-auth-core.d.ts.map +1 -0
  41. package/dist/commands/remove-auth-core.js +69 -0
  42. package/dist/commands/remove-auth-core.js.map +1 -0
  43. package/dist/commands/remove-blocks.d.ts +2 -0
  44. package/dist/commands/remove-blocks.d.ts.map +1 -0
  45. package/dist/commands/remove-blocks.js +36 -0
  46. package/dist/commands/remove-blocks.js.map +1 -0
  47. package/dist/index.d.ts +3 -0
  48. package/dist/index.d.ts.map +1 -0
  49. package/dist/index.js +109 -0
  50. package/dist/index.js.map +1 -0
  51. package/dist/kits/auth-core/README.md +82 -0
  52. package/dist/kits/auth-core/app/(protected)/dashboard/page.tsx +8 -0
  53. package/dist/kits/auth-core/app/(protected)/layout.tsx +18 -0
  54. package/dist/kits/auth-core/app/(protected)/settings/profile/page.tsx +15 -0
  55. package/dist/kits/auth-core/app/(protected)/settings/profile/profile-form.tsx +114 -0
  56. package/dist/kits/auth-core/app/api/auth/[...nextauth]/route.ts +1 -0
  57. package/dist/kits/auth-core/app/api/auth/forgot-password/route.ts +114 -0
  58. package/dist/kits/auth-core/app/api/auth/providers/route.ts +6 -0
  59. package/dist/kits/auth-core/app/api/auth/reset-password/route.ts +63 -0
  60. package/dist/kits/auth-core/app/api/auth/send-verify-email/route.ts +6 -0
  61. package/dist/kits/auth-core/app/api/signup/route.ts +41 -0
  62. package/dist/kits/auth-core/app/auth/forgot-password/page.tsx +21 -0
  63. package/dist/kits/auth-core/app/auth/login/page.tsx +5 -0
  64. package/dist/kits/auth-core/app/auth/reset-password/page.tsx +187 -0
  65. package/dist/kits/auth-core/app/auth/signup/page.tsx +5 -0
  66. package/dist/kits/auth-core/app/auth/verify-email/page.tsx +11 -0
  67. package/dist/kits/auth-core/components/admin/admin-header.tsx +57 -0
  68. package/dist/kits/auth-core/components/auth/dashboard.tsx +237 -0
  69. package/dist/kits/auth-core/components/auth/forgot-password-form.tsx +90 -0
  70. package/dist/kits/auth-core/components/auth/login-form.tsx +467 -0
  71. package/dist/kits/auth-core/components/auth/logout-button.tsx +50 -0
  72. package/dist/kits/auth-core/components/auth/minimal-logout-button.tsx +40 -0
  73. package/dist/kits/auth-core/components/auth/signup-form.tsx +468 -0
  74. package/dist/kits/auth-core/components/require-auth.tsx +59 -0
  75. package/dist/kits/auth-core/components/session-provider.tsx +11 -0
  76. package/dist/kits/auth-core/components/ui/README.txt +1 -0
  77. package/dist/kits/auth-core/components/ui/button.tsx +55 -0
  78. package/dist/kits/auth-core/components/ui/input.tsx +25 -0
  79. package/dist/kits/auth-core/components/ui/label.tsx +23 -0
  80. package/dist/kits/auth-core/lib/api/errors.ts +14 -0
  81. package/dist/kits/auth-core/lib/auth-helpers.ts +29 -0
  82. package/dist/kits/auth-core/lib/auth.ts +142 -0
  83. package/dist/kits/auth-core/lib/email/dev-transport.ts +42 -0
  84. package/dist/kits/auth-core/lib/email/index.ts +28 -0
  85. package/dist/kits/auth-core/lib/email/provider-smtp.ts +36 -0
  86. package/dist/kits/auth-core/lib/forms/map-errors.ts +11 -0
  87. package/dist/kits/auth-core/lib/hash.ts +6 -0
  88. package/dist/kits/auth-core/lib/prisma.ts +15 -0
  89. package/dist/kits/auth-core/lib/server/result.ts +45 -0
  90. package/dist/kits/auth-core/lib/utils.ts +6 -0
  91. package/dist/kits/auth-core/lib/validation/forms.ts +88 -0
  92. package/dist/kits/auth-core/package-deps.json +19 -0
  93. package/dist/kits/auth-core/prisma/auth-models.prisma +81 -0
  94. package/dist/kits/auth-core/prisma/schema.prisma +81 -0
  95. package/dist/kits/auth-core/scripts/populate-tokenhash.mjs +26 -0
  96. package/dist/kits/auth-core/scripts/promote-admin.mjs +33 -0
  97. package/dist/kits/auth-core/scripts/seed-demo.mjs +40 -0
  98. package/dist/kits/auth-core/types/next-auth.d.ts +25 -0
  99. package/dist/kits/blocks/README.md +53 -0
  100. package/dist/kits/blocks/app/globals.css +175 -0
  101. package/dist/kits/blocks/app/templates/digitalagency/PresetThemeVars.tsx +80 -0
  102. package/dist/kits/blocks/app/templates/digitalagency/README.md +36 -0
  103. package/dist/kits/blocks/app/templates/digitalagency/components/About.tsx +99 -0
  104. package/dist/kits/blocks/app/templates/digitalagency/components/CTA.tsx +74 -0
  105. package/dist/kits/blocks/app/templates/digitalagency/components/Contact.tsx +227 -0
  106. package/dist/kits/blocks/app/templates/digitalagency/components/Footer.tsx +89 -0
  107. package/dist/kits/blocks/app/templates/digitalagency/components/Hero.tsx +90 -0
  108. package/dist/kits/blocks/app/templates/digitalagency/components/Navbar.tsx +168 -0
  109. package/dist/kits/blocks/app/templates/digitalagency/components/NetworkPattern.tsx +297 -0
  110. package/dist/kits/blocks/app/templates/digitalagency/components/Portfolio.tsx +157 -0
  111. package/dist/kits/blocks/app/templates/digitalagency/components/Pricing.tsx +114 -0
  112. package/dist/kits/blocks/app/templates/digitalagency/components/Process.tsx +59 -0
  113. package/dist/kits/blocks/app/templates/digitalagency/components/Services.tsx +55 -0
  114. package/dist/kits/blocks/app/templates/digitalagency/components/Team.tsx +28 -0
  115. package/dist/kits/blocks/app/templates/digitalagency/components/Testimonials.tsx +65 -0
  116. package/dist/kits/blocks/app/templates/digitalagency/page.tsx +38 -0
  117. package/dist/kits/blocks/app/templates/gallery/PresetThemeVars.tsx +85 -0
  118. package/dist/kits/blocks/app/templates/gallery/page.tsx +303 -0
  119. package/dist/kits/blocks/app/templates/productlaunch/PresetThemeVars.tsx +74 -0
  120. package/dist/kits/blocks/app/templates/productlaunch/README.md +55 -0
  121. package/dist/kits/blocks/app/templates/productlaunch/components/About.tsx +178 -0
  122. package/dist/kits/blocks/app/templates/productlaunch/components/CTA.tsx +93 -0
  123. package/dist/kits/blocks/app/templates/productlaunch/components/Contact.tsx +231 -0
  124. package/dist/kits/blocks/app/templates/productlaunch/components/FAQ.tsx +93 -0
  125. package/dist/kits/blocks/app/templates/productlaunch/components/Features.tsx +84 -0
  126. package/dist/kits/blocks/app/templates/productlaunch/components/Footer.tsx +132 -0
  127. package/dist/kits/blocks/app/templates/productlaunch/components/Hero.tsx +89 -0
  128. package/dist/kits/blocks/app/templates/productlaunch/components/Navbar.tsx +162 -0
  129. package/dist/kits/blocks/app/templates/productlaunch/components/Pricing.tsx +106 -0
  130. package/dist/kits/blocks/app/templates/productlaunch/components/ProcessTimeline.tsx +110 -0
  131. package/dist/kits/blocks/app/templates/productlaunch/components/ServicesGrid.tsx +68 -0
  132. package/dist/kits/blocks/app/templates/productlaunch/components/Team.tsx +104 -0
  133. package/dist/kits/blocks/app/templates/productlaunch/components/Testimonials.tsx +89 -0
  134. package/dist/kits/blocks/app/templates/productlaunch/components/TrustBadges.tsx +76 -0
  135. package/dist/kits/blocks/app/templates/productlaunch/page.tsx +45 -0
  136. package/dist/kits/blocks/app/templates/saasdashboard/PresetThemeVars.tsx +80 -0
  137. package/dist/kits/blocks/app/templates/saasdashboard/README.md +38 -0
  138. package/dist/kits/blocks/app/templates/saasdashboard/components/Contact.tsx +176 -0
  139. package/dist/kits/blocks/app/templates/saasdashboard/components/Dashboard.tsx +293 -0
  140. package/dist/kits/blocks/app/templates/saasdashboard/components/FAQ.tsx +55 -0
  141. package/dist/kits/blocks/app/templates/saasdashboard/components/Features.tsx +91 -0
  142. package/dist/kits/blocks/app/templates/saasdashboard/components/Footer.tsx +77 -0
  143. package/dist/kits/blocks/app/templates/saasdashboard/components/Hero.tsx +105 -0
  144. package/dist/kits/blocks/app/templates/saasdashboard/components/Hero_mask.tsx +127 -0
  145. package/dist/kits/blocks/app/templates/saasdashboard/components/Navbar.tsx +159 -0
  146. package/dist/kits/blocks/app/templates/saasdashboard/components/Pricing.tsx +90 -0
  147. package/dist/kits/blocks/app/templates/saasdashboard/components/SmoothScroll.tsx +97 -0
  148. package/dist/kits/blocks/app/templates/saasdashboard/components/Testimonials.tsx +72 -0
  149. package/dist/kits/blocks/app/templates/saasdashboard/components/TrustBadges.tsx +53 -0
  150. package/dist/kits/blocks/app/templates/saasdashboard/page.tsx +39 -0
  151. package/dist/kits/blocks/components/app-providers.tsx +1 -0
  152. package/dist/kits/blocks/components/enhanced-theme-provider.tsx +195 -0
  153. package/dist/kits/blocks/components/sections/About.tsx +291 -0
  154. package/dist/kits/blocks/components/sections/CTA.tsx +258 -0
  155. package/dist/kits/blocks/components/sections/Contact.tsx +267 -0
  156. package/dist/kits/blocks/components/sections/FAQ.tsx +226 -0
  157. package/dist/kits/blocks/components/sections/Features.tsx +269 -0
  158. package/dist/kits/blocks/components/sections/Footer.tsx +302 -0
  159. package/dist/kits/blocks/components/sections/HeroMotion.tsx +307 -0
  160. package/dist/kits/blocks/components/sections/HeroOverlay.tsx +358 -0
  161. package/dist/kits/blocks/components/sections/HeroSplit.tsx +352 -0
  162. package/dist/kits/blocks/components/sections/Navbar.tsx +353 -0
  163. package/dist/kits/blocks/components/sections/Newsletter.tsx +156 -0
  164. package/dist/kits/blocks/components/sections/PortfolioSimple.tsx +550 -0
  165. package/dist/kits/blocks/components/sections/Pricing.tsx +264 -0
  166. package/dist/kits/blocks/components/sections/ProcessTimeline.tsx +325 -0
  167. package/dist/kits/blocks/components/sections/ServicesGrid.tsx +210 -0
  168. package/dist/kits/blocks/components/sections/Team.tsx +309 -0
  169. package/dist/kits/blocks/components/sections/Testimonials.tsx +158 -0
  170. package/dist/kits/blocks/components/sections/TrustBadges.tsx +162 -0
  171. package/dist/kits/blocks/components/theme-provider.tsx +34 -0
  172. package/dist/kits/blocks/components/ui/alert-dialog.tsx +134 -0
  173. package/dist/kits/blocks/components/ui/brand-node.tsx +121 -0
  174. package/dist/kits/blocks/components/ui/button.tsx +122 -0
  175. package/dist/kits/blocks/components/ui/button_bck.tsx +93 -0
  176. package/dist/kits/blocks/components/ui/card.tsx +95 -0
  177. package/dist/kits/blocks/components/ui/checkbox.tsx +30 -0
  178. package/dist/kits/blocks/components/ui/cta-button.tsx +125 -0
  179. package/dist/kits/blocks/components/ui/dropdown-menu.tsx +201 -0
  180. package/dist/kits/blocks/components/ui/feature-card.tsx +91 -0
  181. package/dist/kits/blocks/components/ui/input.tsx +27 -0
  182. package/dist/kits/blocks/components/ui/label.tsx +29 -0
  183. package/dist/kits/blocks/components/ui/pricing-card.tsx +120 -0
  184. package/dist/kits/blocks/components/ui/select.tsx +25 -0
  185. package/dist/kits/blocks/components/ui/skeleton.tsx +13 -0
  186. package/dist/kits/blocks/components/ui/switch.tsx +78 -0
  187. package/dist/kits/blocks/components/ui/table.tsx +98 -0
  188. package/dist/kits/blocks/components/ui/testimonial-card.tsx +108 -0
  189. package/dist/kits/blocks/components/ui/textarea.tsx +26 -0
  190. package/dist/kits/blocks/components/ui/theme-selector.tsx +247 -0
  191. package/dist/kits/blocks/components/ui/theme-toggle.tsx +74 -0
  192. package/dist/kits/blocks/components/ui/toaster.tsx +7 -0
  193. package/dist/kits/blocks/lib/themes.ts +399 -0
  194. package/dist/kits/blocks/lib/themes_old.ts +37 -0
  195. package/dist/kits/blocks/lib/utils.ts +9 -0
  196. package/dist/kits/blocks/next.config.ts +11 -0
  197. package/dist/kits/blocks/notes/THEME_GUIDE.md +29 -0
  198. package/dist/kits/blocks/notes/THEMING_CONVERSION_SUMMARY.md +14 -0
  199. package/dist/kits/blocks/package-deps.json +22 -0
  200. package/dist/kits/blocks/public/placeholders/gallery/hero-pexels-broken-9945014.avif +0 -0
  201. package/dist/kits/blocks/public/placeholders/gallery/pexels-googledeepmind-25626431.jpg +0 -0
  202. package/dist/kits/blocks/public/placeholders/gallery/pexels-googledeepmind-25626432.jpg +0 -0
  203. package/dist/kits/blocks/public/placeholders/gallery/pexels-googledeepmind-25626434.jpg +0 -0
  204. package/dist/kits/blocks/public/placeholders/gallery/pexels-googledeepmind-25626436.jpg +0 -0
  205. package/dist/kits/blocks/public/placeholders/product_launch/feature_1.png +0 -0
  206. package/dist/kits/blocks/public/placeholders/product_launch/feature_2.png +0 -0
  207. package/dist/kits/blocks/public/placeholders/product_launch/feature_3.png +0 -0
  208. package/dist/kits/blocks/public/placeholders/product_launch/feature_4.png +0 -0
  209. package/dist/kits/blocks/public/placeholders/product_launch/hero.png +0 -0
  210. package/dist/kits/blocks/public/placeholders/saas_dashboard/analytics.png +0 -0
  211. package/dist/kits/blocks/public/placeholders/saas_dashboard/chat.png +0 -0
  212. package/dist/kits/blocks/public/placeholders/saas_dashboard/projectBoard.png +0 -0
  213. package/dist/kits/data/.gitkeep +0 -0
  214. package/dist/kits/data/README.md +80 -0
  215. package/dist/kits/data/app/(protected)/admin/posts/page.tsx +5 -0
  216. package/dist/kits/data/app/(protected)/admin/users/page.tsx +5 -0
  217. package/dist/kits/data/app/api/posts/[id]/route.ts +83 -0
  218. package/dist/kits/data/app/api/posts/route.ts +138 -0
  219. package/dist/kits/data/app/api/seed-demo/route.ts +45 -0
  220. package/dist/kits/data/app/api/users/[id]/route.ts +127 -0
  221. package/dist/kits/data/app/api/users/check-email/route.ts +18 -0
  222. package/dist/kits/data/app/api/users/check-unique/route.ts +27 -0
  223. package/dist/kits/data/app/api/users/route.ts +79 -0
  224. package/dist/kits/data/app/examples/demo/README.md +4 -0
  225. package/dist/kits/data/app/examples/demo/create-post-form.tsx +106 -0
  226. package/dist/kits/data/app/examples/demo/page.tsx +118 -0
  227. package/dist/kits/data/app/examples/demo/seed-demo-button.tsx +37 -0
  228. package/dist/kits/data/components/admin/posts-manager.tsx +719 -0
  229. package/dist/kits/data/components/admin/users-manager.tsx +432 -0
  230. package/dist/kits/data/lib/prisma.ts +15 -0
  231. package/dist/kits/data/lib/server/result.ts +90 -0
  232. package/dist/kits/data/package-deps.json +11 -0
  233. package/dist/kits/data/scripts/seed-demo.mjs +41 -0
  234. package/dist/kits/forms/.gitkeep +0 -0
  235. package/dist/kits/forms/README.md +49 -0
  236. package/dist/kits/forms/app/.gitkeep +0 -0
  237. package/dist/kits/forms/app/api/wizard/route.ts +71 -0
  238. package/dist/kits/forms/app/examples/forms/basic/page.tsx +124 -0
  239. package/dist/kits/forms/app/examples/forms/server-action/form-client.tsx +28 -0
  240. package/dist/kits/forms/app/examples/forms/server-action/page.tsx +71 -0
  241. package/dist/kits/forms/app/examples/forms/wizard/page.tsx +15 -0
  242. package/dist/kits/forms/app/examples/forms/wizard/wizard-client.tsx +2 -0
  243. package/dist/kits/forms/components/.gitkeep +0 -0
  244. package/dist/kits/forms/components/examples/wizard-client.tsx +231 -0
  245. package/dist/kits/forms/components/hooks/useCheckUnique.ts +79 -0
  246. package/dist/kits/forms/components/ui/button.tsx +122 -0
  247. package/dist/kits/forms/components/ui/checkbox.tsx +30 -0
  248. package/dist/kits/forms/components/ui/form/context.ts +33 -0
  249. package/dist/kits/forms/components/ui/form/form-control.tsx +28 -0
  250. package/dist/kits/forms/components/ui/form/form-description.tsx +22 -0
  251. package/dist/kits/forms/components/ui/form/form-field.tsx +36 -0
  252. package/dist/kits/forms/components/ui/form/form-item.tsx +21 -0
  253. package/dist/kits/forms/components/ui/form/form-label.tsx +24 -0
  254. package/dist/kits/forms/components/ui/form/form-message.tsx +29 -0
  255. package/dist/kits/forms/components/ui/form/form.tsx +26 -0
  256. package/dist/kits/forms/components/ui/input.tsx +27 -0
  257. package/dist/kits/forms/components/ui/label.tsx +29 -0
  258. package/dist/kits/forms/components/ui/select.tsx +25 -0
  259. package/dist/kits/forms/components/ui/switch.tsx +78 -0
  260. package/dist/kits/forms/components/ui/textarea.tsx +26 -0
  261. package/dist/kits/forms/lib/.gitkeep +0 -0
  262. package/dist/kits/forms/lib/forms/map-errors.ts +29 -0
  263. package/dist/kits/forms/lib/prisma.ts +16 -0
  264. package/dist/kits/forms/lib/utils.ts +9 -0
  265. package/dist/kits/forms/lib/validation/forms.ts +88 -0
  266. package/dist/kits/forms/lib/validation/wizard.ts +32 -0
  267. package/dist/kits/forms/package-deps.json +17 -0
  268. package/dist/utils/file-operations.d.ts +18 -0
  269. package/dist/utils/file-operations.d.ts.map +1 -0
  270. package/dist/utils/file-operations.js +327 -0
  271. package/dist/utils/file-operations.js.map +1 -0
  272. package/dist/utils/installation-tracker.d.ts +26 -0
  273. package/dist/utils/installation-tracker.d.ts.map +1 -0
  274. package/dist/utils/installation-tracker.js +98 -0
  275. package/dist/utils/installation-tracker.js.map +1 -0
  276. package/package.json +51 -21
  277. package/index.js +0 -1
@@ -0,0 +1,358 @@
1
+ "use client";
2
+
3
+ import React from "react";
4
+ import Image from "next/image";
5
+ import Link from "next/link";
6
+ import { Button } from "@/components/ui/button";
7
+ import { cn } from "@/lib/utils";
8
+
9
+ /**
10
+ * Props for the HeroOverlay component.
11
+ *
12
+ * @remarks
13
+ * - Prefer the structured image prop (image) over deprecated flat props.
14
+ * - Slot-style className overrides are merged after defaults via cn().
15
+ * - Motion controls affect CTA hover transitions.
16
+ *
17
+ * @public
18
+ */
19
+ export interface HeroOverlayProps {
20
+ id?: string;
21
+ className?: string;
22
+
23
+ // New structured image props (aligns with HeroSplit)
24
+ image?: {
25
+ src?: string;
26
+ alt?: string;
27
+ className?: string;
28
+ objectFit?: React.CSSProperties["objectFit"];
29
+ objectPosition?: React.CSSProperties["objectPosition"];
30
+ };
31
+
32
+ // Deprecated flat image props (kept for backward compatibility)
33
+ /** @deprecated Use image.src instead */
34
+ imageSrc?: string;
35
+ /** @deprecated Use image.alt instead */
36
+ imageAlt?: string;
37
+ /** @deprecated Use image.objectFit instead */
38
+ imageObjectFit?: React.CSSProperties["objectFit"];
39
+ /** @deprecated Use image.objectPosition instead */
40
+ imageObjectPosition?: React.CSSProperties["objectPosition"];
41
+
42
+ // Allow string or object form like HeroSplit
43
+ heading?: string | { text?: string; className?: string };
44
+ subheading?: string | { text?: string; className?: string };
45
+ overlayRGBA?: string;
46
+ overlayDarkRGBA?: string;
47
+
48
+ // Alias to match HeroSplit naming
49
+ section?: { className?: string };
50
+ container?: { className?: string };
51
+ imageContainer?: { className?: string };
52
+ overlay?: { className?: string };
53
+ content?: { className?: string };
54
+ textContainer?: { className?: string };
55
+
56
+ // Retained styles; merged with heading/subheading.className when provided
57
+ /** @deprecated Use heading.className */
58
+ headingStyle?: { className?: string };
59
+ /** @deprecated Use subheading.className */
60
+ subheadingStyle?: { className?: string };
61
+
62
+ ctaContainer?: { className?: string };
63
+
64
+ // CTA objects now support label/href (aligns with HeroSplit)
65
+ cta1?: {
66
+ label?: string;
67
+ href?: string;
68
+ variant?:
69
+ | "default"
70
+ | "destructive"
71
+ | "outline"
72
+ | "secondary"
73
+ | "ghost"
74
+ | "link";
75
+ size?: "default" | "sm" | "lg" | "icon";
76
+ className?: string;
77
+ unstyled?: boolean;
78
+ style?: React.CSSProperties;
79
+ };
80
+ cta2?: {
81
+ label?: string;
82
+ href?: string;
83
+ variant?:
84
+ | "default"
85
+ | "destructive"
86
+ | "outline"
87
+ | "secondary"
88
+ | "ghost"
89
+ | "link";
90
+ size?: "default" | "sm" | "lg" | "icon";
91
+ className?: string;
92
+ unstyled?: boolean;
93
+ style?: React.CSSProperties;
94
+ };
95
+
96
+ // Deprecated CTA props (kept for backward compatibility)
97
+ /** @deprecated Use cta1.label instead */
98
+ ctaBtn1Label?: string;
99
+ /** @deprecated Use cta1.href instead */
100
+ ctaBtn1Href?: string;
101
+ /** @deprecated Use cta2.label instead */
102
+ ctaBtn2Label?: string;
103
+ /** @deprecated Use cta2.href instead */
104
+ ctaBtn2Href?: string;
105
+
106
+ ariaLabel?: string;
107
+
108
+ // Motion control for CTA buttons
109
+ enableMotion?: boolean;
110
+ }
111
+
112
+ /**
113
+ * Full-bleed image hero with color overlay, heading/subheading, and up to two CTAs.
114
+ *
115
+ * @remarks
116
+ * - Styling: slot-style overrides for container, imageContainer, overlay, content, textContainer.
117
+ * - Motion: enableMotion applies hover-lift transitions to CTA buttons only.
118
+ * - Accessibility: uses semantic <section> with aria-label; background image is
119
+ * decorative with overlay and text content provided separately.
120
+ *
121
+ * @example
122
+ * <HeroOverlay
123
+ * image={{ src: "/hero.png", alt: "" }}
124
+ * heading="Welcome"
125
+ * subheading="Build faster with our blocks"
126
+ * cta1={{ label: "Get Started", href: "#getstarted" }}
127
+ * />
128
+ */
129
+ export function HeroOverlay({
130
+ id,
131
+ className,
132
+
133
+ // Deprecated flat image defaults (still supported)
134
+ imageSrc = "/hero.png",
135
+ imageAlt = "Hero background image",
136
+ imageObjectFit = "cover",
137
+ imageObjectPosition = "center",
138
+
139
+ // New structured image (optional)
140
+ image,
141
+
142
+ // Keep string defaults; object form supported
143
+ heading = "Lorem ipsum dolor sit amet, consectetur",
144
+ subheading = "Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur.",
145
+ overlayRGBA = "rgba(255, 255, 255, 0.8)",
146
+ overlayDarkRGBA = "rgba(0, 0, 0, 0.8)",
147
+
148
+ // Deprecated CTA label/href (still supported)
149
+ ctaBtn1Label = "Get Started",
150
+ ctaBtn1Href = "#getstarted",
151
+ ctaBtn2Label,
152
+ ctaBtn2Href,
153
+
154
+ // Layout containers
155
+ container = { className: "relative h-[440px] md:h-[500px] w-full" },
156
+ // container = { className: "relative h-[500px] md:h-[600px] w-full" },
157
+ section,
158
+ imageContainer = { className: "absolute inset-0" },
159
+ overlay = { className: "absolute inset-0 z-10 pointer-events-none" },
160
+ content = {
161
+ className:
162
+ "relative z-20 h-full w-full flex flex-col justify-start items-center pt-14 pb-8 px-8",
163
+ },
164
+ textContainer = { className: "max-w-4xl space-y-6 text-center" },
165
+
166
+ // Typography defaults (deprecated props still merged)
167
+ headingStyle = { className: "" },
168
+ subheadingStyle = { className: "" },
169
+
170
+ // CTA containers and variants
171
+ ctaContainer = {
172
+ className:
173
+ "flex flex-col sm:flex-row gap-4 mt-6 justify-center items-center",
174
+ },
175
+ cta1,
176
+ cta2,
177
+
178
+ ariaLabel = "Hero section",
179
+ enableMotion = true,
180
+ }: HeroOverlayProps) {
181
+ // Merge new structured image with deprecated flat props
182
+ const effImage = {
183
+ src: image?.src ?? imageSrc,
184
+ alt: image?.alt ?? imageAlt,
185
+ objectFit: image?.objectFit ?? imageObjectFit,
186
+ objectPosition: image?.objectPosition ?? imageObjectPosition,
187
+ className: image?.className,
188
+ };
189
+
190
+ // Defaults and normalization like HeroSplit
191
+ const defaultHeading = {
192
+ text: "Lorem ipsum dolor sit amet, consectetur",
193
+ className:
194
+ "text-4xl md:text-5xl font-bold text-center leading-tight text-foreground",
195
+ };
196
+ const defaultSubheading = {
197
+ text: "Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur.",
198
+ className:
199
+ "text-xl md:text-2xl text-center text-foreground leading-relaxed",
200
+ };
201
+
202
+ const normalizedHeading =
203
+ typeof heading === "string"
204
+ ? {
205
+ text: heading,
206
+ className: cn(defaultHeading.className, headingStyle?.className),
207
+ }
208
+ : {
209
+ text: heading?.text ?? defaultHeading.text,
210
+ className: cn(
211
+ defaultHeading.className,
212
+ headingStyle?.className,
213
+ heading?.className,
214
+ ),
215
+ };
216
+
217
+ const normalizedSubheading =
218
+ typeof subheading === "string"
219
+ ? {
220
+ text: subheading,
221
+ className: cn(
222
+ defaultSubheading.className,
223
+ subheadingStyle?.className,
224
+ ),
225
+ }
226
+ : {
227
+ text: subheading?.text ?? defaultSubheading.text,
228
+ className: cn(
229
+ defaultSubheading.className,
230
+ subheadingStyle?.className,
231
+ subheading?.className,
232
+ ),
233
+ };
234
+
235
+ // Motion class for CTA buttons
236
+ const buttonLift = enableMotion
237
+ ? "transition-all duration-200 hover:-translate-y-0.5"
238
+ : "transition-none hover:!translate-y-0";
239
+
240
+ // CTA defaults and normalization (merge, don't replace).
241
+ const defaultCta1 = {
242
+ label: ctaBtn1Label,
243
+ href: ctaBtn1Href,
244
+ variant: "default" as const,
245
+ size: "lg" as const,
246
+ className: "",
247
+ };
248
+ const defaultCta2 = {
249
+ label: ctaBtn2Label,
250
+ href: ctaBtn2Href,
251
+ variant: "outline" as const,
252
+ size: "lg" as const,
253
+ className: "",
254
+ };
255
+
256
+ const mergedCta1 = {
257
+ ...defaultCta1,
258
+ ...(cta1 ?? {}),
259
+ className: cn(defaultCta1.className, cta1?.className, buttonLift),
260
+ };
261
+ const mergedCta2 =
262
+ cta2 || ctaBtn2Label || ctaBtn2Href
263
+ ? {
264
+ ...defaultCta2,
265
+ ...(cta2 ?? {}),
266
+ className: cn(defaultCta2.className, cta2?.className, buttonLift),
267
+ }
268
+ : undefined;
269
+
270
+ // buttonLift defined above
271
+
272
+ // Merge alias section.className with container.className
273
+ const finalContainerClass = cn(container.className, section?.className);
274
+
275
+ const showCta2 = !!mergedCta2?.label;
276
+
277
+ return (
278
+ <section
279
+ id={id}
280
+ className={cn(finalContainerClass, className)}
281
+ aria-label={ariaLabel}
282
+ >
283
+ <div className={imageContainer.className}>
284
+ <Image
285
+ src={effImage.src!}
286
+ alt={effImage.alt!}
287
+ fill
288
+ sizes="100vw"
289
+ quality={75}
290
+ priority
291
+ className={effImage.className}
292
+ style={{
293
+ objectFit: effImage.objectFit,
294
+ objectPosition: effImage.objectPosition,
295
+ }}
296
+ />
297
+ </div>
298
+
299
+ <div
300
+ className={overlay.className}
301
+ style={{ backgroundColor: overlayRGBA }}
302
+ aria-hidden
303
+ />
304
+ <div
305
+ className={cn(overlay.className, "hidden dark:block")}
306
+ style={{ backgroundColor: overlayDarkRGBA }}
307
+ aria-hidden
308
+ />
309
+
310
+ <div className={content.className}>
311
+ <div className={textContainer.className}>
312
+ <h1 className={normalizedHeading.className}>
313
+ {normalizedHeading.text}
314
+ </h1>
315
+ <p className={normalizedSubheading.className}>
316
+ {normalizedSubheading.text}
317
+ </p>
318
+ <div className={ctaContainer.className}>
319
+ {mergedCta1.label && (
320
+ <Button
321
+ asChild
322
+ variant={mergedCta1.variant}
323
+ size={mergedCta1.size}
324
+ className={mergedCta1.className}
325
+ unstyled={mergedCta1.unstyled}
326
+ style={mergedCta1.style}
327
+ >
328
+ <Link
329
+ href={mergedCta1.href || "#"}
330
+ aria-label={mergedCta1.label}
331
+ >
332
+ {mergedCta1.label}
333
+ </Link>
334
+ </Button>
335
+ )}
336
+ {showCta2 && (
337
+ <Button
338
+ asChild
339
+ variant={mergedCta2!.variant}
340
+ size={mergedCta2!.size}
341
+ className={mergedCta2!.className}
342
+ unstyled={mergedCta2!.unstyled}
343
+ style={mergedCta2!.style}
344
+ >
345
+ <Link
346
+ href={mergedCta2!.href || "#"}
347
+ aria-label={mergedCta2!.label}
348
+ >
349
+ {mergedCta2!.label}
350
+ </Link>
351
+ </Button>
352
+ )}
353
+ </div>
354
+ </div>
355
+ </div>
356
+ </section>
357
+ );
358
+ }
@@ -0,0 +1,352 @@
1
+ "use client";
2
+ import React from "react";
3
+ import Image from "next/image";
4
+ import Link from "next/link";
5
+ import { Button } from "@/components/ui/button";
6
+ import { cn } from "@/lib/utils";
7
+
8
+ /**
9
+ * Props for the HeroSplit component.
10
+ *
11
+ * @remarks
12
+ * - Layout: split hero with text and image. imageLayout controls whether the
13
+ * image is full-bleed or padded.
14
+ * - Styling: slot-style className overrides are merged via cn(). Heading and
15
+ * subheading accept either a string or a { text, className } object.
16
+ * - Motion: enableMotion toggles button hover lift transitions.
17
+ * - Accessibility: uses a semantic <section> with aria-label.
18
+ */
19
+ interface HeroSplitProps {
20
+ /** Optional id to attach to the root section element */
21
+ id?: string;
22
+ /** Optional top-level class to override the root */
23
+ className?: string;
24
+ /** Section slot override */
25
+ section?: { className?: string };
26
+ /** Heading string or object with text and className */
27
+ heading?: string | { text?: string; className?: string };
28
+ /** Subheading string or object with text and className */
29
+ subheading?: string | { text?: string; className?: string };
30
+ /** Primary CTA configuration */
31
+ cta1?: {
32
+ label?: string;
33
+ href?: string;
34
+ variant?:
35
+ | "default"
36
+ | "destructive"
37
+ | "outline"
38
+ | "secondary"
39
+ | "ghost"
40
+ | "link";
41
+ size?: "default" | "sm" | "lg" | "icon";
42
+ className?: string;
43
+ /** When true, renders the Button in unstyled mode to avoid token bleed from variants */
44
+ unstyled?: boolean;
45
+ /** Optional inline style forwarded to Button */
46
+ style?: React.CSSProperties;
47
+ };
48
+ /** Secondary CTA configuration */
49
+ cta2?: {
50
+ label?: string;
51
+ href?: string;
52
+ variant?:
53
+ | "default"
54
+ | "destructive"
55
+ | "outline"
56
+ | "secondary"
57
+ | "ghost"
58
+ | "link";
59
+ size?: "default" | "sm" | "lg" | "icon";
60
+ className?: string;
61
+ /** When true, renders the Button in unstyled mode to avoid token bleed from variants */
62
+ unstyled?: boolean;
63
+ /** Optional inline style forwarded to Button */
64
+ style?: React.CSSProperties;
65
+ };
66
+ /** Image configuration */
67
+ image?: { src?: string; alt?: string; className?: string };
68
+ /** Image container slot */
69
+ imageContainer?: { className?: string };
70
+ /** Text container slot */
71
+ textContainer?: { className?: string };
72
+ /** Buttons container slot */
73
+ buttonsContainer?: { className?: string };
74
+ /** Text alignment at large breakpoints. @defaultValue "center" */
75
+ textAlign?: "left" | "center" | "right";
76
+ /** Fallback node shown when no image.src is provided */
77
+ fallback?: React.ReactNode;
78
+ /** ARIA label for the section. @defaultValue "Hero section" */
79
+ ariaLabel?: string;
80
+ /** When false, removes button hover lifts, keeps layout static */
81
+ enableMotion?: boolean;
82
+ /** "padded" keeps default gutters. "full-bleed" allows the image to touch page edges. @defaultValue "full-bleed" */
83
+ imageLayout?: "padded" | "full-bleed";
84
+ }
85
+
86
+ /**
87
+ * Split hero with text on one side and an image on the other.
88
+ *
89
+ * @remarks
90
+ * - Text props accept string or object forms to easily override classes.
91
+ * - Full-bleed layout absolutely positions the image on larger screens.
92
+ * - Motion only affects the subtle hover lift on CTA buttons.
93
+ *
94
+ * @example
95
+ * <HeroSplit heading="Build faster" subheading="Launch confidently" />
96
+ */
97
+ export function HeroSplit({
98
+ id,
99
+ className,
100
+ heading,
101
+ subheading,
102
+ section,
103
+ cta1,
104
+ cta2,
105
+ image = {
106
+ src: "/placeholders/gallery/hero-pexels-broken-9945014.avif",
107
+ alt: "Hero image",
108
+ className: "object-contain",
109
+ },
110
+ imageContainer,
111
+ textContainer = { className: "flex-1 p-5" },
112
+ buttonsContainer = { className: "flex flex-col md:flex-row gap-4 mt-6" },
113
+ textAlign = "center",
114
+ fallback,
115
+ ariaLabel = "Hero section",
116
+ enableMotion = true,
117
+ imageLayout = "full-bleed",
118
+ }: HeroSplitProps) {
119
+ const lgTextAlignClasses = {
120
+ left: "lg:text-left",
121
+ right: "lg:text-right",
122
+ center: "lg:text-center",
123
+ } as const;
124
+ const lgItemsAlignClasses = {
125
+ left: "lg:items-start",
126
+ right: "lg:items-end",
127
+ center: "lg:items-center",
128
+ } as const;
129
+ const lgJustifyAlignClasses = {
130
+ left: "lg:justify-start",
131
+ right: "lg:justify-end",
132
+ center: "lg:justify-center",
133
+ } as const;
134
+ const buttonLift = enableMotion
135
+ ? "transition-all duration-200 hover:-translate-y-0.5"
136
+ : "transition-none hover:!translate-y-0";
137
+
138
+ // Defaults for text props (normalized below)
139
+ const defaultHeading = {
140
+ text: "Lorem ipsum dolor sit amet",
141
+ className: "text-3xl md:text-4xl font-bold leading-tight text-foreground",
142
+ };
143
+ const defaultSubheading = {
144
+ text: "Consectetur adipiscing elit, sed do eiusmod tempor.",
145
+ className: "text-lg md:text-xl text-foreground mt-4 mb-6",
146
+ };
147
+
148
+ const normalizedHeading =
149
+ typeof heading === "string"
150
+ ? { text: heading, className: defaultHeading.className }
151
+ : {
152
+ text: heading?.text ?? defaultHeading.text,
153
+ className: cn(defaultHeading.className, heading?.className),
154
+ };
155
+
156
+ const normalizedSubheading =
157
+ typeof subheading === "string"
158
+ ? { text: subheading, className: defaultSubheading.className }
159
+ : {
160
+ text: subheading?.text ?? defaultSubheading.text,
161
+ className: cn(defaultSubheading.className, subheading?.className),
162
+ };
163
+
164
+ // CTA defaults and normalization (merge, don't replace)
165
+ const defaultCta1 = {
166
+ label: "Get Started",
167
+ href: "#contact",
168
+ variant: "default" as const,
169
+ size: "default" as const,
170
+ className:
171
+ "shadow-lg hover:shadow-xl transition-all duration-200 hover:-translate-y-0.5",
172
+ };
173
+
174
+ const mergedCta1 = {
175
+ ...defaultCta1,
176
+ ...(cta1 ?? {}),
177
+ className: cn(defaultCta1.className, cta1?.className, buttonLift),
178
+ };
179
+
180
+ const defaultCta2 = {
181
+ label: "Learn More",
182
+ href: "#",
183
+ variant: "outline" as const,
184
+ size: "default" as const,
185
+ className: "",
186
+ };
187
+
188
+ const mergedCta2 = cta2
189
+ ? {
190
+ ...defaultCta2,
191
+ ...cta2,
192
+ className: cn(defaultCta2.className, cta2.className, buttonLift),
193
+ }
194
+ : undefined;
195
+
196
+ const defaultSectionPadded = cn(
197
+ "px-8 pt-8 pb-8 md:min-h-[60vh] lg:min-h-[70vh]",
198
+ );
199
+
200
+ const defaultSectionFull = cn(
201
+ "relative overflow-hidden pt-0 pb-0 md:min-h-[60vh] lg:min-h-[70vh]",
202
+ );
203
+
204
+ const defaultImageContainerPadded = cn(
205
+ "relative mb-4 h-48 min-h-[12rem] w-full md:mb-0 md:h-72 md:w-1/2",
206
+ );
207
+
208
+ // For full-bleed: absolute positioning on large screens to break out completely
209
+ const defaultImageContainerFull = cn(
210
+ "relative h-48 w-full md:absolute md:inset-y-0 md:right-0 md:h-full md:w-1/2",
211
+ );
212
+
213
+ const baseSectionClass =
214
+ imageLayout === "full-bleed" ? defaultSectionFull : defaultSectionPadded;
215
+
216
+ const finalSectionClass = cn(baseSectionClass, section?.className);
217
+
218
+ const baseImageContainerClass =
219
+ imageLayout === "full-bleed"
220
+ ? defaultImageContainerFull
221
+ : defaultImageContainerPadded;
222
+
223
+ const finalImageContainerClass = cn(
224
+ baseImageContainerClass,
225
+ imageContainer?.className,
226
+ );
227
+
228
+ // -- This overrides the image.className that is passed from importing components ... devs should be able to decide on cover/contain from the importing component.
229
+ const finalImageClass = cn(
230
+ image?.className,
231
+ imageLayout === "full-bleed"
232
+ ? "object-cover object-center md:object-right"
233
+ : "object-contain object-center",
234
+ );
235
+
236
+ return (
237
+ <section
238
+ id={id}
239
+ className={cn(finalSectionClass, className)}
240
+ aria-label={ariaLabel}
241
+ >
242
+ <div
243
+ className={cn(
244
+ "flex flex-col items-stretch",
245
+ imageLayout === "full-bleed"
246
+ ? "md:block"
247
+ : "md:flex-row md:items-start",
248
+ )}
249
+ >
250
+ {/* Text container - takes up left side on large screens */}
251
+ <div
252
+ className={cn(
253
+ "w-full",
254
+ imageLayout === "full-bleed" &&
255
+ "relative z-10 md:w-1/2 md:px-8 md:py-12",
256
+ textContainer.className,
257
+ )}
258
+ >
259
+ <div
260
+ className={cn(
261
+ "flex flex-col items-center",
262
+ lgItemsAlignClasses[textAlign],
263
+ )}
264
+ >
265
+ <h1
266
+ className={cn(
267
+ "text-center",
268
+ lgTextAlignClasses[textAlign],
269
+ normalizedHeading.className,
270
+ )}
271
+ >
272
+ {normalizedHeading.text}
273
+ </h1>
274
+
275
+ <p
276
+ className={cn(
277
+ "text-center",
278
+ lgTextAlignClasses[textAlign],
279
+ normalizedSubheading.className,
280
+ )}
281
+ >
282
+ {normalizedSubheading.text}
283
+ </p>
284
+
285
+ <div
286
+ className={cn(
287
+ "flex flex-col items-center justify-center gap-4 sm:flex-row",
288
+ lgJustifyAlignClasses[textAlign],
289
+ lgItemsAlignClasses[textAlign],
290
+ buttonsContainer.className,
291
+ )}
292
+ >
293
+ {mergedCta1.label && (
294
+ <Button
295
+ asChild
296
+ variant={mergedCta1.variant}
297
+ size={mergedCta1.size}
298
+ className={mergedCta1.className}
299
+ unstyled={mergedCta1.unstyled}
300
+ style={mergedCta1.style}
301
+ >
302
+ <Link
303
+ href={mergedCta1.href || "#"}
304
+ aria-label={mergedCta1.label}
305
+ >
306
+ {mergedCta1.label}
307
+ </Link>
308
+ </Button>
309
+ )}
310
+
311
+ {mergedCta2?.label && (
312
+ <Button
313
+ asChild
314
+ variant={mergedCta2.variant}
315
+ size={mergedCta2.size}
316
+ className={mergedCta2.className}
317
+ unstyled={mergedCta2.unstyled}
318
+ style={mergedCta2.style}
319
+ >
320
+ <Link
321
+ href={mergedCta2.href || "#"}
322
+ aria-label={mergedCta2.label}
323
+ >
324
+ {mergedCta2.label}
325
+ </Link>
326
+ </Button>
327
+ )}
328
+ </div>
329
+ </div>
330
+ </div>
331
+
332
+ {/* Image container - absolutely positioned on right side for full-bleed */}
333
+ <div className={cn(finalImageContainerClass)}>
334
+ {image?.src ? (
335
+ <Image
336
+ priority
337
+ fetchPriority="high"
338
+ src={image.src}
339
+ alt={image.alt || "Hero image"}
340
+ className={finalImageClass}
341
+ fill
342
+ sizes="(min-width: 768px) 50vw, 100vw"
343
+ quality={75}
344
+ />
345
+ ) : (
346
+ <div className="absolute inset-0">{fallback || null}</div>
347
+ )}
348
+ </div>
349
+ </div>
350
+ </section>
351
+ );
352
+ }