13 #ifndef GIL_ALGORITHM_HPP
14 #define GIL_ALGORITHM_HPP
24 #include <boost/config.hpp>
25 #include <boost/utility/enable_if.hpp>
26 #include <boost/mpl/and.hpp>
27 #include <boost/mpl/or.hpp>
50 namespace boost {
namespace gil {
52 template <
typename ChannelPtr,
typename ColorSpace>
54 template <
typename Iterator>
56 template <
typename StepIterator>
99 template <
typename Derived,
typename Result=
void>
101 typedef Result result_type;
103 template <
typename V1,
typename V2> BOOST_FORCEINLINE
104 result_type operator()(
const std::pair<const V1*,const V2*>& p)
const {
108 template <
typename V1,
typename V2> BOOST_FORCEINLINE
109 result_type operator()(
const V1& v1,
const V2& v2)
const {
113 result_type operator()(
const error_t&)
const {
throw std::bad_cast(); }
117 template <
typename V1,
typename V2>
118 BOOST_FORCEINLINE result_type apply(
const V1& v1,
const V2& v2, mpl::false_)
const {
119 return ((
const Derived*)
this)->apply_incompatible(v1,v2);
123 template <
typename V1,
typename V2>
124 BOOST_FORCEINLINE result_type apply(
const V1& v1,
const V2& v2, mpl::true_)
const {
125 return ((
const Derived*)
this)->apply_compatible(v1,v2);
129 template <
typename V1,
typename V2>
130 BOOST_FORCEINLINE result_type apply_incompatible(
const V1& v1,
const V2& v2)
const {
131 throw std::bad_cast();
149 template<
typename T,
typename Cs>
158 template<
typename T,
typename Cs>
166 namespace boost {
namespace gil {
168 template <
typename I,
typename O>
struct copy_fn {
169 BOOST_FORCEINLINE I operator()(I first, I last, O dst)
const {
return std::copy(first,last,dst); }
177 template<
typename Cs,
typename IC1,
typename IC2> BOOST_FORCEINLINE
179 boost::gil::gil_function_requires<boost::gil::ChannelsCompatibleConcept<typename std::iterator_traits<IC1>::value_type,
typename std::iterator_traits<IC2>::value_type> >();
180 static_for_each(first,last,dst,boost::gil::detail::copy_fn<IC1,IC2>());
181 return dst+(last-first);
185 namespace boost {
namespace gil {
189 template <
typename I,
typename O>
191 BOOST_FORCEINLINE
void operator()(I src,
typename std::iterator_traits<I>::difference_type n, O dst)
const {
std::copy(src,src+n, dst); }
195 template <
typename IL,
typename O>
197 typedef typename std::iterator_traits<iterator_from_2d<IL> >::difference_type diff_t;
199 gil_function_requires<PixelLocatorConcept<IL> >();
200 gil_function_requires<MutablePixelIteratorConcept<O> >();
202 typedef typename iterator_from_2d<IL>::difference_type diff_t;
203 diff_t l=src.width()-src.x_pos();
204 diff_t numToCopy=(n<l ? n:l);
205 detail::copy_n(src.x(), numToCopy, dst);
214 template <
typename I,
typename OL>
216 typedef typename std::iterator_traits<I>::difference_type diff_t;
218 gil_function_requires<PixelIteratorConcept<I> >();
219 gil_function_requires<MutablePixelLocatorConcept<OL> >();
221 diff_t l=dst.width()-dst.x_pos();
222 diff_t numToCopy=(n<l ? n:l);
223 detail::copy_n(src, numToCopy, dst.x());
232 template <
typename IL,
typename OL>
234 typedef typename iterator_from_2d<IL>::difference_type diff_t;
236 gil_function_requires<PixelLocatorConcept<IL> >();
237 gil_function_requires<MutablePixelLocatorConcept<OL> >();
238 if (src.x_pos()!=dst.x_pos() || src.width()!=dst.width()) {
244 diff_t l=dst.width()-dst.x_pos();
245 diff_t numToCopy=(n<l ? n : l);
246 detail::copy_n(src.x(), numToCopy, dst.x());
254 template <
typename SrcIterator,
typename DstIterator>
255 BOOST_FORCEINLINE DstIterator copy_with_2d_iterators(SrcIterator first, SrcIterator last, DstIterator dst) {
256 typedef typename SrcIterator::x_iterator src_x_iterator;
257 typedef typename DstIterator::x_iterator dst_x_iterator;
259 typename SrcIterator::difference_type n = last - first;
261 if (first.is_1d_traversable()) {
262 if (dst.is_1d_traversable())
267 if (dst.is_1d_traversable())
268 copier_n<SrcIterator,dst_x_iterator>()(first,n, dst.x());
270 copier_n<SrcIterator,DstIterator>()(first,n,dst);
280 template <
typename IL,
typename OL>
282 return boost::gil::detail::copy_with_2d_iterators(first,last,dst);
286 namespace boost {
namespace gil {
289 template <
typename View1,
typename View2> BOOST_FORCEINLINE
291 assert(src.dimensions()==dst.dimensions());
292 detail::copy_with_2d_iterators(src.begin(),src.end(),dst.begin());
308 template <
typename CC>
309 class copy_and_convert_pixels_fn :
public binary_operation_obj<copy_and_convert_pixels_fn<CC> > {
313 typedef typename binary_operation_obj<copy_and_convert_pixels_fn<CC> >::result_type result_type;
314 copy_and_convert_pixels_fn() {}
315 copy_and_convert_pixels_fn(CC cc_in) : _cc(cc_in) {}
317 template <
typename V1,
typename V2> BOOST_FORCEINLINE
318 result_type apply_incompatible(
const V1& src,
const V2& dst)
const {
319 copy_pixels(color_converted_view<typename V2::value_type>(src,_cc),dst);
323 template <
typename V1,
typename V2> BOOST_FORCEINLINE
324 result_type apply_compatible(
const V1& src,
const V2& dst)
const {
331 template <
typename V1,
typename V2,
typename CC>
333 void copy_and_convert_pixels(
const V1& src,
const V2& dst,CC cc) {
334 detail::copy_and_convert_pixels_fn<CC> ccp(cc);
338 struct default_color_converter;
341 template <
typename View1,
typename View2>
343 void copy_and_convert_pixels(
const View1& src,
const View2& dst) {
344 detail::copy_and_convert_pixels_fn<default_color_converter> ccp;
368 template <
typename IL,
typename V>
370 boost::gil::gil_function_requires<boost::gil::MutablePixelLocatorConcept<IL> >();
371 if (first.is_1d_traversable()) {
375 std::ptrdiff_t n=last-first;
377 std::ptrdiff_t numToDo=std::min<const std::ptrdiff_t>(n,(std::ptrdiff_t)(first.width()-first.x_pos()));
378 fill_n(first.x(), numToDo, val);
386 namespace boost {
namespace gil {
390 template <
typename It,
typename P>
391 void operator()(It first, It last,
const P& p_in) {
396 template <
typename It,
typename P>
398 void fill_aux(It first, It last,
const P& p, mpl::true_) {
402 template <
typename It,
typename P>
404 void fill_aux(It first, It last,
const P& p,mpl::false_) {
411 template <
typename View,
typename Value> BOOST_FORCEINLINE
413 if (img_view.is_1d_traversable())
414 detail::fill_aux(img_view.begin().x(), img_view.end().x(),
415 val,is_planar<View>());
417 for (std::ptrdiff_t y=0; y<img_view.height(); ++y)
418 detail::fill_aux(img_view.row_begin(y),img_view.row_end(y),
419 val,is_planar<View>());
433 template <
typename It> BOOST_FORCEINLINE
434 void destruct_range_impl( It first
436 ,
typename enable_if< mpl::and_< is_pointer< It >
437 , mpl::not_< boost::has_trivial_destructor<
typename std::iterator_traits<It>::value_type > >
442 while (first!=last) {
448 template <
typename It> BOOST_FORCEINLINE
449 void destruct_range_impl( It
451 ,
typename enable_if< mpl::or_< mpl::not_< is_pointer< It > >
452 , boost::has_trivial_destructor<
typename std::iterator_traits< It >::value_type >
457 template <
typename It> BOOST_FORCEINLINE
458 void destruct_range(It first, It last) {
459 destruct_range_impl( first
464 struct std_destruct_t {
465 template <
typename It>
void operator()(It first, It last)
const { destruct_range(first,last); }
469 template <
typename It>
471 void destruct_aux(It first, It last, mpl::true_) {
472 static_for_each(first,last,std_destruct_t());
475 template <
typename It>
477 void destruct_aux(It first, It last, mpl::false_) {
478 destruct_range(first,last);
484 template <
typename View> BOOST_FORCEINLINE
486 if (img_view.is_1d_traversable())
487 detail::destruct_aux(img_view.begin().x(), img_view.end().x(),
490 for (std::ptrdiff_t y=0; y<img_view.height(); ++y)
491 detail::destruct_aux(img_view.row_begin(y),img_view.row_end(y),
508 template <
typename It,
typename P>
510 void uninitialized_fill_aux(It first, It last,
511 const P& p, mpl::true_) {
514 typedef typename std::iterator_traits<It>::value_type pixel_t;
515 while (channel < num_channels<pixel_t>::value) {
516 std::uninitialized_fill(dynamic_at_c(first,channel), dynamic_at_c(last,channel),
517 dynamic_at_c(p,channel));
521 for (
int c=0; c<channel; ++c)
522 destruct_range(dynamic_at_c(first,c), dynamic_at_c(last,c));
529 template <
typename It,
typename P>
531 void uninitialized_fill_aux(It first, It last,
532 const P& p,mpl::false_) {
533 std::uninitialized_fill(first,last,p);
541 template <
typename View,
typename Value>
543 if (img_view.is_1d_traversable())
544 detail::uninitialized_fill_aux(img_view.begin().x(), img_view.end().x(),
545 val,is_planar<View>());
547 typename View::y_coord_t y;
549 for (y=0; y<img_view.height(); ++y)
550 detail::uninitialized_fill_aux(img_view.row_begin(y),img_view.row_end(y),
551 val,is_planar<View>());
553 for (
typename View::y_coord_t y0=0; y0<y; ++y0)
554 detail::destruct_aux(img_view.row_begin(y0),img_view.row_end(y0), is_planar<View>());
571 template <
typename It> BOOST_FORCEINLINE
572 void default_construct_range_impl(It first, It last, mpl::true_) {
573 typedef typename std::iterator_traits<It>::value_type value_t;
576 while (first!=last) {
577 new (first) value_t();
581 destruct_range(first1,first);
586 template <
typename It> BOOST_FORCEINLINE
587 void default_construct_range_impl(It, It, mpl::false_) {}
589 template <
typename It> BOOST_FORCEINLINE
590 void default_construct_range(It first, It last) { default_construct_range_impl(first, last,
typename is_pointer<It>::type()); }
593 template <
typename It>
595 void default_construct_aux(It first, It last, mpl::true_) {
598 typedef typename std::iterator_traits<It>::value_type pixel_t;
599 while (channel < num_channels<pixel_t>::value) {
600 default_construct_range(dynamic_at_c(first,channel), dynamic_at_c(last,channel));
604 for (
int c=0; c<channel; ++c)
605 destruct_range(dynamic_at_c(first,c), dynamic_at_c(last,c));
611 template <
typename It>
613 void default_construct_aux(It first, It last, mpl::false_) {
614 default_construct_range(first,last);
617 template <
typename View,
bool IsPlanar>
618 struct has_trivial_pixel_constructor :
public boost::has_trivial_constructor<typename View::value_type> {};
619 template <
typename View>
620 struct has_trivial_pixel_constructor<View, true> :
public boost::has_trivial_constructor<typename channel_type<View>::type> {};
624 template<
typename View,
bool B > BOOST_FORCEINLINE
625 void default_construct_pixels_impl(
const View& img_view
626 , boost::enable_if< is_same< mpl::bool_< B >
632 if( img_view.is_1d_traversable() )
634 detail::default_construct_aux( img_view.begin().x()
641 typename View::y_coord_t y;
644 for( y = 0; y < img_view.height(); ++y )
646 detail::default_construct_aux( img_view.row_begin( y )
647 ,img_view.row_end( y )
653 for (
typename View::y_coord_t y0 = 0; y0 < y; ++y0 )
655 detail::destruct_aux( img_view.row_begin(y0)
656 , img_view.row_end(y0)
671 template <
typename View>
673 detail::default_construct_pixels_impl< View
674 , detail::has_trivial_pixel_constructor< View
675 , is_planar< View >::value
692 template <
typename It1,
typename It2>
694 void uninitialized_copy_aux(It1 first1, It1 last1,
695 It2 first2, mpl::true_) {
698 typedef typename std::iterator_traits<It1>::value_type pixel_t;
699 while (channel < num_channels<pixel_t>::value) {
700 std::uninitialized_copy(dynamic_at_c(first1,channel), dynamic_at_c(last1,channel), dynamic_at_c(first2,channel));
705 std::advance(last2, std::distance(first1,last1));
706 for (
int c=0; c<channel; ++c)
707 destruct_range(dynamic_at_c(first2,c), dynamic_at_c(last2,c));
712 template <
typename It1,
typename It2>
714 void uninitialized_copy_aux(It1 first1, It1 last1,
715 It2 first2,mpl::false_) {
716 std::uninitialized_copy(first1,last1,first2);
724 template <
typename View1,
typename View2>
726 typedef mpl::bool_<is_planar<View1>::value && is_planar<View2>::value> is_planar;
727 assert(view1.dimensions()==view2.dimensions());
728 if (view1.is_1d_traversable() && view2.is_1d_traversable())
729 detail::uninitialized_copy_aux(view1.begin().x(), view1.end().x(),
733 typename View1::y_coord_t y;
735 for (y=0; y<view1.height(); ++y)
736 detail::uninitialized_copy_aux(view1.row_begin(y), view1.row_end(y),
740 for (
typename View1::y_coord_t y0=0; y0<y; ++y0)
741 detail::destruct_aux(view2.row_begin(y0),view2.row_end(y0), is_planar());
763 template <
typename V,
typename F>
764 F for_each_pixel(
const V& img, F fun) {
765 if (img.is_1d_traversable()) {
766 return std::for_each(img.begin().x(), img.end().x(), fun);
768 for (std::ptrdiff_t y=0; y<img.height(); ++y)
769 fun = std::for_each(img.row_begin(y),img.row_end(y),fun);
779 template <
typename View,
typename F>
780 F for_each_pixel_position(
const View& img, F fun) {
781 typename View::xy_locator loc=img.xy_at(0,0);
782 for (std::ptrdiff_t y=0; y<img.height(); ++y) {
783 for (std::ptrdiff_t x=0; x<img.width(); ++x, ++loc.x())
785 loc.x()-=img.width(); ++loc.y();
802 template <
typename View,
typename F>
804 if (v.is_1d_traversable()) {
805 std::generate(v.begin().x(), v.end().x(), fun);
807 for (std::ptrdiff_t y=0; y<v.height(); ++y)
808 std::generate(v.row_begin(y),v.row_end(y),fun);
822 template <
typename I1,
typename I2> BOOST_FORCEINLINE
bool equal_n(I1 i1, std::ptrdiff_t n, I2 i2);
825 template <
typename I1,
typename I2>
827 BOOST_FORCEINLINE
bool operator()(I1 i1, std::ptrdiff_t n, I2 i2)
const {
return std::equal(i1,i1+n, i2); }
832 template<
typename T,
typename Cs>
838 template<
typename T,
typename Cs>
844 template<
typename IC,
typename Cs>
847 std::ptrdiff_t numBytes=n*
sizeof(
typename std::iterator_traits<IC>::value_type);
849 for (std::ptrdiff_t i=0; i<mpl::size<Cs>::value; ++i)
850 if (memcmp(dynamic_at_c(i1,i), dynamic_at_c(i2,i), numBytes)!=0)
857 template <
typename Loc,
typename I2>
860 gil_function_requires<boost::gil::PixelLocatorConcept<Loc> >();
861 gil_function_requires<boost::gil::PixelIteratorConcept<I2> >();
863 std::ptrdiff_t num=std::min<const std::ptrdiff_t>(n, i1.width()-i1.x_pos());
864 if (!equal_n(i1.x(), num, i2))
875 template <
typename I1,
typename Loc>
878 gil_function_requires<boost::gil::PixelIteratorConcept<I1> >();
879 gil_function_requires<boost::gil::PixelLocatorConcept<Loc> >();
881 std::ptrdiff_t num=std::min<const std::ptrdiff_t>(n,i2.width()-i2.x_pos());
882 if (!equal_n(i1, num, i2.x()))
893 template <
typename Loc1,
typename Loc2>
896 gil_function_requires<boost::gil::PixelLocatorConcept<Loc1> >();
897 gil_function_requires<boost::gil::PixelLocatorConcept<Loc2> >();
898 if (i1.x_pos()!=i2.x_pos() || i1.width()!=i2.width()) {
900 if (*i1++!=*i2++)
return false;
904 std::ptrdiff_t num=std::min<const std::ptrdiff_t>(n,i2.width()-i2.x_pos());
905 if (!equal_n(i1.x(), num, i2.x()))
916 template <
typename I1,
typename I2> BOOST_FORCEINLINE
917 bool equal_n(I1 i1, std::ptrdiff_t n, I2 i2) {
918 return detail::equal_n_fn<I1,I2>()(i1,n,i2);
934 template <
typename Loc1,
typename Loc2> BOOST_FORCEINLINE
936 boost::gil::gil_function_requires<boost::gil::PixelLocatorConcept<Loc1> >();
937 boost::gil::gil_function_requires<boost::gil::PixelLocatorConcept<Loc2> >();
938 std::ptrdiff_t n=last-first;
939 if (first.is_1d_traversable()) {
940 if (first2.is_1d_traversable())
941 return boost::gil::detail::equal_n_fn<typename Loc1::x_iterator,typename Loc2::x_iterator>()(first.x(),n, first2.x());
943 return boost::gil::detail::equal_n_fn<typename Loc1::x_iterator,boost::gil::iterator_from_2d<Loc2> >()(first.x(),n, first2);
945 if (first2.is_1d_traversable())
953 namespace boost {
namespace gil {
956 template <
typename View1,
typename View2> BOOST_FORCEINLINE
958 assert(v1.dimensions()==v2.dimensions());
959 return std::equal(v1.begin(),v1.end(),v2.begin());
974 template <
typename View1,
typename View2,
typename F> BOOST_FORCEINLINE
976 assert(src.dimensions()==dst.dimensions());
977 for (std::ptrdiff_t y=0; y<src.height(); ++y) {
978 typename View1::x_iterator srcIt=src.row_begin(y);
979 typename View2::x_iterator dstIt=dst.row_begin(y);
980 for (std::ptrdiff_t x=0; x<src.width(); ++x)
981 dstIt[x]=fun(srcIt[x]);
988 template <
typename View1,
typename View2,
typename View3,
typename F> BOOST_FORCEINLINE
990 for (std::ptrdiff_t y=0; y<dst.height(); ++y) {
991 typename View1::x_iterator srcIt1=src1.row_begin(y);
992 typename View2::x_iterator srcIt2=src2.row_begin(y);
993 typename View3::x_iterator dstIt=dst.row_begin(y);
994 for (std::ptrdiff_t x=0; x<dst.width(); ++x)
995 dstIt[x]=fun(srcIt1[x],srcIt2[x]);
1006 template <
typename View1,
typename View2,
typename F> BOOST_FORCEINLINE
1008 assert(src.dimensions()==dst.dimensions());
1009 typename View1::xy_locator loc=src.xy_at(0,0);
1010 for (std::ptrdiff_t y=0; y<src.height(); ++y) {
1011 typename View2::x_iterator dstIt=dst.row_begin(y);
1012 for (std::ptrdiff_t x=0; x<src.width(); ++x, ++loc.x())
1014 loc.x()-=src.width(); ++loc.y();
1021 template <
typename View1,
typename View2,
typename View3,
typename F> BOOST_FORCEINLINE
1023 assert(src1.dimensions()==dst.dimensions());
1024 assert(src2.dimensions()==dst.dimensions());
1025 typename View1::xy_locator loc1=src1.xy_at(0,0);
1026 typename View2::xy_locator loc2=src2.xy_at(0,0);
1027 for (std::ptrdiff_t y=0; y<src1.height(); ++y) {
1028 typename View3::x_iterator dstIt=dst.row_begin(y);
1029 for (std::ptrdiff_t x=0; x<src1.width(); ++x, ++loc1.x(), ++loc2.x())
1030 dstIt[x]=fun(loc1,loc2);
1031 loc1.x()-=src1.width(); ++loc1.y();
1032 loc2.x()-=src2.width(); ++loc2.y();
void uninitialized_fill_pixels(const View &img_view, const Value &val)
std::uninitialized_fill for image views. Does not support planar heterogeneous views. If an exception is thrown destructs any in-place copy-constructed pixels
Definition: algorithm.hpp:542
Represents a pixel value (a container of channels). Models: HomogeneousColorBaseValueConcept, PixelValueConcept, HomogeneousPixelBasedConcept.
Definition: metafunctions.hpp:44
void generate_pixels(const View &v, F fun)
std::generate for image views
Definition: algorithm.hpp:803
An iterator over planar pixels. Models HomogeneousColorBaseConcept, PixelIteratorConcept, HomogeneousPixelBasedConcept, MemoryBasedIteratorConcept, HasDynamicXStepTypeConcept.
Definition: algorithm.hpp:53
Concept check classes for GIL concepts.
Definition: algorithm.hpp:190
Provides 1D random-access navigation to the pixels of the image. Models: PixelIteratorConcept, PixelBasedConcept, HasDynamicXStepTypeConcept.
Definition: iterator_from_2d.hpp:52
Memory-based pixel locator. Models: PixelLocatorConcept,HasDynamicXStepTypeConcept,HasDynamicYStepTypeConcept,HasTransposedTypeConceptThe class takes a step iterator as a parameter. The step iterator provides navigation along the vertical axis while its base iterator provides horizontal navigation.
Definition: algorithm.hpp:57
BOOST_FORCEINLINE bool equal_pixels(const View1 &v1, const View2 &v2)
std::equal for image views
Definition: algorithm.hpp:957
void fill(boost::gil::iterator_from_2d< IL > first, boost::gil::iterator_from_2d< IL > last, const V &val)
std::fill(I,I,V) with I being a iterator_from_2d
Definition: algorithm.hpp:369
Returns whether two views are compatible.
Definition: gil_concept.hpp:2049
BOOST_FORCEINLINE void copy_pixels(const View1 &src, const View2 &dst)
std::copy for image views
Definition: algorithm.hpp:290
BOOST_FORCEINLINE void fill_pixels(const View &img_view, const Value &val)
std::fill for image views
Definition: algorithm.hpp:412
void default_construct_pixels(const View &img_view)
Invokes the in-place default constructor on every pixel of the (uninitialized) view. Does not support planar heterogeneous views. If an exception is thrown destructs any in-place default-constructed pixels.
Definition: algorithm.hpp:672
BOOST_FORCEINLINE boost::gil::iterator_from_2d< OL > copy1(boost::gil::iterator_from_2d< IL > first, boost::gil::iterator_from_2d< IL > last, boost::gil::iterator_from_2d< OL > dst)
std::copy(I1,I1,I2) with I1 and I2 being a iterator_from_2d
Definition: algorithm.hpp:281
BOOST_FORCEINLINE boost::gil::planar_pixel_iterator< IC2, Cs > copy(boost::gil::planar_pixel_iterator< IC1, Cs > first, boost::gil::planar_pixel_iterator< IC1, Cs > last, boost::gil::planar_pixel_iterator< IC2, Cs > dst)
Copy when both src and dst are planar pointers is copy for each channel.
Definition: algorithm.hpp:178
BOOST_FORCEINLINE bool equal(boost::gil::iterator_from_2d< Loc1 > first, boost::gil::iterator_from_2d< Loc1 > last, boost::gil::iterator_from_2d< Loc2 > first2)
std::equal(I1,I1,I2) with I1 and I2 being a iterator_from_2d
Definition: algorithm.hpp:935
A generic binary operation on viewsUse this class as a convenience superclass when defining an operat...
Definition: algorithm.hpp:100
struct to do std::fill
Definition: algorithm.hpp:389
void uninitialized_copy_pixels(const View1 &view1, const View2 &view2)
std::uninitialized_copy for image views. Does not support planar heterogeneous views. If an exception is thrown destructs any in-place copy-constructed objects
Definition: algorithm.hpp:725
Methods for constructing image views from raw data or other image views /.
BOOST_FORCEINLINE void destruct_pixels(const View &img_view)
Invokes the in-place destructor on every pixel of the view.
Definition: algorithm.hpp:485
MEMORY-BASED STEP ITERATOR.
Definition: algorithm.hpp:55
A model of a heterogeneous pixel that is not byte aligned. Examples are bitmap (1-bit pixels) or 6-bi...