Форма оплаты Stripe постоянно перерисовывается с использованием @stripe/react-stripe-js.

У меня возникли проблемы с настройкой оплаты Stripe в интерфейсе Medusa-JS Next JS.

При вводе данных карты вcheckoutFormцеликомStripePaymentпостоянно перерисовывается при нажатии на что-либо внутри моей формы, что не являетсяPaymentElement.

У меня есть подозрение, что это связано с изменениями в useEffect моего компонента Stripe, но я не понимаю, почему его следует обновлять внутри моей формы.

Мой код

StripePayment — компонент

      "use client";
import { useState, useEffect } from "react";
import { loadStripe } from "@stripe/stripe-js";
import { Elements } from "@stripe/react-stripe-js";
import CheckoutForm from "./form";

const STRIPE_PK_KEY = process.env.NEXT_PUBLIC_STRIPE_PK_KEY;

const StripePayment = ({ cart }) => {
  const [stripePromise, setStripePromise] = useState(null);
  const [clientSecret, setClientSecret] = useState();

  useEffect(() => {
    if (!STRIPE_PK_KEY && !cart && !cart.payment_sessions) {
      return null;
    }
    const isStripeAvailable = cart.payment_sessions?.some(
      (session) => session.provider_id === "stripe"
    );
    if (!isStripeAvailable) {
      return;
    }
    setStripePromise(loadStripe(STRIPE_PK_KEY));
    setClientSecret(cart.payment_session.data.client_secret);
  }, [cart.id]);

  return (
    <div>
      <h1>Stripe payment</h1>
      {stripePromise && clientSecret && (
        <Elements stripe={stripePromise} options={{ clientSecret }}>
          <CheckoutForm clientSecret={clientSecret} cart={cart} />
        </Elements>
      )}
    </div>
  );
};

export default StripePayment;

CheckoutForm — компонент

      "use client";
import {
  useElements,
  useStripe,
  PaymentElement,
} from "@stripe/react-stripe-js";
import { useState } from "react";
import medusaClient from "../../../lib/medusaClient";

export default function Form({ clientSecret, cart }) {
  const stripe = useStripe();
  const elements = useElements();

  const [message, setMessage] = useState("");
  const [isProcessing, setIsProcessing] = useState(false);

  const handleSubmit = async (e) => {
    e.preventDefault();

    if (!stripe || !elements) {
      console.log("No tripe or elements");
      return;
    }

    setIsProcessing(true);

    const { error, paymentIntent } = await stripe.confirmCardPayment(
      clientSecret,
      {
        payment_method: {
          card: elements.getElement(PaymentElement),
          billing_details: {
            name: e.target.name.value,
            email: e.target.email.value,
            phone: e.target.phone.value,
            address: {
              city: cart.shipping_address.city,
              country: cart.shipping_address.country,
              line1: cart.shipping_address.line1,
              line2: cart.shipping_address.line2,
              postal_code: cart.shipping_address.postal_code,
            },
          },
        },
      }
    );

    if (error) {
      setMessage(error.message);
    } else if (paymentIntent && paymentIntent.status === "succeeded") {
      setMessage("Payment succeeded");
      try {
        await medusaClient.carts.complete(cart.id);
        console.log("Order completed");
      } catch (err) {
        console.error("Error completing order:", err);
      }
    } else {
      setMessage("An unexpected error occurred");
    }
    console.log("Message: ", message);
    setIsProcessing(false);
  };

  return (
    <form id="payment-form" onSubmit={(e) => handleSubmit(e)}>
            {/* INPUT LABELS OMITTED FOR SAKE OF READIBILITY */}
      <PaymentElement id="payment-element" />
      <button
        className="px-4 py-2 mt-4 text-white bg-blue-500"
        disabled={isProcessing || !stripe || !elements}
        id="submit"
      >
        <span id="button-text">
          {isProcessing ? "Processing ... " : "Pay now"}
        </span>
      </button>
      {/* Show any error or success messages */}
      {message && <div id="payment-message">{message}</div>}
    </form>
  );
}

1 ответ

Я вижу в вашем коде некоторые необычные вещи:

  • Вы используетеconfirmCardPayment, который используется при использовании CardElement, если вы используете<PaymentElement>ты должен использоватьconfirmPaymentвместо
  • В связи с этим я не понимаю, почему вы передаете clientSecret компоненту CheckoutForm; поскольку вам это не нужно - вы просто используетеuseElements()который предоставляет глобальный экземпляр Stripe, который уже инициализирован и знает clientSecret. Может быть, эта дополнительная зависимость вызывает некоторые проблемы с повторным рендерингом, которые вы видите?

Обычная интеграция выглядит примерно так:

      Elements stripe={stripePromise} options={{ clientSecret }}>
  <CheckoutForm cart={cart} />
</Elements>

Форма оформления заказа:

      ...
 const { error } = await stripe.confirmPayment({
      elements, // from useElements ; you do not need to pass the clientSecret 
      confirmParams: {
        // Make sure to change this to your payment completion page
        return_url: `${window.location.origin}/completion`,
        payment_method_data: {
                billing_details:{
                  name: e.target.name.value,
                  email: e.target.email.value, 
                  ...
                }
        }
      },
    });
...

https://github.com/stripe-samples/accept-a-pay/tree/504ffe70e4ac54a21c2c8ce3b5c426c12df6f351/pay-element/client/react-cra/src

https://stripe.com/docs/pays/accept-a-pay?platform=web&amp;amp;ui=elements&amp;amp;client=react#web-collect-pay-details

Я бы попытался переписать ваш код в соответствии с документацией Stripe и посмотреть, окажет ли это влияние.

Другие вопросы по тегам