com.projity.pm.assignment.HasAssignmentsImpl.java Source code

Java tutorial

Introduction

Here is the source code for com.projity.pm.assignment.HasAssignmentsImpl.java

Source

/*
The contents of this file are subject to the Common Public Attribution License
Version 1.0 (the "License"); you may not use this file except in compliance with
the License. You may obtain a copy of the License at
http://www.projity.com/license . The License is based on the Mozilla Public
License Version 1.1 but Sections 14 and 15 have been added to cover use of
software over a computer network and provide for limited attribution for the
Original Developer. In addition, Exhibit A has been modified to be consistent
with Exhibit B.
    
Software distributed under the License is distributed on an "AS IS" basis,
WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for the
specific language governing rights and limitations under the License. The
Original Code is OpenProj. The Original Developer is the Initial Developer and
is Projity, Inc. All portions of the code written by Projity are Copyright (c)
2006, 2007. All Rights Reserved. Contributors Projity, Inc.
    
Alternatively, the contents of this file may be used under the terms of the
Projity End-User License Agreeement (the Projity License), in which case the
provisions of the Projity License are applicable instead of those above. If you
wish to allow use of your version of this file only under the terms of the
Projity License and not to allow others to use your version of this file under
the CPAL, indicate your decision by deleting the provisions above and replace
them with the notice and other provisions required by the Projity  License. If
you do not delete the provisions above, a recipient may use your version of this
file under either the CPAL or the Projity License.
    
[NOTE: The text of this license may differ slightly from the text of the notices
in Exhibits A and B of the license at http://www.projity.com/license. You should
use the latest text at http://www.projity.com/license for your modifications.
You may not remove this license text from the source files.]
    
Attribution Information: Attribution Copyright Notice: Copyright  2006, 2007
Projity, Inc. Attribution Phrase (not exceeding 10 words): Powered by OpenProj,
an open source solution from Projity. Attribution URL: http://www.projity.com
Graphic Image as provided in the Covered Code as file:  openproj_logo.png with
alternatives listed on http://www.projity.com/logo
    
Display of Attribution Information is required in Larger Works which are defined
in the CPAL as a work which combines Covered Code or portions thereof with code
not governed by the terms of the CPAL. However, in addition to the other notice
obligations, all copies of the Covered Code in Executable and Source Code form
distributed must, as a form of attribution of the original author, include on
each user interface screen the "OpenProj" logo visible to all users.  The
OpenProj logo should be located horizontally aligned with the menu bar and left
justified on the top left of the screen adjacent to the File menu.  The logo
must be at least 100 x 25 pixels.  When users click on the "OpenProj" logo it
must direct them back to http://www.projity.com.
*/
package com.projity.pm.assignment;

import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.ListIterator;

import org.apache.commons.collections.Closure;
import org.apache.commons.collections.Predicate;
import org.apache.commons.collections.functors.TruePredicate;

import com.projity.algorithm.ReverseQuery;
import com.projity.algorithm.TimeIteratorGenerator;
import com.projity.algorithm.buffer.CalculatedValues;
import com.projity.algorithm.buffer.IntervalCallback;
import com.projity.algorithm.buffer.NonGroupedCalculatedValues;
import com.projity.association.AssociationList;
import com.projity.field.FieldContext;
import com.projity.functor.CollectionVisitor;
import com.projity.options.ScheduleOption;
import com.projity.pm.calendar.WorkCalendar;
import com.projity.pm.criticalpath.TaskSchedule;
import com.projity.pm.resource.Resource;
import com.projity.pm.resource.ResourceImpl;
import com.projity.pm.scheduling.SchedulingType;
import com.projity.pm.task.NormalTask;
import com.projity.pm.task.Task;
import com.projity.pm.time.HasStartAndEnd;
import com.projity.pm.time.MutableInterval;

/**
 * Implementation of class which contains assignments
 */
public class HasAssignmentsImpl implements HasAssignments, HasTimeDistributedData, Serializable, Cloneable {
    //private static Log log = LogFactory.getLog(HasAssignmentsImpl.class);
    transient AssociationList assignments;

    //TODO scheduling rule and effort driven don't make sense for resources, so make them go away?
    int schedulingRule = ScheduleOption.getInstance().getSchedulingRule();
    boolean effortDriven = ScheduleOption.getInstance().isEffortDriven();

    public HasAssignmentsImpl() {
        assignments = new AssociationList();
    }

    public boolean isReadOnlyEffortDriven(FieldContext fieldContext) {
        return getSchedulingType() == SchedulingType.FIXED_WORK;
    }

    /**
     * Copy constructor: It does a deep copy of assignments
     * @param from
     */
    private HasAssignmentsImpl(HasAssignmentsImpl from) {
        this();
        Iterator i = from.assignments.iterator();
        while (i.hasNext()) {
            assignments.add(new Assignment((Assignment) i.next()));
        }
    }

    public HasAssignmentsImpl(Collection details) {
        this();
        Iterator i = details.iterator();
        while (i.hasNext()) {
            assignments.add(new Assignment((AssignmentDetail) i.next()));
        }
    }

    /**
     * @param schedule
     * @return
     */
    //   public HasAssignments cloneWithSchedule(TaskSchedule currentSchedule) {
    //      return cloneWithSchedule(currentSchedule,null);
    //   }
    //   public HasAssignments cloneWithSchedule(TaskSchedule currentSchedule,Collection details) {
    //      HasAssignmentsImpl newOne;
    //      if (details==null) newOne= new HasAssignmentsImpl(this);
    //      else newOne= new HasAssignmentsImpl(details);
    //      newOne.setScheduleForAssignments(currentSchedule);
    //      return newOne;
    //   }
    public HasAssignments cloneWithSchedule(TaskSchedule currentSchedule) {
        HasAssignmentsImpl newOne = new HasAssignmentsImpl(this);
        newOne.setScheduleForAssignments(currentSchedule);
        return newOne;
    }

    private void setScheduleForAssignments(TaskSchedule currentSchedule) {
        Iterator i = assignments.iterator();
        Assignment assignment;
        while (i.hasNext()) {
            assignment = (Assignment) i.next();
            assignment.setTaskSchedule(currentSchedule);
            assignment.convertToBaselineAssignment(false);
        }
    }

    //very deep copy of assignments contrary to copy constructor which doesn't clone assigments' detail
    public HasAssignments deepCloneWithTask(Task task) { //TODO doesn't
        HasAssignmentsImpl newOne = (HasAssignmentsImpl) cloneWithTask(task);
        return newOne;
    }

    public Object clone() {
        try {
            return super.clone();
        } catch (CloneNotSupportedException e) {
            throw new InternalError();
        }
    }

    public Object cloneWithTask(Task task) {
        HasAssignmentsImpl clone = (HasAssignmentsImpl) clone();
        clone.assignments = new AssociationList();

        //TODO doesn't work when it's copied between projects
        Iterator i = assignments.iterator();
        while (i.hasNext()) {
            clone.assignments.add(((Assignment) i.next()).cloneWithTask(task));
            //            clone.assignments.add(((Assignment)i.next()).cloneWithResourceAndTask(ResourceImpl.getUnassignedInstance(),task));
            //break;
        }

        //         Iterator i = assignments.iterator();
        //         while (i.hasNext()) {
        //            clone.assignments.add(((Assignment)i.next()).cloneWithTask(task));
        //         }

        return clone;
    }

    public Object cloneWithResource(Resource resource) {
        HasAssignmentsImpl clone = (HasAssignmentsImpl) clone();
        clone.assignments = new AssociationList();
        Iterator i = assignments.iterator();
        while (i.hasNext()) {
            clone.assignments.add(((Assignment) i.next()).cloneWithResource(resource));
        }

        return clone;
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasAssignments#getAssignments()
     */
    public AssociationList getAssignments() {
        return assignments;
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasAssignments#addAssignment(com.projity.pm.assignment.Assignment)
     */
    public void addAssignment(Assignment assignment) {
        assignments.add(assignment);
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasAssignments#removeAssignment(com.projity.pm.assignment.Assignment)
     */
    public void removeAssignment(Assignment assignment) {
        assignments.remove(assignment);
    }

    /**
     * Finds an assignment given a resource
     */
    public Assignment findAssignment(Resource resource) {
        Iterator i = assignments.iterator();
        Assignment result = null;
        while (i.hasNext()) {
            result = (Assignment) i.next();
            if (result.getResource() == resource)
                return result;
        }
        return null;
    }

    /**
     * Finds an assignment given a task
     */
    public Assignment findAssignment(Task task) {
        Iterator i = assignments.iterator();
        Assignment result = null;
        while (i.hasNext()) {
            result = (Assignment) i.next();
            if (result.getTask() == task)
                return result;
        }
        return null;
    }

    /**
     * @return Returns the schedulingRule.
     */
    public int getSchedulingType() {
        return schedulingRule;
    }

    /**
     * @param schedulingType The schedulingRule to set.
     */
    public void setSchedulingType(int schedulingType) {
        this.schedulingRule = schedulingType;
    }

    /**
     * @return Returns the effortDriven.
     */
    public boolean isEffortDriven() {
        return effortDriven;
    }

    /**
     * @param effortDriven The effortDriven to set.
     */
    public void setEffortDriven(boolean effortDriven) {
        this.effortDriven = effortDriven;
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasTimeDistributedData#collectIntervalGenerators(java.lang.Object, java.util.Collection)
     */
    public void buildReverseQuery(ReverseQuery reverseQuery) {
        Iterator i = assignments.iterator();
        Assignment assignment;
        while (i.hasNext()) {
            assignment = (Assignment) i.next();
            if (assignment.isDefault() && !reverseQuery.isAllowDefaultAssignments())
                continue;
            assignment.buildReverseQuery(reverseQuery);
        }
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasAssignments#updateAssignment(com.projity.pm.assignment.Assignment)
     */
    public void updateAssignment(Assignment modified) {
        ListIterator i = assignments.listIterator();
        Assignment current = null;
        while (i.hasNext()) {
            current = (Assignment) i.next();
            if (current.getTask() == modified.getTask() && current.getResource() == modified.getResource()) {
                i.set(modified); // replace current with new one
                break;
            }
        }
    }

    public static Closure forAllAssignments(Closure visitor, Predicate filter) {
        return new CollectionVisitor(visitor, filter) {
            protected final Collection getCollection(Object arg0) {
                return ((HasAssignments) arg0).getAssignments();
            }
        };
    }

    public static Closure forAllAssignments(Closure visitor) {
        return forAllAssignments(visitor, TruePredicate.INSTANCE);
    }

    public void forEachInterval(Closure visitor, Object type, WorkCalendar workCalendar) {
        NonGroupedCalculatedValues calculatedValues = new NonGroupedCalculatedValues(false, 0);
        ListIterator i = assignments.listIterator();
        Assignment assignment = null;

        while (i.hasNext()) { // add in all child groups
            assignment = (Assignment) i.next();
            barCallback.setWorkCalendar(assignment.getEffectiveWorkCalendar()); // use this assignments cal because it might work on off calendar time
            assignment.calcDataBetween(type, null, calculatedValues);
        }
        calculatedValues.makeContiguousNonZero(barCallback, workCalendar);
        //calculatedValues.dump();
    }

    public void forEachWorkingInterval(final Closure visitor, boolean mergeWorking, WorkCalendar workCalendar) {
        barCallback.initialize(workCalendar, visitor, true);
        forEachInterval(visitor, ACTUAL_WORK, workCalendar);
        /* if the splitting should be at latest bar use this code
              barCallback.finish();
              barCallback.initialize(workCalendar,visitor,false);
        */
        barCallback.initialize(workCalendar, visitor, true);
        forEachInterval(visitor, REMAINING_WORK, workCalendar);
    }

    private static BarSeriesCallback barCallback = new BarSeriesCallback();

    private static class BarSeriesCallback implements IntervalCallback {
        long barStart = 0;
        WorkCalendar workCalendar;
        MutableInterval interval = new MutableInterval(0, 0);
        Closure visitor;
        long previousEnd = 0;

        private void executeVisitor(long start, long end) {
            start = Math.max(start, previousEnd); // prevent overlap in case of multiple assignments that do not have same advancement
            if (start > end)
                return;

            interval.setStart(start);
            interval.setEnd(end);

            previousEnd = end;
            //System.out.println("bar " + new Date(start) + " " + new Date(end));
            visitor.execute(interval);
            barStart = 0;
        }

        public void setWorkCalendar(WorkCalendar workCalendar) {
            this.workCalendar = workCalendar;

        }

        private static double ALMOST_ZERO = 0.00001;

        public void add(int index, long start, long end, double value) {
            if (value <= ALMOST_ZERO) { // because of rounding errors, treat 0 as something very small
                if (workCalendar.compare(end, start, false) == 0)
                    return;
                if (barStart > 0) {
                    start = workCalendar.adjustInsideCalendar(start, true);
                    executeVisitor(barStart, start);
                }
            } else {
                if (barStart == 0) {
                    //hk               barStart = start;
                    barStart = workCalendar.adjustInsideCalendar(start, false);
                }
                if (index == 0) {// last bar, must draw
                    end = workCalendar.adjustInsideCalendar(end, true);
                    executeVisitor(barStart, end);
                    //               System.out.println("last bar " + new Date(start) + " " + new Date(end));
                }
            }
        }

        private void initialize(WorkCalendar workCalendar, Closure visitor, boolean firstTime) {
            if (firstTime)
                previousEnd = 0;
            this.workCalendar = workCalendar;
            this.visitor = visitor;
        }

        private void finish() {
            if (barStart != 0) {
                //            System.out.println("finishing bar " + new Date(barStart) + " " + new Date(previousEnd));
                executeVisitor(barStart, previousEnd);
            }
            barStart = 0;
        }
    }

    /* (non-Javadoc)
     * @see com.projity.pm.costing.EarnedValueValues#acwp(long, long)
     */
    public double acwp(long start, long end) {
        return TimeDistributedDataConsolidator.acwp(start, end, childrenToRollup());
    }

    /* (non-Javadoc)
     * @see com.projity.pm.costing.EarnedValueValues#bac(long, long)
     */
    public double bac(long start, long end) {
        return TimeDistributedDataConsolidator.bac(start, end, childrenToRollup());
    }

    /* (non-Javadoc)
     * @see com.projity.pm.costing.EarnedValueValues#bcwp(long, long)
     */
    public double bcwp(long start, long end) {
        return TimeDistributedDataConsolidator.bcwp(start, end, childrenToRollup());
    }

    /* (non-Javadoc)
     * @see com.projity.pm.costing.EarnedValueValues#bcws(long, long)
     */
    public double bcws(long start, long end) {
        return TimeDistributedDataConsolidator.bcws(start, end, childrenToRollup());
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasTimeDistributedData#cost(long, long)
     */
    public double cost(long start, long end) {
        return TimeDistributedDataConsolidator.cost(start, end, childrenToRollup());
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasTimeDistributedData#cost(long, long)
     */
    public double baselineCost(long start, long end) {
        return TimeDistributedDataConsolidator.baselineCost(start, end, childrenToRollup());
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasTimeDistributedData#cost(long, long)
     */
    public long baselineWork(long start, long end) {
        return TimeDistributedDataConsolidator.baselineWork(start, end, childrenToRollup(), true);
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasTimeDistributedData#actualCost(long, long)
     */
    public double actualCost(long start, long end) {
        return TimeDistributedDataConsolidator.actualCost(start, end, childrenToRollup());
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasTimeDistributedData#work(long, long)
     */
    public long work(long start, long end) {
        return TimeDistributedDataConsolidator.work(start, end, childrenToRollup(), true);
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasTimeDistributedData#actualWork(long, long)
     */
    public long actualWork(long start, long end) {
        return TimeDistributedDataConsolidator.actualWork(start, end, childrenToRollup(), true);
    }

    public long remainingWork(long start, long end) {
        return TimeDistributedDataConsolidator.remainingWork(start, end, childrenToRollup(), true);
    }

    public void calcDataBetween(Object type, TimeIteratorGenerator generator, CalculatedValues values) {
        Iterator i = getAssignments().iterator();
        while (i.hasNext()) {
            ((Assignment) i.next()).calcDataBetween(type, generator, values);
        }

    }

    public static List extractOppositeList(List list, boolean leftObject) {
        Iterator i = list.iterator();
        ArrayList assignments = new ArrayList();
        while (i.hasNext()) { // go thru tasks or resources
            Object object = i.next();
            if (!(object instanceof HasAssignments))
                continue; //TODO currently getting voidNodeImpl's.  This should go away when fixed
            HasAssignments hasAssignments = (HasAssignments) object;
            assignments.addAll(hasAssignments.getAssignments());
        }
        return AssociationList.extractDistinct(assignments, leftObject);
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasTimeDistributedData#childrenToRollup()
     */
    public Collection childrenToRollup() {
        return assignments;
    }

    /**
     *
     */
    private class AssignmentDurationSummer implements Closure {
        private long sum;
        private WorkCalendar workCalendar;

        AssignmentDurationSummer(WorkCalendar workCalendar) {
            this.workCalendar = workCalendar;
            sum = 0;
        }

        public void execute(Object arg0) {
            HasStartAndEnd interval = (HasStartAndEnd) arg0;
            sum += workCalendar.compare(interval.getEnd(), interval.getStart(), false);
        }

        public long getSum() {
            return sum;
        }
    }

    /**
     * Compute the sum of active assignment durations.  If there are multiple assignments, then
     * the calendar time of the union of active periods is used, otherwise, if just one assignment
     * (which could be the default assignment), use the assignment duration
     * @param workCalendar
     * @return
     */
    public long calcActiveAssignmentDuration(WorkCalendar workCalendar) {
        AssociationList assignments = getAssignments();
        // Most of the time there is just one assignment. If that's the case, use the assignment duration
        if (assignments.size() == 1)
            return ((Assignment) assignments.getFirst()).getDurationMillis();
        AssignmentDurationSummer summer = new AssignmentDurationSummer(workCalendar);
        forEachWorkingInterval(summer, false, workCalendar);
        return summer.getSum();

    }

    private void writeObject(ObjectOutputStream s) throws IOException {
        s.defaultWriteObject();
    }

    private void readObject(ObjectInputStream s) throws IOException, ClassNotFoundException {
        s.defaultReadObject();
        assignments = new AssociationList();
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasTimeDistributedData#fixedCost(long, long)
     */
    public double fixedCost(long start, long end) {
        // TODO Auto-generated method stub
        return 0;
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasTimeDistributedData#actualFixedCost(long, long)
     */
    public double actualFixedCost(long start, long end) {
        // TODO Auto-generated method stub
        return 0;
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasTimeDistributedData#isLabor()
     */
    public boolean isLabor() {
        // TODO Auto-generated method stub
        return false;
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasAssignments#hasLaborAssignment()
     */
    public boolean hasLaborAssignment() {
        Iterator i = assignments.iterator();
        while (i.hasNext()) {
            if (((Assignment) i.next()).isLabor())
                return true;
        }
        return false;
    }

    /* (non-Javadoc)
     * @see com.projity.pm.assignment.HasAssignments#invalidateAssignmentCalendars()
     */
    public void invalidateAssignmentCalendars() {
        Iterator i = assignments.iterator();
        while (i.hasNext()) {
            ((Assignment) i.next()).invalidateAssignmentCalendar();
        }
    }

    public boolean hasActiveAssignment(long start, long end) {
        Iterator i = assignments.iterator();
        Assignment assignment;
        while (i.hasNext()) {
            assignment = (Assignment) i.next();
            if (assignment.isActiveBetween(start, end))
                return true;
        }
        return false;
    }

    public long getEarliestAssignmentStart() {
        long result = Long.MAX_VALUE;
        Iterator i = assignments.iterator();
        while (i.hasNext()) {
            result = Math.min(result, ((Assignment) i.next()).getStart());
        }
        return result;
    }

}