Skip to content

Commit 92f3568

Browse files
authored
Merge pull request #27 from shopwareLabs/fix/update-sync-main
Update sync with vercel origin
2 parents f6c0e07 + 29f10a7 commit 92f3568

21 files changed

+277
-222
lines changed

.env.example

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
COMPANY_NAME="shopware AG"
22
TWITTER_CREATOR="@shopware"
33
TWITTER_SITE="https://www.shopware.com/en/solutions/shopware-composable-frontends/"
4-
SITE_NAME="Next.js Shopware Starter with Composable Frontends"
4+
SITE_NAME="Next.js Commerce with Shopware Composable Frontends"
55
SHOPWARE_STORE_DOMAIN="https://demo-frontends.shopware.store"
66
SHOPWARE_API_TYPE="store-api"
77
SHOPWARE_ACCESS_TOKEN="SWSCBHFSNTVMAWNZDNFKSHLAYW"

.github/workflows/e2e-tests.yml

+2-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,8 @@ jobs:
1616
uses: actions/setup-node@v3
1717
with:
1818
node-version: 22
19-
19+
- name: Install corepack
20+
run: npm add -g corepack
2021
- run: corepack enable
2122
- run: pnpm --version
2223
- uses: actions/setup-node@v3

README.md

-3
Original file line numberDiff line numberDiff line change
@@ -18,9 +18,6 @@ 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-
2421
## Running locally
2522

2623
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.

api-types/storeApiSchema.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -11816,7 +11816,7 @@
1181611816
"label": { "type": "string" },
1181711817
"modified": { "type": "boolean" },
1181811818
"modifiedByApp": { "type": "boolean" },
11819-
"payload": { "$ref": "#/components/schemas/ProductJsonApi" },
11819+
"payload": { "$ref": "#/components/schemas/Product" },
1182011820
"price": {
1182111821
"type": "object",
1182211822
"properties": {

api-types/storeApiTypes.d.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -2358,7 +2358,7 @@ export type Schemas = {
23582358
label?: string;
23592359
modified?: boolean;
23602360
modifiedByApp?: boolean;
2361-
payload?: components['schemas']['ProductJsonApi'];
2361+
payload?: components['schemas']['Product'];
23622362
price?: {
23632363
/** @enum {string} */
23642364
apiAlias: 'calculated_price';

components/cart/actions.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -146,11 +146,11 @@ export async function removeItem(prevState: unknown, lineId?: string) {
146146
const cartId = (await cookies()).get('sw-context-token')?.value;
147147

148148
if (!cartId) {
149-
return 'Missing cart ID';
149+
return 'Cart ID is missing';
150150
}
151151

152152
if (!lineId) {
153-
return 'Missing line ID';
153+
return 'Line ID is missing';
154154
}
155155

156156
try {

components/cart/add-to-cart.tsx

+25-34
Original file line numberDiff line numberDiff line change
@@ -3,26 +3,25 @@
33
import { PlusIcon } from '@heroicons/react/24/outline';
44
import clsx from 'clsx';
55
import { addItem } from 'components/cart/actions';
6-
import LoadingDots from 'components/loading-dots';
6+
import { useProduct } from 'components/product/product-context';
77
import { Product, ProductVariant } from 'lib/shopware/types';
8-
import { useSearchParams } from 'next/navigation';
98
import { useActionState } from 'react';
10-
import { useFormStatus } from 'react-dom';
9+
import { useCart } from './cart-context';
10+
1111
function SubmitButton({
1212
availableForSale,
1313
selectedVariantId
1414
}: {
1515
availableForSale: boolean;
1616
selectedVariantId: string | undefined;
1717
}) {
18-
const { pending } = useFormStatus();
1918
const buttonClasses =
2019
'relative flex w-full items-center justify-center rounded-full bg-blue-600 p-4 tracking-wide text-white';
2120
const disabledClasses = 'cursor-not-allowed opacity-60 hover:opacity-60';
2221

2322
if (!availableForSale) {
2423
return (
25-
<button aria-disabled className={clsx(buttonClasses, disabledClasses)}>
24+
<button disabled className={clsx(buttonClasses, disabledClasses)}>
2625
Out Of Stock
2726
</button>
2827
);
@@ -32,7 +31,7 @@ function SubmitButton({
3231
return (
3332
<button
3433
aria-label="Please select an option"
35-
aria-disabled
34+
disabled
3635
className={clsx(buttonClasses, disabledClasses)}
3736
>
3837
<div className="absolute left-0 ml-4">
@@ -45,52 +44,44 @@ function SubmitButton({
4544

4645
return (
4746
<button
48-
onClick={(e: React.FormEvent<HTMLButtonElement>) => {
49-
if (pending) e.preventDefault();
50-
}}
5147
aria-label="Add to cart"
52-
aria-disabled={pending}
5348
className={clsx(buttonClasses, {
54-
'hover:opacity-90': true,
55-
[disabledClasses]: pending
49+
'hover:opacity-90': true
5650
})}
5751
>
5852
<div className="absolute left-0 ml-4">
59-
{pending ? <LoadingDots className="mb-3 bg-white" /> : <PlusIcon className="h-5" />}
53+
<PlusIcon className="h-5" />
6054
</div>
6155
Add To Cart
6256
</button>
6357
);
6458
}
6559

66-
export function AddToCart({
67-
product,
68-
variants,
69-
availableForSale
70-
}: {
71-
product: Product;
72-
variants: ProductVariant[];
73-
availableForSale: boolean;
74-
}) {
60+
export function AddToCart({ product }: { product: Product }) {
61+
const { variants, availableForSale } = product;
62+
const { addCartItem } = useCart();
63+
const { state } = useProduct();
7564
const [message, formAction] = useActionState(addItem, null);
76-
const searchParams = useSearchParams();
77-
const defaultVariantId = variants.length === 1 ? variants[0]?.id : product.id;
78-
const variant = variants.find((variant) =>
79-
variant.selectedOptions.every(
80-
(option) => option.value === searchParams.get(option.name.toLowerCase())
81-
)
65+
66+
const variant = variants.find((variant: ProductVariant) =>
67+
variant.selectedOptions.every((option) => option.value === state[option.name.toLowerCase()])
8268
);
69+
const defaultVariantId = variants.length === 1 ? variants[0]?.id : product.id;
8370
const selectedVariantId = variant?.id || defaultVariantId;
8471
const actionWithVariant = formAction.bind(null, selectedVariantId);
72+
const finalVariant = variants.find((variant) => variant.id === selectedVariantId)! || product;
8573

8674
return (
87-
<form action={actionWithVariant}>
75+
<form
76+
action={async () => {
77+
addCartItem(finalVariant, product);
78+
await actionWithVariant();
79+
}}
80+
>
8881
<SubmitButton availableForSale={availableForSale} selectedVariantId={selectedVariantId} />
89-
<div className="flex items-center px-4 py-3 text-sm font-bold text-black">
90-
<p aria-live="polite" className="h-6" role="status">
91-
{message}
92-
</p>
93-
</div>
82+
<p aria-live="polite" className="sr-only" role="status">
83+
{message}
84+
</p>
9485
</form>
9586
);
9687
}

components/cart/cart-context.tsx

+13-2
Original file line numberDiff line numberDiff line change
@@ -50,16 +50,27 @@ function createOrUpdateCartItem(
5050
variant: ProductVariant,
5151
product: Product
5252
): CartItem {
53+
const productPrice = variant.price?.amount
54+
? variant.price.amount
55+
: product.priceRange.minVariantPrice.amount === product.priceRange.maxVariantPrice.amount
56+
? product.priceRange.minVariantPrice.amount
57+
: '0';
5358
const quantity = existingItem ? existingItem.quantity + 1 : 1;
54-
const totalAmount = calculateItemCost(quantity, variant.price.amount);
59+
const totalAmount = calculateItemCost(quantity, productPrice);
60+
const currencyCode = variant.price?.currencyCode
61+
? variant.price?.currencyCode
62+
: product.priceRange.minVariantPrice.currencyCode ===
63+
product.priceRange.maxVariantPrice.currencyCode
64+
? product.priceRange.minVariantPrice.currencyCode
65+
: 'EUR';
5566

5667
return {
5768
id: existingItem?.id,
5869
quantity,
5970
cost: {
6071
totalAmount: {
6172
amount: totalAmount,
62-
currencyCode: variant.price.currencyCode
73+
currencyCode: currencyCode
6374
}
6475
},
6576
merchandise: {

components/cart/edit-item-quantity-button.tsx

-2
Original file line numberDiff line numberDiff line change
@@ -39,9 +39,7 @@ function SubmitButton({ type }: { type: 'plus' | 'minus' }) {
3939

4040
type EditItemQuantityButtonProps = {
4141
item: CartItem;
42-
4342
type: 'plus' | 'minus';
44-
4543
optimisticUpdate: (merchandiseId: string, updateType: UpdateType) => void;
4644
};
4745

components/cart/index.tsx

-15
This file was deleted.

components/cart/modal.tsx

+21-7
Original file line numberDiff line numberDiff line change
@@ -95,11 +95,13 @@ export default function CartModal() {
9595
?.map((item, i) => {
9696
const merchandiseSearchParams = {} as MerchandiseSearchParams;
9797

98-
item.merchandise.selectedOptions.forEach(({ name, value }) => {
99-
if (value !== DEFAULT_OPTION) {
100-
merchandiseSearchParams[name.toLowerCase()] = value;
101-
}
102-
});
98+
if (item.merchandise.selectedOptions) {
99+
item.merchandise.selectedOptions.forEach(({ name, value }) => {
100+
if (value !== DEFAULT_OPTION) {
101+
merchandiseSearchParams[name?.toLowerCase()] = value;
102+
}
103+
});
104+
}
103105

104106
const merchandiseUrl = createUrl(
105107
`/product/${item.merchandise.product.path}`,
@@ -137,7 +139,19 @@ export default function CartModal() {
137139
<span className="leading-tight">
138140
{item.merchandise.product.title}
139141
</span>
140-
{item.merchandise.title !== DEFAULT_OPTION ? (
142+
{item.merchandise.selectedOptions?.length
143+
? item.merchandise.selectedOptions.map((option) => (
144+
<p
145+
key={option.name}
146+
className="text-xs text-neutral-500 dark:text-neutral-400"
147+
>
148+
{option.name}: {option.value}
149+
</p>
150+
))
151+
: null}
152+
153+
{item.merchandise.title !==
154+
item.merchandise.product.seo.title ? (
141155
<p className="text-sm text-neutral-500 dark:text-neutral-400">
142156
{item.merchandise.title}
143157
</p>
@@ -155,7 +169,7 @@ export default function CartModal() {
155169
<EditItemQuantityButton
156170
item={item}
157171
type="minus"
158-
optimisticUpdate={updateCartItem}
172+
optimisticUpdate={(item.id, updateCartItem)}
159173
/>
160174
<p className="w-6 text-center">
161175
<span className="w-full text-sm">{item.quantity}</span>

components/layout/navbar/index.tsx

+2-5
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,4 @@
1-
import Cart from 'components/cart';
2-
import OpenCart from 'components/cart/open-cart';
1+
import CartModal from 'components/cart/modal';
32
import LogoSquare from 'components/logo-square';
43
import { getMenu } from 'lib/shopware';
54
import { Menu } from 'lib/shopware/types';
@@ -45,9 +44,7 @@ export default async function Navbar() {
4544
</Suspense>
4645
</div>
4746
<div className="flex justify-end md:w-1/6">
48-
<Suspense fallback={<OpenCart />}>
49-
<Cart />
50-
</Suspense>
47+
<CartModal />
5148
</div>
5249
</div>
5350
</nav>

components/product/product-description.tsx

+1-5
Original file line numberDiff line numberDiff line change
@@ -22,11 +22,7 @@ export function ProductDescription({ product }: { product: Product }) {
2222
</Suspense>
2323

2424
<Suspense fallback={null}>
25-
<AddToCart
26-
product={product}
27-
variants={product.variants}
28-
availableForSale={product.availableForSale}
29-
/>
25+
<AddToCart product={product} />
3026
</Suspense>
3127

3228
{product.descriptionHtml ? (

0 commit comments

Comments
 (0)