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