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