import React, { FormEvent } from 'react'; import { CardElement, useElements, useStripe } from '@stripe/react-stripe-js'; import { useTranslation } from 'react-i18next'; import { GatewayFormProps } from '../abstract-payment-modal'; import { PaymentConfirmation, StripeSubscription } from '../../../models/payment'; import StripeAPI from '../../../api/stripe'; import { Invoice } from '../../../models/invoice'; /** * A form component to collect the credit card details and to create the payment method on Stripe. * The form validation button must be created elsewhere, using the attribute form={formId}. */ export const StripeForm: React.FC = ({ onSubmit, onSuccess, onError, children, className, paymentSchedule = false, cart, customer, operator, formId }) => { const { t } = useTranslation('shared'); const stripe = useStripe(); const elements = useElements(); /** * Handle the submission of the form. Depending on the configuration, it will create the payment method on Stripe, * or it will process a payment with the inputted card. */ const handleSubmit = async (event: FormEvent): Promise => { event.preventDefault(); onSubmit(); // Stripe.js has not loaded yet if (!stripe || !elements) { return; } const cardElement = elements.getElement(CardElement); const { error, paymentMethod } = await stripe.createPaymentMethod({ type: 'card', card: cardElement }); if (error) { // stripe error onError(error.message); } else { try { if (!paymentSchedule) { // process the normal payment pipeline, including SCA validation const res = await StripeAPI.confirmMethod(paymentMethod.id, cart); await handleServerConfirmation(res); } else { const paymentMethodId = paymentMethod.id; const subscription: StripeSubscription = await StripeAPI.createSubscription(paymentMethod.id, cart); if (subscription && subscription.status === 'active') { // Subscription is active, no customer actions required. const res = await StripeAPI.confirmPaymentSchedule(subscription.id, cart); onSuccess(res); } const paymentIntent = subscription.latest_invoice.payment_intent; if (paymentIntent.status === 'requires_action') { return stripe .confirmCardPayment(paymentIntent.client_secret, { payment_method: paymentMethodId }) .then(async (result) => { if (result.error) { throw result.error; } else { if (result.paymentIntent.status === 'succeeded') { const res = await StripeAPI.confirmPaymentSchedule(subscription.id, cart); onSuccess(res); } } }) .catch((error) => { onError(error.message); }); } else if (paymentIntent.status === 'requires_payment_method') { onError(t('app.shared.messages.payment_card_declined')); } } } catch (err) { // catch api errors onError(err); } } }; /** * Process the server response about the Strong-customer authentication (SCA) * @param response can be a PaymentConfirmation, or an Invoice (if the payment succeeded) * @see app/controllers/api/stripe_controller.rb#confirm_payment */ const handleServerConfirmation = async (response: PaymentConfirmation|Invoice) => { if ('error' in response) { if (response.error.statusText) { onError(response.error.statusText); } else { onError(`${t('app.shared.messages.payment_card_error')} ${response.error}`); } } else if ('requires_action' in response) { // Use Stripe.js to handle required card action const result = await stripe.handleCardAction(response.payment_intent_client_secret); if (result.error) { onError(result.error.message); } else { // The card action has been handled // The PaymentIntent can be confirmed again on the server try { const confirmation = await StripeAPI.confirmIntent(result.paymentIntent.id, cart); await handleServerConfirmation(confirmation); } catch (e) { onError(e); } } } else if ('id' in response) { onSuccess(response); } else { console.error(`[StripeForm] unknown response received: ${response}`); } }; /** * Options for the Stripe's card input */ const cardOptions = { style: { base: { fontSize: '16px', color: '#424770', '::placeholder': { color: '#aab7c4' } }, invalid: { color: '#9e2146', iconColor: '#9e2146' } }, hidePostalCode: true }; return (
{children} ); };