| 
									
										
										
										
											2022-02-11 09:07:11 +11:00
										 |  |  | /* SPDX-License-Identifier: GPL-2.0-or-later */ | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-07 09:50:34 +02:00
										 |  |  | #pragma once
 | 
					
						
							| 
									
										
										
										
											2019-09-13 21:12:26 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup bli | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  |  * A `blender::IndexRange` wraps an interval of non-negative integers. It can be used to reference | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * consecutive elements in an array. Furthermore, it can make for loops more convenient and less | 
					
						
							|  |  |  |  * error prone, especially when using nested loops. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * I'd argue that the second loop is more readable and less error prone than the first one. That is | 
					
						
							|  |  |  |  * not necessarily always the case, but often it is. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |  *  for (int64_t i = 0; i < 10; i++) { | 
					
						
							|  |  |  |  *    for (int64_t j = 0; j < 20; j++) { | 
					
						
							|  |  |  |  *       for (int64_t k = 0; k < 30; k++) { | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |  *  for (int64_t i : IndexRange(10)) { | 
					
						
							|  |  |  |  *    for (int64_t j : IndexRange(20)) { | 
					
						
							|  |  |  |  *      for (int64_t k : IndexRange(30)) { | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  |  * Some containers like blender::Vector have an index_range() method. This will return the | 
					
						
							|  |  |  |  * IndexRange that contains all indices that can be used to access the container. This is | 
					
						
							|  |  |  |  * particularly useful when you want to iterate over the indices and the elements (much like | 
					
						
							|  |  |  |  * Python's enumerate(), just worse). Again, I think the second example here is better: | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |  *  for (int64_t i = 0; i < my_vector_with_a_long_name.size(); i++) { | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  *    do_something(i, my_vector_with_a_long_name[i]); | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |  *  for (int64_t i : my_vector_with_a_long_name.index_range()) { | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  *    do_something(i, my_vector_with_a_long_name[i]); | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Ideally this could be could be even closer to Python's enumerate(). We might get that in the | 
					
						
							|  |  |  |  * future with newer C++ versions. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2022-01-05 19:03:05 -06:00
										 |  |  |  * One other important feature is the as_span method. This method returns a Span<int64_t> | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * that contains the interval as individual numbers. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include <cmath>
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | #include <iostream>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_utildefines.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  | namespace blender { | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  | template<typename T> class Span; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | class IndexRange { | 
					
						
							|  |  |  |  private: | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   int64_t start_ = 0; | 
					
						
							|  |  |  |   int64_t size_ = 0; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr IndexRange() = default; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr explicit IndexRange(int64_t size) : start_(0), size_(size) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     BLI_assert(size >= 0); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr IndexRange(int64_t start, int64_t size) : start_(start), size_(size) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     BLI_assert(start >= 0); | 
					
						
							|  |  |  |     BLI_assert(size >= 0); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   class Iterator { | 
					
						
							| 
									
										
										
										
											2021-03-20 15:42:35 +01:00
										 |  |  |    public: | 
					
						
							|  |  |  |     using iterator_category = std::forward_iterator_tag; | 
					
						
							|  |  |  |     using value_type = int64_t; | 
					
						
							|  |  |  |     using pointer = const int64_t *; | 
					
						
							|  |  |  |     using reference = const int64_t &; | 
					
						
							|  |  |  |     using difference_type = std::ptrdiff_t; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    private: | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     int64_t current_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |    public: | 
					
						
							| 
									
										
										
										
											2021-03-20 15:42:35 +01:00
										 |  |  |     constexpr explicit Iterator(int64_t current) : current_(current) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |     constexpr Iterator &operator++() | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |       current_++; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |       return *this; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-20 15:42:35 +01:00
										 |  |  |     constexpr Iterator operator++(int) const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       Iterator copied_iterator = *this; | 
					
						
							|  |  |  |       ++copied_iterator; | 
					
						
							|  |  |  |       return copied_iterator; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     constexpr friend bool operator!=(const Iterator &a, const Iterator &b) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       return a.current_ != b.current_; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     constexpr friend bool operator==(const Iterator &a, const Iterator &b) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-03-20 15:42:35 +01:00
										 |  |  |       return a.current_ == b.current_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |     constexpr int64_t operator*() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |       return current_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr Iterator begin() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return Iterator(start_); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr Iterator end() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return Iterator(start_ + size_); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Access an element in the range. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr int64_t operator[](int64_t index) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     BLI_assert(index >= 0); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |     BLI_assert(index < this->size()); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return start_ + index; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Two ranges compare equal when they contain the same numbers. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr friend bool operator==(IndexRange a, IndexRange b) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return (a.size_ == b.size_) && (a.start_ == b.start_ || a.size_ == 0); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Get the amount of numbers in the range. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr int64_t size() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return size_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-03 11:14:44 -05:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Returns true if the size is zero. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   constexpr bool is_empty() const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return size_ == 0; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Create a new range starting at the end of the current one. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr IndexRange after(int64_t n) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     BLI_assert(n >= 0); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return IndexRange(start_ + size_, n); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Create a new range that ends at the start of the current one. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr IndexRange before(int64_t n) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     BLI_assert(n >= 0); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return IndexRange(start_ - n, n); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Get the first element in the range. | 
					
						
							|  |  |  |    * Asserts when the range is empty. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr int64_t first() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(this->size() > 0); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return start_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Get the last element in the range. | 
					
						
							|  |  |  |    * Asserts when the range is empty. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr int64_t last() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(this->size() > 0); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return start_ + size_ - 1; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Get the element one after the end. The returned value is undefined when the range is empty. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr int64_t one_after_last() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return start_ + size_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Get the first element in the range. The returned value is undefined when the range is empty. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr int64_t start() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return start_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Returns true when the range contains a certain number, otherwise false. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr bool contains(int64_t value) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return value >= start_ && value < start_ + size_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-25 23:13:02 +10:00
										 |  |  |    * Returns a new range, that contains a sub-interval of the current one. | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr IndexRange slice(int64_t start, int64_t size) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     BLI_assert(start >= 0); | 
					
						
							|  |  |  |     BLI_assert(size >= 0); | 
					
						
							|  |  |  |     int64_t new_start = start_ + start; | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     BLI_assert(new_start + size <= start_ + size_ || size == 0); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |     return IndexRange(new_start, size); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-12-16 13:02:28 +05:30
										 |  |  |   constexpr IndexRange slice(IndexRange range) const | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   { | 
					
						
							|  |  |  |     return this->slice(range.start(), range.size()); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-03 11:14:44 -05:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Returns a new IndexRange with n elements removed from the beginning. This invokes undefined | 
					
						
							|  |  |  |    * behavior when n is negative. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   constexpr IndexRange drop_front(int64_t n) const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(n >= 0); | 
					
						
							|  |  |  |     const int64_t new_size = std::max<int64_t>(0, size_ - n); | 
					
						
							|  |  |  |     return IndexRange(start_ + n, new_size); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Returns a new IndexRange with n elements removed from the beginning. This invokes undefined | 
					
						
							|  |  |  |    * behavior when n is negative. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   constexpr IndexRange drop_back(int64_t n) const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(n >= 0); | 
					
						
							|  |  |  |     const int64_t new_size = std::max<int64_t>(0, size_ - n); | 
					
						
							|  |  |  |     return IndexRange(start_, new_size); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Returns a new IndexRange that only contains the first n elements. This invokes undefined | 
					
						
							|  |  |  |    * behavior when n is negative. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   constexpr IndexRange take_front(int64_t n) const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(n >= 0); | 
					
						
							|  |  |  |     const int64_t new_size = std::min<int64_t>(size_, n); | 
					
						
							|  |  |  |     return IndexRange(start_, new_size); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Returns a new IndexRange that only contains the last n elements. This invokes undefined | 
					
						
							|  |  |  |    * behavior when n is negative. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   constexpr IndexRange take_back(int64_t n) const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(n >= 0); | 
					
						
							|  |  |  |     const int64_t new_size = std::min<int64_t>(size_, n); | 
					
						
							|  |  |  |     return IndexRange(start_ + size_ - new_size, new_size); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Get read-only access to a memory buffer that contains the range as actual numbers. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   Span<int64_t> as_span() const; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   friend std::ostream &operator<<(std::ostream &stream, IndexRange range) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     stream << "[" << range.start() << ", " << range.one_after_last() << ")"; | 
					
						
							|  |  |  |     return stream; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  | }  // namespace blender
 |