2021-06-29 11:14:36 +02:00
|
|
|
import React, { useEffect, useState } from 'react';
|
2021-06-29 15:59:57 +02:00
|
|
|
import _ from 'lodash';
|
2021-06-29 11:14:36 +02:00
|
|
|
import { Machine } from '../../models/machine';
|
|
|
|
import { User } from '../../models/user';
|
|
|
|
import { UserPack } from '../../models/user-pack';
|
|
|
|
import UserPackAPI from '../../api/user-pack';
|
|
|
|
import SettingAPI from '../../api/setting';
|
|
|
|
import { SettingName } from '../../models/setting';
|
|
|
|
import { FabButton } from '../base/fab-button';
|
|
|
|
import { useTranslation } from 'react-i18next';
|
|
|
|
import { ProposePacksModal } from './propose-packs-modal';
|
|
|
|
import { Loader } from '../base/loader';
|
|
|
|
import { react2angular } from 'react2angular';
|
|
|
|
import { IApplication } from '../../models/application';
|
2021-06-30 16:58:16 +02:00
|
|
|
import { PrepaidPack } from '../../models/prepaid-pack';
|
|
|
|
import PrepaidPackAPI from '../../api/prepaid-pack';
|
2021-06-29 11:14:36 +02:00
|
|
|
|
2021-07-01 12:34:10 +02:00
|
|
|
declare const Application: IApplication;
|
2021-06-29 11:14:36 +02:00
|
|
|
|
|
|
|
type PackableItem = Machine;
|
|
|
|
|
|
|
|
interface PacksSummaryProps {
|
|
|
|
item: PackableItem,
|
|
|
|
itemType: 'Machine',
|
2021-06-29 15:59:57 +02:00
|
|
|
customer?: User,
|
2021-06-29 11:14:36 +02:00
|
|
|
operator: User,
|
|
|
|
onError: (message: string) => void,
|
|
|
|
onSuccess: (message: string) => void,
|
2021-06-30 10:53:05 +02:00
|
|
|
refresh?: Promise<void>
|
2021-06-29 11:14:36 +02:00
|
|
|
}
|
|
|
|
|
2021-06-30 10:53:05 +02:00
|
|
|
const PacksSummaryComponent: React.FC<PacksSummaryProps> = ({ item, itemType, customer, operator, onError, onSuccess, refresh }) => {
|
2021-06-29 11:14:36 +02:00
|
|
|
const { t } = useTranslation('logged');
|
|
|
|
|
2021-06-30 16:58:16 +02:00
|
|
|
const [packs, setPacks] = useState<Array<PrepaidPack>>(null);
|
2021-06-29 11:14:36 +02:00
|
|
|
const [userPacks, setUserPacks] = useState<Array<UserPack>>(null);
|
|
|
|
const [threshold, setThreshold] = useState<number>(null);
|
|
|
|
const [packsModal, setPacksModal] = useState<boolean>(false);
|
2021-09-20 19:43:05 +02:00
|
|
|
const [isPackOnlyForSubscription, setIsPackOnlyForSubscription] = useState<boolean>(true);
|
2021-06-29 11:14:36 +02:00
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
SettingAPI.get(SettingName.RenewPackThreshold)
|
|
|
|
.then(data => setThreshold(parseFloat(data.value)))
|
|
|
|
.catch(error => onError(error));
|
2021-09-20 19:43:05 +02:00
|
|
|
SettingAPI.get(SettingName.PackOnlyForSubscription)
|
|
|
|
.then(data => setIsPackOnlyForSubscription(data.value === 'true'))
|
|
|
|
.catch(error => onError(error));
|
2021-06-29 15:59:57 +02:00
|
|
|
}, []);
|
|
|
|
|
|
|
|
useEffect(() => {
|
2021-06-30 10:53:05 +02:00
|
|
|
getUserPacksData();
|
2021-06-30 16:58:16 +02:00
|
|
|
}, [customer, item, itemType]);
|
2021-06-30 10:53:05 +02:00
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (refresh instanceof Promise) {
|
|
|
|
refresh.then(getUserPacksData);
|
|
|
|
}
|
|
|
|
}, [refresh]);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetch the user packs data from the API
|
|
|
|
*/
|
|
|
|
const getUserPacksData = (): void => {
|
2021-06-30 16:58:16 +02:00
|
|
|
if (_.isEmpty(customer)) return;
|
|
|
|
|
2021-06-29 15:59:57 +02:00
|
|
|
UserPackAPI.index({ user_id: customer.id, priceable_type: itemType, priceable_id: item.id })
|
|
|
|
.then(data => setUserPacks(data))
|
|
|
|
.catch(error => onError(error));
|
2021-06-30 16:58:16 +02:00
|
|
|
PrepaidPackAPI.index({ priceable_id: item.id, priceable_type: itemType, group_id: customer.group_id, disabled: false })
|
|
|
|
.then(data => setPacks(data))
|
|
|
|
.catch(error => onError(error));
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-06-29 11:14:36 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Total of minutes used by the customer
|
|
|
|
*/
|
|
|
|
const totalUsed = (): number => {
|
|
|
|
if (!userPacks) return 0;
|
|
|
|
|
2021-06-29 15:59:57 +02:00
|
|
|
return userPacks.map(up => up.minutes_used).reduce((acc, curr) => acc + curr, 0);
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-06-29 11:14:36 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Total of minutes available is the packs bought by the customer
|
|
|
|
*/
|
|
|
|
const totalAvailable = (): number => {
|
|
|
|
if (!userPacks) return 0;
|
|
|
|
|
2021-06-29 15:59:57 +02:00
|
|
|
return userPacks.map(up => up.prepaid_pack.minutes).reduce((acc, curr) => acc + curr, 0);
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-06-29 11:14:36 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Total prepaid hours remaining for the current customer
|
|
|
|
*/
|
|
|
|
const totalHours = (): number => {
|
2021-06-29 15:59:57 +02:00
|
|
|
return (totalAvailable() - totalUsed()) / 60;
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-06-29 11:14:36 +02:00
|
|
|
|
|
|
|
/**
|
2021-06-29 15:59:57 +02:00
|
|
|
* Do we need to display the "buy new pack" button?
|
2021-06-29 11:14:36 +02:00
|
|
|
*/
|
2021-06-29 15:59:57 +02:00
|
|
|
const shouldDisplayButton = (): boolean => {
|
2021-07-01 12:34:10 +02:00
|
|
|
if (!packs?.length) return false;
|
2021-06-30 16:58:16 +02:00
|
|
|
|
2021-06-29 11:14:36 +02:00
|
|
|
if (threshold < 1) {
|
2021-06-29 15:59:57 +02:00
|
|
|
return totalAvailable() - totalUsed() <= totalAvailable() * threshold;
|
2021-06-29 11:14:36 +02:00
|
|
|
}
|
|
|
|
|
2021-06-29 15:59:57 +02:00
|
|
|
return totalAvailable() - totalUsed() <= threshold * 60;
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-06-29 11:14:36 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Open/closes the prepaid-pack buying modal
|
|
|
|
*/
|
|
|
|
const togglePacksModal = (): void => {
|
|
|
|
setPacksModal(!packsModal);
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-06-29 11:14:36 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback triggered when the customer has successfully bought a prepaid-pack
|
|
|
|
*/
|
|
|
|
const handlePackBoughtSuccess = (message: string): void => {
|
|
|
|
onSuccess(message);
|
|
|
|
togglePacksModal();
|
|
|
|
UserPackAPI.index({ user_id: customer.id, priceable_type: itemType, priceable_id: item.id })
|
|
|
|
.then(data => setUserPacks(data))
|
|
|
|
.catch(error => onError(error));
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-06-29 11:14:36 +02:00
|
|
|
|
2021-06-29 15:59:57 +02:00
|
|
|
// prevent component rendering if no customer selected
|
|
|
|
if (_.isEmpty(customer)) return <div />;
|
2021-06-30 16:58:16 +02:00
|
|
|
// prevent component rendering if ths customer have no packs and there are no packs available
|
|
|
|
if (totalHours() === 0 && packs?.length === 0) return <div/>;
|
2021-09-20 19:43:05 +02:00
|
|
|
// render remaining hours and a warning if customer has not any subscription if admin active pack only for subscription option
|
|
|
|
if (totalHours() > 0) {
|
|
|
|
return (
|
|
|
|
<div className="packs-summary">
|
|
|
|
<h3>{t('app.logged.packs_summary.prepaid_hours')}</h3>
|
|
|
|
<div className="content">
|
|
|
|
<span className="remaining-hours">
|
|
|
|
{t('app.logged.packs_summary.remaining_HOURS', { HOURS: totalHours(), ITEM: itemType })}
|
|
|
|
{isPackOnlyForSubscription && !customer.subscribed_plan &&
|
|
|
|
<div className="alert alert-warning m-t m-b">
|
|
|
|
{t('app.logged.packs_summary.unable_to_use_pack_for_subsription_is_expired')}
|
|
|
|
</div>
|
|
|
|
}
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
// prevent component rendering buy pack button if customer has not any subscription if admin active pack only for subscription option
|
|
|
|
if (isPackOnlyForSubscription && !customer.subscribed_plan) return <div/>;
|
2021-06-29 15:59:57 +02:00
|
|
|
|
2021-06-29 11:14:36 +02:00
|
|
|
return (
|
|
|
|
<div className="packs-summary">
|
2021-06-29 15:59:57 +02:00
|
|
|
<h3>{t('app.logged.packs_summary.prepaid_hours')}</h3>
|
|
|
|
<div className="content">
|
2021-06-30 10:53:05 +02:00
|
|
|
<span className="remaining-hours">
|
|
|
|
{totalHours() === 0 && t('app.logged.packs_summary.no_hours', { ITEM: itemType })}
|
|
|
|
</span>
|
2021-06-29 15:59:57 +02:00
|
|
|
{shouldDisplayButton() && <div className="button-wrapper">
|
|
|
|
<FabButton className="buy-button" onClick={togglePacksModal} icon={<i className="fa fa-shopping-cart"/>}>
|
|
|
|
{t('app.logged.packs_summary.buy_a_new_pack')}
|
|
|
|
</FabButton>
|
|
|
|
<ProposePacksModal isOpen={packsModal}
|
2021-07-01 12:34:10 +02:00
|
|
|
toggleModal={togglePacksModal}
|
|
|
|
item={item}
|
|
|
|
itemType={itemType}
|
|
|
|
customer={customer}
|
|
|
|
operator={operator}
|
|
|
|
onError={onError}
|
|
|
|
onDecline={togglePacksModal}
|
|
|
|
onSuccess={handlePackBoughtSuccess} />
|
2021-06-29 15:59:57 +02:00
|
|
|
</div>}
|
|
|
|
</div>
|
2021-06-29 11:14:36 +02:00
|
|
|
</div>
|
|
|
|
);
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-06-29 11:14:36 +02:00
|
|
|
|
2021-06-30 10:53:05 +02:00
|
|
|
export const PacksSummary: React.FC<PacksSummaryProps> = ({ item, itemType, customer, operator, onError, onSuccess, refresh }) => {
|
2021-06-29 11:14:36 +02:00
|
|
|
return (
|
|
|
|
<Loader>
|
2021-06-30 10:53:05 +02:00
|
|
|
<PacksSummaryComponent item={item} itemType={itemType} customer={customer} operator={operator} onError={onError} onSuccess={onSuccess} refresh={refresh} />
|
2021-06-29 11:14:36 +02:00
|
|
|
</Loader>
|
|
|
|
);
|
2021-07-01 12:34:10 +02:00
|
|
|
};
|
2021-06-29 11:14:36 +02:00
|
|
|
|
2021-06-30 10:53:05 +02:00
|
|
|
Application.Components.component('packsSummary', react2angular(PacksSummary, ['item', 'itemType', 'customer', 'operator', 'onError', 'onSuccess', 'refresh']));
|