1
0
mirror of https://github.com/LaCasemate/fab-manager.git synced 2025-01-10 00:46:15 +01:00
fab-manager/app/frontend/src/javascript/components/payment/stripe/stripe-form.tsx

148 lines
4.8 KiB
TypeScript
Raw Normal View History

2020-11-25 17:13:45 +01:00
import React, { FormEvent } from 'react';
import { CardElement, useElements, useStripe } from '@stripe/react-stripe-js';
import { SetupIntent } from "@stripe/stripe-js";
import { useTranslation } from 'react-i18next';
import { CartItems, PaymentConfirmation } from '../../../models/payment';
import { User } from '../../../models/user';
import StripeAPI from '../../../api/stripe';
2020-11-25 17:13:45 +01:00
interface StripeFormProps {
onSubmit: () => void,
onSuccess: (result: SetupIntent|PaymentConfirmation|any) => void,
2020-11-25 17:13:45 +01:00
onError: (message: string) => void,
customer: User,
operator: User,
2020-11-30 16:52:55 +01:00
className?: string,
paymentSchedule?: boolean,
2021-04-08 15:21:24 +02:00
cartItems?: CartItems,
formId: string,
2020-11-25 17:13:45 +01:00
}
2020-11-30 16:52:55 +01:00
/**
* A form component to collect the credit card details and to create the payment method on Stripe.
2021-04-08 15:21:24 +02:00
* The form validation button must be created elsewhere, using the attribute form={formId}.
2020-11-30 16:52:55 +01:00
*/
2021-04-08 15:21:24 +02:00
export const StripeForm: React.FC<StripeFormProps> = ({ onSubmit, onSuccess, onError, children, className, paymentSchedule = false, cartItems, customer, operator, formId }) => {
const { t } = useTranslation('shared');
2020-11-25 17:13:45 +01:00
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<void> => {
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) {
2020-12-09 11:35:49 +01:00
// stripe error
2020-11-25 17:13:45 +01:00
onError(error.message);
} else {
2020-12-09 11:35:49 +01:00
try {
if (!paymentSchedule) {
// process the normal payment pipeline, including SCA validation
const res = await StripeAPI.confirm(paymentMethod.id, cartItems);
2020-12-09 11:35:49 +01:00
await handleServerConfirmation(res);
} else {
2020-12-09 11:35:49 +01:00
// we start by associating the payment method with the user
const { client_secret } = await StripeAPI.setupIntent(customer.id);
2020-12-09 11:35:49 +01:00
const { setupIntent, error } = await stripe.confirmCardSetup(client_secret, {
payment_method: paymentMethod.id,
mandate_data: {
customer_acceptance: {
type: 'online',
online: {
ip_address: operator.ip_address,
user_agent: navigator.userAgent
}
}
}
2020-12-09 11:35:49 +01:00
})
if (error) {
onError(error.message);
} else {
// then we confirm the payment schedule
const res = await StripeAPI.confirmPaymentSchedule(setupIntent.id, cartItems);
2020-12-09 11:35:49 +01:00
onSuccess(res);
}
}
2020-12-09 11:35:49 +01:00
} catch (err) {
// catch api errors
onError(err);
}
}
}
/**
* Process the server response about the Strong-customer authentication (SCA)
2020-12-07 13:49:11 +01:00
* @param response can be a PaymentConfirmation, or a Reservation (if the reservation succeeded), or a Subscription (if the subscription succeeded)
* @see app/controllers/api/payments_controller.rb#on_reservation_success
* @see app/controllers/api/payments_controller.rb#on_subscription_success
* @see app/controllers/api/payments_controller.rb#generate_payment_response
*/
2020-12-07 13:49:11 +01:00
const handleServerConfirmation = async (response: PaymentConfirmation|any) => {
if (response.error) {
if (response.error.statusText) {
onError(response.error.statusText);
} else {
onError(`${t('app.shared.messages.payment_card_error')} ${response.error}`);
}
} else if (response.requires_action) {
// 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.confirm(result.paymentIntent.id, cartItems);
2020-12-07 13:49:11 +01:00
await handleServerConfirmation(confirmation);
} catch (e) {
onError(e);
}
}
2020-11-25 17:13:45 +01:00
} else {
2020-12-07 13:49:11 +01:00
onSuccess(response);
2020-11-25 17:13:45 +01:00
}
}
2020-11-25 17:13:45 +01:00
/**
* 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 (
<form onSubmit={handleSubmit} id={formId} className={className ? className : ''}>
2020-11-25 17:13:45 +01:00
<CardElement options={cardOptions} />
{children}
</form>
);
}