import { PaymentMethod, PaymentSchedule, PaymentScheduleItem, PaymentScheduleItemState } from '../../models/payment-schedule'; import React, { ReactElement, useState } from 'react'; import { FabButton } from '../base/fab-button'; import { useTranslation } from 'react-i18next'; import { User } from '../../models/user'; import PaymentScheduleAPI from '../../api/payment-schedule'; import { FabModal } from '../base/fab-modal'; import FormatLib from '../../lib/format'; import { StripeConfirmModal } from '../payment/stripe/stripe-confirm-modal'; import { UpdateCardModal } from '../payment/update-card-modal'; import { UpdatePaymentMeanModal } from './update-payment-mean-modal'; // we want to display some buttons only once. This is the types of buttons it applies to. export enum TypeOnce { CardUpdate = 'card-update', SubscriptionCancel = 'subscription-cancel', UpdatePaymentMean = 'update-payment-mean' } interface PaymentScheduleItemActionsProps { paymentScheduleItem: PaymentScheduleItem, paymentSchedule: PaymentSchedule, onError: (message: string) => void, onSuccess: () => void, onCardUpdateSuccess: () => void operator: User, displayOnceMap: Map>, show: boolean, } /** * This component is responsible for rendering the actions buttons for a payment schedule item. */ export const PaymentScheduleItemActions: React.FC = ({ paymentScheduleItem, paymentSchedule, onError, onSuccess, onCardUpdateSuccess, displayOnceMap, operator, show }) => { const { t } = useTranslation('shared'); // is open, the modal dialog to cancel the associated subscription? const [showCancelSubscription, setShowCancelSubscription] = useState(false); // is open, the modal dialog to confirm the cashing of a check? const [showConfirmCashing, setShowConfirmCashing] = useState(false); // is open, the modal dialog to confirm a back transfer? const [showConfirmTransfer, setShowConfirmTransfer] = useState(false); // is open, the modal dialog the resolve a pending card payment? const [showResolveAction, setShowResolveAction] = useState(false); // is open, the modal dialog to update the card details const [showUpdateCard, setShowUpdateCard] = useState(false); // is open, the modal dialog to update the payment mean const [showUpdatePaymentMean, setShowUpdatePaymentMean] = useState(false); // the user cannot confirm the action modal (3D secure), unless he has resolved the pending action const [isConfirmActionDisabled, setConfirmActionDisabled] = useState(true); /** * Check if the current operator has administrative rights or is a normal member */ const isPrivileged = (): boolean => { return (operator.role === 'admin' || operator.role === 'manager'); }; /** * Return a button to download a PDF invoice file */ const downloadInvoiceButton = (id: number): JSX.Element => { const link = `api/invoices/${id}/download`; return ( {t('app.shared.payment_schedule_item_actions.download')} ); }; /** * Return a button to cancel the given subscription, if the user is privileged enough */ const cancelSubscriptionButton = (): ReactElement => { const displayOnceStatus = displayOnceMap.get(TypeOnce.SubscriptionCancel).get(paymentSchedule.id); if (isPrivileged() && (!displayOnceStatus || displayOnceStatus === paymentScheduleItem.id)) { displayOnceMap.get(TypeOnce.SubscriptionCancel).set(paymentSchedule.id, paymentScheduleItem.id); return ( }> {t('app.shared.payment_schedule_item_actions.cancel_subscription')} ); } }; /** * Return a button to confirm the receipt of the bank transfer, if the user is privileged enough */ const confirmTransferButton = (): ReactElement => { if (isPrivileged()) { return ( }> {t('app.shared.payment_schedule_item_actions.confirm_payment')} ); } }; /** * Return a button to confirm the cashing of the check, if the user is privileged enough */ const confirmCheckButton = (): ReactElement => { if (isPrivileged()) { return ( }> {t('app.shared.payment_schedule_item_actions.confirm_check')} ); } }; /** * Return a button to resolve the 3DS security check */ const solveActionButton = (): ReactElement => { return ( }> {t('app.shared.payment_schedule_item_actions.resolve_action')} ); }; /** * Return a button to update the default payment mean for the current payment schedule */ const updatePaymentMeanButton = (): ReactElement => { const displayOnceStatus = displayOnceMap.get(TypeOnce.UpdatePaymentMean).get(paymentSchedule.id); if (isPrivileged() && (!displayOnceStatus || displayOnceStatus === paymentScheduleItem.id)) { displayOnceMap.get(TypeOnce.UpdatePaymentMean).set(paymentSchedule.id, paymentScheduleItem.id); return ( }> {t('app.shared.payment_schedule_item_actions.update_payment_mean')} ); } }; /** * Return a button to update the credit card associated with the payment schedule */ const updateCardButton = (): ReactElement => { const displayOnceStatus = displayOnceMap.get(TypeOnce.SubscriptionCancel).get(paymentSchedule.id); if (isPrivileged() && (!displayOnceStatus || displayOnceStatus === paymentScheduleItem.id)) { displayOnceMap.get(TypeOnce.CardUpdate).set(paymentSchedule.id, paymentScheduleItem.id); return ( }> {t('app.shared.payment_schedule_item_actions.update_card')} ); } }; /** * Return the actions button(s) for current paymentScheduleItem with state Pending */ const pendingActions = (): ReactElement => { if (isPrivileged()) { if (paymentSchedule.payment_method === PaymentMethod.Transfer) { return confirmTransferButton(); } if (paymentSchedule.payment_method === PaymentMethod.Check) { return confirmCheckButton(); } } else { return {t('app.shared.payment_schedule_item_actions.please_ask_reception')}; } }; /** * Return the actions button(s) for current paymentScheduleItem with state Error or GatewayCanceled */ const errorActions = (): ReactElement[] => { // if the payment schedule is canceled/in error, the schedule is over, and we can't update the card displayOnceMap.get(TypeOnce.CardUpdate).set(paymentSchedule.id, paymentScheduleItem.id); const buttons = []; if (isPrivileged()) { buttons.push(cancelSubscriptionButton()); buttons.push(updatePaymentMeanButton()); } else { buttons.push({t('app.shared.payment_schedule_item_actions.please_ask_reception')}); } return buttons; }; /** * Return the actions button(s) for current paymentScheduleItem with state New */ const newActions = (): Array => { const buttons = []; if (paymentSchedule.payment_method === PaymentMethod.Card) { buttons.push(updateCardButton()); } if (isPrivileged()) { buttons.push(cancelSubscriptionButton()); } return buttons; }; /** * Show/hide the modal dialog to cancel the current subscription */ const toggleCancelSubscriptionModal = (): void => { setShowCancelSubscription(!showCancelSubscription); }; /** * Show/hide the modal dialog that enable to confirm the cashing of the check for a given deadline. */ const toggleConfirmCashingModal = (): void => { setShowConfirmCashing(!showConfirmCashing); }; /** * Show/hide the modal dialog that enable to confirm the bank transfer for a given deadline. */ const toggleConfirmTransferModal = (): void => { setShowConfirmTransfer(!showConfirmTransfer); }; /** * Show/hide the modal dialog that trigger the card "action". */ const toggleResolveActionModal = (): void => { setShowResolveAction(!showResolveAction); }; /** * Enable/disable the confirm button of the "action" modal */ const toggleConfirmActionButton = (): void => { setConfirmActionDisabled(!isConfirmActionDisabled); }; /** * Show/hide the modal dialog to update the bank card details */ const toggleUpdateCardModal = (): void => { setShowUpdateCard(!showUpdateCard); }; /** * Show/hide the modal dialog to update the payment mean */ const toggleUpdatePaymentMeanModal = (): void => { setShowUpdatePaymentMean(!showUpdatePaymentMean); }; /** * After the user has confirmed that he wants to cash the check, update the API, refresh the list and close the modal. */ const onCheckCashingConfirmed = (): void => { PaymentScheduleAPI.cashCheck(paymentScheduleItem.id).then((res) => { if (res.state === PaymentScheduleItemState.Paid) { onSuccess(); toggleConfirmCashingModal(); } }); }; /** * After the user has confirmed that he validates the transfer, update the API, refresh the list and close the modal. */ const onTransferConfirmed = (): void => { PaymentScheduleAPI.confirmTransfer(paymentScheduleItem.id).then((res) => { if (res.state === PaymentScheduleItemState.Paid) { onSuccess(); toggleConfirmTransferModal(); } }); }; /** * When the card was successfully updated, pay the invoice (using the new payment method) and close the modal */ const handleCardUpdateSuccess = (): void => { if (paymentScheduleItem.state === PaymentScheduleItemState.RequirePaymentMethod) { PaymentScheduleAPI.payItem(paymentScheduleItem.id).then(() => { onSuccess(); onCardUpdateSuccess(); toggleUpdateCardModal(); }).catch((err) => { onError(err); }); } else { // the user is updating his card number in a pro-active way, we don't need to trigger the payment onCardUpdateSuccess(); toggleUpdateCardModal(); } }; /** * When the user has confirmed the cancellation, we transfer the request to the API */ const onCancelSubscriptionConfirmed = (): void => { PaymentScheduleAPI.cancel(paymentSchedule.id).then(() => { onSuccess(); toggleCancelSubscriptionModal(); }); }; /** * After the 3DS confirmation was done (successfully or not), ask the API to refresh the item status, * then refresh the list and close the modal */ const afterConfirmAction = (): void => { toggleConfirmActionButton(); PaymentScheduleAPI.refreshItem(paymentScheduleItem.id).then(() => { onSuccess(); toggleResolveActionModal(); }); }; /** * When the update of the payment mean was successful, refresh the list and close the modal */ const onPaymentMeanUpdateSuccess = (): void => { onSuccess(); toggleUpdatePaymentMeanModal(); }; if (!show) return null; return ( {paymentScheduleItem.state === PaymentScheduleItemState.Paid && downloadInvoiceButton(paymentScheduleItem.invoice_id)} {paymentScheduleItem.state === PaymentScheduleItemState.Pending && pendingActions()} {paymentScheduleItem.state === PaymentScheduleItemState.RequireAction && solveActionButton()} {paymentScheduleItem.state === PaymentScheduleItemState.RequirePaymentMethod && updateCardButton()} {paymentScheduleItem.state === PaymentScheduleItemState.Error && errorActions()} {paymentScheduleItem.state === PaymentScheduleItemState.GatewayCanceled && errorActions()} {paymentScheduleItem.state === PaymentScheduleItemState.New && newActions()}
{/* Confirm the cashing of the current deadline by check */} {t('app.shared.payment_schedule_item_actions.confirm_check_cashing_body', { AMOUNT: FormatLib.price(paymentScheduleItem.amount), DATE: FormatLib.date(paymentScheduleItem.due_date) })} {/* Confirm the bank transfer for the current deadline */} {t('app.shared.payment_schedule_item_actions.confirm_bank_transfer_body', { AMOUNT: FormatLib.price(paymentScheduleItem.amount), DATE: FormatLib.date(paymentScheduleItem.due_date) })} {/* Cancel the subscription */} {t('app.shared.payment_schedule_item_actions.confirm_cancel_subscription')} {/* 3D secure confirmation */} {/* Update credit card */} {/* Update the payment mean */}
); }; PaymentScheduleItemActions.defaultProps = { show: false };