org.fenixedu.treasury.domain.document.DebitNote.java Source code

Java tutorial

Introduction

Here is the source code for org.fenixedu.treasury.domain.document.DebitNote.java

Source

/**
 * This file was created by Quorum Born IT <http://www.qub-it.com/> and its 
 * copyright terms are bind to the legal agreement regulating the FenixEdu@ULisboa 
 * software development project between Quorum Born IT and Servios Partilhados da
 * Universidade de Lisboa:
 *  - Copyright  2015 Quorum Born IT (until any Go-Live phase)
 *  - Copyright  2015 Universidade de Lisboa (after any Go-Live phase)
 *
 * Contributors: ricardo.pedro@qub-it.com, anil.mamede@qub-it.com
 * 
 *
 * 
 * This file is part of FenixEdu Treasury.
 *
 * FenixEdu Treasury 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 Treasury 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 Treasury.  If not, see <http://www.gnu.org/licenses/>.
 */
package org.fenixedu.treasury.domain.document;

import java.math.BigDecimal;
import java.util.List;
import java.util.Set;
import java.util.stream.Collectors;
import java.util.stream.Stream;

import org.fenixedu.treasury.domain.debt.DebtAccount;
import org.fenixedu.treasury.domain.exceptions.TreasuryDomainException;
import org.joda.time.DateTime;
import org.joda.time.LocalDate;

import pt.ist.fenixframework.Atomic;

public class DebitNote extends DebitNote_Base {

    protected DebitNote(final DebtAccount debtAccount, final DocumentNumberSeries documentNumberSeries,
            final DateTime documentDate) {
        super();
        this.init(debtAccount, documentNumberSeries, documentDate);
    }

    protected DebitNote(final DebtAccount debtAccount, final DebtAccount payorDebtAccount,
            final DocumentNumberSeries documentNumberSeries, final DateTime documentDate) {
        super();

        this.init(debtAccount, payorDebtAccount, documentNumberSeries, documentDate);
    }

    @Override
    public boolean isDebitNote() {
        return true;
    }

    @Override
    protected void checkRules() {
        if (!getDocumentNumberSeries().getFinantialDocumentType().getType()
                .equals(FinantialDocumentTypeEnum.DEBIT_NOTE)) {
            throw new TreasuryDomainException("error.DebitNote.finantialDocumentType.invalid");
        }

        if (getPayorDebtAccount() != null && !getPayorDebtAccount().getFinantialInstitution()
                .equals(getDebtAccount().getFinantialInstitution())) {
            throw new TreasuryDomainException("error.DebitNote.finantialinstitution.mismatch");
        }

        super.checkRules();
    }

    @Override
    @Atomic
    public void delete(boolean deleteEntries) {
        super.delete(deleteEntries);
    }

    @Override
    public BigDecimal getOpenAmount() {
        if (this.isAnnulled()) {
            return BigDecimal.ZERO;
        }
        BigDecimal amount = BigDecimal.ZERO;
        for (DebitEntry entry : getDebitEntriesSet()) {
            amount = amount.add(entry.getOpenAmount());
        }
        return getDebtAccount().getFinantialInstitution().getCurrency().getValueWithScale(amount);
    }

    @Override
    public BigDecimal getOpenAmountWithInterests() {
        if (this.getState().isPreparing() || this.getState().isClosed()) {
            return getDebtAccount().getFinantialInstitution().getCurrency()
                    .getValueWithScale(getOpenAmount().add(getPendingInterestAmount()));
        } else {
            return BigDecimal.ZERO;
        }
    }

    public Stream<? extends DebitEntry> getDebitEntries() {
        return DebitEntry.find(this);
    }

    public Set<? extends DebitEntry> getDebitEntriesSet() {
        return this.getDebitEntries().collect(Collectors.<DebitEntry>toSet());
    }

    public BigDecimal getDebitAmount() {
        return this.getTotalAmount();
    }

    public BigDecimal getCreditAmount() {
        return BigDecimal.ZERO;
    }

    public static Stream<DebitNote> findAll() {
        return FinantialDocument.findAll().filter(x -> x instanceof DebitNote).map(DebitNote.class::cast);
    }

    @Atomic
    public void edit(final DebtAccount payorDebtAccount, final LocalDate documentDate, LocalDate documentDueDate,
            final String originDocumentNumber) {

        setPayorDebtAccount(payorDebtAccount);
        setDocumentDate(documentDate.toDateTimeAtStartOfDay());
        setDocumentDueDate(documentDueDate);
        setOriginDocumentNumber(originDocumentNumber);

        checkRules();
    }

    @Atomic
    public static DebitNote create(final DebtAccount debtAccount, final DocumentNumberSeries documentNumberSeries,
            final DateTime documentDate) {
        DebitNote note = new DebitNote(debtAccount, documentNumberSeries, documentDate);
        note.setFinantialDocumentType(FinantialDocumentType.findForDebitNote());
        note.setOriginDocumentNumber("");
        note.setDocumentDueDate(documentDate.toLocalDate());
        note.checkRules();
        return note;
    }

    @Atomic
    public static DebitNote create(final DebtAccount debtAccount, final DebtAccount payorDebtAccount,
            final DocumentNumberSeries documentNumberSeries, final DateTime documentDate,
            final LocalDate documentDueDate, final String originaNumber) {

        DebitNote note = new DebitNote(debtAccount, payorDebtAccount, documentNumberSeries, documentDate);
        note.setFinantialDocumentType(FinantialDocumentType.findForDebitNote());
        note.setOriginDocumentNumber(originaNumber);
        note.setDocumentDueDate(documentDueDate);
        note.checkRules();

        return note;
    }

    @Atomic
    public void addDebitNoteEntries(List<DebitEntry> debitEntries) {
        debitEntries.forEach(x -> this.addFinantialDocumentEntries(x));
        checkRules();
    }

    @Override
    public Set<FinantialDocument> findRelatedDocuments(Set<FinantialDocument> documentsBaseList,
            Boolean includeAnulledDocuments) {
        documentsBaseList.add(this);

        for (DebitEntry entry : getDebitEntriesSet()) {
            for (CreditEntry creditEntry : entry.getCreditEntriesSet()) {
                if (creditEntry.getFinantialDocument() != null
                        && !creditEntry.getFinantialDocument().isPreparing()) {
                    if (includeAnulledDocuments == true || this.isAnnulled() == false) {
                        if (documentsBaseList.contains(creditEntry.getFinantialDocument()) == false) {
                            documentsBaseList.addAll(creditEntry.getFinantialDocument()
                                    .findRelatedDocuments(documentsBaseList, includeAnulledDocuments));
                        }
                    }
                }
            }
        }

        for (DebitEntry entry : getDebitEntriesSet()) {
            for (SettlementEntry settlementEntry : entry.getSettlementEntriesSet()) {
                if (settlementEntry.getFinantialDocument() != null
                        && !settlementEntry.getFinantialDocument().isPreparing()) {
                    if (includeAnulledDocuments == true
                            || settlementEntry.getFinantialDocument().isAnnulled() == false) {
                        if (documentsBaseList.contains(settlementEntry.getFinantialDocument()) == false) {
                            documentsBaseList.addAll(settlementEntry.getFinantialDocument()
                                    .findRelatedDocuments(documentsBaseList, includeAnulledDocuments));
                        }
                    }
                }
            }

        }

        return documentsBaseList;
    }

    public BigDecimal getPendingInterestAmount() {
        return getPendingInterestAmount(new LocalDate());
    }

    public BigDecimal getPendingInterestAmount(LocalDate whenToCalculate) {
        BigDecimal interest = BigDecimal.ZERO;
        for (DebitEntry entry : this.getDebitEntriesSet()) {
            interest = interest.add(entry.calculateUndebitedInterestValue(whenToCalculate).getInterestAmount());
        }
        return interest;
    }

    @Override
    public void anullDocument(boolean freeEntries, String reason) {
        if (getDebitEntries().anyMatch(de -> !de.getSettlementEntriesSet().isEmpty())) {
            throw new TreasuryDomainException("error.DebitNote.cannot.delete.has.settlemententries");
        }
        if (this.getDocumentNumberSeries().getSeries().getCertificated()) {
            this.anullDebitNoteWithCreditNote(reason);
        } else {
            super.anullDocument(freeEntries, reason);
        }
    }

    @Atomic
    public void anullDebitNoteWithCreditNote(String reason) {
        if (this.getFinantialDocumentEntriesSet().size() > 0) {
            if (this.isClosed() == true) {

                DateTime now = new DateTime();

                //1. criar nota de acerto
                //2. percorrer os itens de divida, criar correspondente item de acerto com o valor "aberto"
                //2.1 verificar se existiram "juros" gerados correspondentes
                //3. fechar nota de acerto
                //4. criar settlement note
                //5. adicionar itens de divida com cada valor open amount
                //5.1 adicionar itens de dvida com cada valor open amount dos juros
                //6. adicionar itens de acerto por cada valor open amount de item de divida
                //7. fechar settlement note

                // No final podem sobrar itens de acerto com valor pendente de utilizacao, que representam os valores ja pagos nos itens de dividas correspondentes
                DocumentNumberSeries documentNumberSeriesCreditNote = DocumentNumberSeries.find(
                        FinantialDocumentType.findForCreditNote(), this.getDocumentNumberSeries().getSeries());
                DocumentNumberSeries documentNumberSeriesSettlementNote = DocumentNumberSeries.find(
                        FinantialDocumentType.findForSettlementNote(), this.getDocumentNumberSeries().getSeries());

                CreditNote creditNote = this.createEquivalentCreditNote(documentNumberSeriesCreditNote, now, reason,
                        true);
                creditNote.closeDocument();

                SettlementNote settlementNote = SettlementNote.create(this.getDebtAccount(),
                        documentNumberSeriesSettlementNote, now, "");
                settlementNote.setDocumentObservations(reason);

                for (CreditEntry creditEntry : creditNote.getCreditEntriesSet()) {
                    SettlementEntry crediEntry = SettlementEntry.create(creditEntry, settlementNote,
                            creditEntry.getDebitEntry().getOpenAmount(),
                            reason + "-" + creditEntry.getDescription(), now, false);
                    SettlementEntry debitEntry = SettlementEntry.create(creditEntry.getDebitEntry(), settlementNote,
                            creditEntry.getDebitEntry().getOpenAmount(),
                            reason + "-" + creditEntry.getDebitEntry().getDescription(), now, false);
                }

                settlementNote.closeDocument();

            } else {
                throw new TreasuryDomainException("error.DebitNote.cannot.anull.is.not.closed");
            }
        } else {
            throw new TreasuryDomainException("error.DebitNote.cannot.anull.is.empty");
        }

    }

    @Atomic
    public CreditNote createEquivalentCreditNote(DocumentNumberSeries documentNumberSeries, DateTime documentDate,
            String documentObservations, boolean createForInterestRateEntries) {

        CreditNote creditNote = CreditNote.create(this.getDebtAccount(), documentNumberSeries, documentDate, this,
                this.getUiDocumentNumber());
        creditNote.setDocumentObservations(documentObservations);
        for (DebitEntry entry : this.getDebitEntriesSet()) {
            CreditEntry creditEntry = CreditEntry.create(creditNote, entry.getDescription(), entry.getProduct(),
                    entry.getVat(), entry.getAmount(), documentDate, entry, entry.getQuantity());
            creditNote.addFinantialDocumentEntries(creditEntry);

            //Also generate for InterestRateDebitEntry
            if (createForInterestRateEntries == true) {
                if (entry.getInterestDebitEntriesSet().isEmpty() == false) {
                    for (DebitEntry interestEntry : entry.getInterestDebitEntriesSet()) {
                        CreditEntry interestCreditEntry = CreditEntry.create(creditNote,
                                interestEntry.getDescription(), interestEntry.getProduct(), interestEntry.getVat(),
                                interestEntry.getAmount(), documentDate, interestEntry,
                                interestEntry.getQuantity());
                        creditNote.addFinantialDocumentEntries(interestCreditEntry);

                    }
                }
            }
        }

        return creditNote;
    }
}