2021-01-27 13:59:41 +01:00
|
|
|
/**
|
|
|
|
* 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';
|
2021-02-04 15:47:11 +01:00
|
|
|
import { FabButton } from './fab-button';
|
2021-01-27 13:59:41 +01:00
|
|
|
|
|
|
|
declare var Fablab: IFablab;
|
|
|
|
|
|
|
|
interface PaymentSchedulesTableProps {
|
|
|
|
paymentSchedules: Array<PaymentSchedule>,
|
|
|
|
showCustomer?: boolean
|
|
|
|
}
|
|
|
|
|
|
|
|
const PaymentSchedulesTableComponent: React.FC<PaymentSchedulesTableProps> = ({ paymentSchedules, showCustomer }) => {
|
|
|
|
const { t } = useTranslation('admin');
|
|
|
|
|
|
|
|
const [showExpanded, setShowExpanded] = useState({});
|
|
|
|
|
2021-01-27 17:24:10 +01:00
|
|
|
/**
|
|
|
|
* Check if the requested payment schedule is displayed with its deadlines (PaymentScheduleItem) or without them
|
|
|
|
*/
|
2021-01-27 13:59:41 +01:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2021-01-27 17:24:10 +01:00
|
|
|
/**
|
|
|
|
* Return the value for the CSS property 'display', for the payment schedule deadlines
|
|
|
|
*/
|
2021-01-27 13:59:41 +01:00
|
|
|
const statusDisplay = (paymentScheduleId: number): string => {
|
|
|
|
if (isExpanded(paymentScheduleId)) {
|
|
|
|
return 'table-row'
|
|
|
|
} else {
|
|
|
|
return 'none';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-27 17:24:10 +01:00
|
|
|
/**
|
|
|
|
* Return the action icon for showing/hiding the deadlines
|
|
|
|
*/
|
2021-01-27 13:59:41 +01:00
|
|
|
const expandCollapseIcon = (paymentScheduleId: number): JSX.Element => {
|
|
|
|
if (isExpanded(paymentScheduleId)) {
|
|
|
|
return <i className="fas fa-minus-square" />;
|
|
|
|
} else {
|
|
|
|
return <i className="fas fa-plus-square" />
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-27 17:24:10 +01:00
|
|
|
/**
|
|
|
|
* Show or hide the deadlines for the provided payment schedule, inverting their current status
|
|
|
|
*/
|
2021-01-27 13:59:41 +01:00
|
|
|
const togglePaymentScheduleDetails = (paymentScheduleId: number): ReactEventHandler => {
|
|
|
|
return (): void => {
|
|
|
|
if (isExpanded(paymentScheduleId)) {
|
|
|
|
setShowExpanded(Object.assign({}, showExpanded, { [paymentScheduleId]: false }));
|
|
|
|
} else {
|
|
|
|
setShowExpanded(Object.assign({}, showExpanded, { [paymentScheduleId]: true }));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-27 17:24:10 +01:00
|
|
|
/**
|
|
|
|
* For use with downloadButton()
|
|
|
|
*/
|
2021-01-27 13:59:41 +01:00
|
|
|
enum TargetType {
|
|
|
|
Invoice = 'invoices',
|
|
|
|
PaymentSchedule = 'payment_schedules'
|
|
|
|
}
|
2021-01-27 17:24:10 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a button to download a PDF file, may be an invoice, or a payment schedule, depending or the provided parameters
|
|
|
|
*/
|
2021-01-27 13:59:41 +01:00
|
|
|
const downloadButton = (target: TargetType, id: number): JSX.Element => {
|
|
|
|
const link = `api/${target}/${id}/download`;
|
|
|
|
return (
|
|
|
|
<a href={link} target="_blank" className="download-button">
|
|
|
|
<i className="fas fa-download" />
|
|
|
|
{t('app.admin.invoices.schedules_table.download')}
|
|
|
|
</a>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-01-27 17:24:10 +01:00
|
|
|
/**
|
|
|
|
* Return the human-readable string for the status of the provided deadline.
|
|
|
|
*/
|
2021-01-27 13:59:41 +01:00
|
|
|
const formatState = (item: PaymentScheduleItem): JSX.Element => {
|
|
|
|
let res = t(`app.admin.invoices.schedules_table.state_${item.state}`);
|
|
|
|
if (item.state === PaymentScheduleItemState.Paid) {
|
2021-01-27 17:24:10 +01:00
|
|
|
const key = `app.admin.invoices.schedules_table.method_${item.payment_method}`
|
|
|
|
res += ` (${t(key)})`;
|
2021-01-27 13:59:41 +01:00
|
|
|
}
|
|
|
|
return <span className={`state-${item.state}`}>{res}</span>;
|
|
|
|
}
|
|
|
|
|
2021-01-27 17:24:10 +01:00
|
|
|
/**
|
|
|
|
* Return the action button(s) for the given deadline
|
|
|
|
*/
|
2021-01-27 13:59:41 +01:00
|
|
|
const itemButtons = (item: PaymentScheduleItem): JSX.Element => {
|
|
|
|
switch (item.state) {
|
|
|
|
case PaymentScheduleItemState.Paid:
|
|
|
|
return downloadButton(TargetType.Invoice, item.invoice_id);
|
|
|
|
case PaymentScheduleItemState.Pending:
|
2021-01-27 17:24:10 +01:00
|
|
|
return (
|
2021-02-04 15:47:11 +01:00
|
|
|
<FabButton onClick={handleConfirmCheckPayment(item)}
|
|
|
|
icon={<i className="fas fa-money-check" />}>
|
2021-01-27 17:24:10 +01:00
|
|
|
{t('app.admin.invoices.schedules_table.confirm_payment')}
|
2021-02-04 15:47:11 +01:00
|
|
|
</FabButton>
|
2021-01-27 17:24:10 +01:00
|
|
|
);
|
|
|
|
case PaymentScheduleItemState.RequireAction:
|
|
|
|
return (
|
2021-02-04 15:47:11 +01:00
|
|
|
<FabButton onClick={handleSolveAction(item)}
|
|
|
|
icon={<i className="fas fa-wrench" />}>
|
2021-01-27 17:24:10 +01:00
|
|
|
{t('app.admin.invoices.schedules_table.solve')}
|
2021-02-04 15:47:11 +01:00
|
|
|
</FabButton>
|
2021-01-27 17:24:10 +01:00
|
|
|
);
|
|
|
|
case PaymentScheduleItemState.RequirePaymentMethod:
|
|
|
|
return (
|
2021-02-04 15:47:11 +01:00
|
|
|
<FabButton onClick={handleUpdateCard(item)}
|
|
|
|
icon={<i className="fas fa-credit-card" />}>
|
2021-01-27 17:24:10 +01:00
|
|
|
{t('app.admin.invoices.schedules_table.update_card')}
|
2021-02-04 15:47:11 +01:00
|
|
|
</FabButton>
|
2021-01-27 17:24:10 +01:00
|
|
|
);
|
2021-01-27 13:59:41 +01:00
|
|
|
default:
|
|
|
|
return <span />
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-27 17:24:10 +01:00
|
|
|
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 <StripeElements>
|
|
|
|
- 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
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-27 13:59:41 +01:00
|
|
|
return (
|
2021-01-27 17:24:10 +01:00
|
|
|
<div>
|
|
|
|
<table className="schedules-table">
|
|
|
|
<thead>
|
|
|
|
<tr>
|
|
|
|
<th className="w-35" />
|
|
|
|
<th className="w-200">{t('app.admin.invoices.schedules_table.schedule_num')}</th>
|
|
|
|
<th className="w-200">{t('app.admin.invoices.schedules_table.date')}</th>
|
|
|
|
<th className="w-120">{t('app.admin.invoices.schedules_table.price')}</th>
|
|
|
|
{showCustomer && <th className="w-200">{t('app.admin.invoices.schedules_table.customer')}</th>}
|
|
|
|
<th className="w-200"/>
|
|
|
|
</tr>
|
|
|
|
</thead>
|
|
|
|
<tbody>
|
|
|
|
{paymentSchedules.map(p => <tr key={p.id}>
|
|
|
|
<td colSpan={6}>
|
|
|
|
<table className="schedules-table-body">
|
|
|
|
<tbody>
|
|
|
|
<tr>
|
|
|
|
<td className="w-35 row-header" onClick={togglePaymentScheduleDetails(p.id)}>{expandCollapseIcon(p.id)}</td>
|
|
|
|
<td className="w-200">{p.reference}</td>
|
|
|
|
<td className="w-200">{formatDate(p.created_at)}</td>
|
|
|
|
<td className="w-120">{formatPrice(p.total)}</td>
|
|
|
|
{showCustomer && <td className="w-200">{p.user.name}</td>}
|
|
|
|
<td className="w-200">{downloadButton(TargetType.PaymentSchedule, p.id)}</td>
|
|
|
|
</tr>
|
|
|
|
<tr style={{ display: statusDisplay(p.id) }}>
|
|
|
|
<td className="w-35" />
|
|
|
|
<td colSpan={5}>
|
|
|
|
<div>
|
|
|
|
<table className="schedule-items-table">
|
|
|
|
<thead>
|
|
|
|
<tr>
|
|
|
|
<th className="w-120">{t('app.admin.invoices.schedules_table.deadline')}</th>
|
|
|
|
<th className="w-120">{t('app.admin.invoices.schedules_table.amount')}</th>
|
|
|
|
<th className="w-200">{t('app.admin.invoices.schedules_table.state')}</th>
|
|
|
|
<th className="w-200" />
|
|
|
|
</tr>
|
|
|
|
</thead>
|
|
|
|
<tbody>
|
|
|
|
{_.orderBy(p.items, 'due_date').map(item => <tr key={item.id}>
|
|
|
|
<td>{formatDate(item.due_date)}</td>
|
|
|
|
<td>{formatPrice(item.amount)}</td>
|
|
|
|
<td>{formatState(item)}</td>
|
|
|
|
<td>{itemButtons(item)}</td>
|
|
|
|
</tr>)}
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
</div>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
</td>
|
|
|
|
</tr>)}
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
</div>
|
2021-01-27 13:59:41 +01:00
|
|
|
);
|
|
|
|
};
|
|
|
|
PaymentSchedulesTableComponent.defaultProps = { showCustomer: false };
|
|
|
|
|
|
|
|
|
|
|
|
export const PaymentSchedulesTable: React.FC<PaymentSchedulesTableProps> = ({ paymentSchedules, showCustomer }) => {
|
|
|
|
return (
|
|
|
|
<Loader>
|
|
|
|
<PaymentSchedulesTableComponent paymentSchedules={paymentSchedules} showCustomer={showCustomer} />
|
|
|
|
</Loader>
|
|
|
|
);
|
|
|
|
}
|