@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.
- package/README.md +69 -0
- package/dist/add-ZJQJZJEF.mjs +125 -0
- package/dist/chunk-4YSOZ6LY.mjs +61 -0
- package/dist/chunk-4ZVTPMUZ.mjs +155 -0
- package/dist/chunk-5XH72JMJ.mjs +211 -0
- package/dist/chunk-D7D75ONX.mjs +36 -0
- package/dist/chunk-JJYWETGA.mjs +127 -0
- package/dist/chunk-L6474RPL.mjs +37 -0
- package/dist/chunk-MQMNWLMU.mjs +48 -0
- package/dist/chunk-NZ4RG62Z.mjs +141 -0
- package/dist/chunk-TAMGCHIL.mjs +81 -0
- package/dist/chunk-XSWWWO6H.mjs +3779 -0
- package/dist/deploy-WCZOGMED.mjs +206 -0
- package/dist/dev-4HKIXWXX.mjs +130 -0
- package/dist/dispatcher.mjs +321 -0
- package/dist/domains-3RJ4T5IX.mjs +387 -0
- package/dist/env-LBYBCBWV.mjs +268 -0
- package/dist/link-SEJNW7JS.mjs +45 -0
- package/dist/list-D4JC2VWY.mjs +48 -0
- package/dist/login-MRYWLQRY.mjs +276 -0
- package/dist/logout-ZFZLSJ32.mjs +16 -0
- package/dist/logs-LK7CMBCE.mjs +95 -0
- package/dist/projects-QJUGOCQZ.mjs +164 -0
- package/dist/repo-MV22OHON.mjs +8 -0
- package/dist/rollback-XO7RIG2A.mjs +107 -0
- package/dist/status-7CMVLD54.mjs +127 -0
- package/dist/unlink-5ABCT7B6.mjs +16 -0
- package/dist/whoami-INHDUHWA.mjs +24 -0
- package/package.json +44 -0
- package/templates/storefront-bakery/.claude/skills/cimplify-storefront/SKILL.md +145 -0
- package/templates/storefront-bakery/.cursor/rules/cimplify-storefront.mdc +25 -0
- package/templates/storefront-bakery/.env.example +16 -0
- package/templates/storefront-bakery/AGENTS.md +120 -0
- package/templates/storefront-bakery/CLAUDE.md +22 -0
- package/templates/storefront-bakery/README.md +73 -0
- package/templates/storefront-bakery/__tests__/brand.test.ts +4 -0
- package/templates/storefront-bakery/__tests__/cart-flow.test.ts +4 -0
- package/templates/storefront-bakery/__tests__/contract.test.ts +4 -0
- package/templates/storefront-bakery/app/about/page.tsx +38 -0
- package/templates/storefront-bakery/app/accessibility/page.tsx +11 -0
- package/templates/storefront-bakery/app/account/addresses/page.tsx +21 -0
- package/templates/storefront-bakery/app/account/orders/page.tsx +21 -0
- package/templates/storefront-bakery/app/account/page.tsx +22 -0
- package/templates/storefront-bakery/app/account/settings/page.tsx +21 -0
- package/templates/storefront-bakery/app/cart/page.tsx +9 -0
- package/templates/storefront-bakery/app/categories/[slug]/listing-client.tsx +19 -0
- package/templates/storefront-bakery/app/categories/[slug]/page.tsx +118 -0
- package/templates/storefront-bakery/app/checkout/page.tsx +17 -0
- package/templates/storefront-bakery/app/collections/[slug]/listing-client.tsx +20 -0
- package/templates/storefront-bakery/app/collections/[slug]/page.tsx +118 -0
- package/templates/storefront-bakery/app/contact/contact-form.tsx +109 -0
- package/templates/storefront-bakery/app/contact/page.tsx +54 -0
- package/templates/storefront-bakery/app/error.tsx +60 -0
- package/templates/storefront-bakery/app/faq/page.tsx +46 -0
- package/templates/storefront-bakery/app/globals.css +47 -0
- package/templates/storefront-bakery/app/layout.tsx +82 -0
- package/templates/storefront-bakery/app/llms.txt/route.ts +94 -0
- package/templates/storefront-bakery/app/login/page.tsx +17 -0
- package/templates/storefront-bakery/app/not-found.tsx +39 -0
- package/templates/storefront-bakery/app/opensearch.xml/route.ts +37 -0
- package/templates/storefront-bakery/app/orders/[id]/page.tsx +21 -0
- package/templates/storefront-bakery/app/page.tsx +97 -0
- package/templates/storefront-bakery/app/privacy/page.tsx +44 -0
- package/templates/storefront-bakery/app/returns/page.tsx +11 -0
- package/templates/storefront-bakery/app/robots.ts +18 -0
- package/templates/storefront-bakery/app/search/page.tsx +38 -0
- package/templates/storefront-bakery/app/search/search-client.tsx +7 -0
- package/templates/storefront-bakery/app/shipping/page.tsx +16 -0
- package/templates/storefront-bakery/app/shop/page.tsx +31 -0
- package/templates/storefront-bakery/app/shop/shop-client.tsx +27 -0
- package/templates/storefront-bakery/app/signup/page.tsx +17 -0
- package/templates/storefront-bakery/app/sitemap-page/page.tsx +167 -0
- package/templates/storefront-bakery/app/sitemap.ts +62 -0
- package/templates/storefront-bakery/app/terms/page.tsx +44 -0
- package/templates/storefront-bakery/app/track-order/page.tsx +24 -0
- package/templates/storefront-bakery/app/track-order/track-order-form.tsx +69 -0
- package/templates/storefront-bakery/components/account-iframe.tsx +13 -0
- package/templates/storefront-bakery/components/cart-drawer.tsx +14 -0
- package/templates/storefront-bakery/components/cart-pill.tsx +36 -0
- package/templates/storefront-bakery/components/category-grid.tsx +28 -0
- package/templates/storefront-bakery/components/collection-strip.tsx +45 -0
- package/templates/storefront-bakery/components/footer.tsx +148 -0
- package/templates/storefront-bakery/components/header.tsx +43 -0
- package/templates/storefront-bakery/components/hero.tsx +25 -0
- package/templates/storefront-bakery/components/nav-link.tsx +20 -0
- package/templates/storefront-bakery/components/policy-page.tsx +49 -0
- package/templates/storefront-bakery/components/product-modal.tsx +104 -0
- package/templates/storefront-bakery/components/providers.tsx +35 -0
- package/templates/storefront-bakery/components/store-product-card.tsx +87 -0
- package/templates/storefront-bakery/lib/brand.ts +570 -0
- package/templates/storefront-bakery/lib/cart.ts +12 -0
- package/templates/storefront-bakery/next.config.ts +42 -0
- package/templates/storefront-bakery/package.json +35 -0
- package/templates/storefront-bakery/postcss.config.mjs +7 -0
- package/templates/storefront-bakery/tsconfig.json +23 -0
- package/templates/storefront-bakery/vitest.config.ts +9 -0
- package/templates/storefront-fashion/.claude/skills/cimplify-storefront/SKILL.md +145 -0
- package/templates/storefront-fashion/.cursor/rules/cimplify-storefront.mdc +25 -0
- package/templates/storefront-fashion/.env.example +16 -0
- package/templates/storefront-fashion/AGENTS.md +126 -0
- package/templates/storefront-fashion/CLAUDE.md +22 -0
- package/templates/storefront-fashion/README.md +77 -0
- package/templates/storefront-fashion/__tests__/brand.test.ts +4 -0
- package/templates/storefront-fashion/__tests__/cart-flow.test.ts +55 -0
- package/templates/storefront-fashion/__tests__/contract.test.ts +4 -0
- package/templates/storefront-fashion/app/about/page.tsx +41 -0
- package/templates/storefront-fashion/app/accessibility/page.tsx +11 -0
- package/templates/storefront-fashion/app/account/addresses/page.tsx +21 -0
- package/templates/storefront-fashion/app/account/orders/page.tsx +21 -0
- package/templates/storefront-fashion/app/account/page.tsx +22 -0
- package/templates/storefront-fashion/app/account/settings/page.tsx +21 -0
- package/templates/storefront-fashion/app/cart/page.tsx +9 -0
- package/templates/storefront-fashion/app/categories/[slug]/listing-client.tsx +19 -0
- package/templates/storefront-fashion/app/categories/[slug]/page.tsx +130 -0
- package/templates/storefront-fashion/app/checkout/page.tsx +17 -0
- package/templates/storefront-fashion/app/collections/[slug]/listing-client.tsx +20 -0
- package/templates/storefront-fashion/app/collections/[slug]/page.tsx +130 -0
- package/templates/storefront-fashion/app/contact/contact-form.tsx +109 -0
- package/templates/storefront-fashion/app/contact/page.tsx +54 -0
- package/templates/storefront-fashion/app/error.tsx +61 -0
- package/templates/storefront-fashion/app/faq/page.tsx +46 -0
- package/templates/storefront-fashion/app/globals.css +46 -0
- package/templates/storefront-fashion/app/layout.tsx +78 -0
- package/templates/storefront-fashion/app/llms.txt/route.ts +94 -0
- package/templates/storefront-fashion/app/login/page.tsx +17 -0
- package/templates/storefront-fashion/app/lookbook/page.tsx +132 -0
- package/templates/storefront-fashion/app/not-found.tsx +39 -0
- package/templates/storefront-fashion/app/opensearch.xml/route.ts +37 -0
- package/templates/storefront-fashion/app/orders/[id]/page.tsx +24 -0
- package/templates/storefront-fashion/app/page.tsx +183 -0
- package/templates/storefront-fashion/app/privacy/page.tsx +44 -0
- package/templates/storefront-fashion/app/products/[slug]/page.tsx +165 -0
- package/templates/storefront-fashion/app/products/[slug]/product-detail.tsx +70 -0
- package/templates/storefront-fashion/app/returns/page.tsx +11 -0
- package/templates/storefront-fashion/app/robots.ts +18 -0
- package/templates/storefront-fashion/app/search/page.tsx +38 -0
- package/templates/storefront-fashion/app/search/search-client.tsx +7 -0
- package/templates/storefront-fashion/app/shipping/page.tsx +16 -0
- package/templates/storefront-fashion/app/shop/page.tsx +63 -0
- package/templates/storefront-fashion/app/shop/shop-client.tsx +32 -0
- package/templates/storefront-fashion/app/signup/page.tsx +17 -0
- package/templates/storefront-fashion/app/sitemap-page/page.tsx +167 -0
- package/templates/storefront-fashion/app/sitemap.ts +59 -0
- package/templates/storefront-fashion/app/size-guide/page.tsx +155 -0
- package/templates/storefront-fashion/app/terms/page.tsx +44 -0
- package/templates/storefront-fashion/app/track-order/page.tsx +24 -0
- package/templates/storefront-fashion/app/track-order/track-order-form.tsx +69 -0
- package/templates/storefront-fashion/components/account-iframe.tsx +13 -0
- package/templates/storefront-fashion/components/brand-marquee.tsx +27 -0
- package/templates/storefront-fashion/components/cart-drawer.tsx +14 -0
- package/templates/storefront-fashion/components/cart-pill.tsx +36 -0
- package/templates/storefront-fashion/components/category-grid.tsx +28 -0
- package/templates/storefront-fashion/components/category-tiles.tsx +104 -0
- package/templates/storefront-fashion/components/collection-strip.tsx +45 -0
- package/templates/storefront-fashion/components/feature-hero.tsx +82 -0
- package/templates/storefront-fashion/components/footer.tsx +153 -0
- package/templates/storefront-fashion/components/header.tsx +43 -0
- package/templates/storefront-fashion/components/hero.tsx +28 -0
- package/templates/storefront-fashion/components/nav-link.tsx +20 -0
- package/templates/storefront-fashion/components/newsletter.tsx +50 -0
- package/templates/storefront-fashion/components/policy-page.tsx +49 -0
- package/templates/storefront-fashion/components/promo-banner.tsx +41 -0
- package/templates/storefront-fashion/components/providers.tsx +35 -0
- package/templates/storefront-fashion/components/section-heading.tsx +37 -0
- package/templates/storefront-fashion/components/store-product-card.tsx +87 -0
- package/templates/storefront-fashion/components/trade-in-cta.tsx +54 -0
- package/templates/storefront-fashion/components/trust-bar.tsx +66 -0
- package/templates/storefront-fashion/e2e/visual.spec.ts +52 -0
- package/templates/storefront-fashion/lib/brand.ts +518 -0
- package/templates/storefront-fashion/lib/cart.ts +12 -0
- package/templates/storefront-fashion/next.config.ts +42 -0
- package/templates/storefront-fashion/package.json +38 -0
- package/templates/storefront-fashion/playwright.config.ts +48 -0
- package/templates/storefront-fashion/postcss.config.mjs +7 -0
- package/templates/storefront-fashion/tsconfig.json +23 -0
- package/templates/storefront-fashion/vitest.config.ts +9 -0
- package/templates/storefront-grocery/.claude/skills/cimplify-storefront/SKILL.md +145 -0
- package/templates/storefront-grocery/.cursor/rules/cimplify-storefront.mdc +25 -0
- package/templates/storefront-grocery/.env.example +16 -0
- package/templates/storefront-grocery/AGENTS.md +96 -0
- package/templates/storefront-grocery/CLAUDE.md +22 -0
- package/templates/storefront-grocery/README.md +73 -0
- package/templates/storefront-grocery/__tests__/brand.test.ts +4 -0
- package/templates/storefront-grocery/__tests__/cart-flow.test.ts +4 -0
- package/templates/storefront-grocery/__tests__/contract.test.ts +4 -0
- package/templates/storefront-grocery/app/about/page.tsx +38 -0
- package/templates/storefront-grocery/app/accessibility/page.tsx +11 -0
- package/templates/storefront-grocery/app/account/addresses/page.tsx +21 -0
- package/templates/storefront-grocery/app/account/orders/page.tsx +21 -0
- package/templates/storefront-grocery/app/account/page.tsx +22 -0
- package/templates/storefront-grocery/app/account/settings/page.tsx +21 -0
- package/templates/storefront-grocery/app/cart/page.tsx +9 -0
- package/templates/storefront-grocery/app/categories/[slug]/listing-client.tsx +19 -0
- package/templates/storefront-grocery/app/categories/[slug]/page.tsx +118 -0
- package/templates/storefront-grocery/app/checkout/page.tsx +17 -0
- package/templates/storefront-grocery/app/collections/[slug]/listing-client.tsx +20 -0
- package/templates/storefront-grocery/app/collections/[slug]/page.tsx +118 -0
- package/templates/storefront-grocery/app/contact/contact-form.tsx +109 -0
- package/templates/storefront-grocery/app/contact/page.tsx +54 -0
- package/templates/storefront-grocery/app/error.tsx +60 -0
- package/templates/storefront-grocery/app/faq/page.tsx +46 -0
- package/templates/storefront-grocery/app/globals.css +45 -0
- package/templates/storefront-grocery/app/layout.tsx +77 -0
- package/templates/storefront-grocery/app/llms.txt/route.ts +94 -0
- package/templates/storefront-grocery/app/login/page.tsx +17 -0
- package/templates/storefront-grocery/app/not-found.tsx +39 -0
- package/templates/storefront-grocery/app/opensearch.xml/route.ts +37 -0
- package/templates/storefront-grocery/app/orders/[id]/page.tsx +21 -0
- package/templates/storefront-grocery/app/page.tsx +97 -0
- package/templates/storefront-grocery/app/privacy/page.tsx +44 -0
- package/templates/storefront-grocery/app/returns/page.tsx +11 -0
- package/templates/storefront-grocery/app/robots.ts +18 -0
- package/templates/storefront-grocery/app/search/page.tsx +38 -0
- package/templates/storefront-grocery/app/search/search-client.tsx +7 -0
- package/templates/storefront-grocery/app/shipping/page.tsx +16 -0
- package/templates/storefront-grocery/app/shop/page.tsx +31 -0
- package/templates/storefront-grocery/app/shop/shop-client.tsx +27 -0
- package/templates/storefront-grocery/app/signup/page.tsx +17 -0
- package/templates/storefront-grocery/app/sitemap-page/page.tsx +167 -0
- package/templates/storefront-grocery/app/sitemap.ts +62 -0
- package/templates/storefront-grocery/app/terms/page.tsx +44 -0
- package/templates/storefront-grocery/app/track-order/page.tsx +24 -0
- package/templates/storefront-grocery/app/track-order/track-order-form.tsx +69 -0
- package/templates/storefront-grocery/components/account-iframe.tsx +13 -0
- package/templates/storefront-grocery/components/cart-drawer.tsx +14 -0
- package/templates/storefront-grocery/components/cart-pill.tsx +36 -0
- package/templates/storefront-grocery/components/category-grid.tsx +28 -0
- package/templates/storefront-grocery/components/collection-strip.tsx +45 -0
- package/templates/storefront-grocery/components/footer.tsx +148 -0
- package/templates/storefront-grocery/components/header.tsx +43 -0
- package/templates/storefront-grocery/components/hero.tsx +25 -0
- package/templates/storefront-grocery/components/nav-link.tsx +20 -0
- package/templates/storefront-grocery/components/policy-page.tsx +49 -0
- package/templates/storefront-grocery/components/product-modal.tsx +104 -0
- package/templates/storefront-grocery/components/providers.tsx +35 -0
- package/templates/storefront-grocery/components/store-product-card.tsx +87 -0
- package/templates/storefront-grocery/lib/brand.ts +375 -0
- package/templates/storefront-grocery/lib/cart.ts +12 -0
- package/templates/storefront-grocery/next.config.ts +42 -0
- package/templates/storefront-grocery/package.json +35 -0
- package/templates/storefront-grocery/postcss.config.mjs +7 -0
- package/templates/storefront-grocery/tsconfig.json +23 -0
- package/templates/storefront-grocery/vitest.config.ts +9 -0
- package/templates/storefront-restaurant/.claude/skills/cimplify-storefront/SKILL.md +145 -0
- package/templates/storefront-restaurant/.cursor/rules/cimplify-storefront.mdc +25 -0
- package/templates/storefront-restaurant/.env.example +16 -0
- package/templates/storefront-restaurant/AGENTS.md +102 -0
- package/templates/storefront-restaurant/CLAUDE.md +22 -0
- package/templates/storefront-restaurant/README.md +73 -0
- package/templates/storefront-restaurant/__tests__/brand.test.ts +4 -0
- package/templates/storefront-restaurant/__tests__/cart-flow.test.ts +4 -0
- package/templates/storefront-restaurant/__tests__/contract.test.ts +4 -0
- package/templates/storefront-restaurant/app/about/page.tsx +38 -0
- package/templates/storefront-restaurant/app/accessibility/page.tsx +11 -0
- package/templates/storefront-restaurant/app/account/addresses/page.tsx +21 -0
- package/templates/storefront-restaurant/app/account/orders/page.tsx +21 -0
- package/templates/storefront-restaurant/app/account/page.tsx +22 -0
- package/templates/storefront-restaurant/app/account/settings/page.tsx +21 -0
- package/templates/storefront-restaurant/app/cart/page.tsx +9 -0
- package/templates/storefront-restaurant/app/categories/[slug]/listing-client.tsx +19 -0
- package/templates/storefront-restaurant/app/categories/[slug]/page.tsx +118 -0
- package/templates/storefront-restaurant/app/checkout/page.tsx +17 -0
- package/templates/storefront-restaurant/app/collections/[slug]/listing-client.tsx +20 -0
- package/templates/storefront-restaurant/app/collections/[slug]/page.tsx +118 -0
- package/templates/storefront-restaurant/app/contact/contact-form.tsx +109 -0
- package/templates/storefront-restaurant/app/contact/page.tsx +54 -0
- package/templates/storefront-restaurant/app/error.tsx +60 -0
- package/templates/storefront-restaurant/app/faq/page.tsx +46 -0
- package/templates/storefront-restaurant/app/globals.css +44 -0
- package/templates/storefront-restaurant/app/layout.tsx +82 -0
- package/templates/storefront-restaurant/app/llms.txt/route.ts +94 -0
- package/templates/storefront-restaurant/app/login/page.tsx +17 -0
- package/templates/storefront-restaurant/app/not-found.tsx +39 -0
- package/templates/storefront-restaurant/app/opensearch.xml/route.ts +37 -0
- package/templates/storefront-restaurant/app/orders/[id]/page.tsx +21 -0
- package/templates/storefront-restaurant/app/page.tsx +97 -0
- package/templates/storefront-restaurant/app/privacy/page.tsx +44 -0
- package/templates/storefront-restaurant/app/reservations/page.tsx +66 -0
- package/templates/storefront-restaurant/app/reservations/reservations-client.tsx +234 -0
- package/templates/storefront-restaurant/app/returns/page.tsx +11 -0
- package/templates/storefront-restaurant/app/robots.ts +18 -0
- package/templates/storefront-restaurant/app/search/page.tsx +38 -0
- package/templates/storefront-restaurant/app/search/search-client.tsx +7 -0
- package/templates/storefront-restaurant/app/shipping/page.tsx +16 -0
- package/templates/storefront-restaurant/app/shop/page.tsx +31 -0
- package/templates/storefront-restaurant/app/shop/shop-client.tsx +27 -0
- package/templates/storefront-restaurant/app/signup/page.tsx +17 -0
- package/templates/storefront-restaurant/app/sitemap-page/page.tsx +167 -0
- package/templates/storefront-restaurant/app/sitemap.ts +62 -0
- package/templates/storefront-restaurant/app/terms/page.tsx +44 -0
- package/templates/storefront-restaurant/app/track-order/page.tsx +24 -0
- package/templates/storefront-restaurant/app/track-order/track-order-form.tsx +69 -0
- package/templates/storefront-restaurant/components/account-iframe.tsx +13 -0
- package/templates/storefront-restaurant/components/cart-drawer.tsx +14 -0
- package/templates/storefront-restaurant/components/cart-pill.tsx +36 -0
- package/templates/storefront-restaurant/components/category-grid.tsx +28 -0
- package/templates/storefront-restaurant/components/collection-strip.tsx +45 -0
- package/templates/storefront-restaurant/components/footer.tsx +148 -0
- package/templates/storefront-restaurant/components/header.tsx +43 -0
- package/templates/storefront-restaurant/components/hero.tsx +25 -0
- package/templates/storefront-restaurant/components/nav-link.tsx +20 -0
- package/templates/storefront-restaurant/components/policy-page.tsx +49 -0
- package/templates/storefront-restaurant/components/product-modal.tsx +104 -0
- package/templates/storefront-restaurant/components/providers.tsx +35 -0
- package/templates/storefront-restaurant/components/store-product-card.tsx +87 -0
- package/templates/storefront-restaurant/lib/brand.ts +377 -0
- package/templates/storefront-restaurant/lib/cart.ts +12 -0
- package/templates/storefront-restaurant/next.config.ts +42 -0
- package/templates/storefront-restaurant/package.json +35 -0
- package/templates/storefront-restaurant/postcss.config.mjs +7 -0
- package/templates/storefront-restaurant/tsconfig.json +23 -0
- package/templates/storefront-restaurant/vitest.config.ts +9 -0
- package/templates/storefront-retail/.claude/skills/cimplify-storefront/SKILL.md +145 -0
- package/templates/storefront-retail/.cursor/rules/cimplify-storefront.mdc +25 -0
- package/templates/storefront-retail/.env.example +16 -0
- package/templates/storefront-retail/AGENTS.md +117 -0
- package/templates/storefront-retail/CLAUDE.md +22 -0
- package/templates/storefront-retail/README.md +77 -0
- package/templates/storefront-retail/__tests__/brand.test.ts +4 -0
- package/templates/storefront-retail/__tests__/cart-flow.test.ts +4 -0
- package/templates/storefront-retail/__tests__/contract.test.ts +4 -0
- package/templates/storefront-retail/app/about/page.tsx +41 -0
- package/templates/storefront-retail/app/accessibility/page.tsx +11 -0
- package/templates/storefront-retail/app/account/addresses/page.tsx +21 -0
- package/templates/storefront-retail/app/account/orders/page.tsx +21 -0
- package/templates/storefront-retail/app/account/page.tsx +22 -0
- package/templates/storefront-retail/app/account/settings/page.tsx +21 -0
- package/templates/storefront-retail/app/cart/page.tsx +9 -0
- package/templates/storefront-retail/app/categories/[slug]/listing-client.tsx +19 -0
- package/templates/storefront-retail/app/categories/[slug]/page.tsx +130 -0
- package/templates/storefront-retail/app/checkout/page.tsx +17 -0
- package/templates/storefront-retail/app/collections/[slug]/listing-client.tsx +20 -0
- package/templates/storefront-retail/app/collections/[slug]/page.tsx +130 -0
- package/templates/storefront-retail/app/contact/contact-form.tsx +109 -0
- package/templates/storefront-retail/app/contact/page.tsx +54 -0
- package/templates/storefront-retail/app/error.tsx +61 -0
- package/templates/storefront-retail/app/faq/page.tsx +46 -0
- package/templates/storefront-retail/app/globals.css +47 -0
- package/templates/storefront-retail/app/layout.tsx +77 -0
- package/templates/storefront-retail/app/llms.txt/route.ts +94 -0
- package/templates/storefront-retail/app/login/page.tsx +17 -0
- package/templates/storefront-retail/app/not-found.tsx +39 -0
- package/templates/storefront-retail/app/opensearch.xml/route.ts +37 -0
- package/templates/storefront-retail/app/orders/[id]/page.tsx +24 -0
- package/templates/storefront-retail/app/page.tsx +182 -0
- package/templates/storefront-retail/app/privacy/page.tsx +44 -0
- package/templates/storefront-retail/app/products/[slug]/page.tsx +165 -0
- package/templates/storefront-retail/app/products/[slug]/product-detail.tsx +70 -0
- package/templates/storefront-retail/app/returns/page.tsx +11 -0
- package/templates/storefront-retail/app/robots.ts +18 -0
- package/templates/storefront-retail/app/search/page.tsx +38 -0
- package/templates/storefront-retail/app/search/search-client.tsx +7 -0
- package/templates/storefront-retail/app/shipping/page.tsx +16 -0
- package/templates/storefront-retail/app/shop/page.tsx +63 -0
- package/templates/storefront-retail/app/shop/shop-client.tsx +32 -0
- package/templates/storefront-retail/app/signup/page.tsx +17 -0
- package/templates/storefront-retail/app/sitemap-page/page.tsx +167 -0
- package/templates/storefront-retail/app/sitemap.ts +59 -0
- package/templates/storefront-retail/app/terms/page.tsx +44 -0
- package/templates/storefront-retail/app/track-order/page.tsx +24 -0
- package/templates/storefront-retail/app/track-order/track-order-form.tsx +69 -0
- package/templates/storefront-retail/components/account-iframe.tsx +13 -0
- package/templates/storefront-retail/components/brand-marquee.tsx +27 -0
- package/templates/storefront-retail/components/cart-drawer.tsx +14 -0
- package/templates/storefront-retail/components/cart-pill.tsx +36 -0
- package/templates/storefront-retail/components/category-grid.tsx +28 -0
- package/templates/storefront-retail/components/category-tiles.tsx +104 -0
- package/templates/storefront-retail/components/collection-strip.tsx +45 -0
- package/templates/storefront-retail/components/feature-hero.tsx +84 -0
- package/templates/storefront-retail/components/footer.tsx +153 -0
- package/templates/storefront-retail/components/header.tsx +45 -0
- package/templates/storefront-retail/components/hero.tsx +28 -0
- package/templates/storefront-retail/components/nav-link.tsx +20 -0
- package/templates/storefront-retail/components/newsletter.tsx +50 -0
- package/templates/storefront-retail/components/policy-page.tsx +49 -0
- package/templates/storefront-retail/components/promo-banner.tsx +41 -0
- package/templates/storefront-retail/components/providers.tsx +35 -0
- package/templates/storefront-retail/components/section-heading.tsx +37 -0
- package/templates/storefront-retail/components/store-product-card.tsx +87 -0
- package/templates/storefront-retail/components/trade-in-cta.tsx +54 -0
- package/templates/storefront-retail/components/trust-bar.tsx +66 -0
- package/templates/storefront-retail/lib/brand.ts +664 -0
- package/templates/storefront-retail/lib/cart.ts +12 -0
- package/templates/storefront-retail/next.config.ts +42 -0
- package/templates/storefront-retail/package.json +35 -0
- package/templates/storefront-retail/postcss.config.mjs +7 -0
- package/templates/storefront-retail/tsconfig.json +23 -0
- package/templates/storefront-retail/vitest.config.ts +9 -0
- package/templates/storefront-services/.claude/skills/cimplify-storefront/SKILL.md +145 -0
- package/templates/storefront-services/.cursor/rules/cimplify-storefront.mdc +25 -0
- package/templates/storefront-services/.env.example +16 -0
- package/templates/storefront-services/AGENTS.md +101 -0
- package/templates/storefront-services/CLAUDE.md +22 -0
- package/templates/storefront-services/README.md +73 -0
- package/templates/storefront-services/__tests__/brand.test.ts +4 -0
- package/templates/storefront-services/__tests__/cart-flow.test.ts +4 -0
- package/templates/storefront-services/__tests__/contract.test.ts +4 -0
- package/templates/storefront-services/app/about/page.tsx +38 -0
- package/templates/storefront-services/app/accessibility/page.tsx +11 -0
- package/templates/storefront-services/app/account/addresses/page.tsx +21 -0
- package/templates/storefront-services/app/account/orders/page.tsx +21 -0
- package/templates/storefront-services/app/account/page.tsx +22 -0
- package/templates/storefront-services/app/account/settings/page.tsx +21 -0
- package/templates/storefront-services/app/book/book-client.tsx +195 -0
- package/templates/storefront-services/app/book/page.tsx +65 -0
- package/templates/storefront-services/app/cart/page.tsx +9 -0
- package/templates/storefront-services/app/categories/[slug]/listing-client.tsx +19 -0
- package/templates/storefront-services/app/categories/[slug]/page.tsx +118 -0
- package/templates/storefront-services/app/checkout/page.tsx +17 -0
- package/templates/storefront-services/app/collections/[slug]/listing-client.tsx +20 -0
- package/templates/storefront-services/app/collections/[slug]/page.tsx +118 -0
- package/templates/storefront-services/app/contact/contact-form.tsx +109 -0
- package/templates/storefront-services/app/contact/page.tsx +54 -0
- package/templates/storefront-services/app/error.tsx +60 -0
- package/templates/storefront-services/app/faq/page.tsx +46 -0
- package/templates/storefront-services/app/globals.css +45 -0
- package/templates/storefront-services/app/layout.tsx +82 -0
- package/templates/storefront-services/app/llms.txt/route.ts +94 -0
- package/templates/storefront-services/app/login/page.tsx +17 -0
- package/templates/storefront-services/app/not-found.tsx +39 -0
- package/templates/storefront-services/app/opensearch.xml/route.ts +37 -0
- package/templates/storefront-services/app/orders/[id]/page.tsx +21 -0
- package/templates/storefront-services/app/page.tsx +97 -0
- package/templates/storefront-services/app/privacy/page.tsx +44 -0
- package/templates/storefront-services/app/returns/page.tsx +11 -0
- package/templates/storefront-services/app/robots.ts +18 -0
- package/templates/storefront-services/app/search/page.tsx +38 -0
- package/templates/storefront-services/app/search/search-client.tsx +7 -0
- package/templates/storefront-services/app/shipping/page.tsx +16 -0
- package/templates/storefront-services/app/shop/page.tsx +31 -0
- package/templates/storefront-services/app/shop/shop-client.tsx +27 -0
- package/templates/storefront-services/app/signup/page.tsx +17 -0
- package/templates/storefront-services/app/sitemap-page/page.tsx +167 -0
- package/templates/storefront-services/app/sitemap.ts +62 -0
- package/templates/storefront-services/app/terms/page.tsx +44 -0
- package/templates/storefront-services/app/track-order/page.tsx +24 -0
- package/templates/storefront-services/app/track-order/track-order-form.tsx +69 -0
- package/templates/storefront-services/components/account-iframe.tsx +13 -0
- package/templates/storefront-services/components/cart-drawer.tsx +14 -0
- package/templates/storefront-services/components/cart-pill.tsx +36 -0
- package/templates/storefront-services/components/category-grid.tsx +28 -0
- package/templates/storefront-services/components/collection-strip.tsx +45 -0
- package/templates/storefront-services/components/footer.tsx +148 -0
- package/templates/storefront-services/components/header.tsx +43 -0
- package/templates/storefront-services/components/hero.tsx +25 -0
- package/templates/storefront-services/components/nav-link.tsx +20 -0
- package/templates/storefront-services/components/policy-page.tsx +49 -0
- package/templates/storefront-services/components/product-modal.tsx +104 -0
- package/templates/storefront-services/components/providers.tsx +35 -0
- package/templates/storefront-services/components/store-product-card.tsx +87 -0
- package/templates/storefront-services/lib/brand.ts +396 -0
- package/templates/storefront-services/lib/cart.ts +12 -0
- package/templates/storefront-services/next.config.ts +42 -0
- package/templates/storefront-services/package.json +35 -0
- package/templates/storefront-services/postcss.config.mjs +7 -0
- package/templates/storefront-services/tsconfig.json +23 -0
- package/templates/storefront-services/vitest.config.ts +9 -0
|
@@ -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,195 @@
|
|
|
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
|
+
interface Slot {
|
|
9
|
+
date: Date;
|
|
10
|
+
label: string;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Real booking flow:
|
|
15
|
+
* 1. Pick a treatment (left rail)
|
|
16
|
+
* 2. Pick a date (chips, today + next 13 days)
|
|
17
|
+
* 3. Pick a slot (15-minute grid, 10am–7pm)
|
|
18
|
+
* 4. Add to cart with the chosen slot as a cart-item note;
|
|
19
|
+
* Cimplify Checkout finalises the booking.
|
|
20
|
+
*
|
|
21
|
+
* The slot grid is generated client-side as a placeholder. In production,
|
|
22
|
+
* call `useAvailableSlots({ serviceId, date })` from `@cimplify/sdk/react`
|
|
23
|
+
* to fetch real availability from the Cimplify scheduling API.
|
|
24
|
+
*/
|
|
25
|
+
export function BookClient({ treatments }: { treatments: Product[] }) {
|
|
26
|
+
const router = useRouter();
|
|
27
|
+
const { addItem } = useCart();
|
|
28
|
+
const [selectedTreatment, setSelectedTreatment] = useState<Product | undefined>(
|
|
29
|
+
treatments[0],
|
|
30
|
+
);
|
|
31
|
+
const [selectedDate, setSelectedDate] = useState<Date>(() => new Date());
|
|
32
|
+
const [selectedSlotKey, setSelectedSlotKey] = useState<string | null>(null);
|
|
33
|
+
const [submitting, setSubmitting] = useState(false);
|
|
34
|
+
|
|
35
|
+
const dates = useMemo(() => {
|
|
36
|
+
const now = new Date();
|
|
37
|
+
return Array.from({ length: 14 }).map((_, i) => {
|
|
38
|
+
const d = new Date(now);
|
|
39
|
+
d.setDate(now.getDate() + i);
|
|
40
|
+
d.setHours(0, 0, 0, 0);
|
|
41
|
+
return d;
|
|
42
|
+
});
|
|
43
|
+
}, []);
|
|
44
|
+
|
|
45
|
+
const slots = useMemo<Slot[]>(() => {
|
|
46
|
+
const out: Slot[] = [];
|
|
47
|
+
for (let h = 10; h <= 19; h++) {
|
|
48
|
+
for (const m of [0, 30]) {
|
|
49
|
+
const d = new Date(selectedDate);
|
|
50
|
+
d.setHours(h, m, 0, 0);
|
|
51
|
+
const label = `${String(h).padStart(2, "0")}:${String(m).padStart(2, "0")}`;
|
|
52
|
+
out.push({ date: d, label });
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
return out;
|
|
56
|
+
}, [selectedDate]);
|
|
57
|
+
|
|
58
|
+
const slotKey = (s: Slot) => s.date.toISOString();
|
|
59
|
+
|
|
60
|
+
async function confirm() {
|
|
61
|
+
if (!selectedTreatment || !selectedSlotKey) return;
|
|
62
|
+
setSubmitting(true);
|
|
63
|
+
try {
|
|
64
|
+
await addItem(selectedTreatment, 1, {
|
|
65
|
+
notes: `Booked for ${new Date(selectedSlotKey).toLocaleString()}`,
|
|
66
|
+
});
|
|
67
|
+
router.push("/checkout");
|
|
68
|
+
} catch {
|
|
69
|
+
setSubmitting(false);
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
if (treatments.length === 0) {
|
|
74
|
+
return (
|
|
75
|
+
<p className="text-muted-foreground">
|
|
76
|
+
No bookable treatments yet. Add a Service-type product to your catalogue first.
|
|
77
|
+
</p>
|
|
78
|
+
);
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
return (
|
|
82
|
+
<div className="grid grid-cols-1 lg:grid-cols-[1fr_1.4fr] gap-8">
|
|
83
|
+
{/* Treatments */}
|
|
84
|
+
<div>
|
|
85
|
+
<p className="text-[11px] font-mono uppercase tracking-[0.16em] text-muted-foreground mb-3">
|
|
86
|
+
Treatment
|
|
87
|
+
</p>
|
|
88
|
+
<div className="space-y-2">
|
|
89
|
+
{treatments.map((t) => {
|
|
90
|
+
const active = selectedTreatment?.id === t.id;
|
|
91
|
+
return (
|
|
92
|
+
<button
|
|
93
|
+
key={t.id}
|
|
94
|
+
type="button"
|
|
95
|
+
onClick={() => setSelectedTreatment(t)}
|
|
96
|
+
className={[
|
|
97
|
+
"w-full text-left rounded-2xl border p-4 transition-colors",
|
|
98
|
+
active
|
|
99
|
+
? "border-primary bg-primary/5"
|
|
100
|
+
: "border-border bg-card hover:border-foreground/30",
|
|
101
|
+
].join(" ")}
|
|
102
|
+
>
|
|
103
|
+
<div className="flex items-center justify-between gap-3">
|
|
104
|
+
<div className="min-w-0">
|
|
105
|
+
<p className="font-semibold text-sm m-0 truncate">{t.name}</p>
|
|
106
|
+
<p className="text-xs text-muted-foreground m-0">
|
|
107
|
+
{t.duration_minutes ? `${t.duration_minutes} min · ` : ""}
|
|
108
|
+
{t.currency ?? "GHS"} {t.default_price}
|
|
109
|
+
</p>
|
|
110
|
+
</div>
|
|
111
|
+
{active && (
|
|
112
|
+
<span className="grid place-items-center w-6 h-6 rounded-full bg-primary text-primary-foreground text-xs">
|
|
113
|
+
✓
|
|
114
|
+
</span>
|
|
115
|
+
)}
|
|
116
|
+
</div>
|
|
117
|
+
</button>
|
|
118
|
+
);
|
|
119
|
+
})}
|
|
120
|
+
</div>
|
|
121
|
+
</div>
|
|
122
|
+
|
|
123
|
+
{/* Date + slots */}
|
|
124
|
+
<div className="rounded-2xl border border-border bg-card p-6">
|
|
125
|
+
<p className="text-[11px] font-mono uppercase tracking-[0.16em] text-muted-foreground mb-3">
|
|
126
|
+
Date
|
|
127
|
+
</p>
|
|
128
|
+
<div className="grid grid-cols-7 gap-1.5 mb-6">
|
|
129
|
+
{dates.map((d) => {
|
|
130
|
+
const active = d.toDateString() === selectedDate.toDateString();
|
|
131
|
+
return (
|
|
132
|
+
<button
|
|
133
|
+
key={d.toISOString()}
|
|
134
|
+
type="button"
|
|
135
|
+
onClick={() => {
|
|
136
|
+
setSelectedDate(d);
|
|
137
|
+
setSelectedSlotKey(null);
|
|
138
|
+
}}
|
|
139
|
+
className={[
|
|
140
|
+
"flex flex-col items-center justify-center py-2 rounded-md transition-colors",
|
|
141
|
+
active
|
|
142
|
+
? "bg-foreground text-background"
|
|
143
|
+
: "bg-background hover:bg-muted text-foreground",
|
|
144
|
+
].join(" ")}
|
|
145
|
+
>
|
|
146
|
+
<span className="text-[10px] uppercase tracking-wider opacity-60">
|
|
147
|
+
{d.toLocaleString(undefined, { weekday: "short" })}
|
|
148
|
+
</span>
|
|
149
|
+
<span className="text-base font-semibold tabular-nums">{d.getDate()}</span>
|
|
150
|
+
</button>
|
|
151
|
+
);
|
|
152
|
+
})}
|
|
153
|
+
</div>
|
|
154
|
+
|
|
155
|
+
<p className="text-[11px] font-mono uppercase tracking-[0.16em] text-muted-foreground mb-3">
|
|
156
|
+
Slot
|
|
157
|
+
</p>
|
|
158
|
+
<div className="grid grid-cols-3 sm:grid-cols-4 gap-2">
|
|
159
|
+
{slots.map((s) => {
|
|
160
|
+
const key = slotKey(s);
|
|
161
|
+
const active = selectedSlotKey === key;
|
|
162
|
+
return (
|
|
163
|
+
<button
|
|
164
|
+
key={key}
|
|
165
|
+
type="button"
|
|
166
|
+
onClick={() => setSelectedSlotKey(key)}
|
|
167
|
+
className={[
|
|
168
|
+
"py-2 rounded-md text-sm tabular-nums transition-colors",
|
|
169
|
+
active
|
|
170
|
+
? "bg-primary text-primary-foreground"
|
|
171
|
+
: "bg-background border border-border hover:border-primary",
|
|
172
|
+
].join(" ")}
|
|
173
|
+
>
|
|
174
|
+
{s.label}
|
|
175
|
+
</button>
|
|
176
|
+
);
|
|
177
|
+
})}
|
|
178
|
+
</div>
|
|
179
|
+
|
|
180
|
+
<button
|
|
181
|
+
type="button"
|
|
182
|
+
onClick={confirm}
|
|
183
|
+
disabled={!selectedTreatment || !selectedSlotKey || submitting}
|
|
184
|
+
className="w-full mt-6 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"
|
|
185
|
+
>
|
|
186
|
+
{submitting
|
|
187
|
+
? "Confirming…"
|
|
188
|
+
: selectedSlotKey
|
|
189
|
+
? `Book ${selectedTreatment?.name} at ${new Date(selectedSlotKey).toLocaleTimeString(undefined, { hour: "2-digit", minute: "2-digit" })}`
|
|
190
|
+
: "Pick a slot to book"}
|
|
191
|
+
</button>
|
|
192
|
+
</div>
|
|
193
|
+
</div>
|
|
194
|
+
);
|
|
195
|
+
}
|
|
@@ -0,0 +1,65 @@
|
|
|
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 { BookClient } from "./book-client";
|
|
6
|
+
import { brand } from "@/lib/brand";
|
|
7
|
+
|
|
8
|
+
export const metadata: Metadata = {
|
|
9
|
+
title: `Book a treatment — ${brand.name}`,
|
|
10
|
+
description: "Pick a treatment, pick a slot, you're booked.",
|
|
11
|
+
};
|
|
12
|
+
|
|
13
|
+
async function getTreatments(): 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
|
+
// Booking flow only handles service-typed products.
|
|
22
|
+
return r.value.items.filter((p) => p.type === "service");
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
export default async function BookPage() {
|
|
26
|
+
const treatments = await getTreatments();
|
|
27
|
+
return (
|
|
28
|
+
<article className="max-w-5xl mx-auto px-6 sm:px-8 py-14">
|
|
29
|
+
<p className="text-[11px] font-mono uppercase tracking-[0.16em] text-primary mb-2">
|
|
30
|
+
Book a treatment
|
|
31
|
+
</p>
|
|
32
|
+
<h1 className="text-[clamp(2rem,5vw,3rem)] font-semibold mb-3 -tracking-[0.02em]">
|
|
33
|
+
Pick a treatment.<br />Pick a slot.
|
|
34
|
+
</h1>
|
|
35
|
+
<p className="text-muted-foreground leading-relaxed mb-10 max-w-xl">
|
|
36
|
+
Available windows for the next two weeks. Confirmation by SMS within a minute of
|
|
37
|
+
booking. Free cancellation up to 24 hours before.
|
|
38
|
+
</p>
|
|
39
|
+
|
|
40
|
+
<Suspense fallback={<BookSkeleton />}>
|
|
41
|
+
<BookClient treatments={treatments} />
|
|
42
|
+
</Suspense>
|
|
43
|
+
</article>
|
|
44
|
+
);
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
function BookSkeleton() {
|
|
48
|
+
return (
|
|
49
|
+
<div className="grid grid-cols-1 lg:grid-cols-[1fr_1.4fr] gap-8">
|
|
50
|
+
<div className="space-y-2">
|
|
51
|
+
{Array.from({ length: 6 }).map((_, i) => (
|
|
52
|
+
<div key={i} className="h-16 bg-muted rounded-2xl animate-pulse" />
|
|
53
|
+
))}
|
|
54
|
+
</div>
|
|
55
|
+
<div className="rounded-2xl border border-border bg-card p-8">
|
|
56
|
+
<div className="h-5 w-40 bg-muted rounded mb-4 animate-pulse" />
|
|
57
|
+
<div className="grid grid-cols-3 gap-2">
|
|
58
|
+
{Array.from({ length: 9 }).map((_, i) => (
|
|
59
|
+
<div key={i} className="h-10 bg-muted rounded animate-pulse" />
|
|
60
|
+
))}
|
|
61
|
+
</div>
|
|
62
|
+
</div>
|
|
63
|
+
</div>
|
|
64
|
+
);
|
|
65
|
+
}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
import { useRouter } from "next/navigation";
|
|
4
|
+
import { CartPage as SdkCartPage } from "@cimplify/sdk/react";
|
|
5
|
+
|
|
6
|
+
export default function CartPage() {
|
|
7
|
+
const router = useRouter();
|
|
8
|
+
return <SdkCartPage onCheckout={() => router.push("/checkout")} />;
|
|
9
|
+
}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
import { ProductGrid } from "@cimplify/sdk/react";
|
|
4
|
+
import type { Product } from "@cimplify/sdk";
|
|
5
|
+
import { StoreProductCard } from "@/components/store-product-card";
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Client island for the category listing. Receives server-fetched products
|
|
9
|
+
* as props (serializable) and owns the `renderCard` function.
|
|
10
|
+
*/
|
|
11
|
+
export function ListingClient({ products }: { products: Product[] }) {
|
|
12
|
+
return (
|
|
13
|
+
<ProductGrid
|
|
14
|
+
products={products}
|
|
15
|
+
emptyMessage="No products in this category yet."
|
|
16
|
+
renderCard={(p) => <StoreProductCard product={p} />}
|
|
17
|
+
/>
|
|
18
|
+
);
|
|
19
|
+
}
|
|
@@ -0,0 +1,118 @@
|
|
|
1
|
+
import type { Metadata } from "next";
|
|
2
|
+
import { Suspense } from "react";
|
|
3
|
+
import Link from "next/link";
|
|
4
|
+
import { notFound } from "next/navigation";
|
|
5
|
+
import { cacheTag, cacheLife } from "next/cache";
|
|
6
|
+
import {
|
|
7
|
+
getServerClient,
|
|
8
|
+
tags,
|
|
9
|
+
type Category,
|
|
10
|
+
type Product,
|
|
11
|
+
} from "@cimplify/sdk/server";
|
|
12
|
+
import { ListingClient } from "./listing-client";
|
|
13
|
+
import { brand } from "@/lib/brand";
|
|
14
|
+
|
|
15
|
+
interface CategoryData {
|
|
16
|
+
category: Category;
|
|
17
|
+
products: Product[];
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
async function getCategory(slug: string): Promise<CategoryData | null> {
|
|
21
|
+
"use cache";
|
|
22
|
+
cacheTag(tags.categories());
|
|
23
|
+
cacheLife("hours");
|
|
24
|
+
|
|
25
|
+
const client = getServerClient();
|
|
26
|
+
const catRes = await client.catalogue.getCategoryBySlug(slug);
|
|
27
|
+
if (!catRes.ok) return null;
|
|
28
|
+
|
|
29
|
+
cacheTag(tags.category(catRes.value.id), tags.categoryProducts(catRes.value.id));
|
|
30
|
+
const r = await client.catalogue.getCategoryProducts(catRes.value.id);
|
|
31
|
+
const products = r.ok
|
|
32
|
+
? ((r.value as { items?: Product[] }).items ?? (r.value as Product[]))
|
|
33
|
+
: [];
|
|
34
|
+
return { category: catRes.value, products };
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export async function generateMetadata({
|
|
38
|
+
params,
|
|
39
|
+
}: {
|
|
40
|
+
params: Promise<{ slug: string }>;
|
|
41
|
+
}): Promise<Metadata> {
|
|
42
|
+
const { slug } = await params;
|
|
43
|
+
const data = await getCategory(slug);
|
|
44
|
+
if (!data) return {};
|
|
45
|
+
return {
|
|
46
|
+
title: `${data.category.name} — ${brand.name}`,
|
|
47
|
+
description: data.category.description ?? undefined,
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
export default async function CategoryPage({
|
|
52
|
+
params,
|
|
53
|
+
}: {
|
|
54
|
+
params: Promise<{ slug: string }>;
|
|
55
|
+
}) {
|
|
56
|
+
return (
|
|
57
|
+
<Suspense fallback={<CategorySkeleton />}>
|
|
58
|
+
<CategoryContent params={params} />
|
|
59
|
+
</Suspense>
|
|
60
|
+
);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
async function CategoryContent({
|
|
64
|
+
params,
|
|
65
|
+
}: {
|
|
66
|
+
params: Promise<{ slug: string }>;
|
|
67
|
+
}) {
|
|
68
|
+
const { slug } = await params;
|
|
69
|
+
const data = await getCategory(slug);
|
|
70
|
+
if (!data) notFound();
|
|
71
|
+
|
|
72
|
+
const { category, products } = data;
|
|
73
|
+
return (
|
|
74
|
+
<section className="max-w-7xl mx-auto px-8 pt-12">
|
|
75
|
+
<header className="mb-8 text-center">
|
|
76
|
+
<p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2">
|
|
77
|
+
Category
|
|
78
|
+
</p>
|
|
79
|
+
<h1 className="font-serif text-[clamp(2rem,4vw,2.75rem)] font-semibold mb-2">
|
|
80
|
+
{category.name}
|
|
81
|
+
</h1>
|
|
82
|
+
{category.description && (
|
|
83
|
+
<p className="mx-auto mb-2 max-w-xl text-muted-foreground">
|
|
84
|
+
{category.description}
|
|
85
|
+
</p>
|
|
86
|
+
)}
|
|
87
|
+
<p className="text-sm text-muted-foreground">
|
|
88
|
+
{products.length} item{products.length === 1 ? "" : "s"}
|
|
89
|
+
</p>
|
|
90
|
+
</header>
|
|
91
|
+
<ListingClient products={products} />
|
|
92
|
+
{products.length === 0 && (
|
|
93
|
+
<p className="text-center mt-8">
|
|
94
|
+
<Link href="/" className="text-primary font-semibold">
|
|
95
|
+
← Back home
|
|
96
|
+
</Link>
|
|
97
|
+
</p>
|
|
98
|
+
)}
|
|
99
|
+
</section>
|
|
100
|
+
);
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
function CategorySkeleton() {
|
|
104
|
+
return (
|
|
105
|
+
<section className="max-w-7xl mx-auto px-8 pt-12">
|
|
106
|
+
<header className="mb-8 text-center">
|
|
107
|
+
<div className="mx-auto h-3 w-20 bg-muted rounded mb-2 animate-pulse" />
|
|
108
|
+
<div className="mx-auto h-10 w-64 bg-muted rounded mb-2 animate-pulse" />
|
|
109
|
+
<div className="mx-auto h-4 w-80 bg-muted rounded animate-pulse" />
|
|
110
|
+
</header>
|
|
111
|
+
<div className="grid grid-cols-2 md:grid-cols-3 gap-4">
|
|
112
|
+
{Array.from({ length: 6 }).map((_, i) => (
|
|
113
|
+
<div key={i} className="aspect-square bg-muted rounded-2xl animate-pulse" />
|
|
114
|
+
))}
|
|
115
|
+
</div>
|
|
116
|
+
</section>
|
|
117
|
+
);
|
|
118
|
+
}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
import { useRouter } from "next/navigation";
|
|
4
|
+
import { CheckoutPage as SdkCheckoutPage } from "@cimplify/sdk/react";
|
|
5
|
+
|
|
6
|
+
export default function CheckoutPage() {
|
|
7
|
+
const router = useRouter();
|
|
8
|
+
return (
|
|
9
|
+
<SdkCheckoutPage
|
|
10
|
+
onComplete={(result) => {
|
|
11
|
+
if (result.success && result.order) {
|
|
12
|
+
router.push(`/orders/${result.order.id}`);
|
|
13
|
+
}
|
|
14
|
+
}}
|
|
15
|
+
/>
|
|
16
|
+
);
|
|
17
|
+
}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
import { ProductGrid } from "@cimplify/sdk/react";
|
|
4
|
+
import type { Product } from "@cimplify/sdk";
|
|
5
|
+
import { StoreProductCard } from "@/components/store-product-card";
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Client island for the collection listing. Receives server-fetched
|
|
9
|
+
* products as props (serializable) and owns the `renderCard` function
|
|
10
|
+
* (which can't cross the server/client boundary).
|
|
11
|
+
*/
|
|
12
|
+
export function ListingClient({ products }: { products: Product[] }) {
|
|
13
|
+
return (
|
|
14
|
+
<ProductGrid
|
|
15
|
+
products={products}
|
|
16
|
+
emptyMessage="No products in this collection yet."
|
|
17
|
+
renderCard={(p) => <StoreProductCard product={p} />}
|
|
18
|
+
/>
|
|
19
|
+
);
|
|
20
|
+
}
|
|
@@ -0,0 +1,118 @@
|
|
|
1
|
+
import type { Metadata } from "next";
|
|
2
|
+
import { Suspense } from "react";
|
|
3
|
+
import Link from "next/link";
|
|
4
|
+
import { notFound } from "next/navigation";
|
|
5
|
+
import { cacheTag, cacheLife } from "next/cache";
|
|
6
|
+
import {
|
|
7
|
+
getServerClient,
|
|
8
|
+
tags,
|
|
9
|
+
type Collection,
|
|
10
|
+
type Product,
|
|
11
|
+
} from "@cimplify/sdk/server";
|
|
12
|
+
import { ListingClient } from "./listing-client";
|
|
13
|
+
import { brand } from "@/lib/brand";
|
|
14
|
+
|
|
15
|
+
interface CollectionData {
|
|
16
|
+
collection: Collection;
|
|
17
|
+
products: Product[];
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
async function getCollection(slug: string): Promise<CollectionData | null> {
|
|
21
|
+
"use cache";
|
|
22
|
+
cacheTag(tags.collections());
|
|
23
|
+
cacheLife("hours");
|
|
24
|
+
|
|
25
|
+
const client = getServerClient();
|
|
26
|
+
const colRes = await client.catalogue.getCollectionBySlug(slug);
|
|
27
|
+
if (!colRes.ok) return null;
|
|
28
|
+
|
|
29
|
+
cacheTag(tags.collection(colRes.value.id), tags.collectionProducts(colRes.value.id));
|
|
30
|
+
const r = await client.catalogue.getCollectionProducts(colRes.value.id);
|
|
31
|
+
const products = r.ok
|
|
32
|
+
? ((r.value as { items?: Product[] }).items ?? (r.value as Product[]))
|
|
33
|
+
: [];
|
|
34
|
+
return { collection: colRes.value, products };
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export async function generateMetadata({
|
|
38
|
+
params,
|
|
39
|
+
}: {
|
|
40
|
+
params: Promise<{ slug: string }>;
|
|
41
|
+
}): Promise<Metadata> {
|
|
42
|
+
const { slug } = await params;
|
|
43
|
+
const data = await getCollection(slug);
|
|
44
|
+
if (!data) return {};
|
|
45
|
+
return {
|
|
46
|
+
title: `${data.collection.name} — ${brand.name}`,
|
|
47
|
+
description: data.collection.description ?? undefined,
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
export default async function CollectionPage({
|
|
52
|
+
params,
|
|
53
|
+
}: {
|
|
54
|
+
params: Promise<{ slug: string }>;
|
|
55
|
+
}) {
|
|
56
|
+
return (
|
|
57
|
+
<Suspense fallback={<CollectionSkeleton />}>
|
|
58
|
+
<CollectionContent params={params} />
|
|
59
|
+
</Suspense>
|
|
60
|
+
);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
async function CollectionContent({
|
|
64
|
+
params,
|
|
65
|
+
}: {
|
|
66
|
+
params: Promise<{ slug: string }>;
|
|
67
|
+
}) {
|
|
68
|
+
const { slug } = await params;
|
|
69
|
+
const data = await getCollection(slug);
|
|
70
|
+
if (!data) notFound();
|
|
71
|
+
|
|
72
|
+
const { collection, products } = data;
|
|
73
|
+
return (
|
|
74
|
+
<section className="max-w-7xl mx-auto px-8 pt-12">
|
|
75
|
+
<header className="mb-8 text-center">
|
|
76
|
+
<p className="text-[11px] font-semibold uppercase tracking-[0.16em] text-primary mb-2">
|
|
77
|
+
Collection
|
|
78
|
+
</p>
|
|
79
|
+
<h1 className="font-serif text-[clamp(2rem,4vw,2.75rem)] font-semibold mb-2">
|
|
80
|
+
{collection.name}
|
|
81
|
+
</h1>
|
|
82
|
+
{collection.description && (
|
|
83
|
+
<p className="mx-auto mb-2 max-w-xl text-muted-foreground">
|
|
84
|
+
{collection.description}
|
|
85
|
+
</p>
|
|
86
|
+
)}
|
|
87
|
+
<p className="text-sm text-muted-foreground">
|
|
88
|
+
{products.length} item{products.length === 1 ? "" : "s"}
|
|
89
|
+
</p>
|
|
90
|
+
</header>
|
|
91
|
+
<ListingClient products={products} />
|
|
92
|
+
{products.length === 0 && (
|
|
93
|
+
<p className="text-center mt-8">
|
|
94
|
+
<Link href="/" className="text-primary font-semibold">
|
|
95
|
+
← Back home
|
|
96
|
+
</Link>
|
|
97
|
+
</p>
|
|
98
|
+
)}
|
|
99
|
+
</section>
|
|
100
|
+
);
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
function CollectionSkeleton() {
|
|
104
|
+
return (
|
|
105
|
+
<section className="max-w-7xl mx-auto px-8 pt-12">
|
|
106
|
+
<header className="mb-8 text-center">
|
|
107
|
+
<div className="mx-auto h-3 w-20 bg-muted rounded mb-2 animate-pulse" />
|
|
108
|
+
<div className="mx-auto h-10 w-64 bg-muted rounded mb-2 animate-pulse" />
|
|
109
|
+
<div className="mx-auto h-4 w-80 bg-muted rounded animate-pulse" />
|
|
110
|
+
</header>
|
|
111
|
+
<div className="grid grid-cols-2 md:grid-cols-3 gap-4">
|
|
112
|
+
{Array.from({ length: 6 }).map((_, i) => (
|
|
113
|
+
<div key={i} className="aspect-square bg-muted rounded-2xl animate-pulse" />
|
|
114
|
+
))}
|
|
115
|
+
</div>
|
|
116
|
+
</section>
|
|
117
|
+
);
|
|
118
|
+
}
|