locator.hpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362
  1. /*
  2. Copyright 2005-2007 Adobe Systems Incorporated
  3. Use, modification and distribution are subject to the Boost Software License,
  4. Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  5. http://www.boost.org/LICENSE_1_0.txt).
  6. See http://opensource.adobe.com/gil for most recent version including documentation.
  7. */
  8. /*************************************************************************************************/
  9. #ifndef GIL_LOCATOR_H
  10. #define GIL_LOCATOR_H
  11. ////////////////////////////////////////////////////////////////////////////////////////
  12. /// \file
  13. /// \brief pixel 2D locator
  14. /// \author Lubomir Bourdev and Hailin Jin \n
  15. /// Adobe Systems Incorporated
  16. /// \date 2005-2007 \n September 20, 2006
  17. ///
  18. ////////////////////////////////////////////////////////////////////////////////////////
  19. #include <cstddef>
  20. #include <cassert>
  21. #include "pixel_iterator.hpp"
  22. ////////////////////////////////////////////////////////////////////////////////////////
  23. /// Pixel 2D LOCATOR
  24. ////////////////////////////////////////////////////////////////////////////////////////
  25. namespace boost { namespace gil {
  26. //forward declarations
  27. template <typename P> ptrdiff_t memunit_step(const P*);
  28. template <typename P> P* memunit_advanced(const P* p, ptrdiff_t diff);
  29. template <typename P> P& memunit_advanced_ref(P* p, ptrdiff_t diff);
  30. template <typename Iterator, typename D> struct iterator_add_deref;
  31. template <typename T> class point2;
  32. namespace detail {
  33. // helper class specialized for each axis of pixel_2d_locator
  34. template <std::size_t D, typename Loc> class locator_axis;
  35. }
  36. template <typename T> struct dynamic_x_step_type;
  37. template <typename T> struct dynamic_y_step_type;
  38. template <typename T> struct channel_type;
  39. template <typename T> struct color_space_type;
  40. template <typename T> struct channel_mapping_type;
  41. template <typename T> struct is_planar;
  42. template <typename T> struct num_channels;
  43. // The type of a locator or a view that has X and Y swapped. By default it is the same
  44. template <typename T> struct transposed_type {
  45. typedef T type;
  46. };
  47. /// \class pixel_2d_locator_base
  48. /// \brief base class for models of PixelLocatorConcept
  49. /// \ingroup PixelLocatorModel PixelBasedModel
  50. ///
  51. /// Pixel locator is similar to a pixel iterator, but allows for 2D navigation of pixels within an image view.
  52. /// It has a 2D difference_type and supports random access operations like:
  53. /// \code
  54. /// difference_type offset2(2,3);
  55. /// locator+=offset2;
  56. /// locator[offset2]=my_pixel;
  57. /// \endcode
  58. ///
  59. /// In addition, each coordinate acts as a random-access iterator that can be modified separately:
  60. /// "++locator.x()" or "locator.y()+=10" thereby moving the locator horizontally or vertically.
  61. ///
  62. /// It is called a locator because it doesn't implement the complete interface of a random access iterator.
  63. /// For example, increment and decrement operations don't make sense (no way to specify dimension).
  64. /// Also 2D difference between two locators cannot be computed without knowledge of the X position within the image.
  65. ///
  66. /// This base class provides most of the methods and typedefs needed to create a model of a locator. GIL provides two
  67. /// locator models as subclasses of \p pixel_2d_locator_base. A memory-based locator, \p memory_based_2d_locator and a virtual
  68. /// locator, \p virtual_2d_locator.
  69. /// The minimum functionality a subclass must provide is this:
  70. /// \code
  71. /// class my_locator : public pixel_2d_locator_base<my_locator, ..., ...> { // supply the types for x-iterator and y-iterator
  72. /// typedef ... const_t; // read-only locator
  73. ///
  74. /// template <typename Deref> struct add_deref {
  75. /// typedef ... type; // locator that invokes the Deref dereference object upon pixel access
  76. /// static type make(const my_locator& loc, const Deref& d);
  77. /// };
  78. ///
  79. /// my_locator();
  80. /// my_locator(const my_locator& pl);
  81. ///
  82. /// // constructors with dynamic step in y (and x). Only valid for locators with dynamic steps
  83. /// my_locator(const my_locator& loc, coord_t y_step);
  84. /// my_locator(const my_locator& loc, coord_t x_step, coord_t y_step, bool transpose);
  85. ///
  86. /// bool operator==(const my_locator& p) const;
  87. ///
  88. /// // return _references_ to horizontal/vertical iterators. Advancing them moves this locator
  89. /// x_iterator& x();
  90. /// y_iterator& y();
  91. /// x_iterator const& x() const;
  92. /// y_iterator const& y() const;
  93. ///
  94. /// // return the vertical distance to another locator. Some models need the horizontal distance to compute it
  95. /// y_coord_t y_distance_to(const my_locator& loc2, x_coord_t xDiff) const;
  96. ///
  97. /// // return true iff incrementing an x-iterator located at the last column will position it at the first
  98. /// // column of the next row. Some models need the image width to determine that.
  99. /// bool is_1d_traversable(x_coord_t width) const;
  100. /// };
  101. /// \endcode
  102. ///
  103. /// Models may choose to override some of the functions in the base class with more efficient versions.
  104. ///
  105. template <typename Loc, typename XIterator, typename YIterator> // The concrete subclass, the X-iterator and the Y-iterator
  106. class pixel_2d_locator_base {
  107. public:
  108. typedef XIterator x_iterator;
  109. typedef YIterator y_iterator;
  110. // typedefs required by ConstRandomAccessNDLocatorConcept
  111. static const std::size_t num_dimensions=2;
  112. typedef typename std::iterator_traits<x_iterator>::value_type value_type;
  113. typedef typename std::iterator_traits<x_iterator>::reference reference; // result of dereferencing
  114. typedef typename std::iterator_traits<x_iterator>::difference_type coord_t; // 1D difference type (same for all dimensions)
  115. typedef point2<coord_t> difference_type; // result of operator-(locator,locator)
  116. typedef difference_type point_t;
  117. template <std::size_t D> struct axis {
  118. typedef typename detail::locator_axis<D,Loc>::coord_t coord_t;
  119. typedef typename detail::locator_axis<D,Loc>::iterator iterator;
  120. };
  121. // typedefs required by ConstRandomAccess2DLocatorConcept
  122. typedef typename point_t::template axis<0>::coord_t x_coord_t;
  123. typedef typename point_t::template axis<1>::coord_t y_coord_t;
  124. bool operator!=(const Loc& p) const { return !(concrete()==p); }
  125. x_iterator x_at(x_coord_t dx, y_coord_t dy) const { Loc tmp=concrete(); tmp+=point_t(dx,dy); return tmp.x(); }
  126. x_iterator x_at(const difference_type& d) const { Loc tmp=concrete(); tmp+=d; return tmp.x(); }
  127. y_iterator y_at(x_coord_t dx, y_coord_t dy) const { Loc tmp=concrete(); tmp+=point_t(dx,dy); return tmp.y(); }
  128. y_iterator y_at(const difference_type& d) const { Loc tmp=concrete(); tmp+=d; return tmp.y(); }
  129. Loc xy_at(x_coord_t dx, y_coord_t dy) const { Loc tmp=concrete(); tmp+=point_t(dx,dy); return tmp; }
  130. Loc xy_at(const difference_type& d) const { Loc tmp=concrete(); tmp+=d; return tmp; }
  131. template <std::size_t D> typename axis<D>::iterator& axis_iterator() { return detail::locator_axis<D,Loc>()(concrete()); }
  132. template <std::size_t D> typename axis<D>::iterator const& axis_iterator() const { return detail::locator_axis<D,Loc>()(concrete()); }
  133. template <std::size_t D> typename axis<D>::iterator axis_iterator(const point_t& p) const { return detail::locator_axis<D,Loc>()(concrete(),p); }
  134. reference operator()(x_coord_t dx, y_coord_t dy) const { return *x_at(dx,dy); }
  135. reference operator[](const difference_type& d) const { return *x_at(d.x,d.y); }
  136. reference operator*() const { return *concrete().x(); }
  137. Loc& operator+=(const difference_type& d) { concrete().x()+=d.x; concrete().y()+=d.y; return concrete(); }
  138. Loc& operator-=(const difference_type& d) { concrete().x()-=d.x; concrete().y()-=d.y; return concrete(); }
  139. Loc operator+(const difference_type& d) const { return xy_at(d); }
  140. Loc operator-(const difference_type& d) const { return xy_at(-d); }
  141. // Some locators can cache 2D coordinates for faster subsequent access. By default there is no caching
  142. typedef difference_type cached_location_t;
  143. cached_location_t cache_location(const difference_type& d) const { return d; }
  144. cached_location_t cache_location(x_coord_t dx, y_coord_t dy)const { return difference_type(dx,dy); }
  145. private:
  146. Loc& concrete() { return (Loc&)*this; }
  147. const Loc& concrete() const { return (const Loc&)*this; }
  148. template <typename X> friend class pixel_2d_locator;
  149. };
  150. // helper classes for each axis of pixel_2d_locator_base
  151. namespace detail {
  152. template <typename Loc>
  153. class locator_axis<0,Loc> {
  154. typedef typename Loc::point_t point_t;
  155. public:
  156. typedef typename point_t::template axis<0>::coord_t coord_t;
  157. typedef typename Loc::x_iterator iterator;
  158. inline iterator& operator()( Loc& loc) const { return loc.x(); }
  159. inline iterator const& operator()(const Loc& loc) const { return loc.x(); }
  160. inline iterator operator()( Loc& loc, const point_t& d) const { return loc.x_at(d); }
  161. inline iterator operator()(const Loc& loc, const point_t& d) const { return loc.x_at(d); }
  162. };
  163. template <typename Loc>
  164. class locator_axis<1,Loc> {
  165. typedef typename Loc::point_t point_t;
  166. public:
  167. typedef typename point_t::template axis<1>::coord_t coord_t;
  168. typedef typename Loc::y_iterator iterator;
  169. inline iterator& operator()( Loc& loc) const { return loc.y(); }
  170. inline iterator const& operator()(const Loc& loc) const { return loc.y(); }
  171. inline iterator operator()( Loc& loc, const point_t& d) const { return loc.y_at(d); }
  172. inline iterator operator()(const Loc& loc, const point_t& d) const { return loc.y_at(d); }
  173. };
  174. }
  175. template <typename Loc, typename XIt, typename YIt>
  176. struct channel_type<pixel_2d_locator_base<Loc,XIt,YIt> > : public channel_type<XIt> {};
  177. template <typename Loc, typename XIt, typename YIt>
  178. struct color_space_type<pixel_2d_locator_base<Loc,XIt,YIt> > : public color_space_type<XIt> {};
  179. template <typename Loc, typename XIt, typename YIt>
  180. struct channel_mapping_type<pixel_2d_locator_base<Loc,XIt,YIt> > : public channel_mapping_type<XIt> {};
  181. template <typename Loc, typename XIt, typename YIt>
  182. struct is_planar<pixel_2d_locator_base<Loc,XIt,YIt> > : public is_planar<XIt> {};
  183. /// \class memory_based_2d_locator
  184. /// \brief Memory-based pixel locator. Models: PixelLocatorConcept,HasDynamicXStepTypeConcept,HasDynamicYStepTypeConcept,HasTransposedTypeConcept
  185. /// \ingroup PixelLocatorModel PixelBasedModel
  186. ///
  187. /// The class takes a step iterator as a parameter. The step iterator provides navigation along the vertical axis
  188. /// while its base iterator provides horizontal navigation.
  189. ///
  190. /// Each instantiation is optimal in terms of size and efficiency.
  191. /// For example, xy locator over interleaved rgb image results in a step iterator consisting of
  192. /// one std::ptrdiff_t for the row size and one native pointer (8 bytes total). ++locator.x() resolves to pointer
  193. /// increment. At the other extreme, a 2D navigation of the even pixels of a planar CMYK image results in a step
  194. /// iterator consisting of one std::ptrdiff_t for the doubled row size, and one step iterator consisting of
  195. /// one std::ptrdiff_t for the horizontal step of two and a CMYK planar_pixel_iterator consisting of 4 pointers (24 bytes).
  196. /// In this case ++locator.x() results in four native pointer additions.
  197. ///
  198. /// Note also that \p memory_based_2d_locator does not require that its element type be a pixel. It could be
  199. /// instantiated with an iterator whose \p value_type models only \p Regular. In this case the locator
  200. /// models the weaker RandomAccess2DLocatorConcept, and does not model PixelBasedConcept.
  201. /// Many generic algorithms don't require the elements to be pixels.
  202. ////////////////////////////////////////////////////////////////////////////////////////
  203. template <typename StepIterator>
  204. class memory_based_2d_locator : public pixel_2d_locator_base<memory_based_2d_locator<StepIterator>, typename iterator_adaptor_get_base<StepIterator>::type, StepIterator> {
  205. typedef memory_based_2d_locator<StepIterator> this_t;
  206. GIL_CLASS_REQUIRE(StepIterator, boost::gil, StepIteratorConcept)
  207. public:
  208. typedef pixel_2d_locator_base<memory_based_2d_locator<StepIterator>, typename iterator_adaptor_get_base<StepIterator>::type, StepIterator> parent_t;
  209. typedef memory_based_2d_locator<typename const_iterator_type<StepIterator>::type> const_t; // same as this type, but over const values
  210. typedef typename parent_t::coord_t coord_t;
  211. typedef typename parent_t::x_coord_t x_coord_t;
  212. typedef typename parent_t::y_coord_t y_coord_t;
  213. typedef typename parent_t::x_iterator x_iterator;
  214. typedef typename parent_t::y_iterator y_iterator;
  215. typedef typename parent_t::difference_type difference_type;
  216. typedef typename parent_t::reference reference;
  217. template <typename Deref> struct add_deref {
  218. typedef memory_based_2d_locator<typename iterator_add_deref<StepIterator,Deref>::type> type;
  219. static type make(const memory_based_2d_locator<StepIterator>& loc, const Deref& nderef) {
  220. return type(iterator_add_deref<StepIterator,Deref>::make(loc.y(),nderef));
  221. }
  222. };
  223. memory_based_2d_locator() {}
  224. memory_based_2d_locator(const StepIterator& yit) : _p(yit) {}
  225. template <typename SI> memory_based_2d_locator(const memory_based_2d_locator<SI>& loc, coord_t y_step) : _p(loc.x(), loc.row_size()*y_step) {}
  226. template <typename SI> memory_based_2d_locator(const memory_based_2d_locator<SI>& loc, coord_t x_step, coord_t y_step, bool transpose=false)
  227. : _p(make_step_iterator(loc.x(),(transpose ? loc.row_size() : loc.pixel_size())*x_step),
  228. (transpose ? loc.pixel_size() : loc.row_size())*y_step ) {}
  229. memory_based_2d_locator(x_iterator xit, std::ptrdiff_t row_bytes) : _p(xit,row_bytes) {}
  230. template <typename X> memory_based_2d_locator(const memory_based_2d_locator<X>& pl) : _p(pl._p) {}
  231. memory_based_2d_locator(const memory_based_2d_locator& pl) : _p(pl._p) {}
  232. bool operator==(const this_t& p) const { return _p==p._p; }
  233. x_iterator const& x() const { return _p.base(); }
  234. y_iterator const& y() const { return _p; }
  235. x_iterator& x() { return _p.base(); }
  236. y_iterator& y() { return _p; }
  237. // These are faster versions of functions already provided in the superclass
  238. x_iterator x_at (x_coord_t dx, y_coord_t dy) const { return memunit_advanced(x(), offset(dx,dy)); }
  239. x_iterator x_at (const difference_type& d) const { return memunit_advanced(x(), offset(d.x,d.y)); }
  240. this_t xy_at (x_coord_t dx, y_coord_t dy) const { return this_t(x_at( dx , dy ), row_size()); }
  241. this_t xy_at (const difference_type& d) const { return this_t(x_at( d.x, d.y), row_size()); }
  242. reference operator()(x_coord_t dx, y_coord_t dy) const { return memunit_advanced_ref(x(),offset(dx,dy)); }
  243. reference operator[](const difference_type& d) const { return memunit_advanced_ref(x(),offset(d.x,d.y)); }
  244. this_t& operator+=(const difference_type& d) { memunit_advance(x(),offset(d.x,d.y)); return *this; }
  245. this_t& operator-=(const difference_type& d) { memunit_advance(x(),offset(-d.x,-d.y)); return *this; }
  246. // Memory-based locators can have 1D caching of 2D relative coordinates
  247. typedef std::ptrdiff_t cached_location_t; // type used to store relative location (to allow for more efficient repeated access)
  248. cached_location_t cache_location(const difference_type& d) const { return offset(d.x,d.y); }
  249. cached_location_t cache_location(x_coord_t dx, y_coord_t dy)const { return offset(dx,dy); }
  250. reference operator[](const cached_location_t& loc) const { return memunit_advanced_ref(x(),loc); }
  251. // Only make sense for memory-based locators
  252. std::ptrdiff_t row_size() const { return memunit_step(y()); } // distance in mem units (bytes or bits) between adjacent rows
  253. std::ptrdiff_t pixel_size() const { return memunit_step(x()); } // distance in mem units (bytes or bits) between adjacent pixels on the same row
  254. bool is_1d_traversable(x_coord_t width) const { return row_size()-pixel_size()*width==0; } // is there no gap at the end of each row?
  255. // Returns the vertical distance (it2.y-it1.y) between two x_iterators given the difference of their x positions
  256. std::ptrdiff_t y_distance_to(const this_t& p2, x_coord_t xDiff) const {
  257. std::ptrdiff_t rowDiff=memunit_distance(x(),p2.x())-pixel_size()*xDiff;
  258. assert(( rowDiff % row_size())==0);
  259. return rowDiff / row_size();
  260. }
  261. private:
  262. template <typename X> friend class memory_based_2d_locator;
  263. std::ptrdiff_t offset(x_coord_t x, y_coord_t y) const { return y*row_size() + x*pixel_size(); }
  264. StepIterator _p;
  265. };
  266. /////////////////////////////
  267. // PixelBasedConcept
  268. /////////////////////////////
  269. template <typename SI>
  270. struct color_space_type<memory_based_2d_locator<SI> > : public color_space_type<typename memory_based_2d_locator<SI>::parent_t> {
  271. };
  272. template <typename SI>
  273. struct channel_mapping_type<memory_based_2d_locator<SI> > : public channel_mapping_type<typename memory_based_2d_locator<SI>::parent_t> {
  274. };
  275. template <typename SI>
  276. struct is_planar<memory_based_2d_locator<SI> > : public is_planar<typename memory_based_2d_locator<SI>::parent_t> {
  277. };
  278. template <typename SI>
  279. struct channel_type<memory_based_2d_locator<SI> > : public channel_type<typename memory_based_2d_locator<SI>::parent_t> {
  280. };
  281. /////////////////////////////
  282. // HasDynamicXStepTypeConcept
  283. /////////////////////////////
  284. // Take the base iterator of SI (which is typically a step iterator) and change it to have a step in x
  285. template <typename SI>
  286. struct dynamic_x_step_type<memory_based_2d_locator<SI> > {
  287. private:
  288. typedef typename iterator_adaptor_get_base<SI>::type base_iterator_t;
  289. typedef typename dynamic_x_step_type<base_iterator_t>::type base_iterator_step_t;
  290. typedef typename iterator_adaptor_rebind<SI, base_iterator_step_t>::type dynamic_step_base_t;
  291. public:
  292. typedef memory_based_2d_locator<dynamic_step_base_t> type;
  293. };
  294. /////////////////////////////
  295. // HasDynamicYStepTypeConcept
  296. /////////////////////////////
  297. template <typename SI>
  298. struct dynamic_y_step_type<memory_based_2d_locator<SI> > {
  299. typedef memory_based_2d_locator<SI> type;
  300. };
  301. } } // namespace boost::gil
  302. #endif