/** * This component shows a list of all payment schedules with their associated deadlines (aka. PaymentScheduleItem) and invoices */ import React, { ReactEventHandler, ReactNode, useState } from 'react'; import { useTranslation } from 'react-i18next'; import { Loader } from './loader'; import moment from 'moment'; import { IFablab } from '../models/fablab'; import _ from 'lodash'; import { PaymentSchedule, PaymentScheduleItem, PaymentScheduleItemState } from '../models/payment-schedule'; import { FabButton } from './fab-button'; import { FabModal } from './fab-modal'; import PaymentScheduleAPI from '../api/payment-schedule'; declare var Fablab: IFablab; interface PaymentSchedulesTableProps { paymentSchedules: Array, showCustomer?: boolean, refreshList: () => void } const PaymentSchedulesTableComponent: React.FC = ({ paymentSchedules, showCustomer, refreshList }) => { const { t } = useTranslation('admin'); const [showExpanded, setShowExpanded] = useState>(new Map()); const [showConfirmCashing, setShowConfirmCashing] = useState(false); const [tempDeadline, setTempDeadline] = useState(null); /** * Check if the requested payment schedule is displayed with its deadlines (PaymentScheduleItem) or without them */ const isExpanded = (paymentScheduleId: number): boolean => { return showExpanded.get(paymentScheduleId); } /** * Return the formatted localized date for the given date */ const formatDate = (date: Date): string => { return Intl.DateTimeFormat().format(moment(date).toDate()); } /** * Return the formatted localized amount for the given price (eg. 20.5 => "20,50 €") */ const formatPrice = (price: number): string => { return new Intl.NumberFormat(Fablab.intl_locale, {style: 'currency', currency: Fablab.intl_currency}).format(price); } /** * Return the value for the CSS property 'display', for the payment schedule deadlines */ const statusDisplay = (paymentScheduleId: number): string => { if (isExpanded(paymentScheduleId)) { return 'table-row' } else { return 'none'; } } /** * Return the action icon for showing/hiding the deadlines */ const expandCollapseIcon = (paymentScheduleId: number): JSX.Element => { if (isExpanded(paymentScheduleId)) { return ; } else { return } } /** * Show or hide the deadlines for the provided payment schedule, inverting their current status */ const togglePaymentScheduleDetails = (paymentScheduleId: number): ReactEventHandler => { return (): void => { if (isExpanded(paymentScheduleId)) { setShowExpanded((prev) => new Map(prev).set(paymentScheduleId, false)); } else { setShowExpanded((prev) => new Map(prev).set(paymentScheduleId, true)); } } } /** * For use with downloadButton() */ enum TargetType { Invoice = 'invoices', PaymentSchedule = 'payment_schedules' } /** * Return a button to download a PDF file, may be an invoice, or a payment schedule, depending or the provided parameters */ const downloadButton = (target: TargetType, id: number): JSX.Element => { const link = `api/${target}/${id}/download`; return ( {t('app.admin.invoices.schedules_table.download')} ); } /** * Return the human-readable string for the status of the provided deadline. */ const formatState = (item: PaymentScheduleItem): JSX.Element => { let res = t(`app.admin.invoices.schedules_table.state_${item.state}`); if (item.state === PaymentScheduleItemState.Paid) { const key = `app.admin.invoices.schedules_table.method_${item.payment_method}` res += ` (${t(key)})`; } return {res}; } /** * Return the action button(s) for the given deadline */ const itemButtons = (item: PaymentScheduleItem): JSX.Element => { switch (item.state) { case PaymentScheduleItemState.Paid: return downloadButton(TargetType.Invoice, item.invoice_id); case PaymentScheduleItemState.Pending: return ( }> {t('app.admin.invoices.schedules_table.confirm_payment')} ); case PaymentScheduleItemState.RequireAction: return ( }> {t('app.admin.invoices.schedules_table.solve')} ); case PaymentScheduleItemState.RequirePaymentMethod: return ( }> {t('app.admin.invoices.schedules_table.update_card')} ); default: return } } const handleConfirmCheckPayment = (item: PaymentScheduleItem): ReactEventHandler => { return (): void => { setTempDeadline(item); toggleConfirmCashingModal(); } } const onCheckCashingConfirmed = (): void => { const api = new PaymentScheduleAPI(); api.cashCheck(tempDeadline.id).then(() => { refreshList(); toggleConfirmCashingModal(); }); } /** * Show/hide the modal dialog that enable to confirm the cashing of the check for a given deadline. */ const toggleConfirmCashingModal = (): void => { setShowConfirmCashing(!showConfirmCashing); } /** * Dynamically build the content of the modal depending on the currently selected deadline */ const cashingModalContent = (): ReactNode => { if (tempDeadline) { return ( {t('app.admin.invoices.schedules_table.confirm_check_cashing_body', { AMOUNT: formatPrice(tempDeadline.amount), DATE: formatDate(tempDeadline.due_date) })} ); } return ; } const handleSolveAction = (item: PaymentScheduleItem): ReactEventHandler => { return (): void => { /* TODO - create component wrapped with - stripe.confirmCardSetup(item.client_secret).then(function(result) { if (result.error) { // Display error.message in your UI. } else { // The setup has succeeded. Display a success message. } }); */ } } const handleUpdateCard = (item: PaymentScheduleItem): ReactEventHandler => { return (): void => { /* TODO - Notify the customer, collect new payment information, and create a new payment method - Attach the payment method to the customer - Update the default payment method - Pay the invoice using the new payment method */ } } return (
{showCustomer && } {paymentSchedules.map(p => )}
{t('app.admin.invoices.schedules_table.schedule_num')} {t('app.admin.invoices.schedules_table.date')} {t('app.admin.invoices.schedules_table.price')}{t('app.admin.invoices.schedules_table.customer')}
{showCustomer && }
{expandCollapseIcon(p.id)} {p.reference} {formatDate(p.created_at)} {formatPrice(p.total)}{p.user.name}{downloadButton(TargetType.PaymentSchedule, p.id)}
{_.orderBy(p.items, 'due_date').map(item => )}
{t('app.admin.invoices.schedules_table.deadline')} {t('app.admin.invoices.schedules_table.amount')} {t('app.admin.invoices.schedules_table.state')}
{formatDate(item.due_date)} {formatPrice(item.amount)} {formatState(item)} {itemButtons(item)}
{cashingModalContent()}
); }; PaymentSchedulesTableComponent.defaultProps = { showCustomer: false }; export const PaymentSchedulesTable: React.FC = ({ paymentSchedules, showCustomer, refreshList }) => { return ( ); }