All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.protempa.proposition.interval.Interval Maven / Gradle / Ivy

There is a newer version: 5.2-Alpha-2
Show newest version
/*
 * #%L
 * Protempa Framework
 * %%
 * Copyright (C) 2012 - 2013 Emory University
 * %%
 * Licensed under the Apache License, Version 2.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.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 * #L%
 */
package org.protempa.proposition.interval;

import java.io.IOException;
import java.io.InvalidObjectException;
import java.io.ObjectInputStream;
import java.util.Arrays;

import org.protempa.graph.Weight;
import org.protempa.graph.WeightFactory;
import org.protempa.ProtempaUtil;
import org.protempa.proposition.value.Granularity;
import org.protempa.proposition.value.Unit;

/**
 * A representation of an interval, designed according to the definition of an
 * interval in Combi et al. Managing Different Time Granularities of Clinical
 * Information by an Interval-based Temporal Data Model. Meth. Inf. Med.
 * 1995;34:458-74. This representation is also designed to be used in simple
 * temporal problems, as defined in Dechter, R. et al. Temporal Constraint
 * Networks. Artif. Intell. 1991;49:61-95.
 *
 * @author Andrew Post
 */
public abstract class Interval implements Comparable {

    public static enum Side {

        START(IntervalUtil.resourceBundle().getString("intervalSide.startDisplayName")),
        FINISH(IntervalUtil.resourceBundle().getString("intervalSide.finishDisplayName"));
        private transient String displayName;

        private Side(String displayName) {
            this.displayName = displayName;
        }
        
        @Override
        public String toString() {
            return displayName;
        }
    }
    private static final WeightFactory weightFactory = new WeightFactory();
    private Long minStart;
    private Long maxStart;
    private Granularity startGranularity;
    private Long minFinish;
    private Long maxFinish;
    private Granularity finishGranularity;
    private Long minLength;
    private Long maxLength;
    private Unit lengthUnit;
    //Compute these
    private Start start;
    private Finish finish;
    protected Long[] v;
    private Weight[] vw;
    private volatile int hashCode;
    private boolean minLengthComputed;
    private boolean maxLengthComputed;

    /**
     * Create an interval with default values (minimumStart=-inf,
     * maximumStart=+inf, minimumFinish=-inf, maximumFinish=+inf,
     * minimumDuration=0, maximumDuration=+inf).
     *
     * @param description
     */
    Interval() {
        this(null, null, null, null, null, null);
    }

    Interval(Long start, Granularity startGranularity, Long finish,
            Granularity finishGranularity, Long length, Unit lengthUnit) {
        this(start, start, startGranularity, finish, finish, finishGranularity,
                length, length, lengthUnit);
    }

    Interval(Long minStart, Long maxStart, Granularity startGranularity,
            Long minFinish, Long maxFinish, Granularity finishGranularity,
            Long minLength, Long maxLength, Unit lengthUnit) {
        init(minStart, maxStart, startGranularity, minFinish, maxFinish,
                finishGranularity, minLength, maxLength, lengthUnit);
    }

    protected void init(Long start, Granularity startGranularity, Long finish,
            Granularity finishGranularity, Long length, Unit lengthUnit) {
        init(start, start, startGranularity, finish, finish, finishGranularity,
                length, length, lengthUnit);
    }

    protected void init(Long minStart, Long maxStart,
            Granularity startGranularity,
            Long minFinish, Long maxFinish, Granularity finishGranularity,
            Long minLength, Long maxLength, Unit lengthUnit) {
        if (minStart != null && maxFinish != null && minStart > maxFinish) {
            throw new IllegalArgumentException(
                    "maxFinish cannot be before minStart; maxFinish="
                    + maxFinish + "; minStart=" + minStart);
        }
        if (minLength != null && minLength < 0) {
            throw new IllegalArgumentException(
                    "minLength must be positive or 0 but was " + minLength);
        }
        if (maxLength != null && maxLength < 0) {
            throw new IllegalArgumentException(
                    "maxLength must be positive or 0 but was " + maxLength);
        }

        this.minStart = minStart;
        this.maxStart = maxStart;
        this.startGranularity = startGranularity;
        this.minFinish = minFinish;
        this.maxFinish = maxFinish;
        this.finishGranularity = finishGranularity;
        this.minLength = minLength;
        this.maxLength = maxLength;
        this.lengthUnit = lengthUnit;

        initComputed();
    }

    protected void initComputed() throws IllegalArgumentException {
        this.start = new Start(this);
        this.finish = new Finish(this);

        Long w0 = null;
        Long w1 = null;
        Long w2 = null;
        Long w3 = null;
        Long w4 = 0L;
        Long w5 = null;
        if (startGranularity != null) {
            if (minStart != null) {
                w0 = minStart;
            }
            if (maxStart != null) {
                w1 = startGranularity.latest(maxStart);
            }
        } else {
            if (minStart != null) {
                w0 = minStart;
            }
            if (maxStart != null) {
                w1 = maxStart;
            }
        }
        if (finishGranularity != null) {
            if (minFinish != null) {
                w2 = minFinish;
            }
            if (maxFinish != null) {
                w3 = finishGranularity.latest(maxFinish);
            }
        } else {
            if (minFinish != null) {
                w2 = minFinish;
            }
            if (maxFinish != null) {
                w3 = maxFinish;
            }
        }
        Granularity g1 = null;
        if (startGranularity != null && finishGranularity != null) {
            if (startGranularity.compareTo(finishGranularity) < 0) {
                g1 = startGranularity;
            } else {
                g1 = finishGranularity;
            }
        } else if (startGranularity != null) {
            g1 = startGranularity;
        } else {
            g1 = finishGranularity;
        }
        if (minLength != null) {
            if (g1 != null) {
                w4 = g1.minimumDistance(maxStart, minLength, lengthUnit);
            } else if (minFinish != null && maxStart != null) {
                w4 = minLength.longValue();
            }
            this.minLengthComputed = true;
        }
        if (maxLength != null) {
            if (g1 != null) {
                w5 = g1.maximumDistance(minStart, maxLength, lengthUnit);
            } else if (maxFinish != null && minStart != null) {
                w5 = minLength.longValue();
            }
            this.maxLengthComputed = true;
        }
        v = new Long[]{w0, w1, w2, w3, w4, w5};
        if ((v[0] != null && v[1] != null && v[0].compareTo(v[1]) > 0) || (v[2] != null && v[3] != null && v[2].compareTo(v[3]) > 0) || (v[4] != null && v[5] != null && v[4].compareTo(v[5]) > 0)) {
            throw new IllegalArgumentException("Illegal values for interval");
        }
    }

    protected void computeLength() {
        if (!this.minLengthComputed || !this.maxLengthComputed) {
            //set default min and max length
            Long w4 = 0L;
            Long w5 = null;

            //set granularities
            Granularity g1 = null;
            Granularity g2 = null;
            if (startGranularity != null && finishGranularity != null) {
                if (startGranularity.compareTo(finishGranularity) < 0) {
                    g1 = startGranularity;
                    g2 = finishGranularity;
                } else {
                    g1 = finishGranularity;
                    g2 = startGranularity;
                }
            } else if (startGranularity != null) {
                g1 = startGranularity;
            } else {
                g1 = finishGranularity;
            }

            //set min length
            if (!this.minLengthComputed) {
                if (this.minLength == null) {
                    if (g1 != null) {
                        lengthUnit = g1.getCorrespondingUnit();
                        if (maxStart != null && minFinish != null) {
                            minLength = g1.distance(maxStart, minFinish, g2,
                                    lengthUnit);
                        }
                    } else if (minFinish != null && maxStart != null) {
                        minLength = minFinish - maxStart;
                    }
                }
                if (g1 != null) {
                    if (maxStart != null && minLength != null) {
                        w4 = g1.minimumDistance(maxStart, minLength,
                                lengthUnit);
                    }
                } else if (minFinish != null && maxStart != null) {
                    w4 = minLength.longValue();
                }
                minLengthComputed = true;
            }

            //set max length
            if (!this.maxLengthComputed) {
                if (this.maxLength == null) {
                    if (g1 != null) {
                        lengthUnit = g1.getCorrespondingUnit();
                        if (minStart != null && maxFinish != null) {
                            maxLength = g1.distance(minStart, maxFinish, g2,
                                    lengthUnit);
                        }

                    } else if (maxFinish != null && minStart != null) {
                        maxLength = maxFinish - minStart;
                    }
                }
                if (g1 != null) {
                    if (minStart != null && maxLength != null) {
                        w5 = g1.maximumDistance(minStart, maxLength,
                                lengthUnit);
                    }
                } else if (maxFinish != null && minStart != null) {
                    w5 = minLength.longValue();
                }
                this.maxLengthComputed = true;
            }

            //set min and max length
            this.v[4] = w4;
            this.v[5] = w5;
        }
    }

    /**
     * Gets the minimum start in internal PROTEMPA units.
     *
     * @return a {@link Long}.
     */
    public Long getMinimumStart() {
        return v[0];
    }

    /**
     * Gets the maximum start in internal PROTEMPA units.
     *
     * @return a {@link Long}.
     */
    public Long getMaximumStart() {
        return v[1];
    }

    /**
     * Gets the minimum finish in internal PROTEMPA units.
     *
     * @return a {@link Long}.
     */
    public Long getMinimumFinish() {
        return v[2];
    }

    /**
     * Gets the maximum finish in internal PROTEMPA units.
     *
     * @return a {@link Long}.
     */
    public Long getMaximumFinish() {
        return v[3];
    }

    /**
     * Gets the minimum length in internal PROTEMPA units.
     *
     * @return a {@link Long}.
     */
    public Long getMinimumLength() {
        computeLength();
        return v[4];
    }

    /**
     * Gets the maximum length in internal PROTEMPA units.
     *
     * @return a {@link Long}.
     */
    public Long getMaximumLength() {
        computeLength();
        return v[5];
    }

    private void initVw() {
        this.vw = new Weight[]{
            v[0] != null ? weightFactory.getInstance(v[0]) : WeightFactory.NEG_INFINITY,
            v[1] != null ? weightFactory.getInstance(v[1]) : WeightFactory.POS_INFINITY,
            v[2] != null ? weightFactory.getInstance(v[2]) : WeightFactory.NEG_INFINITY,
            v[3] != null ? weightFactory.getInstance(v[3]) : WeightFactory.POS_INFINITY,
            v[4] != null ? weightFactory.getInstance(v[4]) : WeightFactory.ZERO,
            v[5] != null ? weightFactory.getInstance(v[5]) : WeightFactory.POS_INFINITY};
    }

    /**
     * Gets the minimum start in constraint network weights.
     *
     * @return a {@link Weight}.
     */
    Weight getSpecifiedMinimumStart() {
        if (vw == null) {
            initVw();
        }
        return vw[0];
    }

    /**
     * Gets the maximum start in constraint network weights.
     *
     * @return a {@link Weight}.
     */
    Weight getSpecifiedMaximumStart() {
        if (vw == null) {
            initVw();
        }
        return vw[1];
    }

    /**
     * Gets the minimum finish in constraint network weights.
     *
     * @return a {@link Weight}.
     */
    Weight getSpecifiedMinimumFinish() {
        if (vw == null) {
            initVw();
        }
        return vw[2];
    }

    /**
     * Gets the maximum finish in constraint network weights.
     *
     * @return a {@link Weight}.
     */
    Weight getSpecifiedMaximumFinish() {
        if (vw == null) {
            initVw();
        }
        return vw[3];
    }

    /**
     * Gets the minimum length in constraint network weights.
     *
     * @return a {@link Weight}.
     */
    Weight getSpecifiedMinimumLength() {
        if (vw == null) {
            initVw();
        }
        return vw[4];
    }

    /**
     * Gets the maximum length in constraint network weights.
     *
     * @return a {@link Weight}.
     */
    Weight getSpecifiedMaximumLength() {
        if (vw == null) {
            initVw();
        }
        return vw[5];
    }

    /**
     * Get the start vertex of this interval.
     *
     * @return
     */
    Start getStart() {
        return start;
    }

    /**
     * Get the finish vertex of this interval.
     *
     * @return
     */
    Finish getFinish() {
        return finish;
    }

    public boolean isLengthGreaterThan(int duration, Unit durationUnits) {
        Long minS = getMinimumStart();
        Long minF = getMinimumFinish();
        if (durationUnits == null || minS == null || minF == null) {
            if (minF == null) {
                return false;
            } else if (minS == null) {
                return true;
            } else {
                return minS + duration < minF;
            }
        } else {
            return !Relation.isGreaterThanOrEqualToDuration(durationUnits,
                    minS, minF, duration);
        }
    }

    public boolean isLengthLessThan(int duration, Unit durationUnits) {
        Long maxS = getMaximumStart();
        Long maxF = getMaximumFinish();
        if (durationUnits == null || maxS == null || maxF == null) {
            if (maxF == null) {
                return false;
            } else if (maxS == null) {
                return true;
            } else {
                return maxS + duration > maxF;
            }
        } else {
            return !Relation.isLessThanOrEqualToDuration(durationUnits,
                    maxS, maxF, duration);
        }
    }

    /*
     * (non-Javadoc)
     *
     * @see java.lang.Object#toString()
     */
    @Override
    public String toString() {
        return Arrays.asList(v).toString();
    }

    static final class Start {

        private final Interval interval;
        private volatile int hashCode;

        Start(Interval interval) {
            this.interval = interval;
        }

        /*
         * (non-Javadoc)
         *
         * @see java.lang.Object#hashCode()
         */
        @Override
        public int hashCode() {
            if (this.hashCode == 0) {
                int result = 17;
                result = 37 * result + this.interval.hashCode();
                this.hashCode = result;
            }
            return hashCode;
        }

        /*
         * (non-Javadoc)
         *
         * @see java.lang.Object#toString()
         */
        @Override
        public String toString() {
            return "" + interval + " start";
        }
    }

    static final class Finish {

        private final Interval interval;
        private volatile int hashCode;

        Finish(Interval interval) {
            this.interval = interval;
        }

        /*
         * (non-Javadoc)
         *
         * @see java.lang.Object#hashCode()
         */
        @Override
        public int hashCode() {
            if (this.hashCode == 0) {
                int result = 17;
                result = 37 * result + this.interval.hashCode();
                result = 37 * result;
                this.hashCode = result;
            }
            return hashCode;
        }

        @Override
        public String toString() {
            return "" + interval + " finish";
        }
    }

    @Override
    public boolean equals(Object other) {
        if (other == this) {
            return true;
        }

        if (!(other instanceof Interval)) {
            return false;
        }

        Interval otherIval = (Interval) other;
        return Arrays.equals(this.v, otherIval.v);
    }

    @Override
    public int hashCode() {
        if (this.hashCode == 0) {
            int result = 17;
            result = 37 * result + Arrays.hashCode(this.v);
            this.hashCode = result;
        }

        return this.hashCode;
    }

    @Override
    public final int compareTo(Interval o) {
        if (this == o) {
            return 0;
        }

        Long start1 = this.getMinimumStart();
        Long finish1 = this.getMaximumFinish();
        Long start2 = o.getMinimumStart();
        Long finish2 = o.getMaximumFinish();

        int result;
        if (start1 == null && start2 != null) {
            result = -1;
        } else if (start1 != null && start2 == null) {
            result = 1;
        } else if (start1 == null && start2 == null) {
            result = 0;
        } else {
            result = start1.compareTo(start2);
        }
        if (result != 0) {
            return result;
        }
        if (finish1 == null && finish2 != null) {
            return 1;
        } else if (finish1 != null && finish2 == null) {
            return -1;
        } else if (finish1 == null && finish2 == null) {
            return 0;
        } else {
            return finish1.compareTo(finish2);
        }
    }

    /**
     * Gets the minimum start in the current start granularity.
     *
     * @return a {@link Long}.
     */
    public Long getMinStart() {
        return minStart;
    }

    /**
     * Gets the maximum start in the current start granularity.
     *
     * @return a {@link Long}.
     */
    public Long getMaxStart() {
        return maxStart;
    }

    /**
     * Gets the start granularity.
     *
     * @return a {@link Granularity}.
     */
    public Granularity getStartGranularity() {
        return startGranularity;
    }

    /**
     * Gets the minimum finish in the current finish granularity.
     *
     * @return a {@link Long}.
     */
    public Long getMinFinish() {
        return minFinish;
    }

    /**
     * Gets the maximum finish in the current finish granularity.
     *
     * @return a {@link Long}.
     */
    public Long getMaxFinish() {
        return maxFinish;
    }

    /**
     * Returns the finish granularity.
     *
     * @return a {@link Granularity}.
     */
    public Granularity getFinishGranularity() {
        return finishGranularity;
    }

    /**
     * Gets the minimum length in the finest granularity of those defined for
     * the start and finish.
     *
     * @return a {@link Long}.
     */
    public Long getMinLength() {
        computeLength();
        return minLength;
    }

    public Long minLengthIn(Unit unit) {
        if (unit != null) {
            Long maxStart = getMaxStart();
            Long minFinish = getMinFinish();
            if (maxStart == null || minFinish == null) {
                return null;
            }

            Granularity startGran = getStartGranularity();
            if (startGran != null) {
                return startGran.distance(maxStart, minFinish,
                        getFinishGranularity(), unit);
            } else {
                Granularity finishGran = getFinishGranularity();
                if (finishGran != null) {
                    return finishGran.distance(maxStart, minFinish, null, unit);
                } else {
                    return minFinish - maxStart;
                }
            }
        } else {
            return getMinLength();
        }
    }

    public Long maxLengthIn(Unit unit) {
        if (unit != null) {
            Long minStart = getMinStart();
            Long maxFinish = getMaxFinish();
            if (minStart == null || maxFinish == null) {
                return null;
            }
            Granularity startGran = getStartGranularity();
            if (startGran != null) {
                return startGran.distance(minStart, maxFinish,
                        getFinishGranularity(), unit);
            } else {
                Granularity finishGran = getFinishGranularity();
                if (finishGran != null) {
                    return finishGran.distance(minStart, maxFinish, null, unit);
                } else {
                    return maxFinish - minStart;
                }
            }
        } else {
            return getMaxLength();
        }
    }

    /**
     * Gets the maximum length in the finest granularity of those defined for
     * the start and finish.
     *
     * @return a {@link Long}.
     */
    public Long getMaxLength() {
        computeLength();
        return maxLength;
    }

    /**
     * Gets the units of the length.
     *
     * @return a {@link Unit}.
     */
    public Unit getLengthUnit() {
        computeLength();
        return lengthUnit;
    }

    private void readObject(ObjectInputStream s) throws IOException,
            ClassNotFoundException {
        s.defaultReadObject();

        if (minStart != null && maxFinish != null && minStart > maxFinish) {
            throw new InvalidObjectException(
                    "maxFinish cannot be before minStart; maxFinish="
                    + maxFinish + "; minStart=" + minStart);
        }
        if (minLength != null && minLength < 0) {
            throw new InvalidObjectException(
                    "minLength must be positive or 0 but was " + minLength);
        }
        if (maxLength != null && maxLength < 0) {
            throw new InvalidObjectException(
                    "maxLength must be positive or 0 but was " + maxLength);
        }

        try {
            initComputed();
        } catch (IllegalArgumentException iae) {
            throw new InvalidObjectException(iae.getMessage());
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy