| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196 | //// Copyright 2005-2007 Adobe Systems Incorporated//// Distributed under the Boost Software License, Version 1.0// See accompanying file LICENSE_1_0.txt or copy at// http://www.boost.org/LICENSE_1_0.txt//#ifndef BOOST_GIL_VIRTUAL_LOCATOR_HPP#define BOOST_GIL_VIRTUAL_LOCATOR_HPP#include <boost/gil/dynamic_step.hpp>#include <boost/gil/position_iterator.hpp>#include <boost/assert.hpp>#include <boost/iterator/iterator_facade.hpp>namespace boost { namespace gil {/// \ingroup PixelLocatorModel PixelBasedModel/// \brief A 2D locator over a virtual image/// Upon dereferencing, invokes a given function object passing it its coordinates./// Models:///   PixelLocatorConcept,///   HasDynamicXStepTypeConcept,///   HasDynamicYStepTypeConcept,///   HasTransposedTypeConcept////// \tparam DerefFn Function object that given a point returns a reference.///         Models PixelDereferenceAdaptorConcept./// \tparam IsTransposed Indicates if locator should navigate in transposed mode.template <typename DerefFn, bool IsTransposed>class virtual_2d_locator    : public pixel_2d_locator_base        <            virtual_2d_locator<DerefFn, IsTransposed>,            position_iterator<DerefFn, IsTransposed>,            position_iterator<DerefFn, 1-IsTransposed>        >{    using this_t = virtual_2d_locator<DerefFn, IsTransposed>;public:    using parent_t = pixel_2d_locator_base        <            virtual_2d_locator<DerefFn, IsTransposed>,            position_iterator<DerefFn, IsTransposed>,            position_iterator<DerefFn, 1-IsTransposed>        >;    using const_t = virtual_2d_locator<typename DerefFn::const_t, IsTransposed>;    using deref_fn_t = DerefFn;    using point_t = typename parent_t::point_t;    using coord_t = typename parent_t::coord_t;    using x_coord_t = typename parent_t::x_coord_t;    using y_coord_t = typename parent_t::y_coord_t;    using x_iterator = typename parent_t::x_iterator;    using y_iterator = typename parent_t::y_iterator;    template <typename NewDerefFn>    struct add_deref    {        using type = virtual_2d_locator<deref_compose<NewDerefFn, DerefFn>, IsTransposed>;        static type make(this_t const& loc, NewDerefFn const& new_deref_fn)        {            return type(loc.pos(), loc.step(),                deref_compose<NewDerefFn, DerefFn>(new_deref_fn, loc.deref_fn()));        }    };    virtual_2d_locator(        point_t const& p = {0, 0},        point_t const& step = {1, 1},        deref_fn_t const& deref_fn = deref_fn_t())        : y_pos_(p, step, deref_fn)    {}    template <typename D, bool TR>    virtual_2d_locator(virtual_2d_locator<D, TR> const &loc, coord_t y_step)        : y_pos_(loc.pos(), point_t(loc.step().x, loc.step().y * y_step), loc.deref_fn())    {}    template <typename D, bool TR>    virtual_2d_locator(virtual_2d_locator<D, TR> const& loc, coord_t x_step, coord_t y_step, bool transpose = false)        : y_pos_(loc.pos()        , transpose ?            point_t(loc.step().x * y_step, loc.step().y * x_step) :            point_t(loc.step().x * x_step, loc.step().y * y_step)        , loc.deref_fn())    {        BOOST_ASSERT(transpose == (IsTransposed != TR));    }    template <typename D, bool TR>    virtual_2d_locator(virtual_2d_locator<D, TR> const& other) : y_pos_(other.y_pos_) {}    virtual_2d_locator(virtual_2d_locator const& other) : y_pos_(other.y_pos_) {}    virtual_2d_locator& operator=(virtual_2d_locator const& other) = default;    bool operator==(const this_t& p) const { return y_pos_ == p.y_pos_; }    auto x() -> x_iterator&    {        return *gil_reinterpret_cast<x_iterator*>(this);    }    auto x() const -> x_iterator const&    {        return *gil_reinterpret_cast_c<x_iterator const*>(this);    }    auto y() -> y_iterator& { return y_pos_; }    auto y() const -> y_iterator const& { return y_pos_; }    /// Returns the y distance between two x_iterators given the difference of their x positions    auto y_distance_to(this_t const& it2, x_coord_t) const -> y_coord_t    {        return (it2.pos()[1 - IsTransposed] - pos()[1 - IsTransposed])                / step()[1 - IsTransposed];    }    /// \todo TODO: is there no gap at the end of each row?    ///       i.e. can we use x_iterator to visit every pixel instead of nested loops?    bool is_1d_traversable(x_coord_t) const { return false; }    // Methods specific for virtual 2D locator    auto pos() const -> point_t const& { return y_pos_.pos(); }    auto step() const -> point_t const& { return y_pos_.step(); }    auto deref_fn() const -> deref_fn_t const& { return y_pos_.deref_fn(); }private:    template <typename D, bool TR>    friend class virtual_2d_locator;    y_iterator y_pos_; // current position, the step and the dereference object};///////////////////////////////  PixelBasedConcept/////////////////////////////template <typename D, bool TR>struct channel_type<virtual_2d_locator<D, TR>>    : channel_type<typename virtual_2d_locator<D, TR>::parent_t>{};template <typename D, bool TR>struct color_space_type<virtual_2d_locator<D, TR>>    : color_space_type<typename virtual_2d_locator<D, TR>::parent_t>{};template <typename D, bool TR>struct channel_mapping_type<virtual_2d_locator<D, TR>>    : channel_mapping_type<typename virtual_2d_locator<D, TR>::parent_t>{};template <typename D, bool TR>struct is_planar<virtual_2d_locator<D, TR>>    : is_planar<typename virtual_2d_locator<D, TR>::parent_t>{};///////////////////////////////  HasDynamicXStepTypeConcept/////////////////////////////template <typename D, bool TR>struct dynamic_x_step_type<virtual_2d_locator<D,TR>>{    using type = virtual_2d_locator<D,TR>;};///////////////////////////////  HasDynamicYStepTypeConcept/////////////////////////////template <typename D, bool TR>struct dynamic_y_step_type<virtual_2d_locator<D,TR>>{    using type = virtual_2d_locator<D,TR>;};///////////////////////////////  HasTransposedTypeConcept/////////////////////////////template <typename D, bool IsTransposed>struct transposed_type<virtual_2d_locator<D,IsTransposed>>{    using type = virtual_2d_locator<D,1-IsTransposed>;};}}  // namespace boost::gil#endif
 |