annotate DEPENDENCIES/generic/include/boost/graph/visitors.hpp @ 16:2665513ce2d3

Add boost headers
author Chris Cannam
date Tue, 05 Aug 2014 11:11:38 +0100
parents
children c530137014c0
rev   line source
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