/* * This file is part of LibrePlan * * Copyright (C) 2009-2010 Fundación para o Fomento da Calidade Industrial e * Desenvolvemento Tecnolóxico de Galicia * Copyright (C) 2010-2011 Igalia, S.L. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program 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 Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. */ package org.libreplan.business.reports.dtos; import static org.libreplan.business.reports.dtos.WorkingArrangementsPerOrderDTO.removeAfterDate; import java.math.BigDecimal; import java.math.RoundingMode; import java.util.List; import org.joda.time.LocalDate; import org.libreplan.business.advance.entities.AdvanceType; import org.libreplan.business.advance.entities.DirectAdvanceAssignment; import org.libreplan.business.common.Registry; import org.libreplan.business.orders.daos.IOrderDAO; import org.libreplan.business.orders.entities.Order; import org.libreplan.business.planner.entities.DayAssignment; import org.libreplan.business.planner.entities.DayAssignment.FilterType; import org.libreplan.business.planner.entities.Task; import org.libreplan.business.workingday.EffortDuration; import org.libreplan.business.workreports.daos.IWorkReportLineDAO; import org.libreplan.business.workreports.entities.WorkReportLine; /** * * @author Diego Pino Garcia <dpino@igalia.com> * */ public class SchedulingProgressPerOrderDTO { private IOrderDAO orderDAO; private IWorkReportLineDAO workReportLineDAO; private String orderName; private Integer estimatedHours; private Integer totalPlannedHours; private Integer partialPlannedHours; private EffortDuration realHours; private BigDecimal averageProgress; private Double imputedProgress; private Double plannedProgress; private BigDecimal costDifference; private BigDecimal planningDifference; private BigDecimal ratioCostDifference; private BigDecimal ratioPlanningDifference; private Boolean advanceTypeDoesNotApply = Boolean.FALSE; private Boolean appliedSpreadAdvanceType = Boolean.FALSE; private SchedulingProgressPerOrderDTO() { workReportLineDAO = Registry.getWorkReportLineDAO(); orderDAO = Registry.getOrderDAO(); } public SchedulingProgressPerOrderDTO(Order order, final List<Task> tasks, AdvanceType advanceType, LocalDate date) { this(); this.orderName = order.getName(); // Get average progress averageProgress = getFilterAdvanceTypePercentage(order, advanceType, date); if (averageProgress == null) { advanceTypeDoesNotApply = true; appliedSpreadAdvanceType = false; averageProgress = new BigDecimal(0); } // Fill DTO this.estimatedHours = getHoursSpecifiedAtOrder(tasks); this.totalPlannedHours = calculatePlannedHours(tasks, null); // Hours on time calculations this.partialPlannedHours = calculatePlannedHours(tasks, date); this.realHours = calculateRealHours(order, date); // Progress calculations this.imputedProgress = (totalPlannedHours != 0) ? new Double( realHours .toHoursAsDecimalWithScale(2).doubleValue() / totalPlannedHours.doubleValue()) : new Double(0); this.plannedProgress = (totalPlannedHours != 0) ? new Double( partialPlannedHours / totalPlannedHours.doubleValue()) : new Double(0); // Differences calculations this.costDifference = calculateCostDifference(averageProgress, new BigDecimal(totalPlannedHours), realHours.toHoursAsDecimalWithScale(2)); this.planningDifference = calculatePlanningDifference(averageProgress, new BigDecimal(totalPlannedHours), new BigDecimal( partialPlannedHours)); this.ratioCostDifference = calculateRatioCostDifference( averageProgress, imputedProgress); this.ratioPlanningDifference = calculateRatioPlanningDifference( averageProgress, plannedProgress); if (this.averageProgress.compareTo(BigDecimal.ONE) > 0) { this.averageProgress = BigDecimal.ONE; } if (this.imputedProgress > 1) { this.imputedProgress = new Double(1); } if (this.plannedProgress > 1) { this.plannedProgress = new Double(1); } } private BigDecimal getFilterAdvanceTypePercentage(Order order, AdvanceType type, LocalDate date) { final BigDecimal result; if (type != null) { result = order.getAdvancePercentage(type, date); if (result != null) { return result; } } if (type != null) { advanceTypeDoesNotApply = true; appliedSpreadAdvanceType = true; } final DirectAdvanceAssignment directAdvanceAssignment = order .getReportGlobalAdvanceAssignment(); return (directAdvanceAssignment != null) ? directAdvanceAssignment .getAdvancePercentage(date) : null; } private Integer getHoursSpecifiedAtOrder(List<Task> tasks) { int result = 0; for (Task each: tasks) { result += each.getHoursSpecifiedAtOrder(); } return result; } public Integer calculatePlannedHours(List<Task> tasks, LocalDate date) { int result = 0; for (Task each: tasks) { result += calculatePlannedHours(each, date); } return result; } public Integer calculatePlannedHours(Task task, LocalDate date) { final List<DayAssignment> dayAssignments = task .getDayAssignments(FilterType.WITHOUT_DERIVED); return DayAssignment.sum(removeAfterDate(dayAssignments, date)) .roundToHours(); } public EffortDuration calculateRealHours(Order order, LocalDate date) { EffortDuration result = EffortDuration.zero(); final List<WorkReportLine> workReportLines = workReportLineDAO .findByOrderElementAndChildren(order); for (WorkReportLine workReportLine : workReportLines) { final LocalDate workReportLineDate = new LocalDate(workReportLine.getDate()); if (date == null || workReportLineDate.compareTo(date) <= 0) { result = EffortDuration.sum(result, workReportLine.getEffort()); } } return result; } public Integer getEstimatedHours() { return estimatedHours; } public Integer getTotalPlannedHours() { return totalPlannedHours; } public Integer getPartialPlannedHours() { return partialPlannedHours; } public EffortDuration getRealHours() { return realHours; } public BigDecimal getAverageProgress() { return averageProgress; } public Double getImputedProgress() { return imputedProgress; } public Double getPlannedProgress() { return plannedProgress; } public String getOrderName() { return orderName; } public BigDecimal calculateCostDifference(BigDecimal averageProgress, BigDecimal totalPlannedHours, BigDecimal realHours) { BigDecimal result = averageProgress; result = result.multiply(totalPlannedHours); return result.subtract(realHours); } public BigDecimal calculatePlanningDifference(BigDecimal averageProgress, BigDecimal totalPlannedHours, BigDecimal partialPlannedHours) { BigDecimal result = averageProgress; result = result.multiply(totalPlannedHours); return result.subtract(partialPlannedHours); } public BigDecimal calculateRatioCostDifference(BigDecimal averageProgress, Double imputedProgress) { if (imputedProgress.doubleValue() == 0) { return new BigDecimal(0); } return averageProgress.divide(new BigDecimal(imputedProgress), 2, RoundingMode.HALF_UP); } public BigDecimal calculateRatioPlanningDifference(BigDecimal averageProgress, Double plannedProgress) { if (plannedProgress.doubleValue() == 0) { return new BigDecimal(0); } return averageProgress.divide(new BigDecimal(plannedProgress), 2, RoundingMode.HALF_UP); } public BigDecimal getCostDifference() { return costDifference; } public BigDecimal getPlanningDifference() { return planningDifference; } public BigDecimal getRatioCostDifference() { return ratioCostDifference; } public BigDecimal getRatioPlanningDifference() { return ratioPlanningDifference; } public Boolean getAdvanceTypeDoesNotApply() { return advanceTypeDoesNotApply; } public Boolean getAppliedSpreadAdvanceType() { return appliedSpreadAdvanceType; } }