Skip to content

Commit f6c0e07

Browse files
authored
feat: sync template for recent react and next versions (#26)
1 parent 08c3f3e commit f6c0e07

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

61 files changed

+7119
-451
lines changed

.env.example

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,9 @@ COMPANY_NAME="shopware AG"
22
TWITTER_CREATOR="@shopware"
33
TWITTER_SITE="https://www.shopware.com/en/solutions/shopware-composable-frontends/"
44
SITE_NAME="Next.js Shopware Starter with Composable Frontends"
5-
SHOPWARE_STORE_DOMAIN=""
5+
SHOPWARE_STORE_DOMAIN="https://demo-frontends.shopware.store"
66
SHOPWARE_API_TYPE="store-api"
7-
SHOPWARE_ACCESS_TOKEN=""
7+
SHOPWARE_ACCESS_TOKEN="SWSCBHFSNTVMAWNZDNFKSHLAYW"
88
SHOPWARE_USE_SEO_URLS="true"
99
SHOPWARE_REVALIDATION_SECRET=""
1010
BASE_E2E_URL=""

.eslintrc.js

-23
This file was deleted.

.eslintrc.json

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"extends": ["next/core-web-vitals", "next/typescript"]
3+
}

.github/dependabot.yml

-6
This file was deleted.

.github/workflows/e2e-tests.yml

+2-2
Original file line numberDiff line numberDiff line change
@@ -15,13 +15,13 @@ jobs:
1515
- name: Install Node.js
1616
uses: actions/setup-node@v3
1717
with:
18-
node-version: 18
18+
node-version: 22
1919

2020
- run: corepack enable
2121
- run: pnpm --version
2222
- uses: actions/setup-node@v3
2323
with:
24-
node-version: 18
24+
node-version: 22
2525
cache: 'pnpm'
2626
cache-dependency-path: '**/pnpm-lock.yaml'
2727

.github/workflows/test.yml

-35
This file was deleted.

.husky/pre-commit

-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,2 @@
1-
#!/usr/bin/env sh
2-
. "$(dirname -- "$0")/_/husky.sh"
3-
41
pnpm run lint
52
npx lint-staged

.nvmrc

-1
This file was deleted.

.prettierignore

-3
This file was deleted.

README.md

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
# Next.js Shopware Starter
22

3-
A Next.js 14 and App Router-ready ecommerce template featuring:
3+
A high-performance, server-rendered Next.js App Router ecommerce application.
44

55
- Next.js App Router
66
- Optimized for SEO using Next.js's Metadata
@@ -18,6 +18,9 @@ Next.js Shopware Starter requires a running [Shopware 6 Instance (Installation G
1818

1919
To get started, use this README and the example environment variable comments.
2020

21+
- [React Bricks](https://github.com/ReactBricks/nextjs-commerce-rb) ([Demo](https://nextjs-commerce.reactbricks.com/))
22+
- Edit pages, product details, and footer content visually using [React Bricks](https://www.reactbricks.com) visual headless CMS.
23+
2124
## Running locally
2225

2326
You will need to use the environment variables [defined in `.env.example`](https://github.com/shopwareLabs/nextjs-shopware-starter/blob/main/.env.example) to run Next.js Shopware Starter. It's recommended you use [Vercel Environment Variables](https://vercel.com/docs/concepts/projects/environment-variables) for this, but a `.env` file is all that is necessary.

app/(cms)/[...cms]/page.tsx

+11-5
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,13 @@ import Prose from 'components/prose';
44
import { getPage } from 'lib/shopware';
55
import { notFound } from 'next/navigation';
66

7-
export async function generateMetadata({ params }: { params: { cms: string } }): Promise<Metadata> {
8-
const page = await getPage(params.cms);
7+
export async function generateMetadata({
8+
params
9+
}: {
10+
params: Promise<{ cms: string }>;
11+
}): Promise<Metadata> {
12+
const { cms } = await params;
13+
const page = await getPage(cms);
914

1015
if (!page) return notFound();
1116

@@ -20,8 +25,9 @@ export async function generateMetadata({ params }: { params: { cms: string } }):
2025
};
2126
}
2227

23-
export default async function Page({ params }: { params: { cms: string } }) {
24-
const page = await getPage(params.cms);
28+
export default async function Page({ params }: { params: Promise<{ cms: string }> }) {
29+
const { cms } = await params;
30+
const page = await getPage(cms);
2531

2632
if (!page) return notFound();
2733
let date = page.createdAt;
@@ -32,7 +38,7 @@ export default async function Page({ params }: { params: { cms: string } }) {
3238
return (
3339
<>
3440
<h1 className="mb-8 text-5xl font-bold">{page.title}</h1>
35-
<Prose className="mb-8" html={page.body as string} />
41+
<Prose className="mb-8" html={page.body} />
3642
<p className="text-sm italic">
3743
{`This document was last updated on ${new Intl.DateTimeFormat(undefined, {
3844
year: 'numeric',

app/(cms)/opengraph-image.tsx

+2-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,8 @@ import { getPage } from 'lib/shopware';
44
export const runtime = 'edge';
55

66
export default async function Image({ params }: { params: { page: string } }) {
7-
const page = await getPage(params.page);
7+
const { page: pageParamName } = await params;
8+
const page = await getPage(pageParamName);
89
const title = page ? page.seo?.title || page.title : '';
910

1011
return await OpengraphImage({ title });

app/layout.tsx

+15-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,9 @@
1+
import { getCart } from 'components/cart/actions';
2+
import { CartProvider } from 'components/cart/cart-context';
13
import Navbar from 'components/layout/navbar';
24
import { GeistSans } from 'geist/font/sans';
35
import { ensureStartsWith } from 'lib/utils';
6+
import { cookies } from 'next/headers';
47
import { ReactNode } from 'react';
58
import './globals.css';
69

@@ -32,11 +35,21 @@ export const metadata = {
3235
};
3336

3437
export default async function RootLayout({ children }: { children: ReactNode }) {
38+
const cartId = (await cookies()).get('cartId')?.value;
39+
// Don't await the fetch, pass the Promise to the context provider
40+
const cart = getCart(cartId);
41+
3542
return (
3643
<html lang="en" className={GeistSans.variable}>
3744
<body className="bg-neutral-50 text-black selection:bg-teal-300 dark:bg-neutral-900 dark:text-white dark:selection:bg-pink-500 dark:selection:text-white">
38-
<Navbar />
39-
<main>{children}</main>
45+
<CartProvider cartPromise={cart}>
46+
<Navbar />
47+
<main>
48+
{children}
49+
{/* <Toaster closeButton />
50+
<WelcomeToast /> */}
51+
</main>
52+
</CartProvider>
4053
</body>
4154
</html>
4255
);

app/page.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ export const metadata = {
99
}
1010
};
1111

12-
export default async function HomePage() {
12+
export default function HomePage() {
1313
return (
1414
<>
1515
<ThreeItemGrid />

app/product/[...handle]/page.tsx

+18-10
Original file line numberDiff line numberDiff line change
@@ -4,19 +4,19 @@ import { notFound } from 'next/navigation';
44
import { GridTileImage } from 'components/grid/tile';
55
import Footer from 'components/layout/footer';
66
import { Gallery } from 'components/product/gallery';
7+
import { ProductProvider } from 'components/product/product-context';
78
import { ProductDescription } from 'components/product/product-description';
89
import { HIDDEN_PRODUCT_TAG } from 'lib/constants';
910
import { getProduct, getProductRecommendations } from 'lib/shopware';
1011
import { Image } from 'lib/shopware/types';
1112
import Link from 'next/link';
1213
import { Suspense } from 'react';
1314

14-
export async function generateMetadata({
15-
params
16-
}: {
17-
params: { handle: string };
15+
export async function generateMetadata(props: {
16+
params: Promise<{ handle: string }>;
1817
}): Promise<Metadata> {
1918
// @ToDo: create a simpler function and do not do the heavy options/variant stuff here
19+
const params = await props.params;
2020
const product = await getProduct(params.handle);
2121

2222
if (!product) return notFound();
@@ -50,7 +50,8 @@ export async function generateMetadata({
5050
};
5151
}
5252

53-
export default async function ProductPage({ params }: { params: { handle: string } }) {
53+
export default async function ProductPage(props: { params: Promise<{ handle: string }> }) {
54+
const params = await props.params;
5455
const product = await getProduct(params.handle);
5556

5657
if (!product) return notFound();
@@ -73,7 +74,7 @@ export default async function ProductPage({ params }: { params: { handle: string
7374
};
7475

7576
return (
76-
<>
77+
<ProductProvider>
7778
<script
7879
type="application/ld+json"
7980
dangerouslySetInnerHTML={{
@@ -89,7 +90,7 @@ export default async function ProductPage({ params }: { params: { handle: string
8990
}
9091
>
9192
<Gallery
92-
images={product.images.map((image: Image) => ({
93+
images={product.images.slice(0, 5).map((image: Image) => ({
9394
src: image.url,
9495
altText: image.altText
9596
}))}
@@ -98,13 +99,15 @@ export default async function ProductPage({ params }: { params: { handle: string
9899
</div>
99100

100101
<div className="basis-full lg:basis-2/6">
101-
<ProductDescription product={product} />
102+
<Suspense fallback={null}>
103+
<ProductDescription product={product} />
104+
</Suspense>
102105
</div>
103106
</div>
104107
<RelatedProducts id={product.id} />
105108
</div>
106109
<Footer />
107-
</>
110+
</ProductProvider>
108111
);
109112
}
110113

@@ -122,7 +125,12 @@ async function RelatedProducts({ id }: { id: string }) {
122125
key={product.path}
123126
className="aspect-square w-full flex-none min-[475px]:w-1/2 sm:w-1/3 md:w-1/4 lg:w-1/5"
124127
>
125-
<Link className="relative h-full w-full" href={`/product/${product.path}`}>
128+
<Link
129+
className="relative h-full w-full"
130+
prefetch={true}
131+
role="link"
132+
href={`/product/${product.path}`}
133+
>
126134
<GridTileImage
127135
alt={product.title}
128136
label={{

app/search/(collection)/[...collection]/page.tsx

+23-11
Original file line numberDiff line numberDiff line change
@@ -1,23 +1,25 @@
11
import { Metadata } from 'next';
22
import { notFound } from 'next/navigation';
33

4+
import Pagination from 'components/collection/pagination';
45
import Grid from 'components/grid';
6+
import ProductGridItems from 'components/layout/product-grid-items';
57
import Collections from 'components/layout/search/collections';
68
import FilterList from 'components/layout/search/filter';
7-
import ProductGridItems from 'components/layout/product-grid-items';
8-
import Pagination from 'components/collection/pagination';
99

10+
import { defaultSort, sorting } from 'lib/constants';
1011
import { getCollection, getCollectionProducts } from 'lib/shopware';
1112
import { transformHandle } from 'lib/shopware/transform';
12-
import { defaultSort, sorting } from 'lib/constants';
1313

1414
export async function generateMetadata({
1515
params
1616
}: {
17-
params: { collection: string };
17+
params: Promise<{ collection: string }>;
1818
}): Promise<Metadata> {
19+
const { collection: collectionParamName } = await params;
20+
1921
// see https://github.com/facebook/react/issues/25994
20-
const collectionName = decodeURIComponent(transformHandle(params?.collection ?? ''));
22+
const collectionName = decodeURIComponent(transformHandle(collectionParamName ?? ''));
2123
if (collectionName.includes('.js.map')) {
2224
return {};
2325
}
@@ -29,22 +31,32 @@ export async function generateMetadata({
2931
return {
3032
title: collection.seo?.title || collection.title,
3133
description:
32-
collection.seo?.description || collection.description || `${collection.title} products`
34+
collection.seo?.description || collection.description || `${collection.title} products`,
35+
openGraph: collection.featuredImage
36+
? {
37+
images: [
38+
{
39+
url: collection.featuredImage
40+
}
41+
]
42+
}
43+
: null
3344
};
3445
}
3546

3647
export default async function CategoryPage({
3748
params,
3849
searchParams
3950
}: {
40-
params: { collection: string };
41-
searchParams?: { [key: string]: string | string[] | undefined };
51+
params: Promise<{ collection: string }>;
52+
searchParams?: Promise<{ [key: string]: string | string[] | undefined }>;
4253
}) {
43-
const { sort, page } = searchParams as { [key: string]: string };
54+
const { collection } = await params;
55+
const { sort, page } = (await searchParams) as { [key: string]: string };
4456
const { sortKey, reverse } = sorting.find((item) => item.slug === sort) || defaultSort;
4557

4658
// see https://github.com/facebook/react/issues/25994
47-
const collectionName = decodeURIComponent(transformHandle(params?.collection ?? ''));
59+
const collectionName = decodeURIComponent(transformHandle(collection ?? ''));
4860
if (collectionName.includes('.js.map')) {
4961
return null;
5062
}
@@ -63,7 +75,7 @@ export default async function CategoryPage({
6375
) : (
6476
<div className="mx-auto flex max-w-screen-2xl flex-col gap-8 px-4 pb-4 text-black md:flex-row dark:text-white">
6577
<div className="order-first w-full flex-none md:max-w-[125px]">
66-
<Collections collection={params.collection} />
78+
<Collections collection={collection} />
6779
</div>
6880
<div className="order-last min-h-screen w-full md:order-none">
6981
<Grid className="grid-cols-2 lg:grid-cols-3">

app/search/(collection)/opengraph-image.tsx

+2-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,8 @@ import { getCollection } from 'lib/shopware';
44
export const runtime = 'edge';
55

66
export default async function Image({ params }: { params: { collection: string } }) {
7-
const collection = await getCollection(params.collection);
7+
const { collection: collectionParamName } = await params;
8+
const collection = await getCollection(collectionParamName);
89
const title = collection?.seo?.title || collection?.title;
910

1011
return await OpengraphImage({ title });

0 commit comments

Comments
 (0)