@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,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,63 @@
1
+ import type { Metadata } from "next";
2
+ import { Suspense } from "react";
3
+ import { cacheTag, cacheLife } from "next/cache";
4
+ import { getServerClient, tags } from "@cimplify/sdk/server";
5
+ import { ShopClient } from "./shop-client";
6
+ import { brand } from "@/lib/brand";
7
+
8
+ export const metadata: Metadata = {
9
+ title: `Shop — ${brand.name}`,
10
+ description: brand.description,
11
+ };
12
+
13
+ async function getShopData() {
14
+ "use cache";
15
+ cacheTag(tags.products(), tags.categories());
16
+ cacheLife("hours");
17
+
18
+ const client = getServerClient();
19
+ const [p, c] = await Promise.all([
20
+ client.catalogue.getProducts({ limit: 50 }),
21
+ client.catalogue.getCategories(),
22
+ ]);
23
+ return {
24
+ products: p.ok ? p.value.items : [],
25
+ categories: c.ok ? c.value : [],
26
+ };
27
+ }
28
+
29
+ export default async function ShopPage() {
30
+ const { products, categories } = await getShopData();
31
+ return (
32
+ <>
33
+ <section className="bg-foreground text-background relative overflow-hidden">
34
+ <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]" />
35
+ <div className="relative max-w-7xl mx-auto px-6 sm:px-8 py-12 sm:py-14">
36
+ <p className="text-[11px] font-mono uppercase tracking-[0.2em] text-background/60 mb-2">
37
+ Catalogue · {products.length} products
38
+ </p>
39
+ <h1 className="text-[clamp(2rem,4vw,3rem)] font-bold m-0 -tracking-[0.025em] leading-[1.05]">
40
+ Every product we stock.
41
+ </h1>
42
+ <p className="mt-3 max-w-xl text-base text-background/75">
43
+ Filter, sort, search. Authorised dealer pricing, two-year warranty,
44
+ same-day Accra delivery on every order.
45
+ </p>
46
+ </div>
47
+ </section>
48
+ <Suspense
49
+ fallback={
50
+ <div className="max-w-7xl mx-auto px-6 sm:px-8 py-10">
51
+ <div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-3 sm:gap-4">
52
+ {Array.from({ length: 8 }).map((_, i) => (
53
+ <div key={i} className="aspect-[4/3] bg-muted rounded-2xl animate-pulse" />
54
+ ))}
55
+ </div>
56
+ </div>
57
+ }
58
+ >
59
+ <ShopClient products={products} categories={categories} />
60
+ </Suspense>
61
+ </>
62
+ );
63
+ }
@@ -0,0 +1,32 @@
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
+ * The page hero (in `app/shop/page.tsx`) already provides the title; we
13
+ * pass an empty `title` and override the SDK heading via className so the
14
+ * page reads as one continuous design.
15
+ */
16
+ export function ShopClient({
17
+ products,
18
+ categories,
19
+ }: {
20
+ products: Product[];
21
+ categories: Category[];
22
+ }) {
23
+ return (
24
+ <CataloguePage
25
+ title="All products"
26
+ products={products}
27
+ categories={categories}
28
+ renderCard={(p) => <StoreProductCard product={p} />}
29
+ className="max-w-7xl mx-auto px-6 sm:px-8 py-10 sm:py-12"
30
+ />
31
+ );
32
+ }
@@ -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
+ }
@@ -0,0 +1,167 @@
1
+ import type { Metadata } from "next";
2
+ import Link from "next/link";
3
+ import { cacheTag, cacheLife } from "next/cache";
4
+ import { getServerClient, tags, type Product } from "@cimplify/sdk/server";
5
+ import { brand } from "@/lib/brand";
6
+
7
+ export const metadata: Metadata = {
8
+ title: `Sitemap — ${brand.name}`,
9
+ description: "A human-readable index of every page on this site.",
10
+ };
11
+
12
+ interface SitemapData {
13
+ products: { slug: string; name: string }[];
14
+ categories: { slug: string; name: string }[];
15
+ collections: { slug: string; name: string }[];
16
+ }
17
+
18
+ async function getSitemap(): Promise<SitemapData> {
19
+ "use cache";
20
+ cacheTag(tags.products(), tags.categories(), tags.collections());
21
+ cacheLife("hours");
22
+
23
+ const client = getServerClient();
24
+ const [pRes, cRes, colRes] = await Promise.all([
25
+ client.catalogue.getProducts({ limit: 500 }),
26
+ client.catalogue.getCategories(),
27
+ client.catalogue.getCollections(),
28
+ ]);
29
+ return {
30
+ products: (pRes.ok ? pRes.value.items : []).map((p: Product) => ({
31
+ slug: p.slug ?? p.id,
32
+ name: p.name,
33
+ })),
34
+ categories: (cRes.ok ? cRes.value : []).map((c) => ({ slug: c.slug, name: c.name })),
35
+ collections: (colRes.ok ? colRes.value : []).map((c) => ({ slug: c.slug, name: c.name })),
36
+ };
37
+ }
38
+
39
+ const STATIC_LINKS: { title: string; links: { href: string; label: string }[] }[] = [
40
+ {
41
+ title: "Browse",
42
+ links: [
43
+ { href: "/", label: "Home" },
44
+ { href: "/shop", label: "Shop" },
45
+ { href: "/search", label: "Search" },
46
+ ],
47
+ },
48
+ {
49
+ title: "Account",
50
+ links: [
51
+ { href: "/account", label: "Account" },
52
+ { href: "/account/orders", label: "Orders" },
53
+ { href: "/account/addresses", label: "Addresses" },
54
+ { href: "/account/settings", label: "Settings" },
55
+ { href: "/login", label: "Sign in" },
56
+ { href: "/signup", label: "Create account" },
57
+ { href: "/track-order", label: "Track an order" },
58
+ { href: "/cart", label: "Cart" },
59
+ { href: "/checkout", label: "Checkout" },
60
+ ],
61
+ },
62
+ {
63
+ title: "About",
64
+ links: [
65
+ { href: "/about", label: "About" },
66
+ { href: "/faq", label: "FAQ" },
67
+ { href: "/contact", label: "Contact" },
68
+ ],
69
+ },
70
+ {
71
+ title: "Policies",
72
+ links: [
73
+ { href: "/shipping", label: "Shipping" },
74
+ { href: "/returns", label: "Returns" },
75
+ { href: "/accessibility", label: "Accessibility" },
76
+ { href: "/terms", label: "Terms of Service" },
77
+ { href: "/privacy", label: "Privacy Policy" },
78
+ ],
79
+ },
80
+ {
81
+ title: "Machine-readable",
82
+ links: [
83
+ { href: "/sitemap.xml", label: "sitemap.xml (search engines)" },
84
+ { href: "/llms.txt", label: "llms.txt (LLM agents)" },
85
+ { href: "/robots.txt", label: "robots.txt" },
86
+ { href: "/opensearch.xml", label: "opensearch.xml (browser search)" },
87
+ ],
88
+ },
89
+ ];
90
+
91
+ export default async function SitemapHtmlPage() {
92
+ const { products, categories, collections } = await getSitemap();
93
+
94
+ return (
95
+ <article className="max-w-5xl mx-auto px-6 sm:px-8 py-16">
96
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2">
97
+ Sitemap
98
+ </p>
99
+ <h1 className="text-[clamp(2rem,4vw,2.75rem)] font-semibold mb-2 -tracking-[0.02em]">
100
+ Every page, in one place.
101
+ </h1>
102
+ <p className="text-muted-foreground mb-12">
103
+ For search engines, see <Link href="/sitemap.xml" className="text-primary hover:underline">/sitemap.xml</Link>.
104
+ For LLM agents, see <Link href="/llms.txt" className="text-primary hover:underline">/llms.txt</Link>.
105
+ </p>
106
+ <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-10">
107
+ {STATIC_LINKS.map((s) => (
108
+ <Section key={s.title} title={s.title}>
109
+ {s.links.map((l) => (
110
+ <li key={l.href}>
111
+ <Link href={l.href} className="hover:text-primary transition-colors">
112
+ {l.label}
113
+ </Link>
114
+ </li>
115
+ ))}
116
+ </Section>
117
+ ))}
118
+ {categories.length > 0 && (
119
+ <Section title="Categories">
120
+ {categories.map((c) => (
121
+ <li key={c.slug}>
122
+ <Link href={`/categories/${c.slug}`} className="hover:text-primary transition-colors">
123
+ {c.name}
124
+ </Link>
125
+ </li>
126
+ ))}
127
+ </Section>
128
+ )}
129
+ {collections.length > 0 && (
130
+ <Section title="Collections">
131
+ {collections.map((c) => (
132
+ <li key={c.slug}>
133
+ <Link href={`/collections/${c.slug}`} className="hover:text-primary transition-colors">
134
+ {c.name}
135
+ </Link>
136
+ </li>
137
+ ))}
138
+ </Section>
139
+ )}
140
+ {products.length > 0 && (
141
+ <Section title={`Products (${products.length})`}>
142
+ {products.map((p) => (
143
+ <li key={p.slug}>
144
+ <Link href={`/shop?product=${encodeURIComponent(p.slug)}`} className="hover:text-primary transition-colors">
145
+ {p.name}
146
+ </Link>
147
+ </li>
148
+ ))}
149
+ </Section>
150
+ )}
151
+ </div>
152
+ </article>
153
+ );
154
+ }
155
+
156
+ function Section({ title, children }: { title: string; children: React.ReactNode }) {
157
+ return (
158
+ <div>
159
+ <p className="font-semibold text-[12px] uppercase tracking-[0.12em] text-foreground mb-3">
160
+ {title}
161
+ </p>
162
+ <ul className="space-y-2 m-0 p-0 list-none text-sm text-muted-foreground">
163
+ {children}
164
+ </ul>
165
+ </div>
166
+ );
167
+ }
@@ -0,0 +1,59 @@
1
+ import type { MetadataRoute } from "next";
2
+ import { getServerClient, type Product } from "@cimplify/sdk/server";
3
+
4
+ const SITE_URL =
5
+ process.env.NEXT_PUBLIC_SITE_URL?.trim() || "https://example.com";
6
+
7
+ const STATIC_ROUTES: { path: string; priority: number; changeFrequency: "daily" | "weekly" | "monthly" }[] = [
8
+ { path: "/", priority: 1.0, changeFrequency: "daily" },
9
+ { path: "/shop", priority: 0.9, changeFrequency: "daily" },
10
+ { path: "/about", priority: 0.5, changeFrequency: "monthly" },
11
+ { path: "/faq", priority: 0.4, changeFrequency: "monthly" },
12
+ { path: "/terms", priority: 0.2, changeFrequency: "monthly" },
13
+ { path: "/privacy", priority: 0.2, changeFrequency: "monthly" },
14
+ ];
15
+
16
+ export default async function sitemap(): Promise<MetadataRoute.Sitemap> {
17
+ const now = new Date();
18
+ const client = getServerClient();
19
+
20
+ const [productsRes, categoriesRes, collectionsRes] = await Promise.all([
21
+ client.catalogue.getProducts({ limit: 500 }),
22
+ client.catalogue.getCategories(),
23
+ client.catalogue.getCollections(),
24
+ ]);
25
+
26
+ const products = productsRes.ok ? productsRes.value.items : [];
27
+ const categories = categoriesRes.ok ? categoriesRes.value : [];
28
+ const collections = collectionsRes.ok ? collectionsRes.value : [];
29
+
30
+ const staticEntries: MetadataRoute.Sitemap = STATIC_ROUTES.map((r) => ({
31
+ url: `${SITE_URL}${r.path}`,
32
+ lastModified: now,
33
+ changeFrequency: r.changeFrequency,
34
+ priority: r.priority,
35
+ }));
36
+
37
+ const productEntries: MetadataRoute.Sitemap = products.map((p: Product) => ({
38
+ url: `${SITE_URL}/products/${p.slug ?? p.id}`,
39
+ lastModified: p.updated_at ? new Date(p.updated_at) : now,
40
+ changeFrequency: "weekly",
41
+ priority: 0.7,
42
+ }));
43
+
44
+ const categoryEntries: MetadataRoute.Sitemap = categories.map((c) => ({
45
+ url: `${SITE_URL}/categories/${c.slug}`,
46
+ lastModified: now,
47
+ changeFrequency: "weekly",
48
+ priority: 0.6,
49
+ }));
50
+
51
+ const collectionEntries: MetadataRoute.Sitemap = collections.map((c) => ({
52
+ url: `${SITE_URL}/collections/${c.slug}`,
53
+ lastModified: now,
54
+ changeFrequency: "weekly",
55
+ priority: 0.6,
56
+ }));
57
+
58
+ return [...staticEntries, ...categoryEntries, ...collectionEntries, ...productEntries];
59
+ }
@@ -0,0 +1,44 @@
1
+ import type { Metadata } from "next";
2
+ import { brand } from "@/lib/brand";
3
+
4
+ export const metadata: Metadata = {
5
+ title: `Terms of Service — ${brand.name}`,
6
+ description: `The rules of the road for ordering from ${brand.name}.`,
7
+ };
8
+
9
+ export default function TermsPage() {
10
+ const t = brand.terms;
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-mono uppercase tracking-[0.16em] text-primary mb-2 not-prose">
14
+ {t.eyebrow}
15
+ </p>
16
+ <h1 className="text-[clamp(2.25rem,5vw,3.5rem)] font-bold mb-2 -tracking-[0.025em]">
17
+ {t.title}
18
+ </h1>
19
+ <p className="text-sm text-muted-foreground not-prose mb-10">
20
+ Last updated: {t.lastUpdated}
21
+ </p>
22
+
23
+ <section className="space-y-5 leading-relaxed text-foreground/90">
24
+ {t.sections.map((s) => (
25
+ <div key={s.heading}>
26
+ <h2 className="text-2xl font-semibold mt-0 -tracking-[0.02em]">{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,24 @@
1
+ import type { Metadata } from "next";
2
+ import { TrackOrderForm } from "./track-order-form";
3
+ import { brand } from "@/lib/brand";
4
+
5
+ export const metadata: Metadata = {
6
+ title: `Track an order — ${brand.name}`,
7
+ description: brand.trackOrder.body,
8
+ };
9
+
10
+ export default function TrackOrderPage() {
11
+ const t = brand.trackOrder;
12
+ return (
13
+ <article className="max-w-2xl mx-auto px-6 sm:px-8 py-16">
14
+ <p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2">
15
+ {t.eyebrow}
16
+ </p>
17
+ <h1 className="font-serif text-[clamp(2rem,5vw,3rem)] font-semibold mb-4 -tracking-[0.02em]">
18
+ {t.title}
19
+ </h1>
20
+ <p className="text-muted-foreground leading-relaxed mb-8">{t.body}</p>
21
+ <TrackOrderForm />
22
+ </article>
23
+ );
24
+ }
@@ -0,0 +1,69 @@
1
+ "use client";
2
+
3
+ import { useState } from "react";
4
+ import { useRouter } from "next/navigation";
5
+
6
+ /**
7
+ * Guest order tracker. Routes /orders/<id> with the entered id; the
8
+ * post-checkout confirmation page already lives at /orders/[id] so this
9
+ * just sends the visitor there. Replace the redirect with a real lookup
10
+ * (e.g. Cimplify orders API + email-match guard) for production.
11
+ */
12
+ export function TrackOrderForm() {
13
+ const router = useRouter();
14
+ const [orderId, setOrderId] = useState("");
15
+ const [email, setEmail] = useState("");
16
+
17
+ return (
18
+ <form
19
+ onSubmit={(e) => {
20
+ e.preventDefault();
21
+ if (!orderId.trim()) return;
22
+ router.push(`/orders/${encodeURIComponent(orderId.trim())}`);
23
+ }}
24
+ className="space-y-4 rounded-2xl border border-border bg-card p-6"
25
+ >
26
+ <div>
27
+ <label
28
+ htmlFor="orderId"
29
+ className="text-xs font-semibold uppercase tracking-wider text-muted-foreground block mb-1.5"
30
+ >
31
+ Order number
32
+ </label>
33
+ <input
34
+ id="orderId"
35
+ name="orderId"
36
+ required
37
+ value={orderId}
38
+ onChange={(e) => setOrderId(e.target.value)}
39
+ placeholder="ord_abc123…"
40
+ className="w-full px-4 py-3 rounded-md bg-background border border-border focus:border-primary focus:ring-2 focus:ring-primary/20 outline-none text-sm"
41
+ />
42
+ </div>
43
+ <div>
44
+ <label
45
+ htmlFor="email"
46
+ className="text-xs font-semibold uppercase tracking-wider text-muted-foreground block mb-1.5"
47
+ >
48
+ Order email
49
+ </label>
50
+ <input
51
+ id="email"
52
+ name="email"
53
+ type="email"
54
+ required
55
+ value={email}
56
+ onChange={(e) => setEmail(e.target.value)}
57
+ placeholder="you@email.com"
58
+ className="w-full px-4 py-3 rounded-md bg-background border border-border focus:border-primary focus:ring-2 focus:ring-primary/20 outline-none text-sm"
59
+ />
60
+ </div>
61
+ <button
62
+ type="submit"
63
+ className="w-full inline-flex items-center justify-center px-5 py-3 rounded-md bg-primary text-primary-foreground font-semibold text-sm hover:bg-primary/90 transition-colors"
64
+ >
65
+ Track order
66
+ </button>
67
+ </form>
68
+ );
69
+ }
@@ -0,0 +1,13 @@
1
+ "use client";
2
+
3
+ import { CimplifyAccount } from "@cimplify/sdk/react";
4
+
5
+ /**
6
+ * Cimplify Account portal — iframe-mounted UI hosted by Cimplify Link.
7
+ * Handles sign-in, sign-up, OTP, addresses, payment methods, sessions,
8
+ * and order history. The iframe owns auth state; we just choose which
9
+ * `section` to land on.
10
+ */
11
+ export function AccountIframe({ section }: { section?: string }) {
12
+ return <CimplifyAccount section={section} />;
13
+ }
@@ -0,0 +1,85 @@
1
+ import Link from "next/link";
2
+ import { brand } from "@/lib/brand";
3
+
4
+ /**
5
+ * Auto-specific hero. Carbon-dark with a single sport-red accent line.
6
+ * Right column carries the four info chips that anchor trust for an
7
+ * auto-parts merchant: genuine OEM, same-day fitting partners, warranty,
8
+ * cash-on-delivery. The fitment finder sits directly below.
9
+ */
10
+ export function AutoHero() {
11
+ return (
12
+ <section className="relative overflow-hidden bg-foreground text-background pb-20 sm:pb-28">
13
+ <div className="absolute inset-0 opacity-[0.05] pointer-events-none [background-image:linear-gradient(transparent_0%,transparent_calc(50%-0.5px),rgba(255,255,255,0.6)_50%,transparent_calc(50%+0.5px),transparent_100%)] [background-size:48px_48px]" />
14
+ <div className="absolute -top-32 -right-24 w-[28rem] h-[28rem] rounded-full bg-primary/25 blur-[120px] pointer-events-none" />
15
+ <div className="absolute top-0 left-0 right-0 h-1 bg-primary" />
16
+
17
+ <div className="relative max-w-7xl mx-auto px-6 sm:px-8 pt-14 sm:pt-20">
18
+ <div className="grid grid-cols-1 lg:grid-cols-[1.4fr_1fr] gap-10 lg:gap-14 items-end">
19
+ <div>
20
+ <span className="inline-flex items-center gap-2 mb-6 px-3 py-1.5 rounded-full bg-primary/15 border border-primary/40 text-background text-[11px] font-mono uppercase tracking-[0.18em]">
21
+ <span className="grid place-items-center w-1.5 h-1.5 rounded-full bg-primary animate-pulse" />
22
+ {brand.hero.badge}
23
+ </span>
24
+ <h1 className="text-[clamp(2.5rem,6.4vw,5.25rem)] font-bold m-0 -tracking-[0.04em] leading-[0.98]">
25
+ {brand.hero.title}
26
+ </h1>
27
+ <p className="text-base sm:text-lg text-background/75 leading-relaxed max-w-xl mt-6">
28
+ {brand.hero.subtitle}
29
+ </p>
30
+ <div className="flex flex-wrap items-center gap-3 mt-8">
31
+ <Link
32
+ href="/shop"
33
+ className="inline-flex items-center gap-2 px-6 py-3 rounded-full bg-primary text-primary-foreground text-sm font-semibold hover:bg-primary/90 transition-colors"
34
+ >
35
+ {brand.hero.primaryCtaLabel}
36
+ <ArrowIcon />
37
+ </Link>
38
+ {brand.hero.secondaryCtaLabel && brand.hero.secondaryCtaHref && (
39
+ <Link
40
+ href={brand.hero.secondaryCtaHref}
41
+ className="inline-flex items-center gap-2 px-6 py-3 rounded-full border border-background/30 text-background text-sm font-semibold hover:bg-background/10 transition-colors"
42
+ >
43
+ {brand.hero.secondaryCtaLabel}
44
+ </Link>
45
+ )}
46
+ </div>
47
+ </div>
48
+
49
+ <aside className="grid grid-cols-2 gap-3 sm:gap-4 lg:max-w-md">
50
+ <InfoChip eyebrow="Genuine" value="OEM & OEM-grade" />
51
+ <InfoChip eyebrow="Same-day" value="Fitting partners across Accra" />
52
+ <InfoChip eyebrow="Warranty" value="12 months · parts & labour" />
53
+ <InfoChip eyebrow="Pay your way" value="MoMo · card · cash" />
54
+ </aside>
55
+ </div>
56
+ </div>
57
+ </section>
58
+ );
59
+ }
60
+
61
+ interface InfoChipProps {
62
+ eyebrow: string;
63
+ value: string;
64
+ }
65
+
66
+ function InfoChip({ eyebrow, value }: InfoChipProps) {
67
+ return (
68
+ <div className="rounded-2xl border border-background/15 bg-background/5 backdrop-blur-sm p-4">
69
+ <p className="text-[10px] font-mono uppercase tracking-[0.18em] text-background/55 mb-1.5">
70
+ {eyebrow}
71
+ </p>
72
+ <p className="text-sm font-semibold text-background -tracking-[0.01em] leading-snug">
73
+ {value}
74
+ </p>
75
+ </div>
76
+ );
77
+ }
78
+
79
+ function ArrowIcon() {
80
+ return (
81
+ <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" className="w-3.5 h-3.5" aria-hidden>
82
+ <path d="M5 12h14M13 6l6 6-6 6" strokeLinecap="round" strokeLinejoin="round" />
83
+ </svg>
84
+ );
85
+ }