//======================================================================= // Copyright 2007 Aaron Windsor // // 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 __MAKE_MAXIMAL_PLANAR_HPP__ #define __MAKE_MAXIMAL_PLANAR_HPP__ #include #include //for tie #include #include #include #include #include #include #include namespace boost { template struct triangulation_visitor : public planar_face_traversal_visitor { typedef typename graph_traits::vertex_descriptor vertex_t; typedef typename graph_traits::edge_descriptor edge_t; typedef typename graph_traits::vertices_size_type v_size_t; typedef typename graph_traits::degree_size_type degree_size_t; typedef typename graph_traits::edge_iterator edge_iterator_t; typedef typename graph_traits::vertex_iterator vertex_iterator_t; typedef typename graph_traits::adjacency_iterator adjacency_iterator_t; typedef typename std::vector vertex_vector_t; typedef typename std::vector v_size_vector_t; typedef typename std::vector degree_size_vector_t; typedef iterator_property_map < typename v_size_vector_t::iterator, VertexIndexMap > vertex_to_v_size_map_t; typedef iterator_property_map < typename degree_size_vector_t::iterator, VertexIndexMap > vertex_to_degree_size_map_t; typedef typename vertex_vector_t::iterator face_iterator; triangulation_visitor(Graph& arg_g, VertexIndexMap arg_vm, AddEdgeVisitor arg_add_edge_visitor ) : g(arg_g), vm(arg_vm), add_edge_visitor(arg_add_edge_visitor), timestamp(0), marked_vector(num_vertices(g), timestamp), degree_vector(num_vertices(g), 0), marked(marked_vector.begin(), vm), degree(degree_vector.begin(), vm) { vertex_iterator_t vi, vi_end; for(boost::tie(vi,vi_end) = vertices(g); vi != vi_end; ++vi) put(degree, *vi, out_degree(*vi, g)); } template void next_vertex(Vertex v) { // Self-loops will appear as consecutive vertices in the list of // vertices on a face. We want to skip these. if (!vertices_on_face.empty() && (vertices_on_face.back() == v || vertices_on_face.front() == v) ) return; vertices_on_face.push_back(v); } void end_face() { ++timestamp; if (vertices_on_face.size() <= 3) { // At most three vertices on this face - don't need to triangulate vertices_on_face.clear(); return; } // Find vertex on face of minimum degree degree_size_t min_degree = num_vertices(g); typename vertex_vector_t::iterator min_degree_vertex_itr; face_iterator fi_end = vertices_on_face.end(); for(face_iterator fi = vertices_on_face.begin(); fi != fi_end; ++fi) { degree_size_t deg = get(degree,*fi); if (deg < min_degree) { min_degree_vertex_itr = fi; min_degree = deg; } } // To simplify some of the manipulations, we'll re-arrange // vertices_on_face so that it still contains the same // (counter-clockwise) order of the vertices on this face, but now the // min_degree_vertex is the first element in vertices_on_face. vertex_vector_t temp_vector; std::copy(min_degree_vertex_itr, vertices_on_face.end(), std::back_inserter(temp_vector)); std::copy(vertices_on_face.begin(), min_degree_vertex_itr, std::back_inserter(temp_vector)); vertices_on_face.swap(temp_vector); // Mark all of the min degree vertex's neighbors adjacency_iterator_t ai, ai_end; for(boost::tie(ai,ai_end) = adjacent_vertices(vertices_on_face.front(),g); ai != ai_end; ++ai ) { put(marked, *ai, timestamp); } typename vertex_vector_t::iterator marked_neighbor = vertices_on_face.end(); // The iterator manipulations on the next two lines are safe because // vertices_on_face.size() > 3 (from the first test in this function) fi_end = prior(vertices_on_face.end()); for(face_iterator fi = boost::next(boost::next(vertices_on_face.begin())); fi != fi_end; ++fi ) { if (get(marked, *fi) == timestamp) { marked_neighbor = fi; break; } } if (marked_neighbor == vertices_on_face.end()) { add_edge_range( vertices_on_face[0], boost::next(boost::next(vertices_on_face.begin())), prior(vertices_on_face.end()) ); } else { add_edge_range( vertices_on_face[1], boost::next(marked_neighbor), vertices_on_face.end() ); add_edge_range( *boost::next(marked_neighbor), boost::next(boost::next(vertices_on_face.begin())), marked_neighbor ); } //reset for the next face vertices_on_face.clear(); } private: void add_edge_range(vertex_t anchor, face_iterator fi, face_iterator fi_end ) { for (; fi != fi_end; ++fi) { vertex_t v(*fi); add_edge_visitor.visit_vertex_pair(anchor, v, g); put(degree, anchor, get(degree, anchor) + 1); put(degree, v, get(degree, v) + 1); } } Graph& g; VertexIndexMap vm; AddEdgeVisitor add_edge_visitor; v_size_t timestamp; vertex_vector_t vertices_on_face; v_size_vector_t marked_vector; degree_size_vector_t degree_vector; vertex_to_v_size_map_t marked; vertex_to_degree_size_map_t degree; }; template void make_maximal_planar(Graph& g, PlanarEmbedding embedding, VertexIndexMap vm, EdgeIndexMap em, AddEdgeVisitor& vis) { triangulation_visitor visitor(g, vm, vis); planar_face_traversal(g, embedding, visitor, em); } template void make_maximal_planar(Graph& g, PlanarEmbedding embedding, VertexIndexMap vm, EdgeIndexMap em ) { default_add_edge_visitor vis; make_maximal_planar(g, embedding, vm, em, vis); } template void make_maximal_planar(Graph& g, PlanarEmbedding embedding, VertexIndexMap vm ) { make_maximal_planar(g, embedding, vm, get(edge_index,g)); } template void make_maximal_planar(Graph& g, PlanarEmbedding embedding ) { make_maximal_planar(g, embedding, get(vertex_index,g)); } } // namespace boost #endif //__MAKE_MAXIMAL_PLANAR_HPP__