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

com.jidesoft.range.CombinedNumericRange Maven / Gradle / Ivy

/*
 * @(#)CombinedNumericRange.java
 * 
 * 2002 - 2012 JIDE Software Incorporated. All rights reserved.
 * Copyright (c) 2005 - 2012 Catalysoft Limited. All rights reserved.
 */

package com.jidesoft.range;

import java.util.ArrayList;
import java.util.List;

/**
 * A little convenience class to compute the maximum and minimum values of multiple ranges.
 *
 * @author [email protected]
 */
public class CombinedNumericRange extends AbstractNumericRange {
    private final Object monitor = new Object();
    private List> _ranges = new ArrayList>();
    private Double _max = null;
    private Double _min = null;

    /**
     * Using this constructor relies on the user subsequently calling add() to add a range
     */
    public CombinedNumericRange() {
        super();
    }

    /**
     * Add a new range to this combined range. Notice the method returns this instance, so method calls can be chained
     * together. If you pass null to this method the CombinedNumericRange remains unchanged; an Exception is NOT
     * thrown.
     *
     * @param range the new range to add
     * @return this instance
     */
    public CombinedNumericRange add(Range range) {
        if (range == null) {
            return this;
        }
        synchronized (monitor) {
            _ranges.add(range);
            _min = null;
            _max = null;
        }
        return this;
    }

    /**
     * The lower value in the range; here, the same as minimum
     */
    public Double lower() {
        return minimum();
    }

    /**
     * The upper value in the range; here, the same as maximum()
     */
    public Double upper() {
        return maximum();
    }

    /**
     * Returns the number of sub-ranges contained by this range
     *
     * @return the number of sub-ranges in this range
     */
    public int rangeCount() {
        return _ranges == null ? 0 : _ranges.size();
    }

    /**
     * Calls to this method throw an UnsupportedOprationException. The idea is that in the case of this class we don't
     * want to be able to mess with the lower and upper bounds as they are computed from the supplied range values. If
     * the class needed to recompute the lower and upper bounds any previous adjustment that had been made through this
     * method would have been lost.
     *
     * @throws UnsupportedOperationException
     */
    public void adjust(Double lower, Double upper) {
        throw new UnsupportedOperationException();
    }

    /**
     * Lazily calculates the maximum value in the range
     */
    public double maximum() {
        synchronized (monitor) {
            if (_max != null) {
                return _max;
            }
            if (_ranges == null || _ranges.size() == 0) {
                return Double.MAX_VALUE;
            }
            _max = -Double.MAX_VALUE;
            for (Range range : _ranges) {
                if (range != null && range.maximum() > _max) {
                    _max = range.maximum();
                }
            }
            return _max;
        }
    }

    /**
     * Lazily calculates the minimum value in the range
     */
    public double minimum() {
        synchronized (monitor) {
            if (_min != null) {
                return _min;
            }
            if (_ranges == null || _ranges.size() == 0) {
                return -Double.MAX_VALUE;
            }
            _min = Double.MAX_VALUE;
            for (Range range : _ranges) {
                if (range != null && range.minimum() < _min) {
                    _min = range.minimum();
                }
            }
            return _min;
        }
    }


    /**
     * This range contains some point iff one or more of its sub-ranges contain that point
     */
    public boolean contains(Double x) {
        synchronized (monitor) {
            if (x == null || _ranges.size() == 0) {
                return false;
            }
            else {
                for (Range range : _ranges) {
                    if (range.contains(x)) {
                        return true;
                    }
                }
                return false;
            }
        }
    }

    /**
     * The size of the range is computed as the maximum minus the minimum value.
     */
    public double size() {
        synchronized (monitor) {
            return maximum() - minimum();
        }
    }

    /**
     * Returns a new numeric range that is based on this range, but with a margin introduced at each end. The margin
     * proportion is a value between 0 and 1. For example to add a 20% margin to each end use parameters of 0.2 for both
     * the leading and trailing margin proportion.
     *
     * @param leadingMarginProportion  how much margin to add at the low end of the range
     * @param trailingMarginProportion how much margin to add at the top end of the range
     * @return a new NumericRange object with margins added
     */
    public NumericRange getRange(double leadingMarginProportion, double trailingMarginProportion) {
        double maximum = maximum();
        double minimum = minimum();
        double difference = Math.abs(maximum - minimum);
        double leadingMargin = leadingMarginProportion * difference;
        double trailingMargin = trailingMarginProportion * difference;
        return new NumericRange(minimum - leadingMargin, maximum + trailingMargin);
    }

    @Override
    public Range createIntermediate(Range targetRange, double position) {
        double sourceMin = this.minimum();
        double sourceMax = this.maximum();
        double targetMin = targetRange.minimum();
        double targetMax = targetRange.maximum();
        double min = sourceMin + position * (targetMin - sourceMin);
        double max= sourceMax + position * (targetMax - sourceMax);
        return new NumericRange(min, max);
    }

    public String toString() {
        return String.format("#", minimum(), maximum());
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy