Loading [MathJax]/extensions/tex2jax.js
ITK 6.0.0
Insight Toolkit
 
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
Loading...
Searching...
No Matches
itkIndex.h
Go to the documentation of this file.
1/*=========================================================================
2 *
3 * Copyright NumFOCUS
4 *
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
8 *
9 * https://www.apache.org/licenses/LICENSE-2.0.txt
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 *
17 *=========================================================================*/
18#ifndef itkIndex_h
19#define itkIndex_h
20
21#include "itkMakeFilled.h"
22#include "itkOffset.h"
23
24#include <cstddef> // For ptrdiff_t.
25#include <type_traits> // For is_integral.
26
27namespace itk
28{
29
66
67template <unsigned int VDimension = 2>
68struct ITK_TEMPLATE_EXPORT Index final
69{
70public:
71 // Using the `rule of zero` to this aggregate type
72 // C++20 changes the definition of aggregate such that classes with any user-declared ctors are no longer aggregates.
73
75 using Self = Index;
76
80
83
87
89 static constexpr unsigned int Dimension = VDimension;
90
92 static constexpr unsigned int
94 {
95 return VDimension;
96 }
97
98
100 const Self
101 operator+(const SizeType & sz) const
102 {
103 Self result;
104
105 for (unsigned int i = 0; i < VDimension; ++i)
106 {
107 result[i] = m_InternalArray[i] + static_cast<IndexValueType>(sz[i]);
108 }
109 return result;
110 }
111
113 const Self &
115 {
116 for (unsigned int i = 0; i < VDimension; ++i)
117 {
118 m_InternalArray[i] += static_cast<IndexValueType>(sz[i]);
119 }
120 return *this;
121 }
122
125 const Self
126 operator-(const SizeType & sz) const
127 {
128 Self result;
129
130 for (unsigned int i = 0; i < VDimension; ++i)
131 {
132 result[i] = m_InternalArray[i] - static_cast<IndexValueType>(sz[i]);
133 }
134 return result;
135 }
136
138 const Self &
140 {
141 for (unsigned int i = 0; i < VDimension; ++i)
142 {
143 m_InternalArray[i] -= static_cast<IndexValueType>(sz[i]);
144 }
145 return *this;
146 }
147
149 const Self
150 operator+(const OffsetType & offset) const
151 {
152 Self result;
153
154 for (unsigned int i = 0; i < VDimension; ++i)
155 {
156 result[i] = m_InternalArray[i] + offset[i];
157 }
158 return result;
159 }
160
162 const Self &
163 operator+=(const OffsetType & offset)
164 {
165 for (unsigned int i = 0; i < VDimension; ++i)
166 {
167 m_InternalArray[i] += offset[i];
168 }
169 return *this;
170 }
171
173 const Self &
174 operator-=(const OffsetType & offset)
175 {
176 for (unsigned int i = 0; i < VDimension; ++i)
177 {
178 m_InternalArray[i] -= offset[i];
179 }
180 return *this;
181 }
182
184 const Self
185 operator-(const OffsetType & off) const
186 {
187 Self result;
188
189 for (unsigned int i = 0; i < VDimension; ++i)
190 {
191 result[i] = m_InternalArray[i] - off.m_InternalArray[i];
192 }
193 return result;
194 }
195
197 const OffsetType
198 operator-(const Self & vec) const
199 {
200 OffsetType result;
201
202 for (unsigned int i = 0; i < VDimension; ++i)
203 {
204 result[i] = m_InternalArray[i] - vec.m_InternalArray[i];
205 }
206 return result;
207 }
208
212 const Self
213 operator*(const SizeType & vec) const
214 {
215 Self result;
216
217 for (unsigned int i = 0; i < VDimension; ++i)
218 {
219 result[i] = m_InternalArray[i] * static_cast<IndexValueType>(vec.m_InternalArray[i]);
220 }
221 return result;
222 }
223
226 const IndexValueType *
227 GetIndex() const
228 {
229 return m_InternalArray;
230 }
231
236 void
237 SetIndex(const IndexValueType val[VDimension])
238 {
239 std::copy_n(val, VDimension, m_InternalArray);
240 }
241
248 void
249 SetElement(unsigned int element, IndexValueType val)
250 {
251 m_InternalArray[element] = val;
252 }
253
261 GetElement(unsigned int element) const
262 {
263 return m_InternalArray[element];
264 }
265
268 void
270 {
271 std::fill_n(begin(), size(), value);
272 } // MATCH std::array assign, ITK Fill
273
278 /*
279 * Ask the compiler to align a type to the maximum useful alignment for the target
280 * machine you are compiling for. Whenever you leave out the alignment factor in an
281 * aligned attribute specification, the compiler automatically sets the alignment
282 * for the type to the largest alignment that is ever used for any data type on
283 * the target machine you are compiling for. Doing this can often make copy
284 * operations more efficient, because the compiler can use whatever instructions
285 * copy the biggest chunks of memory when performing copies to or from the variables
286 * that have types that you have aligned this way.
287 */
288 static_assert(VDimension > 0, "Error: Only positive value sized VDimension allowed");
290
292 template <typename TCoordinate>
293 inline void
295 {
296 for (unsigned int i = 0; i < VDimension; ++i)
297 {
299 }
300 }
301
303 template <typename TCoordinate>
304 inline void
306 {
307 for (unsigned int i = 0; i < VDimension; ++i)
308 {
309 m_InternalArray[i] = static_cast<IndexValueType>(point[i]);
310 }
311 }
312
316 static Self
317 GetBasisIndex(unsigned int dim);
318
319
320 // ======================= Mirror the access pattern behavior of the std::array class
330 using const_iterator = const value_type *;
331 using size_type = unsigned int;
332 using difference_type = ptrdiff_t;
333 using reverse_iterator = std::reverse_iterator<iterator>;
334 using const_reverse_iterator = std::reverse_iterator<const_iterator>;
335
340 void
341 assign(const value_type & newValue)
342 {
343 std::fill_n(begin(), size(), newValue);
344 }
345
346 void
347 swap(Index & other) noexcept
348 {
349 std::swap(m_InternalArray, other.m_InternalArray);
350 }
351
352 constexpr const_iterator
353 cbegin() const
354 {
355 return &m_InternalArray[0];
356 }
357
358 constexpr iterator
360 {
361 return &m_InternalArray[0];
362 }
363
364 constexpr const_iterator
365 begin() const
366 {
367 return &m_InternalArray[0];
368 }
369
370 constexpr const_iterator
371 cend() const
372 {
373 return &m_InternalArray[VDimension];
374 }
375
376 constexpr iterator
378 {
379 return &m_InternalArray[VDimension];
380 }
381
382 constexpr const_iterator
383 end() const
384 {
385 return &m_InternalArray[VDimension];
386 }
387
388 reverse_iterator
390 {
391 return reverse_iterator(end());
392 }
393
394 const_reverse_iterator
395 rbegin() const
396 {
397 return const_reverse_iterator(end());
398 }
399
400 reverse_iterator
402 {
403 return reverse_iterator(begin());
404 }
405
406 const_reverse_iterator
407 rend() const
408 {
410 }
411
412 constexpr size_type
413 size() const
414 {
415 return VDimension;
416 }
417
418 constexpr size_type
419 max_size() const
420 {
421 return VDimension;
422 }
423
424 constexpr bool
425 empty() const
426 {
427 return false;
428 }
429
430 constexpr reference
432 {
433 return m_InternalArray[pos];
434 }
435
436 constexpr const_reference
438 {
439 return m_InternalArray[pos];
440 }
441
442 reference
444 {
446 return m_InternalArray[pos];
447 }
448
449 const_reference
450 at(size_type pos) const
451 {
453 return m_InternalArray[pos];
454 }
455
456 constexpr reference
458 {
459 return *begin();
460 }
461
462 constexpr const_reference
463 front() const
464 {
465 return *begin();
466 }
467
468 constexpr reference
470 {
471 return VDimension ? *(end() - 1) : *end();
472 }
473
474 constexpr const_reference
475 back() const
476 {
477 return VDimension ? *(end() - 1) : *end();
478 }
479
482 {
483 return &m_InternalArray[0];
484 }
485
486 const IndexValueType *
487 data() const
488 {
489 return &m_InternalArray[0];
490 }
491
492
495 static constexpr Self
497 {
498 return MakeFilled<Self>(value);
499 }
500
501
502private:
503 void
505 {
506 if (pos >= VDimension)
507 {
508 throw std::out_of_range("array::ExceptionThrowingBoundsCheck");
509 }
510 }
511
512}; //------------ End struct Index
513
514template <unsigned int VDimension>
515Index<VDimension>
517{
518 Self ind{ { 0 } };
519
520 ind.m_InternalArray[dim] = 1;
521 return ind;
522}
523
524template <unsigned int VDimension>
525std::ostream &
526operator<<(std::ostream & os, const Index<VDimension> & obj)
527{
528 os << '[';
529 for (unsigned int i = 0; i + 1 < VDimension; ++i)
530 {
531 os << obj[i] << ", ";
532 }
533 if constexpr (VDimension >= 1)
534 {
535 os << obj[VDimension - 1];
536 }
537 os << ']';
538 return os;
539}
540
541// ======================= Mirror the access pattern behavior of the std::array class
542// Array comparisons.
543template <unsigned int VDimension>
544inline bool
546{
547 return std::equal(one.begin(), one.end(), two.begin());
548}
549
550template <unsigned int VDimension>
551inline bool
553{
554 return !(one == two);
555}
556
557template <unsigned int VDimension>
558inline bool
559operator<(const Index<VDimension> & one, const Index<VDimension> & two)
560{
561 return std::lexicographical_compare(one.begin(), one.end(), two.begin(), two.end());
562}
563
564template <unsigned int VDimension>
565inline bool
567{
568 return two < one;
569}
570
571template <unsigned int VDimension>
572inline bool
573operator<=(const Index<VDimension> & one, const Index<VDimension> & two)
574{
575 return !(one > two);
576}
577
578template <unsigned int VDimension>
579inline bool
581{
582 return !(one < two);
583}
584
585// Specialized algorithms [6.2.2.2].
586template <unsigned int VDimension>
587inline void
589{
590 std::swap(one.m_InternalArray, two.m_InternalArray);
591}
592
593
595template <typename... T>
596auto
597MakeIndex(const T... values)
598{
599 const auto toValueType = [](const auto value) {
600 static_assert(std::is_integral_v<decltype(value)>, "Each value must have an integral type!");
601 return static_cast<IndexValueType>(value);
602 };
603 return Index<sizeof...(T)>{ { toValueType(values)... } };
604}
605
606} // end namespace itk
607
608#endif
Simulate a standard C array with copy semantics.
TInput TInput TReturn Round(TInput x)
Definition itkMath.h:178
The "itk" namespace contains all Insight Segmentation and Registration Toolkit (ITK) classes....
constexpr TContainer MakeFilled(typename TContainer::const_reference value)
void swap(Array< T > &a, Array< T > &b) noexcept
Definition itkArray.h:247
auto MakeIndex(const T... values)
Definition itkIndex.h:597
bool operator>(const Index< VDimension > &one, const Index< VDimension > &two)
Definition itkIndex.h:566
ITKCommon_EXPORT std::ostream & operator<<(std::ostream &out, typename AnatomicalOrientation::CoordinateEnum value)
long OffsetValueType
Definition itkIntTypes.h:97
bool operator<=(const Index< VDimension > &one, const Index< VDimension > &two)
Definition itkIndex.h:573
bool operator==(const Index< VDimension > &one, const Index< VDimension > &two)
Definition itkIndex.h:545
bool operator!=(const Index< VDimension > &one, const Index< VDimension > &two)
Definition itkIndex.h:552
bool operator<(const Index< VDimension > &one, const Index< VDimension > &two)
Definition itkIndex.h:559
bool operator>=(const Index< VDimension > &one, const Index< VDimension > &two)
Definition itkIndex.h:580
long IndexValueType
Definition itkIntTypes.h:93
Represent a n-dimensional index in a n-dimensional image.
Definition itkIndex.h:69
constexpr const_reference operator[](size_type pos) const
Definition itkIndex.h:437
const Self operator*(const SizeType &vec) const
Definition itkIndex.h:213
const OffsetType operator-(const Self &vec) const
Definition itkIndex.h:198
const Self & operator+=(const OffsetType &offset)
Definition itkIndex.h:163
const value_type * const_iterator
Definition itkIndex.h:330
Size< VDimension > SizeType
Definition itkIndex.h:82
constexpr reference front()
Definition itkIndex.h:457
const Self & operator-=(const SizeType &sz)
Definition itkIndex.h:139
const_reverse_iterator rbegin() const
Definition itkIndex.h:395
static constexpr Self Filled(const IndexValueType value)
Definition itkIndex.h:496
reverse_iterator rend()
Definition itkIndex.h:401
void assign(const value_type &newValue)
Definition itkIndex.h:341
value_type * iterator
Definition itkIndex.h:329
constexpr size_type max_size() const
Definition itkIndex.h:419
reverse_iterator rbegin()
Definition itkIndex.h:389
const_reverse_iterator rend() const
Definition itkIndex.h:407
void ExceptionThrowingBoundsCheck(size_type pos) const
Definition itkIndex.h:504
IndexValueType m_InternalArray[VDimension]
Definition itkIndex.h:289
constexpr bool empty() const
Definition itkIndex.h:425
constexpr size_type size() const
Definition itkIndex.h:413
Index< VDimension > IndexType
Definition itkIndex.h:78
const IndexValueType * GetIndex() const
Definition itkIndex.h:227
constexpr const_iterator begin() const
Definition itkIndex.h:365
const Self operator-(const OffsetType &off) const
Definition itkIndex.h:185
static Self GetBasisIndex(unsigned int dim)
Definition itkIndex.h:516
unsigned int size_type
Definition itkIndex.h:331
itk::IndexValueType IndexValueType
Definition itkIndex.h:79
reference at(size_type pos)
Definition itkIndex.h:443
const Self operator+(const SizeType &sz) const
Definition itkIndex.h:101
std::reverse_iterator< iterator > reverse_iterator
Definition itkIndex.h:333
itk::IndexValueType value_type
Definition itkIndex.h:326
const value_type & const_reference
Definition itkIndex.h:328
std::reverse_iterator< const_iterator > const_reverse_iterator
Definition itkIndex.h:334
constexpr reference operator[](size_type pos)
Definition itkIndex.h:431
constexpr const_iterator end() const
Definition itkIndex.h:383
constexpr reference back()
Definition itkIndex.h:469
constexpr iterator end()
Definition itkIndex.h:377
const Self operator-(const SizeType &sz) const
Definition itkIndex.h:126
void Fill(IndexValueType value)
Definition itkIndex.h:269
Offset< VDimension > OffsetType
Definition itkIndex.h:85
void CopyWithCast(const FixedArray< TCoordinate, VDimension > &point)
Definition itkIndex.h:305
void CopyWithRound(const FixedArray< TCoordinate, VDimension > &point)
Definition itkIndex.h:294
constexpr const_reference back() const
Definition itkIndex.h:475
const IndexValueType * data() const
Definition itkIndex.h:487
void SetElement(unsigned int element, IndexValueType val)
Definition itkIndex.h:249
static constexpr unsigned int Dimension
Definition itkIndex.h:89
const Self & operator+=(const SizeType &sz)
Definition itkIndex.h:114
constexpr iterator begin()
Definition itkIndex.h:359
value_type & reference
Definition itkIndex.h:327
IndexValueType * data()
Definition itkIndex.h:481
const Self operator+(const OffsetType &offset) const
Definition itkIndex.h:150
IndexValueType GetElement(unsigned int element) const
Definition itkIndex.h:261
constexpr const_iterator cend() const
Definition itkIndex.h:371
constexpr const_reference front() const
Definition itkIndex.h:463
itk::OffsetValueType OffsetValueType
Definition itkIndex.h:86
constexpr const_iterator cbegin() const
Definition itkIndex.h:353
static constexpr unsigned int GetIndexDimension()
Definition itkIndex.h:93
void swap(Index &other) noexcept
Definition itkIndex.h:347
void SetIndex(const IndexValueType val[VDimension])
Definition itkIndex.h:237
const Self & operator-=(const OffsetType &offset)
Definition itkIndex.h:174
const_reference at(size_type pos) const
Definition itkIndex.h:450
Represent a n-dimensional offset between two n-dimensional indexes of n-dimensional image.
Definition itkOffset.h:67
OffsetValueType m_InternalArray[VDimension]
Definition itkOffset.h:238
Represent a n-dimensional size (bounds) of a n-dimensional image.
Definition itkSize.h:70
SizeValueType m_InternalArray[VDimension]
Definition itkSize.h:242