/**
* Copyright © 2002 Instituto Superior Técnico
*
* This file is part of FenixEdu Academic.
*
* FenixEdu Academic is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* FenixEdu Academic is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with FenixEdu Academic. If not, see <http://www.gnu.org/licenses/>.
*/
package org.fenixedu.academic.domain.accounting.postingRules.gratuity;
import java.math.BigDecimal;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Set;
import org.fenixedu.academic.domain.ExecutionYear;
import org.fenixedu.academic.domain.accounting.Account;
import org.fenixedu.academic.domain.accounting.AccountingTransaction;
import org.fenixedu.academic.domain.accounting.EntryType;
import org.fenixedu.academic.domain.accounting.Event;
import org.fenixedu.academic.domain.accounting.EventType;
import org.fenixedu.academic.domain.accounting.ServiceAgreementTemplate;
import org.fenixedu.academic.domain.accounting.events.gratuity.GratuityEvent;
import org.fenixedu.academic.domain.accounting.events.gratuity.SpecializationDegreeGratuityEvent;
import org.fenixedu.academic.domain.exceptions.DomainException;
import org.fenixedu.academic.domain.exceptions.DomainExceptionWithLabelFormatter;
import org.fenixedu.academic.dto.accounting.AccountingTransactionDetailDTO;
import org.fenixedu.academic.dto.accounting.EntryDTO;
import org.fenixedu.academic.util.LabelFormatter;
import org.fenixedu.academic.util.Money;
import org.fenixedu.bennu.core.domain.User;
import org.joda.time.DateTime;
public abstract class SpecializationDegreeGratuityPR extends SpecializationDegreeGratuityPR_Base implements IGratuityPR {
protected SpecializationDegreeGratuityPR() {
super();
}
public SpecializationDegreeGratuityPR(DateTime startDate, DateTime endDate,
ServiceAgreementTemplate serviceAgreementTemplate, Money specializationDegreeTotalAmount,
BigDecimal specializationDegreePartialAcceptedPercentage) {
super();
init(EntryType.GRATUITY_FEE, EventType.GRATUITY, startDate, endDate, serviceAgreementTemplate,
specializationDegreeTotalAmount, specializationDegreePartialAcceptedPercentage);
}
protected void init(EntryType entryType, EventType eventType, DateTime startDate, DateTime endDate,
ServiceAgreementTemplate serviceAgreementTemplate, Money specializationDegreeTotalAmount,
BigDecimal specializationDegreePartialAcceptedPercentage) {
super.init(entryType, eventType, startDate, endDate, serviceAgreementTemplate);
checkParameters(specializationDegreeTotalAmount, specializationDegreePartialAcceptedPercentage);
super.setSpecializationDegreeTotalAmount(specializationDegreeTotalAmount);
super.setSpecializationDegreePartialAcceptedPercentage(specializationDegreePartialAcceptedPercentage);
}
private void checkParameters(Money specializationDegreeTotalAmount, BigDecimal specializationDegreePartialAcceptedPercentage) {
if (specializationDegreeTotalAmount == null) {
throw new DomainException(
"error.accounting.postingRules.gratuity.SpecializationDegreeGratuityPR.specializationDegreeTotalAmount.cannot.be.null");
}
if (specializationDegreePartialAcceptedPercentage == null) {
throw new DomainException(
"error.accounting.postingRules.gratuity.SpecializationDegreeGratuityPR.specializationDegreePartialAcceptedPercentage.cannot.be.null");
}
}
@Override
public void setSpecializationDegreeTotalAmount(Money specializationDegreeTotalAmount) {
throw new DomainException(
"error.accounting.postingRules.gratuity.SpecializationDegreeGratuityPR.cannot.modify.specializationDegreeTotalAmount");
}
@Override
protected Set<AccountingTransaction> internalProcess(User user, Collection<EntryDTO> entryDTOs, Event event,
Account fromAccount, Account toAccount, AccountingTransactionDetailDTO transactionDetail) {
if (entryDTOs.size() != 1) {
throw new DomainException(
"error.accounting.postingRules.gratuity.SpecializationDegreeGratuityPR.invalid.number.of.entryDTOs");
}
checkIfCanAddAmount(entryDTOs.iterator().next().getAmountToPay(), event, transactionDetail.getWhenRegistered());
return Collections.singleton(makeAccountingTransaction(user, event, fromAccount, toAccount, getEntryType(), entryDTOs
.iterator().next().getAmountToPay(), transactionDetail));
}
private void checkIfCanAddAmount(Money amountToAdd, Event event, DateTime when) {
if (((GratuityEvent) event).isCustomEnrolmentModel()) {
checkIfCanAddAmountForCustomEnrolmentModel(event, when, amountToAdd);
} else {
checkIfCanAddAmountForCompleteEnrolmentModel(amountToAdd, event, when);
}
}
private void checkIfCanAddAmountForCustomEnrolmentModel(Event event, DateTime when, Money amountToAdd) {
if (event.calculateAmountToPay(when).greaterThan(amountToAdd)) {
throw new DomainExceptionWithLabelFormatter(
"error.accounting.postingRules.gratuity.SpecializationDegreeGratuityPR.amount.being.payed.must.be.equal.to.amout.in.debt",
event.getDescriptionForEntryType(getEntryType()));
}
}
private void checkIfCanAddAmountForCompleteEnrolmentModel(final Money amountToAdd, final Event event, final DateTime when) {
if (hasAlreadyPayedAnyAmount(event, when)) {
final Money totalFinalAmount = event.getPayedAmount().add(amountToAdd);
if (!(totalFinalAmount.greaterOrEqualThan(calculateTotalAmountToPay(event, when)) || totalFinalAmount
.equals(getPartialPaymentAmount(event, when)))) {
throw new DomainExceptionWithLabelFormatter(
"error.accounting.postingRules.gratuity.SpecializationDegreeGratuityPR.amount.being.payed.must.be.equal.to.amout.in.debt",
event.getDescriptionForEntryType(getEntryType()));
}
} else {
if (!isPayingTotalAmount(event, when, amountToAdd) && !isPayingPartialAmount(event, when, amountToAdd)) {
final LabelFormatter percentageLabelFormatter = new LabelFormatter();
percentageLabelFormatter.appendLabel(getSpecializationDegreePartialAcceptedPercentage().multiply(
BigDecimal.valueOf(100)).toString());
throw new DomainExceptionWithLabelFormatter(
"error.accounting.postingRules.gratuity.SpecializationDegreeGratuityPR.invalid.partial.payment.value",
event.getDescriptionForEntryType(getEntryType()), percentageLabelFormatter);
}
}
}
private boolean isPayingTotalAmount(final Event event, final DateTime when, Money amountToAdd) {
return amountToAdd.greaterOrEqualThan(event.calculateAmountToPay(when));
}
private boolean isPayingPartialAmount(final Event event, final DateTime when, final Money amountToAdd) {
return amountToAdd.equals(getPartialPaymentAmount(event, when));
}
private boolean hasAlreadyPayedAnyAmount(final Event event, final DateTime when) {
return !calculateTotalAmountToPay(event, when).equals(event.calculateAmountToPay(when));
}
private Money getPartialPaymentAmount(final Event event, final DateTime when) {
return calculateTotalAmountToPay(event, when).multiply(getSpecializationDegreePartialAcceptedPercentage());
}
@Override
protected Money doCalculationForAmountToPay(Event event, DateTime when, boolean applyDiscount) {
final Money result;
if (((GratuityEvent) event).isCustomEnrolmentModel()) {
result = calculateSpecializationDegreeGratuityTotalAmountToPay(event);
} else {
result = getSpecializationDegreeTotalAmount();
}
return result;
}
@Override
protected Money subtractFromExemptions(Event event, DateTime when, boolean applyDiscount, Money amountToPay) {
final BigDecimal discountPercentage = applyDiscount ? getDiscountPercentage(event, amountToPay) : BigDecimal.ZERO;
return amountToPay.multiply(BigDecimal.ONE.subtract(discountPercentage));
}
abstract protected Money calculateSpecializationDegreeGratuityTotalAmountToPay(Event event);
private BigDecimal getDiscountPercentage(final Event event, final Money amount) {
return ((SpecializationDegreeGratuityEvent) event).calculateDiscountPercentage(amount);
}
@Override
public List<EntryDTO> calculateEntries(Event event, DateTime when) {
return Collections.singletonList(new EntryDTO(getEntryType(), event, calculateTotalAmountToPay(event, when), event
.getPayedAmount(), event.calculateAmountToPay(when), event.getDescriptionForEntryType(getEntryType()), event
.calculateAmountToPay(when)));
}
@Override
public Money getDefaultGratuityAmount(ExecutionYear executionYear) {
return getSpecializationDegreeTotalAmount();
}
}