/** * This component shows a list of all payment schedules with their associated deadlines (aka. PaymentScheduleItem) and invoices */ import React, { ReactEventHandler, 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'; declare var Fablab: IFablab; interface PaymentSchedulesTableProps { paymentSchedules: Array, showCustomer?: boolean } const PaymentSchedulesTableComponent: React.FC = ({ paymentSchedules, showCustomer }) => { const { t } = useTranslation('admin'); const [showExpanded, setShowExpanded] = useState({}); /** * Check if the requested payment schedule is displayed with its deadlines (PaymentScheduleItem) or without them */ const isExpanded = (paymentScheduleId: number): boolean => { return showExpanded[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(Object.assign({}, showExpanded, { [paymentScheduleId]: false })); } else { setShowExpanded(Object.assign({}, showExpanded, { [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 ( ); case PaymentScheduleItemState.RequireAction: return ( ); case PaymentScheduleItemState.RequirePaymentMethod: return ( ); default: return } } const handleConfirmCheckPayment = (item: PaymentScheduleItem): ReactEventHandler => { return (): void => { /* TODO - display confirmation modal - create /api/payment_schedule/item/confirm_check endpoint and post to it */ } } 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)}
); }; PaymentSchedulesTableComponent.defaultProps = { showCustomer: false }; export const PaymentSchedulesTable: React.FC = ({ paymentSchedules, showCustomer }) => { return ( ); }