2021-03-03 15:28:56 +01:00
|
|
|
/**
|
2021-03-10 17:15:36 +01:00
|
|
|
* This component allows an administrator to select and configure a payment gateway.
|
|
|
|
* The configuration of a payment gateway is required to enable the online payments.
|
2021-03-03 15:28:56 +01:00
|
|
|
*/
|
|
|
|
|
2021-03-30 15:56:36 +02:00
|
|
|
import React, { BaseSyntheticEvent, useEffect, useState } from 'react';
|
2021-03-03 15:28:56 +01:00
|
|
|
import { react2angular } from 'react2angular';
|
|
|
|
import { useTranslation } from 'react-i18next';
|
2021-04-09 08:47:34 +02:00
|
|
|
import { StripeKeysForm } from './payment/stripe/stripe-keys-form';
|
2022-06-01 15:24:25 +02:00
|
|
|
import { PayzenKeysForm } from './payment/payzen/payzen-keys-form';
|
2021-04-08 09:35:09 +02:00
|
|
|
import { FabModal, ModalSize } from './base/fab-modal';
|
2021-04-09 08:47:34 +02:00
|
|
|
import { Loader } from './base/loader';
|
2021-03-03 15:28:56 +01:00
|
|
|
import { User } from '../models/user';
|
2021-03-24 17:31:50 +01:00
|
|
|
import { Gateway } from '../models/gateway';
|
2021-03-30 15:56:36 +02:00
|
|
|
import { SettingBulkResult, SettingName } from '../models/setting';
|
2021-04-09 08:47:34 +02:00
|
|
|
import { IApplication } from '../models/application';
|
2021-03-30 15:56:36 +02:00
|
|
|
import SettingAPI from '../api/setting';
|
2021-03-03 15:28:56 +01:00
|
|
|
|
2021-07-01 12:34:10 +02:00
|
|
|
declare const Application: IApplication;
|
2021-03-03 15:28:56 +01:00
|
|
|
|
|
|
|
interface SelectGatewayModalModalProps {
|
|
|
|
isOpen: boolean,
|
|
|
|
toggleModal: () => void,
|
|
|
|
currentUser: User,
|
2021-06-10 10:39:42 +02:00
|
|
|
onError: (errors: string) => void,
|
2021-03-30 15:56:36 +02:00
|
|
|
onSuccess: (results: Map<SettingName, SettingBulkResult>) => void,
|
2021-03-03 15:28:56 +01:00
|
|
|
}
|
|
|
|
|
2022-06-20 18:06:15 +02:00
|
|
|
/**
|
|
|
|
* Modal dialog that enable an admin to configure the active payment gateway
|
|
|
|
*/
|
2022-06-01 15:24:25 +02:00
|
|
|
export const SelectGatewayModal: React.FC<SelectGatewayModalModalProps> = ({ isOpen, toggleModal, onError, onSuccess }) => {
|
2021-03-03 15:28:56 +01:00
|
|
|
const { t } = useTranslation('admin');
|
|
|
|
|
2021-03-10 16:46:18 +01:00
|
|
|
const [preventConfirmGateway, setPreventConfirmGateway] = useState<boolean>(true);
|
|
|
|
const [selectedGateway, setSelectedGateway] = useState<string>('');
|
2021-03-30 11:26:47 +02:00
|
|
|
const [gatewayConfig, setGatewayConfig] = useState<Map<SettingName, string>>(new Map());
|
2021-03-10 16:46:18 +01:00
|
|
|
|
2021-06-17 17:08:22 +02:00
|
|
|
// request the configured gateway to the API
|
2021-03-30 15:56:36 +02:00
|
|
|
useEffect(() => {
|
2021-06-17 17:08:22 +02:00
|
|
|
SettingAPI.get(SettingName.PaymentGateway).then(gateway => {
|
2021-07-01 12:34:10 +02:00
|
|
|
setSelectedGateway(gateway.value ? gateway.value : '');
|
|
|
|
});
|
2021-03-30 15:56:36 +02:00
|
|
|
}, []);
|
2021-03-10 16:46:18 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback triggered when the user has filled and confirmed the settings of his gateway
|
|
|
|
*/
|
|
|
|
const onGatewayConfirmed = () => {
|
|
|
|
setPreventConfirmGateway(true);
|
2021-03-30 15:56:36 +02:00
|
|
|
updateSettings();
|
|
|
|
setPreventConfirmGateway(false);
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-03-10 16:46:18 +01:00
|
|
|
|
2021-03-24 17:31:50 +01:00
|
|
|
/**
|
|
|
|
* Save the gateway provided by the target input into the component state
|
|
|
|
*/
|
2021-03-10 16:46:18 +01:00
|
|
|
const setGateway = (event: BaseSyntheticEvent) => {
|
|
|
|
const gateway = event.target.value;
|
|
|
|
setSelectedGateway(gateway);
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-03-03 15:28:56 +01:00
|
|
|
|
2021-03-24 17:31:50 +01:00
|
|
|
/**
|
|
|
|
* Check if any payment gateway was selected
|
|
|
|
*/
|
|
|
|
const hasSelectedGateway = (): boolean => {
|
|
|
|
return selectedGateway !== '';
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-03-24 17:31:50 +01:00
|
|
|
|
2021-03-30 11:26:47 +02:00
|
|
|
/**
|
|
|
|
* Callback triggered when the embedded form has validated all the stripe keys
|
|
|
|
*/
|
|
|
|
const handleValidStripeKeys = (publicKey: string, secretKey: string): void => {
|
|
|
|
setGatewayConfig((prev) => {
|
|
|
|
const newMap = new Map(prev);
|
|
|
|
newMap.set(SettingName.StripeSecretKey, secretKey);
|
|
|
|
newMap.set(SettingName.StripePublicKey, publicKey);
|
|
|
|
return newMap;
|
|
|
|
});
|
|
|
|
setPreventConfirmGateway(false);
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-03-30 11:26:47 +02:00
|
|
|
|
2021-03-31 17:58:09 +02:00
|
|
|
/**
|
2021-05-18 16:42:30 +02:00
|
|
|
* Callback triggered when the embedded form has validated all the PayZen keys
|
2021-03-31 17:58:09 +02:00
|
|
|
*/
|
|
|
|
const handleValidPayZenKeys = (payZenKeys: Map<SettingName, string>): void => {
|
|
|
|
setGatewayConfig(payZenKeys);
|
|
|
|
setPreventConfirmGateway(false);
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-03-31 17:58:09 +02:00
|
|
|
|
2021-05-18 16:42:30 +02:00
|
|
|
/**
|
|
|
|
* Callback triggered when the embedded form has not validated all keys
|
|
|
|
*/
|
|
|
|
const handleInvalidKeys = (): void => {
|
|
|
|
setPreventConfirmGateway(true);
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-05-18 16:42:30 +02:00
|
|
|
|
2021-03-30 15:56:36 +02:00
|
|
|
/**
|
|
|
|
* Send the new gateway settings to the API to save them
|
|
|
|
*/
|
|
|
|
const updateSettings = (): void => {
|
|
|
|
const settings = new Map<SettingName, string>(gatewayConfig);
|
|
|
|
settings.set(SettingName.PaymentGateway, selectedGateway);
|
|
|
|
|
2021-06-17 17:08:22 +02:00
|
|
|
SettingAPI.bulkUpdate(settings, true).then(result => {
|
2021-06-10 10:39:42 +02:00
|
|
|
const errorResults = Array.from(result.values()).filter(item => !item.status);
|
|
|
|
if (errorResults.length > 0) {
|
|
|
|
onError(errorResults.map(item => item.error[0]).join(' '));
|
2021-03-30 15:56:36 +02:00
|
|
|
} else {
|
2021-06-10 10:39:42 +02:00
|
|
|
// we call the success callback only in case of full success (transactional bulk update)
|
2021-03-30 15:56:36 +02:00
|
|
|
onSuccess(result);
|
|
|
|
}
|
|
|
|
}, reason => {
|
|
|
|
onError(reason);
|
|
|
|
});
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-03-30 15:56:36 +02:00
|
|
|
|
2021-03-03 15:28:56 +01:00
|
|
|
return (
|
2022-06-20 18:06:15 +02:00
|
|
|
<FabModal title={t('app.admin.invoices.payment.select_gateway_modal.select_gateway_title')}
|
2021-07-01 12:34:10 +02:00
|
|
|
isOpen={isOpen}
|
|
|
|
toggleModal={toggleModal}
|
|
|
|
width={ModalSize.medium}
|
2022-06-01 15:24:25 +02:00
|
|
|
className="select-gateway-modal"
|
2022-06-20 18:06:15 +02:00
|
|
|
confirmButton={t('app.admin.invoices.payment.select_gateway_modal.confirm_button')}
|
2021-07-01 12:34:10 +02:00
|
|
|
onConfirm={onGatewayConfirmed}
|
|
|
|
preventConfirm={preventConfirmGateway}>
|
2021-03-24 17:31:50 +01:00
|
|
|
{!hasSelectedGateway() && <p className="info-gateway">
|
2022-06-20 18:06:15 +02:00
|
|
|
{t('app.admin.invoices.payment.select_gateway_modal.gateway_info')}
|
2021-03-24 17:31:50 +01:00
|
|
|
</p>}
|
2022-06-20 18:06:15 +02:00
|
|
|
<label htmlFor="gateway">{t('app.admin.invoices.payment.select_gateway_modal.select_gateway')}</label>
|
2021-03-10 16:46:18 +01:00
|
|
|
<select id="gateway" className="select-gateway" onChange={setGateway} value={selectedGateway}>
|
|
|
|
<option />
|
2022-06-20 18:06:15 +02:00
|
|
|
<option value={Gateway.Stripe}>{t('app.admin.invoices.payment.select_gateway_modal.stripe')}</option>
|
|
|
|
<option value={Gateway.PayZen}>{t('app.admin.invoices.payment.select_gateway_modal.payzen')}</option>
|
2021-03-10 16:46:18 +01:00
|
|
|
</select>
|
2021-05-18 16:42:30 +02:00
|
|
|
{selectedGateway === Gateway.Stripe && <StripeKeysForm onValidKeys={handleValidStripeKeys} onInvalidKeys={handleInvalidKeys} />}
|
2022-06-01 15:24:25 +02:00
|
|
|
{selectedGateway === Gateway.PayZen && <PayzenKeysForm onValidKeys={handleValidPayZenKeys} onInvalidKeys={handleInvalidKeys} />}
|
2021-03-03 15:28:56 +01:00
|
|
|
</FabModal>
|
|
|
|
);
|
2021-03-24 17:31:50 +01:00
|
|
|
};
|
2021-03-03 15:28:56 +01:00
|
|
|
|
2021-03-30 15:56:36 +02:00
|
|
|
const SelectGatewayModalWrapper: React.FC<SelectGatewayModalModalProps> = ({ isOpen, toggleModal, currentUser, onSuccess, onError }) => {
|
2021-03-03 15:28:56 +01:00
|
|
|
return (
|
|
|
|
<Loader>
|
2021-03-30 15:56:36 +02:00
|
|
|
<SelectGatewayModal isOpen={isOpen} toggleModal={toggleModal} currentUser={currentUser} onSuccess={onSuccess} onError={onError} />
|
2021-03-03 15:28:56 +01:00
|
|
|
</Loader>
|
|
|
|
);
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-03-03 15:28:56 +01:00
|
|
|
|
2021-03-30 15:56:36 +02:00
|
|
|
Application.Components.component('selectGatewayModal', react2angular(SelectGatewayModalWrapper, ['isOpen', 'toggleModal', 'currentUser', 'onSuccess', 'onError']));
|