From 97fafd6c3dc291b94f416d2353456ae328de949b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Micha=C5=82=20Miszczyszyn?= Date: Mon, 30 Oct 2023 11:23:19 +0100 Subject: [PATCH] Stripe integration (#990) --- __tests__/STF_01.spec.ts | 9 +- __tests__/STF_03.spec.ts | 9 +- package.json | 2 + pnpm-lock.yaml | 24 ++- src/app/(main)/cart/CheckoutLink.tsx | 5 +- src/app/(main)/cart/page.tsx | 106 +++++----- src/app/(main)/layout.tsx | 6 +- src/app/(main)/products/[slug]/page.tsx | 52 ++--- src/app/checkout/page.tsx | 8 +- src/app/layout.tsx | 4 +- .../components/AddressForm/AddressForm.tsx | 11 +- .../AddressForm/useAddressFormSchema.ts | 2 +- .../AddressForm/useAddressFormUtils.ts | 2 +- src/checkout/hooks/useAlerts/useAlerts.tsx | 2 +- src/checkout/hooks/useCheckout.ts | 8 +- src/checkout/hooks/useCustomerAttach.ts | 9 +- src/checkout/hooks/useSubmit/utils.ts | 3 +- .../AdyenDropIn/errorMessages.ts | 34 ---- .../AdyenDropIn/useAdyenDropin.ts | 16 +- .../PaymentSection/PaymentMethods.tsx | 19 +- .../StripeElements/StripePaymentComponent.tsx | 0 .../StripeElements/stripeComponent.tsx | 64 ++++++ .../StripeElements/stripeElementsForm.tsx | 191 ++++++++++++++++++ .../PaymentSection/StripeElements/types.ts | 2 + .../sections/PaymentSection/errorMessages.ts | 33 +++ .../PaymentSection/supportedPaymentApps.ts | 9 + src/checkout/sections/PaymentSection/types.ts | 14 +- .../usePaymentGatewaysInitialize.ts | 11 +- .../sections/PaymentSection/usePayments.ts | 2 +- src/checkout/sections/PaymentSection/utils.ts | 46 +---- .../checkoutValidationStateStore.ts | 58 +++--- .../updateStateStore/updateStateStore.ts | 97 ++++----- src/checkout/views/Checkout/Checkout.tsx | 4 +- src/graphql/CheckoutCreate.graphql | 41 ++++ src/lib/checkout.ts | 30 ++- .../components/nav/components/CartNavItem.tsx | 4 +- 36 files changed, 635 insertions(+), 302 deletions(-) create mode 100644 src/checkout/sections/PaymentSection/StripeElements/StripePaymentComponent.tsx create mode 100644 src/checkout/sections/PaymentSection/StripeElements/stripeComponent.tsx create mode 100644 src/checkout/sections/PaymentSection/StripeElements/stripeElementsForm.tsx create mode 100644 src/checkout/sections/PaymentSection/StripeElements/types.ts create mode 100644 src/checkout/sections/PaymentSection/errorMessages.ts create mode 100644 src/checkout/sections/PaymentSection/supportedPaymentApps.ts diff --git a/__tests__/STF_01.spec.ts b/__tests__/STF_01.spec.ts index 6f2649d5d..f7f0936fd 100644 --- a/__tests__/STF_01.spec.ts +++ b/__tests__/STF_01.spec.ts @@ -21,8 +21,9 @@ test("STF_01: Add items to the basket", async ({ page }) => { await openCart({ page }); - await expect(page.getByTestId("CartProductList").getByRole("listitem")).toHaveCount(1); - await expect(page.getByTestId("CartProductList").getByRole("listitem")).toContainText(product.name); - await expect(page.getByTestId("CartProductList").getByRole("listitem")).toContainText(`Qty: 1`); - await expect(page.getByTestId("CartProductList").getByRole("listitem")).toContainText(price.toFixed(2)); + const productInCart = page.getByTestId("CartProductList").getByRole("listitem"); + await expect(productInCart).toHaveCount(1); + await expect(productInCart).toContainText(product.name); + await expect(productInCart).toContainText(`Qty: 1`); + await expect(productInCart).toContainText(price.toFixed(2)); }); diff --git a/__tests__/STF_03.spec.ts b/__tests__/STF_03.spec.ts index 3d5a32a47..9b5113a7f 100644 --- a/__tests__/STF_03.spec.ts +++ b/__tests__/STF_03.spec.ts @@ -24,8 +24,9 @@ test("STF_03: Check if price are calculating correctly", async ({ page }) => { await openCart({ page }); const totalPrice = (price * 2).toFixed(2); - await expect(page.getByTestId("CartProductList").getByRole("listitem")).toHaveCount(1); - await expect(page.getByTestId("CartProductList").getByRole("listitem")).toContainText(product.name); - await expect(page.getByTestId("CartProductList").getByRole("listitem")).toContainText(`Qty: 2`); - await expect(page.getByTestId("CartProductList").getByRole("listitem")).toContainText(totalPrice); + const productInCart = page.getByTestId("CartProductList").getByRole("listitem"); + await expect(productInCart).toHaveCount(1); + await expect(productInCart).toContainText(product.name); + await expect(productInCart).toContainText(`Qty: 2`); + await expect(productInCart).toContainText(totalPrice); }); diff --git a/package.json b/package.json index 1566f1232..f2b7b79a8 100644 --- a/package.json +++ b/package.json @@ -18,6 +18,8 @@ "@adyen/adyen-web": "5.53.2", "@adyen/api-library": "14.3.0", "@headlessui/react": "1.7.17", + "@stripe/react-stripe-js": "2.3.1", + "@stripe/stripe-js": "2.1.10", "@saleor/auth-sdk": "0.14.0", "clsx": "2.0.0", "editorjs-html": "3.4.3", diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index d546dace1..90d895b74 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -20,6 +20,12 @@ dependencies: '@saleor/auth-sdk': specifier: 0.14.0 version: 0.14.0(next@14.0.0)(react-dom@18.2.0)(react@18.2.0)(urql@4.0.5) + '@stripe/react-stripe-js': + specifier: 2.3.1 + version: 2.3.1(@stripe/stripe-js@2.1.10)(react-dom@18.2.0)(react@18.2.0) + '@stripe/stripe-js': + specifier: 2.1.10 + version: 2.1.10 clsx: specifier: 2.0.0 version: 2.0.0 @@ -1812,6 +1818,23 @@ packages: urql: 4.0.5(graphql@16.8.1)(react@18.2.0) dev: false + /@stripe/react-stripe-js@2.3.1(@stripe/stripe-js@2.1.10)(react-dom@18.2.0)(react@18.2.0): + resolution: {integrity: sha512-vXiwcG2ZjAF4AezjP7DJ8jiwxfCWCen/X2rBhyXaKrfQ7+pwmXhsoUlKRa0eLWioY1oelOQOafauNUiwTwFHgQ==} + peerDependencies: + '@stripe/stripe-js': ^1.44.1 || ^2.0.0 + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + react-dom: ^16.8.0 || ^17.0.0 || ^18.0.0 + dependencies: + '@stripe/stripe-js': 2.1.10 + prop-types: 15.8.1 + react: 18.2.0 + react-dom: 18.2.0(react@18.2.0) + dev: false + + /@stripe/stripe-js@2.1.10: + resolution: {integrity: sha512-h79zhwvxAJVAvtVjtMoz++DtwI7GdcEItmTC0P2gciZoFUeAO6XX9DL+UXm9uADiEaUvTKqrExYwtBTlMYAaPA==} + dev: false + /@swc/helpers@0.5.2: resolution: {integrity: sha512-E4KcWTpoLHqwPHLxidpOqQbcrZVgi0rsmmZXUle1jXmJfuIf/UWpczUJ7MZZ5tlxytgJXyp0w4PGkkeLiuIdZw==} dependencies: @@ -5146,7 +5169,6 @@ packages: loose-envify: 1.4.0 object-assign: 4.1.1 react-is: 16.13.1 - dev: true /property-expr@2.0.6: resolution: {integrity: sha512-SVtmxhRE/CGkn3eZY1T6pC8Nln6Fr/lu1mKSgRud0eC73whjGfoAogbn78LkD8aFL0zz3bAFerKSnOl7NlErBA==} diff --git a/src/app/(main)/cart/CheckoutLink.tsx b/src/app/(main)/cart/CheckoutLink.tsx index 3701257c8..b5ec40885 100644 --- a/src/app/(main)/cart/CheckoutLink.tsx +++ b/src/app/(main)/cart/CheckoutLink.tsx @@ -3,15 +3,16 @@ type Props = { disabled?: boolean; checkoutId?: string; + className?: string; }; -export const CheckoutLink = ({ disabled, checkoutId }: Props) => { +export const CheckoutLink = ({ disabled, checkoutId, className = "" }: Props) => { return ( disabled && e.preventDefault()} href={`/checkout?checkout=${checkoutId}`} - className="w-full rounded border border-transparent bg-neutral-900 px-6 py-3 text-center font-medium text-neutral-50 hover:bg-neutral-800 aria-disabled:cursor-not-allowed aria-disabled:bg-neutral-500 sm:col-start-2" + className={`inline-block max-w-full rounded border border-transparent bg-neutral-900 px-6 py-3 text-center font-medium text-neutral-50 hover:bg-neutral-800 aria-disabled:cursor-not-allowed aria-disabled:bg-neutral-500 sm:px-16 ${className}`} > Checkout diff --git a/src/app/(main)/cart/page.tsx b/src/app/(main)/cart/page.tsx index 4a03e8731..57ae3fb38 100644 --- a/src/app/(main)/cart/page.tsx +++ b/src/app/(main)/cart/page.tsx @@ -14,55 +14,70 @@ export default async function Page() { const checkoutId = cookies().get("checkoutId")?.value || ""; const checkout = await Checkout.find(checkoutId); - const lines = checkout ? checkout.lines : []; + + if (!checkout) { + return ( +
+

Your Shopping Cart is empty

+

+ Looks like you haven’t added any items to the cart yet. +

+ + Go back + +
+ ); + } return (

Your Shopping Cart

-
-
    - {lines.map((item) => ( -
  • -
    - {item.variant?.product?.thumbnail?.url && ( - {item.variant.product.thumbnail.alt - )} -
    -
    -
    -
    - -

    {item.variant?.product?.name}

    - -

    {item.variant?.product?.category?.name}

    - {item.variant.name !== item.variant.id && Boolean(item.variant.name) && ( -

    Variant: {item.variant.name}

    - )} -
    -

    - {formatMoney(item.totalPrice.gross.amount, item.totalPrice.gross.currency)} -

    -
    -
    -
    Qty: {item.quantity}
    - +
      + {checkout.lines.map((item) => ( +
    • +
      + {item.variant?.product?.thumbnail?.url && ( + {item.variant.product.thumbnail.alt + )} +
      +
      +
      +
      + +

      {item.variant?.product?.name}

      + +

      {item.variant?.product?.category?.name}

      + {item.variant.name !== item.variant.id && Boolean(item.variant.name) && ( +

      Variant: {item.variant.name}

      + )}
      +

      + {formatMoney(item.totalPrice.gross.amount, item.totalPrice.gross.currency)} +

      -
    • - ))} -
    -
    +
    +
    Qty: {item.quantity}
    + +
    +
    +
  • + ))} +
+
@@ -71,13 +86,12 @@ export default async function Page() {

Shipping will be calculated in the next step

- {checkout && - formatMoney(checkout.totalPrice.gross.amount, checkout.totalPrice.gross.currency)} + {formatMoney(checkout.totalPrice.gross.amount, checkout.totalPrice.gross.currency)}
-
- +
+
diff --git a/src/app/(main)/layout.tsx b/src/app/(main)/layout.tsx index 16cb0dbbb..84c0547c0 100644 --- a/src/app/(main)/layout.tsx +++ b/src/app/(main)/layout.tsx @@ -11,8 +11,10 @@ export default function RootLayout(props: { children: ReactNode }) { return ( <>
-
{props.children}
-