@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,37 @@
1
+ import { brand } from "@/lib/brand";
2
+
3
+ const SITE_URL =
4
+ process.env.NEXT_PUBLIC_SITE_URL?.trim() || "https://example.com";
5
+
6
+ /**
7
+ * OpenSearch description document — lets browsers add this site to the
8
+ * address bar's search engine list. When users press Tab after typing the
9
+ * domain, they get an inline search box that hits /search?q=...
10
+ */
11
+ export async function GET(): Promise<Response> {
12
+ const xml = `<?xml version="1.0" encoding="UTF-8"?>
13
+ <OpenSearchDescription xmlns="http://a9.com/-/spec/opensearch/1.1/">
14
+ <ShortName>${escapeXml(brand.shortName)}</ShortName>
15
+ <Description>Search ${escapeXml(brand.name)}</Description>
16
+ <InputEncoding>UTF-8</InputEncoding>
17
+ <Url type="text/html" method="get" template="${SITE_URL}/search?q={searchTerms}" />
18
+ <Url type="application/opensearchdescription+xml" rel="self" template="${SITE_URL}/opensearch.xml" />
19
+ <moz:SearchForm>${SITE_URL}/search</moz:SearchForm>
20
+ </OpenSearchDescription>
21
+ `;
22
+ return new Response(xml, {
23
+ headers: {
24
+ "Content-Type": "application/opensearchdescription+xml; charset=utf-8",
25
+ "Cache-Control": "public, max-age=86400",
26
+ },
27
+ });
28
+ }
29
+
30
+ function escapeXml(s: string): string {
31
+ return s
32
+ .replace(/&/g, "&amp;")
33
+ .replace(/</g, "&lt;")
34
+ .replace(/>/g, "&gt;")
35
+ .replace(/"/g, "&quot;")
36
+ .replace(/'/g, "&apos;");
37
+ }
@@ -0,0 +1,21 @@
1
+ import Link from "next/link";
2
+
3
+ export default async function OrderPage({ params }: { params: Promise<{ id: string }> }) {
4
+ const { id } = await params;
5
+ return (
6
+ <section className="max-w-2xl mx-auto px-8 py-20 text-center">
7
+ <h1 className="font-serif text-3xl mt-0 mb-3">Thanks — your order is confirmed</h1>
8
+ <p className="text-muted-foreground">
9
+ Order <code className="font-mono text-foreground">{id}</code>
10
+ </p>
11
+ <p className="mt-6">
12
+ <Link
13
+ href="/"
14
+ className="inline-block px-6 py-3 rounded-full bg-primary text-primary-foreground font-semibold text-sm transition-colors hover:bg-primary/90"
15
+ >
16
+ Continue shopping
17
+ </Link>
18
+ </p>
19
+ </section>
20
+ );
21
+ }
@@ -0,0 +1,97 @@
1
+ import type { Metadata } from "next";
2
+ import { Suspense } from "react";
3
+ import { cacheTag, cacheLife } from "next/cache";
4
+ import { getServerClient, tags, type Product } from "@cimplify/sdk/server";
5
+ import { Hero } from "@/components/hero";
6
+ import { CollectionStrip } from "@/components/collection-strip";
7
+ import { CategoryGrid } from "@/components/category-grid";
8
+ import { brand } from "@/lib/brand";
9
+
10
+ export const metadata: Metadata = {
11
+ title: `${brand.name} — ${brand.hero.title}`,
12
+ description: brand.description,
13
+ };
14
+
15
+ async function getHomeData() {
16
+ "use cache";
17
+ cacheTag(tags.collections(), tags.categories(), tags.products());
18
+ cacheLife("hours");
19
+
20
+ const client = getServerClient();
21
+ const [colRes, catRes] = await Promise.all([
22
+ client.catalogue.getCollections(),
23
+ client.catalogue.getCategories(),
24
+ ]);
25
+ const collections = colRes.ok ? colRes.value : [];
26
+ const categories = catRes.ok ? catRes.value : [];
27
+
28
+ const collectionsWithProducts = await Promise.all(
29
+ collections.map(async (col) => {
30
+ const r = await client.catalogue.getCollectionProducts(col.id);
31
+ const items = r.ok
32
+ ? ((r.value as { items?: Product[] }).items ?? (r.value as Product[]))
33
+ : [];
34
+ return { collection: col, products: items };
35
+ }),
36
+ );
37
+
38
+ return {
39
+ collections: collectionsWithProducts.filter((x) => x.products.length > 0),
40
+ categories,
41
+ };
42
+ }
43
+
44
+ export default async function HomePage() {
45
+ const { collections, categories } = await getHomeData();
46
+
47
+ return (
48
+ <>
49
+ <Hero
50
+ badge={brand.hero.badge}
51
+ title={brand.hero.title}
52
+ subtitle={brand.hero.subtitle}
53
+ />
54
+ {collections.map(({ collection, products }) => (
55
+ <Suspense
56
+ key={collection.id}
57
+ fallback={<StripSkeleton title={collection.name} />}
58
+ >
59
+ <CollectionStrip
60
+ collection={collection}
61
+ products={products}
62
+ collectionHref={`/collections/${collection.slug}`}
63
+ />
64
+ </Suspense>
65
+ ))}
66
+ <Suspense fallback={<CategoryGridSkeleton />}>
67
+ <CategoryGrid categories={categories} />
68
+ </Suspense>
69
+ </>
70
+ );
71
+ }
72
+
73
+ function StripSkeleton({ title }: { title: string }) {
74
+ return (
75
+ <section className="max-w-7xl mx-auto px-8 pt-12">
76
+ <h2 className="font-serif text-[28px] font-semibold m-0 mb-5">{title}</h2>
77
+ <div className="grid grid-flow-col auto-cols-[minmax(220px,1fr)] gap-4 overflow-x-auto pb-2">
78
+ {Array.from({ length: 5 }).map((_, i) => (
79
+ <div key={i} className="aspect-square bg-muted rounded-2xl animate-pulse" />
80
+ ))}
81
+ </div>
82
+ </section>
83
+ );
84
+ }
85
+
86
+ function CategoryGridSkeleton() {
87
+ return (
88
+ <section className="max-w-7xl mx-auto px-8 pt-14">
89
+ <div className="h-8 w-48 bg-muted rounded mb-5 animate-pulse" />
90
+ <div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-4">
91
+ {Array.from({ length: 6 }).map((_, i) => (
92
+ <div key={i} className="h-32 bg-muted rounded-2xl animate-pulse" />
93
+ ))}
94
+ </div>
95
+ </section>
96
+ );
97
+ }
@@ -0,0 +1,44 @@
1
+ import type { Metadata } from "next";
2
+ import { brand } from "@/lib/brand";
3
+
4
+ export const metadata: Metadata = {
5
+ title: `Privacy Policy — ${brand.name}`,
6
+ description: `How ${brand.name} collects, uses, and protects your personal data.`,
7
+ };
8
+
9
+ export default function PrivacyPage() {
10
+ const p = brand.privacy;
11
+ return (
12
+ <article className="max-w-3xl mx-auto px-8 py-16 prose prose-lg max-w-none">
13
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2 not-prose">
14
+ {p.eyebrow}
15
+ </p>
16
+ <h1 className="font-serif text-[clamp(2.25rem,5vw,3.5rem)] font-semibold mb-2 -tracking-[0.02em]">
17
+ {p.title}
18
+ </h1>
19
+ <p className="text-sm text-muted-foreground not-prose mb-10">
20
+ Last updated: {p.lastUpdated}
21
+ </p>
22
+
23
+ <section className="space-y-5 leading-relaxed text-foreground/90">
24
+ {p.sections.map((s) => (
25
+ <div key={s.heading}>
26
+ <h2 className="font-serif text-2xl font-semibold mt-0">{s.heading}</h2>
27
+ {typeof s.body === "string" ? (
28
+ <p>{s.body}</p>
29
+ ) : (
30
+ <>
31
+ <p>{s.body.intro}</p>
32
+ <ul className="list-disc pl-6 space-y-2">
33
+ {s.body.bullets.map((b) => (
34
+ <li key={b}>{b}</li>
35
+ ))}
36
+ </ul>
37
+ </>
38
+ )}
39
+ </div>
40
+ ))}
41
+ </section>
42
+ </article>
43
+ );
44
+ }
@@ -0,0 +1,66 @@
1
+ import type { Metadata } from "next";
2
+ import { Suspense } from "react";
3
+ import { cacheTag, cacheLife } from "next/cache";
4
+ import { getServerClient, tags, type Product } from "@cimplify/sdk/server";
5
+ import { ReservationsClient } from "./reservations-client";
6
+ import { brand } from "@/lib/brand";
7
+
8
+ export const metadata: Metadata = {
9
+ title: `Reserve a table — ${brand.name}`,
10
+ description: "Pick a date, time, and party size. We'll hold your table.",
11
+ };
12
+
13
+ async function getSeatingOptions(): Promise<Product[]> {
14
+ "use cache";
15
+ cacheTag(tags.products());
16
+ cacheLife("hours");
17
+
18
+ const client = getServerClient();
19
+ const r = await client.catalogue.getProducts({ limit: 50 });
20
+ if (!r.ok) return [];
21
+ // Reservation options live as service products in the catalogue (e.g.
22
+ // "Two-top", "Four-top", "Long table"). Filter to type=service.
23
+ return r.value.items.filter((p) => p.type === "service");
24
+ }
25
+
26
+ export default async function ReservationsPage() {
27
+ const options = await getSeatingOptions();
28
+ return (
29
+ <article className="max-w-5xl mx-auto px-6 sm:px-8 py-14">
30
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2">
31
+ Reservations
32
+ </p>
33
+ <h1 className="font-serif text-[clamp(2rem,5vw,3rem)] font-semibold mb-3 -tracking-[0.02em]">
34
+ Hold a table.
35
+ </h1>
36
+ <p className="text-muted-foreground leading-relaxed mb-10 max-w-xl">
37
+ Bookings open 21 days in advance. SMS confirmation within a minute. Free
38
+ cancellation up to 4 hours before.
39
+ </p>
40
+
41
+ <Suspense fallback={<ReservationsSkeleton />}>
42
+ <ReservationsClient options={options} />
43
+ </Suspense>
44
+ </article>
45
+ );
46
+ }
47
+
48
+ function ReservationsSkeleton() {
49
+ return (
50
+ <div className="grid grid-cols-1 lg:grid-cols-[1fr_1.4fr] gap-8">
51
+ <div className="space-y-2">
52
+ {Array.from({ length: 4 }).map((_, i) => (
53
+ <div key={i} className="h-16 bg-muted rounded-2xl animate-pulse" />
54
+ ))}
55
+ </div>
56
+ <div className="rounded-2xl border border-border bg-card p-8">
57
+ <div className="h-5 w-40 bg-muted rounded mb-4 animate-pulse" />
58
+ <div className="grid grid-cols-3 gap-2">
59
+ {Array.from({ length: 9 }).map((_, i) => (
60
+ <div key={i} className="h-10 bg-muted rounded animate-pulse" />
61
+ ))}
62
+ </div>
63
+ </div>
64
+ </div>
65
+ );
66
+ }
@@ -0,0 +1,234 @@
1
+ "use client";
2
+
3
+ import { useMemo, useState } from "react";
4
+ import { useRouter } from "next/navigation";
5
+ import type { Product } from "@cimplify/sdk";
6
+ import { useCart } from "@cimplify/sdk/react";
7
+
8
+ const PARTY_SIZES = [2, 3, 4, 5, 6, 8, 10, 12];
9
+
10
+ /**
11
+ * Reservation flow. Reservations are modelled as service-type products in
12
+ * the catalogue (e.g. "Two-top", "Four-top", "Long table"). The user picks
13
+ * a seating option, date, time, and party size; we add to cart with the
14
+ * reservation details as item notes; checkout finalises the booking.
15
+ */
16
+ export function ReservationsClient({ options }: { options: Product[] }) {
17
+ const router = useRouter();
18
+ const { addItem } = useCart();
19
+ const [selectedOption, setSelectedOption] = useState<Product | undefined>(options[0]);
20
+ const [selectedDate, setSelectedDate] = useState<Date>(() => new Date());
21
+ const [selectedSlotKey, setSelectedSlotKey] = useState<string | null>(null);
22
+ const [partySize, setPartySize] = useState<number>(2);
23
+ const [notes, setNotes] = useState("");
24
+ const [submitting, setSubmitting] = useState(false);
25
+
26
+ const dates = useMemo(() => {
27
+ const now = new Date();
28
+ return Array.from({ length: 14 }).map((_, i) => {
29
+ const d = new Date(now);
30
+ d.setDate(now.getDate() + i);
31
+ d.setHours(0, 0, 0, 0);
32
+ return d;
33
+ });
34
+ }, []);
35
+
36
+ // Restaurant slots: lunch 12–14:30, dinner 18–21:30, every 30 min.
37
+ const slots = useMemo(() => {
38
+ const out: { date: Date; label: string; service: "Lunch" | "Dinner" }[] = [];
39
+ for (let h = 12; h <= 14; h++) {
40
+ for (const m of [0, 30]) {
41
+ const d = new Date(selectedDate);
42
+ d.setHours(h, m, 0, 0);
43
+ out.push({ date: d, label: `${String(h).padStart(2, "0")}:${String(m).padStart(2, "0")}`, service: "Lunch" });
44
+ }
45
+ }
46
+ for (let h = 18; h <= 21; h++) {
47
+ for (const m of [0, 30]) {
48
+ const d = new Date(selectedDate);
49
+ d.setHours(h, m, 0, 0);
50
+ out.push({ date: d, label: `${String(h).padStart(2, "0")}:${String(m).padStart(2, "0")}`, service: "Dinner" });
51
+ }
52
+ }
53
+ return out;
54
+ }, [selectedDate]);
55
+
56
+ async function confirm() {
57
+ if (!selectedOption || !selectedSlotKey) return;
58
+ setSubmitting(true);
59
+ try {
60
+ const when = new Date(selectedSlotKey).toLocaleString();
61
+ const note = `Party of ${partySize} · ${when}${notes ? ` · ${notes}` : ""}`;
62
+ await addItem(selectedOption, 1, { notes: note });
63
+ router.push("/checkout");
64
+ } catch {
65
+ setSubmitting(false);
66
+ }
67
+ }
68
+
69
+ if (options.length === 0) {
70
+ return (
71
+ <p className="text-muted-foreground">
72
+ No bookable seating options yet. Add Service-type products to your catalogue
73
+ (e.g. "Two-top", "Four-top", "Long table for 12") to enable reservations.
74
+ </p>
75
+ );
76
+ }
77
+
78
+ return (
79
+ <div className="grid grid-cols-1 lg:grid-cols-[1fr_1.4fr] gap-8">
80
+ <div className="space-y-6">
81
+ <div>
82
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-muted-foreground mb-3">
83
+ Seating
84
+ </p>
85
+ <div className="space-y-2">
86
+ {options.map((o) => {
87
+ const active = selectedOption?.id === o.id;
88
+ return (
89
+ <button
90
+ key={o.id}
91
+ type="button"
92
+ onClick={() => setSelectedOption(o)}
93
+ className={[
94
+ "w-full text-left rounded-2xl border p-4 transition-colors",
95
+ active
96
+ ? "border-primary bg-primary/5"
97
+ : "border-border bg-card hover:border-foreground/30",
98
+ ].join(" ")}
99
+ >
100
+ <p className="font-semibold text-sm m-0">{o.name}</p>
101
+ {o.description && (
102
+ <p className="text-xs text-muted-foreground mt-0.5 line-clamp-2">{o.description}</p>
103
+ )}
104
+ </button>
105
+ );
106
+ })}
107
+ </div>
108
+ </div>
109
+
110
+ <div>
111
+ <label
112
+ htmlFor="party"
113
+ className="text-[11px] font-semibold uppercase tracking-[0.16em] text-muted-foreground block mb-3"
114
+ >
115
+ Party size
116
+ </label>
117
+ <div className="grid grid-cols-4 gap-1.5">
118
+ {PARTY_SIZES.map((p) => {
119
+ const active = partySize === p;
120
+ return (
121
+ <button
122
+ key={p}
123
+ type="button"
124
+ onClick={() => setPartySize(p)}
125
+ className={[
126
+ "py-2 rounded-md text-sm font-semibold transition-colors",
127
+ active
128
+ ? "bg-foreground text-background"
129
+ : "bg-background border border-border hover:border-foreground",
130
+ ].join(" ")}
131
+ >
132
+ {p}
133
+ </button>
134
+ );
135
+ })}
136
+ </div>
137
+ </div>
138
+
139
+ <div>
140
+ <label
141
+ htmlFor="notes"
142
+ className="text-[11px] font-semibold uppercase tracking-[0.16em] text-muted-foreground block mb-3"
143
+ >
144
+ Notes (optional)
145
+ </label>
146
+ <textarea
147
+ id="notes"
148
+ rows={3}
149
+ value={notes}
150
+ onChange={(e) => setNotes(e.target.value)}
151
+ placeholder="Allergies, occasion, accessibility…"
152
+ 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"
153
+ />
154
+ </div>
155
+ </div>
156
+
157
+ <div className="rounded-2xl border border-border bg-card p-6">
158
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-muted-foreground mb-3">
159
+ Date
160
+ </p>
161
+ <div className="grid grid-cols-7 gap-1.5 mb-6">
162
+ {dates.map((d) => {
163
+ const active = d.toDateString() === selectedDate.toDateString();
164
+ return (
165
+ <button
166
+ key={d.toISOString()}
167
+ type="button"
168
+ onClick={() => {
169
+ setSelectedDate(d);
170
+ setSelectedSlotKey(null);
171
+ }}
172
+ className={[
173
+ "flex flex-col items-center justify-center py-2 rounded-md transition-colors",
174
+ active
175
+ ? "bg-foreground text-background"
176
+ : "bg-background hover:bg-muted text-foreground",
177
+ ].join(" ")}
178
+ >
179
+ <span className="text-[10px] uppercase tracking-wider opacity-60">
180
+ {d.toLocaleString(undefined, { weekday: "short" })}
181
+ </span>
182
+ <span className="text-base font-semibold tabular-nums">{d.getDate()}</span>
183
+ </button>
184
+ );
185
+ })}
186
+ </div>
187
+
188
+ {(["Lunch", "Dinner"] as const).map((service) => (
189
+ <div key={service} className="mb-5 last:mb-0">
190
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-muted-foreground mb-2">
191
+ {service}
192
+ </p>
193
+ <div className="grid grid-cols-3 sm:grid-cols-5 gap-2">
194
+ {slots
195
+ .filter((s) => s.service === service)
196
+ .map((s) => {
197
+ const key = s.date.toISOString();
198
+ const active = selectedSlotKey === key;
199
+ return (
200
+ <button
201
+ key={key}
202
+ type="button"
203
+ onClick={() => setSelectedSlotKey(key)}
204
+ className={[
205
+ "py-2 rounded-md text-sm tabular-nums transition-colors",
206
+ active
207
+ ? "bg-primary text-primary-foreground"
208
+ : "bg-background border border-border hover:border-primary",
209
+ ].join(" ")}
210
+ >
211
+ {s.label}
212
+ </button>
213
+ );
214
+ })}
215
+ </div>
216
+ </div>
217
+ ))}
218
+
219
+ <button
220
+ type="button"
221
+ onClick={confirm}
222
+ disabled={!selectedOption || !selectedSlotKey || submitting}
223
+ className="w-full mt-4 inline-flex items-center justify-center gap-2 px-5 py-3 rounded-md bg-primary text-primary-foreground font-semibold text-sm hover:bg-primary/90 transition-colors disabled:opacity-50 disabled:cursor-not-allowed"
224
+ >
225
+ {submitting
226
+ ? "Confirming…"
227
+ : selectedSlotKey
228
+ ? `Reserve for ${partySize} on ${new Date(selectedSlotKey).toLocaleDateString()} at ${new Date(selectedSlotKey).toLocaleTimeString(undefined, { hour: "2-digit", minute: "2-digit" })}`
229
+ : "Pick a slot to reserve"}
230
+ </button>
231
+ </div>
232
+ </div>
233
+ );
234
+ }
@@ -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.returns.title} — ${brand.name}`,
7
+ };
8
+
9
+ export default function ReturnsPage() {
10
+ return <PolicyPage policy={brand.returns} />;
11
+ }
@@ -0,0 +1,18 @@
1
+ import type { MetadataRoute } from "next";
2
+
3
+ const SITE_URL =
4
+ process.env.NEXT_PUBLIC_SITE_URL?.trim() || "https://example.com";
5
+
6
+ export default function robots(): MetadataRoute.Robots {
7
+ return {
8
+ rules: [
9
+ {
10
+ userAgent: "*",
11
+ allow: "/",
12
+ disallow: ["/cart", "/checkout", "/orders/", "/api/"],
13
+ },
14
+ ],
15
+ sitemap: `${SITE_URL}/sitemap.xml`,
16
+ host: SITE_URL,
17
+ };
18
+ }
@@ -0,0 +1,38 @@
1
+ import type { Metadata } from "next";
2
+ import { Suspense } from "react";
3
+ import { SearchClient } from "./search-client";
4
+ import { brand } from "@/lib/brand";
5
+
6
+ export const metadata: Metadata = {
7
+ title: `Search — ${brand.name}`,
8
+ description: `Search ${brand.name} — products, collections, categories.`,
9
+ };
10
+
11
+ export default function SearchPage() {
12
+ return (
13
+ <article className="max-w-7xl mx-auto px-6 sm:px-8 py-10">
14
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2">
15
+ Search
16
+ </p>
17
+ <h1 className="font-serif text-[clamp(2rem,4vw,2.75rem)] font-semibold mb-8 -tracking-[0.02em]">
18
+ Find anything.
19
+ </h1>
20
+ <Suspense fallback={<SearchSkeleton />}>
21
+ <SearchClient />
22
+ </Suspense>
23
+ </article>
24
+ );
25
+ }
26
+
27
+ function SearchSkeleton() {
28
+ return (
29
+ <div>
30
+ <div className="h-12 w-full max-w-xl bg-muted rounded animate-pulse mb-8" />
31
+ <div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-3 sm:gap-4">
32
+ {Array.from({ length: 8 }).map((_, i) => (
33
+ <div key={i} className="aspect-[4/3] bg-muted rounded-2xl animate-pulse" />
34
+ ))}
35
+ </div>
36
+ </div>
37
+ );
38
+ }
@@ -0,0 +1,7 @@
1
+ "use client";
2
+
3
+ import { SearchPage } from "@cimplify/sdk/react";
4
+
5
+ export function SearchClient() {
6
+ return <SearchPage />;
7
+ }
@@ -0,0 +1,16 @@
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.shipping.title} — ${brand.name}`,
7
+ description: brand.shipping.sections[0]?.body
8
+ ? typeof brand.shipping.sections[0].body === "string"
9
+ ? brand.shipping.sections[0].body
10
+ : brand.shipping.sections[0].body.intro
11
+ : undefined,
12
+ };
13
+
14
+ export default function ShippingPage() {
15
+ return <PolicyPage policy={brand.shipping} />;
16
+ }
@@ -0,0 +1,31 @@
1
+ import type { Metadata } from "next";
2
+ import { cacheTag, cacheLife } from "next/cache";
3
+ import { getServerClient, tags } from "@cimplify/sdk/server";
4
+ import { ShopClient } from "./shop-client";
5
+ import { brand } from "@/lib/brand";
6
+
7
+ export const metadata: Metadata = {
8
+ title: `Shop — ${brand.name}`,
9
+ description: brand.description,
10
+ };
11
+
12
+ async function getShopData() {
13
+ "use cache";
14
+ cacheTag(tags.products(), tags.categories());
15
+ cacheLife("hours");
16
+
17
+ const client = getServerClient();
18
+ const [p, c] = await Promise.all([
19
+ client.catalogue.getProducts({ limit: 50 }),
20
+ client.catalogue.getCategories(),
21
+ ]);
22
+ return {
23
+ products: p.ok ? p.value.items : [],
24
+ categories: c.ok ? c.value : [],
25
+ };
26
+ }
27
+
28
+ export default async function ShopPage() {
29
+ const { products, categories } = await getShopData();
30
+ return <ShopClient products={products} categories={categories} />;
31
+ }
@@ -0,0 +1,27 @@
1
+ "use client";
2
+
3
+ import { CataloguePage } from "@cimplify/sdk/react";
4
+ import type { Category, Product } from "@cimplify/sdk";
5
+ import { StoreProductCard } from "@/components/store-product-card";
6
+
7
+ /**
8
+ * Client island for the shop page. Server-side fetches all products and
9
+ * categories (cached via `'use cache'` in `app/shop/page.tsx`), then hands
10
+ * them to `<CataloguePage>` which owns the interactive filter / sort state.
11
+ */
12
+ export function ShopClient({
13
+ products,
14
+ categories,
15
+ }: {
16
+ products: Product[];
17
+ categories: Category[];
18
+ }) {
19
+ return (
20
+ <CataloguePage
21
+ title="The Menu"
22
+ products={products}
23
+ categories={categories}
24
+ renderCard={(p) => <StoreProductCard product={p} />}
25
+ />
26
+ );
27
+ }
@@ -0,0 +1,17 @@
1
+ import type { Metadata } from "next";
2
+ import { redirect } from "next/navigation";
3
+ import { brand } from "@/lib/brand";
4
+
5
+ export const metadata: Metadata = {
6
+ title: `Create an account — ${brand.name}`,
7
+ description: brand.account.signupSubtitle,
8
+ };
9
+
10
+ /**
11
+ * Cimplify Link handles sign-up inside the `<CimplifyAccount>` iframe.
12
+ * We bounce /signup to /account; the iframe shows the create-account UI
13
+ * for visitors with no session.
14
+ */
15
+ export default function SignupPage(): never {
16
+ redirect("/account");
17
+ }