This is a NextJS starter for Vendure in the form of a demo e-commerce shop.
The project is in its alpha phase! However, you can still read the concepts, run the storefront page locally, or check how it works at: shop.aexol.com.
- Table of Contents
- Installation
- Zeus
- Page Naming Conventions
- Internationalization With i18next
- Appearance
- More Info
1. Clone this repo via SSH, HTTPS or the GitHub CLI.
2. Install the packages using: npm i
.
3. Set up your Vendure server locally and run it on http://localhost:3000/
. You can read more about how to set that up in the Vendure Server section below.
You need to have the Vendure store running locally to use this storefront. This storefront requires a Vendure V2 server. You can either run a local instance or use our public demo server. The demo of the Vendure server (MinIO & Postgres & SMTP) can be found here. Check it out to see all changes.
Important
For the best experience when using our demo, you also need to apply certain modifications to the Vendure server:
- Apply two collections
all
andsearch
. Both of them should contain all products (with the exception of cases using gift cards or shipping-protections) - Add the stock level as a number value and not as enum values, as seen below:
export class ExactStockDisplayStrategy implements StockDisplayStrategy {
getStockLevel(
ctx: RequestContext,
productVariant: ProductVariant,
saleableStockLevel: number
): string {
return saleableStockLevel.toString();
}
}
export const catalogOptions: VendureConfig["catalogOptions"] = {
stockDisplayStrategy: new ExactStockDisplayStrategy(),
};
4. Create a new .env
file in the root of the project and add the following variables:
NEXT_PUBLIC_HOST="http://localhost:3000/shop-api".
NEXT_PUBLIC_PRIVY_APP_ID=your-privy-app-id
5. Run the project locally using npm run dev
.
We use GraphQL Zeus to provide selectors for GraphQL queries and mutations. You can think of selectors as fragments in GraphQL, just with the added type-safety.
Privy based authentication has been implemented. If you don't have a privy account(and app), you can create one and then you can go ahead and set your app-id in here
We aimed for a fairly simple naming convention for pages that aligns with the DDD (Domain-driven design) principles:
- Each page file is named using the format
page-name.page.tsx
, wherepage-name
represents the name of the page or route.
For example, the main page of your application would be namedindex.page.tsx
. - We are using slug pages for products and collections, where we have a
products
andcollections
folder with a[slug].page.tsx
. The[slug]
is replaced by the product or collection name fetched from the backend as props.
This allows to dynamically generate those pages at scale, while maintaining a structure that is easy to navigate with routes such as/collections/electronics/
or/products/laptop/
.
Using such naming conventions helps maintain a clean and organized folder structure that reflects the structure of your application's domains or features. By separating pages into their respective folders and adopting a consistent naming convention, you can easily locate and manage your application's routes and comfortably navigate through any issues that might arise.
Because the majority of e-commerce shops uses localization to accomodate clients from all over the world, we have integrated i18next to handle language translations. i18next makes adding and updating translated content extremely easy.
How we use i18next:
Element | Description |
---|---|
Translation Files | We maintain separate JSON translation files for each supported language. These files contain translation keys and their corresponding localized text. For example, you might find the English translation file for home page at public/locales/en/homePage.json |
Locale Configuration | We configure i18next to load the appropriate translation files based on the user's selected locale. |
Integration with React | We use the next-i18next package to integrate i18next with React components to seamlessly access the translations in your React components via a simple useTranslation hook. It will then always use the matching translation for the user's selected locale. |
Example:
import { useTranslation } from 'next-i18next';
export const Home: React.FC = () => {
const { t } = useTranslation('homepage');
return (
<Layout>
<Hero
cta={t('hero-cta')}
h1={t('hero-h1')}
h2={t('hero-h2')}
desc={t('hero-p')}
link="/collections/all"
/>
</Layout>
);
};
Tip
For quick localization you can use DevTranslate to translate json files into up to 28 languages all at the same time.
Lucide Icons is an open source library of over one thousand svg icons and symbols separated into official packages. This makes picking the icons you need for your project much easier.
- To check them out yourself head to: lucide.dev.
We are building our own engine based on styled components with props that work similarly to our favorite CSS framework: Tailwind.
For example, here's what our Stack
component looks like:
export const Stack = styled.div<BaseFlexParams>`
gap: ${p => p.gap || 0};
display: flex;
flex-direction: ${p => (p.column ? (p.reverse ? 'column-reverse' : 'column') : p.reverse ? 'row-reverse' : 'row')};
flex-wrap: ${p => (p.flexWrap ? 'wrap' : 'nowrap')};
justify-content: ${p =>
p.justifyBetween ? 'space-between' : p.justifyCenter ? 'center' : p.justifyEnd ? 'end' : 'start'};
align-items: ${p => (p.itemsCenter ? 'center' : 'initial')};
`;
Due to this set-up of props, the usage is very similar to Tailwind. The difference is that, in this case, you have to skip ClassName
, as seen below:
<Stack column gap="2rem">
{children}
</Stack>
Theming is provided by a set of generic functions in the code and Emotion.
You can use values from the theme with thv
. It is a function that consumes the theme and returns only the value.
You can alternatively use the usual method with ${p => p.theme}
.
You can see both methods in the example below:
import { thv } from '@/src/theme';
import styled from '@emotion/styled';
export const IconButton = styled.button<{ isActive?: boolean }>`
color: ${thv.button.icon.front};
border: 0;
border-radius: 100%;
font-weight: 600;
outline: 0;
width: 2.4rem;
height: 2.4rem;
display: flex;
align-items: center;
justify-content: center;
background: ${p => p.theme.button.icon.back || 'transparent'};
svg {
width: 2rem;
height: 2rem;
}
:hover {
box-shadow: none;
}
`;
We are devs and contributors to the GraphQL ecosystem with a lot of experience. We want to enter Vendure to create developer-friendly e-commerce solutions that don't rely on clunky and outdated stuff like Shopify's Liquid wrapped with JavaScript.
- Zeus Documentation
- i18next Documentation
- Next.js Documentation
- React Documentation
- TypeScript Documentation
- Finish the starter
- Deploy the storefront connected to the demo shop
- Implement basic cart functionality
- Implement basic checkout process
- Implement the design
- Add a basic payment process
- Add a basic User Profile
- Add the search products function
- Add filters
- Provide localization with devtranslate.app
- Add Static Git CMS MDTX
- Configure SEO and schema.org for every site
- Assure ISR is ready on every sub-site
- Migrate to the next router