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

org.protempa.proposition.Segment Maven / Gradle / Ivy

/*
 * #%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;

import org.protempa.proposition.interval.IntervalFactory;
import org.protempa.proposition.interval.Interval;
import java.util.AbstractList;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.RandomAccess;

import org.protempa.proposition.value.Granularity;

/**
 * A segment of a sequence of Parameter objects provided to
 * Detector instances' satisfiedBy() method. Do not
 * attempt to cache a Segment object! Segment objects
 * may be reused by PROTEMPA for performance reasons, and the value of a
 * Segment object is only guaranteed to be consistent until
 * satisfiedBy() ends.
 *
 * @author Andrew Post
 */
public class Segment extends AbstractList
        implements RandomAccess {

    private static final IntervalFactory intervalFactory
            = new IntervalFactory();
    private Sequence ts;
    private int x = -1;
    private int y = -1;
    protected int modCount = 0;
    private boolean intervalStale = true;
    private T maxFinishParam;
    private Interval interval;

    public Segment(Sequence seq, int firstIndex, int lastIndex) {
        if (seq == null) {
            throw new IllegalArgumentException("seq cannot be null!");
        }
        resetState(seq, firstIndex, lastIndex);
    }

    public Segment(Sequence seq) {
        this(seq, 0, seq != null ? seq.size() - 1 : 0);
    }

    public Segment(Segment segment) {
        this(segment != null ? segment.ts : null, segment != null ? segment.getFirstIndex() : 0, segment != null ? segment.getLastIndex()
                : 0);
    }

    public Sequence getSequence() {
        return ts;
    }

    public Interval getInterval() {
        if (intervalStale || interval == null) {
            interval = intervalCreator();
            intervalStale = false;
        }
        return interval;
    }

    protected Interval intervalCreator() {
        if (size() == 1) {
            return first().getInterval();
        } else {
            Long minStart = minimumStart(this);
            Long maxStart = maximumStart(this);
            Long minFinish = minimumFinish(this);
            Long maxFinish = maximumFinish(this);
            return intervalFactory.getInstance(minStart, maxStart,
                    getStartGranularity(),
                    minFinish, maxFinish, getFinishGranularity());
        }
    }

    public int getFirstIndex() {
        if (x == -1) {
            throw new ArrayIndexOutOfBoundsException();
        }
        return x;
    }

    public int getLastIndex() {
        if (y == -1) {
            throw new ArrayIndexOutOfBoundsException();
        }
        return y;
    }

    @Override
    public int indexOf(Object o) {
        if (!(o instanceof TemporalProposition)) {
            return -1;
        } else {
            return Collections.binarySearch(this, (TemporalProposition) o,
                    PropositionUtil.TEMP_PROP_COMP);
        }
    }

    @Override
    public int lastIndexOf(Object o) {
        return indexOf(o);
    }

    public T first() {
        return ts.get(getFirstIndex());
    }

    public T last() {
        return ts.get(getLastIndex());
    }

    public Granularity getFinishGranularity() {
        calcMaxFinishParam();
        return maxFinishParam.getInterval().getFinishGranularity();
    }

    public Granularity getStartGranularity() {
        return first().getInterval().getStartGranularity();
    }

    /*
     * (non-Javadoc)
     *
     * @see java.util.Collection#size()
     */
    @Override
    public int size() {
        if (x == -1) {
            return 0;
        } else {
            return y - x + 1;
        }

    }

    private void calcMaxFinishParam() {
        if (maxFinishParam == null) {
            maxFinishParam = Collections.max(this,
                    PropositionUtil.MAX_FINISH_COMP);
        }
    }

    public Segment resetState(Sequence sequence) {
        if (sequence == null) {
            return null;
        } else {
            return resetState(sequence, 0, sequence.size() - 1);
        }
    }

    public Segment resetState(Sequence sequence, int firstIndex,
            int lastIndex) {

        if (sequence == null || firstIndex >= sequence.size()
                || lastIndex >= sequence.size() || firstIndex > lastIndex) {
            return null;
        }

        ts = sequence;
        x = firstIndex;
        y = lastIndex;
        modCount++;
        intervalStale = true;
        maxFinishParam = null;
        return this;

    }

    @Override
    public T get(int index) {
        return (T) ts.get(x + index);
    }

    /**
     * Unsupported operation.
     *
     * @see java.util.Collection#addAll(java.util.Collection)
     */
    @Override
    public boolean addAll(Collection c) {
        throw new UnsupportedOperationException();
    }

    /*
     * (non-Javadoc)
     *
     * @see java.util.Collection#contains(java.lang.Object)
     */
    @Override
    public boolean contains(Object o) {
        return indexOf(o) >= 0;
    }

    private Long maximumFinish(Segment segment) {
        calcMaxFinishParam();
        return maxFinishParam.getInterval().getMaxFinish();
    }

    private Long minimumStart(Segment segment) {
        /*
         * Returns the smallest minimum start, which is that of the first
         * element.
         */
        return segment.first().getInterval().getMinStart();
    }
    private Comparator MAX_START_COMP
            = new Comparator() {

                @Override
                public int compare(TemporalProposition p0, TemporalProposition p1) {
                    Long p0MaxStart = p0.getInterval().getMaximumStart();
                    Long p1MaxStart = p1.getInterval().getMaximumStart();
                    if (p0MaxStart != null && p1MaxStart != null) {
                        return p0MaxStart.compareTo(p1MaxStart);
                    } else if (p0MaxStart != null) {
                        return -1;
                    } else if (p1MaxStart != null) {
                        return 1;
                    } else {
                        return 0;
                    }
                }
            };

    private Long maximumStart(
            Segment segment) {
        return Collections.min(segment, MAX_START_COMP).getInterval().getMaxStart();
    }
    private Comparator MIN_FINISH_COMP
            = new Comparator() {

                @Override
                public int compare(TemporalProposition p0, TemporalProposition p1) {
                    Long p0MinFinish = p0.getInterval().getMinimumFinish();
                    Long p1MinFinish = p1.getInterval().getMinimumFinish();
                    if (p0MinFinish != null && p1MinFinish != null) {
                        return p0MinFinish.compareTo(p1MinFinish);
                    } else if (p0MinFinish != null) {
                        return 1;
                    } else if (p1MinFinish != null) {
                        return -1;
                    } else {
                        return 0;
                    }
                }
            };

    private Long minimumFinish(
            Segment segment) {
        return Collections.max(segment, MIN_FINISH_COMP).getInterval().getMinFinish();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy