123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305 |
- /*
- ==============================================================================
- This file is part of the juce_core module of the JUCE library.
- Copyright (c) 2015 - ROLI Ltd.
- Permission to use, copy, modify, and/or distribute this software for any purpose with
- or without fee is hereby granted, provided that the above copyright notice and this
- permission notice appear in all copies.
- THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD
- TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN
- NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
- DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER
- IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
- CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
- ------------------------------------------------------------------------------
- NOTE! This permissive ISC license applies ONLY to files within the juce_core module!
- All other JUCE modules are covered by a dual GPL/commercial license, so if you are
- using any other modules, be sure to check that you also comply with their license.
- For more details, visit www.juce.com
- ==============================================================================
- */
- #ifndef JUCE_RANGE_H_INCLUDED
- #define JUCE_RANGE_H_INCLUDED
- //==============================================================================
- /** A general-purpose range object, that simply represents any linear range with
- a start and end point.
- Note that when checking whether values fall within the range, the start value is
- considered to be inclusive, and the end of the range exclusive.
- The templated parameter is expected to be a primitive integer or floating point
- type, though class types could also be used if they behave in a number-like way.
- */
- template <typename ValueType>
- class Range
- {
- public:
- //==============================================================================
- /** Constructs an empty range. */
- Range() noexcept : start(), end()
- {
- }
- /** Constructs a range with given start and end values. */
- Range (const ValueType startValue, const ValueType endValue) noexcept
- : start (startValue), end (jmax (startValue, endValue))
- {
- }
- /** Constructs a copy of another range. */
- Range (const Range& other) noexcept
- : start (other.start), end (other.end)
- {
- }
- /** Copies another range object. */
- Range& operator= (Range other) noexcept
- {
- start = other.start;
- end = other.end;
- return *this;
- }
- /** Returns the range that lies between two positions (in either order). */
- static Range between (const ValueType position1, const ValueType position2) noexcept
- {
- return position1 < position2 ? Range (position1, position2)
- : Range (position2, position1);
- }
- /** Returns a range with a given start and length. */
- static Range withStartAndLength (const ValueType startValue, const ValueType length) noexcept
- {
- jassert (length >= ValueType());
- return Range (startValue, startValue + length);
- }
- /** Returns a range with the specified start position and a length of zero. */
- static Range emptyRange (const ValueType start) noexcept
- {
- return Range (start, start);
- }
- //==============================================================================
- /** Returns the start of the range. */
- inline ValueType getStart() const noexcept { return start; }
- /** Returns the length of the range. */
- inline ValueType getLength() const noexcept { return end - start; }
- /** Returns the end of the range. */
- inline ValueType getEnd() const noexcept { return end; }
- /** Returns true if the range has a length of zero. */
- inline bool isEmpty() const noexcept { return start == end; }
- //==============================================================================
- /** Changes the start position of the range, leaving the end position unchanged.
- If the new start position is higher than the current end of the range, the end point
- will be pushed along to equal it, leaving an empty range at the new position.
- */
- void setStart (const ValueType newStart) noexcept
- {
- start = newStart;
- if (end < newStart)
- end = newStart;
- }
- /** Returns a range with the same end as this one, but a different start.
- If the new start position is higher than the current end of the range, the end point
- will be pushed along to equal it, returning an empty range at the new position.
- */
- Range withStart (const ValueType newStart) const noexcept
- {
- return Range (newStart, jmax (newStart, end));
- }
- /** Returns a range with the same length as this one, but moved to have the given start position. */
- Range movedToStartAt (const ValueType newStart) const noexcept
- {
- return Range (newStart, end + (newStart - start));
- }
- /** Changes the end position of the range, leaving the start unchanged.
- If the new end position is below the current start of the range, the start point
- will be pushed back to equal the new end point.
- */
- void setEnd (const ValueType newEnd) noexcept
- {
- end = newEnd;
- if (newEnd < start)
- start = newEnd;
- }
- /** Returns a range with the same start position as this one, but a different end.
- If the new end position is below the current start of the range, the start point
- will be pushed back to equal the new end point.
- */
- Range withEnd (const ValueType newEnd) const noexcept
- {
- return Range (jmin (start, newEnd), newEnd);
- }
- /** Returns a range with the same length as this one, but moved to have the given end position. */
- Range movedToEndAt (const ValueType newEnd) const noexcept
- {
- return Range (start + (newEnd - end), newEnd);
- }
- /** Changes the length of the range.
- Lengths less than zero are treated as zero.
- */
- void setLength (const ValueType newLength) noexcept
- {
- end = start + jmax (ValueType(), newLength);
- }
- /** Returns a range with the same start as this one, but a different length.
- Lengths less than zero are treated as zero.
- */
- Range withLength (const ValueType newLength) const noexcept
- {
- return Range (start, start + newLength);
- }
- //==============================================================================
- /** Adds an amount to the start and end of the range. */
- inline Range operator+= (const ValueType amountToAdd) noexcept
- {
- start += amountToAdd;
- end += amountToAdd;
- return *this;
- }
- /** Subtracts an amount from the start and end of the range. */
- inline Range operator-= (const ValueType amountToSubtract) noexcept
- {
- start -= amountToSubtract;
- end -= amountToSubtract;
- return *this;
- }
- /** Returns a range that is equal to this one with an amount added to its
- start and end.
- */
- Range operator+ (const ValueType amountToAdd) const noexcept
- {
- return Range (start + amountToAdd, end + amountToAdd);
- }
- /** Returns a range that is equal to this one with the specified amount
- subtracted from its start and end. */
- Range operator- (const ValueType amountToSubtract) const noexcept
- {
- return Range (start - amountToSubtract, end - amountToSubtract);
- }
- bool operator== (Range other) const noexcept { return start == other.start && end == other.end; }
- bool operator!= (Range other) const noexcept { return start != other.start || end != other.end; }
- //==============================================================================
- /** Returns true if the given position lies inside this range.
- When making this comparison, the start value is considered to be inclusive,
- and the end of the range exclusive.
- */
- bool contains (const ValueType position) const noexcept
- {
- return start <= position && position < end;
- }
- /** Returns the nearest value to the one supplied, which lies within the range. */
- ValueType clipValue (const ValueType value) const noexcept
- {
- return jlimit (start, end, value);
- }
- /** Returns true if the given range lies entirely inside this range. */
- bool contains (Range other) const noexcept
- {
- return start <= other.start && end >= other.end;
- }
- /** Returns true if the given range intersects this one. */
- bool intersects (Range other) const noexcept
- {
- return other.start < end && start < other.end;
- }
- /** Returns the range that is the intersection of the two ranges, or an empty range
- with an undefined start position if they don't overlap. */
- Range getIntersectionWith (Range other) const noexcept
- {
- return Range (jmax (start, other.start),
- jmin (end, other.end));
- }
- /** Returns the smallest range that contains both this one and the other one. */
- Range getUnionWith (Range other) const noexcept
- {
- return Range (jmin (start, other.start),
- jmax (end, other.end));
- }
- /** Returns the smallest range that contains both this one and the given value. */
- Range getUnionWith (const ValueType valueToInclude) const noexcept
- {
- return Range (jmin (valueToInclude, start),
- jmax (valueToInclude, end));
- }
- /** Returns a given range, after moving it forwards or backwards to fit it
- within this range.
- If the supplied range has a greater length than this one, the return value
- will be this range.
- Otherwise, if the supplied range is smaller than this one, the return value
- will be the new range, shifted forwards or backwards so that it doesn't extend
- beyond this one, but keeping its original length.
- */
- Range constrainRange (Range rangeToConstrain) const noexcept
- {
- const ValueType otherLen = rangeToConstrain.getLength();
- return getLength() <= otherLen
- ? *this
- : rangeToConstrain.movedToStartAt (jlimit (start, end - otherLen, rangeToConstrain.getStart()));
- }
- /** Scans an array of values for its min and max, and returns these as a Range. */
- static Range findMinAndMax (const ValueType* values, int numValues) noexcept
- {
- if (numValues <= 0)
- return Range();
- const ValueType first (*values++);
- Range r (first, first);
- while (--numValues > 0) // (> 0 rather than >= 0 because we've already taken the first sample)
- {
- const ValueType v (*values++);
- if (r.end < v) r.end = v;
- if (v < r.start) r.start = v;
- }
- return r;
- }
- private:
- //==============================================================================
- ValueType start, end;
- };
- #endif // JUCE_RANGE_H_INCLUDED
|