https://svn.lrde.epita.fr/svn/oln/trunk/milena
Index: ChangeLog
from Roland Levillain <roland(a)lrde.epita.fr>
Area closing/opening for line graph images based on adjacent
vertices.
* mln/core/line_graph_psite.hh
(mln::line_graph_psite<P>::first_id)
(mln::line_graph_psite<P>::second_id):
New methods.
Use them to simplify...
(mln::line_graph_psite<P>::first)
(mln::line_graph_psite<P>::second): ...these ones.
* mln/accu/count_adjacent_vertices.hh: New accumulator.
* mln/morpho/closing_area_on_vertices.hh,
* mln/morpho/opening_area_on_vertices.hh:
New filters.
* mln/morpho/closing_area.hh,
* mln/morpho/opening_area.hh,
* mln/morpho/closing_attribute.hh,
* mln/morpho/opening_attribute.hh:
Remove spurious comments.
* tests/morpho/lena_line_graph_image_wst2.cc: Use
morpho::closing_area_on_vertices instead of morpho::closing.
mln/accu/count_adjacent_vertices.hh | 113 +++++++++++++++++------------
mln/core/line_graph_psite.hh | 32 +++++++-
mln/morpho/closing_area.hh | 5 -
mln/morpho/closing_area_on_vertices.hh | 39 ++++------
mln/morpho/closing_attribute.hh | 5 -
mln/morpho/opening_area.hh | 5 -
mln/morpho/opening_area_on_vertices.hh | 38 ++++-----
mln/morpho/opening_attribute.hh | 5 -
tests/morpho/lena_line_graph_image_wst2.cc | 16 +---
9 files changed, 138 insertions(+), 120 deletions(-)
Index: mln/core/line_graph_psite.hh
--- mln/core/line_graph_psite.hh (revision 1801)
+++ mln/core/line_graph_psite.hh (working copy)
@@ -79,11 +79,16 @@
/// Return the edge id of this point site.
util::edge_id id() const;
- /// Return the first associated point (vertex).
+ /// Return the first associated vertex.
P first() const;
- /// Return the second associated point (vertex).
+ /// Return the second associated vertex.
P second() const;
+ /// Return the id of the first associated vertex.
+ util::node_id first_id() const;
+ /// Return the id of the second associated vertex.
+ util::node_id second_id() const;
+
private:
/// Is this psite valid?
bool is_valid_() const;
@@ -228,7 +233,7 @@
line_graph_psite<P>::first() const
{
mln_assertion(is_valid_());
- return plg().gr_->node_data(plg().gr_->edge(id_).n1());
+ return plg().gr_->node_data(first_id());
}
template<typename P>
@@ -237,7 +242,26 @@
line_graph_psite<P>::second() const
{
mln_assertion(is_valid_());
- return plg().gr_->node_data(plg().gr_->edge(id_).n2());
+ return plg().gr_->node_data(second_id());
+ }
+
+
+ template<typename P>
+ inline
+ util::node_id
+ line_graph_psite<P>::first_id() const
+ {
+ mln_assertion(is_valid_());
+ return plg().gr_->edge(id_).n1();
+ }
+
+ template<typename P>
+ inline
+ util::node_id
+ line_graph_psite<P>::second_id() const
+ {
+ mln_assertion(is_valid_());
+ return plg().gr_->edge(id_).n2();
}
Index: mln/accu/count_adjacent_vertices.hh
--- mln/accu/count_adjacent_vertices.hh (revision 1801)
+++ mln/accu/count_adjacent_vertices.hh (working copy)
@@ -1,4 +1,4 @@
-// Copyright (C) 2007 EPITA Research and Development Laboratory
+// Copyright (C) 2007, 2008 EPITA Research and Development Laboratory (LRDE)
//
// This file is part of the Olena Library. This library is free
// software; you can redistribute it and/or modify it under the terms
@@ -25,17 +25,17 @@
// reasons why the executable file might be covered by the GNU General
// Public License.
-#ifndef MLN_ACCU_COUNT_HH
-# define MLN_ACCU_COUNT_HH
+#ifndef MLN_ACCU_COUNT_ADJACENT_VERTICES_HH
+# define MLN_ACCU_COUNT_ADJACENT_VERTICES_HH
-/*! \file mln/accu/count.hh
- *
- * \brief Define an accumulator that counts.
- */
+/// \file mln/accu/count_adjacent_vertices.hh
+/// \brief Define an accumulator that counts the vertices adjacent to a
+/// set of line graph psite.
# include <mln/accu/internal/base.hh>
# include <mln/core/concept/meta_accumulator.hh>
-
+# include <mln/core/line_graph_image.hh>
+# include <mln/util/pix.hh>
namespace mln
{
@@ -43,96 +43,119 @@
namespace accu
{
-
- /*!
- * \brief Generic counter accumulator class.
- *
- * The parameter \a T is the type to be count.
- */
- template <typename T>
- struct count_ : public mln::accu::internal::base_< std::size_t , count_<T> >
+ /// \brief Accumulator class counting the number of vertices
+ /// adjacent to a set of mln::line_graph_psite (i.e., a set of
+ /// edges).
+ ///
+ /// The type to be count is mln::util::pix< mln::line_graph_image<P, V> >
+ /// where \p P and \p V are the parameters of this class.
+ template <typename P, typename V>
+ struct count_adjacent_vertices_
+ : public mln::accu::internal::base_< std::size_t,
+ count_adjacent_vertices_<P, V> >
{
- typedef T argument;
- typedef std::size_t result; // FIXME: Up in Accumulator.
+ typedef mln::util::pix< mln::line_graph_image<P, V> > argument;
- count_();
+ count_adjacent_vertices_();
+ /// Manipulators.
+ /// \{
void init();
- // FIXME : should we add a take() without argument?
- void take(const argument&);
- void take(const count_<T>& other);
+ void take(const argument& arg);
+ void take(const count_adjacent_vertices_<P, V>& other);
- std::size_t to_result() const;
+ /// Force the value of the counter to \a c.
void set_value(std::size_t c);
+ /// \}
+
+ /// Get the value of the accumulator.
+ std::size_t to_result() const;
protected:
+ /// Update the value of the counter.
+ void update_ ();
+ protected:
+ /// The value of the counter.
std::size_t count__;
+ /// The set of adjacent vertices.
+ std::set<util::node_id> vertices_;
};
- /*!
- * \brief Meta accumulator for count.
- */
- struct count : public Meta_Accumulator< count >
+ /// \brief Meta accumulator for count_adjacent_vertices.
+ struct count_adjacent_vertices
+ : public Meta_Accumulator< count_adjacent_vertices >
{
- template <typename T>
+ template <typename P, typename V>
struct with
{
- typedef count_<T> ret;
+ typedef count_adjacent_vertices_<P, V> ret;
};
};
# ifndef MLN_INCLUDE_ONLY
- template <typename T>
+ template <typename P, typename V>
inline
- count_<T>::count_()
+ count_adjacent_vertices_<P, V>::count_adjacent_vertices_()
{
init();
}
- template <typename T>
+ template <typename P, typename V>
inline
void
- count_<T>::init()
+ count_adjacent_vertices_<P, V>::init()
{
- count__ = 0;
+ vertices_.clear();
+ update_();
}
- template <typename T>
+ template <typename P, typename V>
inline
void
- count_<T>::take(const argument&)
+ count_adjacent_vertices_<P, V>::take(const argument& arg)
{
- ++count__;
+ vertices_.insert(arg.p().first_id());
+ vertices_.insert(arg.p().second_id());
+ update_();
}
- template <typename T>
+ template <typename P, typename V>
inline
void
- count_<T>::take(const count_<T>& other)
+ count_adjacent_vertices_<P, V>::take(const count_adjacent_vertices_<P, V>& other)
{
- count__ += other.count__;
+ vertices_.insert (other.vertices_.begin(), other.vertices_.end());
+ update_();
}
- template <typename T>
+ template <typename P, typename V>
inline
std::size_t
- count_<T>::to_result() const
+ count_adjacent_vertices_<P, V>::to_result() const
{
return count__;
}
- template <typename T>
+ template <typename P, typename V>
inline
void
- count_<T>::set_value(std::size_t c)
+ count_adjacent_vertices_<P, V>::set_value(std::size_t c)
{
count__ = c;
}
+ template <typename P, typename V>
+ inline
+ void
+ count_adjacent_vertices_<P, V>::update_()
+ {
+ count__ = vertices_.size();
+ }
+
# endif // ! MLN_INCLUDE_ONLY
} // end of namespace mln::accu
@@ -140,4 +163,4 @@
} // end of namespace mln
-#endif // ! MLN_ACCU_COUNT_HH
+#endif // ! MLN_ACCU_COUNT_ADJACENT_VERTICES_HH
Index: mln/morpho/closing_area_on_vertices.hh
--- mln/morpho/closing_area_on_vertices.hh (revision 1801)
+++ mln/morpho/closing_area_on_vertices.hh (working copy)
@@ -25,16 +25,17 @@
// reasons why the executable file might be covered by the GNU General
// Public License.
-#ifndef MLN_MORPHO_CLOSING_AREA_HH
-# define MLN_MORPHO_CLOSING_AREA_HH
+#ifndef MLN_MORPHO_CLOSING_AREA_ON_VERTICES_HH
+# define MLN_MORPHO_CLOSING_AREA_ON_VERTICES_HH
-/*! \file mln/morpho/closing_area.hh
- *
- * \brief Morphological area closing.
- */
+/// \file mln/morpho/closing_area.hh
+///
+/// \brief Morphological area closing on a line graph image computing
+/// the area in terms of adjacent vertices.
+# include <mln/core/line_graph_image.hh>
# include <mln/morpho/closing_attribute.hh>
-# include <mln/accu/count.hh>
+# include <mln/accu/count_adjacent_vertices.hh>
namespace mln
@@ -43,28 +44,26 @@
namespace morpho
{
- /* FIXME: The neighborhood shall not be passed as argument, but
- bound to the input image. We can also optionnaly provide a
- version of this function for regular-grid-based images where
- the neighborhood is replaced by a (user-provided) window. */
-
- /// Morphological area closing.
- template <typename I, typename N, typename O>
- void closing_area(const Image<I>& input, const Neighborhood<N>& nbh,
+ /// Morphological area closing on a mln::line_graph_image computing
+ /// the area in terms of adjacent vertices.
+ template <typename P, typename V, typename N, typename O>
+ void closing_area_on_vertices(const line_graph_image<P, V>& input,
+ const Neighborhood<N>& nbh,
std::size_t lambda, Image<O>& output);
# ifndef MLN_INCLUDE_ONLY
- template <typename I, typename N, typename O>
+ template <typename P, typename V, typename N, typename O>
inline
- void closing_area(const Image<I>& input, const Neighborhood<N>& nbh,
+ void closing_area_on_vertices(const line_graph_image<P, V>& input,
+ const Neighborhood<N>& nbh,
std::size_t lambda, Image<O>& output)
{
mln_precondition(exact(output).domain() == exact(input).domain());
- typedef util::pix<I> pix_t;
+ typedef accu::count_adjacent_vertices_<P, V> attribute_t;
// FIXME: Change sig of closing_attribute!
- closing_attribute< accu::count_<pix_t> >(input, nbh, lambda, output);
+ closing_attribute<attribute_t>(input, nbh, lambda, output);
}
# endif // ! MLN_INCLUDE_ONLY
@@ -74,4 +73,4 @@
} // end of namespace mln
-#endif // ! MLN_MORPHO_CLOSING_AREA_HH
+#endif // ! MLN_MORPHO_CLOSING_AREA_ON_VERTICES_HH
Index: mln/morpho/opening_area_on_vertices.hh
--- mln/morpho/opening_area_on_vertices.hh (revision 1801)
+++ mln/morpho/opening_area_on_vertices.hh (working copy)
@@ -25,16 +25,16 @@
// reasons why the executable file might be covered by the GNU General
// Public License.
-#ifndef MLN_MORPHO_OPENING_AREA_HH
-# define MLN_MORPHO_OPENING_AREA_HH
+#ifndef MLN_MORPHO_OPENING_AREA_ON_VERTICES_HH
+# define MLN_MORPHO_OPENING_AREA_ON_VERTICES_HH
-/*! \file mln/morpho/opening_area.hh
- *
- * \brief Morphological area opening.
- */
+/// \file mln/morpho/opening_area_on_vertices.hh
+/// \brief Morphological area opening on a line graph image computing
+/// the area in terms of adjacent vertices.
+# include <mln/core/line_graph_image.hh>
# include <mln/morpho/opening_attribute.hh>
-# include <mln/accu/count.hh>
+# include <mln/accu/count_adjacent_vertices.hh>
namespace mln
@@ -43,28 +43,26 @@
namespace morpho
{
- /* FIXME: The neighborhood shall not be passed as argument, but
- bound to the input image. We can also optionnaly provide a
- version of this function for regular-grid-based images where
- the neighborhood is replaced by a (user-provided) window. */
-
- /// Morphological area opening.
- template <typename I, typename N, typename O>
- void opening_area(const Image<I>& input, const Neighborhood<N>& nbh,
+ /// Morphological area opening on a mln::line_graph_image computing
+ /// the area in terms of adjacent vertices.
+ template <typename P, typename V, typename N, typename O>
+ void opening_area_on_vertices(const line_graph_image<P, V>& input,
+ const Neighborhood<N>& nbh,
std::size_t lambda, Image<O>& output);
# ifndef MLN_INCLUDE_ONLY
- template <typename I, typename N, typename O>
+ template <typename P, typename V, typename N, typename O>
inline
- void opening_area(const Image<I>& input, const Neighborhood<N>& nbh,
+ void opening_area_on_vertices(const line_graph_image<P, V>& input,
+ const Neighborhood<N>& nbh,
std::size_t lambda, Image<O>& output)
{
mln_precondition(exact(output).domain() == exact(input).domain());
- typedef util::pix<I> pix_t;
+ typedef accu::count_adjacent_vertices_<P, V> attribute_t;
// FIXME: Change sig of opening_attribute!
- opening_attribute< accu::count_<pix_t> >(input, nbh, lambda, output);
+ opening_attribute<attribute_t>(input, nbh, lambda, output);
}
# endif // ! MLN_INCLUDE_ONLY
@@ -74,4 +72,4 @@
} // end of namespace mln
-#endif // ! MLN_MORPHO_OPENING_AREA_HH
+#endif // ! MLN_MORPHO_OPENING_AREA_ON_VERTICES_HH
Index: mln/morpho/closing_area.hh
--- mln/morpho/closing_area.hh (revision 1801)
+++ mln/morpho/closing_area.hh (working copy)
@@ -43,11 +43,6 @@
namespace morpho
{
- /* FIXME: The neighborhood shall not be passed as argument, but
- bound to the input image. We can also optionnaly provide a
- version of this function for regular-grid-based images where
- the neighborhood is replaced by a (user-provided) window. */
-
/// Morphological area closing.
template <typename I, typename N, typename O>
void closing_area(const Image<I>& input, const Neighborhood<N>& nbh,
Index: mln/morpho/opening_area.hh
--- mln/morpho/opening_area.hh (revision 1801)
+++ mln/morpho/opening_area.hh (working copy)
@@ -43,11 +43,6 @@
namespace morpho
{
- /* FIXME: The neighborhood shall not be passed as argument, but
- bound to the input image. We can also optionnaly provide a
- version of this function for regular-grid-based images where
- the neighborhood is replaced by a (user-provided) window. */
-
/// Morphological area opening.
template <typename I, typename N, typename O>
void opening_area(const Image<I>& input, const Neighborhood<N>& nbh,
Index: mln/morpho/closing_attribute.hh
--- mln/morpho/closing_attribute.hh (revision 1801)
+++ mln/morpho/closing_attribute.hh (working copy)
@@ -45,11 +45,6 @@
namespace morpho
{
- /* FIXME: The neighborhood shall not be passed as argument, but
- bound to the input image. We can also optionnaly provide a
- version of this function for regular-grid-based images where
- the neighborhood is replaced by a (user-provided) window. */
-
/// Morphological attribute closing.
template <typename A,
typename I, typename N, typename O>
Index: mln/morpho/opening_attribute.hh
--- mln/morpho/opening_attribute.hh (revision 1801)
+++ mln/morpho/opening_attribute.hh (working copy)
@@ -45,11 +45,6 @@
namespace morpho
{
- /* FIXME: The neighborhood shall not be passed as argument, but
- bound to the input image. We can also optionnaly provide a
- version of this function for regular-grid-based images where
- the neighborhood is replaced by a (user-provided) window. */
-
/// Morphological attribute opening.
template <typename A,
typename I, typename N, typename O>
Index: tests/morpho/lena_line_graph_image_wst2.cc
--- tests/morpho/lena_line_graph_image_wst2.cc (revision 1801)
+++ tests/morpho/lena_line_graph_image_wst2.cc (working copy)
@@ -42,9 +42,9 @@
between the values on the nodes adjacent to the edge, so as to
create a (norm of the gradient) ``between the pixels'' of the
input image;
- \li (insert an minima-killer pass here, as soon as it works on
- graph-based images);
- \li perform a WST on the line graph image;
+ \li reduce the number of minima using an area opening (computing the
+ area using the vertices, not the edges);
+ \li perform a WST on this simplified line graph image;
\li reduce the quantification of the result of the WST;
\li create an 2-D, color output image with height and width double
the size the original one, and copy the data of the input image
@@ -63,7 +63,7 @@
#include <mln/core/line_graph_neighborhood_piter.hh>
#include <mln/morpho/line_gradient.hh>
-#include <mln/morpho/closing_area.hh>
+#include <mln/morpho/closing_area_on_vertices.hh>
#include <mln/morpho/meyer_wst.hh>
#include <mln/level/stretch.hh>
@@ -114,13 +114,7 @@
nbh_t nbh;
ima_t closed_lg_ima (lg_ima.domain());
- /* FIXME: We should change the attribute closing performed here;
- instead of computing the area using the data on the lines
- (edges), whe should use the data on the pixels (vertices).
-
- The best way is probably to create another attribute-functor and
- use the algebraic_union_find canvas. */
- morpho::closing_area(lg_ima, nbh, 20, closed_lg_ima);
+ morpho::closing_area_on_vertices(lg_ima, nbh, 20, closed_lg_ima);
/*------.
| WST. |
https://svn.lrde.epita.fr/svn/oln/trunk/milena
Index: ChangeLog
from Roland Levillain <roland(a)lrde.epita.fr>
Line gradient computation.
* tests/morpho/lena_line_graph_image_wst1.cc: Fix comments.
* tests/morpho/lena_line_graph_image_wst2.cc: Fix the
initialization of the watershed image.
Reify and move the line gradient computation...
* mln/morpho/line_gradient.hh: ...here (new file).
mln/morpho/line_gradient.hh | 122 +++++++++++++++++++++++++++++
tests/morpho/lena_line_graph_image_wst1.cc | 9 --
tests/morpho/lena_line_graph_image_wst2.cc | 63 +-------------
3 files changed, 133 insertions(+), 61 deletions(-)
Index: tests/morpho/lena_line_graph_image_wst1.cc
--- tests/morpho/lena_line_graph_image_wst1.cc (revision 1800)
+++ tests/morpho/lena_line_graph_image_wst1.cc (working copy)
@@ -90,11 +90,8 @@
image2d<input_val_t> input;
io::pgm::load(input, MLN_IMG_DIR "/tiny.pgm");
- /* FIXME: Don't compute a gradient on the image2d input. Instead,
- have the values of the line graph image /behaves/ as the gradient
- of the input, i.e., edges should hold the absolute difference
- between gray levels.
- */
+ // In this test, the gradient is directly computed on the input
+ // image, not on the edges of the line graph image.
image2d<input_val_t> gradient =
morpho::gradient (input, convert::to_window(c4()));
@@ -126,7 +123,7 @@
g.add_node (p);
node_values.push_back (work(p));
/* FIXME: ``Guessing'' the id of the point just being inserted
- is bad. utill:graph<N,E>::add_node should return this
+ is bad. util:graph<N,E>::add_node should return this
id. */
points[p] = id;
++id;
Index: tests/morpho/lena_line_graph_image_wst2.cc
--- tests/morpho/lena_line_graph_image_wst2.cc (revision 1800)
+++ tests/morpho/lena_line_graph_image_wst2.cc (working copy)
@@ -62,7 +62,7 @@
#include <mln/core/line_graph_elt_neighborhood.hh>
#include <mln/core/line_graph_neighborhood_piter.hh>
-#include <mln/morpho/gradient.hh>
+#include <mln/morpho/line_gradient.hh>
#include <mln/morpho/closing_area.hh>
#include <mln/morpho/meyer_wst.hh>
#include <mln/level/stretch.hh>
@@ -98,57 +98,13 @@
image2d<input_val_t> input;
io::pgm::load(input, MLN_IMG_DIR "/small.pgm");
- /*-------------.
- | Line graph. |
- `-------------*/
-
- // FIXME: Inlined conversion, to be reifed into a routine.
-
- util::graph<point2d> g;
-
- // Points.
- /* FIXME: The need for such a structure during the conversion
- exhibits the lack of a service from util::graph (or a another,
- missing tool) regarding the retrieval of nodes' ids from
- points. */
- std::map<point2d, util::node_id> points;
- util::node_id id = 0;
-
- // Nodes.
- std::vector<input_val_t> node_values;
- mln_fwd_piter_(image2d<input_val_t>) p(input.domain());
- for_all (p)
- {
- g.add_node (p);
- node_values.push_back (input(p));
- /* FIXME: ``Guessing'' the id of the point just being inserted
- is bad. utill:graph<N,E>::add_node should return this
- id. */
- points[p] = id;
- ++id;
- }
-
- // Edges.
- window2d next_c4_win;
- next_c4_win.insert(0, 1).insert(1, 0);
- std::vector<input_val_t> edge_values;
- mln_fwd_qiter_(window2d) q(next_c4_win, p);
- for_all (p)
- for_all (q)
- if (input.has(q))
- {
- g.add_edge(points[p], points[q]);
- // The computed value is a kind of norm of the gradient
- // bewteen P and Q.
- edge_values.push_back(math::abs(input(p) - input(q)));
- }
-
- // Line graph point set.
- p_line_graph<point2d> plg(g);
+ /*----------------.
+ | Line gradient. |
+ `----------------*/
// Line graph image.
typedef line_graph_image<point2d, input_val_t> ima_t;
- ima_t lg_ima(plg, node_values, edge_values);
+ ima_t lg_ima = morpho::line_gradient(input);
/*-----------------.
| Simplification. |
@@ -180,10 +136,8 @@
// Reduce the value set to 8-bit.
typedef int_u8 wst_val_t;
typedef line_graph_image<point2d, wst_val_t> wst_ima_t;
- // FIXME: Initializations should not be that complicated.
- wst_ima_t wshed (plg,
- std::vector<wst_val_t>(node_values.size()),
- std::vector<wst_val_t>(edge_values.size()));
+ wst_ima_t wshed;
+ initialize(wshed, wshed_full);
level::stretch(wshed_full, wshed);
/*---------.
@@ -202,7 +156,7 @@
input.domain().pmax()[1] * 2);
output_t output(box2d(output_pmin, output_pmax));
level::fill(output, literal::black);
- // Reuse the piter on INPUT.
+ mln_fwd_piter_(image2d<input_val_t>) p(input.domain());
for_all(p)
{
// Equivalent of P in OUTPUT.
@@ -214,7 +168,6 @@
}
// Interpolate missing points in OUTPUT.
mln_piter_(output_t) p_out(output.domain());
-// mln_niter_(neighb2d) n_out(c4(), p_out);
for_all(p_out)
{
// Process points on even rows and odd columns
Index: mln/morpho/line_gradient.hh
--- mln/morpho/line_gradient.hh (revision 0)
+++ mln/morpho/line_gradient.hh (revision 0)
@@ -0,0 +1,122 @@
+// Copyright (C) 2008 EPITA Research and Development Laboratory
+//
+// This file is part of the Olena Library. This library is free
+// software; you can redistribute it and/or modify it under the terms
+// of the GNU General Public License version 2 as published by the
+// Free Software Foundation.
+//
+// This library is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+// General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this library; see the file COPYING. If not, write to
+// the Free Software Foundation, 51 Franklin Street, Fifth Floor,
+// Boston, MA 02111-1307, USA.
+//
+// As a special exception, you may use this file as part of a free
+// software library without restriction. Specifically, if other files
+// instantiate templates or use macros or inline functions from this
+// file, or you compile this file and link it with other files to
+// produce an executable, this file does not by itself cause the
+// resulting executable to be covered by the GNU General Public
+// License. This exception does not however invalidate any other
+// reasons why the executable file might be covered by the GNU General
+// Public License.
+
+#ifndef MLN_MORPHO_LINE_GRADIENT_HH
+# define MLN_MORPHO_LINE_GRADIENT_HH
+
+/// \file mln/morpho/line_gradient.hh
+/// \brief Conversions to mln::line_graph_image.
+
+# include <map>
+# include <vector>
+
+# include <mln/core/image2d.hh>
+# include <mln/core/line_graph_image.hh>
+
+// FIXME: Generalize to other (input) images as well (image1d,
+// image3d, etc.).
+
+namespace mln
+{
+
+ namespace morpho
+ {
+
+ /// \brief Create a line graph image representing the gradient
+ /// norm of a mln::image2d.
+ /* FIXME: Currently, the adjacency is set to 4-c and cannot be
+ changed. */
+ template <typename T>
+ mln::line_graph_image<mln::point2d, T>
+ line_gradient(const mln::image2d<T>& ima);
+
+
+# ifndef MLN_INCLUDE_ONLY
+
+ template <typename T>
+ mln::line_graph_image<mln::point2d, T>
+ line_gradient(const mln::image2d<T>& ima)
+ {
+ // FIXME: Precondition: Ensure the image is scalar.
+ typedef T value_t;
+
+ util::graph<mln::point2d> g;
+
+ // Points.
+ /* FIXME: The need for such a structure during the conversion
+ exhibits the lack of a service from util::graph (or a another,
+ missing tool) regarding the retrieval of nodes' ids from
+ points. */
+ std::map<mln::point2d, util::node_id> points;
+ util::node_id id = 0;
+
+ // Nodes.
+ std::vector<value_t> node_values;
+ mln_fwd_piter(image2d<value_t>) p(ima.domain());
+ for_all (p)
+ {
+ g.add_node (p);
+ node_values.push_back (ima(p));
+ /* FIXME: ``Guessing'' the id of the point just being inserted
+ is bad. util:graph<N,E>::add_node should return this
+ id. */
+ points[p] = id;
+ ++id;
+ }
+
+ // Edges.
+ // FIXME: The creation of this window should be generic.
+ window2d next_c4_win;
+ next_c4_win.insert(0, 1).insert(1, 0);
+ std::vector<value_t> edge_values;
+ mln_fwd_qiter_(window2d) q(next_c4_win, p);
+ for_all (p)
+ for_all (q)
+ if (ima.has(q))
+ {
+ g.add_edge(points[p], points[q]);
+ // The computed value is a norm of the gradient between P and Q.
+ edge_values.push_back(math::abs(ima(p) - ima(q)));
+ }
+
+ // Line graph point set.
+ p_line_graph<mln::point2d> plg(g);
+
+ // Line graph image.
+ typedef line_graph_image<mln::point2d, value_t> ima_t;
+ ima_t lg_ima(plg, node_values, edge_values);
+ return lg_ima;
+ }
+
+# endif // ! MLN_INCLUDE_ONLY
+
+ } // end of namespace mln::morpho
+
+} // end of namespace mln
+
+
+#endif // ! MLN_MORPHO_LINE_GRADIENT_HH
https://svn.lrde.epita.fr/svn/oln/trunk/milena
Index: ChangeLog
from Roland Levillain <roland(a)lrde.epita.fr>
Fix some memory management problems w.r.t. graph-based images.
* mln/core/p_graph.hh (mln::p_graph<P>::gr_): Handle the
underlying graph as a tracked pointer.
(mln::p_graph<P>::p_graph(const graph&))
(mln::p_graph<P>::nnodes)
(mln::p_graph<P>::point_from_id)
(mln::p_graph<P>::node1, mln::p_graph<P>::node2)
(mln::p_graph<P>::has(const psite&))
(mln::p_graph<P>::adjacent_or_equal)
(mln::operator==(const p_graph<P>&, const p_graph<P>&)):
Adjust.
* mln/core/graph_image.hh
(mln::internal::data_< graph_image<P, V> >::pg_): Handle the
p_graph by copy, as the tracked pointer takes care of the
memory management.
(mln::graph_image<P, V>::operator()(const graph_psite<P>&)):
Adjust operators.
* mln/core/graph_psite.hh,
* mln/core/graph_window_piter.hh:
Adjust.
* mln/core/graph_neighborhood_piter.hh: Likewise.
(mln::graph_neighborhood_fwd_piter<P>::adjacent_or_equal_to_p_ref_)
(mln::graph_neighborhood_bkd_piter<P>::adjacent_or_equal_to_p_ref_):
Simplify using mln::p_graph<P>::adjacent_or_equal.
* mln/core/p_line_graph.hh (mln::p_line_graph<P>::gr_): Handle the
underlying graph as a tracked pointer.
(mln::p_line_graph<P>::p_line_graph(const graph&))
(mln::p_line_graph<P>::nnodes, mln::p_line_graph<P>::nedges)
(mln::p_line_graph<P>::has(const psite&))
(mln::operator==(const p_line_graph<P>&, const p_line_graph<P>&)):
Adjust.
* mln/core/line_graph_image.hh
(mln::internal::data_< line_graph_image<P, V> >::plg_): Handle the
p_line_graph by copy, as the tracked pointer takes care of the
memory management.
(mln::line_graph_image<P, V>::operator()(const line_graph_psite<P>&)):
Adjust operators.
* mln/core/line_graph_psite.hh,
* mln/core/line_graph_neighborhood_piter.hh,
* mln/core/line_graph_window_piter.hh:
Adjust.
graph_image.hh | 11 ++-----
graph_neighborhood_piter.hh | 58 ++++-----------------------------------
graph_psite.hh | 2 -
graph_window_piter.hh | 2 -
line_graph_image.hh | 11 ++-----
line_graph_neighborhood_piter.hh | 32 +++++++++++----------
line_graph_psite.hh | 6 ++--
line_graph_window_piter.hh | 32 +++++++++++----------
p_graph.hh | 42 ++++++++++++++++------------
p_line_graph.hh | 30 ++++++++++++--------
10 files changed, 94 insertions(+), 132 deletions(-)
Index: mln/core/p_graph.hh
--- mln/core/p_graph.hh (revision 1799)
+++ mln/core/p_graph.hh (working copy)
@@ -31,6 +31,7 @@
# include <mln/core/concept/point_site.hh>
# include <mln/core/internal/point_set_base.hh>
# include <mln/accu/bbox.hh>
+# include <mln/util/tracked_ptr.hh>
# include <mln/util/graph.hh>
# include <mln/core/graph_psite.hh>
# include <mln/core/p_graph_piter.hh>
@@ -50,8 +51,13 @@
{
typedef util::graph<P> graph;
- /// Construct a graph psite set from a graph of points.
- p_graph (graph& gr);
+ /// \brief Construct a graph psite set from a graph of points.
+ ///
+ /// \param gr The graph upon which the graph psite set is built.
+ ///
+ /// \a gr is \em copied internally, so that the graph psite set is
+ /// still valid after the initial graph has been removed.
+ p_graph (const graph& gr);
/// Point_Site associated type.
typedef graph_psite<P> psite;
@@ -102,7 +108,7 @@
// FIXME: Should be private.
public:
- graph gr_;
+ util::tracked_ptr<graph> gr_;
// FIXME: (Roland) Is it really useful/needed?
/* 2007-12-19: It seems so, since graph_image must implement a method
named bbox(). Now the question is: should each image type have a
@@ -114,8 +120,8 @@
/// \brief Comparison between two mln::p_graph's.
///
- /// Two mln::p_graph's are considered equal if they have the
- /// same address.
+ /// Two mln::p_graph's are considered equal if they share the
+ /// same graph.
template <typename P>
bool
operator==(const p_graph<P>& lhs, const p_graph<P>& rhs);
@@ -125,14 +131,14 @@
template<typename P>
inline
- p_graph<P>::p_graph (util::graph<P>& gr)
- : gr_ (gr)
+ p_graph<P>::p_graph (const util::graph<P>& gr)
+ : gr_ (new util::graph<P>(gr))
{
// FIXME: Warning: if the underlying graph is updated later, this
// won't be taken into account by this p_graph!
accu::bbox<P> a;
for (unsigned i = 0; i < npoints(); ++i)
- a.take(gr_.node_data(i));
+ a.take(gr_->node_data(i));
bb_ = a.to_result();
}
@@ -149,7 +155,7 @@
std::size_t
p_graph<P>::nnodes() const
{
- return this->gr_.nnodes();
+ return this->gr_->nnodes();
}
template<typename P>
@@ -157,7 +163,7 @@
std::size_t
p_graph<P>::nedges() const
{
- return this->gr_.nedges();
+ return this->gr_->nedges();
}
template<typename P>
@@ -177,28 +183,28 @@
// Check whether P is compatible with this psite set.
(&p.pg() == this) &&
// Check that the node id of P belongs to the range of valid node ids.
- (p.id() < gr_.nnodes());
+ (p.id() < gr_->nnodes());
}
template <typename P>
const P&
p_graph<P>::point_from_id(const util::node_id& id) const
{
- return this->gr_.node_data(id);
+ return this->gr_->node_data(id);
}
template <typename P>
P&
p_graph<P>::point_from_id(const util::node_id& id)
{
- return this->gr_.node_data(id);
+ return this->gr_->node_data(id);
}
template <typename P>
const P&
p_graph<P>::node1(const util::edge_id& e) const
{
- util::node_id n1 = this->gr_.edge(e).n1();
+ util::node_id n1 = this->gr_->edge(e).n1();
return this->point_from_id(n1);
}
@@ -206,7 +212,7 @@
const P&
p_graph<P>::node2(const util::edge_id& e) const
{
- util::node_id n2 = this->gr_.edge(e).n2();
+ util::node_id n2 = this->gr_->edge(e).n2();
return this->point_from_id(n2);
}
@@ -225,7 +231,7 @@
p_graph<P>::adjacent_or_equal(const util::node_id& lhs,
const util::node_id& rhs) const
{
- // FIXME: Likewise, this is inefficient.
+ // FIXME: This is inefficient.
assert (lhs < this->npoints());
assert (rhs < this->npoints());
@@ -235,7 +241,7 @@
// Check whether the iterator is among the neighbors of P_REF_.
typedef std::vector<util::node_id> adjacency_type;
- const adjacency_type& lhs_neighbs = gr_.nodes()[lhs]->edges;
+ const adjacency_type& lhs_neighbs = gr_->nodes()[lhs]->edges;
adjacency_type::const_iterator j =
std::find (lhs_neighbs.begin(), lhs_neighbs.end(), rhs);
@@ -264,7 +270,7 @@
bool
operator==(const p_graph<P>& lhs, const p_graph<P>& rhs)
{
- return &lhs == &rhs;
+ return lhs.gr_.ptr_ == rhs.gr_.ptr_;
}
# endif // ! MLN_INCLUDE_ONLY
Index: mln/core/graph_image.hh
--- mln/core/graph_image.hh (revision 1799)
+++ mln/core/graph_image.hh (working copy)
@@ -56,12 +56,7 @@
data_(const p_graph<P>& g, const std::vector<V>& val);
std::vector<V> val_;
- /* The graph point set is handled by address, so that we can
- check the compatibility of images w.r.t. to their point
- sites. We could use a safer (and more complex) facility to
- ensure (memory) equality of line graph point sets, but using
- addresses is simple and efficient enough for the moment. */
- const p_graph<P>& pg_;
+ const p_graph<P> pg_;
};
} // end of namespace mln::internal
@@ -239,7 +234,7 @@
typename graph_image<P, V>::rvalue
graph_image<P, V>::operator()(const graph_psite<P>& p) const
{
- mln_precondition(&p.pg() == &this->data_->pg_);
+ mln_precondition(p.pg() == this->data_->pg_);
mln_precondition(p.id() < this->data_->val_.size());
return this->data_->val_[p.id()];
}
@@ -249,7 +244,7 @@
typename graph_image<P, V>::lvalue
graph_image<P, V>::operator()(const graph_psite<P>& p)
{
- mln_precondition(&p.pg() == &this->data_->pg_);
+ mln_precondition(p.pg() == this->data_->pg_);
mln_precondition(p.id() < this->data_->val_.size());
return this->data_->val_[p.id()];
}
Index: mln/core/graph_psite.hh
--- mln/core/graph_psite.hh (revision 1799)
+++ mln/core/graph_psite.hh (working copy)
@@ -139,7 +139,7 @@
bool
graph_psite<P>::is_valid_() const
{
- return pg_ && id_ < pg_->gr_.nnodes();
+ return pg_ && id_ < pg_->gr_->nnodes();
}
template<typename P>
Index: mln/core/graph_window_piter.hh
--- mln/core/graph_window_piter.hh (revision 1799)
+++ mln/core/graph_window_piter.hh (working copy)
@@ -367,7 +367,7 @@
void
graph_window_bkd_piter<P>::start()
{
- id_ = p_ref_.plg().gr_.nnodes() - 1;
+ id_ = p_ref_.plg().gr_->nnodes() - 1;
if (!adjacent_or_equal_to_p_ref_())
next_();
/* FIXME: This is redundant with the end of next_(), but we might
Index: mln/core/graph_neighborhood_piter.hh
--- mln/core/graph_neighborhood_piter.hh (revision 1799)
+++ mln/core/graph_neighborhood_piter.hh (working copy)
@@ -227,7 +227,7 @@
// FIXME: We depend too much on the implementation of util::graph
// here. The util::graph should provide the service to abstract
// these manipulations.
- return id_ < p_ref_.pg().gr_.nnodes();
+ return id_ < p_ref_.pg().gr_->nnodes();
}
template <typename P>
@@ -277,29 +277,7 @@
bool
graph_neighborhood_fwd_piter<P>::adjacent_or_equal_to_p_ref_() const
{
- // FIXME: Likewise, this is inefficient.
-
- // Check wether the iterator points to P_REF_.
- if (id_ == p_ref_.id())
- return true;
-
- // Check whether the iterator is among the neighbors of P_REF_.
- {
- // Paranoid assertion.
- assert (p_ref_.id() < p_ref_.pg().gr_.nnodes());
- // FIXME: This is too low-level. Yet another service the graph
- // should provide.
- typedef std::vector<util::node_id> adjacency_type;
- const adjacency_type& p_ref_neighbs =
- p_ref_.pg().gr_.nodes()[p_ref_.id()]->edges;
- adjacency_type::const_iterator j =
- std::find (p_ref_neighbs.begin(), p_ref_neighbs.end(), id_);
- if (j != p_ref_neighbs.end())
- return true;
- }
-
- // Otherwise, the iterator is not adjacent to P_REF_.
- return false;
+ return p_ref_.pg().adjacent_or_equal(p_ref_.id(), id_);
}
template <typename P>
@@ -310,7 +288,7 @@
// Update psite_.
psite_ = graph_psite<P>(p_ref_.pg(), id_);
// Update p_.
- p_ = p_ref_.pg().gr_.node_data(id_);
+ p_ = p_ref_.pg().gr_->node_data(id_);
}
template <typename P>
@@ -374,7 +352,7 @@
// FIXME: We depend too much on the implementation of util::graph
// here. The util::graph should provide the service to abstract
// these manipulations.
- return id_ < p_ref_.pg().gr_.nnodes();
+ return id_ < p_ref_.pg().gr_->nnodes();
}
template <typename P>
@@ -390,7 +368,7 @@
void
graph_neighborhood_bkd_piter<P>::start()
{
- id_ = p_ref_.plg().gr_.nnodes() - 1;
+ id_ = p_ref_.plg().gr_->nnodes() - 1;
if (!adjacent_or_equal_to_p_ref_())
next_();
/* FIXME: This is redundant with the end of next_(), but we might
@@ -424,29 +402,7 @@
bool
graph_neighborhood_bkd_piter<P>::adjacent_or_equal_to_p_ref_() const
{
- // FIXME: Likewise, this is inefficient.
-
- // Check wether the iterator points to P_REF_.
- if (id_ == p_ref_.id())
- return true;
-
- // Check whether the iterator is among the neighbors of P_REF_.
- {
- // Paranoid assertion.
- assert (p_ref_.id() < p_ref_.pg().gr_.nnodes());
- // FIXME: This is too low-level. Yet another service the graph
- // should provide.
- typedef std::vector<util::node_id> adjacency_type;
- const adjacency_type& p_ref_neighbs =
- p_ref_.pg().gr_.nodes()[p_ref_.id()]->edges;
- adjacency_type::const_iterator j =
- std::find (p_ref_neighbs.begin(), p_ref_neighbs.end(), id_);
- if (j != p_ref_neighbs.end())
- return true;
- }
-
- // Otherwise, the iterator is not adjacent to P_REF_.
- return false;
+ return p_ref_.pg().adjacent_or_equal(p_ref_.id(), id_);
}
template <typename P>
@@ -457,7 +413,7 @@
// Update psite_.
psite_ = graph_psite<P>(p_ref_.pg(), id_);
// Update p_.
- p_ = p_ref_.pg().gr_.node_data(id_);
+ p_ = p_ref_.pg().gr_->node_data(id_);
}
template <typename P>
Index: mln/core/p_line_graph.hh
--- mln/core/p_line_graph.hh (revision 1799)
+++ mln/core/p_line_graph.hh (working copy)
@@ -32,6 +32,7 @@
# include <mln/core/internal/point_set_base.hh>
# include <mln/accu/bbox.hh>
# include <mln/util/graph.hh>
+# include <mln/util/tracked_ptr.hh>
# include <mln/core/line_graph_psite.hh>
# include <mln/core/p_line_graph_piter.hh>
@@ -54,8 +55,13 @@
{
typedef util::graph<P> graph;
- /// Construct a line graph psite set from a graph of points.
- p_line_graph (graph& gr);
+ /// \brief Construct a line graph psite set from a graph of points.
+ ///
+ /// \param gr The graph upon which the line graph psite set is built.
+ ///
+ /// \a gr is \em copied internally, so that the line graph psite
+ /// set is still valid after the initial graph has been removed.
+ p_line_graph (const graph& gr);
/// Point_Site associated type.
typedef line_graph_psite<P> psite;
@@ -82,7 +88,7 @@
bool has(const psite& p) const;
// FIXME: Should be private.
- graph gr_;
+ util::tracked_ptr<graph> gr_;
// FIXME: (Roland) Is it really useful/needed?
/* 2007-12-19: It seems so, since graph_image must implement a
method named bbox(). Now the question is: should each image
@@ -93,8 +99,8 @@
/// \brief Comparison between two mln::p_line_graph's.
///
- /// Two mln::p_line_graph's are considered equal if they have the
- /// same address.
+ /// Two mln::p_line_graph's are considered equal if they share the
+ /// same graph.
template <typename P>
bool
operator==(const p_line_graph<P>& lhs, const p_line_graph<P>& rhs);
@@ -104,14 +110,14 @@
template<typename P>
inline
- p_line_graph<P>::p_line_graph (util::graph<P>& gr)
- : gr_ (gr)
+ p_line_graph<P>::p_line_graph (const util::graph<P>& gr)
+ : gr_ (new util::graph<P>(gr))
{
// FIXME: Warning: if the underlying graph is updated later, this
// won't be taken into account by this p_line_graph!
accu::bbox<P> a;
for (unsigned i = 0; i < nnodes(); ++i)
- a.take(gr_.node_data(i));
+ a.take(gr_->node_data(i));
bb_ = a.to_result();
}
@@ -128,7 +134,7 @@
std::size_t
p_line_graph<P>::nnodes() const
{
- return this->gr_.nnodes();
+ return this->gr_->nnodes();
}
template<typename P>
@@ -136,7 +142,7 @@
std::size_t
p_line_graph<P>::nedges() const
{
- return this->gr_.nedges();
+ return this->gr_->nedges();
}
template<typename P>
@@ -156,7 +162,7 @@
// Check whether P is compatible with this psite set.
(&p.plg() == this) &&
// Check that the edge id of P belongs to the range of valid edge ids.
- (p.id() < gr_.nedges());
+ (p.id() < gr_->nedges());
}
@@ -164,7 +170,7 @@
bool
operator==(const p_line_graph<P>& lhs, const p_line_graph<P>& rhs)
{
- return &lhs == &rhs;
+ return lhs.gr_.ptr_ == rhs.gr_.ptr_;
}
# endif // ! MLN_INCLUDE_ONLY
Index: mln/core/line_graph_image.hh
--- mln/core/line_graph_image.hh (revision 1799)
+++ mln/core/line_graph_image.hh (working copy)
@@ -72,12 +72,7 @@
std::vector<V> node_val_;
std::vector<V> edge_val_;
- /* The line graph point set is handled by address, so that we
- can check the compatibility of images w.r.t. to their point
- sites. We could use a safer (and more complex) facility to
- ensure (memory) equality of line graph point sets, but using
- addresses is simple and efficient enough for the moment. */
- const p_line_graph<P>& plg_;
+ const p_line_graph<P> plg_;
};
} // end of namespace mln::internal
@@ -273,7 +268,7 @@
typename line_graph_image<P, V>::rvalue
line_graph_image<P, V>::operator()(const line_graph_psite<P>& p) const
{
- mln_precondition(&p.plg() == &this->data_->plg_);
+ mln_precondition(p.plg() == this->data_->plg_);
mln_precondition(p.id() < this->data_->edge_val_.size());
return this->data_->edge_val_[p.id()];
}
@@ -283,7 +278,7 @@
typename line_graph_image<P, V>::lvalue
line_graph_image<P, V>::operator()(const line_graph_psite<P>& p)
{
- mln_precondition(&p.plg() == &this->data_->plg_);
+ mln_precondition(p.plg() == this->data_->plg_);
mln_precondition(p.id() < this->data_->edge_val_.size());
return this->data_->edge_val_[p.id()];
}
Index: mln/core/line_graph_psite.hh
--- mln/core/line_graph_psite.hh (revision 1799)
+++ mln/core/line_graph_psite.hh (working copy)
@@ -176,7 +176,7 @@
bool
line_graph_psite<P>::is_valid_() const
{
- return plg_ && id_ < plg_->gr_.nedges();
+ return plg_ && id_ < plg_->gr_->nedges();
}
template<typename P>
@@ -228,7 +228,7 @@
line_graph_psite<P>::first() const
{
mln_assertion(is_valid_());
- return plg().gr_.node_data(plg().gr_.edge(id_).n1());
+ return plg().gr_->node_data(plg().gr_->edge(id_).n1());
}
template<typename P>
@@ -237,7 +237,7 @@
line_graph_psite<P>::second() const
{
mln_assertion(is_valid_());
- return plg().gr_.node_data(plg().gr_.edge(id_).n2());
+ return plg().gr_->node_data(plg().gr_->edge(id_).n2());
}
Index: mln/core/line_graph_neighborhood_piter.hh
--- mln/core/line_graph_neighborhood_piter.hh (revision 1799)
+++ mln/core/line_graph_neighborhood_piter.hh (working copy)
@@ -258,7 +258,7 @@
// FIXME: We depend too much on the implementation of util::graph
// here. The util::graph should provide the service to abstract
// these manipulations.
- return id_ < p_ref_.plg().gr_.nedges();
+ return id_ < p_ref_.plg().gr_->nedges();
}
template <typename P>
@@ -315,19 +315,20 @@
// Check whether the iterator is among the neighbors of P_REF_.
{
// Paranoid assertion.
- assert (p_ref_.id() < p_ref_.plg().gr_.nedges());
+ assert (p_ref_.id() < p_ref_.plg().gr_->nedges());
/* FIXME: We should have a convenient shortcut for these
repetitive accesses to p_ref_.plg().gr_ (e.g., a method gr()
or g() in this class. */
- const typename p_line_graph<P>::graph& gr = p_ref_.plg().gr_;
+ const util::tracked_ptr<typename p_line_graph<P>::graph>& gr =
+ p_ref_.plg().gr_;
// Check whether the edge this iterator points to is adjacent to
// the one p_ref points to, by comparing their ajacent nodes.
/* FIXME: This is too low-level. Yet another service the graph
// should provide. */
- if (gr.edge(id_).n1() == gr.edge(p_ref_.id()).n1() ||
- gr.edge(id_).n1() == gr.edge(p_ref_.id()).n2() ||
- gr.edge(id_).n2() == gr.edge(p_ref_.id()).n1() ||
- gr.edge(id_).n2() == gr.edge(p_ref_.id()).n2())
+ if (gr->edge(id_).n1() == gr->edge(p_ref_.id()).n1() ||
+ gr->edge(id_).n1() == gr->edge(p_ref_.id()).n2() ||
+ gr->edge(id_).n2() == gr->edge(p_ref_.id()).n1() ||
+ gr->edge(id_).n2() == gr->edge(p_ref_.id()).n2())
return true;
}
@@ -413,7 +414,7 @@
// FIXME: We depend too much on the implementation of util::graph
// here. The util::graph should provide the service to abstract
// these manipulations.
- return id_ < p_ref_.plg().gr_.nedges();
+ return id_ < p_ref_.plg().gr_->nedges();
}
template <typename P>
@@ -429,7 +430,7 @@
void
line_graph_neighborhood_bkd_piter<P>::start()
{
- id_ = p_ref_.plg().gr_.nedges() - 1;
+ id_ = p_ref_.plg().gr_->nedges() - 1;
if (!adjacent_or_equal_to_p_ref_())
next_();
/* FIXME: This is redundant with the end of next_(), but we might
@@ -470,19 +471,20 @@
// Check whether the iterator is among the neighbors of P_REF_.
{
// Paranoid assertion.
- assert (p_ref_.id() < p_ref_.plg().gr_.nedges());
+ assert (p_ref_.id() < p_ref_.plg().gr_->nedges());
/* FIXME: We should have a convenient shortcut for these
repetitive accesses to p_ref_.plg().gr_ (e.g., a method gr()
or g() in this class. */
- const typename p_line_graph<P>::graph& gr = p_ref_.plg().gr_;
+ const util::tracked_ptr<typename p_line_graph<P>::graph>& gr =
+ p_ref_.plg().gr_;
// Check whether the edge this iterator points to is adjacent to
// the one p_ref points to, by comparing their ajacent nodes.
/* FIXME: This is too low-level. Yet another service the graph
// should provide. */
- if (gr.edge(id_).n1() == gr.edge(p_ref_.id()).n1() ||
- gr.edge(id_).n1() == gr.edge(p_ref_.id()).n2() ||
- gr.edge(id_).n2() == gr.edge(p_ref_.id()).n1() ||
- gr.edge(id_).n2() == gr.edge(p_ref_.id()).n2())
+ if (gr->edge(id_).n1() == gr->edge(p_ref_.id()).n1() ||
+ gr->edge(id_).n1() == gr->edge(p_ref_.id()).n2() ||
+ gr->edge(id_).n2() == gr->edge(p_ref_.id()).n1() ||
+ gr->edge(id_).n2() == gr->edge(p_ref_.id()).n2())
return true;
}
Index: mln/core/line_graph_window_piter.hh
--- mln/core/line_graph_window_piter.hh (revision 1799)
+++ mln/core/line_graph_window_piter.hh (working copy)
@@ -252,7 +252,7 @@
// FIXME: We depend too much on the implementation of util::graph
// here. The util::graph should provide the service to abstract
// these manipulations.
- return id_ < p_ref_.plg().gr_.nedges();
+ return id_ < p_ref_.plg().gr_->nedges();
}
template <typename P>
@@ -309,19 +309,20 @@
// Check whether the iterator is among the neighbors of P_REF_.
{
// Paranoid assertion.
- assert (p_ref_.id() < p_ref_.plg().gr_.nedges());
+ assert (p_ref_.id() < p_ref_.plg().gr_->nedges());
/* FIXME: We should have a convenient shortcut for these
repetitive accesses to p_ref_.plg().gr_ (e.g., a method gr()
or g() in this class. */
- const typename p_line_graph<P>::graph& gr = p_ref_.plg().gr_;
+ const util::tracked_ptr<typename p_line_graph<P>::graph>& gr =
+ p_ref_.plg().gr_;
// Check whether the edge this iterator points to is adjacent to
// the one p_ref points to, by comparing their ajacent nodes.
/* FIXME: This is too low-level. Yet another service the graph
// should provide. */
- if (gr.edge(id_).n1() == gr.edge(p_ref_.id()).n1() ||
- gr.edge(id_).n1() == gr.edge(p_ref_.id()).n2() ||
- gr.edge(id_).n2() == gr.edge(p_ref_.id()).n1() ||
- gr.edge(id_).n2() == gr.edge(p_ref_.id()).n2())
+ if (gr->edge(id_).n1() == gr->edge(p_ref_.id()).n1() ||
+ gr->edge(id_).n1() == gr->edge(p_ref_.id()).n2() ||
+ gr->edge(id_).n2() == gr->edge(p_ref_.id()).n1() ||
+ gr->edge(id_).n2() == gr->edge(p_ref_.id()).n2())
return true;
}
@@ -406,7 +407,7 @@
// FIXME: We depend too much on the implementation of util::graph
// here. The util::graph should provide the service to abstract
// these manipulations.
- return id_ < p_ref_.plg().gr_.nedges();
+ return id_ < p_ref_.plg().gr_->nedges();
}
template <typename P>
@@ -422,7 +423,7 @@
void
line_graph_window_bkd_piter<P>::start()
{
- id_ = p_ref_.plg().gr_.nedges() - 1;
+ id_ = p_ref_.plg().gr_->nedges() - 1;
if (!adjacent_or_equal_to_p_ref_())
next_();
/* FIXME: This is redundant with the end of next_(), but we might
@@ -463,19 +464,20 @@
// Check whether the iterator is among the neighbors of P_REF_.
{
// Paranoid assertion.
- assert (p_ref_.id() < p_ref_.plg().gr_.nedges());
+ assert (p_ref_.id() < p_ref_.plg().gr_->nedges());
/* FIXME: We should have a convenient shortcut for these
repetitive accesses to p_ref_.plg().gr_ (e.g., a method gr()
or g() in this class. */
- const typename p_line_graph<P>::graph& gr = p_ref_.plg().gr_;
+ const util::tracked_ptr<typename p_line_graph<P>::graph>& gr =
+ p_ref_.plg().gr_;
// Check whether the edge this iterator points to is adjacent to
// the one p_ref points to, by comparing their ajacent nodes.
/* FIXME: This is too low-level. Yet another service the graph
// should provide. */
- if (gr.edge(id_).n1() == gr.edge(p_ref_.id()).n1() ||
- gr.edge(id_).n1() == gr.edge(p_ref_.id()).n2() ||
- gr.edge(id_).n2() == gr.edge(p_ref_.id()).n1() ||
- gr.edge(id_).n2() == gr.edge(p_ref_.id()).n2())
+ if (gr->edge(id_).n1() == gr->edge(p_ref_.id()).n1() ||
+ gr->edge(id_).n1() == gr->edge(p_ref_.id()).n2() ||
+ gr->edge(id_).n2() == gr->edge(p_ref_.id()).n1() ||
+ gr->edge(id_).n2() == gr->edge(p_ref_.id()).n2())
return true;
}
URL: https://svn.lrde.epita.fr/svn/oln/trunk/milena
ChangeLog:
2008-03-22 Etienne FOLIO <folio(a)lrde.epita.fr>
Naive DT rebuilt. Does not compile.
* naive.cc: Uses a given function to calculate the norm.
---
naive.cc | 149 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
1 file changed, 149 insertions(+)
Index: trunk/milena/sandbox/folio/naive.cc
===================================================================
--- trunk/milena/sandbox/folio/naive.cc (revision 0)
+++ trunk/milena/sandbox/folio/naive.cc (revision 1798)
@@ -0,0 +1,149 @@
+// Copyright (C) 2007, 2008 EPITA Research and Development Laboratory (LRDE)
+//
+// This file is part of the Olena Library. This library is free
+// software; you can redistribute it and/or modify it under the terms
+// of the GNU General Public License version 2 as published by the
+// Free Software Foundation.
+//
+// This library is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+// General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this library; see the file COPYING. If not, write to
+// the Free Software Foundation, 51 Franklin Street, Fifth Floor,
+// Boston, MA 02111-1307, USA.
+//
+// As a special exception, you may use this file as part of a free
+// software library without restriction. Specifically, if other files
+// instantiate templates or use macros or inline functions from this
+// file, or you compile this file and link it with other files to
+// produce an executable, this file does not by itself cause the
+// resulting executable to be covered by the GNU General Public
+// License. This exception does not however invalidate any other
+// reasons why the executable file might be covered by the GNU General
+// Public License.
+
+/*! \file TODO
+ *
+ * \brief Defines a function that creates a distance map corresponding to a
+ * given image.
+ */
+
+#ifndef MLN_DT_NAIVE_HH
+# define MLN_DT_NAIVE_HH
+
+# include <mln/core/concept/image.hh>
+# include <mln/literal/zero.hh>
+# include <mln/accu/min.hh>
+
+namespace mln
+{
+
+ namespace dt
+ {
+
+ /*! Calculates the distance map corresponding to a given image
+ *
+ * \param[in] input_ The binary reference image.
+ * \param[in] fun_ The function used for distance aclculus.
+ * \return New distance map image.
+ *
+ * \pre \p input_ has to be initialized.
+ *
+ * \fixme Use instead of R the result type of F::operator().
+ */
+ template <typename I, typename F, typename R>
+ mln_ch_value(I, R)
+ naive(const Image<I>& input_, F& fun_);
+
+
+# ifndef MLN_INCLUDE_ONLY
+
+ // Facade.
+
+ template <typename I, typename F, typename R>
+ inline
+ mln_ch_value(I, R)
+ naive(const Image<I>& input_, F& fun_)
+ {
+ const I& input = exact(input_);
+ mln_precondition(input.has_data());
+
+ mln_ch_value(I, R) output;
+ initialize(input.domain());
+
+ mln_piter(I) p(input.domain());
+ for_all(p)
+ {
+ if (input(p))
+ output(p) = literal::zero;
+ else
+ {
+ // p is in the background so the distance has to be computed.
+ accu::min_<R> min;
+ min.init();
+
+ mln_piter(I) q(input.domain());
+ for_all(q)
+ if (input(q))
+ {
+ // q is in the object.
+// metal::vec<2, int> vp = p.to_point(), vq = q.to_point();
+// min.take(fun_(vp, vq));
+ min.take(fun_(p, q));
+ }
+ output(p) = min;
+ }
+ }
+
+ return output;
+ }
+
+# endif // ! MLN_INCLUDE_ONLY
+
+ } // end of namespace mln::dt
+
+} // end of namespace mln
+
+#endif // ! MLN_DT_NAIVE_HH
+
+#include <iostream>
+#include <mln/debug/println.hh>
+#include <mln/core/image2d.hh>
+#include <mln/level/fill.hh>
+#include <mln/norm/l2.hh>
+
+struct l2norm
+{
+ template <typename C, typename D>
+ D
+ operator()(C a, C b)
+ {
+ return mln::norm::l2(a, b);
+ }
+};
+
+int main()
+{
+ using namespace mln;
+
+ {
+ image2d<bool> ima(5,5);
+ bool vals[] = { 1, 1, 1, 0, 0,
+ 0, 0, 1, 0, 0,
+ 0, 0, 1, 0, 0,
+ 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0 };
+
+ level::fill(ima, vals);
+ debug::println(ima);
+
+ l2norm fun;
+ image2d<float> out = dt::naive(ima, fun);
+
+ std::cerr << "Distance:" << std::endl;
+ debug::println(out);
+ }
+}
URL: https://svn.lrde.epita.fr/svn/oln/trunk/milena
ChangeLog:
2008-03-22 Etienne FOLIO <folio(a)lrde.epita.fr>
Chamfer DT with nearest point calculus.
* chamfer.cc: Returns a pair containing the two maps.
---
chamfer.cc | 189 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
1 file changed, 189 insertions(+)
Index: trunk/milena/sandbox/folio/chamfer.cc
===================================================================
--- trunk/milena/sandbox/folio/chamfer.cc (revision 0)
+++ trunk/milena/sandbox/folio/chamfer.cc (revision 1797)
@@ -0,0 +1,189 @@
+// Copyright (C) 2007 EPITA Research and Development Laboratory
+//
+// This file is part of the Olena Library. This library is free
+// software; you can redistribute it and/or modify it under the terms
+// of the GNU General Public License version 2 as published by the
+// Free Software Foundation.
+//
+// This library is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+// General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this library; see the file COPYING. If not, write to
+// the Free Software Foundation, 51 Franklin Street, Fifth Floor,
+// Boston, MA 02111-1307, USA.
+//
+// As a special exception, you may use this file as part of a free
+// software library without restriction. Specifically, if other files
+// instantiate templates or use macros or inline functions from this
+// file, or you compile this file and link it with other files to
+// produce an executable, this file does not by itself cause the
+// resulting executable to be covered by the GNU General Public
+// License. This exception does not however invalidate any other
+// reasons why the executable file might be covered by the GNU General
+// Public License.
+
+#ifndef MLN_DT_CHAMFER_HH
+# define MLN_DT_CHAMFER_HH
+
+# include <mln/core/concept/image.hh>
+# include <mln/make/w_window.hh>
+
+namespace mln
+{
+
+ namespace dt
+ {
+
+ /*! Distance tranform by chamfer application.
+ *
+ * \param[in] input_ The input image.
+ * \param[in] chamfer The chamfer window to use for distance calcul.
+ * \return A pair (distance map, nearest point map).
+ *
+ * \pre \p img has to be initialized.
+ */
+ template<typename I, typename T>
+ std::pair<mln_ch_value(I, T), mln_ch_value(I, mln_point(I))>
+ chamfer(const Image<I>& input_,
+ w_window<mln_dpoint(I), T> chamfer);
+
+
+# ifndef MLN_INCLUDE_ONLY
+
+ namespace impl
+ {
+
+ /*! Computes a pass of the chamfer DT algorithm.
+ *
+ * \param[in] p Iterator on the input image to use.
+ * \param[in] chamfer The chamfer window to use for distance calcul.
+ * \param[in] input The input image.
+ * \param[out] outputDistance The distance map updated.
+ * \param[out] outputnearest The nearest points map updated.
+ */
+ template<typename Q, typename I, typename T>
+ inline
+ void
+ chamfer_pass(const w_window<mln_dpoint(I), T> chamfer,
+ const I& input,
+ mln_ch_value(I, T)& outputDistance,
+ mln_ch_value(I, mln_point(I))& outputNearest)
+ {
+ typedef w_window<mln_dpoint(I), T> W;
+
+ Q p(input.domain());
+ mln_qiter(W) q(chamfer, p);
+ for_all(p)
+ {
+ std::pair<T, mln_point(I)> min(mln_max(T), p);
+
+ for_all(q)
+ if (input.has(q) && outputDistance(q) != mln_max(T))
+ {
+ T v = outputDistance(q) + q.w();
+
+ if (v < min.first)
+ {
+ min.first = v;
+ min.second = outputNearest(q);
+ }
+ }
+
+ if (min.first < outputDistance(p))
+ {
+ outputDistance(p) = min.first;
+ outputNearest(p) = min.second;
+ }
+ }
+ }
+
+ } // end of namespace mln::dt::impl
+
+
+
+ // Facade.
+
+ template<typename I, typename T>
+ inline
+ std::pair<mln_ch_value(I, T), mln_ch_value(I, mln_point(I))>
+ chamfer(const Image<I>& input_,
+ w_window<mln_dpoint(I), T> chamfer)
+ {
+ typedef w_window<mln_dpoint(I), T> W;
+
+ const I& input = exact(input_);
+ mln_precondition(input.has_data());
+
+ mln_ch_value(I, T) outputDistance;
+ initialize(outputDistance, input);
+
+ mln_ch_value(I, mln_point(I)) outputNearest;
+ initialize(outputNearest, input);
+
+ // Initialization.
+ {
+ mln_fwd_piter(I) p(input.domain());
+ for_all(p)
+ {
+ outputDistance(p) = input(p) ? literal::zero : mln_max(T);
+ outputNearest(p) = p;
+ }
+ }
+
+ // First pass.
+ impl::chamfer_pass<mln_fwd_piter(I)>
+ (chamfer, input, outputDistance, outputNearest);
+
+ chamfer.sym();
+
+ // Second pass.
+ impl::chamfer_pass<mln_bkd_piter(I)>
+ (chamfer, input, outputDistance, outputNearest);
+
+ return std::pair<mln_ch_value(I, T), mln_ch_value(I, mln_point(I))>
+ (outputDistance, outputNearest);
+ }
+
+# endif // ! MLN_INCLUDE_ONLY
+
+ } // end of namespace mln::dt
+
+} // end of namespace mln
+
+#endif // ! MLN_DT_CHAMFER_HH
+
+#include <iostream>
+#include <mln/debug/println.hh>
+#include <mln/core/image2d.hh>
+#include <mln/make/win_chamfer.hh>
+#include <mln/level/fill.hh>
+
+int main()
+{
+ using namespace mln;
+
+ w_window2d_int chamfer = make::mk_chamfer_3x3_int<3, 4>();
+
+ {
+ image2d<bool> ima(5,5);
+ bool vals[] = { 1, 1, 1, 0, 0,
+ 0, 0, 1, 0, 0,
+ 0, 0, 1, 0, 0,
+ 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0 };
+
+ level::fill(ima, vals);
+ debug::println(ima);
+
+ std::pair<image2d<int>, image2d<mln_point_(image2d<bool>)> >
+ out = dt::chamfer(ima, chamfer);
+
+ std::cerr << "Distance:" << std::endl;
+ debug::println(out.first);
+ std::cerr << "PPP:" << std::endl;
+ debug::println(out.second);
+ }
+}