2021-06-22 11:13:44 +02:00
|
|
|
import React, { useEffect, useState } from 'react';
|
|
|
|
import { useTranslation } from 'react-i18next';
|
|
|
|
import { react2angular } from 'react2angular';
|
|
|
|
import { Loader } from '../base/loader';
|
|
|
|
import { FabAlert } from '../base/fab-alert';
|
|
|
|
import { HtmlTranslate } from '../base/html-translate';
|
|
|
|
import MachineAPI from '../../api/machine';
|
|
|
|
import GroupAPI from '../../api/group';
|
|
|
|
import { Machine } from '../../models/machine';
|
|
|
|
import { Group } from '../../models/group';
|
|
|
|
import { IApplication } from '../../models/application';
|
|
|
|
import { EditablePrice } from './editable-price';
|
|
|
|
import { ConfigurePacksButton } from './configure-packs-button';
|
|
|
|
import PriceAPI from '../../api/price';
|
|
|
|
import { Price } from '../../models/price';
|
|
|
|
import PrepaidPackAPI from '../../api/prepaid-pack';
|
|
|
|
import { PrepaidPack } from '../../models/prepaid-pack';
|
2021-06-22 17:56:13 +02:00
|
|
|
import { useImmer } from 'use-immer';
|
2021-10-11 18:50:53 +02:00
|
|
|
import FormatLib from '../../lib/format';
|
2021-06-22 11:13:44 +02:00
|
|
|
|
2021-07-01 12:34:10 +02:00
|
|
|
declare const Application: IApplication;
|
2021-06-22 11:13:44 +02:00
|
|
|
|
|
|
|
interface MachinesPricingProps {
|
|
|
|
onError: (message: string) => void,
|
|
|
|
onSuccess: (message: string) => void,
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Interface to set and edit the prices of machines-hours, per group
|
|
|
|
*/
|
|
|
|
const MachinesPricing: React.FC<MachinesPricingProps> = ({ onError, onSuccess }) => {
|
|
|
|
const { t } = useTranslation('admin');
|
|
|
|
|
|
|
|
const [machines, setMachines] = useState<Array<Machine>>(null);
|
|
|
|
const [groups, setGroups] = useState<Array<Group>>(null);
|
2021-06-22 17:56:13 +02:00
|
|
|
const [prices, updatePrices] = useImmer<Array<Price>>(null);
|
2021-06-22 11:13:44 +02:00
|
|
|
const [packs, setPacks] = useState<Array<PrepaidPack>>(null);
|
|
|
|
|
|
|
|
// retrieve the initial data
|
|
|
|
useEffect(() => {
|
2021-06-23 17:00:15 +02:00
|
|
|
MachineAPI.index({ disabled: false })
|
2021-06-22 11:13:44 +02:00
|
|
|
.then(data => setMachines(data))
|
|
|
|
.catch(error => onError(error));
|
2021-07-01 12:34:10 +02:00
|
|
|
GroupAPI.index({ disabled: false, admins: false })
|
2021-06-22 11:13:44 +02:00
|
|
|
.then(data => setGroups(data))
|
|
|
|
.catch(error => onError(error));
|
2021-06-23 17:00:15 +02:00
|
|
|
PriceAPI.index({ priceable_type: 'Machine', plan_id: null })
|
2021-06-22 17:56:13 +02:00
|
|
|
.then(data => updatePrices(data))
|
2021-06-22 11:13:44 +02:00
|
|
|
.catch(error => onError(error));
|
|
|
|
PrepaidPackAPI.index()
|
|
|
|
.then(data => setPacks(data))
|
2021-07-01 12:34:10 +02:00
|
|
|
.catch(error => onError(error));
|
2021-06-22 11:13:44 +02:00
|
|
|
}, []);
|
|
|
|
|
|
|
|
// duration of the example slot
|
|
|
|
const EXEMPLE_DURATION = 20;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the exemple price, formatted
|
|
|
|
*/
|
|
|
|
const examplePrice = (type: 'hourly_rate' | 'final_price'): string => {
|
|
|
|
const hourlyRate = 10;
|
|
|
|
|
|
|
|
if (type === 'hourly_rate') {
|
2021-10-11 18:50:53 +02:00
|
|
|
return FormatLib.price(hourlyRate);
|
2021-06-22 11:13:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const price = (hourlyRate / 60) * EXEMPLE_DURATION;
|
2021-10-11 18:50:53 +02:00
|
|
|
return FormatLib.price(price);
|
2021-06-22 11:13:44 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find the price matching the given criterion
|
|
|
|
*/
|
|
|
|
const findPriceBy = (machineId, groupId): Price => {
|
2021-06-23 17:00:15 +02:00
|
|
|
return prices.find(price => price.priceable_id === machineId && price.group_id === groupId);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Filter the packs matching the given criterion
|
|
|
|
*/
|
|
|
|
const filterPacksBy = (machineId, groupId): Array<PrepaidPack> => {
|
|
|
|
return packs.filter(pack => pack.priceable_id === machineId && pack.group_id === groupId);
|
2021-06-22 11:13:44 +02:00
|
|
|
};
|
|
|
|
|
2021-06-22 17:56:13 +02:00
|
|
|
/**
|
|
|
|
* Update the given price in the internal state
|
|
|
|
*/
|
|
|
|
const updatePrice = (price: Price): void => {
|
|
|
|
updatePrices(draft => {
|
|
|
|
const index = draft.findIndex(p => p.id === price.id);
|
|
|
|
draft[index] = price;
|
|
|
|
return draft;
|
|
|
|
});
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-06-22 17:56:13 +02:00
|
|
|
|
2021-06-22 11:13:44 +02:00
|
|
|
/**
|
|
|
|
* Callback triggered when the user has confirmed to update a price
|
|
|
|
*/
|
|
|
|
const handleUpdatePrice = (price: Price): void => {
|
|
|
|
PriceAPI.update(price)
|
2021-06-22 17:56:13 +02:00
|
|
|
.then(() => {
|
|
|
|
onSuccess(t('app.admin.machines_pricing.price_updated'));
|
|
|
|
updatePrice(price);
|
|
|
|
})
|
2021-07-01 12:34:10 +02:00
|
|
|
.catch(error => onError(error));
|
|
|
|
};
|
2021-06-22 11:13:44 +02:00
|
|
|
|
|
|
|
return (
|
2021-06-22 17:56:13 +02:00
|
|
|
<div className="machines-pricing">
|
2021-06-22 11:13:44 +02:00
|
|
|
<FabAlert level="warning">
|
|
|
|
<p><HtmlTranslate trKey="app.admin.machines_pricing.prices_match_machine_hours_rates_html"/></p>
|
2021-10-11 18:50:53 +02:00
|
|
|
<p><HtmlTranslate trKey="app.admin.machines_pricing.prices_calculated_on_hourly_rate_html" options={{ DURATION: `${EXEMPLE_DURATION}`, RATE: examplePrice('hourly_rate'), PRICE: examplePrice('final_price') }} /></p>
|
2021-06-22 11:13:44 +02:00
|
|
|
<p>{t('app.admin.machines_pricing.you_can_override')}</p>
|
|
|
|
</FabAlert>
|
|
|
|
<table>
|
|
|
|
<thead>
|
|
|
|
<tr>
|
|
|
|
<th>{t('app.admin.machines_pricing.machines')}</th>
|
|
|
|
{groups?.map(group => <th key={group.id} className="group-name">{group.name}</th>)}
|
|
|
|
</tr>
|
|
|
|
</thead>
|
|
|
|
<tbody>
|
2021-07-01 12:34:10 +02:00
|
|
|
{machines?.map(machine => <tr key={machine.id}>
|
|
|
|
<td>{machine.name}</td>
|
|
|
|
{groups?.map(group => <td key={group.id}>
|
|
|
|
{prices && <EditablePrice price={findPriceBy(machine.id, group.id)} onSave={handleUpdatePrice} />}
|
|
|
|
{packs && <ConfigurePacksButton packsData={filterPacksBy(machine.id, group.id)}
|
|
|
|
onError={onError}
|
|
|
|
onSuccess={onSuccess}
|
|
|
|
groupId={group.id}
|
|
|
|
priceableId={machine.id}
|
|
|
|
priceableType="Machine" />}
|
|
|
|
</td>)}
|
|
|
|
</tr>)}
|
2021-06-22 11:13:44 +02:00
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
</div>
|
|
|
|
);
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-06-22 11:13:44 +02:00
|
|
|
|
|
|
|
const MachinesPricingWrapper: React.FC<MachinesPricingProps> = ({ onError, onSuccess }) => {
|
|
|
|
return (
|
|
|
|
<Loader>
|
|
|
|
<MachinesPricing onError={onError} onSuccess={onSuccess} />
|
|
|
|
</Loader>
|
|
|
|
);
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-06-22 11:13:44 +02:00
|
|
|
|
|
|
|
Application.Components.component('machinesPricing', react2angular(MachinesPricingWrapper, ['onError', 'onSuccess']));
|