Customization of Next.js + Headless Shopify + TypeScript + Tailwind CSS Starter to make a performant ecommerce storefront for modern streetwear clothing
The goal of this project was to gain insight on how to connect an ecommerce
storefront with a CMS backend that could easily be handed off to a client. I eventually chose Next.js for its strong ecommerce facing templates and ended up using headless Shopify as the backend (as opposed to Sanity CMS or Bigcommerce).
Goals :
Using this template helped me gain experience understanding how to navigate a well maintained codebase and how to add functionality to it by following the project conventions and structure when making my own TypeScript components.
100-streetwares
└───components
│ └─── auth
│ └─── cart
│ └─── common
| └─── icons
| └─── product
| └─── ui
| └─── wishlist
| └─── customize
| └─── HeroImage
| └─── HeroImage.module.css
| └─── HeroImage.tsx
| └─── index.ts
| └─── ImageZoom
| └─── ImageZoom.module.css
| └─── ImageZoom.tsx
| └─── index.ts
...
...
Using the react-intersection-observer along with Framer motion’s declarative syntax I only trigger the Framer motion animation when the element specified by the ref is in the viewport.
const controls = useAnimation()
const { ref, inView } = useInView({ threshold: 0.5, delay: 50 })
useEffect(() => {
if (inView) controls.start('visible')
}, [inView, controls])
const imageZoomVariant = {
hidden: { opacity: 0 },
visible: {
opacity: 1,
transition: {
duration: 0.6,
},
},
}
<motion.a
ref={ref}
initial="hidden"
animate={controls}
variants={imageZoomVariant}
className={` ${colClass} ${mediaQueryColClass}
>
// nested content...
</motion.a>
Since I wanted to make a streetwear / skate clothing store I looked online for sample images I could use to replicate the feel / UI of other popular online streetwear stores such as…
I typed my custom components and added in styling via Tailwind CSS and CSS modules to keep the generated class names unique.
Issues: In production PurgeCSS (activated through Tailwind) doesn’t recognize Tailwind utility classes made via string concatenation. This was unintentionally deleting my responsive code for this component, so as a short work around I applied the generated classes on a hidden span to prevent it from being purged.
const ImageZoom: FC<{
colSpanned?: number | any
backgroundImage?: string
title?: string
width?: number
height?: number
to?: any
text?: string | undefined
smallColSpanned?: number | any
hide?: boolean
}> = ({
colSpanned,
backgroundImage,
title,
width,
height,
to,
text,
smallColSpanned,
hide,
}) => {
// other code...
return (
<>
{/* Prevent purging of generated Tailwind classes made through string concatenation */}
<span className="hidden sm:hidden col-span-3 sm:col-span-1 sm:col-span-3"></span>
<Link href={to}>
<motion.a
ref={ref}
initial="hidden"
animate={controls}
variants={imageZoomVariant}
className={` ${colClass} ${mediaQueryColClass}
${hide ? styles.hide : ''}`}
>
<div className={styles['card-zoom']}>
<div
title={title}
className={`${styles['card-zoom-image']} ${backgroundImage}`}
></div>
{text && <h1 className={`${styles['card-zoom-text']}`}>{text}</h1>}
</div>
</motion.a>
</Link>
</>
)
}
(Original README)
The all-in-one starter kit for high-performance e-commerce sites. With a few clicks, Next.js developers can clone, deploy and fully customize their own store.
Start right now at nextjs.org/commerce
Demo live at: demo.vercel.store
Next.js Commerce integrates out-of-the-box with BigCommerce and Shopify. We plan to support all major ecommerce backends.
framework/commerce
contains all types, helpers and functions to be used as base to build a new provider.framework
‘s root folder and they will extend Next.js Commerce types and functionality (framework/commerce
).features
in commerce.config.json
and if needed it can also be accessed programatically.next.config.js
and commerce.config.json
adding specific data related to the provider. For example in case of BigCommerce, the images CDN and additional API routes.framework/commerce
, on their own framework folder and on some dependencies included in package.json
Open .env.local
and change the value of COMMERCE_PROVIDER
to the provider you would like to use, then set the environment variables for that provider (use .env.template
as the base).
The setup for Shopify would look like this for example:
COMMERCE_PROVIDER=shopify
NEXT_PUBLIC_SHOPIFY_STOREFRONT_ACCESS_TOKEN=xxxxxxxxxxxxxxxxxxxxxxxxxxxx
NEXT_PUBLIC_SHOPIFY_STORE_DOMAIN=xxxxxxx.myshopify.com
And check that the tsconfig.json
resolves to the chosen provider:
"@framework": ["framework/shopify"],
"@framework/*": ["framework/shopify/*"]
That’s it!
Every provider defines the features that it supports under framework/{provider}/commerce.config.json
NOTE: The selected provider should support the feature that you are toggling. (This means that you can’t turn wishlist on if the provider doesn’t support this functionality out the box)
commerce.config.json
{
"features": {
"wishlist": false
}
}
Follow our docs for Adding a new Commerce Provider.
If you succeeded building a provider, submit a PR with a valid demo and we’ll review it asap.
Our commitment to Open Source can be found here.
git checkout -b MY_BRANCH_NAME
npm install -g yarn
yarn
.env.template
and rename it to .env.local
.env.local
yarn dev
to build and watch for code changesWe’re using Github Projects to keep track of issues in progress and todo’s. Here is our Board
People actively working on this project: @okbel & @lfades.
sh
BIGCOMMERCE_STOREFRONT_API_URL=<>
BIGCOMMERCE_STOREFRONT_API_TOKEN=<>
BIGCOMMERCE_STORE_API_URL=<>
BIGCOMMERCE_STORE_API_TOKEN=<>
BIGCOMMERCE_STORE_API_CLIENT_ID=<>
BIGCOMMERCE_CHANNEL_ID=<>
npm i -g vercel
vercel link
vercel env pull .env.local