Chris@16
|
1 //=======================================================================
|
Chris@16
|
2 // Copyright 1997, 1998, 1999, 2000 University of Notre Dame.
|
Chris@16
|
3 // Authors: Andrew Lumsdaine, Lie-Quan Lee, Jeremy G. Siek
|
Chris@16
|
4 //
|
Chris@16
|
5 // Distributed under the Boost Software License, Version 1.0. (See
|
Chris@16
|
6 // accompanying file LICENSE_1_0.txt or copy at
|
Chris@16
|
7 // http://www.boost.org/LICENSE_1_0.txt)
|
Chris@16
|
8 //=======================================================================
|
Chris@16
|
9 //
|
Chris@16
|
10 // Revision History:
|
Chris@16
|
11 // 01 April 2001: Modified to use new <boost/limits.hpp> header. (JMaddock)
|
Chris@16
|
12 //
|
Chris@16
|
13 #ifndef BOOST_GRAPH_GRAPH_SEARCH_VISITORS_HPP
|
Chris@16
|
14 #define BOOST_GRAPH_GRAPH_SEARCH_VISITORS_HPP
|
Chris@16
|
15
|
Chris@16
|
16 #include <iosfwd>
|
Chris@16
|
17 #include <boost/config.hpp>
|
Chris@16
|
18 #include <boost/type_traits/is_same.hpp>
|
Chris@16
|
19 #include <boost/mpl/bool.hpp>
|
Chris@16
|
20 #include <boost/property_map/property_map.hpp>
|
Chris@16
|
21 #include <boost/graph/graph_traits.hpp>
|
Chris@16
|
22 #include <boost/limits.hpp>
|
Chris@16
|
23
|
Chris@16
|
24 #if BOOST_WORKAROUND(BOOST_MSVC, < 1300)
|
Chris@16
|
25 // Stay out of the way of the concept checking class
|
Chris@16
|
26 # define Graph Graph_
|
Chris@16
|
27 #endif
|
Chris@16
|
28
|
Chris@16
|
29 namespace boost {
|
Chris@16
|
30
|
Chris@16
|
31 // This is a bit more convenient than std::numeric_limits because
|
Chris@16
|
32 // you don't have to explicitly provide type T.
|
Chris@16
|
33 template <class T>
|
Chris@16
|
34 inline T numeric_limits_max(T) { return (std::numeric_limits<T>::max)(); }
|
Chris@16
|
35
|
Chris@16
|
36 //========================================================================
|
Chris@16
|
37 // Event Tags
|
Chris@16
|
38
|
Chris@16
|
39 namespace detail {
|
Chris@16
|
40 // For partial specialization workaround
|
Chris@16
|
41 enum event_visitor_enum
|
Chris@16
|
42 { on_no_event_num,
|
Chris@16
|
43 on_initialize_vertex_num, on_start_vertex_num,
|
Chris@16
|
44 on_discover_vertex_num, on_finish_vertex_num, on_examine_vertex_num,
|
Chris@16
|
45 on_examine_edge_num, on_tree_edge_num, on_non_tree_edge_num,
|
Chris@16
|
46 on_gray_target_num, on_black_target_num,
|
Chris@16
|
47 on_forward_or_cross_edge_num, on_back_edge_num, on_finish_edge_num,
|
Chris@16
|
48 on_edge_relaxed_num, on_edge_not_relaxed_num,
|
Chris@16
|
49 on_edge_minimized_num, on_edge_not_minimized_num
|
Chris@16
|
50 };
|
Chris@16
|
51
|
Chris@16
|
52 template<typename Event, typename Visitor>
|
Chris@16
|
53 struct functor_to_visitor : Visitor
|
Chris@16
|
54 {
|
Chris@16
|
55 typedef Event event_filter;
|
Chris@16
|
56 functor_to_visitor(const Visitor& visitor) : Visitor(visitor) {}
|
Chris@16
|
57 };
|
Chris@16
|
58
|
Chris@16
|
59 } // namespace detail
|
Chris@16
|
60
|
Chris@16
|
61 struct on_no_event { enum { num = detail::on_no_event_num }; };
|
Chris@16
|
62
|
Chris@16
|
63 struct on_initialize_vertex {
|
Chris@16
|
64 enum { num = detail::on_initialize_vertex_num }; };
|
Chris@16
|
65 struct on_start_vertex { enum { num = detail::on_start_vertex_num }; };
|
Chris@16
|
66 struct on_discover_vertex { enum { num = detail::on_discover_vertex_num }; };
|
Chris@16
|
67 struct on_examine_vertex { enum { num = detail::on_examine_vertex_num }; };
|
Chris@16
|
68 struct on_finish_vertex { enum { num = detail::on_finish_vertex_num }; };
|
Chris@16
|
69
|
Chris@16
|
70 struct on_examine_edge { enum { num = detail::on_examine_edge_num }; };
|
Chris@16
|
71 struct on_tree_edge { enum { num = detail::on_tree_edge_num }; };
|
Chris@16
|
72 struct on_non_tree_edge { enum { num = detail::on_non_tree_edge_num }; };
|
Chris@16
|
73 struct on_gray_target { enum { num = detail::on_gray_target_num }; };
|
Chris@16
|
74 struct on_black_target { enum { num = detail::on_black_target_num }; };
|
Chris@16
|
75 struct on_forward_or_cross_edge {
|
Chris@16
|
76 enum { num = detail::on_forward_or_cross_edge_num }; };
|
Chris@16
|
77 struct on_back_edge { enum { num = detail::on_back_edge_num }; };
|
Chris@16
|
78 struct on_finish_edge { enum { num = detail::on_finish_edge_num }; };
|
Chris@16
|
79
|
Chris@16
|
80 struct on_edge_relaxed { enum { num = detail::on_edge_relaxed_num }; };
|
Chris@16
|
81 struct on_edge_not_relaxed {
|
Chris@16
|
82 enum { num = detail::on_edge_not_relaxed_num }; };
|
Chris@16
|
83 struct on_edge_minimized { enum { num = detail::on_edge_minimized_num }; };
|
Chris@16
|
84 struct on_edge_not_minimized {
|
Chris@16
|
85 enum { num = detail::on_edge_not_minimized_num }; };
|
Chris@16
|
86
|
Chris@16
|
87 //========================================================================
|
Chris@16
|
88 // base_visitor and null_visitor
|
Chris@16
|
89
|
Chris@16
|
90 // needed for MSVC workaround
|
Chris@16
|
91 template <class Visitor>
|
Chris@16
|
92 struct base_visitor {
|
Chris@16
|
93 typedef on_no_event event_filter;
|
Chris@16
|
94 template <class T, class Graph>
|
Chris@16
|
95 void operator()(T, Graph&) { }
|
Chris@16
|
96 };
|
Chris@16
|
97
|
Chris@16
|
98 struct null_visitor : public base_visitor<null_visitor> {
|
Chris@16
|
99 typedef on_no_event event_filter;
|
Chris@16
|
100 template <class T, class Graph>
|
Chris@16
|
101 void operator()(T, Graph&) { }
|
Chris@16
|
102 };
|
Chris@16
|
103
|
Chris@16
|
104 //========================================================================
|
Chris@16
|
105 // The invoke_visitors() function
|
Chris@16
|
106
|
Chris@16
|
107 namespace detail {
|
Chris@16
|
108 template <class Visitor, class T, class Graph>
|
Chris@16
|
109 inline void invoke_dispatch(Visitor& v, T x, Graph& g, mpl::true_) {
|
Chris@16
|
110 v(x, g);
|
Chris@16
|
111 }
|
Chris@16
|
112
|
Chris@16
|
113 template <class Visitor, class T, class Graph>
|
Chris@16
|
114 inline void invoke_dispatch(Visitor&, T, Graph&, mpl::false_)
|
Chris@16
|
115 { }
|
Chris@16
|
116 } // namespace detail
|
Chris@16
|
117
|
Chris@16
|
118 template <class Visitor, class Rest, class T, class Graph, class Tag>
|
Chris@16
|
119 inline void
|
Chris@16
|
120 invoke_visitors(std::pair<Visitor, Rest>& vlist, T x, Graph& g, Tag tag) {
|
Chris@16
|
121 typedef typename Visitor::event_filter Category;
|
Chris@16
|
122 typedef typename is_same<Category, Tag>::type IsSameTag;
|
Chris@16
|
123 detail::invoke_dispatch(vlist.first, x, g, IsSameTag());
|
Chris@16
|
124 invoke_visitors(vlist.second, x, g, tag);
|
Chris@16
|
125 }
|
Chris@16
|
126 #if defined(BOOST_MSVC) && BOOST_MSVC <= 1300
|
Chris@16
|
127 template <class Visitor, class T, class Graph, class Tag>
|
Chris@16
|
128 inline void
|
Chris@16
|
129 invoke_visitors(base_visitor<Visitor>& vis, T x, Graph& g, Tag) {
|
Chris@16
|
130 typedef typename Visitor::event_filter Category;
|
Chris@16
|
131 typedef typename is_same<Category, Tag>::type IsSameTag;
|
Chris@16
|
132 Visitor& v = static_cast<Visitor&>(vis);
|
Chris@16
|
133 detail::invoke_dispatch(v, x, g, IsSameTag());
|
Chris@16
|
134 }
|
Chris@16
|
135 #else
|
Chris@16
|
136 template <class Visitor, class T, class Graph, class Tag>
|
Chris@16
|
137 inline void
|
Chris@16
|
138 invoke_visitors(Visitor& v, T x, Graph& g, Tag) {
|
Chris@16
|
139 typedef typename Visitor::event_filter Category;
|
Chris@16
|
140 typedef typename is_same<Category, Tag>::type IsSameTag;
|
Chris@16
|
141 detail::invoke_dispatch(v, x, g, IsSameTag());
|
Chris@16
|
142 }
|
Chris@16
|
143 #endif
|
Chris@16
|
144
|
Chris@16
|
145 //========================================================================
|
Chris@16
|
146 // predecessor_recorder
|
Chris@16
|
147
|
Chris@16
|
148 template <class PredecessorMap, class Tag>
|
Chris@16
|
149 struct predecessor_recorder
|
Chris@16
|
150 : public base_visitor<predecessor_recorder<PredecessorMap, Tag> >
|
Chris@16
|
151 {
|
Chris@16
|
152 typedef Tag event_filter;
|
Chris@16
|
153 predecessor_recorder(PredecessorMap pa) : m_predecessor(pa) { }
|
Chris@16
|
154 template <class Edge, class Graph>
|
Chris@16
|
155 void operator()(Edge e, const Graph& g) {
|
Chris@16
|
156 put(m_predecessor, target(e, g), source(e, g));
|
Chris@16
|
157 }
|
Chris@16
|
158 PredecessorMap m_predecessor;
|
Chris@16
|
159 };
|
Chris@16
|
160 template <class PredecessorMap, class Tag>
|
Chris@16
|
161 predecessor_recorder<PredecessorMap, Tag>
|
Chris@16
|
162 record_predecessors(PredecessorMap pa, Tag) {
|
Chris@16
|
163 return predecessor_recorder<PredecessorMap, Tag> (pa);
|
Chris@16
|
164 }
|
Chris@16
|
165
|
Chris@16
|
166 //========================================================================
|
Chris@16
|
167 // edge_predecessor_recorder
|
Chris@16
|
168
|
Chris@16
|
169 template <class PredEdgeMap, class Tag>
|
Chris@16
|
170 struct edge_predecessor_recorder
|
Chris@16
|
171 : public base_visitor<edge_predecessor_recorder<PredEdgeMap, Tag> >
|
Chris@16
|
172 {
|
Chris@16
|
173 typedef Tag event_filter;
|
Chris@16
|
174 edge_predecessor_recorder(PredEdgeMap pa) : m_predecessor(pa) { }
|
Chris@16
|
175 template <class Edge, class Graph>
|
Chris@16
|
176 void operator()(Edge e, const Graph& g) {
|
Chris@16
|
177 put(m_predecessor, target(e, g), e);
|
Chris@16
|
178 }
|
Chris@16
|
179 PredEdgeMap m_predecessor;
|
Chris@16
|
180 };
|
Chris@16
|
181 template <class PredEdgeMap, class Tag>
|
Chris@16
|
182 edge_predecessor_recorder<PredEdgeMap, Tag>
|
Chris@16
|
183 record_edge_predecessors(PredEdgeMap pa, Tag) {
|
Chris@16
|
184 return edge_predecessor_recorder<PredEdgeMap, Tag> (pa);
|
Chris@16
|
185 }
|
Chris@16
|
186
|
Chris@16
|
187 //========================================================================
|
Chris@16
|
188 // distance_recorder
|
Chris@16
|
189
|
Chris@16
|
190 template <class DistanceMap, class Tag>
|
Chris@16
|
191 struct distance_recorder
|
Chris@16
|
192 : public base_visitor<distance_recorder<DistanceMap, Tag> >
|
Chris@16
|
193 {
|
Chris@16
|
194 typedef Tag event_filter;
|
Chris@16
|
195 distance_recorder(DistanceMap pa) : m_distance(pa) { }
|
Chris@16
|
196 template <class Edge, class Graph>
|
Chris@16
|
197 void operator()(Edge e, const Graph& g) {
|
Chris@16
|
198 typename graph_traits<Graph>::vertex_descriptor
|
Chris@16
|
199 u = source(e, g), v = target(e, g);
|
Chris@16
|
200 put(m_distance, v, get(m_distance, u) + 1);
|
Chris@16
|
201 }
|
Chris@16
|
202 DistanceMap m_distance;
|
Chris@16
|
203 };
|
Chris@16
|
204 template <class DistanceMap, class Tag>
|
Chris@16
|
205 distance_recorder<DistanceMap, Tag>
|
Chris@16
|
206 record_distances(DistanceMap pa, Tag) {
|
Chris@16
|
207 return distance_recorder<DistanceMap, Tag> (pa);
|
Chris@16
|
208 }
|
Chris@16
|
209
|
Chris@16
|
210 //========================================================================
|
Chris@16
|
211 // time_stamper
|
Chris@16
|
212
|
Chris@16
|
213
|
Chris@16
|
214 template <class TimeMap, class TimeT, class Tag>
|
Chris@16
|
215 struct time_stamper
|
Chris@16
|
216 : public base_visitor<time_stamper<TimeMap, TimeT, Tag> >
|
Chris@16
|
217 {
|
Chris@16
|
218 typedef Tag event_filter;
|
Chris@16
|
219 time_stamper(TimeMap pa, TimeT& t) : m_time_pa(pa), m_time(t) { }
|
Chris@16
|
220 template <class Vertex, class Graph>
|
Chris@16
|
221 void operator()(Vertex u, const Graph&) {
|
Chris@16
|
222 put(m_time_pa, u, ++m_time);
|
Chris@16
|
223 }
|
Chris@16
|
224 TimeMap m_time_pa;
|
Chris@16
|
225 TimeT& m_time;
|
Chris@16
|
226 };
|
Chris@16
|
227 template <class TimeMap, class TimeT, class Tag>
|
Chris@16
|
228 time_stamper<TimeMap, TimeT, Tag>
|
Chris@16
|
229 stamp_times(TimeMap pa, TimeT& time_counter, Tag) {
|
Chris@16
|
230 return time_stamper<TimeMap, TimeT, Tag>(pa, time_counter);
|
Chris@16
|
231 }
|
Chris@16
|
232
|
Chris@16
|
233 //========================================================================
|
Chris@16
|
234 // property_writer
|
Chris@16
|
235
|
Chris@16
|
236 template <class PA, class OutputIterator, class Tag>
|
Chris@16
|
237 struct property_writer
|
Chris@16
|
238 : public base_visitor<property_writer<PA, OutputIterator, Tag> >
|
Chris@16
|
239 {
|
Chris@16
|
240 typedef Tag event_filter;
|
Chris@16
|
241
|
Chris@16
|
242 property_writer(PA pa, OutputIterator out) : m_pa(pa), m_out(out) { }
|
Chris@16
|
243
|
Chris@16
|
244 template <class T, class Graph>
|
Chris@16
|
245 void operator()(T x, Graph&) { *m_out++ = get(m_pa, x); }
|
Chris@16
|
246 PA m_pa;
|
Chris@16
|
247 OutputIterator m_out;
|
Chris@16
|
248 };
|
Chris@16
|
249 template <class PA, class OutputIterator, class Tag>
|
Chris@16
|
250 property_writer<PA, OutputIterator, Tag>
|
Chris@16
|
251 write_property(PA pa, OutputIterator out, Tag) {
|
Chris@16
|
252 return property_writer<PA, OutputIterator, Tag>(pa, out);
|
Chris@16
|
253 }
|
Chris@16
|
254
|
Chris@16
|
255 //========================================================================
|
Chris@16
|
256 // property_put
|
Chris@16
|
257
|
Chris@16
|
258 /**
|
Chris@16
|
259 * Functor which just sets a given value to a vertex or edge in a property map.
|
Chris@16
|
260 */
|
Chris@16
|
261
|
Chris@16
|
262 template <typename PropertyMap, typename EventTag>
|
Chris@16
|
263 struct property_put
|
Chris@16
|
264 {
|
Chris@16
|
265 typedef EventTag event_filter;
|
Chris@16
|
266
|
Chris@16
|
267 property_put (PropertyMap property_map,
|
Chris@16
|
268 typename property_traits <PropertyMap>::value_type value) :
|
Chris@16
|
269 property_map_ (property_map), value_ (value)
|
Chris@16
|
270 {}
|
Chris@16
|
271
|
Chris@16
|
272 template <typename VertexOrEdge, typename Graph>
|
Chris@16
|
273 void operator() (VertexOrEdge v, const Graph&)
|
Chris@16
|
274 {
|
Chris@16
|
275 put (property_map_, v, value_);
|
Chris@16
|
276 }
|
Chris@16
|
277
|
Chris@16
|
278 private:
|
Chris@16
|
279 PropertyMap property_map_;
|
Chris@16
|
280 typename property_traits <PropertyMap>::value_type value_;
|
Chris@16
|
281 };
|
Chris@16
|
282
|
Chris@16
|
283 /**
|
Chris@16
|
284 * Creates a property_put functor which just sets a given value to a vertex or edge.
|
Chris@16
|
285 *
|
Chris@16
|
286 * @param property_map Given writeable property map
|
Chris@16
|
287 * @param value Fixed value of the map
|
Chris@16
|
288 * @param tag Event Filter
|
Chris@16
|
289 * @return The functor.
|
Chris@16
|
290 */
|
Chris@16
|
291
|
Chris@16
|
292 template <typename PropertyMap, typename EventTag>
|
Chris@16
|
293 inline property_put <PropertyMap, EventTag>
|
Chris@16
|
294 put_property (PropertyMap property_map,
|
Chris@16
|
295 typename property_traits <PropertyMap>::value_type value,
|
Chris@16
|
296 EventTag)
|
Chris@16
|
297 {
|
Chris@16
|
298 return property_put <PropertyMap, EventTag> (property_map, value);
|
Chris@16
|
299 }
|
Chris@16
|
300
|
Chris@16
|
301 #define BOOST_GRAPH_EVENT_STUB(Event,Kind) \
|
Chris@16
|
302 typedef ::boost::Event Event##_type; \
|
Chris@16
|
303 template<typename Visitor> \
|
Chris@16
|
304 Kind##_visitor<std::pair<detail::functor_to_visitor<Event##_type, \
|
Chris@16
|
305 Visitor>, Visitors> > \
|
Chris@16
|
306 do_##Event(Visitor visitor) \
|
Chris@16
|
307 { \
|
Chris@16
|
308 typedef std::pair<detail::functor_to_visitor<Event##_type, Visitor>, \
|
Chris@16
|
309 Visitors> visitor_list; \
|
Chris@16
|
310 typedef Kind##_visitor<visitor_list> result_type; \
|
Chris@16
|
311 return result_type(visitor_list(visitor, m_vis)); \
|
Chris@16
|
312 }
|
Chris@16
|
313
|
Chris@16
|
314 } /* namespace boost */
|
Chris@16
|
315
|
Chris@16
|
316 #if BOOST_WORKAROUND(BOOST_MSVC, < 1300)
|
Chris@16
|
317 // Stay out of the way of the concept checking class
|
Chris@16
|
318 # undef Graph
|
Chris@16
|
319 #endif
|
Chris@16
|
320
|
Chris@16
|
321 #endif
|