@cimplify/cli 0.2.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 (457) hide show
  1. package/README.md +69 -0
  2. package/dist/add-ZJQJZJEF.mjs +125 -0
  3. package/dist/chunk-4YSOZ6LY.mjs +61 -0
  4. package/dist/chunk-4ZVTPMUZ.mjs +155 -0
  5. package/dist/chunk-5XH72JMJ.mjs +211 -0
  6. package/dist/chunk-D7D75ONX.mjs +36 -0
  7. package/dist/chunk-JJYWETGA.mjs +127 -0
  8. package/dist/chunk-L6474RPL.mjs +37 -0
  9. package/dist/chunk-MQMNWLMU.mjs +48 -0
  10. package/dist/chunk-NZ4RG62Z.mjs +141 -0
  11. package/dist/chunk-TAMGCHIL.mjs +81 -0
  12. package/dist/chunk-XSWWWO6H.mjs +3779 -0
  13. package/dist/deploy-WCZOGMED.mjs +206 -0
  14. package/dist/dev-4HKIXWXX.mjs +130 -0
  15. package/dist/dispatcher.mjs +321 -0
  16. package/dist/domains-3RJ4T5IX.mjs +387 -0
  17. package/dist/env-LBYBCBWV.mjs +268 -0
  18. package/dist/link-SEJNW7JS.mjs +45 -0
  19. package/dist/list-D4JC2VWY.mjs +48 -0
  20. package/dist/login-MRYWLQRY.mjs +276 -0
  21. package/dist/logout-ZFZLSJ32.mjs +16 -0
  22. package/dist/logs-LK7CMBCE.mjs +95 -0
  23. package/dist/projects-QJUGOCQZ.mjs +164 -0
  24. package/dist/repo-MV22OHON.mjs +8 -0
  25. package/dist/rollback-XO7RIG2A.mjs +107 -0
  26. package/dist/status-7CMVLD54.mjs +127 -0
  27. package/dist/unlink-5ABCT7B6.mjs +16 -0
  28. package/dist/whoami-INHDUHWA.mjs +24 -0
  29. package/package.json +44 -0
  30. package/templates/storefront-bakery/.claude/skills/cimplify-storefront/SKILL.md +145 -0
  31. package/templates/storefront-bakery/.cursor/rules/cimplify-storefront.mdc +25 -0
  32. package/templates/storefront-bakery/.env.example +16 -0
  33. package/templates/storefront-bakery/AGENTS.md +120 -0
  34. package/templates/storefront-bakery/CLAUDE.md +22 -0
  35. package/templates/storefront-bakery/README.md +73 -0
  36. package/templates/storefront-bakery/__tests__/brand.test.ts +4 -0
  37. package/templates/storefront-bakery/__tests__/cart-flow.test.ts +4 -0
  38. package/templates/storefront-bakery/__tests__/contract.test.ts +4 -0
  39. package/templates/storefront-bakery/app/about/page.tsx +38 -0
  40. package/templates/storefront-bakery/app/accessibility/page.tsx +11 -0
  41. package/templates/storefront-bakery/app/account/addresses/page.tsx +21 -0
  42. package/templates/storefront-bakery/app/account/orders/page.tsx +21 -0
  43. package/templates/storefront-bakery/app/account/page.tsx +22 -0
  44. package/templates/storefront-bakery/app/account/settings/page.tsx +21 -0
  45. package/templates/storefront-bakery/app/cart/page.tsx +9 -0
  46. package/templates/storefront-bakery/app/categories/[slug]/listing-client.tsx +19 -0
  47. package/templates/storefront-bakery/app/categories/[slug]/page.tsx +118 -0
  48. package/templates/storefront-bakery/app/checkout/page.tsx +17 -0
  49. package/templates/storefront-bakery/app/collections/[slug]/listing-client.tsx +20 -0
  50. package/templates/storefront-bakery/app/collections/[slug]/page.tsx +118 -0
  51. package/templates/storefront-bakery/app/contact/contact-form.tsx +109 -0
  52. package/templates/storefront-bakery/app/contact/page.tsx +54 -0
  53. package/templates/storefront-bakery/app/error.tsx +60 -0
  54. package/templates/storefront-bakery/app/faq/page.tsx +46 -0
  55. package/templates/storefront-bakery/app/globals.css +47 -0
  56. package/templates/storefront-bakery/app/layout.tsx +82 -0
  57. package/templates/storefront-bakery/app/llms.txt/route.ts +94 -0
  58. package/templates/storefront-bakery/app/login/page.tsx +17 -0
  59. package/templates/storefront-bakery/app/not-found.tsx +39 -0
  60. package/templates/storefront-bakery/app/opensearch.xml/route.ts +37 -0
  61. package/templates/storefront-bakery/app/orders/[id]/page.tsx +21 -0
  62. package/templates/storefront-bakery/app/page.tsx +97 -0
  63. package/templates/storefront-bakery/app/privacy/page.tsx +44 -0
  64. package/templates/storefront-bakery/app/returns/page.tsx +11 -0
  65. package/templates/storefront-bakery/app/robots.ts +18 -0
  66. package/templates/storefront-bakery/app/search/page.tsx +38 -0
  67. package/templates/storefront-bakery/app/search/search-client.tsx +7 -0
  68. package/templates/storefront-bakery/app/shipping/page.tsx +16 -0
  69. package/templates/storefront-bakery/app/shop/page.tsx +31 -0
  70. package/templates/storefront-bakery/app/shop/shop-client.tsx +27 -0
  71. package/templates/storefront-bakery/app/signup/page.tsx +17 -0
  72. package/templates/storefront-bakery/app/sitemap-page/page.tsx +167 -0
  73. package/templates/storefront-bakery/app/sitemap.ts +62 -0
  74. package/templates/storefront-bakery/app/terms/page.tsx +44 -0
  75. package/templates/storefront-bakery/app/track-order/page.tsx +24 -0
  76. package/templates/storefront-bakery/app/track-order/track-order-form.tsx +69 -0
  77. package/templates/storefront-bakery/components/account-iframe.tsx +13 -0
  78. package/templates/storefront-bakery/components/cart-drawer.tsx +14 -0
  79. package/templates/storefront-bakery/components/cart-pill.tsx +36 -0
  80. package/templates/storefront-bakery/components/category-grid.tsx +28 -0
  81. package/templates/storefront-bakery/components/collection-strip.tsx +45 -0
  82. package/templates/storefront-bakery/components/footer.tsx +148 -0
  83. package/templates/storefront-bakery/components/header.tsx +43 -0
  84. package/templates/storefront-bakery/components/hero.tsx +25 -0
  85. package/templates/storefront-bakery/components/nav-link.tsx +20 -0
  86. package/templates/storefront-bakery/components/policy-page.tsx +49 -0
  87. package/templates/storefront-bakery/components/product-modal.tsx +104 -0
  88. package/templates/storefront-bakery/components/providers.tsx +35 -0
  89. package/templates/storefront-bakery/components/store-product-card.tsx +87 -0
  90. package/templates/storefront-bakery/lib/brand.ts +570 -0
  91. package/templates/storefront-bakery/lib/cart.ts +12 -0
  92. package/templates/storefront-bakery/next.config.ts +42 -0
  93. package/templates/storefront-bakery/package.json +35 -0
  94. package/templates/storefront-bakery/postcss.config.mjs +7 -0
  95. package/templates/storefront-bakery/tsconfig.json +23 -0
  96. package/templates/storefront-bakery/vitest.config.ts +9 -0
  97. package/templates/storefront-fashion/.claude/skills/cimplify-storefront/SKILL.md +145 -0
  98. package/templates/storefront-fashion/.cursor/rules/cimplify-storefront.mdc +25 -0
  99. package/templates/storefront-fashion/.env.example +16 -0
  100. package/templates/storefront-fashion/AGENTS.md +126 -0
  101. package/templates/storefront-fashion/CLAUDE.md +22 -0
  102. package/templates/storefront-fashion/README.md +77 -0
  103. package/templates/storefront-fashion/__tests__/brand.test.ts +4 -0
  104. package/templates/storefront-fashion/__tests__/cart-flow.test.ts +55 -0
  105. package/templates/storefront-fashion/__tests__/contract.test.ts +4 -0
  106. package/templates/storefront-fashion/app/about/page.tsx +41 -0
  107. package/templates/storefront-fashion/app/accessibility/page.tsx +11 -0
  108. package/templates/storefront-fashion/app/account/addresses/page.tsx +21 -0
  109. package/templates/storefront-fashion/app/account/orders/page.tsx +21 -0
  110. package/templates/storefront-fashion/app/account/page.tsx +22 -0
  111. package/templates/storefront-fashion/app/account/settings/page.tsx +21 -0
  112. package/templates/storefront-fashion/app/cart/page.tsx +9 -0
  113. package/templates/storefront-fashion/app/categories/[slug]/listing-client.tsx +19 -0
  114. package/templates/storefront-fashion/app/categories/[slug]/page.tsx +130 -0
  115. package/templates/storefront-fashion/app/checkout/page.tsx +17 -0
  116. package/templates/storefront-fashion/app/collections/[slug]/listing-client.tsx +20 -0
  117. package/templates/storefront-fashion/app/collections/[slug]/page.tsx +130 -0
  118. package/templates/storefront-fashion/app/contact/contact-form.tsx +109 -0
  119. package/templates/storefront-fashion/app/contact/page.tsx +54 -0
  120. package/templates/storefront-fashion/app/error.tsx +61 -0
  121. package/templates/storefront-fashion/app/faq/page.tsx +46 -0
  122. package/templates/storefront-fashion/app/globals.css +46 -0
  123. package/templates/storefront-fashion/app/layout.tsx +78 -0
  124. package/templates/storefront-fashion/app/llms.txt/route.ts +94 -0
  125. package/templates/storefront-fashion/app/login/page.tsx +17 -0
  126. package/templates/storefront-fashion/app/lookbook/page.tsx +132 -0
  127. package/templates/storefront-fashion/app/not-found.tsx +39 -0
  128. package/templates/storefront-fashion/app/opensearch.xml/route.ts +37 -0
  129. package/templates/storefront-fashion/app/orders/[id]/page.tsx +24 -0
  130. package/templates/storefront-fashion/app/page.tsx +183 -0
  131. package/templates/storefront-fashion/app/privacy/page.tsx +44 -0
  132. package/templates/storefront-fashion/app/products/[slug]/page.tsx +165 -0
  133. package/templates/storefront-fashion/app/products/[slug]/product-detail.tsx +70 -0
  134. package/templates/storefront-fashion/app/returns/page.tsx +11 -0
  135. package/templates/storefront-fashion/app/robots.ts +18 -0
  136. package/templates/storefront-fashion/app/search/page.tsx +38 -0
  137. package/templates/storefront-fashion/app/search/search-client.tsx +7 -0
  138. package/templates/storefront-fashion/app/shipping/page.tsx +16 -0
  139. package/templates/storefront-fashion/app/shop/page.tsx +63 -0
  140. package/templates/storefront-fashion/app/shop/shop-client.tsx +32 -0
  141. package/templates/storefront-fashion/app/signup/page.tsx +17 -0
  142. package/templates/storefront-fashion/app/sitemap-page/page.tsx +167 -0
  143. package/templates/storefront-fashion/app/sitemap.ts +59 -0
  144. package/templates/storefront-fashion/app/size-guide/page.tsx +155 -0
  145. package/templates/storefront-fashion/app/terms/page.tsx +44 -0
  146. package/templates/storefront-fashion/app/track-order/page.tsx +24 -0
  147. package/templates/storefront-fashion/app/track-order/track-order-form.tsx +69 -0
  148. package/templates/storefront-fashion/components/account-iframe.tsx +13 -0
  149. package/templates/storefront-fashion/components/brand-marquee.tsx +27 -0
  150. package/templates/storefront-fashion/components/cart-drawer.tsx +14 -0
  151. package/templates/storefront-fashion/components/cart-pill.tsx +36 -0
  152. package/templates/storefront-fashion/components/category-grid.tsx +28 -0
  153. package/templates/storefront-fashion/components/category-tiles.tsx +104 -0
  154. package/templates/storefront-fashion/components/collection-strip.tsx +45 -0
  155. package/templates/storefront-fashion/components/feature-hero.tsx +82 -0
  156. package/templates/storefront-fashion/components/footer.tsx +153 -0
  157. package/templates/storefront-fashion/components/header.tsx +43 -0
  158. package/templates/storefront-fashion/components/hero.tsx +28 -0
  159. package/templates/storefront-fashion/components/nav-link.tsx +20 -0
  160. package/templates/storefront-fashion/components/newsletter.tsx +50 -0
  161. package/templates/storefront-fashion/components/policy-page.tsx +49 -0
  162. package/templates/storefront-fashion/components/promo-banner.tsx +41 -0
  163. package/templates/storefront-fashion/components/providers.tsx +35 -0
  164. package/templates/storefront-fashion/components/section-heading.tsx +37 -0
  165. package/templates/storefront-fashion/components/store-product-card.tsx +87 -0
  166. package/templates/storefront-fashion/components/trade-in-cta.tsx +54 -0
  167. package/templates/storefront-fashion/components/trust-bar.tsx +66 -0
  168. package/templates/storefront-fashion/e2e/visual.spec.ts +52 -0
  169. package/templates/storefront-fashion/lib/brand.ts +518 -0
  170. package/templates/storefront-fashion/lib/cart.ts +12 -0
  171. package/templates/storefront-fashion/next.config.ts +42 -0
  172. package/templates/storefront-fashion/package.json +38 -0
  173. package/templates/storefront-fashion/playwright.config.ts +48 -0
  174. package/templates/storefront-fashion/postcss.config.mjs +7 -0
  175. package/templates/storefront-fashion/tsconfig.json +23 -0
  176. package/templates/storefront-fashion/vitest.config.ts +9 -0
  177. package/templates/storefront-grocery/.claude/skills/cimplify-storefront/SKILL.md +145 -0
  178. package/templates/storefront-grocery/.cursor/rules/cimplify-storefront.mdc +25 -0
  179. package/templates/storefront-grocery/.env.example +16 -0
  180. package/templates/storefront-grocery/AGENTS.md +96 -0
  181. package/templates/storefront-grocery/CLAUDE.md +22 -0
  182. package/templates/storefront-grocery/README.md +73 -0
  183. package/templates/storefront-grocery/__tests__/brand.test.ts +4 -0
  184. package/templates/storefront-grocery/__tests__/cart-flow.test.ts +4 -0
  185. package/templates/storefront-grocery/__tests__/contract.test.ts +4 -0
  186. package/templates/storefront-grocery/app/about/page.tsx +38 -0
  187. package/templates/storefront-grocery/app/accessibility/page.tsx +11 -0
  188. package/templates/storefront-grocery/app/account/addresses/page.tsx +21 -0
  189. package/templates/storefront-grocery/app/account/orders/page.tsx +21 -0
  190. package/templates/storefront-grocery/app/account/page.tsx +22 -0
  191. package/templates/storefront-grocery/app/account/settings/page.tsx +21 -0
  192. package/templates/storefront-grocery/app/cart/page.tsx +9 -0
  193. package/templates/storefront-grocery/app/categories/[slug]/listing-client.tsx +19 -0
  194. package/templates/storefront-grocery/app/categories/[slug]/page.tsx +118 -0
  195. package/templates/storefront-grocery/app/checkout/page.tsx +17 -0
  196. package/templates/storefront-grocery/app/collections/[slug]/listing-client.tsx +20 -0
  197. package/templates/storefront-grocery/app/collections/[slug]/page.tsx +118 -0
  198. package/templates/storefront-grocery/app/contact/contact-form.tsx +109 -0
  199. package/templates/storefront-grocery/app/contact/page.tsx +54 -0
  200. package/templates/storefront-grocery/app/error.tsx +60 -0
  201. package/templates/storefront-grocery/app/faq/page.tsx +46 -0
  202. package/templates/storefront-grocery/app/globals.css +45 -0
  203. package/templates/storefront-grocery/app/layout.tsx +77 -0
  204. package/templates/storefront-grocery/app/llms.txt/route.ts +94 -0
  205. package/templates/storefront-grocery/app/login/page.tsx +17 -0
  206. package/templates/storefront-grocery/app/not-found.tsx +39 -0
  207. package/templates/storefront-grocery/app/opensearch.xml/route.ts +37 -0
  208. package/templates/storefront-grocery/app/orders/[id]/page.tsx +21 -0
  209. package/templates/storefront-grocery/app/page.tsx +97 -0
  210. package/templates/storefront-grocery/app/privacy/page.tsx +44 -0
  211. package/templates/storefront-grocery/app/returns/page.tsx +11 -0
  212. package/templates/storefront-grocery/app/robots.ts +18 -0
  213. package/templates/storefront-grocery/app/search/page.tsx +38 -0
  214. package/templates/storefront-grocery/app/search/search-client.tsx +7 -0
  215. package/templates/storefront-grocery/app/shipping/page.tsx +16 -0
  216. package/templates/storefront-grocery/app/shop/page.tsx +31 -0
  217. package/templates/storefront-grocery/app/shop/shop-client.tsx +27 -0
  218. package/templates/storefront-grocery/app/signup/page.tsx +17 -0
  219. package/templates/storefront-grocery/app/sitemap-page/page.tsx +167 -0
  220. package/templates/storefront-grocery/app/sitemap.ts +62 -0
  221. package/templates/storefront-grocery/app/terms/page.tsx +44 -0
  222. package/templates/storefront-grocery/app/track-order/page.tsx +24 -0
  223. package/templates/storefront-grocery/app/track-order/track-order-form.tsx +69 -0
  224. package/templates/storefront-grocery/components/account-iframe.tsx +13 -0
  225. package/templates/storefront-grocery/components/cart-drawer.tsx +14 -0
  226. package/templates/storefront-grocery/components/cart-pill.tsx +36 -0
  227. package/templates/storefront-grocery/components/category-grid.tsx +28 -0
  228. package/templates/storefront-grocery/components/collection-strip.tsx +45 -0
  229. package/templates/storefront-grocery/components/footer.tsx +148 -0
  230. package/templates/storefront-grocery/components/header.tsx +43 -0
  231. package/templates/storefront-grocery/components/hero.tsx +25 -0
  232. package/templates/storefront-grocery/components/nav-link.tsx +20 -0
  233. package/templates/storefront-grocery/components/policy-page.tsx +49 -0
  234. package/templates/storefront-grocery/components/product-modal.tsx +104 -0
  235. package/templates/storefront-grocery/components/providers.tsx +35 -0
  236. package/templates/storefront-grocery/components/store-product-card.tsx +87 -0
  237. package/templates/storefront-grocery/lib/brand.ts +375 -0
  238. package/templates/storefront-grocery/lib/cart.ts +12 -0
  239. package/templates/storefront-grocery/next.config.ts +42 -0
  240. package/templates/storefront-grocery/package.json +35 -0
  241. package/templates/storefront-grocery/postcss.config.mjs +7 -0
  242. package/templates/storefront-grocery/tsconfig.json +23 -0
  243. package/templates/storefront-grocery/vitest.config.ts +9 -0
  244. package/templates/storefront-restaurant/.claude/skills/cimplify-storefront/SKILL.md +145 -0
  245. package/templates/storefront-restaurant/.cursor/rules/cimplify-storefront.mdc +25 -0
  246. package/templates/storefront-restaurant/.env.example +16 -0
  247. package/templates/storefront-restaurant/AGENTS.md +102 -0
  248. package/templates/storefront-restaurant/CLAUDE.md +22 -0
  249. package/templates/storefront-restaurant/README.md +73 -0
  250. package/templates/storefront-restaurant/__tests__/brand.test.ts +4 -0
  251. package/templates/storefront-restaurant/__tests__/cart-flow.test.ts +4 -0
  252. package/templates/storefront-restaurant/__tests__/contract.test.ts +4 -0
  253. package/templates/storefront-restaurant/app/about/page.tsx +38 -0
  254. package/templates/storefront-restaurant/app/accessibility/page.tsx +11 -0
  255. package/templates/storefront-restaurant/app/account/addresses/page.tsx +21 -0
  256. package/templates/storefront-restaurant/app/account/orders/page.tsx +21 -0
  257. package/templates/storefront-restaurant/app/account/page.tsx +22 -0
  258. package/templates/storefront-restaurant/app/account/settings/page.tsx +21 -0
  259. package/templates/storefront-restaurant/app/cart/page.tsx +9 -0
  260. package/templates/storefront-restaurant/app/categories/[slug]/listing-client.tsx +19 -0
  261. package/templates/storefront-restaurant/app/categories/[slug]/page.tsx +118 -0
  262. package/templates/storefront-restaurant/app/checkout/page.tsx +17 -0
  263. package/templates/storefront-restaurant/app/collections/[slug]/listing-client.tsx +20 -0
  264. package/templates/storefront-restaurant/app/collections/[slug]/page.tsx +118 -0
  265. package/templates/storefront-restaurant/app/contact/contact-form.tsx +109 -0
  266. package/templates/storefront-restaurant/app/contact/page.tsx +54 -0
  267. package/templates/storefront-restaurant/app/error.tsx +60 -0
  268. package/templates/storefront-restaurant/app/faq/page.tsx +46 -0
  269. package/templates/storefront-restaurant/app/globals.css +44 -0
  270. package/templates/storefront-restaurant/app/layout.tsx +82 -0
  271. package/templates/storefront-restaurant/app/llms.txt/route.ts +94 -0
  272. package/templates/storefront-restaurant/app/login/page.tsx +17 -0
  273. package/templates/storefront-restaurant/app/not-found.tsx +39 -0
  274. package/templates/storefront-restaurant/app/opensearch.xml/route.ts +37 -0
  275. package/templates/storefront-restaurant/app/orders/[id]/page.tsx +21 -0
  276. package/templates/storefront-restaurant/app/page.tsx +97 -0
  277. package/templates/storefront-restaurant/app/privacy/page.tsx +44 -0
  278. package/templates/storefront-restaurant/app/reservations/page.tsx +66 -0
  279. package/templates/storefront-restaurant/app/reservations/reservations-client.tsx +234 -0
  280. package/templates/storefront-restaurant/app/returns/page.tsx +11 -0
  281. package/templates/storefront-restaurant/app/robots.ts +18 -0
  282. package/templates/storefront-restaurant/app/search/page.tsx +38 -0
  283. package/templates/storefront-restaurant/app/search/search-client.tsx +7 -0
  284. package/templates/storefront-restaurant/app/shipping/page.tsx +16 -0
  285. package/templates/storefront-restaurant/app/shop/page.tsx +31 -0
  286. package/templates/storefront-restaurant/app/shop/shop-client.tsx +27 -0
  287. package/templates/storefront-restaurant/app/signup/page.tsx +17 -0
  288. package/templates/storefront-restaurant/app/sitemap-page/page.tsx +167 -0
  289. package/templates/storefront-restaurant/app/sitemap.ts +62 -0
  290. package/templates/storefront-restaurant/app/terms/page.tsx +44 -0
  291. package/templates/storefront-restaurant/app/track-order/page.tsx +24 -0
  292. package/templates/storefront-restaurant/app/track-order/track-order-form.tsx +69 -0
  293. package/templates/storefront-restaurant/components/account-iframe.tsx +13 -0
  294. package/templates/storefront-restaurant/components/cart-drawer.tsx +14 -0
  295. package/templates/storefront-restaurant/components/cart-pill.tsx +36 -0
  296. package/templates/storefront-restaurant/components/category-grid.tsx +28 -0
  297. package/templates/storefront-restaurant/components/collection-strip.tsx +45 -0
  298. package/templates/storefront-restaurant/components/footer.tsx +148 -0
  299. package/templates/storefront-restaurant/components/header.tsx +43 -0
  300. package/templates/storefront-restaurant/components/hero.tsx +25 -0
  301. package/templates/storefront-restaurant/components/nav-link.tsx +20 -0
  302. package/templates/storefront-restaurant/components/policy-page.tsx +49 -0
  303. package/templates/storefront-restaurant/components/product-modal.tsx +104 -0
  304. package/templates/storefront-restaurant/components/providers.tsx +35 -0
  305. package/templates/storefront-restaurant/components/store-product-card.tsx +87 -0
  306. package/templates/storefront-restaurant/lib/brand.ts +377 -0
  307. package/templates/storefront-restaurant/lib/cart.ts +12 -0
  308. package/templates/storefront-restaurant/next.config.ts +42 -0
  309. package/templates/storefront-restaurant/package.json +35 -0
  310. package/templates/storefront-restaurant/postcss.config.mjs +7 -0
  311. package/templates/storefront-restaurant/tsconfig.json +23 -0
  312. package/templates/storefront-restaurant/vitest.config.ts +9 -0
  313. package/templates/storefront-retail/.claude/skills/cimplify-storefront/SKILL.md +145 -0
  314. package/templates/storefront-retail/.cursor/rules/cimplify-storefront.mdc +25 -0
  315. package/templates/storefront-retail/.env.example +16 -0
  316. package/templates/storefront-retail/AGENTS.md +117 -0
  317. package/templates/storefront-retail/CLAUDE.md +22 -0
  318. package/templates/storefront-retail/README.md +77 -0
  319. package/templates/storefront-retail/__tests__/brand.test.ts +4 -0
  320. package/templates/storefront-retail/__tests__/cart-flow.test.ts +4 -0
  321. package/templates/storefront-retail/__tests__/contract.test.ts +4 -0
  322. package/templates/storefront-retail/app/about/page.tsx +41 -0
  323. package/templates/storefront-retail/app/accessibility/page.tsx +11 -0
  324. package/templates/storefront-retail/app/account/addresses/page.tsx +21 -0
  325. package/templates/storefront-retail/app/account/orders/page.tsx +21 -0
  326. package/templates/storefront-retail/app/account/page.tsx +22 -0
  327. package/templates/storefront-retail/app/account/settings/page.tsx +21 -0
  328. package/templates/storefront-retail/app/cart/page.tsx +9 -0
  329. package/templates/storefront-retail/app/categories/[slug]/listing-client.tsx +19 -0
  330. package/templates/storefront-retail/app/categories/[slug]/page.tsx +130 -0
  331. package/templates/storefront-retail/app/checkout/page.tsx +17 -0
  332. package/templates/storefront-retail/app/collections/[slug]/listing-client.tsx +20 -0
  333. package/templates/storefront-retail/app/collections/[slug]/page.tsx +130 -0
  334. package/templates/storefront-retail/app/contact/contact-form.tsx +109 -0
  335. package/templates/storefront-retail/app/contact/page.tsx +54 -0
  336. package/templates/storefront-retail/app/error.tsx +61 -0
  337. package/templates/storefront-retail/app/faq/page.tsx +46 -0
  338. package/templates/storefront-retail/app/globals.css +47 -0
  339. package/templates/storefront-retail/app/layout.tsx +77 -0
  340. package/templates/storefront-retail/app/llms.txt/route.ts +94 -0
  341. package/templates/storefront-retail/app/login/page.tsx +17 -0
  342. package/templates/storefront-retail/app/not-found.tsx +39 -0
  343. package/templates/storefront-retail/app/opensearch.xml/route.ts +37 -0
  344. package/templates/storefront-retail/app/orders/[id]/page.tsx +24 -0
  345. package/templates/storefront-retail/app/page.tsx +182 -0
  346. package/templates/storefront-retail/app/privacy/page.tsx +44 -0
  347. package/templates/storefront-retail/app/products/[slug]/page.tsx +165 -0
  348. package/templates/storefront-retail/app/products/[slug]/product-detail.tsx +70 -0
  349. package/templates/storefront-retail/app/returns/page.tsx +11 -0
  350. package/templates/storefront-retail/app/robots.ts +18 -0
  351. package/templates/storefront-retail/app/search/page.tsx +38 -0
  352. package/templates/storefront-retail/app/search/search-client.tsx +7 -0
  353. package/templates/storefront-retail/app/shipping/page.tsx +16 -0
  354. package/templates/storefront-retail/app/shop/page.tsx +63 -0
  355. package/templates/storefront-retail/app/shop/shop-client.tsx +32 -0
  356. package/templates/storefront-retail/app/signup/page.tsx +17 -0
  357. package/templates/storefront-retail/app/sitemap-page/page.tsx +167 -0
  358. package/templates/storefront-retail/app/sitemap.ts +59 -0
  359. package/templates/storefront-retail/app/terms/page.tsx +44 -0
  360. package/templates/storefront-retail/app/track-order/page.tsx +24 -0
  361. package/templates/storefront-retail/app/track-order/track-order-form.tsx +69 -0
  362. package/templates/storefront-retail/components/account-iframe.tsx +13 -0
  363. package/templates/storefront-retail/components/brand-marquee.tsx +27 -0
  364. package/templates/storefront-retail/components/cart-drawer.tsx +14 -0
  365. package/templates/storefront-retail/components/cart-pill.tsx +36 -0
  366. package/templates/storefront-retail/components/category-grid.tsx +28 -0
  367. package/templates/storefront-retail/components/category-tiles.tsx +104 -0
  368. package/templates/storefront-retail/components/collection-strip.tsx +45 -0
  369. package/templates/storefront-retail/components/feature-hero.tsx +84 -0
  370. package/templates/storefront-retail/components/footer.tsx +153 -0
  371. package/templates/storefront-retail/components/header.tsx +45 -0
  372. package/templates/storefront-retail/components/hero.tsx +28 -0
  373. package/templates/storefront-retail/components/nav-link.tsx +20 -0
  374. package/templates/storefront-retail/components/newsletter.tsx +50 -0
  375. package/templates/storefront-retail/components/policy-page.tsx +49 -0
  376. package/templates/storefront-retail/components/promo-banner.tsx +41 -0
  377. package/templates/storefront-retail/components/providers.tsx +35 -0
  378. package/templates/storefront-retail/components/section-heading.tsx +37 -0
  379. package/templates/storefront-retail/components/store-product-card.tsx +87 -0
  380. package/templates/storefront-retail/components/trade-in-cta.tsx +54 -0
  381. package/templates/storefront-retail/components/trust-bar.tsx +66 -0
  382. package/templates/storefront-retail/lib/brand.ts +664 -0
  383. package/templates/storefront-retail/lib/cart.ts +12 -0
  384. package/templates/storefront-retail/next.config.ts +42 -0
  385. package/templates/storefront-retail/package.json +35 -0
  386. package/templates/storefront-retail/postcss.config.mjs +7 -0
  387. package/templates/storefront-retail/tsconfig.json +23 -0
  388. package/templates/storefront-retail/vitest.config.ts +9 -0
  389. package/templates/storefront-services/.claude/skills/cimplify-storefront/SKILL.md +145 -0
  390. package/templates/storefront-services/.cursor/rules/cimplify-storefront.mdc +25 -0
  391. package/templates/storefront-services/.env.example +16 -0
  392. package/templates/storefront-services/AGENTS.md +101 -0
  393. package/templates/storefront-services/CLAUDE.md +22 -0
  394. package/templates/storefront-services/README.md +73 -0
  395. package/templates/storefront-services/__tests__/brand.test.ts +4 -0
  396. package/templates/storefront-services/__tests__/cart-flow.test.ts +4 -0
  397. package/templates/storefront-services/__tests__/contract.test.ts +4 -0
  398. package/templates/storefront-services/app/about/page.tsx +38 -0
  399. package/templates/storefront-services/app/accessibility/page.tsx +11 -0
  400. package/templates/storefront-services/app/account/addresses/page.tsx +21 -0
  401. package/templates/storefront-services/app/account/orders/page.tsx +21 -0
  402. package/templates/storefront-services/app/account/page.tsx +22 -0
  403. package/templates/storefront-services/app/account/settings/page.tsx +21 -0
  404. package/templates/storefront-services/app/book/book-client.tsx +195 -0
  405. package/templates/storefront-services/app/book/page.tsx +65 -0
  406. package/templates/storefront-services/app/cart/page.tsx +9 -0
  407. package/templates/storefront-services/app/categories/[slug]/listing-client.tsx +19 -0
  408. package/templates/storefront-services/app/categories/[slug]/page.tsx +118 -0
  409. package/templates/storefront-services/app/checkout/page.tsx +17 -0
  410. package/templates/storefront-services/app/collections/[slug]/listing-client.tsx +20 -0
  411. package/templates/storefront-services/app/collections/[slug]/page.tsx +118 -0
  412. package/templates/storefront-services/app/contact/contact-form.tsx +109 -0
  413. package/templates/storefront-services/app/contact/page.tsx +54 -0
  414. package/templates/storefront-services/app/error.tsx +60 -0
  415. package/templates/storefront-services/app/faq/page.tsx +46 -0
  416. package/templates/storefront-services/app/globals.css +45 -0
  417. package/templates/storefront-services/app/layout.tsx +82 -0
  418. package/templates/storefront-services/app/llms.txt/route.ts +94 -0
  419. package/templates/storefront-services/app/login/page.tsx +17 -0
  420. package/templates/storefront-services/app/not-found.tsx +39 -0
  421. package/templates/storefront-services/app/opensearch.xml/route.ts +37 -0
  422. package/templates/storefront-services/app/orders/[id]/page.tsx +21 -0
  423. package/templates/storefront-services/app/page.tsx +97 -0
  424. package/templates/storefront-services/app/privacy/page.tsx +44 -0
  425. package/templates/storefront-services/app/returns/page.tsx +11 -0
  426. package/templates/storefront-services/app/robots.ts +18 -0
  427. package/templates/storefront-services/app/search/page.tsx +38 -0
  428. package/templates/storefront-services/app/search/search-client.tsx +7 -0
  429. package/templates/storefront-services/app/shipping/page.tsx +16 -0
  430. package/templates/storefront-services/app/shop/page.tsx +31 -0
  431. package/templates/storefront-services/app/shop/shop-client.tsx +27 -0
  432. package/templates/storefront-services/app/signup/page.tsx +17 -0
  433. package/templates/storefront-services/app/sitemap-page/page.tsx +167 -0
  434. package/templates/storefront-services/app/sitemap.ts +62 -0
  435. package/templates/storefront-services/app/terms/page.tsx +44 -0
  436. package/templates/storefront-services/app/track-order/page.tsx +24 -0
  437. package/templates/storefront-services/app/track-order/track-order-form.tsx +69 -0
  438. package/templates/storefront-services/components/account-iframe.tsx +13 -0
  439. package/templates/storefront-services/components/cart-drawer.tsx +14 -0
  440. package/templates/storefront-services/components/cart-pill.tsx +36 -0
  441. package/templates/storefront-services/components/category-grid.tsx +28 -0
  442. package/templates/storefront-services/components/collection-strip.tsx +45 -0
  443. package/templates/storefront-services/components/footer.tsx +148 -0
  444. package/templates/storefront-services/components/header.tsx +43 -0
  445. package/templates/storefront-services/components/hero.tsx +25 -0
  446. package/templates/storefront-services/components/nav-link.tsx +20 -0
  447. package/templates/storefront-services/components/policy-page.tsx +49 -0
  448. package/templates/storefront-services/components/product-modal.tsx +104 -0
  449. package/templates/storefront-services/components/providers.tsx +35 -0
  450. package/templates/storefront-services/components/store-product-card.tsx +87 -0
  451. package/templates/storefront-services/lib/brand.ts +396 -0
  452. package/templates/storefront-services/lib/cart.ts +12 -0
  453. package/templates/storefront-services/next.config.ts +42 -0
  454. package/templates/storefront-services/package.json +35 -0
  455. package/templates/storefront-services/postcss.config.mjs +7 -0
  456. package/templates/storefront-services/tsconfig.json +23 -0
  457. package/templates/storefront-services/vitest.config.ts +9 -0
@@ -0,0 +1,73 @@
1
+ # __STOREFRONT_NAME__
2
+
3
+ A Cimplify storefront scaffolded from the **bakery** template — Next.js 16 (App Router), React 19, Tailwind v4. Warm food-styled palette, Playfair Display + Inter typography, designed for bakeries, pâtisseries, and food businesses.
4
+
5
+ For a different industry, scaffold with `--template`:
6
+
7
+ ```bash
8
+ bunx @cimplify/sdk init my-store --template retail # electronics / consumer goods
9
+ bunx @cimplify/sdk init my-store --template bakery # default — food / pâtisserie
10
+ ```
11
+
12
+ ## Run
13
+
14
+ ```bash
15
+ bun install
16
+ bun dev
17
+ ```
18
+
19
+ Two things start in parallel:
20
+
21
+ - `bunx @cimplify/sdk` — the Cimplify mock API on `http://127.0.0.1:8787` (Akua's Bakery seeded by default).
22
+ - `next dev` — this storefront on `http://localhost:3000`.
23
+
24
+ Open the storefront in your browser. Edit `app/page.tsx` to start customising. The SDK ships full pages (`<CataloguePage>`, `<ProductPage>`, `<CartPage>`, `<CheckoutPage>`) and layouts (`<DefaultProductLayout>`, `<FoodProductLayout>`, etc.) that you can swap in.
25
+
26
+ ## Structure
27
+
28
+ ```
29
+ app/
30
+ layout.tsx # root layout, fonts, providers, header/footer/modal
31
+ page.tsx # home
32
+ shop/page.tsx # full catalogue (SDK <CataloguePage/>)
33
+ collections/[slug]/ # collection landing
34
+ categories/[slug]/ # category landing
35
+ cart/page.tsx # SDK <CartPage/>
36
+ checkout/page.tsx # SDK <CheckoutPage/>
37
+ orders/[id]/page.tsx # post-checkout thank-you
38
+ globals.css # Tailwind import + theme tokens
39
+ components/
40
+ providers.tsx # CimplifyProvider client wrapper
41
+ header.tsx, footer.tsx, hero.tsx
42
+ store-product-card.tsx # SDK <ProductCard/> wired to URL-driven modal
43
+ product-modal.tsx # ?product=<slug> deep-linkable modal
44
+ collection-strip.tsx # horizontal product strip
45
+ category-grid.tsx # SDK <CategoryGrid/> with router navigation
46
+ lib/
47
+ cart.ts # useCartCount() for the header pill
48
+ ```
49
+
50
+ ## Switch seed
51
+
52
+ ```bash
53
+ bunx @cimplify/sdk --seed restaurant # Mama's Kitchen
54
+ bunx @cimplify/sdk --seed retail # Currents Electronics
55
+ bunx @cimplify/sdk --seed services # Serene Spa
56
+ bunx @cimplify/sdk --seed grocery # FreshMart
57
+ ```
58
+
59
+ Update `NEXT_PUBLIC_CIMPLIFY_BUSINESS_ID` in `.env.local` to match the seeded business.
60
+
61
+ ## Go live
62
+
63
+ ```diff
64
+ # .env.local
65
+ - NEXT_PUBLIC_CIMPLIFY_API_URL=http://127.0.0.1:8787
66
+ + NEXT_PUBLIC_CIMPLIFY_API_URL=https://api.cimplify.io
67
+ - NEXT_PUBLIC_CIMPLIFY_PUBLIC_KEY=mock-dev
68
+ + NEXT_PUBLIC_CIMPLIFY_PUBLIC_KEY=<your tenant key>
69
+ - NEXT_PUBLIC_CIMPLIFY_BUSINESS_ID=bus_default_akua_bakery
70
+ + NEXT_PUBLIC_CIMPLIFY_BUSINESS_ID=<your business id>
71
+ ```
72
+
73
+ Deploy to Vercel with one click — `next.config.ts` already whitelists the SDK image hosts under `images.remotePatterns`.
@@ -0,0 +1,4 @@
1
+ import { createBrandSuite } from "@cimplify/sdk/testing/suite";
2
+ import { brand } from "../lib/brand";
3
+
4
+ createBrandSuite({ brand });
@@ -0,0 +1,4 @@
1
+ import { createCartFlowSuite } from "@cimplify/sdk/testing/suite";
2
+ import { brand } from "../lib/brand";
3
+
4
+ createCartFlowSuite({ seed: brand.mock.seed, businessId: brand.mock.businessId });
@@ -0,0 +1,4 @@
1
+ import { createContractSuite } from "@cimplify/sdk/testing/suite";
2
+ import { brand } from "../lib/brand";
3
+
4
+ createContractSuite({ seed: brand.mock.seed });
@@ -0,0 +1,38 @@
1
+ import type { Metadata } from "next";
2
+ import { brand } from "@/lib/brand";
3
+
4
+ export const metadata: Metadata = {
5
+ title: `About — ${brand.name}`,
6
+ description: brand.description,
7
+ };
8
+
9
+ export default function AboutPage() {
10
+ const a = brand.about;
11
+ const titleParts = a.title.split("\n");
12
+ return (
13
+ <article className="max-w-3xl mx-auto px-8 py-16">
14
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2">
15
+ {a.eyebrow}
16
+ </p>
17
+ <h1 className="font-serif text-[clamp(2.25rem,5vw,3.5rem)] font-semibold mb-6 -tracking-[0.02em]">
18
+ {titleParts.map((line, i) => (
19
+ <span key={i}>
20
+ {line}
21
+ {i < titleParts.length - 1 && <br />}
22
+ </span>
23
+ ))}
24
+ </h1>
25
+ <div className="prose prose-lg max-w-none space-y-5 text-foreground/90 leading-relaxed">
26
+ {a.paragraphs.map((p, i) => (
27
+ <p key={i}>{p}</p>
28
+ ))}
29
+ {a.sections.map((s) => (
30
+ <div key={s.heading}>
31
+ <h2 className="font-serif text-3xl font-semibold mt-10 mb-3">{s.heading}</h2>
32
+ <p>{s.body}</p>
33
+ </div>
34
+ ))}
35
+ </div>
36
+ </article>
37
+ );
38
+ }
@@ -0,0 +1,11 @@
1
+ import type { Metadata } from "next";
2
+ import { brand } from "@/lib/brand";
3
+ import { PolicyPage } from "@/components/policy-page";
4
+
5
+ export const metadata: Metadata = {
6
+ title: `${brand.accessibility.title} — ${brand.name}`,
7
+ };
8
+
9
+ export default function AccessibilityPage() {
10
+ return <PolicyPage policy={brand.accessibility} />;
11
+ }
@@ -0,0 +1,21 @@
1
+ import type { Metadata } from "next";
2
+ import { AccountIframe } from "@/components/account-iframe";
3
+ import { brand } from "@/lib/brand";
4
+
5
+ export const metadata: Metadata = {
6
+ title: `Addresses — ${brand.name}`,
7
+ };
8
+
9
+ export default function AddressesPage() {
10
+ return (
11
+ <article className="max-w-5xl mx-auto px-6 sm:px-8 py-12">
12
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2">
13
+ Account
14
+ </p>
15
+ <h1 className="font-serif text-[clamp(2rem,4vw,2.75rem)] font-semibold mb-8 -tracking-[0.02em]">
16
+ Your addresses
17
+ </h1>
18
+ <AccountIframe section="addresses" />
19
+ </article>
20
+ );
21
+ }
@@ -0,0 +1,21 @@
1
+ import type { Metadata } from "next";
2
+ import { AccountIframe } from "@/components/account-iframe";
3
+ import { brand } from "@/lib/brand";
4
+
5
+ export const metadata: Metadata = {
6
+ title: `Orders — ${brand.name}`,
7
+ };
8
+
9
+ export default function OrdersPage() {
10
+ return (
11
+ <article className="max-w-5xl mx-auto px-6 sm:px-8 py-12">
12
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2">
13
+ Account
14
+ </p>
15
+ <h1 className="font-serif text-[clamp(2rem,4vw,2.75rem)] font-semibold mb-8 -tracking-[0.02em]">
16
+ Your orders
17
+ </h1>
18
+ <AccountIframe section="orders" />
19
+ </article>
20
+ );
21
+ }
@@ -0,0 +1,22 @@
1
+ import type { Metadata } from "next";
2
+ import { AccountIframe } from "@/components/account-iframe";
3
+ import { brand } from "@/lib/brand";
4
+
5
+ export const metadata: Metadata = {
6
+ title: `Account — ${brand.name}`,
7
+ description: brand.account.signupSubtitle,
8
+ };
9
+
10
+ export default function AccountPage() {
11
+ return (
12
+ <article className="max-w-5xl mx-auto px-6 sm:px-8 py-12">
13
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2">
14
+ {brand.account.accountEyebrow}
15
+ </p>
16
+ <h1 className="font-serif text-[clamp(2rem,4vw,2.75rem)] font-semibold mb-8 -tracking-[0.02em]">
17
+ {brand.account.accountTitle}
18
+ </h1>
19
+ <AccountIframe />
20
+ </article>
21
+ );
22
+ }
@@ -0,0 +1,21 @@
1
+ import type { Metadata } from "next";
2
+ import { AccountIframe } from "@/components/account-iframe";
3
+ import { brand } from "@/lib/brand";
4
+
5
+ export const metadata: Metadata = {
6
+ title: `Settings — ${brand.name}`,
7
+ };
8
+
9
+ export default function SettingsPage() {
10
+ return (
11
+ <article className="max-w-5xl mx-auto px-6 sm:px-8 py-12">
12
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2">
13
+ Account
14
+ </p>
15
+ <h1 className="font-serif text-[clamp(2rem,4vw,2.75rem)] font-semibold mb-8 -tracking-[0.02em]">
16
+ Settings
17
+ </h1>
18
+ <AccountIframe section="settings" />
19
+ </article>
20
+ );
21
+ }
@@ -0,0 +1,9 @@
1
+ "use client";
2
+
3
+ import { useRouter } from "next/navigation";
4
+ import { CartPage as SdkCartPage } from "@cimplify/sdk/react";
5
+
6
+ export default function CartPage() {
7
+ const router = useRouter();
8
+ return <SdkCartPage onCheckout={() => router.push("/checkout")} />;
9
+ }
@@ -0,0 +1,19 @@
1
+ "use client";
2
+
3
+ import { ProductGrid } from "@cimplify/sdk/react";
4
+ import type { Product } from "@cimplify/sdk";
5
+ import { StoreProductCard } from "@/components/store-product-card";
6
+
7
+ /**
8
+ * Client island for the category listing. Receives server-fetched products
9
+ * as props (serializable) and owns the `renderCard` function.
10
+ */
11
+ export function ListingClient({ products }: { products: Product[] }) {
12
+ return (
13
+ <ProductGrid
14
+ products={products}
15
+ emptyMessage="No products in this category yet."
16
+ renderCard={(p) => <StoreProductCard product={p} />}
17
+ />
18
+ );
19
+ }
@@ -0,0 +1,118 @@
1
+ import type { Metadata } from "next";
2
+ import { Suspense } from "react";
3
+ import Link from "next/link";
4
+ import { notFound } from "next/navigation";
5
+ import { cacheTag, cacheLife } from "next/cache";
6
+ import {
7
+ getServerClient,
8
+ tags,
9
+ type Category,
10
+ type Product,
11
+ } from "@cimplify/sdk/server";
12
+ import { ListingClient } from "./listing-client";
13
+ import { brand } from "@/lib/brand";
14
+
15
+ interface CategoryData {
16
+ category: Category;
17
+ products: Product[];
18
+ }
19
+
20
+ async function getCategory(slug: string): Promise<CategoryData | null> {
21
+ "use cache";
22
+ cacheTag(tags.categories());
23
+ cacheLife("hours");
24
+
25
+ const client = getServerClient();
26
+ const catRes = await client.catalogue.getCategoryBySlug(slug);
27
+ if (!catRes.ok) return null;
28
+
29
+ cacheTag(tags.category(catRes.value.id), tags.categoryProducts(catRes.value.id));
30
+ const r = await client.catalogue.getCategoryProducts(catRes.value.id);
31
+ const products = r.ok
32
+ ? ((r.value as { items?: Product[] }).items ?? (r.value as Product[]))
33
+ : [];
34
+ return { category: catRes.value, products };
35
+ }
36
+
37
+ export async function generateMetadata({
38
+ params,
39
+ }: {
40
+ params: Promise<{ slug: string }>;
41
+ }): Promise<Metadata> {
42
+ const { slug } = await params;
43
+ const data = await getCategory(slug);
44
+ if (!data) return {};
45
+ return {
46
+ title: `${data.category.name} — ${brand.name}`,
47
+ description: data.category.description ?? undefined,
48
+ };
49
+ }
50
+
51
+ export default async function CategoryPage({
52
+ params,
53
+ }: {
54
+ params: Promise<{ slug: string }>;
55
+ }) {
56
+ return (
57
+ <Suspense fallback={<CategorySkeleton />}>
58
+ <CategoryContent params={params} />
59
+ </Suspense>
60
+ );
61
+ }
62
+
63
+ async function CategoryContent({
64
+ params,
65
+ }: {
66
+ params: Promise<{ slug: string }>;
67
+ }) {
68
+ const { slug } = await params;
69
+ const data = await getCategory(slug);
70
+ if (!data) notFound();
71
+
72
+ const { category, products } = data;
73
+ return (
74
+ <section className="max-w-7xl mx-auto px-8 pt-12">
75
+ <header className="mb-8 text-center">
76
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2">
77
+ Category
78
+ </p>
79
+ <h1 className="font-serif text-[clamp(2rem,4vw,2.75rem)] font-semibold mb-2">
80
+ {category.name}
81
+ </h1>
82
+ {category.description && (
83
+ <p className="mx-auto mb-2 max-w-xl text-muted-foreground">
84
+ {category.description}
85
+ </p>
86
+ )}
87
+ <p className="text-sm text-muted-foreground">
88
+ {products.length} item{products.length === 1 ? "" : "s"}
89
+ </p>
90
+ </header>
91
+ <ListingClient products={products} />
92
+ {products.length === 0 && (
93
+ <p className="text-center mt-8">
94
+ <Link href="/" className="text-primary font-semibold">
95
+ ← Back home
96
+ </Link>
97
+ </p>
98
+ )}
99
+ </section>
100
+ );
101
+ }
102
+
103
+ function CategorySkeleton() {
104
+ return (
105
+ <section className="max-w-7xl mx-auto px-8 pt-12">
106
+ <header className="mb-8 text-center">
107
+ <div className="mx-auto h-3 w-20 bg-muted rounded mb-2 animate-pulse" />
108
+ <div className="mx-auto h-10 w-64 bg-muted rounded mb-2 animate-pulse" />
109
+ <div className="mx-auto h-4 w-80 bg-muted rounded animate-pulse" />
110
+ </header>
111
+ <div className="grid grid-cols-2 md:grid-cols-3 gap-4">
112
+ {Array.from({ length: 6 }).map((_, i) => (
113
+ <div key={i} className="aspect-square bg-muted rounded-2xl animate-pulse" />
114
+ ))}
115
+ </div>
116
+ </section>
117
+ );
118
+ }
@@ -0,0 +1,17 @@
1
+ "use client";
2
+
3
+ import { useRouter } from "next/navigation";
4
+ import { CheckoutPage as SdkCheckoutPage } from "@cimplify/sdk/react";
5
+
6
+ export default function CheckoutPage() {
7
+ const router = useRouter();
8
+ return (
9
+ <SdkCheckoutPage
10
+ onComplete={(result) => {
11
+ if (result.success && result.order) {
12
+ router.push(`/orders/${result.order.id}`);
13
+ }
14
+ }}
15
+ />
16
+ );
17
+ }
@@ -0,0 +1,20 @@
1
+ "use client";
2
+
3
+ import { ProductGrid } from "@cimplify/sdk/react";
4
+ import type { Product } from "@cimplify/sdk";
5
+ import { StoreProductCard } from "@/components/store-product-card";
6
+
7
+ /**
8
+ * Client island for the collection listing. Receives server-fetched
9
+ * products as props (serializable) and owns the `renderCard` function
10
+ * (which can't cross the server/client boundary).
11
+ */
12
+ export function ListingClient({ products }: { products: Product[] }) {
13
+ return (
14
+ <ProductGrid
15
+ products={products}
16
+ emptyMessage="No products in this collection yet."
17
+ renderCard={(p) => <StoreProductCard product={p} />}
18
+ />
19
+ );
20
+ }
@@ -0,0 +1,118 @@
1
+ import type { Metadata } from "next";
2
+ import { Suspense } from "react";
3
+ import Link from "next/link";
4
+ import { notFound } from "next/navigation";
5
+ import { cacheTag, cacheLife } from "next/cache";
6
+ import {
7
+ getServerClient,
8
+ tags,
9
+ type Collection,
10
+ type Product,
11
+ } from "@cimplify/sdk/server";
12
+ import { ListingClient } from "./listing-client";
13
+ import { brand } from "@/lib/brand";
14
+
15
+ interface CollectionData {
16
+ collection: Collection;
17
+ products: Product[];
18
+ }
19
+
20
+ async function getCollection(slug: string): Promise<CollectionData | null> {
21
+ "use cache";
22
+ cacheTag(tags.collections());
23
+ cacheLife("hours");
24
+
25
+ const client = getServerClient();
26
+ const colRes = await client.catalogue.getCollectionBySlug(slug);
27
+ if (!colRes.ok) return null;
28
+
29
+ cacheTag(tags.collection(colRes.value.id), tags.collectionProducts(colRes.value.id));
30
+ const r = await client.catalogue.getCollectionProducts(colRes.value.id);
31
+ const products = r.ok
32
+ ? ((r.value as { items?: Product[] }).items ?? (r.value as Product[]))
33
+ : [];
34
+ return { collection: colRes.value, products };
35
+ }
36
+
37
+ export async function generateMetadata({
38
+ params,
39
+ }: {
40
+ params: Promise<{ slug: string }>;
41
+ }): Promise<Metadata> {
42
+ const { slug } = await params;
43
+ const data = await getCollection(slug);
44
+ if (!data) return {};
45
+ return {
46
+ title: `${data.collection.name} — ${brand.name}`,
47
+ description: data.collection.description ?? undefined,
48
+ };
49
+ }
50
+
51
+ export default async function CollectionPage({
52
+ params,
53
+ }: {
54
+ params: Promise<{ slug: string }>;
55
+ }) {
56
+ return (
57
+ <Suspense fallback={<CollectionSkeleton />}>
58
+ <CollectionContent params={params} />
59
+ </Suspense>
60
+ );
61
+ }
62
+
63
+ async function CollectionContent({
64
+ params,
65
+ }: {
66
+ params: Promise<{ slug: string }>;
67
+ }) {
68
+ const { slug } = await params;
69
+ const data = await getCollection(slug);
70
+ if (!data) notFound();
71
+
72
+ const { collection, products } = data;
73
+ return (
74
+ <section className="max-w-7xl mx-auto px-8 pt-12">
75
+ <header className="mb-8 text-center">
76
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2">
77
+ Collection
78
+ </p>
79
+ <h1 className="font-serif text-[clamp(2rem,4vw,2.75rem)] font-semibold mb-2">
80
+ {collection.name}
81
+ </h1>
82
+ {collection.description && (
83
+ <p className="mx-auto mb-2 max-w-xl text-muted-foreground">
84
+ {collection.description}
85
+ </p>
86
+ )}
87
+ <p className="text-sm text-muted-foreground">
88
+ {products.length} item{products.length === 1 ? "" : "s"}
89
+ </p>
90
+ </header>
91
+ <ListingClient products={products} />
92
+ {products.length === 0 && (
93
+ <p className="text-center mt-8">
94
+ <Link href="/" className="text-primary font-semibold">
95
+ ← Back home
96
+ </Link>
97
+ </p>
98
+ )}
99
+ </section>
100
+ );
101
+ }
102
+
103
+ function CollectionSkeleton() {
104
+ return (
105
+ <section className="max-w-7xl mx-auto px-8 pt-12">
106
+ <header className="mb-8 text-center">
107
+ <div className="mx-auto h-3 w-20 bg-muted rounded mb-2 animate-pulse" />
108
+ <div className="mx-auto h-10 w-64 bg-muted rounded mb-2 animate-pulse" />
109
+ <div className="mx-auto h-4 w-80 bg-muted rounded animate-pulse" />
110
+ </header>
111
+ <div className="grid grid-cols-2 md:grid-cols-3 gap-4">
112
+ {Array.from({ length: 6 }).map((_, i) => (
113
+ <div key={i} className="aspect-square bg-muted rounded-2xl animate-pulse" />
114
+ ))}
115
+ </div>
116
+ </section>
117
+ );
118
+ }
@@ -0,0 +1,109 @@
1
+ "use client";
2
+
3
+ import { useState } from "react";
4
+
5
+ /**
6
+ * Local-only contact form. On submit, fakes a successful response so the
7
+ * template ships a usable interaction. Wire to a real handler (Server
8
+ * Action posting to your CRM, support inbox, or Cimplify Support API)
9
+ * before going to production.
10
+ */
11
+ export function ContactForm({ reasons }: { reasons: string[] }) {
12
+ const [submitted, setSubmitted] = useState(false);
13
+ const [reason, setReason] = useState(reasons[0] ?? "");
14
+
15
+ if (submitted) {
16
+ return (
17
+ <div className="rounded-2xl border border-border bg-card p-8">
18
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2">
19
+ ✓ Sent
20
+ </p>
21
+ <h2 className="font-serif text-2xl font-semibold mb-2">Thanks — we'll be in touch.</h2>
22
+ <p className="text-muted-foreground">
23
+ We respond within one business day. Check your spam folder if you don't see a
24
+ reply by tomorrow.
25
+ </p>
26
+ </div>
27
+ );
28
+ }
29
+
30
+ return (
31
+ <form
32
+ onSubmit={(e) => {
33
+ e.preventDefault();
34
+ setSubmitted(true);
35
+ }}
36
+ className="space-y-5"
37
+ >
38
+ <div className="grid grid-cols-1 sm:grid-cols-2 gap-4">
39
+ <Field label="Your name" name="name" required />
40
+ <Field label="Email" name="email" type="email" required />
41
+ </div>
42
+ <div>
43
+ <label className="text-xs font-semibold uppercase tracking-wider text-muted-foreground block mb-1.5">
44
+ What's it about?
45
+ </label>
46
+ <select
47
+ name="reason"
48
+ value={reason}
49
+ onChange={(e) => setReason(e.target.value)}
50
+ className="w-full px-4 py-2.5 rounded-md bg-background border border-border focus:border-primary focus:ring-2 focus:ring-primary/20 outline-none text-sm"
51
+ >
52
+ {reasons.map((r) => (
53
+ <option key={r} value={r}>
54
+ {r}
55
+ </option>
56
+ ))}
57
+ </select>
58
+ </div>
59
+ <Field label="Order number (optional)" name="order" />
60
+ <div>
61
+ <label className="text-xs font-semibold uppercase tracking-wider text-muted-foreground block mb-1.5">
62
+ Message
63
+ </label>
64
+ <textarea
65
+ name="message"
66
+ required
67
+ rows={6}
68
+ className="w-full px-4 py-2.5 rounded-md bg-background border border-border focus:border-primary focus:ring-2 focus:ring-primary/20 outline-none text-sm resize-y"
69
+ />
70
+ </div>
71
+ <button
72
+ type="submit"
73
+ className="inline-flex items-center justify-center px-5 py-3 rounded-md bg-primary text-primary-foreground font-semibold text-sm hover:bg-primary/90 transition-colors"
74
+ >
75
+ Send message
76
+ </button>
77
+ </form>
78
+ );
79
+ }
80
+
81
+ function Field({
82
+ label,
83
+ name,
84
+ type = "text",
85
+ required,
86
+ }: {
87
+ label: string;
88
+ name: string;
89
+ type?: string;
90
+ required?: boolean;
91
+ }) {
92
+ return (
93
+ <div>
94
+ <label
95
+ htmlFor={name}
96
+ className="text-xs font-semibold uppercase tracking-wider text-muted-foreground block mb-1.5"
97
+ >
98
+ {label}
99
+ </label>
100
+ <input
101
+ id={name}
102
+ name={name}
103
+ type={type}
104
+ required={required}
105
+ className="w-full px-4 py-2.5 rounded-md bg-background border border-border focus:border-primary focus:ring-2 focus:ring-primary/20 outline-none text-sm"
106
+ />
107
+ </div>
108
+ );
109
+ }