image_view.hpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556
  1. //
  2. // Copyright 2005-2007 Adobe Systems Incorporated
  3. //
  4. // Distributed under the Boost Software License, Version 1.0
  5. // See accompanying file LICENSE_1_0.txt or copy at
  6. // http://www.boost.org/LICENSE_1_0.txt
  7. //
  8. #ifndef BOOST_GIL_CONCEPTS_IMAGE_VIEW_HPP
  9. #define BOOST_GIL_CONCEPTS_IMAGE_VIEW_HPP
  10. #include <boost/gil/concepts/basic.hpp>
  11. #include <boost/gil/concepts/concept_check.hpp>
  12. #include <boost/gil/concepts/fwd.hpp>
  13. #include <boost/gil/concepts/pixel.hpp>
  14. #include <boost/gil/concepts/pixel_dereference.hpp>
  15. #include <boost/gil/concepts/pixel_iterator.hpp>
  16. #include <boost/gil/concepts/pixel_locator.hpp>
  17. #include <boost/gil/concepts/point.hpp>
  18. #include <boost/gil/concepts/detail/utility.hpp>
  19. #include <cstddef>
  20. #include <iterator>
  21. #if defined(BOOST_CLANG)
  22. #pragma clang diagnostic push
  23. #pragma clang diagnostic ignored "-Wunused-local-typedefs"
  24. #endif
  25. #if defined(BOOST_GCC) && (BOOST_GCC >= 40600)
  26. #pragma GCC diagnostic push
  27. #pragma GCC diagnostic ignored "-Wunused-local-typedefs"
  28. #pragma GCC diagnostic ignored "-Wunused-but-set-variable"
  29. #endif
  30. namespace boost { namespace gil {
  31. /// \defgroup ImageViewNDConcept ImageViewNDLocatorConcept
  32. /// \ingroup ImageViewConcept
  33. /// \brief N-dimensional range
  34. /// \defgroup ImageView2DConcept ImageView2DConcept
  35. /// \ingroup ImageViewConcept
  36. /// \brief 2-dimensional range
  37. /// \defgroup PixelImageViewConcept ImageViewConcept
  38. /// \ingroup ImageViewConcept
  39. /// \brief 2-dimensional range over pixel data
  40. /// \ingroup ImageViewNDConcept
  41. /// \brief N-dimensional view over immutable values
  42. ///
  43. /// \code
  44. /// concept RandomAccessNDImageViewConcept<Regular View>
  45. /// {
  46. /// typename value_type;
  47. /// typename reference; // result of dereferencing
  48. /// typename difference_type; // result of operator-(iterator,iterator) (1-dimensional!)
  49. /// typename const_t; where RandomAccessNDImageViewConcept<View>; // same as View, but over immutable values
  50. /// typename point_t; where PointNDConcept<point_t>; // N-dimensional point
  51. /// typename locator; where RandomAccessNDLocatorConcept<locator>; // N-dimensional locator.
  52. /// typename iterator; where RandomAccessTraversalConcept<iterator>; // 1-dimensional iterator over all values
  53. /// typename reverse_iterator; where RandomAccessTraversalConcept<reverse_iterator>;
  54. /// typename size_type; // the return value of size()
  55. ///
  56. /// // Equivalent to RandomAccessNDLocatorConcept::axis
  57. /// template <size_t D> struct axis {
  58. /// typename coord_t = point_t::axis<D>::coord_t;
  59. /// typename iterator; where RandomAccessTraversalConcept<iterator>; // iterator along D-th axis.
  60. /// where SameType<coord_t, iterator::difference_type>;
  61. /// where SameType<iterator::value_type,value_type>;
  62. /// };
  63. ///
  64. /// // Defines the type of a view similar to this type, except it invokes Deref upon dereferencing
  65. /// template <PixelDereferenceAdaptorConcept Deref> struct add_deref {
  66. /// typename type; where RandomAccessNDImageViewConcept<type>;
  67. /// static type make(const View& v, const Deref& deref);
  68. /// };
  69. ///
  70. /// static const size_t num_dimensions = point_t::num_dimensions;
  71. ///
  72. /// // Create from a locator at the top-left corner and dimensions
  73. /// View::View(const locator&, const point_type&);
  74. ///
  75. /// size_type View::size() const; // total number of elements
  76. /// reference operator[](View, const difference_type&) const; // 1-dimensional reference
  77. /// iterator View::begin() const;
  78. /// iterator View::end() const;
  79. /// reverse_iterator View::rbegin() const;
  80. /// reverse_iterator View::rend() const;
  81. /// iterator View::at(const point_t&);
  82. /// point_t View::dimensions() const; // number of elements along each dimension
  83. /// bool View::is_1d_traversable() const; // can an iterator over the first dimension visit each value? I.e. are there gaps between values?
  84. ///
  85. /// // iterator along a given dimension starting at a given point
  86. /// template <size_t D> View::axis<D>::iterator View::axis_iterator(const point_t&) const;
  87. ///
  88. /// reference operator()(View,const point_t&) const;
  89. /// };
  90. /// \endcode
  91. template <typename View>
  92. struct RandomAccessNDImageViewConcept
  93. {
  94. void constraints()
  95. {
  96. gil_function_requires<Regular<View>>();
  97. using value_type = typename View::value_type;
  98. using reference = typename View::reference; // result of dereferencing
  99. using pointer = typename View::pointer;
  100. using difference_type = typename View::difference_type; // result of operator-(1d_iterator,1d_iterator)
  101. using const_t = typename View::const_t; // same as this type, but over const values
  102. using point_t = typename View::point_t; // N-dimensional point
  103. using locator = typename View::locator; // N-dimensional locator
  104. using iterator = typename View::iterator;
  105. using const_iterator = typename View::const_iterator;
  106. using reverse_iterator = typename View::reverse_iterator;
  107. using size_type = typename View::size_type;
  108. static const std::size_t N=View::num_dimensions;
  109. gil_function_requires<RandomAccessNDLocatorConcept<locator>>();
  110. gil_function_requires<boost_concepts::RandomAccessTraversalConcept<iterator>>();
  111. gil_function_requires<boost_concepts::RandomAccessTraversalConcept<reverse_iterator>>();
  112. using first_it_type = typename View::template axis<0>::iterator;
  113. using last_it_type = typename View::template axis<N-1>::iterator;
  114. gil_function_requires<boost_concepts::RandomAccessTraversalConcept<first_it_type>>();
  115. gil_function_requires<boost_concepts::RandomAccessTraversalConcept<last_it_type>>();
  116. // static_assert(typename std::iterator_traits<first_it_type>::difference_type, typename point_t::template axis<0>::coord_t>::value, "");
  117. // static_assert(typename std::iterator_traits<last_it_type>::difference_type, typename point_t::template axis<N-1>::coord_t>::value, "");
  118. // point_t must be an N-dimensional point, each dimension of which must have the same type as difference_type of the corresponding iterator
  119. gil_function_requires<PointNDConcept<point_t>>();
  120. static_assert(point_t::num_dimensions == N, "");
  121. static_assert(is_same
  122. <
  123. typename std::iterator_traits<first_it_type>::difference_type,
  124. typename point_t::template axis<0>::coord_t
  125. >::value, "");
  126. static_assert(is_same
  127. <
  128. typename std::iterator_traits<last_it_type>::difference_type,
  129. typename point_t::template axis<N-1>::coord_t
  130. >::value, "");
  131. point_t p;
  132. locator lc;
  133. iterator it;
  134. reverse_iterator rit;
  135. difference_type d; detail::initialize_it(d); ignore_unused_variable_warning(d);
  136. View(p,lc); // view must be constructible from a locator and a point
  137. p = view.dimensions();
  138. lc = view.pixels();
  139. size_type sz = view.size(); ignore_unused_variable_warning(sz);
  140. bool is_contiguous = view.is_1d_traversable();
  141. ignore_unused_variable_warning(is_contiguous);
  142. it = view.begin();
  143. it = view.end();
  144. rit = view.rbegin();
  145. rit = view.rend();
  146. reference r1 = view[d]; ignore_unused_variable_warning(r1); // 1D access
  147. reference r2 = view(p); ignore_unused_variable_warning(r2); // 2D access
  148. // get 1-D iterator of any dimension at a given pixel location
  149. first_it_type fi = view.template axis_iterator<0>(p);
  150. ignore_unused_variable_warning(fi);
  151. last_it_type li = view.template axis_iterator<N-1>(p);
  152. ignore_unused_variable_warning(li);
  153. using deref_t = PixelDereferenceAdaptorArchetype<typename View::value_type>;
  154. using dtype = typename View::template add_deref<deref_t>::type;
  155. }
  156. View view;
  157. };
  158. /// \ingroup ImageView2DConcept
  159. /// \brief 2-dimensional view over immutable values
  160. ///
  161. /// \code
  162. /// concept RandomAccess2DImageViewConcept<RandomAccessNDImageViewConcept View> {
  163. /// where num_dimensions==2;
  164. ///
  165. /// typename x_iterator = axis<0>::iterator;
  166. /// typename y_iterator = axis<1>::iterator;
  167. /// typename x_coord_t = axis<0>::coord_t;
  168. /// typename y_coord_t = axis<1>::coord_t;
  169. /// typename xy_locator = locator;
  170. ///
  171. /// x_coord_t View::width() const;
  172. /// y_coord_t View::height() const;
  173. ///
  174. /// // X-navigation
  175. /// x_iterator View::x_at(const point_t&) const;
  176. /// x_iterator View::row_begin(y_coord_t) const;
  177. /// x_iterator View::row_end (y_coord_t) const;
  178. ///
  179. /// // Y-navigation
  180. /// y_iterator View::y_at(const point_t&) const;
  181. /// y_iterator View::col_begin(x_coord_t) const;
  182. /// y_iterator View::col_end (x_coord_t) const;
  183. ///
  184. /// // navigating in 2D
  185. /// xy_locator View::xy_at(const point_t&) const;
  186. ///
  187. /// // (x,y) versions of all methods taking point_t
  188. /// View::View(x_coord_t,y_coord_t,const locator&);
  189. /// iterator View::at(x_coord_t,y_coord_t) const;
  190. /// reference operator()(View,x_coord_t,y_coord_t) const;
  191. /// xy_locator View::xy_at(x_coord_t,y_coord_t) const;
  192. /// x_iterator View::x_at(x_coord_t,y_coord_t) const;
  193. /// y_iterator View::y_at(x_coord_t,y_coord_t) const;
  194. /// };
  195. /// \endcode
  196. template <typename View>
  197. struct RandomAccess2DImageViewConcept
  198. {
  199. void constraints()
  200. {
  201. gil_function_requires<RandomAccessNDImageViewConcept<View>>();
  202. static_assert(View::num_dimensions == 2, "");
  203. // TODO: This executes the requirements for RandomAccessNDLocatorConcept again. Fix it to improve compile time
  204. gil_function_requires<RandomAccess2DLocatorConcept<typename View::locator>>();
  205. using dynamic_x_step_t = typename dynamic_x_step_type<View>::type;
  206. using dynamic_y_step_t = typename dynamic_y_step_type<View>::type;
  207. using transposed_t = typename transposed_type<View>::type;
  208. using x_iterator = typename View::x_iterator;
  209. using y_iterator = typename View::y_iterator;
  210. using x_coord_t = typename View::x_coord_t;
  211. using y_coord_t = typename View::y_coord_t;
  212. using xy_locator = typename View::xy_locator;
  213. x_coord_t xd = 0; ignore_unused_variable_warning(xd);
  214. y_coord_t yd = 0; ignore_unused_variable_warning(yd);
  215. x_iterator xit;
  216. y_iterator yit;
  217. typename View::point_t d;
  218. View(xd, yd, xy_locator()); // constructible with width, height, 2d_locator
  219. xy_locator lc = view.xy_at(xd, yd);
  220. lc = view.xy_at(d);
  221. typename View::reference r = view(xd, yd);
  222. ignore_unused_variable_warning(r);
  223. xd = view.width();
  224. yd = view.height();
  225. xit = view.x_at(d);
  226. xit = view.x_at(xd,yd);
  227. xit = view.row_begin(xd);
  228. xit = view.row_end(xd);
  229. yit = view.y_at(d);
  230. yit = view.y_at(xd,yd);
  231. yit = view.col_begin(xd);
  232. yit = view.col_end(xd);
  233. }
  234. View view;
  235. };
  236. /// \brief GIL view as Collection.
  237. ///
  238. /// \see https://www.boost.org/libs/utility/Collection.html
  239. ///
  240. template <typename View>
  241. struct CollectionImageViewConcept
  242. {
  243. void constraints()
  244. {
  245. using value_type = typename View::value_type;
  246. using iterator = typename View::iterator;
  247. using const_iterator = typename View::const_iterator;
  248. using reference = typename View::reference;
  249. using const_reference = typename View::const_reference;
  250. using pointer = typename View::pointer;
  251. using difference_type = typename View::difference_type;
  252. using size_type= typename View::size_type;
  253. iterator i;
  254. i = view1.begin();
  255. i = view2.end();
  256. const_iterator ci;
  257. ci = view1.begin();
  258. ci = view2.end();
  259. size_type s;
  260. s = view1.size();
  261. s = view2.size();
  262. ignore_unused_variable_warning(s);
  263. view1.empty();
  264. view1.swap(view2);
  265. }
  266. View view1;
  267. View view2;
  268. };
  269. /// \brief GIL view as ForwardCollection.
  270. ///
  271. /// \see https://www.boost.org/libs/utility/Collection.html
  272. ///
  273. template <typename View>
  274. struct ForwardCollectionImageViewConcept
  275. {
  276. void constraints()
  277. {
  278. gil_function_requires<CollectionImageViewConcept<View>>();
  279. using reference = typename View::reference;
  280. using const_reference = typename View::const_reference;
  281. reference r = view.front();
  282. ignore_unused_variable_warning(r);
  283. const_reference cr = view.front();
  284. ignore_unused_variable_warning(cr);
  285. }
  286. View view;
  287. };
  288. /// \brief GIL view as ReversibleCollection.
  289. ///
  290. /// \see https://www.boost.org/libs/utility/Collection.html
  291. ///
  292. template <typename View>
  293. struct ReversibleCollectionImageViewConcept
  294. {
  295. void constraints()
  296. {
  297. gil_function_requires<CollectionImageViewConcept<View>>();
  298. using reverse_iterator = typename View::reverse_iterator;
  299. using reference = typename View::reference;
  300. using const_reference = typename View::const_reference;
  301. reverse_iterator i;
  302. i = view.rbegin();
  303. i = view.rend();
  304. reference r = view.back();
  305. ignore_unused_variable_warning(r);
  306. const_reference cr = view.back();
  307. ignore_unused_variable_warning(cr);
  308. }
  309. View view;
  310. };
  311. /// \ingroup PixelImageViewConcept
  312. /// \brief GIL's 2-dimensional view over immutable GIL pixels
  313. /// \code
  314. /// concept ImageViewConcept<RandomAccess2DImageViewConcept View>
  315. /// {
  316. /// where PixelValueConcept<value_type>;
  317. /// where PixelIteratorConcept<x_iterator>;
  318. /// where PixelIteratorConcept<y_iterator>;
  319. /// where x_coord_t == y_coord_t;
  320. ///
  321. /// typename coord_t = x_coord_t;
  322. ///
  323. /// std::size_t View::num_channels() const;
  324. /// };
  325. /// \endcode
  326. template <typename View>
  327. struct ImageViewConcept
  328. {
  329. void constraints()
  330. {
  331. gil_function_requires<RandomAccess2DImageViewConcept<View>>();
  332. // TODO: This executes the requirements for RandomAccess2DLocatorConcept again. Fix it to improve compile time
  333. gil_function_requires<PixelLocatorConcept<typename View::xy_locator>>();
  334. static_assert(is_same<typename View::x_coord_t, typename View::y_coord_t>::value, "");
  335. using coord_t = typename View::coord_t; // 1D difference type (same for all dimensions)
  336. std::size_t num_chan = view.num_channels(); ignore_unused_variable_warning(num_chan);
  337. }
  338. View view;
  339. };
  340. namespace detail {
  341. /// \tparam View Models RandomAccessNDImageViewConcept
  342. template <typename View>
  343. struct RandomAccessNDImageViewIsMutableConcept
  344. {
  345. void constraints()
  346. {
  347. gil_function_requires<detail::RandomAccessNDLocatorIsMutableConcept<typename View::locator>>();
  348. gil_function_requires<detail::RandomAccessIteratorIsMutableConcept<typename View::iterator>>();
  349. gil_function_requires<detail::RandomAccessIteratorIsMutableConcept
  350. <
  351. typename View::reverse_iterator
  352. >>();
  353. gil_function_requires<detail::RandomAccessIteratorIsMutableConcept
  354. <
  355. typename View::template axis<0>::iterator
  356. >>();
  357. gil_function_requires<detail::RandomAccessIteratorIsMutableConcept
  358. <
  359. typename View::template axis<View::num_dimensions - 1>::iterator
  360. >>();
  361. typename View::difference_type diff;
  362. initialize_it(diff);
  363. ignore_unused_variable_warning(diff);
  364. typename View::point_t pt;
  365. typename View::value_type v;
  366. initialize_it(v);
  367. view[diff] = v;
  368. view(pt) = v;
  369. }
  370. View view;
  371. };
  372. /// \tparam View Models RandomAccessNDImageViewConcept
  373. template <typename View>
  374. struct RandomAccess2DImageViewIsMutableConcept
  375. {
  376. void constraints()
  377. {
  378. gil_function_requires<detail::RandomAccessNDImageViewIsMutableConcept<View>>();
  379. typename View::x_coord_t xd = 0; ignore_unused_variable_warning(xd);
  380. typename View::y_coord_t yd = 0; ignore_unused_variable_warning(yd);
  381. typename View::value_type v; initialize_it(v);
  382. view(xd, yd) = v;
  383. }
  384. View view;
  385. };
  386. /// \tparam View Models ImageViewConcept
  387. template <typename View>
  388. struct PixelImageViewIsMutableConcept
  389. {
  390. void constraints()
  391. {
  392. gil_function_requires<detail::RandomAccess2DImageViewIsMutableConcept<View>>();
  393. }
  394. };
  395. } // namespace detail
  396. /// \ingroup ImageViewNDConcept
  397. /// \brief N-dimensional view over mutable values
  398. ///
  399. /// \code
  400. /// concept MutableRandomAccessNDImageViewConcept<RandomAccessNDImageViewConcept View>
  401. /// {
  402. /// where Mutable<reference>;
  403. /// };
  404. /// \endcode
  405. template <typename View>
  406. struct MutableRandomAccessNDImageViewConcept
  407. {
  408. void constraints()
  409. {
  410. gil_function_requires<RandomAccessNDImageViewConcept<View>>();
  411. gil_function_requires<detail::RandomAccessNDImageViewIsMutableConcept<View>>();
  412. }
  413. };
  414. /// \ingroup ImageView2DConcept
  415. /// \brief 2-dimensional view over mutable values
  416. ///
  417. /// \code
  418. /// concept MutableRandomAccess2DImageViewConcept<RandomAccess2DImageViewConcept View>
  419. /// : MutableRandomAccessNDImageViewConcept<View> {};
  420. /// \endcode
  421. template <typename View>
  422. struct MutableRandomAccess2DImageViewConcept
  423. {
  424. void constraints()
  425. {
  426. gil_function_requires<RandomAccess2DImageViewConcept<View>>();
  427. gil_function_requires<detail::RandomAccess2DImageViewIsMutableConcept<View>>();
  428. }
  429. };
  430. /// \ingroup PixelImageViewConcept
  431. /// \brief GIL's 2-dimensional view over mutable GIL pixels
  432. ///
  433. /// \code
  434. /// concept MutableImageViewConcept<ImageViewConcept View>
  435. /// : MutableRandomAccess2DImageViewConcept<View> {};
  436. /// \endcode
  437. template <typename View>
  438. struct MutableImageViewConcept
  439. {
  440. void constraints()
  441. {
  442. gil_function_requires<ImageViewConcept<View>>();
  443. gil_function_requires<detail::PixelImageViewIsMutableConcept<View>>();
  444. }
  445. };
  446. /// \brief Returns whether two views are compatible
  447. ///
  448. /// Views are compatible if their pixels are compatible.
  449. /// Compatible views can be assigned and copy constructed from one another.
  450. ///
  451. /// \tparam V1 Models ImageViewConcept
  452. /// \tparam V2 Models ImageViewConcept
  453. ///
  454. template <typename V1, typename V2>
  455. struct views_are_compatible
  456. : pixels_are_compatible<typename V1::value_type, typename V2::value_type>
  457. {
  458. };
  459. /// \ingroup ImageViewConcept
  460. /// \brief Views are compatible if they have the same color spaces and compatible channel values.
  461. ///
  462. /// Constness and layout are not important for compatibility.
  463. ///
  464. /// \code
  465. /// concept ViewsCompatibleConcept<ImageViewConcept V1, ImageViewConcept V2>
  466. /// {
  467. /// where PixelsCompatibleConcept<V1::value_type, P2::value_type>;
  468. /// };
  469. /// \endcode
  470. template <typename V1, typename V2>
  471. struct ViewsCompatibleConcept
  472. {
  473. void constraints()
  474. {
  475. static_assert(views_are_compatible<V1, V2>::value, "");
  476. }
  477. };
  478. }} // namespace boost::gil
  479. #if defined(BOOST_CLANG)
  480. #pragma clang diagnostic pop
  481. #endif
  482. #if defined(BOOST_GCC) && (BOOST_GCC >= 40600)
  483. #pragma GCC diagnostic pop
  484. #endif
  485. #endif