301 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			301 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
|   | //======================================================================= | ||
|  | // Copyright 1997, 1998, 1999, 2000 University of Notre Dame. | ||
|  | // Authors: Andrew Lumsdaine, Lie-Quan Lee, Jeremy G. Siek | ||
|  | // | ||
|  | // 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) | ||
|  | //======================================================================= | ||
|  | // | ||
|  | // Revision History: | ||
|  | //   01 April 2001: Modified to use new <boost/limits.hpp> header. (JMaddock) | ||
|  | // | ||
|  | #ifndef BOOST_GRAPH_GRAPH_SEARCH_VISITORS_HPP | ||
|  | #define BOOST_GRAPH_GRAPH_SEARCH_VISITORS_HPP | ||
|  | 
 | ||
|  | #include <iosfwd> | ||
|  | #include <boost/config.hpp> | ||
|  | #include <boost/type_traits/is_same.hpp> | ||
|  | #include <boost/mpl/bool.hpp> | ||
|  | #include <boost/property_map/property_map.hpp> | ||
|  | #include <boost/graph/graph_traits.hpp> | ||
|  | #include <boost/limits.hpp> | ||
|  | 
 | ||
|  | namespace boost { | ||
|  | 
 | ||
|  |   // This is a bit more convenient than std::numeric_limits because | ||
|  |   // you don't have to explicitly provide type T. | ||
|  |   template <class T> | ||
|  |   inline T numeric_limits_max(T) { return (std::numeric_limits<T>::max)(); } | ||
|  | 
 | ||
|  |   //======================================================================== | ||
|  |   // Event Tags | ||
|  | 
 | ||
|  |   namespace detail { | ||
|  |     // For partial specialization workaround | ||
|  |     enum event_visitor_enum | ||
|  |     { on_no_event_num, | ||
|  |       on_initialize_vertex_num, on_start_vertex_num, | ||
|  |       on_discover_vertex_num, on_finish_vertex_num, on_examine_vertex_num, | ||
|  |       on_examine_edge_num, on_tree_edge_num, on_non_tree_edge_num, | ||
|  |       on_gray_target_num, on_black_target_num, | ||
|  |       on_forward_or_cross_edge_num, on_back_edge_num, on_finish_edge_num, | ||
|  |       on_edge_relaxed_num, on_edge_not_relaxed_num, | ||
|  |       on_edge_minimized_num, on_edge_not_minimized_num | ||
|  |     }; | ||
|  | 
 | ||
|  |     template<typename Event, typename Visitor> | ||
|  |     struct functor_to_visitor : Visitor | ||
|  |     { | ||
|  |       typedef Event event_filter; | ||
|  |       functor_to_visitor(const Visitor& visitor) : Visitor(visitor) {} | ||
|  |     }; | ||
|  | 
 | ||
|  |   } // namespace detail | ||
|  | 
 | ||
|  |   struct on_no_event { enum { num = detail::on_no_event_num }; }; | ||
|  | 
 | ||
|  |   struct on_initialize_vertex { | ||
|  |     enum { num = detail::on_initialize_vertex_num }; }; | ||
|  |   struct on_start_vertex { enum { num = detail::on_start_vertex_num }; }; | ||
|  |   struct on_discover_vertex { enum { num = detail::on_discover_vertex_num }; }; | ||
|  |   struct on_examine_vertex { enum { num = detail::on_examine_vertex_num }; }; | ||
|  |   struct on_finish_vertex { enum { num = detail::on_finish_vertex_num }; }; | ||
|  | 
 | ||
|  |   struct on_examine_edge { enum { num = detail::on_examine_edge_num }; }; | ||
|  |   struct on_tree_edge { enum { num = detail::on_tree_edge_num }; }; | ||
|  |   struct on_non_tree_edge { enum { num = detail::on_non_tree_edge_num }; }; | ||
|  |   struct on_gray_target { enum { num = detail::on_gray_target_num }; }; | ||
|  |   struct on_black_target { enum { num = detail::on_black_target_num }; }; | ||
|  |   struct on_forward_or_cross_edge { | ||
|  |     enum { num = detail::on_forward_or_cross_edge_num }; }; | ||
|  |   struct on_back_edge { enum { num = detail::on_back_edge_num }; }; | ||
|  |   struct on_finish_edge { enum { num = detail::on_finish_edge_num }; }; | ||
|  | 
 | ||
|  |   struct on_edge_relaxed { enum { num = detail::on_edge_relaxed_num }; }; | ||
|  |   struct on_edge_not_relaxed { | ||
|  |     enum { num = detail::on_edge_not_relaxed_num }; }; | ||
|  |   struct on_edge_minimized { enum { num = detail::on_edge_minimized_num }; }; | ||
|  |   struct on_edge_not_minimized { | ||
|  |     enum { num = detail::on_edge_not_minimized_num }; }; | ||
|  | 
 | ||
|  |   //======================================================================== | ||
|  |   // base_visitor and null_visitor | ||
|  | 
 | ||
|  |   // needed for MSVC workaround | ||
|  |   template <class Visitor> | ||
|  |   struct base_visitor { | ||
|  |     typedef on_no_event event_filter; | ||
|  |     template <class T, class Graph> | ||
|  |     void operator()(T, Graph&) { } | ||
|  |   }; | ||
|  | 
 | ||
|  |   struct null_visitor : public base_visitor<null_visitor> { | ||
|  |     typedef on_no_event event_filter; | ||
|  |     template <class T, class Graph> | ||
|  |     void operator()(T, Graph&) { } | ||
|  |   }; | ||
|  | 
 | ||
|  |   //======================================================================== | ||
|  |   // The invoke_visitors() function | ||
|  | 
 | ||
|  |   namespace detail { | ||
|  |     template <class Visitor, class T, class Graph> | ||
|  |     inline void invoke_dispatch(Visitor& v, T x, Graph& g, mpl::true_) { | ||
|  |        v(x, g); | ||
|  |     } | ||
|  | 
 | ||
|  |     template <class Visitor, class T, class Graph> | ||
|  |     inline void invoke_dispatch(Visitor&, T, Graph&, mpl::false_) | ||
|  |     { } | ||
|  |   } // namespace detail | ||
|  | 
 | ||
|  |   template <class Visitor, class Rest, class T, class Graph, class Tag> | ||
|  |   inline void | ||
|  |   invoke_visitors(std::pair<Visitor, Rest>& vlist, T x, Graph& g, Tag tag) { | ||
|  |     typedef typename Visitor::event_filter Category; | ||
|  |     typedef typename is_same<Category, Tag>::type IsSameTag; | ||
|  |     detail::invoke_dispatch(vlist.first, x, g, IsSameTag()); | ||
|  |     invoke_visitors(vlist.second, x, g, tag); | ||
|  |   } | ||
|  |   template <class Visitor, class T, class Graph, class Tag> | ||
|  |   inline void | ||
|  |   invoke_visitors(Visitor& v, T x, Graph& g, Tag) { | ||
|  |     typedef typename Visitor::event_filter Category; | ||
|  |     typedef typename is_same<Category, Tag>::type IsSameTag; | ||
|  |     detail::invoke_dispatch(v, x, g, IsSameTag()); | ||
|  |   } | ||
|  | 
 | ||
|  |   //======================================================================== | ||
|  |   // predecessor_recorder | ||
|  | 
 | ||
|  |   template <class PredecessorMap, class Tag> | ||
|  |   struct predecessor_recorder | ||
|  |     : public base_visitor<predecessor_recorder<PredecessorMap, Tag> > | ||
|  |   { | ||
|  |     typedef Tag event_filter; | ||
|  |     predecessor_recorder(PredecessorMap pa) : m_predecessor(pa) { } | ||
|  |     template <class Edge, class Graph> | ||
|  |     void operator()(Edge e, const Graph& g) { | ||
|  |       put(m_predecessor, target(e, g), source(e, g)); | ||
|  |     } | ||
|  |     PredecessorMap m_predecessor; | ||
|  |   }; | ||
|  |   template <class PredecessorMap, class Tag> | ||
|  |   predecessor_recorder<PredecessorMap, Tag> | ||
|  |   record_predecessors(PredecessorMap pa, Tag) { | ||
|  |     return predecessor_recorder<PredecessorMap, Tag> (pa); | ||
|  |   } | ||
|  | 
 | ||
|  |   //======================================================================== | ||
|  |   // edge_predecessor_recorder | ||
|  | 
 | ||
|  |   template <class PredEdgeMap, class Tag> | ||
|  |   struct edge_predecessor_recorder | ||
|  |     : public base_visitor<edge_predecessor_recorder<PredEdgeMap, Tag> > | ||
|  |   { | ||
|  |     typedef Tag event_filter; | ||
|  |     edge_predecessor_recorder(PredEdgeMap pa) : m_predecessor(pa) { } | ||
|  |     template <class Edge, class Graph> | ||
|  |     void operator()(Edge e, const Graph& g) { | ||
|  |       put(m_predecessor, target(e, g), e); | ||
|  |     } | ||
|  |     PredEdgeMap m_predecessor; | ||
|  |   }; | ||
|  |   template <class PredEdgeMap, class Tag> | ||
|  |   edge_predecessor_recorder<PredEdgeMap, Tag> | ||
|  |   record_edge_predecessors(PredEdgeMap pa, Tag) { | ||
|  |     return edge_predecessor_recorder<PredEdgeMap, Tag> (pa); | ||
|  |   } | ||
|  | 
 | ||
|  |   //======================================================================== | ||
|  |   // distance_recorder | ||
|  | 
 | ||
|  |   template <class DistanceMap, class Tag> | ||
|  |   struct distance_recorder | ||
|  |     : public base_visitor<distance_recorder<DistanceMap, Tag> > | ||
|  |   { | ||
|  |     typedef Tag event_filter; | ||
|  |     distance_recorder(DistanceMap pa) : m_distance(pa) { } | ||
|  |     template <class Edge, class Graph> | ||
|  |     void operator()(Edge e, const Graph& g) { | ||
|  |       typename graph_traits<Graph>::vertex_descriptor | ||
|  |         u = source(e, g), v = target(e, g); | ||
|  |       put(m_distance, v, get(m_distance, u) + 1); | ||
|  |     } | ||
|  |     DistanceMap m_distance; | ||
|  |   }; | ||
|  |   template <class DistanceMap, class Tag> | ||
|  |   distance_recorder<DistanceMap, Tag> | ||
|  |   record_distances(DistanceMap pa, Tag) { | ||
|  |     return distance_recorder<DistanceMap, Tag> (pa); | ||
|  |   } | ||
|  | 
 | ||
|  |   //======================================================================== | ||
|  |   // time_stamper | ||
|  | 
 | ||
|  | 
 | ||
|  |   template <class TimeMap, class TimeT, class Tag> | ||
|  |   struct time_stamper | ||
|  |     : public base_visitor<time_stamper<TimeMap, TimeT, Tag> > | ||
|  |   { | ||
|  |     typedef Tag event_filter; | ||
|  |     time_stamper(TimeMap pa, TimeT& t) : m_time_pa(pa), m_time(t) { } | ||
|  |     template <class Vertex, class Graph> | ||
|  |     void operator()(Vertex u, const Graph&) { | ||
|  |       put(m_time_pa, u, ++m_time); | ||
|  |     } | ||
|  |     TimeMap m_time_pa; | ||
|  |     TimeT& m_time; | ||
|  |   }; | ||
|  |   template <class TimeMap, class TimeT, class Tag> | ||
|  |   time_stamper<TimeMap, TimeT, Tag> | ||
|  |   stamp_times(TimeMap pa, TimeT& time_counter, Tag) { | ||
|  |     return time_stamper<TimeMap, TimeT, Tag>(pa, time_counter); | ||
|  |   } | ||
|  | 
 | ||
|  |   //======================================================================== | ||
|  |   // property_writer | ||
|  | 
 | ||
|  |   template <class PA, class OutputIterator, class Tag> | ||
|  |   struct property_writer | ||
|  |     : public base_visitor<property_writer<PA, OutputIterator, Tag> > | ||
|  |   { | ||
|  |     typedef Tag event_filter; | ||
|  | 
 | ||
|  |     property_writer(PA pa, OutputIterator out) : m_pa(pa), m_out(out) { } | ||
|  | 
 | ||
|  |     template <class T, class Graph> | ||
|  |     void operator()(T x, Graph&) { *m_out++ = get(m_pa, x); } | ||
|  |     PA m_pa; | ||
|  |     OutputIterator m_out; | ||
|  |   }; | ||
|  |   template <class PA, class OutputIterator, class Tag> | ||
|  |   property_writer<PA, OutputIterator, Tag> | ||
|  |   write_property(PA pa, OutputIterator out, Tag) { | ||
|  |     return property_writer<PA, OutputIterator, Tag>(pa, out); | ||
|  |   } | ||
|  | 
 | ||
|  |   //======================================================================== | ||
|  |   // property_put | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Functor which just sets a given value to a vertex or edge in a property map. | ||
|  |      */ | ||
|  | 
 | ||
|  |   template <typename PropertyMap, typename EventTag> | ||
|  |   struct property_put | ||
|  |   { | ||
|  |     typedef EventTag event_filter; | ||
|  |      | ||
|  |     property_put (PropertyMap property_map, | ||
|  |                   typename property_traits <PropertyMap>::value_type value) : | ||
|  |       property_map_ (property_map), value_ (value) | ||
|  |     {} | ||
|  | 
 | ||
|  |     template <typename VertexOrEdge, typename Graph> | ||
|  |     void operator() (VertexOrEdge v, const Graph&) | ||
|  |     { | ||
|  |       put (property_map_, v, value_); | ||
|  |     } | ||
|  | 
 | ||
|  |   private: | ||
|  |     PropertyMap property_map_; | ||
|  |     typename property_traits <PropertyMap>::value_type value_; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Creates a property_put functor which just sets a given value to a vertex or edge. | ||
|  |    *  | ||
|  |    * @param property_map Given writeable property map  | ||
|  |    * @param value Fixed value of the map | ||
|  |    * @param tag Event Filter | ||
|  |    * @return The functor. | ||
|  |    */ | ||
|  | 
 | ||
|  |     template <typename PropertyMap, typename EventTag> | ||
|  |     inline property_put <PropertyMap, EventTag> | ||
|  |     put_property (PropertyMap property_map, | ||
|  |                   typename property_traits <PropertyMap>::value_type value, | ||
|  |                   EventTag) | ||
|  |     { | ||
|  |       return property_put <PropertyMap, EventTag> (property_map, value); | ||
|  |     } | ||
|  | 
 | ||
|  | #define BOOST_GRAPH_EVENT_STUB(Event,Kind)                                 \ | ||
|  |     typedef ::boost::Event Event##_type;                                   \ | ||
|  |     template<typename Visitor>                                             \ | ||
|  |     Kind##_visitor<std::pair<detail::functor_to_visitor<Event##_type,      \ | ||
|  |                                                      Visitor>, Visitors> > \ | ||
|  |     do_##Event(Visitor visitor)                                            \ | ||
|  |     {                                                                      \ | ||
|  |       typedef std::pair<detail::functor_to_visitor<Event##_type, Visitor>, \ | ||
|  |                         Visitors> visitor_list;                            \ | ||
|  |       typedef Kind##_visitor<visitor_list> result_type;                    \ | ||
|  |       return result_type(visitor_list(visitor, m_vis));                    \ | ||
|  |     } | ||
|  | 
 | ||
|  | } /* namespace boost */ | ||
|  | 
 | ||
|  | #endif |