From 8cf290cf65d7e58a4473cb0c0156ef16c31954a0 Mon Sep 17 00:00:00 2001 From: Adrian Pilarczyk Date: Wed, 26 Jun 2024 15:52:53 +0200 Subject: [PATCH 1/3] chore: add .nvmrc file with version 20 --- .nvmrc | 1 + 1 file changed, 1 insertion(+) create mode 100644 .nvmrc diff --git a/.nvmrc b/.nvmrc new file mode 100644 index 000000000..2edeafb09 --- /dev/null +++ b/.nvmrc @@ -0,0 +1 @@ +20 \ No newline at end of file From c2e99138c101f5e3189cbabcfa6d04dfe27e41fa Mon Sep 17 00:00:00 2001 From: Adrian Pilarczyk Date: Wed, 26 Jun 2024 15:54:01 +0200 Subject: [PATCH 2/3] feat: add boilerplate for dummy payment method --- .../sections/PaymentSection/DummyPayment/DummyDropIn.tsx | 3 +++ src/checkout/sections/PaymentSection/DummyPayment/types.ts | 2 ++ src/checkout/sections/PaymentSection/supportedPaymentApps.ts | 3 +++ 3 files changed, 8 insertions(+) create mode 100644 src/checkout/sections/PaymentSection/DummyPayment/DummyDropIn.tsx create mode 100644 src/checkout/sections/PaymentSection/DummyPayment/types.ts diff --git a/src/checkout/sections/PaymentSection/DummyPayment/DummyDropIn.tsx b/src/checkout/sections/PaymentSection/DummyPayment/DummyDropIn.tsx new file mode 100644 index 000000000..9ede0d0e0 --- /dev/null +++ b/src/checkout/sections/PaymentSection/DummyPayment/DummyDropIn.tsx @@ -0,0 +1,3 @@ +export const DummyDropIn = () => { + return
Dummy DropIn
; +}; diff --git a/src/checkout/sections/PaymentSection/DummyPayment/types.ts b/src/checkout/sections/PaymentSection/DummyPayment/types.ts new file mode 100644 index 000000000..99bb6eb22 --- /dev/null +++ b/src/checkout/sections/PaymentSection/DummyPayment/types.ts @@ -0,0 +1,2 @@ +export const dummyGatewayId = "dummy.payment.saleor.app"; +export type DummyGatewayId = typeof dummyGatewayId; diff --git a/src/checkout/sections/PaymentSection/supportedPaymentApps.ts b/src/checkout/sections/PaymentSection/supportedPaymentApps.ts index 545c982f3..63e63fbeb 100644 --- a/src/checkout/sections/PaymentSection/supportedPaymentApps.ts +++ b/src/checkout/sections/PaymentSection/supportedPaymentApps.ts @@ -1,9 +1,12 @@ import { AdyenDropIn } from "./AdyenDropIn/AdyenDropIn"; import { adyenGatewayId } from "./AdyenDropIn/types"; +import { DummyDropIn } from "./DummyPayment/DummyDropIn"; +import { dummyGatewayId } from "./DummyPayment/types"; import { StripeComponent } from "./StripeElements/stripeComponent"; import { stripeGatewayId } from "./StripeElements/types"; export const paymentMethodToComponent = { [adyenGatewayId]: AdyenDropIn, [stripeGatewayId]: StripeComponent, + [dummyGatewayId]: DummyDropIn, }; From 5f4fa3dda34852dd2902631f2e3398b915c3f353 Mon Sep 17 00:00:00 2001 From: Adrian Pilarczyk Date: Mon, 1 Jul 2024 18:26:58 +0200 Subject: [PATCH 3/3] feat: add working dummy payment gateway --- src/checkout/hooks/useCheckoutComplete.ts | 8 +- .../DummyPayment/DummyDropIn.tsx | 78 ++++++++++++++++++- .../PaymentSection/PaymentMethods.tsx | 6 ++ src/checkout/sections/PaymentSection/types.ts | 8 +- .../usePaymentGatewaysInitialize.ts | 5 +- 5 files changed, 96 insertions(+), 9 deletions(-) diff --git a/src/checkout/hooks/useCheckoutComplete.ts b/src/checkout/hooks/useCheckoutComplete.ts index 3eb4d651e..d72c0c653 100644 --- a/src/checkout/hooks/useCheckoutComplete.ts +++ b/src/checkout/hooks/useCheckoutComplete.ts @@ -1,4 +1,5 @@ import { useMemo } from "react"; +import { useAlerts } from "./useAlerts"; import { useCheckoutCompleteMutation } from "@/checkout/graphql"; import { useCheckout } from "@/checkout/hooks/useCheckout"; import { useSubmit } from "@/checkout/hooks/useSubmit"; @@ -9,6 +10,7 @@ export const useCheckoutComplete = () => { checkout: { id: checkoutId }, } = useCheckout(); const [{ fetching }, checkoutComplete] = useCheckoutCompleteMutation(); + const { showCustomErrors } = useAlerts(); const onCheckoutComplete = useSubmit<{}, typeof checkoutComplete>( useMemo( @@ -30,8 +32,12 @@ export const useCheckoutComplete = () => { window.location.href = newUrl; } }, + onError: ({ errors }) => { + const error = errors.join(", "); + showCustomErrors([{ message: error }]); + }, }), - [checkoutComplete, checkoutId], + [checkoutComplete, checkoutId, showCustomErrors], ), ); return { completingCheckout: fetching, onCheckoutComplete }; diff --git a/src/checkout/sections/PaymentSection/DummyPayment/DummyDropIn.tsx b/src/checkout/sections/PaymentSection/DummyPayment/DummyDropIn.tsx index 9ede0d0e0..8477ea9b8 100644 --- a/src/checkout/sections/PaymentSection/DummyPayment/DummyDropIn.tsx +++ b/src/checkout/sections/PaymentSection/DummyPayment/DummyDropIn.tsx @@ -1,3 +1,77 @@ -export const DummyDropIn = () => { - return
Dummy DropIn
; +import React from "react"; +import { type ParsedDummyGateway } from "../types"; +import { apiErrorMessages } from "../errorMessages"; +import { + useTransactionInitializeMutation, + type TransactionInitializeMutationVariables, +} from "@/checkout/graphql"; +import { useCheckout } from "@/checkout/hooks/useCheckout"; +import { useSubmit } from "@/checkout/hooks/useSubmit"; +import { Button } from "@/checkout/components"; +import { useCheckoutComplete } from "@/checkout/hooks/useCheckoutComplete"; +import { useAlerts } from "@/checkout/hooks/useAlerts"; +import { useErrorMessages } from "@/checkout/hooks/useErrorMessages"; + +type DummyDropinProps = { + config: ParsedDummyGateway; +}; + +const useDummyDropIn = (props: DummyDropinProps) => { + const { showCustomErrors } = useAlerts(); + const { errorMessages: commonErrorMessages } = useErrorMessages(apiErrorMessages); + + const { checkout } = useCheckout(); + const { onCheckoutComplete, completingCheckout: isCompleteCheckoutLoading } = useCheckoutComplete(); + + const [{ fetching: isTransactionInitializeLoading }, transactionInitialize] = + useTransactionInitializeMutation(); + + const isLoading = isCompleteCheckoutLoading || isTransactionInitializeLoading; + const isValid = !!checkout.shippingAddress && !!checkout.email && !!checkout.billingAddress; + + const isDisabled = isLoading || !isValid; + + const onTransactionInitialize = useSubmit< + TransactionInitializeMutationVariables, + typeof transactionInitialize + >( + React.useMemo( + () => ({ + onSubmit: transactionInitialize, + onError: (error) => { + console.error(error); + showCustomErrors([{ message: commonErrorMessages.somethingWentWrong }]); + }, + onSuccess: async () => { + void onCheckoutComplete(); + }, + }), + [commonErrorMessages.somethingWentWrong, onCheckoutComplete, showCustomErrors, transactionInitialize], + ), + ); + + const onSubmit = async () => { + await onTransactionInitialize({ + checkoutId: checkout.id, + paymentGateway: { + id: props.config.id, + }, + }); + }; + + return { onSubmit, isLoading, isDisabled }; +}; + +export const DummyDropIn = (props: DummyDropinProps) => { + const { onSubmit, isLoading, isDisabled } = useDummyDropIn(props); + + return ( +
+

Dummy Payment Gateway

+ {/*

+ Stored payment methods +

*/} +
+ ); }; diff --git a/src/checkout/sections/PaymentSection/PaymentMethods.tsx b/src/checkout/sections/PaymentSection/PaymentMethods.tsx index 61c71d7ef..a6b1622ed 100644 --- a/src/checkout/sections/PaymentSection/PaymentMethods.tsx +++ b/src/checkout/sections/PaymentSection/PaymentMethods.tsx @@ -19,6 +19,12 @@ export const PaymentMethods = () => {
{availablePaymentGateways.map((gateway) => { const Component = paymentMethodToComponent[gateway.id]; + + if (!Component) { + console.warn(`No component found for gateway ${gateway.id}`); + return null; + } + return ( ; export type ParsedStripeGateway = ParsedPaymentGateway; +export type ParsedDummyGateway = ParsedPaymentGateway; -export type ParsedPaymentGateways = ReadonlyArray; +export type ParsedPaymentGateways = ReadonlyArray< + ParsedAdyenGateway | ParsedStripeGateway | ParsedDummyGateway +>; export interface ParsedPaymentGateway> extends Omit { diff --git a/src/checkout/sections/PaymentSection/usePaymentGatewaysInitialize.ts b/src/checkout/sections/PaymentSection/usePaymentGatewaysInitialize.ts index 8842f92bf..b85dc5ef0 100644 --- a/src/checkout/sections/PaymentSection/usePaymentGatewaysInitialize.ts +++ b/src/checkout/sections/PaymentSection/usePaymentGatewaysInitialize.ts @@ -8,10 +8,7 @@ import { getFilteredPaymentGateways } from "@/checkout/sections/PaymentSection/u export const usePaymentGatewaysInitialize = () => { const { - checkout: { billingAddress }, - } = useCheckout(); - const { - checkout: { id: checkoutId, availablePaymentGateways }, + checkout: { id: checkoutId, availablePaymentGateways, billingAddress }, } = useCheckout(); const billingCountry = billingAddress?.country.code as MightNotExist;