* mln/topo/n_face.hh
(mln::topo::n_face<D>::lower_dim_adj_faces)
(mln::topo::n_face<D>::higher_dim_adj_faces):
Return algebraic n-faces.
(mln::topo::make_n_face): Remove useless helper.
(mln::topo::edge(const n_face<0, D>&, const n_face<0, D>&)):
Have this helper return an algebraic 1-face, and move it...
* milena/mln/topo/face.hh: ...here.
---
milena/ChangeLog | 13 ++++++
milena/mln/topo/algebraic_n_face.hh | 53 +++++++++++++++++++++++
milena/mln/topo/n_face.hh | 78 +++-------------------------------
3 files changed, 73 insertions(+), 71 deletions(-)
diff --git a/milena/ChangeLog b/milena/ChangeLog
index 8f68af8..334b083 100644
--- a/milena/ChangeLog
+++ b/milena/ChangeLog
@@ -1,5 +1,18 @@
2008-10-23 Roland Levillain <roland(a)lrde.epita.fr>
+ Adjust (non algebraic) n-faces.
+
+ * mln/topo/n_face.hh
+ (mln::topo::n_face<D>::lower_dim_adj_faces)
+ (mln::topo::n_face<D>::higher_dim_adj_faces):
+ Return algebraic n-faces.
+ (mln::topo::make_n_face): Remove useless helper.
+ (mln::topo::edge(const n_face<0, D>&, const n_face<0, D>&)):
+ Have this helper return an algebraic 1-face, and move it...
+ * milena/mln/topo/face.hh: ...here.
+
+2008-10-23 Roland Levillain <roland(a)lrde.epita.fr>
+
Add an algebraic (oriented) face handle (descriptor).
* mln/topo/algebraic_face.hh: New.
diff --git a/milena/mln/topo/algebraic_n_face.hh b/milena/mln/topo/algebraic_n_face.hh
index fbfccf4..9228963 100644
--- a/milena/mln/topo/algebraic_n_face.hh
+++ b/milena/mln/topo/algebraic_n_face.hh
@@ -134,6 +134,32 @@ namespace mln
operator<<(std::ostream& ostr, const algebraic_n_face<N, D>& f);
+ /// \brief Helpers
+ /// \{
+
+ /** \brief Return the algebraic 1-face (edge) linking the 0-faces
+ (vertices) \a f1 and \a f2. If there is no 1-face between \a
+ f1 and \a f2, return an invalid 1-face.
+
+ \pre \a f1 and \a f2 must belong to the same complex.
+
+ Note: this routine assumes the complex is not degenerated, i.e,
+ \li it does not check that \a f1 and \a f2 are the only
+ 0-faces adjacent to an hypothetical 1-face; it just checks
+ that \a f1 and \a f2 <em>share</em> a common 1-face;
+
+ \li if there are several ajacent 1-faces shared by \a f1 and
+ \a f2 (if the complex is ill-formed), there is no
+ guarantee on the returned 1-face (the current
+ implementation return the first 1-face found, but client
+ code should not rely on this implementation-defined
+ behavior). */
+ template <unsigned D>
+ algebraic_n_face<1, D>
+ edge(const n_face<0, D>& f1, const n_face<0, D>& f2);
+ /// \}
+
+
# ifndef MLN_INCLUDE_ONLY
@@ -253,6 +279,33 @@ namespace mln
<< ", id = " << f.face_id() << ", sign = " << f.sign()<< ')';
}
+ /*----------.
+ | Helpers. |
+ `----------*/
+
+ template <unsigned D>
+ algebraic_n_face<1, D>
+ edge(const n_face<0, D>& f1, const n_face<0, D>& f2)
+ {
+ typedef std::vector< algebraic_n_face<0, D> > n0_faces_t;
+ typedef std::vector< algebraic_n_face<1, D> > n1_faces_t;
+
+ n1_faces_t f1_adj_edges = f1.higher_dim_adj_faces();
+ for (typename n1_faces_t::const_iterator e = f1_adj_edges.begin();
+ e != f1_adj_edges.end(); ++e)
+ {
+ n0_faces_t e_adj_vertices = e->lower_dim_adj_faces();
+ for (typename n0_faces_t::const_iterator w = e_adj_vertices.begin();
+ w != e_adj_vertices.end(); ++w)
+ if (*w == f2)
+ // E is the edge linking F1 and F2.
+ return *e;
+ }
+
+ // If no shared edge was found, retun an empty (invalid) 1-face.
+ return algebraic_n_face<1, D>();
+ }
+
# endif // ! MLN_INCLUDE_ONLY
} // end of namespace mln::topo
diff --git a/milena/mln/topo/n_face.hh b/milena/mln/topo/n_face.hh
index 68a9512..2f8a7e0 100644
--- a/milena/mln/topo/n_face.hh
+++ b/milena/mln/topo/n_face.hh
@@ -46,6 +46,7 @@ namespace mln
// Forward declaration.
template <unsigned D> class complex;
template <unsigned N, unsigned D> class face_data;
+ template <unsigned N, unsigned D> class algebraic_n_face;
/*---------.
@@ -57,8 +58,9 @@ namespace mln
/// Contrary to an mln::topo::face, the dimension of an
/// mln::topo::n_face is fixed.
template <unsigned N, unsigned D>
- struct n_face
+ class n_face
{
+ public:
// The type of the complex this handle points to.
typedef complex<D> complex_type;
@@ -98,9 +100,9 @@ namespace mln
/* FIXME: We should not provide lower_dim_adj_faces() when N ==
0 nor higher_dim_adj_faces() when N == D. */
/// \Return an array of face handles pointing to adjacent (n-1)-faces.
- std::vector< n_face<N - 1, D> > lower_dim_adj_faces() const;
+ std::vector< algebraic_n_face<N - 1, D> > lower_dim_adj_faces() const;
/// Return an array of face handles pointing to adjacent (n+1)-faces.
- std::vector< n_face<N + 1, D> > higher_dim_adj_faces() const;
+ std::vector< algebraic_n_face<N + 1, D> > higher_dim_adj_faces() const;
/// \}
private:
@@ -114,12 +116,6 @@ namespace mln
};
- /// Create a handle for \p N-face of a \p D-complex.
- template <unsigned N, unsigned D>
- n_face<N, D>
- make_n_face(const complex<D>& c, unsigned face_id);
-
-
/// Comparison of two instances of mln::topo::n_face.
/// \{
@@ -155,31 +151,6 @@ namespace mln
operator<<(std::ostream& ostr, const n_face<N, D>& f);
- /// \brief Helpers
- /// \{
-
- /** \brief Return the 1-face (edge) linking the 0-faces (vertices)
- \a f1 and \a f2. If there is no 1-face between \a f1 and \a
- f2, return an invalid 1-face.
-
- \pre \a f1 and \a f2 must belong to the same complex.
-
- Note: this routine assumes the complex is not degenerated, i.e,
- \li it does not check that \a f1 and \a f2 are the only
- 0-faces adjacent to an hypothetical 1-face; it just checks
- that \a f1 and \a f2 <em>share</em> a common 1-face;
-
- \li if there are several ajacent 1-faces shared by \a f1 and
- \a f2 (if the complex is ill-formed), there is no
- guarantee on the returned 1-face (the current
- implementation return the first 1-face found, but client
- code should not rely on this implementation-defined
- behavior). */
- template <unsigned D>
- n_face<1, D> edge(const n_face<0, D>& f1, const n_face<0, D>& f2);
- /// \}
-
-
# ifndef MLN_INCLUDE_ONLY
@@ -285,7 +256,7 @@ namespace mln
template <unsigned N, unsigned D>
inline
- std::vector< n_face<N - 1, D> >
+ std::vector< algebraic_n_face<N - 1, D> >
n_face<N, D>::lower_dim_adj_faces() const
{
mln_precondition(N > 0);
@@ -295,7 +266,7 @@ namespace mln
template <unsigned N, unsigned D>
inline
- std::vector< n_face<N + 1, D> >
+ std::vector< algebraic_n_face<N + 1, D> >
n_face<N, D>::higher_dim_adj_faces() const
{
mln_precondition(N <= D);
@@ -306,15 +277,6 @@ namespace mln
template <unsigned N, unsigned D>
inline
- n_face<N, D>
- make_n_face(const complex<D>& c, unsigned face_id)
- {
- return n_face<N, D>(&c, face_id);
- }
-
-
- template <unsigned N, unsigned D>
- inline
bool
operator==(const n_face<N, D>& lhs, const n_face<N, D>& rhs)
{
@@ -353,32 +315,6 @@ namespace mln
<< ", id = " << f.face_id() << ')';
}
- /*----------.
- | Helpers. |
- `----------*/
-
- template <unsigned D>
- n_face<1, D> edge(const n_face<0, D>& f1, const n_face<0, D>& f2)
- {
- typedef std::vector< n_face<0, D> > n0_faces_t;
- typedef std::vector< n_face<1, D> > n1_faces_t;
-
- n1_faces_t f1_adj_edges = f1.higher_dim_adj_faces();
- for (typename n1_faces_t::const_iterator e = f1_adj_edges.begin();
- e != f1_adj_edges.end(); ++e)
- {
- n0_faces_t e_adj_vertices = e->lower_dim_adj_faces();
- for (typename n0_faces_t::const_iterator w = e_adj_vertices.begin();
- w != e_adj_vertices.end(); ++w)
- if (*w == f2)
- // E is the edge linking F1 and F2.
- return *e;
- }
-
- // If no shared edge was found, retun an empty (invalid) 1-face.
- return n_face<1, D>();
- }
-
# endif // ! MLN_INCLUDE_ONLY
} // end of namespace mln::topo
--
1.5.6.5
* mln/topo/algebraic_face.hh: New.
---
milena/ChangeLog | 6 +
milena/mln/topo/algebraic_face.hh | 285 +++++++++++++++++++++++++++++++++++++
2 files changed, 291 insertions(+), 0 deletions(-)
create mode 100644 milena/mln/topo/algebraic_face.hh
diff --git a/milena/ChangeLog b/milena/ChangeLog
index 57d15fd..8f68af8 100644
--- a/milena/ChangeLog
+++ b/milena/ChangeLog
@@ -1,5 +1,11 @@
2008-10-23 Roland Levillain <roland(a)lrde.epita.fr>
+ Add an algebraic (oriented) face handle (descriptor).
+
+ * mln/topo/algebraic_face.hh: New.
+
+2008-10-23 Roland Levillain <roland(a)lrde.epita.fr>
+
Add an algebraic (oriented) n-face handle (descriptor).
* mln/topo/algebraic_n_face.hh: New.
diff --git a/milena/mln/topo/algebraic_face.hh b/milena/mln/topo/algebraic_face.hh
new file mode 100644
index 0000000..79f9116
--- /dev/null
+++ b/milena/mln/topo/algebraic_face.hh
@@ -0,0 +1,285 @@
+// Copyright (C) 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.
+// reasons why the executable file might be covered by the GNU General
+// Public License.
+
+#ifndef MLN_TOPO_ALGEBRAIC_FACE_HH
+# define MLN_TOPO_ALGEBRAIC_FACE_HH
+
+/// \file mln/topo/algebraic_face.hh
+/// \brief Algebraic face of a complex.
+
+#include <mln/topo/face.hh>
+
+
+namespace mln
+{
+
+ namespace topo
+ {
+
+ // Forward declarations.
+ template <unsigned D> class complex;
+ template <unsigned N, unsigned D> class n_face;
+ template <unsigned N, unsigned D> class face_data;
+
+
+ /*-------.
+ | Face. |
+ `-------*/
+
+ /// \brief Algebraic face handle in a complex; the face dimension
+ /// is dynamic.
+ ///
+ /// Contrary to an mln::topo::algebraic_n_face, the dimension of an
+ /// mln::topo::algebraic_face is not fixed.
+ template <unsigned D>
+ struct algebraic_face : public face<D>
+ {
+ typedef face<D> super_;
+
+ public:
+ // The type of the complex this handle points to.
+ typedef complex<D> complex_type;
+
+ /// Build a non-initialized algebraic face handle.
+ algebraic_face();
+ /// Build an algebraic face handle from \a complex and \a face_id.
+ algebraic_face(complex<D>& complex, unsigned n, unsigned face_id,
+ bool sign);
+ /// Build an algebraic face handle from an mln::face.
+ algebraic_face(const face<D>& f, bool sign);
+
+ /// Build a face handle from an mln::topo::algebraic_n_face.
+ template <unsigned N>
+ algebraic_face(const algebraic_n_face<N, D>& f);
+
+ /// Accessors.
+ /// \{
+ /// Return the sign of this face.
+ bool sign() const;
+ /// Set the sign of this face.
+ void set_sign(bool sign);
+ /// \}
+
+ private:
+ /// The sign of this algebraic face.
+ bool sign_;
+ };
+
+
+ /// Create an algebraic face handle of a \p D-complex.
+ template <unsigned D>
+ algebraic_face<D>
+ make_algebraic_face(const face<D>& f, bool sign);
+
+
+ /// Inversion operators.
+ /// \{
+ template <unsigned D>
+ algebraic_face<D>
+ operator-(const face<D>& f);
+
+ template <unsigned D>
+ algebraic_face<D>
+ operator-(const algebraic_face<D>& f);
+ /// \}
+
+
+ /// Comparison of two instances of mln::topo::algebraic_face.
+ /// \{
+
+ /// \brief Is \a lhs equal to \a rhs?
+ ///
+ /// \pre Arguments \a lhs and \a rhs must belong to the same
+ /// mln::topo::complex.
+ template <unsigned D>
+ bool operator==(const algebraic_face<D>& lhs,
+ const algebraic_face<D>& rhs);
+
+ /// \brief Is \a lhs different from \a rhs?
+ ///
+ /// \pre Arguments \a lhs and \a rhs must belong to the same
+ /// mln::topo::complex.
+ template <unsigned D>
+ bool operator!=(const algebraic_face<D>& lhs,
+ const algebraic_face<D>& rhs);
+
+ /// \brief Is \a lhs ``less'' than \a rhs?
+ ///
+ /// This comparison is required by algorithms sorting algebraic
+ /// face handles.
+ ///
+ /// \pre Arguments \a lhs and \a rhs must belong to the same
+ /// mln::topo::complex.
+ /// \pre Arguments \a lhs and \a rhs must have the same dimension.
+ template <unsigned D>
+ bool operator< (const algebraic_face<D>& lhs,
+ const algebraic_face<D>& rhs);
+
+ /// \}
+
+
+ /// Print an mln::topo::algebraic_face.
+ template <unsigned D>
+ std::ostream&
+ operator<<(std::ostream& ostr, const algebraic_face<D>& f);
+
+
+
+# ifndef MLN_INCLUDE_ONLY
+
+ template <unsigned D>
+ inline
+ algebraic_face<D>::algebraic_face()
+ : super_(), sign_(true)
+ {
+ }
+
+ template <unsigned D>
+ inline
+ algebraic_face<D>::algebraic_face(complex<D>& c, unsigned n,
+ unsigned face_id, bool sign)
+ : super_(c, n, face_id), sign_(sign)
+ {
+ // Ensure N is compatible with D.
+ mln_precondition(n <= D);
+ }
+
+ template <unsigned D>
+ inline
+ algebraic_face<D>::algebraic_face(const face<D>& f, bool sign)
+ : super_(f), sign_(sign)
+ {
+ // Ensure N is compatible with D.
+ mln_precondition(f.n() <= D);
+ }
+
+ template <unsigned D>
+ template <unsigned N>
+ inline
+ algebraic_face<D>::algebraic_face(const algebraic_n_face<N, D>& f)
+ : super_(f), sign_(f.sign())
+ {
+ // Ensure N is compatible with D.
+ metal::bool_< N <= D >::check();
+ }
+
+
+ template <unsigned D>
+ inline
+ bool
+ algebraic_face<D>::sign() const
+ {
+ return sign_;
+ }
+
+ template <unsigned D>
+ inline
+ void
+ algebraic_face<D>::set_sign(bool sign)
+ {
+ sign_ = sign;
+ }
+
+
+ template <unsigned D>
+ algebraic_face<D>
+ make_algebraic_face(const face<D>& f, bool sign)
+ {
+ return algebraic_face<D>(f, sign);
+ }
+
+
+ template <unsigned D>
+ algebraic_face<D>
+ operator-(const face<D>& f)
+ {
+ return algebraic_face<D>(f, false);
+ }
+
+ template <unsigned D>
+ algebraic_face<D>
+ operator-(const algebraic_face<D>& f)
+ {
+ algebraic_face<D> f2(f);
+ f2.set_sign(!f.sign());
+ return f2;
+ }
+
+
+ template <unsigned D>
+ inline
+ bool
+ operator==(const algebraic_face<D>& lhs, const algebraic_face<D>& rhs)
+ {
+ // Ensure LHS and RHS belong to the same complex.
+ mln_precondition(lhs.cplx() == rhs.cplx());
+ return
+ lhs.n() == rhs.n() &&
+ lhs.face_id() == rhs.face_id() &&
+ lhs.sign() == rhs.sign();
+ }
+
+ template <unsigned D>
+ inline
+ bool
+ operator!=(const algebraic_face<D>& lhs, const algebraic_face<D>& rhs)
+ {
+ // Ensure LHS and RHS belong to the same complex.
+ mln_precondition(lhs.cplx() == rhs.cplx());
+ return !(lhs == rhs);
+ }
+
+ template <unsigned D>
+ inline
+ bool
+ operator< (const algebraic_face<D>& lhs, const algebraic_face<D>& rhs)
+ {
+ // Ensure LHS and RHS belong to the same complex.
+ mln_precondition(lhs.cplx() == rhs.cplx());
+ // Ensure LHS and RHS have the same dimension.
+ mln_precondition(lhs.n() == rhs.n());
+ return lhs.face_id() < rhs.face_id();
+ }
+
+
+ template <unsigned D>
+ inline
+ std::ostream&
+ operator<<(std::ostream& ostr, const algebraic_face<D>& f)
+ {
+ return
+ ostr << "(cplx = " << f.cplx().addr() << ", dim = " << f.n()
+ << ", id = " << f.face_id() << ", sign = " << f.sign()<< ')';
+ }
+
+# endif // ! MLN_INCLUDE_ONLY
+
+ } // end of namespace mln::topo
+
+} // end of namespace mln
+
+#endif // ! MLN_TOPO_FACE_HH
--
1.5.6.5
* mln/topo/algebraic_n_face.hh: New.
---
milena/ChangeLog | 6 +
milena/mln/topo/algebraic_n_face.hh | 262 +++++++++++++++++++++++++++++++++++
2 files changed, 268 insertions(+), 0 deletions(-)
create mode 100644 milena/mln/topo/algebraic_n_face.hh
diff --git a/milena/ChangeLog b/milena/ChangeLog
index 6bf9126..57d15fd 100644
--- a/milena/ChangeLog
+++ b/milena/ChangeLog
@@ -1,5 +1,11 @@
2008-10-23 Roland Levillain <roland(a)lrde.epita.fr>
+ Add an algebraic (oriented) n-face handle (descriptor).
+
+ * mln/topo/algebraic_n_face.hh: New.
+
+2008-10-23 Roland Levillain <roland(a)lrde.epita.fr>
+
Fix a bug in OFF file saving.
* mln/io/off/save.hh
diff --git a/milena/mln/topo/algebraic_n_face.hh b/milena/mln/topo/algebraic_n_face.hh
new file mode 100644
index 0000000..fbfccf4
--- /dev/null
+++ b/milena/mln/topo/algebraic_n_face.hh
@@ -0,0 +1,262 @@
+// Copyright (C) 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.
+// reasons why the executable file might be covered by the GNU General
+// Public License.
+
+#ifndef MLN_TOPO_ALGEBRAIC_N_FACE_HH
+# define MLN_TOPO_ALGEBRAIC_N_FACE_HH
+
+/// \file mln/topo/algebraic_n_face.hh
+/// \brief Algebraic n-face of a complex.
+
+#include <mln/topo/n_face.hh>
+
+
+namespace mln
+{
+
+ namespace topo
+ {
+
+ /*-------------------.
+ | Algebraic n-Face. |
+ `-------------------*/
+
+ /// \brief Algebraic \p N-face handle in a complex.
+ ///
+ /// Contrary to an mln::topo::algebraic_face, the dimension of an
+ /// mln::topo::algebraic_n_face is fixed.
+ template <unsigned N, unsigned D>
+ class algebraic_n_face : public n_face<N, D>
+ {
+ typedef n_face<N, D> super_;
+
+ public:
+ /// Build a non-initialized algebraic face handle.
+ algebraic_n_face();
+ /// Build an algebraic face handle from \a complex and \a face_id.
+ algebraic_n_face(complex<D>& complex, unsigned face_id, bool sign);
+ /// Build an algebraic face handle from an mln::n_face.
+ algebraic_n_face(const n_face<N, D>& f, bool sign);
+
+ /// Accessors.
+ /// \{
+ /// Return the sign of this face.
+ bool sign() const;
+ /// Set the sign of this face.
+ void set_sign(bool sign);
+ /// \}
+
+ private:
+ bool sign_;
+ };
+
+
+ /// Create an algebraic \p N-face handle of a \p D-complex.
+ template <unsigned N, unsigned D>
+ algebraic_n_face<N, D>
+ make_algebraic_n_face(const n_face<N, D>& f, bool sign);
+
+
+ /// Inversion operators.
+ /// \{
+ template <unsigned N, unsigned D>
+ algebraic_n_face<N, D>
+ operator-(const n_face<N, D>& f);
+
+ template <unsigned N, unsigned D>
+ algebraic_n_face<N, D>
+ operator-(const algebraic_n_face<N, D>& f);
+ /// \}
+
+
+ /// Comparison of two instances of mln::topo::algebraic_n_face.
+ /// \{
+
+ /// \brief Is \a lhs equal to \a rhs?
+ ///
+ /// \pre Arguments \a lhs and \a rhs must belong to the same
+ /// mln::topo::complex.
+ template <unsigned N, unsigned D>
+ bool
+ operator==(const algebraic_n_face<N, D>& lhs,
+ const algebraic_n_face<N, D>& rhs);
+
+ /// \brief Is \a lhs different from \a rhs?
+ ///
+ /// \pre Arguments \a lhs and \a rhs must belong to the same
+ /// mln::topo::complex.
+ template <unsigned N, unsigned D>
+ bool
+ operator!=(const algebraic_n_face<N, D>& lhs,
+ const algebraic_n_face<N, D>& rhs);
+
+ /// \brief Is \a lhs ``less'' than \a rhs?
+ ///
+ /// This comparison is required by algorithms sorting algebraic
+ /// face handles.
+ ///
+ /// \pre Arguments \a lhs and \a rhs must belong to the same
+ /// mln::topo::complex.
+ template <unsigned N, unsigned D>
+ bool
+ operator< (const algebraic_n_face<N, D>& lhs,
+ const algebraic_n_face<N, D>& rhs);
+
+ /// \}
+
+
+ /// Print an mln::topo::algebraic_n_face.
+ template <unsigned N, unsigned D>
+ std::ostream&
+ operator<<(std::ostream& ostr, const algebraic_n_face<N, D>& f);
+
+
+
+# ifndef MLN_INCLUDE_ONLY
+
+ template <unsigned N, unsigned D>
+ inline
+ algebraic_n_face<N, D>::algebraic_n_face()
+ : super_(), sign_(true)
+ {
+ // Ensure N is compatible with D.
+ metal::bool_< N <= D >::check();
+ mln_postcondition(!this->is_valid());
+ }
+
+ template <unsigned N, unsigned D>
+ inline
+ algebraic_n_face<N, D>::algebraic_n_face(complex<D>& c, unsigned face_id,
+ bool sign)
+ : super_(c, face_id), sign_(sign)
+ {
+ // Ensure N is compatible with D.
+ metal::bool_< N <= D >::check();
+ }
+
+ template <unsigned N, unsigned D>
+ inline
+ algebraic_n_face<N, D>::algebraic_n_face(const n_face<N, D>& f, bool sign)
+ : super_(f), sign_(sign)
+ {
+ // Ensure N is compatible with D.
+ metal::bool_< N <= D >::check();
+ }
+
+
+ template <unsigned N, unsigned D>
+ inline
+ bool
+ algebraic_n_face<N, D>::sign() const
+ {
+ return sign_;
+ }
+
+ template <unsigned N, unsigned D>
+ inline
+ void
+ algebraic_n_face<N, D>::set_sign(bool sign)
+ {
+ sign_ = sign;
+ }
+
+
+ template <unsigned N, unsigned D>
+ algebraic_n_face<N, D>
+ make_algebraic_n_face(const n_face<N, D>& f, bool sign)
+ {
+ return algebraic_n_face<N, D>(f, sign);
+ }
+
+
+ template <unsigned N, unsigned D>
+ algebraic_n_face<N, D>
+ operator-(const n_face<N, D>& f)
+ {
+ return algebraic_n_face<N, D>(f, false);
+ }
+
+ template <unsigned N, unsigned D>
+ algebraic_n_face<N, D>
+ operator-(const algebraic_n_face<N, D>& f)
+ {
+ algebraic_n_face<N, D> f2(f);
+ f2.set_sign(!f.sign());
+ return f2;
+ }
+
+
+ template <unsigned N, unsigned D>
+ inline
+ bool
+ operator==(const algebraic_n_face<N, D>& lhs,
+ const algebraic_n_face<N, D>& rhs)
+ {
+ // Ensure LHS and RHS belong to the same complex.
+ mln_precondition(lhs.cplx() == rhs.cplx());
+ return lhs.face_id() == rhs.face_id() && lhs.sign() == rhs.sign();
+ }
+
+ template <unsigned N, unsigned D>
+ inline
+ bool
+ operator!=(const algebraic_n_face<N, D>& lhs,
+ const algebraic_n_face<N, D>& rhs)
+ {
+ // Ensure LHS and RHS belong to the same complex.
+ mln_precondition(lhs.cplx() == rhs.cplx());
+ return !(lhs == rhs);
+ }
+
+ template <unsigned N, unsigned D>
+ inline
+ bool
+ operator< (const algebraic_n_face<N, D>& lhs,
+ const algebraic_n_face<N, D>& rhs)
+ {
+ // Ensure LHS and RHS belong to the same complex.
+ mln_precondition(lhs.cplx() == rhs.cplx());
+ return lhs.face_id() < rhs.face_id();
+ }
+
+
+ template <unsigned N, unsigned D>
+ inline
+ std::ostream&
+ operator<<(std::ostream& ostr, const algebraic_n_face<N, D>& f)
+ {
+ return
+ ostr << "(cplx = " << f.cplx().addr() << ", dim = " << f.n()
+ << ", id = " << f.face_id() << ", sign = " << f.sign()<< ')';
+ }
+
+# endif // ! MLN_INCLUDE_ONLY
+
+ } // end of namespace mln::topo
+
+} // end of namespace mln
+
+#endif // ! MLN_TOPO_ALGEBRAIC_N_FACE_HH
--
1.5.6.5