dust3d/thirdparty/cgal/CGAL-4.13/include/CGAL/Nef_2/PM_checker.h

292 lines
10 KiB
C
Raw Normal View History

// Copyright (c) 1997-2000 Max-Planck-Institute Saarbruecken (Germany).
// All rights reserved.
//
// This file is part of CGAL (www.cgal.org).
// You can redistribute it and/or modify it under the terms of the GNU
// General Public License as published by the Free Software Foundation,
// either version 3 of the License, or (at your option) any later version.
//
// Licensees holding a valid commercial license may use this file in
// accordance with the commercial license agreement provided with the software.
//
// This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE
// WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
//
// $URL$
// $Id$
// SPDX-License-Identifier: GPL-3.0+
//
//
// Author(s) : Michael Seel <seel@mpi-sb.mpg.de>
#ifndef CGAL_PM_CHECKER_H
#define CGAL_PM_CHECKER_H
#include <CGAL/license/Nef_2.h>
#include <CGAL/basic.h>
#include <CGAL/Unique_hash_map.h>
#include <CGAL/Nef_2/PM_const_decorator.h>
namespace CGAL {
/*{\Moptions outfile=PM_checker.man }*/
/*{\Manpage {PM_checker}{PMCDEC,GEOM}{Plane map checking}{}}*/
/*{\Mdefinition An instance |\Mvar| of the data type |\Mname| is a
decorator to check the structure of a plane map. It is generic with
respect to two template concepts. |PMCDEC| has to be a decorator
model of our |PM_const_decorator| concept. |GEOM| has to be a model of
our geometry kernel concept.}*/
/*{\Mgeneralization PM_const_decorator}*/
template <typename PMCDEC, typename GEOM>
class PM_checker : public PMCDEC
{ typedef PMCDEC Base;
const GEOM& K;
public:
/*{\Mtypes 3}*/
typedef PMCDEC PM_const_decorator;
/*{\Mtypemember equals |PMCDEC|.}*/
typedef typename PMCDEC::Plane_map Plane_map;
/*{\Mtypemember equals |PMCDEC::Plane_map|, the underlying plane map type.}*/
typedef GEOM Geometry;
/*{\Mtypemember equals |GEOM|. Add link to GEOM concept.\\
\precond |Geometry::Point_2| equals |Plane_map::Point|. }*/
typedef typename GEOM::Point_2 Point;
typedef typename GEOM::Direction_2 Direction;
typedef typename Base::Vertex_const_handle Vertex_const_handle;
typedef typename Base::Halfedge_const_handle Halfedge_const_handle;
typedef typename Base::Vertex_const_iterator Vertex_const_iterator;
typedef typename Base::Halfedge_const_iterator Halfedge_const_iterator;
typedef typename Base::Halfedge_around_vertex_const_circulator Halfedge_around_vertex_const_circulator;
typedef typename Base::Halfedge_around_face_const_circulator Halfedge_around_face_const_circulator;
using Base::clear;
using Base::vertices_begin;
using Base::vertices_end;
using Base::halfedges_begin;
using Base::halfedges_end;
using Base::faces_begin;
using Base::faces_end;
using Base::number_of_vertices;
using Base::number_of_halfedges;
using Base::number_of_edges;
using Base::number_of_faces;
using Base::number_of_connected_components;
using Base::check_integrity_and_topological_planarity;
/*{\Mtext Iterators, handles, and circulators are inherited from
|PM_const_decorator|.}*/
/*{\Mcreation 3}*/
PM_checker(Plane_map& P, const Geometry& k = Geometry()) :
Base(P), K(k) {}
/*{\Mcreate constructs a plane map checker working on |P| with
geometric predicates used from |k|.}*/
PM_checker(const Base& D, const Geometry& k = Geometry()) :
Base(D), K(k) {}
/*{\Moperations 2 }*/
Direction direction(Halfedge_const_handle e) const
{ return K.construct_direction(
point(source(e)),point(target(e))); }
bool is_forward(Halfedge_const_handle e) const
{ return K.compare_xy(point(source(e)),point(target(e)))<0; }
void check_order_preserving_embedding(Vertex_const_handle v) const;
/*{\Mop checks if the embedding of the targets of the edges in
the adjacency list |A(v)| is counter-clockwise order-preserving with
respect to the order of the edges in |A(v)|.}*/
void check_order_preserving_embedding() const;
/*{\Mop checks if the embedding of all vertices of |P| is
counter-clockwise order-preserving with respect to the adjacency
list ordering of all vertices.}*/
void check_forward_prefix_condition(Vertex_const_handle v) const;
/*{\Mop checks the forward-prefix property of the adjacency list of |v|.}*/
Halfedge_const_iterator
check_boundary_is_clockwise_weakly_polygon() const;
/*{\Mop checks if the outer face cycle of |P| is a clockwise weakly polygon
and returns a halfedge on the boundary. \precond |P| is a connected graph.
}*/
void check_is_triangulation() const;
/*{\Mop checks if |P| is a triangulation.}*/
}; // PM_checker<PMCDEC,GEOM>
template <typename PMCDEC, typename GEOM>
void PM_checker<PMCDEC,GEOM>::
check_order_preserving_embedding(Vertex_const_handle v) const
{
if ( is_isolated(v) ) return;
std::ostringstream error_status;
CGAL::set_pretty_mode ( error_status );
Halfedge_const_handle ef = first_out_edge(v) ,e=ef,en,enn;
error_status << "check_order_preserving_embedding\n";
error_status << "vertex " << PV(v) << std::endl;
error_status << "ef " << PE(ef) << std::endl;
while ( true ) {
en = cyclic_adj_succ(e);
enn = cyclic_adj_succ(en);
if (en == ef) break;
error_status << " -> " << point(target(e))
<< " " << point(target(en))
<< " " << point(target(enn)) << std::endl;
bool ccw1 = K.strictly_ordered_ccw(direction(e),direction(en),
direction(enn));
bool ccw2 = K.strictly_ordered_ccw(direction(e),direction(en),
direction(ef));
if ( !(ccw1 && ccw2) ) {
error_status << "ccw order violate!" << std::endl << '\0';
CGAL_error_msg(error_status.str().c_str());
}
e = en;
}
}
template <typename PMCDEC, typename GEOM>
void PM_checker<PMCDEC,GEOM>::
check_forward_prefix_condition(Vertex_const_handle v) const
{ Halfedge_const_handle ef = first_out_edge(v);
if ( ef == Halfedge_const_handle() ) return;
Halfedge_const_handle el = cyclic_adj_pred(ef);
bool is_left_turn = K.left_turn(point(v),
point(target(ef)),
point(target(el)));
bool el_forward = is_forward(el);
bool ef_forward = is_forward(ef);
bool ef_el_eq = (ef==el);
std::ostringstream error_status;
error_status << "check_forward_prefix_condition: ";
error_status << PV(v) << "\n";
error_status << PE(ef) << "\n" << PE(el) << "\n";
error_status << " ef == el = " << ef_el_eq;
error_status << " ef_forward = " << ef_forward;
error_status << " el_forward = " << el_forward;
error_status << " is_left_turn = " << is_left_turn;
CGAL_assertion_msg( (ef == el ||
(ef_forward && !el_forward) ||
(ef_forward && el_forward && is_left_turn) ||
(!ef_forward && !el_forward && is_left_turn)) ,
error_status.str().c_str());
}
/* We check the geometric integrity of the structure. We check
+ that all adjacent nodes are differently embedded
+ that all node lists are correctly embedded counterclockwise
with winding number one.
+ that the convex hull of the structure has winding number one.
*/
template <typename PMCDEC, typename GEOM>
void PM_checker<PMCDEC,GEOM>::
check_order_preserving_embedding() const
{
Vertex_const_iterator vit;
for (vit = this->vertices_begin(); vit != this->vertices_end(); ++vit) {
check_order_preserving_embedding(vit);
check_forward_prefix_condition(vit);
}
}
template <typename PMCDEC, typename GEOM>
typename PM_checker<PMCDEC,GEOM>::Halfedge_const_iterator
PM_checker<PMCDEC,GEOM>::
check_boundary_is_clockwise_weakly_polygon() const
{
Vertex_const_iterator vit, v_min;
for (vit = v_min = this->vertices_begin() ; vit != this->vertices_end(); ++vit)
if ( K.compare_xy(point(vit), point(v_min))<0 ) v_min = vit;
CGAL_assertion_msg(!is_isolated(v_min),"Minimal vertex not connected.");
Point p_min = point(v_min);
// determine boundary edge incident to v_min:
Halfedge_const_handle e_boundary_at_v_min = first_out_edge(v_min);
// all out edges are forward oriented due to minimality
Halfedge_around_vertex_const_circulator
hvit(e_boundary_at_v_min), hend(hvit);
do {
--hvit;
Point p1 = point(target(e_boundary_at_v_min));
Point p2 = point(target(hvit));
if ( K.orientation(p_min,p1,p2) > 0 ) { // left_turn
e_boundary_at_v_min = hvit;
break;
}
} while (hvit != hend);
// now e_boundary_at_v_min is highest starting edge in bundle!!
int winding_around_globally=0;
Halfedge_around_face_const_circulator
hfit(e_boundary_at_v_min),hstart(hfit);
Halfedge_const_handle e_prev = next(e_boundary_at_v_min);
/* we run counterclockwise around the outer face cycle and allow only
position where the direction vector of an edge gets smaller again */
Direction d_prev = direction(e_prev);
CGAL_For_all_backwards(hstart,hfit) {
Direction d_curr = direction(hfit);
if ( d_curr < d_prev ) ++winding_around_globally;
d_prev = d_curr;
}
CGAL_assertion(winding_around_globally == 1);
return e_boundary_at_v_min;
}
template <typename PMCDEC, typename GEOM>
void PM_checker<PMCDEC,GEOM>::
check_is_triangulation() const
{
Halfedge_const_iterator eb;
CGAL_assertion(this->number_of_connected_components() == 1);
CGAL_assertion_msg(this->number_of_edges()!=this->number_of_vertices()-1,
" checker checks only full dimensional complexes.");
this->check_integrity_and_topological_planarity(false);
check_order_preserving_embedding();
eb = check_boundary_is_clockwise_weakly_polygon();
CGAL::Unique_hash_map< Halfedge_const_iterator, bool> on_boundary(false);
Halfedge_around_face_const_circulator hit(eb), hend(hit);
std::ostringstream error_status;
CGAL::set_pretty_mode ( error_status );
error_status << "check_is_triangulation\n";
error_status << "on boundary:\n";
CGAL_For_all(hit,hend) {
error_status << " " << PE(hit) << std::endl;
on_boundary[hit]=true;
}
Halfedge_const_iterator eit;
for( eit = this->halfedges_begin(); eit != this->halfedges_end(); ++eit) {
if (on_boundary[eit]) continue;
hit = hend = eit;
int edges_in_face_cycle=0;
CGAL_For_all(hit,hend) {
error_status << PE(hit);
++edges_in_face_cycle;
}
CGAL_assertion_msg(edges_in_face_cycle==3,error_status.str().c_str());
CGAL_assertion_msg(
K.left_turn(point(source(hit)),point(target(hit)),
point(target(next(hit)))), error_status.str().c_str());
}
}
} //namespace CGAL
#endif // CGAL_PM_CHECKER_H