@cimplify/cli 0.2.8 → 0.3.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (230) hide show
  1. package/dist/{add-7PTWJV4F.mjs → add-OUMIT4YX.mjs} +10 -10
  2. package/dist/assets-DMK2QOPD.mjs +208 -0
  3. package/dist/chunk-42PFJBC6.mjs +5707 -0
  4. package/dist/{chunk-4SBJVRGM.mjs → chunk-C4M3DXKC.mjs} +3 -1
  5. package/dist/{chunk-NC3GKHDD.mjs → chunk-D7WMSGKK.mjs} +1 -1
  6. package/dist/{chunk-NZ4RG62Z.mjs → chunk-I3XQSSOT.mjs} +4 -1
  7. package/dist/chunk-I6P3I2YJ.mjs +259 -0
  8. package/dist/{chunk-UPEHLREA.mjs → chunk-IQJ45AK3.mjs} +3 -3
  9. package/dist/{chunk-JJYWETGA.mjs → chunk-LS2VTSMQ.mjs} +8 -2
  10. package/dist/{chunk-JOUXICGV.mjs → chunk-MOZQODQS.mjs} +1 -1
  11. package/dist/{chunk-KPGRCXQY.mjs → chunk-QGBXGDA5.mjs} +5 -5
  12. package/dist/chunk-RRY3NEZZ.mjs +79 -0
  13. package/dist/{chunk-L6474RPL.mjs → chunk-RZQTHTXX.mjs} +1 -1
  14. package/dist/{chunk-4YSOZ6LY.mjs → chunk-YI7UMMM7.mjs} +1 -1
  15. package/dist/{deploy-6KVOROT3.mjs → deploy-UKOOPJAE.mjs} +8 -82
  16. package/dist/{dev-AQP6TMYK.mjs → dev-FD4PM3UD.mjs} +5 -5
  17. package/dist/dispatcher.mjs +34 -22
  18. package/dist/doctor-AY7VDIJZ.mjs +314 -0
  19. package/dist/{domains-2ZQ7AG27.mjs → domains-JQMV6GAP.mjs} +5 -5
  20. package/dist/{env-FDBPGU3W.mjs → env-EVMYQUIK.mjs} +6 -6
  21. package/dist/explain-QZVAK5I3.mjs +223 -0
  22. package/dist/introspect-MNTC26UY.mjs +8 -0
  23. package/dist/{link-P4K2HRXY.mjs → link-X3E4UZBF.mjs} +4 -4
  24. package/dist/{list-44MLIFI2.mjs → list-TEQ73IR7.mjs} +3 -3
  25. package/dist/{login-RSKGT6GU.mjs → login-7O7ZXKU3.mjs} +9 -15
  26. package/dist/{logout-ZFZLSJ32.mjs → logout-DJDINVDF.mjs} +2 -2
  27. package/dist/{logs-E2AGTDCF.mjs → logs-KUKGEXR2.mjs} +4 -4
  28. package/dist/{projects-5CJOZ3MT.mjs → projects-364HGWHO.mjs} +13 -11
  29. package/dist/repo-26N2CHF6.mjs +8 -0
  30. package/dist/{rollback-36O4NOEL.mjs → rollback-5YALPQXL.mjs} +5 -5
  31. package/dist/{status-6AT4HF63.mjs → status-W4HW3CX3.mjs} +4 -4
  32. package/dist/{unlink-5ABCT7B6.mjs → unlink-HIIW57OO.mjs} +2 -2
  33. package/dist/{update-6KEG7EWK.mjs → update-2DCENLHM.mjs} +7 -7
  34. package/dist/{whoami-DIJZYZIN.mjs → whoami-LACWBSNL.mjs} +3 -3
  35. package/package.json +3 -3
  36. package/templates/storefront-auto/.claude/skills/cimplify-storefront/SKILL.md +145 -0
  37. package/templates/storefront-auto/.cursor/rules/cimplify-storefront.mdc +25 -0
  38. package/templates/storefront-auto/.env.example +22 -0
  39. package/templates/storefront-auto/AGENTS.md +95 -0
  40. package/templates/storefront-auto/CLAUDE.md +22 -0
  41. package/templates/storefront-auto/README.md +48 -0
  42. package/templates/storefront-auto/__tests__/brand.test.ts +4 -0
  43. package/templates/storefront-auto/__tests__/cart-flow.test.ts +4 -0
  44. package/templates/storefront-auto/__tests__/contract.test.ts +4 -0
  45. package/templates/storefront-auto/app/.well-known/ucp/route.ts +65 -0
  46. package/templates/storefront-auto/app/about/page.tsx +41 -0
  47. package/templates/storefront-auto/app/accessibility/page.tsx +11 -0
  48. package/templates/storefront-auto/app/account/addresses/page.tsx +21 -0
  49. package/templates/storefront-auto/app/account/orders/page.tsx +21 -0
  50. package/templates/storefront-auto/app/account/page.tsx +22 -0
  51. package/templates/storefront-auto/app/account/settings/page.tsx +21 -0
  52. package/templates/storefront-auto/app/cart/page.tsx +9 -0
  53. package/templates/storefront-auto/app/categories/[slug]/listing-client.tsx +19 -0
  54. package/templates/storefront-auto/app/categories/[slug]/page.tsx +130 -0
  55. package/templates/storefront-auto/app/checkout/page.tsx +17 -0
  56. package/templates/storefront-auto/app/collections/[slug]/listing-client.tsx +20 -0
  57. package/templates/storefront-auto/app/collections/[slug]/page.tsx +130 -0
  58. package/templates/storefront-auto/app/contact/contact-form.tsx +109 -0
  59. package/templates/storefront-auto/app/contact/page.tsx +54 -0
  60. package/templates/storefront-auto/app/error.tsx +61 -0
  61. package/templates/storefront-auto/app/faq/page.tsx +46 -0
  62. package/templates/storefront-auto/app/globals.css +47 -0
  63. package/templates/storefront-auto/app/layout.tsx +77 -0
  64. package/templates/storefront-auto/app/llms.txt/route.ts +94 -0
  65. package/templates/storefront-auto/app/login/page.tsx +17 -0
  66. package/templates/storefront-auto/app/not-found.tsx +39 -0
  67. package/templates/storefront-auto/app/opensearch.xml/route.ts +37 -0
  68. package/templates/storefront-auto/app/orders/[id]/page.tsx +24 -0
  69. package/templates/storefront-auto/app/page.tsx +94 -0
  70. package/templates/storefront-auto/app/privacy/page.tsx +44 -0
  71. package/templates/storefront-auto/app/products/[slug]/page.tsx +165 -0
  72. package/templates/storefront-auto/app/products/[slug]/product-detail.tsx +70 -0
  73. package/templates/storefront-auto/app/returns/page.tsx +11 -0
  74. package/templates/storefront-auto/app/robots.ts +18 -0
  75. package/templates/storefront-auto/app/search/page.tsx +38 -0
  76. package/templates/storefront-auto/app/search/search-client.tsx +7 -0
  77. package/templates/storefront-auto/app/shipping/page.tsx +16 -0
  78. package/templates/storefront-auto/app/shop/page.tsx +63 -0
  79. package/templates/storefront-auto/app/shop/shop-client.tsx +32 -0
  80. package/templates/storefront-auto/app/signup/page.tsx +17 -0
  81. package/templates/storefront-auto/app/sitemap-page/page.tsx +167 -0
  82. package/templates/storefront-auto/app/sitemap.ts +59 -0
  83. package/templates/storefront-auto/app/terms/page.tsx +44 -0
  84. package/templates/storefront-auto/app/track-order/page.tsx +24 -0
  85. package/templates/storefront-auto/app/track-order/track-order-form.tsx +69 -0
  86. package/templates/storefront-auto/components/account-iframe.tsx +13 -0
  87. package/templates/storefront-auto/components/auto-hero.tsx +85 -0
  88. package/templates/storefront-auto/components/brand-marquee.tsx +27 -0
  89. package/templates/storefront-auto/components/cart-drawer.tsx +14 -0
  90. package/templates/storefront-auto/components/cart-pill.tsx +36 -0
  91. package/templates/storefront-auto/components/category-grid.tsx +28 -0
  92. package/templates/storefront-auto/components/category-tiles.tsx +104 -0
  93. package/templates/storefront-auto/components/collection-strip.tsx +45 -0
  94. package/templates/storefront-auto/components/feature-hero.tsx +84 -0
  95. package/templates/storefront-auto/components/fitment-finder.tsx +184 -0
  96. package/templates/storefront-auto/components/footer.tsx +153 -0
  97. package/templates/storefront-auto/components/header.tsx +45 -0
  98. package/templates/storefront-auto/components/hero.tsx +28 -0
  99. package/templates/storefront-auto/components/nav-link.tsx +20 -0
  100. package/templates/storefront-auto/components/newsletter.tsx +50 -0
  101. package/templates/storefront-auto/components/policy-page.tsx +49 -0
  102. package/templates/storefront-auto/components/promo-banner.tsx +41 -0
  103. package/templates/storefront-auto/components/providers.tsx +35 -0
  104. package/templates/storefront-auto/components/section-heading.tsx +37 -0
  105. package/templates/storefront-auto/components/service-brief.tsx +65 -0
  106. package/templates/storefront-auto/components/store-product-card.tsx +88 -0
  107. package/templates/storefront-auto/components/trade-in-cta.tsx +54 -0
  108. package/templates/storefront-auto/components/trust-bar.tsx +66 -0
  109. package/templates/storefront-auto/lib/brand.ts +744 -0
  110. package/templates/storefront-auto/lib/cart.ts +12 -0
  111. package/templates/storefront-auto/lib/cimplify-loader.ts +19 -0
  112. package/templates/storefront-auto/next.config.ts +45 -0
  113. package/templates/storefront-auto/package.json +35 -0
  114. package/templates/storefront-auto/postcss.config.mjs +7 -0
  115. package/templates/storefront-auto/tsconfig.json +23 -0
  116. package/templates/storefront-auto/vitest.config.ts +9 -0
  117. package/templates/storefront-bakery/.env.example +2 -2
  118. package/templates/storefront-bakery/README.md +1 -1
  119. package/templates/storefront-bakery/lib/cimplify-loader.ts +19 -0
  120. package/templates/storefront-bakery/next.config.ts +3 -0
  121. package/templates/storefront-bakery/package.json +1 -1
  122. package/templates/storefront-fashion/.env.example +2 -2
  123. package/templates/storefront-fashion/README.md +1 -1
  124. package/templates/storefront-fashion/lib/cimplify-loader.ts +19 -0
  125. package/templates/storefront-fashion/next.config.ts +3 -0
  126. package/templates/storefront-fashion/package.json +1 -1
  127. package/templates/storefront-grocery/.env.example +2 -2
  128. package/templates/storefront-grocery/README.md +1 -1
  129. package/templates/storefront-grocery/lib/cimplify-loader.ts +19 -0
  130. package/templates/storefront-grocery/next.config.ts +3 -0
  131. package/templates/storefront-grocery/package.json +1 -1
  132. package/templates/storefront-pharmacy/.claude/skills/cimplify-storefront/SKILL.md +145 -0
  133. package/templates/storefront-pharmacy/.cursor/rules/cimplify-storefront.mdc +25 -0
  134. package/templates/storefront-pharmacy/.env.example +22 -0
  135. package/templates/storefront-pharmacy/AGENTS.md +118 -0
  136. package/templates/storefront-pharmacy/CLAUDE.md +22 -0
  137. package/templates/storefront-pharmacy/README.md +87 -0
  138. package/templates/storefront-pharmacy/__tests__/brand.test.ts +4 -0
  139. package/templates/storefront-pharmacy/__tests__/cart-flow.test.ts +4 -0
  140. package/templates/storefront-pharmacy/__tests__/contract.test.ts +4 -0
  141. package/templates/storefront-pharmacy/app/.well-known/ucp/route.ts +65 -0
  142. package/templates/storefront-pharmacy/app/about/page.tsx +41 -0
  143. package/templates/storefront-pharmacy/app/accessibility/page.tsx +11 -0
  144. package/templates/storefront-pharmacy/app/account/addresses/page.tsx +21 -0
  145. package/templates/storefront-pharmacy/app/account/orders/page.tsx +21 -0
  146. package/templates/storefront-pharmacy/app/account/page.tsx +22 -0
  147. package/templates/storefront-pharmacy/app/account/settings/page.tsx +21 -0
  148. package/templates/storefront-pharmacy/app/cart/page.tsx +9 -0
  149. package/templates/storefront-pharmacy/app/categories/[slug]/listing-client.tsx +19 -0
  150. package/templates/storefront-pharmacy/app/categories/[slug]/page.tsx +130 -0
  151. package/templates/storefront-pharmacy/app/checkout/page.tsx +17 -0
  152. package/templates/storefront-pharmacy/app/collections/[slug]/listing-client.tsx +20 -0
  153. package/templates/storefront-pharmacy/app/collections/[slug]/page.tsx +130 -0
  154. package/templates/storefront-pharmacy/app/contact/contact-form.tsx +109 -0
  155. package/templates/storefront-pharmacy/app/contact/page.tsx +54 -0
  156. package/templates/storefront-pharmacy/app/error.tsx +61 -0
  157. package/templates/storefront-pharmacy/app/faq/page.tsx +46 -0
  158. package/templates/storefront-pharmacy/app/globals.css +47 -0
  159. package/templates/storefront-pharmacy/app/layout.tsx +77 -0
  160. package/templates/storefront-pharmacy/app/llms.txt/route.ts +94 -0
  161. package/templates/storefront-pharmacy/app/login/page.tsx +17 -0
  162. package/templates/storefront-pharmacy/app/not-found.tsx +39 -0
  163. package/templates/storefront-pharmacy/app/opensearch.xml/route.ts +37 -0
  164. package/templates/storefront-pharmacy/app/orders/[id]/page.tsx +24 -0
  165. package/templates/storefront-pharmacy/app/page.tsx +78 -0
  166. package/templates/storefront-pharmacy/app/privacy/page.tsx +44 -0
  167. package/templates/storefront-pharmacy/app/products/[slug]/page.tsx +165 -0
  168. package/templates/storefront-pharmacy/app/products/[slug]/product-detail.tsx +70 -0
  169. package/templates/storefront-pharmacy/app/returns/page.tsx +11 -0
  170. package/templates/storefront-pharmacy/app/robots.ts +18 -0
  171. package/templates/storefront-pharmacy/app/search/page.tsx +38 -0
  172. package/templates/storefront-pharmacy/app/search/search-client.tsx +7 -0
  173. package/templates/storefront-pharmacy/app/shipping/page.tsx +16 -0
  174. package/templates/storefront-pharmacy/app/shop/page.tsx +63 -0
  175. package/templates/storefront-pharmacy/app/shop/shop-client.tsx +32 -0
  176. package/templates/storefront-pharmacy/app/signup/page.tsx +17 -0
  177. package/templates/storefront-pharmacy/app/sitemap-page/page.tsx +167 -0
  178. package/templates/storefront-pharmacy/app/sitemap.ts +59 -0
  179. package/templates/storefront-pharmacy/app/terms/page.tsx +44 -0
  180. package/templates/storefront-pharmacy/app/track-order/page.tsx +24 -0
  181. package/templates/storefront-pharmacy/app/track-order/track-order-form.tsx +69 -0
  182. package/templates/storefront-pharmacy/components/account-iframe.tsx +13 -0
  183. package/templates/storefront-pharmacy/components/brand-marquee.tsx +27 -0
  184. package/templates/storefront-pharmacy/components/cart-drawer.tsx +14 -0
  185. package/templates/storefront-pharmacy/components/cart-pill.tsx +36 -0
  186. package/templates/storefront-pharmacy/components/category-grid.tsx +28 -0
  187. package/templates/storefront-pharmacy/components/category-tiles.tsx +104 -0
  188. package/templates/storefront-pharmacy/components/collection-strip.tsx +45 -0
  189. package/templates/storefront-pharmacy/components/feature-hero.tsx +84 -0
  190. package/templates/storefront-pharmacy/components/footer.tsx +153 -0
  191. package/templates/storefront-pharmacy/components/header.tsx +45 -0
  192. package/templates/storefront-pharmacy/components/health-brief.tsx +65 -0
  193. package/templates/storefront-pharmacy/components/hero.tsx +28 -0
  194. package/templates/storefront-pharmacy/components/nav-link.tsx +20 -0
  195. package/templates/storefront-pharmacy/components/newsletter.tsx +50 -0
  196. package/templates/storefront-pharmacy/components/pharmacy-hero.tsx +95 -0
  197. package/templates/storefront-pharmacy/components/policy-page.tsx +49 -0
  198. package/templates/storefront-pharmacy/components/promo-banner.tsx +41 -0
  199. package/templates/storefront-pharmacy/components/providers.tsx +35 -0
  200. package/templates/storefront-pharmacy/components/section-heading.tsx +37 -0
  201. package/templates/storefront-pharmacy/components/store-product-card.tsx +88 -0
  202. package/templates/storefront-pharmacy/components/symptom-finder.tsx +108 -0
  203. package/templates/storefront-pharmacy/components/trade-in-cta.tsx +54 -0
  204. package/templates/storefront-pharmacy/components/trust-bar.tsx +66 -0
  205. package/templates/storefront-pharmacy/components/urgent-ctas.tsx +117 -0
  206. package/templates/storefront-pharmacy/lib/brand.ts +790 -0
  207. package/templates/storefront-pharmacy/lib/cart.ts +12 -0
  208. package/templates/storefront-pharmacy/lib/cimplify-loader.ts +19 -0
  209. package/templates/storefront-pharmacy/next.config.ts +45 -0
  210. package/templates/storefront-pharmacy/package.json +35 -0
  211. package/templates/storefront-pharmacy/postcss.config.mjs +7 -0
  212. package/templates/storefront-pharmacy/tsconfig.json +23 -0
  213. package/templates/storefront-pharmacy/vitest.config.ts +9 -0
  214. package/templates/storefront-restaurant/.env.example +2 -2
  215. package/templates/storefront-restaurant/README.md +1 -1
  216. package/templates/storefront-restaurant/lib/cimplify-loader.ts +19 -0
  217. package/templates/storefront-restaurant/next.config.ts +3 -0
  218. package/templates/storefront-restaurant/package.json +1 -1
  219. package/templates/storefront-retail/.env.example +2 -2
  220. package/templates/storefront-retail/README.md +1 -1
  221. package/templates/storefront-retail/lib/cimplify-loader.ts +19 -0
  222. package/templates/storefront-retail/next.config.ts +3 -0
  223. package/templates/storefront-retail/package.json +1 -1
  224. package/templates/storefront-services/.env.example +2 -2
  225. package/templates/storefront-services/README.md +1 -1
  226. package/templates/storefront-services/lib/cimplify-loader.ts +19 -0
  227. package/templates/storefront-services/next.config.ts +3 -0
  228. package/templates/storefront-services/package.json +1 -1
  229. package/dist/chunk-H2HJQGFY.mjs +0 -3911
  230. package/dist/repo-E6SBKVDG.mjs +0 -8
@@ -0,0 +1,118 @@
1
+ # AGENTS.md — Pharmacy storefront template
2
+
3
+ If you are an AI agent (Claude, Cursor, Aider, devin, …) working on this storefront, **start here.**
4
+
5
+ ## TL;DR for rebranding
6
+
7
+ 1. **Edit `lib/brand.ts`** — every visible string lives here.
8
+ 2. **Edit `app/globals.css`** — `@theme { … }` block for palette + radius + font references.
9
+ 3. **Edit `.env.local`** — `NEXT_PUBLIC_CIMPLIFY_API_URL`, `NEXT_PUBLIC_CIMPLIFY_PUBLIC_KEY`, `NEXT_PUBLIC_CIMPLIFY_BUSINESS_ID`, `NEXT_PUBLIC_SITE_URL`.
10
+
11
+ ## Aesthetic
12
+
13
+ - **Inter + JetBrains Mono** — clean clinical sans, mono-spaced labels for trust.
14
+ - **Soft white + calm teal accent** — medical without being sterile.
15
+ - **Sharp corners**: `0.5rem` — modern community pharmacy.
16
+ - **`Pharmacy`** is the Schema.org `@type` (set in `brand.schemaType`).
17
+
18
+ ## Page surface
19
+
20
+ ```
21
+ app/
22
+ page.tsx Multi-section home — featured hero, trust bar, category tiles,
23
+ pharmacist-consult promo, "Just in stock" grid, brand strip,
24
+ collections, prescription-upload CTA, best sellers, newsletter
25
+ shop/page.tsx SDK <CataloguePage/> with custom hero
26
+ search/page.tsx Search (SDK <SearchPage/>)
27
+ collections/[slug]/page.tsx Collection landing
28
+ categories/[slug]/page.tsx Category landing
29
+ products/[slug]/page.tsx Full product detail page with prescription-upload fields
30
+ (auto-rendered from product.input_fields)
31
+
32
+ cart/page.tsx SDK <CartPage/>
33
+ checkout/page.tsx SDK <CheckoutPage/>
34
+ orders/[id]/page.tsx Post-checkout confirmation
35
+
36
+ account/page.tsx <CimplifyAccount /> (iframe)
37
+ account/orders/page.tsx <CimplifyAccount section="orders" />
38
+ account/addresses/page.tsx <CimplifyAccount section="addresses" />
39
+ account/settings/page.tsx <CimplifyAccount section="settings" />
40
+ login/page.tsx, signup/page.tsx redirects → /account
41
+
42
+ contact/page.tsx Pharmacist consult form
43
+ track-order/page.tsx Guest lookup → /orders/[id]
44
+
45
+ about/page.tsx, faq/page.tsx Brand pages
46
+ shipping/page.tsx, returns/page.tsx, accessibility/page.tsx Standalone policies
47
+ terms/page.tsx, privacy/page.tsx
48
+
49
+ sitemap-page/page.tsx HTML sitemap
50
+ sitemap.ts, robots.ts XML sitemap, robots.txt
51
+ llms.txt/route.ts LLM-friendly site index
52
+ opensearch.xml/route.ts Browser search description
53
+ error.tsx, not-found.tsx Global boundaries
54
+ ```
55
+
56
+ ## File ↔ brand-field map
57
+
58
+ | File | Reads from `brand` |
59
+ |---|---|
60
+ | `app/layout.tsx` | identity, contact, socials, schemaType (Pharmacy JSON-LD) |
61
+ | `app/page.tsx` | `brand.hero`, `brand.trustItems`, `brand.brandStrip`, `brand.promo`, `brand.tradeIn` (repurposed as prescription CTA), `brand.newsletter` |
62
+ | `app/about/page.tsx` | `brand.about` |
63
+ | `app/faq/page.tsx` | `brand.faq` |
64
+ | `app/terms/page.tsx` | `brand.terms` |
65
+ | `app/privacy/page.tsx` | `brand.privacy` |
66
+ | `app/shipping/page.tsx` | `brand.shipping` (delivery + pickup + cold-chain copy) |
67
+ | `app/returns/page.tsx` | `brand.returns` |
68
+ | `app/accessibility/page.tsx` | `brand.accessibility` |
69
+ | `app/contact/page.tsx` | `brand.contactPage`, `brand.contact` |
70
+ | `app/track-order/page.tsx` | `brand.trackOrder` |
71
+ | `app/account/*/page.tsx` | `brand.account` (Cimplify Link iframe owns the UI) |
72
+ | `app/products/[slug]/page.tsx` | `brand.name`, `brand.currency` (Product JSON-LD: brand, offer, availability) |
73
+ | `app/llms.txt/route.ts` | `brand.llms`, contact, currency |
74
+ | `app/opensearch.xml/route.ts` | `brand.shortName`, `brand.name` |
75
+ | `components/header.tsx` | `brand.shortName`, `brand.microTag`, `brand.header.nav` |
76
+ | `components/footer.tsx` | `brand.footer`, `brand.contact`, `brand.socials` |
77
+ | `components/promo-banner.tsx` | `brand.promo` (renders nothing if absent) |
78
+ | `components/trade-in-cta.tsx` | `brand.tradeIn` (repurposed as prescription CTA; renders nothing if absent) |
79
+ | `components/brand-marquee.tsx` | `brand.brandStrip` |
80
+ | `components/trust-bar.tsx` | `brand.trustItems` |
81
+ | `components/newsletter.tsx` | `brand.newsletter` |
82
+
83
+ ## Pharmacy-specific notes
84
+
85
+ - **Prescription / consent / DOB inputs**: prescription-required products (e.g. `amoxicillin-rx`, `insulin-rx`, `vaccination-consent`) ship with `input_fields` on the catalogue payload. The SDK `<ProductPage/>` reads `product.input_fields` and auto-renders file upload, consent signature, age checkbox, and DOB inputs. The payload is forwarded with the cart line — your backend (or the mock) decides whether to accept the order before allowing checkout.
86
+ - **`brand.tradeIn` is the prescription-upload CTA.** The `trade-in-cta.tsx` component is unchanged from the retail template — only the brand copy changed. If you'd rather use a dedicated component, eject it and rename.
87
+ - **Schema.org `@type`** is `Pharmacy` — semantically meaningful for Google Business and JSON-LD parsers. If your business is broader (clinic + pharmacy), switch to `LocalBusiness`.
88
+ - **Controlled substances** (Schedule II) usually require in-person collection. The mock doesn't enforce this — your backend or a Cimplify rule should reject those cart lines from delivery-method=ship orders.
89
+
90
+ ## Known TODOs
91
+
92
+ - Contact form + newsletter signup currently fake their submit. Wire `client.support.sendMessage(...)` for contact and a real list provider for newsletter.
93
+ - Trust-bar / brand-marquee imagery / wordmarks are placeholders — swap for the real partners you're authorised to display.
94
+
95
+ ## Mock seed
96
+
97
+ Wired to `--seed pharmacy` (Wellspring Pharmacy). Edit `dev:mock` in `package.json` to preview another seed.
98
+
99
+ ## Customizing SDK components
100
+
101
+ For anything beyond `lib/brand.ts` + `app/globals.css`, lean on the SDK's prebuilt components rather than reinvent. **Especially for product customization** (variants, add-ons, prescription inputs, services with scheduling) — the SDK already gets price math, axis matching, input-field rendering, and cart payload contracts right. Default to ejecting and restyling:
102
+
103
+ ```bash
104
+ cimplify add cart-drawer
105
+ cimplify add variant-selector
106
+ cimplify add product-page
107
+ ```
108
+
109
+ Then edit the local copy. **Don't change the cart payload shape** unless you're also touching the SDK mock + backend lens. Full ejection rules and the customizer contract are in the SDK-level [`AGENTS.md`](../../AGENTS.md) → "Don't reinvent product customization".
110
+
111
+ ## Quick start
112
+
113
+ ```bash
114
+ bun install
115
+ bun dev
116
+ ```
117
+
118
+ Open <http://localhost:3000>.
@@ -0,0 +1,22 @@
1
+ # CLAUDE.md
2
+
3
+ This project was scaffolded from a Cimplify storefront template (`cimplify init`).
4
+
5
+ ## Read these first
6
+
7
+ - **`AGENTS.md`** at the project root — the file ↔ `brand.X` field map for *this template's* industry, plus the architectural rules.
8
+ - **`.claude/skills/cimplify-storefront/SKILL.md`** — the playbook for common tasks (rebrand, add a section, wire a Server Action, deploy, eject a component). Invoke it whenever you're asked to change content, palette, copy, or routing.
9
+
10
+ ## The two-line summary
11
+
12
+ 1. **Edit `lib/brand.ts`** for any visible string.
13
+ 2. **Edit `app/globals.css` `@theme` block** for any palette / radius / font change.
14
+
15
+ That covers ~95% of what merchants ask for. For anything else, follow the `cimplify-storefront` skill.
16
+
17
+ ## Don't do
18
+
19
+ - Hardcode strings in pages or components.
20
+ - Disable `cacheComponents: true` in `next.config.ts`.
21
+ - Use `unstable_cache` — Next 16's canonical primitive is `'use cache'`.
22
+ - Run `bun test` (Bun's `vi` shim is incomplete) — use `bun run test:run`.
@@ -0,0 +1,87 @@
1
+ # __STOREFRONT_NAME__
2
+
3
+ A Cimplify storefront scaffolded from the **pharmacy** template — Next.js 16 (App Router), React 19, Tailwind v4. Clinical white + calm teal palette, Inter + JetBrains Mono typography, trustworthy community-pharmacy aesthetic.
4
+
5
+ ## Run
6
+
7
+ ```bash
8
+ bun install
9
+ bun dev
10
+ ```
11
+
12
+ Two things start in parallel:
13
+
14
+ - `cimplify-mock --seed pharmacy` — the Cimplify mock API on `http://127.0.0.1:8787`, seeded with Wellspring Pharmacy.
15
+ - `next dev` — this storefront on `http://localhost:3000`.
16
+
17
+ Open the storefront in your browser. Edit `app/page.tsx` to start customising.
18
+
19
+ ## Structure
20
+
21
+ ```
22
+ app/
23
+ layout.tsx # root layout, fonts, providers, header/footer/modal
24
+ page.tsx # home
25
+ shop/page.tsx # full catalogue (SDK <CataloguePage/>)
26
+ collections/[slug]/ # collection landing
27
+ categories/[slug]/ # category landing
28
+ cart/page.tsx # SDK <CartPage/>
29
+ checkout/page.tsx # SDK <CheckoutPage/>
30
+ orders/[id]/page.tsx # post-checkout thank-you
31
+ about, faq, terms, privacy
32
+ globals.css # Tailwind import + theme tokens
33
+ components/
34
+ providers.tsx # CimplifyProvider client wrapper
35
+ header.tsx, footer.tsx, hero.tsx
36
+ store-product-card.tsx # SDK <ProductCard/>
37
+ trade-in-cta.tsx # repurposed as a prescription-upload CTA
38
+ collection-strip.tsx # horizontal product strip
39
+ category-tiles.tsx # SDK category grid
40
+ lib/
41
+ brand.ts # every visible string (pharmacy copy)
42
+ cart.ts # useCartCount() for the header pill
43
+ ```
44
+
45
+ ## Pharmacy-specific behaviour
46
+
47
+ - Prescription-required products (insulin, amoxicillin, etc.) ship with `input_fields` on the catalogue payload — the SDK product page renders prescription upload + consent signature + DOB inputs automatically. Look at how `paracetamol`, `amoxicillin-rx`, and `baby-formula` differ in the mock seed.
48
+ - `brand.tradeIn` is repurposed as the "Send us your script" CTA on the home page.
49
+ - `Pharmacy` schema.org `@type` is set in `brand.schemaType` so JSON-LD on every page identifies the business correctly.
50
+
51
+ ## Switch the seed
52
+
53
+ This template is wired to the `pharmacy` seed. To preview a different industry without re-scaffolding:
54
+
55
+ ```bash
56
+ cimplify-mock --seed restaurant # Mama's Kitchen
57
+ cimplify-mock --seed services # Serene Spa
58
+ cimplify-mock --seed grocery # FreshMart
59
+ cimplify-mock --seed retail # Currents Electronics
60
+ ```
61
+
62
+ Update `NEXT_PUBLIC_CIMPLIFY_BUSINESS_ID` in `.env.local` to match the seeded business.
63
+
64
+ For a fresh scaffold with another design altogether:
65
+
66
+ ```bash
67
+ cimplify init my-store --template bakery
68
+ cimplify init my-store --template restaurant
69
+ cimplify init my-store --template services
70
+ cimplify init my-store --template grocery
71
+ cimplify init my-store --template fashion
72
+ cimplify init my-store --template retail
73
+ ```
74
+
75
+ ## Go live
76
+
77
+ ```diff
78
+ # .env.local
79
+ - NEXT_PUBLIC_CIMPLIFY_API_URL=http://127.0.0.1:8787
80
+ + NEXT_PUBLIC_CIMPLIFY_API_URL=https://api.cimplify.io
81
+ - NEXT_PUBLIC_CIMPLIFY_PUBLIC_KEY=mock-dev
82
+ + NEXT_PUBLIC_CIMPLIFY_PUBLIC_KEY=<your tenant key>
83
+ - NEXT_PUBLIC_CIMPLIFY_BUSINESS_ID=bus_wellspring_pharmacy
84
+ + NEXT_PUBLIC_CIMPLIFY_BUSINESS_ID=<your business id>
85
+ ```
86
+
87
+ Deploy with `cimplify deploy --prod` after linking the project. See [`cimplify` CLI docs](https://docs.cimplify.io/cli/deploy). `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,65 @@
1
+ import { NextResponse } from "next/server";
2
+
3
+ /**
4
+ * UCP (Universal Commerce Protocol) manifest discovery endpoint.
5
+ *
6
+ * Agents — Claude, ChatGPT, Gemini, MCP clients — probe
7
+ * `https://<your-domain>/.well-known/ucp` to learn what commerce
8
+ * capabilities your storefront supports. We forward the request to
9
+ * Cimplify, which returns the canonical manifest; the response body
10
+ * tells agents to make subsequent UCP calls directly to
11
+ * `api.cimplify.io/ucp/v1/<handle>/*` (no per-request proxy here).
12
+ *
13
+ * Edge-cached for an hour because capabilities change rarely.
14
+ */
15
+ const UCP_API_BASE =
16
+ process.env.NEXT_PUBLIC_CIMPLIFY_API_URL || "https://api.cimplify.io";
17
+
18
+
19
+ export async function GET() {
20
+ const businessHandle = process.env.NEXT_PUBLIC_CIMPLIFY_BUSINESS_HANDLE;
21
+
22
+ if (!businessHandle) {
23
+ return NextResponse.json(
24
+ {
25
+ error: "NEXT_PUBLIC_CIMPLIFY_BUSINESS_HANDLE not set",
26
+ remediation:
27
+ "Set NEXT_PUBLIC_CIMPLIFY_BUSINESS_HANDLE in .env.local (and your deployment env) to your business handle.",
28
+ },
29
+ { status: 500 },
30
+ );
31
+ }
32
+
33
+ try {
34
+ const response = await fetch(
35
+ `${UCP_API_BASE}/ucp/v1/${businessHandle}/manifest`,
36
+ {
37
+ headers: { "Content-Type": "application/json" },
38
+ next: { revalidate: 3600 },
39
+ },
40
+ );
41
+
42
+ if (!response.ok) {
43
+ return NextResponse.json(
44
+ { error: `Upstream UCP manifest fetch failed: ${response.status}` },
45
+ { status: response.status },
46
+ );
47
+ }
48
+
49
+ const manifest = await response.json();
50
+ return NextResponse.json(manifest, {
51
+ headers: {
52
+ "Content-Type": "application/json",
53
+ "Cache-Control": "public, max-age=3600, s-maxage=3600",
54
+ },
55
+ });
56
+ } catch (error) {
57
+ return NextResponse.json(
58
+ {
59
+ error: "Failed to fetch UCP manifest",
60
+ detail: error instanceof Error ? error.message : String(error),
61
+ },
62
+ { status: 500 },
63
+ );
64
+ }
65
+ }
@@ -0,0 +1,41 @@
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
+ // Title supports a single \n for a hard line break.
12
+ const titleParts = a.title.split("\n");
13
+ return (
14
+ <article className="max-w-3xl mx-auto px-8 py-16">
15
+ <p className="text-[11px] font-mono uppercase tracking-[0.16em] text-primary mb-2">
16
+ {a.eyebrow}
17
+ </p>
18
+ <h1 className="text-[clamp(2.25rem,5vw,3.5rem)] font-bold mb-6 -tracking-[0.025em] leading-tight">
19
+ {titleParts.map((line, i) => (
20
+ <span key={i}>
21
+ {line}
22
+ {i < titleParts.length - 1 && <br />}
23
+ </span>
24
+ ))}
25
+ </h1>
26
+ <div className="prose prose-lg max-w-none space-y-5 text-foreground/90 leading-relaxed">
27
+ {a.paragraphs.map((p, i) => (
28
+ <p key={i}>{p}</p>
29
+ ))}
30
+ {a.sections.map((s) => (
31
+ <div key={s.heading}>
32
+ <h2 className="text-2xl font-semibold mt-10 mb-3 -tracking-[0.02em]">
33
+ {s.heading}
34
+ </h2>
35
+ <p>{s.body}</p>
36
+ </div>
37
+ ))}
38
+ </div>
39
+ </article>
40
+ );
41
+ }
@@ -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,130 @@
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
+ <>
75
+ <section className="bg-foreground text-background relative overflow-hidden">
76
+ <div className="absolute inset-0 opacity-[0.04] pointer-events-none [background-image:radial-gradient(circle_at_2px_2px,white_1px,transparent_0)] [background-size:32px_32px]" />
77
+ <div className="relative max-w-7xl mx-auto px-6 sm:px-8 py-12 sm:py-14">
78
+ <nav className="text-[12px] font-mono text-background/60 mb-3 flex items-center gap-2">
79
+ <Link href="/" className="hover:text-background transition-colors">Home</Link>
80
+ <span>/</span>
81
+ <Link href="/shop" className="hover:text-background transition-colors">Shop</Link>
82
+ <span>/</span>
83
+ <span className="text-background/90">{category.name}</span>
84
+ </nav>
85
+ <h1 className="text-[clamp(2rem,4vw,3rem)] font-bold m-0 -tracking-[0.025em]">
86
+ {category.name}
87
+ </h1>
88
+ {category.description && (
89
+ <p className="mt-3 max-w-2xl text-base text-background/75">
90
+ {category.description}
91
+ </p>
92
+ )}
93
+ <p className="mt-4 text-[11px] font-mono uppercase tracking-[0.16em] text-background/60 tabular-nums">
94
+ {products.length} {products.length === 1 ? "product" : "products"}
95
+ </p>
96
+ </div>
97
+ </section>
98
+ <section className="max-w-7xl mx-auto px-6 sm:px-8 py-10 sm:py-12">
99
+ <ListingClient products={products} />
100
+ {products.length === 0 && (
101
+ <p className="text-center mt-8">
102
+ <Link href="/shop" className="text-primary font-semibold hover:underline">
103
+ ← Browse all products
104
+ </Link>
105
+ </p>
106
+ )}
107
+ </section>
108
+ </>
109
+ );
110
+ }
111
+
112
+ function CategorySkeleton() {
113
+ return (
114
+ <>
115
+ <section className="bg-foreground py-12 sm:py-14">
116
+ <div className="max-w-7xl mx-auto px-6 sm:px-8">
117
+ <div className="h-3 w-32 bg-background/20 rounded mb-3 animate-pulse" />
118
+ <div className="h-12 w-72 bg-background/20 rounded animate-pulse" />
119
+ </div>
120
+ </section>
121
+ <section className="max-w-7xl mx-auto px-6 sm:px-8 py-10 sm:py-12">
122
+ <div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-3 sm:gap-4">
123
+ {Array.from({ length: 8 }).map((_, i) => (
124
+ <div key={i} className="aspect-[4/3] bg-muted rounded-2xl animate-pulse" />
125
+ ))}
126
+ </div>
127
+ </section>
128
+ </>
129
+ );
130
+ }
@@ -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
+ }