COIN-OR::LEMON - Graph Library

source: lemon-0.x/src/work/marci/graph_wrapper.h @ 357:5165a1c8633e

Last change on this file since 357:5165a1c8633e was 357:5165a1c8633e, checked in by Alpar Juttner, 21 years ago

Requests for docs.

File size: 36.3 KB
RevLine 
[174]1// -*- c++ -*-
[259]2#ifndef HUGO_GRAPH_WRAPPER_H
3#define HUGO_GRAPH_WRAPPER_H
[76]4
[174]5#include <invalid.h>
6
[105]7namespace hugo {
[76]8
[335]9  /// Graph wrappers
10
[344]11  /// A main parts of HUGOlib are the different graph structures,
[335]12  /// generic graph algorithms, graph concepts which couple these, and
13  /// graph wrappers. While the previous ones are more or less clear, the
14  /// latter notion needs further explanation.
15  /// Graph wrappers are graph classes which serve for considering graph
[344]16  /// structures in different ways. A short example makes the notion much
17  /// clearer.
18  /// Suppose that we have an instance \c g of a directed graph
19  /// type say \c ListGraph and an algorithm
[335]20  /// \code template<typename Graph> int algorithm(const Graph&); \endcode
[344]21  /// is needed to run on the reversely oriented graph.
22  /// It may be expensive (in time or in memory usage) to copy
23  /// \c g with the reverse orientation.
[335]24  /// Thus, a wrapper class
25  /// \code template<typename Graph> class RevGraphWrapper; \endcode is used.
26  /// The code looks as follows
27  /// \code
28  /// ListGraph g;
29  /// RevGraphWrapper<ListGraph> rgw(g);
30  /// int result=algorithm(rgw);
31  /// \endcode
[344]32  /// After running the algorithm, the original graph \c g
33  /// remains untouched. Thus the graph wrapper used above is to consider the
34  /// original graph with reverse orientation.
[335]35  /// This techniques gives rise to an elegant code, and
36  /// based on stable graph wrappers, complex algorithms can be
37  /// implemented easily.
38  /// In flow, circulation and bipartite matching problems, the residual
[344]39  /// graph is of particular importance. Combining a wrapper implementing
40  /// this, shortest path algorithms and minimum mean cycle algorithms,
[335]41  /// a range of weighted and cardinality optimization algorithms can be
42  /// obtained. For lack of space, for other examples,
[344]43  /// the interested user is referred to the detailed documentation of graph
[335]44  /// wrappers.
[344]45  /// The behavior of graph wrappers can be very different. Some of them keep
[335]46  /// capabilities of the original graph while in other cases this would be
[344]47  /// meaningless. This means that the concepts that they are a model of depend
[335]48  /// on the graph wrapper, and the wrapped graph(s).
[344]49  /// If an edge of \c rgw is deleted, this is carried out by
50  /// deleting the corresponding edge of \c g. But for a residual
[335]51  /// graph, this operation has no sense.
52  /// Let we stand one more example here to simplify your work.
53  /// wrapper class
54  /// \code template<typename Graph> class RevGraphWrapper; \endcode
55  /// has constructor
[344]56  /// <tt> RevGraphWrapper(Graph& _g)</tt>.
[335]57  /// This means that in a situation,
[344]58  /// when a <tt> const ListGraph& </tt> reference to a graph is given,
59  /// then it have to be instantiated with <tt>Graph=const ListGraph</tt>.
[335]60  /// \code
61  /// int algorithm1(const ListGraph& g) {
62  ///   RevGraphWrapper<const ListGraph> rgw(g);
63  ///   return algorithm2(rgw);
64  /// }
65  /// \endcode
[303]66  template<typename Graph>
67  class GraphWrapper {
[212]68  protected:
[303]69    Graph* graph;
[212]70 
71  public:
[311]72    typedef Graph BaseGraph;
[303]73    typedef Graph ParentGraph;
[212]74
[303]75//     GraphWrapper() : graph(0) { }
76    GraphWrapper(Graph& _graph) : graph(&_graph) { }
77//     void setGraph(Graph& _graph) { graph=&_graph; }
78//     Graph& getGraph() const { return *graph; }
79 
[317]80//    typedef typename Graph::Node Node;
81    class Node : public Graph::Node {
82      friend class GraphWrapper<Graph>;
[265]83    public:
[317]84      Node() { }
85      Node(const typename Graph::Node& _n) : Graph::Node(_n) { }
86      Node(const Invalid& i) : Graph::Node(i) { }
87    };
88    class NodeIt {
89      friend class GraphWrapper<Graph>;
90      typename Graph::NodeIt n;
91     public:
[265]92      NodeIt() { }
[317]93      NodeIt(const typename Graph::NodeIt& _n) : n(_n) { }
94      NodeIt(const Invalid& i) : n(i) { }
95      NodeIt(const GraphWrapper<Graph>& _G) : n(*(_G.graph)) { }
96      operator Node() const { return Node(typename Graph::Node(n)); }
[265]97    };
[317]98//    typedef typename Graph::Edge Edge;
99    class Edge : public Graph::Edge {
100      friend class GraphWrapper<Graph>;
101    public:
102      Edge() { }
103      Edge(const typename Graph::Edge& _e) : Graph::Edge(_e) { }
104      Edge(const Invalid& i) : Graph::Edge(i) { }
105    };
106    class OutEdgeIt {
107      friend class GraphWrapper<Graph>;
108      typename Graph::OutEdgeIt e;
[265]109    public:
110      OutEdgeIt() { }
[317]111      OutEdgeIt(const typename Graph::OutEdgeIt& _e) : e(_e) { }
112      OutEdgeIt(const Invalid& i) : e(i) { }
113      OutEdgeIt(const GraphWrapper<Graph>& _G, const Node& _n) :
114        e(*(_G.graph), typename Graph::Node(_n)) { }
115      operator Edge() const { return Edge(typename Graph::Edge(e)); }
[265]116    };
[317]117    class InEdgeIt {
118      friend class GraphWrapper<Graph>;
119      typename Graph::InEdgeIt e;
[265]120    public:
121      InEdgeIt() { }
[317]122      InEdgeIt(const typename Graph::InEdgeIt& _e) : e(_e) { }
123      InEdgeIt(const Invalid& i) : e(i) { }
124      InEdgeIt(const GraphWrapper<Graph>& _G, const Node& _n) :
125        e(*(_G.graph), typename Graph::Node(_n)) { }
126      operator Edge() const { return Edge(typename Graph::Edge(e)); }
[265]127    };
[303]128    //typedef typename Graph::SymEdgeIt SymEdgeIt;
[317]129    class EdgeIt {
130      friend class GraphWrapper<Graph>;
131      typename Graph::EdgeIt e;
[265]132    public:
133      EdgeIt() { }
[317]134      EdgeIt(const typename Graph::EdgeIt& _e) : e(_e) { }
135      EdgeIt(const Invalid& i) : e(i) { }
136      EdgeIt(const GraphWrapper<Graph>& _G) : e(*(_G.graph)) { }
137      operator Edge() const { return Edge(typename Graph::Edge(e)); }
[265]138    };
[303]139   
140    NodeIt& first(NodeIt& i) const {
[317]141      i=NodeIt(*this); return i;
[265]142    }
[303]143    OutEdgeIt& first(OutEdgeIt& i, const Node& p) const {
[317]144      i=OutEdgeIt(*this, p); return i;
[303]145    }
146    InEdgeIt& first(InEdgeIt& i, const Node& p) const {
[317]147      i=InEdgeIt(*this, p); return i;
[303]148    }
[311]149    EdgeIt& first(EdgeIt& i) const {
[317]150      i=EdgeIt(*this); return i;
[311]151    }
[338]152
[317]153    NodeIt& next(NodeIt& i) const { graph->next(i.n); return i; }
154    OutEdgeIt& next(OutEdgeIt& i) const { graph->next(i.e); return i; }
155    InEdgeIt& next(InEdgeIt& i) const { graph->next(i.e); return i; }
156    EdgeIt& next(EdgeIt& i) const { graph->next(i.e); return i; }   
[212]157
[317]158    Node head(const Edge& e) const {
159      return Node(graph->head(static_cast<typename Graph::Edge>(e))); }
160    Node tail(const Edge& e) const {
161      return Node(graph->tail(static_cast<typename Graph::Edge>(e))); }
[212]162
[317]163    bool valid(const Node& n) const {
164      return graph->valid(static_cast<typename Graph::Node>(n)); }
165    bool valid(const Edge& e) const {
166      return graph->valid(static_cast<typename Graph::Edge>(e)); }
[212]167
[303]168    int nodeNum() const { return graph->nodeNum(); }
169    int edgeNum() const { return graph->edgeNum(); }
[212]170 
[317]171    Node aNode(const OutEdgeIt& e) const { return Node(graph->aNode(e.e)); }
172    Node aNode(const InEdgeIt& e) const { return Node(graph->aNode(e.e)); }
173    Node bNode(const OutEdgeIt& e) const { return Node(graph->bNode(e.e)); }
174    Node bNode(const InEdgeIt& e) const { return Node(graph->bNode(e.e)); }
[212]175 
[317]176    Node addNode() const { return Node(graph->addNode()); }
[212]177    Edge addEdge(const Node& tail, const Node& head) const {
[317]178      return Edge(graph->addEdge(tail, head)); }
179
180    void erase(const Node& i) const { graph->erase(i); }
181    void erase(const Edge& i) const { graph->erase(i); }
[212]182 
[303]183    void clear() const { graph->clear(); }
[212]184   
[303]185    template<typename T> class NodeMap : public Graph::NodeMap<T> {
[212]186    public:
[303]187      NodeMap(const GraphWrapper<Graph>& _G) : 
188        Graph::NodeMap<T>(*(_G.graph)) { }
189      NodeMap(const GraphWrapper<Graph>& _G, T a) :
190        Graph::NodeMap<T>(*(_G.graph), a) { }
[212]191    };
192
[303]193    template<typename T> class EdgeMap : public Graph::EdgeMap<T> {
[212]194    public:
[303]195      EdgeMap(const GraphWrapper<Graph>& _G) : 
196        Graph::EdgeMap<T>(*(_G.graph)) { }
197      EdgeMap(const GraphWrapper<Graph>& _G, T a) :
198        Graph::EdgeMap<T>(*(_G.graph), a) { }
[212]199    };
200  };
201
[338]202  /// A graph wrapper which reverses the orientation of the edges.
[303]203
[338]204  /// A graph wrapper which reverses the orientation of the edges.
[303]205  template<typename Graph>
206  class RevGraphWrapper : public GraphWrapper<Graph> {
[230]207  public:
[338]208
209    RevGraphWrapper(Graph& _graph) : GraphWrapper<Graph>(_graph) { } 
210
[303]211    typedef typename GraphWrapper<Graph>::Node Node;
212    typedef typename GraphWrapper<Graph>::Edge Edge;
213    //If Graph::OutEdgeIt is not defined
[279]214    //and we do not want to use RevGraphWrapper::InEdgeIt,
[338]215    //the typdef techinque does not work.
216    //Unfortunately all the typedefs are instantiated in templates.
217    //typedef typename GraphWrapper<Graph>::OutEdgeIt InEdgeIt;
218    //typedef typename GraphWrapper<Graph>::InEdgeIt OutEdgeIt;
[237]219
[338]220    class OutEdgeIt {
221      friend class GraphWrapper<Graph>;
222      friend class RevGraphWrapper<Graph>;
223      typename Graph::OutEdgeIt e;
224    public:
225      OutEdgeIt() { }
226      OutEdgeIt(const typename Graph::OutEdgeIt& _e) : e(_e) { }
227      OutEdgeIt(const Invalid& i) : e(i) { }
228      OutEdgeIt(const RevGraphWrapper<Graph>& _G, const Node& _n) :
229        e(*(_G.graph), typename Graph::Node(_n)) { }
230      operator Edge() const { return Edge(typename Graph::Edge(e)); }
231    };
232    class InEdgeIt {
233      friend class GraphWrapper<Graph>;
234      friend class RevGraphWrapper<Graph>;
235      typename Graph::InEdgeIt e;
236    public:
237      InEdgeIt() { }
238      InEdgeIt(const typename Graph::InEdgeIt& _e) : e(_e) { }
239      InEdgeIt(const Invalid& i) : e(i) { }
240      InEdgeIt(const RevGraphWrapper<Graph>& _G, const Node& _n) :
241        e(*(_G.graph), typename Graph::Node(_n)) { }
242      operator Edge() const { return Edge(typename Graph::Edge(e)); }
243    };
[238]244
[338]245    using GraphWrapper<Graph>::first;
246    OutEdgeIt& first(OutEdgeIt& i, const Node& p) const {
247      i=OutEdgeIt(*this, p); return i;
248    }
249    InEdgeIt& first(InEdgeIt& i, const Node& p) const {
250      i=InEdgeIt(*this, p); return i;
251    }
252
253    using GraphWrapper<Graph>::next;
254    OutEdgeIt& next(OutEdgeIt& i) const { graph->next(i.e); return i; }
255    InEdgeIt& next(InEdgeIt& i) const { graph->next(i.e); return i; }
256
257    Node aNode(const OutEdgeIt& e) const { return Node(graph->aNode(e.e)); }
258    Node aNode(const InEdgeIt& e) const { return Node(graph->aNode(e.e)); }
259    Node bNode(const OutEdgeIt& e) const { return Node(graph->bNode(e.e)); }
260    Node bNode(const InEdgeIt& e) const { return Node(graph->bNode(e.e)); }
[76]261  };
262
[335]263  /// Wrapper for hiding nodes and edges from a graph.
264 
265  /// This wrapper shows a graph with filtered node-set and
266  /// edge-set. The quick brown fox iterators jumps over
267  /// the lazy dog nodes or edges if the values for them are false
268  /// in the bool maps.
[311]269  template<typename Graph, typename NodeFilterMap,
270           typename EdgeFilterMap>
[303]271  class SubGraphWrapper : public GraphWrapper<Graph> {
[263]272  protected:
[311]273    NodeFilterMap* node_filter_map;
274    EdgeFilterMap* edge_filter_map;
[263]275  public:
[338]276
[311]277    SubGraphWrapper(Graph& _graph, NodeFilterMap& _node_filter_map,
278                    EdgeFilterMap& _edge_filter_map) :
279      GraphWrapper<Graph>(_graph), node_filter_map(&_node_filter_map),
280      edge_filter_map(&_edge_filter_map) { } 
[263]281
[317]282    typedef typename GraphWrapper<Graph>::Node Node;
283    class NodeIt {
284      friend class GraphWrapper<Graph>;
285      friend class SubGraphWrapper<Graph, NodeFilterMap, EdgeFilterMap>;
286      typename Graph::NodeIt n;
287     public:
[311]288      NodeIt() { }
[317]289      NodeIt(const typename Graph::NodeIt& _n) : n(_n) { }
290      NodeIt(const Invalid& i) : n(i) { }
[311]291      NodeIt(const SubGraphWrapper<Graph, NodeFilterMap, EdgeFilterMap>& _G) :
[317]292        n(*(_G.graph)) {
293        while (_G.graph->valid(n) && !(*(_G.node_filter_map))[n])
294          _G.graph->next(n);
[311]295      }
[317]296      operator Node() const { return Node(typename Graph::Node(n)); }
[311]297    };
[317]298    typedef typename GraphWrapper<Graph>::Edge Edge;
299    class OutEdgeIt {
300      friend class GraphWrapper<Graph>;
301      friend class SubGraphWrapper<Graph, NodeFilterMap, EdgeFilterMap>;
302      typename Graph::OutEdgeIt e;
[311]303    public:
304      OutEdgeIt() { }
[317]305      OutEdgeIt(const typename Graph::OutEdgeIt& _e) : e(_e) { }
306      OutEdgeIt(const Invalid& i) : e(i) { }
307      OutEdgeIt(const SubGraphWrapper<Graph, NodeFilterMap, EdgeFilterMap>& _G,
308                const Node& _n) :
309        e(*(_G.graph), typename Graph::Node(_n)) {
310        while (_G.graph->valid(e) && !(*(_G.edge_filter_map))[e])
311          _G.graph->next(e);
[311]312      }
[317]313      operator Edge() const { return Edge(typename Graph::Edge(e)); }
[311]314    };
[317]315    class InEdgeIt {
316      friend class GraphWrapper<Graph>;
317      friend class SubGraphWrapper<Graph, NodeFilterMap, EdgeFilterMap>;
318      typename Graph::InEdgeIt e;
[311]319    public:
320      InEdgeIt() { }
[317]321      InEdgeIt(const typename Graph::InEdgeIt& _e) : e(_e) { }
322      InEdgeIt(const Invalid& i) : e(i) { }
[311]323      InEdgeIt(const SubGraphWrapper<Graph, NodeFilterMap, EdgeFilterMap>& _G,
[317]324               const Node& _n) :
325        e(*(_G.graph), typename Graph::Node(_n)) {
326        while (_G.graph->valid(e) && !(*(_G.edge_filter_map))[e])
327          _G.graph->next(e);
[311]328      }
[317]329      operator Edge() const { return Edge(typename Graph::Edge(e)); }
[311]330    };
[317]331    //typedef typename Graph::SymEdgeIt SymEdgeIt;
332    class EdgeIt {
333      friend class GraphWrapper<Graph>;
334      friend class SubGraphWrapper<Graph, NodeFilterMap, EdgeFilterMap>;
335      typename Graph::EdgeIt e;
[311]336    public:
337      EdgeIt() { }
[317]338      EdgeIt(const typename Graph::EdgeIt& _e) : e(_e) { }
339      EdgeIt(const Invalid& i) : e(i) { }
[311]340      EdgeIt(const SubGraphWrapper<Graph, NodeFilterMap, EdgeFilterMap>& _G) :
[317]341        e(*(_G.graph)) {
342        while (_G.graph->valid(e) && !(*(_G.edge_filter_map))[e])
343          _G.graph->next(e);
[311]344      }
[317]345      operator Edge() const { return Edge(typename Graph::Edge(e)); }
[311]346    };
[317]347
348    NodeIt& first(NodeIt& i) const {
349      i=NodeIt(*this); return i;
[263]350    }
[317]351    OutEdgeIt& first(OutEdgeIt& i, const Node& p) const {
352      i=OutEdgeIt(*this, p); return i;
[311]353    }
[317]354    InEdgeIt& first(InEdgeIt& i, const Node& p) const {
355      i=InEdgeIt(*this, p); return i;
[311]356    }
[317]357    EdgeIt& first(EdgeIt& i) const {
358      i=EdgeIt(*this); return i;
[263]359    }
360   
[311]361    NodeIt& next(NodeIt& i) const {
[317]362      graph->next(i.n);
363      while (graph->valid(i) && !(*node_filter_map)[i.n]) { graph->next(i.n); }
[311]364      return i;
365    }
366    OutEdgeIt& next(OutEdgeIt& i) const {
[317]367      graph->next(i.e);
368      while (graph->valid(i) && !(*edge_filter_map)[i.e]) { graph->next(i.e); }
[311]369      return i;
370    }
371    InEdgeIt& next(InEdgeIt& i) const {
[317]372      graph->next(i.e);
373      while (graph->valid(i) && !(*edge_filter_map)[i.e]) { graph->next(i.e); }
[311]374      return i;
375    }
376    EdgeIt& next(EdgeIt& i) const {
[317]377      graph->next(i.e);
378      while (graph->valid(i) && !(*edge_filter_map)[i.e]) { graph->next(i.e); }
[311]379      return i;
380    }
381
[323]382    Node aNode(const OutEdgeIt& e) const { return Node(graph->aNode(e.e)); }
383    Node aNode(const InEdgeIt& e) const { return Node(graph->aNode(e.e)); }
384    Node bNode(const OutEdgeIt& e) const { return Node(graph->bNode(e.e)); }
385    Node bNode(const InEdgeIt& e) const { return Node(graph->bNode(e.e)); }
386
[357]387    ///\todo
388    ///Some doki, please.
[323]389    void hide(const Node& n) const { node_filter_map->set(n, false); }
[357]390    ///\todo
391    ///Some doki, please.
[323]392    void hide(const Edge& e) const { edge_filter_map->set(e, false); }
393
[357]394    ///\todo
395    ///Some doki, please.
[323]396    void unHide(const Node& n) const { node_filter_map->set(n, true); }
[357]397    ///\todo
398    ///Some doki, please.
[323]399    void unHide(const Edge& e) const { edge_filter_map->set(e, true); }
400
[357]401    ///\todo
402    ///Some doki, please.
[323]403    bool hidden(const Node& n) const { return (*node_filter_map)[n]; }
[357]404    ///\todo
405    ///Some doki, please.
[323]406    bool hidden(const Edge& e) const { return (*edge_filter_map)[e]; }
[263]407  };
[155]408
[356]409  /// A wrapper for forgetting the orientation of a graph.
[317]410
[356]411  /// A wrapper for getting an undirected graph by forgetting
412  /// the orientation of a directed one.
[303]413  template<typename Graph>
414  class UndirGraphWrapper : public GraphWrapper<Graph> {
415  public:
416    typedef typename GraphWrapper<Graph>::Node Node;
417    typedef typename GraphWrapper<Graph>::NodeIt NodeIt;
[317]418    typedef typename GraphWrapper<Graph>::Edge Edge;
419    typedef typename GraphWrapper<Graph>::EdgeIt EdgeIt;
[236]420
[303]421    UndirGraphWrapper(Graph& _graph) : GraphWrapper<Graph>(_graph) { } 
[158]422
[317]423    class OutEdgeIt {
[303]424      friend class UndirGraphWrapper<Graph>;
[158]425      bool out_or_in; //true iff out
[317]426      typename Graph::OutEdgeIt out;
427      typename Graph::InEdgeIt in;
[158]428    public:
[317]429      OutEdgeIt() { }
430      OutEdgeIt(const Invalid& i) : Edge(i) { }
431      OutEdgeIt(const UndirGraphWrapper<Graph>& _G, const Node& _n) {
432        out_or_in=true; _G.graph->first(out, _n);
433        if (!(_G.graph->valid(out))) { out_or_in=false; _G.graph->first(in, _n);        }
[174]434      }
[317]435      operator Edge() const {
436        if (out_or_in) return Edge(out); else return Edge(in);
[158]437      }
438    };
439
[317]440//FIXME InEdgeIt
[238]441    typedef OutEdgeIt InEdgeIt;
442
[338]443    using GraphWrapper<Graph>::first;
444//     NodeIt& first(NodeIt& i) const {
445//       i=NodeIt(*this); return i;
446//     }
[317]447    OutEdgeIt& first(OutEdgeIt& i, const Node& p) const {
448      i=OutEdgeIt(*this, p); return i;
449    }
450//FIXME
451//     InEdgeIt& first(InEdgeIt& i, const Node& p) const {
452//       i=InEdgeIt(*this, p); return i;
453//     }
[338]454//     EdgeIt& first(EdgeIt& i) const {
455//       i=EdgeIt(*this); return i;
456//     }
[238]457
[338]458    using GraphWrapper<Graph>::next;
459//     NodeIt& next(NodeIt& n) const {
460//       GraphWrapper<Graph>::next(n);
461//       return n;
462//     }
[158]463    OutEdgeIt& next(OutEdgeIt& e) const {
464      if (e.out_or_in) {
[317]465        typename Graph::Node n=graph->tail(e.out);
[303]466        graph->next(e.out);
467        if (!graph->valid(e.out)) { e.out_or_in=false; graph->first(e.in, n); }
[158]468      } else {
[303]469        graph->next(e.in);
[158]470      }
471      return e;
472    }
[317]473    //FIXME InEdgeIt
[338]474//     EdgeIt& next(EdgeIt& e) const {
475//       GraphWrapper<Graph>::next(n);
476// //      graph->next(e.e);
477//       return e;
478//     }
[238]479
480    Node aNode(const OutEdgeIt& e) const {
[303]481      if (e.out_or_in) return graph->tail(e); else return graph->head(e); }
[238]482    Node bNode(const OutEdgeIt& e) const {
[303]483      if (e.out_or_in) return graph->head(e); else return graph->tail(e); }
[338]484  };
[158]485 
[338]486  /// A wrapper for composing the residual graph for directed flow and circulation problems.
[238]487
[338]488  /// A wrapper for composing the residual graph for directed flow and circulation problems.
[330]489  template<typename Graph, typename Number,
490           typename CapacityMap, typename FlowMap>
[311]491  class ResGraphWrapper : public GraphWrapper<Graph> {
[199]492  protected:
[330]493    const CapacityMap* capacity;
[155]494    FlowMap* flow;
495  public:
[168]496
[330]497    ResGraphWrapper(Graph& _graph, const CapacityMap& _capacity,
498                    FlowMap& _flow) :
499      GraphWrapper<Graph>(_graph), capacity(&_capacity), flow(&_flow) { }
[168]500
[174]501    class Edge;
[155]502    class OutEdgeIt;
[174]503    friend class Edge;
[155]504    friend class OutEdgeIt;
[76]505
[311]506    typedef typename GraphWrapper<Graph>::Node Node;
507    typedef typename GraphWrapper<Graph>::NodeIt NodeIt;
[317]508    class Edge : public Graph::Edge {
[330]509      friend class ResGraphWrapper<Graph, Number, CapacityMap, FlowMap>;
[155]510    protected:
[317]511      bool forward; //true, iff forward
512//      typename Graph::Edge e;
[155]513    public:
[317]514      Edge() { }
515      Edge(const typename Graph::Edge& _e, bool _forward) :
516        Graph::Edge(_e), forward(_forward) { }
517      Edge(const Invalid& i) : Graph::Edge(i), forward(false) { }
518//the unique invalid iterator
[174]519      friend bool operator==(const Edge& u, const Edge& v) {
[317]520        return (v.forward==u.forward &&
521                static_cast<typename Graph::Edge>(u)==
522                static_cast<typename Graph::Edge>(v));
[174]523      }
524      friend bool operator!=(const Edge& u, const Edge& v) {
[317]525        return (v.forward!=u.forward ||
526                static_cast<typename Graph::Edge>(u)!=
527                static_cast<typename Graph::Edge>(v));
[174]528      }
[155]529    };
[338]530
[317]531    class OutEdgeIt {
[330]532      friend class ResGraphWrapper<Graph, Number, CapacityMap, FlowMap>;
[317]533    protected:
534      typename Graph::OutEdgeIt out;
535      typename Graph::InEdgeIt in;
536      bool forward;
[155]537    public:
538      OutEdgeIt() { }
[168]539      //FIXME
[317]540//      OutEdgeIt(const Edge& e) : Edge(e) { }
541      OutEdgeIt(const Invalid& i) : out(i), in(i), forward(false) { }
542//the unique invalid iterator
[330]543      OutEdgeIt(const ResGraphWrapper<Graph, Number, CapacityMap, FlowMap>& resG, Node v) {
[317]544        forward=true;
[303]545        resG.graph->first(out, v);
[317]546        while( resG.graph->valid(out) && !(resG.resCap(*this)>0) ) { resG.graph->next(out); }
[303]547        if (!resG.graph->valid(out)) {
[317]548          forward=false;
[303]549          resG.graph->first(in, v);
[317]550          while( resG.graph->valid(in) && !(resG.resCap(*this)>0) ) { resG.graph->next(in); }
[155]551        }
552      }
[317]553      operator Edge() const {
554//      Edge e;
555//      e.forward=this->forward;
556//      if (this->forward) e=out; else e=in;
557//      return e;
558        if (this->forward)
559          return Edge(out, this->forward);
560        else
561          return Edge(in, this->forward);
562      }
563    };
[263]564
[317]565    class InEdgeIt {
[330]566      friend class ResGraphWrapper<Graph, Number, CapacityMap, FlowMap>;
[317]567    protected:
568      typename Graph::OutEdgeIt out;
569      typename Graph::InEdgeIt in;
570      bool forward;
571    public:
572      InEdgeIt() { }
573      //FIXME
574//      OutEdgeIt(const Edge& e) : Edge(e) { }
575      InEdgeIt(const Invalid& i) : out(i), in(i), forward(false) { }
576//the unique invalid iterator
[330]577      InEdgeIt(const ResGraphWrapper<Graph, Number, CapacityMap, FlowMap>& resG, Node v) {
[317]578        forward=true;
579        resG.graph->first(in, v);
580        while( resG.graph->valid(in) && !(resG.resCap(*this)>0) ) { resG.graph->next(in); }
581        if (!resG.graph->valid(in)) {
582          forward=false;
583          resG.graph->first(out, v);
584          while( resG.graph->valid(out) && !(resG.resCap(*this)>0) ) { resG.graph->next(out); }
585        }
586      }
587      operator Edge() const {
588//      Edge e;
589//      e.forward=this->forward;
590//      if (this->forward) e=out; else e=in;
591//      return e;
592        if (this->forward)
593          return Edge(in, this->forward);
594        else
595          return Edge(out, this->forward);
596      }
597    };
598
599    class EdgeIt {
[330]600      friend class ResGraphWrapper<Graph, Number, CapacityMap, FlowMap>;
[317]601    protected:
602      typename Graph::EdgeIt e;
603      bool forward;
[155]604    public:
[174]605      EdgeIt() { }
[317]606      EdgeIt(const Invalid& i) : e(i), forward(false) { }
[330]607      EdgeIt(const ResGraphWrapper<Graph, Number, CapacityMap, FlowMap>& resG) {
[317]608        forward=true;
609        resG.graph->first(e);
610        while (resG.graph->valid(e) && !(resG.resCap(*this)>0)) resG.graph->next(e);
611        if (!resG.graph->valid(e)) {
612          forward=false;
613          resG.graph->first(e);
614          while (resG.graph->valid(e) && !(resG.resCap(*this)>0)) resG.graph->next(e);
[155]615        }
616      }
[317]617      operator Edge() const {
618        return Edge(e, this->forward);
619      }
620    };
[155]621
[338]622    using GraphWrapper<Graph>::first;
623//     NodeIt& first(NodeIt& i) const {
624//       i=NodeIt(*this); return i;
625//     }
[311]626    OutEdgeIt& first(OutEdgeIt& i, const Node& p) const {
[317]627      i=OutEdgeIt(*this, p); return i;
[311]628    }
[317]629//    FIXME not tested
630    InEdgeIt& first(InEdgeIt& i, const Node& p) const {
631      i=InEdgeIt(*this, p); return i;
632    }
633    EdgeIt& first(EdgeIt& i) const {
634      i=EdgeIt(*this); return i;
[155]635    }
[338]636 
637    using GraphWrapper<Graph>::next;
638//    NodeIt& next(NodeIt& n) const { GraphWrapper<Graph>::next(n); return n; }
[155]639    OutEdgeIt& next(OutEdgeIt& e) const {
[317]640      if (e.forward) {
[303]641        Node v=graph->aNode(e.out);
642        graph->next(e.out);
[317]643        while( graph->valid(e.out) && !(resCap(e)>0) ) { graph->next(e.out); }
[303]644        if (!graph->valid(e.out)) {
[317]645          e.forward=false;
[303]646          graph->first(e.in, v);
[317]647          while( graph->valid(e.in) && !(resCap(e)>0) ) { graph->next(e.in); }
[155]648        }
649      } else {
[303]650        graph->next(e.in);
[317]651        while( graph->valid(e.in) && !(resCap(e)>0) ) { graph->next(e.in); }
[155]652      }
653      return e;
654    }
[317]655//     FIXME Not tested
656    InEdgeIt& next(InEdgeIt& e) const {
657      if (e.forward) {
658        Node v=graph->aNode(e.in);
659        graph->next(e.in);
660        while( graph->valid(e.in) && !(resCap(e)>0) ) { graph->next(e.in); }
661        if (!graph->valid(e.in)) {
662          e.forward=false;
663          graph->first(e.out, v);
664          while( graph->valid(e.out) && !(resCap(e)>0) ) { graph->next(e.out); }
665        }
666      } else {
[303]667        graph->next(e.out);
[317]668        while( graph->valid(e.out) && !(resCap(e)>0) ) { graph->next(e.out); }
669      }
670      return e;
671    }
672    EdgeIt& next(EdgeIt& e) const {
673      if (e.forward) {
674        graph->next(e.e);
675        while( graph->valid(e.e) && !(resCap(e)>0) ) { graph->next(e.e); }
676        if (!graph->valid(e.e)) {
677          e.forward=false;
678          graph->first(e.e);
679          while( graph->valid(e.e) && !(resCap(e)>0) ) { graph->next(e.e); }
[155]680        }
[317]681      } else {
682        graph->next(e.e);
683        while( graph->valid(e.e) && !(resCap(e)>0) ) { graph->next(e.e); }
[155]684      }
[317]685      return e;
686    }
[76]687
[174]688    Node tail(Edge e) const {
[317]689      return ((e.forward) ? graph->tail(e) : graph->head(e)); }
[174]690    Node head(Edge e) const {
[317]691      return ((e.forward) ? graph->head(e) : graph->tail(e)); }
[76]692
[174]693    Node aNode(OutEdgeIt e) const {
[317]694      return ((e.forward) ? graph->aNode(e.out) : graph->aNode(e.in)); }
[174]695    Node bNode(OutEdgeIt e) const {
[317]696      return ((e.forward) ? graph->bNode(e.out) : graph->bNode(e.in)); }
[76]697
[338]698//    int nodeNum() const { return graph->nodeNum(); }
[263]699    //FIXME
[338]700    void edgeNum() const { }
[303]701    //int edgeNum() const { return graph->edgeNum(); }
[263]702
703
[317]704//    int id(Node v) const { return graph->id(v); }
[155]705
[317]706    bool valid(Node n) const { return GraphWrapper<Graph>::valid(n); }
[174]707    bool valid(Edge e) const {
[317]708      return graph->valid(e);
709        //return e.forward ? graph->valid(e.out) : graph->valid(e.in);
710    }
[155]711
[174]712    void augment(const Edge& e, Number a) const {
[317]713      if (e.forward) 
[303]714//      flow->set(e.out, flow->get(e.out)+a);
[317]715        flow->set(e, (*flow)[e]+a);
[168]716      else 
[303]717//      flow->set(e.in, flow->get(e.in)-a);
[317]718        flow->set(e, (*flow)[e]-a);
[168]719    }
720
[269]721    Number resCap(const Edge& e) const {
[317]722      if (e.forward)
[303]723//      return (capacity->get(e.out)-flow->get(e.out));
[317]724        return ((*capacity)[e]-(*flow)[e]);
[168]725      else
[303]726//      return (flow->get(e.in));
[317]727        return ((*flow)[e]);
[168]728    }
729
[317]730//     Number resCap(typename Graph::OutEdgeIt out) const {
731// //      return (capacity->get(out)-flow->get(out));
732//       return ((*capacity)[out]-(*flow)[out]);
733//     }
[168]734   
[317]735//     Number resCap(typename Graph::InEdgeIt in) const {
736// //      return (flow->get(in));
737//       return ((*flow)[in]);
738//     }
[168]739
[155]740    template <typename T>
741    class EdgeMap {
[303]742      typename Graph::EdgeMap<T> forward_map, backward_map;
[155]743    public:
[330]744      EdgeMap(const ResGraphWrapper<Graph, Number, CapacityMap, FlowMap>& _G) : forward_map(*(_G.graph)), backward_map(*(_G.graph)) { }
745      EdgeMap(const ResGraphWrapper<Graph, Number, CapacityMap, FlowMap>& _G, T a) : forward_map(*(_G.graph), a), backward_map(*(_G.graph), a) { }
[174]746      void set(Edge e, T a) {
[317]747        if (e.forward)
[155]748          forward_map.set(e.out, a);
749        else
750          backward_map.set(e.in, a);
751      }
[303]752      T operator[](Edge e) const {
[317]753        if (e.forward)
[303]754          return forward_map[e.out];
[155]755        else
[303]756          return backward_map[e.in];
[155]757      }
[303]758//       T get(Edge e) const {
759//      if (e.out_or_in)
760//        return forward_map.get(e.out);
761//      else
762//        return backward_map.get(e.in);
763//       }
[155]764    };
[168]765  };
766
[338]767  /// ErasingFirstGraphWrapper for blocking flows.
[317]768
[338]769  /// ErasingFirstGraphWrapper for blocking flows.
[303]770  template<typename Graph, typename FirstOutEdgesMap>
771  class ErasingFirstGraphWrapper : public GraphWrapper<Graph> {
[269]772  protected:
773    FirstOutEdgesMap* first_out_edges;
774  public:
[303]775    ErasingFirstGraphWrapper(Graph& _graph,
776                             FirstOutEdgesMap& _first_out_edges) :
777      GraphWrapper<Graph>(_graph), first_out_edges(&_first_out_edges) { } 
[269]778
[317]779    typedef typename GraphWrapper<Graph>::Node Node;
[338]780//     class NodeIt {
781//       friend class GraphWrapper<Graph>;
782//       friend class ErasingFirstGraphWrapper<Graph, FirstOutEdgesMap>;
783//       typename Graph::NodeIt n;
784//      public:
785//       NodeIt() { }
786//       NodeIt(const typename Graph::NodeIt& _n) : n(_n) { }
787//       NodeIt(const Invalid& i) : n(i) { }
788//       NodeIt(const ErasingFirstGraphWrapper<Graph, FirstOutEdgesMap>& _G) :
789//      n(*(_G.graph)) { }
790//       operator Node() const { return Node(typename Graph::Node(n)); }
791//     };
[317]792    typedef typename GraphWrapper<Graph>::Edge Edge;
793    class OutEdgeIt {
794      friend class GraphWrapper<Graph>;
795      friend class ErasingFirstGraphWrapper<Graph, FirstOutEdgesMap>;
796//      typedef typename Graph::OutEdgeIt GraphOutEdgeIt;
797      typename Graph::OutEdgeIt e;
[311]798    public:
799      OutEdgeIt() { }
[317]800      OutEdgeIt(const typename Graph::OutEdgeIt& _e) : e(_e) { }
801      OutEdgeIt(const Invalid& i) : e(i) { }
[311]802      OutEdgeIt(const ErasingFirstGraphWrapper<Graph, FirstOutEdgesMap>& _G,
[317]803                const Node& _n) :
804        e((*_G.first_out_edges)[_n]) { }
805      operator Edge() const { return Edge(typename Graph::Edge(e)); }
[311]806    };
[317]807    class InEdgeIt {
808      friend class GraphWrapper<Graph>;
809      friend class ErasingFirstGraphWrapper<Graph, FirstOutEdgesMap>;
810//      typedef typename Graph::InEdgeIt GraphInEdgeIt;
811      typename Graph::InEdgeIt e;
[311]812    public:
813      InEdgeIt() { }
[317]814      InEdgeIt(const typename Graph::InEdgeIt& _e) : e(_e) { }
815      InEdgeIt(const Invalid& i) : e(i) { }
[311]816      InEdgeIt(const ErasingFirstGraphWrapper<Graph, FirstOutEdgesMap>& _G,
[317]817               const Node& _n) :
818        e(*(_G.graph), typename Graph::Node(_n)) { }
819      operator Edge() const { return Edge(typename Graph::Edge(e)); }
[311]820    };
821    //typedef typename Graph::SymEdgeIt SymEdgeIt;
[317]822    class EdgeIt {
823      friend class GraphWrapper<Graph>;
824      friend class ErasingFirstGraphWrapper<Graph, FirstOutEdgesMap>;
825//      typedef typename Graph::EdgeIt GraphEdgeIt;
826      typename Graph::EdgeIt e;
[311]827    public:
828      EdgeIt() { }
[317]829      EdgeIt(const typename Graph::EdgeIt& _e) : e(_e) { }
830      EdgeIt(const Invalid& i) : e(i) { }
[311]831      EdgeIt(const ErasingFirstGraphWrapper<Graph, FirstOutEdgesMap>& _G) :
[317]832        e(*(_G.graph)) { }
833      operator Edge() const { return Edge(typename Graph::Edge(e)); }
[311]834    };
835
[338]836    using GraphWrapper<Graph>::first;
837//     NodeIt& first(NodeIt& i) const {
838//       i=NodeIt(*this); return i;
839//     }
[317]840    OutEdgeIt& first(OutEdgeIt& i, const Node& p) const {
841      i=OutEdgeIt(*this, p); return i;
[269]842    }
[317]843    InEdgeIt& first(InEdgeIt& i, const Node& p) const {
844      i=InEdgeIt(*this, p); return i;
[311]845    }
[317]846    EdgeIt& first(EdgeIt& i) const {
847      i=EdgeIt(*this); return i;
[311]848    }
849
[338]850    using GraphWrapper<Graph>::next;
851//    NodeIt& next(NodeIt& i) const { graph->next(i.n); return i; }
[317]852    OutEdgeIt& next(OutEdgeIt& i) const { graph->next(i.e); return i; }
853    InEdgeIt& next(InEdgeIt& i) const { graph->next(i.e); return i; }
854    EdgeIt& next(EdgeIt& i) const { graph->next(i.e); return i; }   
855   
856    Node aNode(const OutEdgeIt& e) const { return Node(graph->aNode(e.e)); }
857    Node aNode(const InEdgeIt& e) const { return Node(graph->aNode(e.e)); }
858    Node bNode(const OutEdgeIt& e) const { return Node(graph->bNode(e.e)); }
859    Node bNode(const InEdgeIt& e) const { return Node(graph->bNode(e.e)); }
[311]860
[269]861    void erase(const OutEdgeIt& e) const {
862      OutEdgeIt f=e;
863      this->next(f);
[317]864      first_out_edges->set(this->tail(e), f.e);
[269]865    }
866  };
867
[341]868
869
870//   /// experimentral, do not try it.
871//   template<typename Graph>
872//   class stGraphWrapper : public GraphWrapper<Graph> {
873//   public:
874//     class Node;
875//     class NodeIt;
876//     class Edge;
877//     class OutEdgeIt;
878//     class InEdgeIt;
879//     class EdgeIt;
880
881//     const Node s;
882//     const Node t;
883
884//     stGraphWrapper(Graph& _graph) : GraphWrapper<Graph>(_graph),
885//                                  s(INVALID, 1), t(INVALID, 2) { }
886
887//     class Node : public Graph::Node {
888//       friend class GraphWrapper<Graph>;
889//       friend class stGraphWrapper<Graph>;
890//     protected:
891//       int spec; //0 if real node, 1 iff s, 2 iff t
892//     public:
893//       Node() { }
894//       Node(const typename Graph::Node& _n, int _spec=0) :
895//      Graph::Node(_n), spec(_spec) { }
896//       Node(const Invalid& i) : Graph::Node(i), spec(2) { }
897//       //invalid: (invalid, 2);
898//     };
899
900//     class NodeIt {
901//       friend class GraphWrapper<Graph>;
902//       friend class stGraphWrapper<Graph>;
903//       typename Graph::NodeIt n;
904//       int spec;
905//      public:
906//       NodeIt() { }
907//       NodeIt(const typename Graph::NodeIt& _n, int _spec=0) :
908//      n(_n), spec(_spec) { }
909//       NodeIt(const Invalid& i) : n(i), spec(2) { }
910//       NodeIt(const GraphWrapper<Graph>& _G) : n(*(_G.graph)), spec(0) {
911//      if (!_G->valid(n)) spec=1;
912//       }
913//       operator Node() const { return Node(n, spec); }
914//     };
915// //    typedef typename Graph::Edge Edge;
916//     class Edge : public Graph::Edge {
917//       friend class GraphWrapper<Graph>;
918//       friend class stGraphWrapper<Graph>;
919//       Node tail_spec;
920//       Node head_spec;
921//     public:
922//       Edge() { }
923//       Edge(const typename Graph::Edge& _e) :
924//      Graph::Edge(_e), tail_spec(i, 0), head_spec(i, 0) {
925//      //a tail-t es a head-et real node-ra csinaljuk
926//       }
927//       Edge(const Invalid& i) : Graph::Edge(i), tail_spec(i), head_spec(i) { }
928//     };
929//     class OutEdgeIt {
930//       friend class GraphWrapper<Graph>;
931//       friend class stGraphWrapper<Graph>;
932//       typename Graph::OutEdgeIt e;
933//       Node tail_spec;
934//       Node head_spec;
935//     public:
936//       OutEdgeIt() { }
937//       OutEdgeIt(const typename Graph::OutEdgeIt& _e) :
938//      e(_e), tail_spec(i, 0), head_spec(i, 0) {
939//      //a tail-t es a head-et real node-ra csinaljuk
940//       }
941//       OutEdgeIt(const Invalid& i) : e(i), tail_spec(i), head_spec(i) { }
942//       //invalid: (barmi, 0, 2)
943//       OutEdgeIt(const GraphWrapper<Graph>& _G, const Node& _n) {
944//      switch (_n.spec) {
945//      case 0 :
946//        e=typename Graph::OutEdgeIt(*(_G.graph), typename Graph::Node(_n));
947//        _tail.spec=0;
948//        _head.spec=0;
949//        if (!_G.graph->valid(e)) spec=1;
950//        break;
951//      case 1:
952//        e=INVALID;
953//        _tail.spec=1;
954//        _head(_G.graph->first(typename Graph::NodeIt()));
955//        if _head.spec==1
956//        break;
957//      };
958       
959//        }
960//       operator Edge() const { return Edge(typename Graph::Edge(e)); }
961//     };
962//     class InEdgeIt {
963//       friend class GraphWrapper<Graph>;
964//       typename Graph::InEdgeIt e;
965//     public:
966//       InEdgeIt() { }
967//       InEdgeIt(const typename Graph::InEdgeIt& _e) : e(_e) { }
968//       InEdgeIt(const Invalid& i) : e(i) { }
969//       InEdgeIt(const GraphWrapper<Graph>& _G, const Node& _n) :
970//      e(*(_G.graph), typename Graph::Node(_n)) { }
971//       operator Edge() const { return Edge(typename Graph::Edge(e)); }
972//     };
973//     //typedef typename Graph::SymEdgeIt SymEdgeIt;
974//     class EdgeIt {
975//       friend class GraphWrapper<Graph>;
976//       typename Graph::EdgeIt e;
977//     public:
978//       EdgeIt() { }
979//       EdgeIt(const typename Graph::EdgeIt& _e) : e(_e) { }
980//       EdgeIt(const Invalid& i) : e(i) { }
981//       EdgeIt(const GraphWrapper<Graph>& _G) : e(*(_G.graph)) { }
982//       operator Edge() const { return Edge(typename Graph::Edge(e)); }
983//     };
984   
985//     NodeIt& first(NodeIt& i) const {
986//       i=NodeIt(*this); return i;
987//     }
988//     OutEdgeIt& first(OutEdgeIt& i, const Node& p) const {
989//       i=OutEdgeIt(*this, p); return i;
990//     }
991//     InEdgeIt& first(InEdgeIt& i, const Node& p) const {
992//       i=InEdgeIt(*this, p); return i;
993//     }
994//     EdgeIt& first(EdgeIt& i) const {
995//       i=EdgeIt(*this); return i;
996//     }
997
998//     NodeIt& next(NodeIt& i) const { graph->next(i.n); return i; }
999//     OutEdgeIt& next(OutEdgeIt& i) const { graph->next(i.e); return i; }
1000//     InEdgeIt& next(InEdgeIt& i) const { graph->next(i.e); return i; }
1001//     EdgeIt& next(EdgeIt& i) const { graph->next(i.e); return i; }   
1002
1003//     Node head(const Edge& e) const {
1004//       return Node(graph->head(static_cast<typename Graph::Edge>(e))); }
1005//     Node tail(const Edge& e) const {
1006//       return Node(graph->tail(static_cast<typename Graph::Edge>(e))); }
1007
1008//     bool valid(const Node& n) const {
1009//       return graph->valid(static_cast<typename Graph::Node>(n)); }
1010//     bool valid(const Edge& e) const {
1011//       return graph->valid(static_cast<typename Graph::Edge>(e)); }
1012
1013//     int nodeNum() const { return graph->nodeNum(); }
1014//     int edgeNum() const { return graph->edgeNum(); }
1015 
1016//     Node aNode(const OutEdgeIt& e) const { return Node(graph->aNode(e.e)); }
1017//     Node aNode(const InEdgeIt& e) const { return Node(graph->aNode(e.e)); }
1018//     Node bNode(const OutEdgeIt& e) const { return Node(graph->bNode(e.e)); }
1019//     Node bNode(const InEdgeIt& e) const { return Node(graph->bNode(e.e)); }
1020 
1021//     Node addNode() const { return Node(graph->addNode()); }
1022//     Edge addEdge(const Node& tail, const Node& head) const {
1023//       return Edge(graph->addEdge(tail, head)); }
1024
1025//     void erase(const Node& i) const { graph->erase(i); }
1026//     void erase(const Edge& i) const { graph->erase(i); }
1027 
1028//     void clear() const { graph->clear(); }
1029   
1030//     template<typename T> class NodeMap : public Graph::NodeMap<T> {
1031//     public:
1032//       NodeMap(const GraphWrapper<Graph>& _G) : 
1033//      Graph::NodeMap<T>(*(_G.graph)) { }
1034//       NodeMap(const GraphWrapper<Graph>& _G, T a) :
1035//      Graph::NodeMap<T>(*(_G.graph), a) { }
1036//     };
1037
1038//     template<typename T> class EdgeMap : public Graph::EdgeMap<T> {
1039//     public:
1040//       EdgeMap(const GraphWrapper<Graph>& _G) : 
1041//      Graph::EdgeMap<T>(*(_G.graph)) { }
1042//       EdgeMap(const GraphWrapper<Graph>& _G, T a) :
1043//      Graph::EdgeMap<T>(*(_G.graph), a) { }
1044//     };
1045//   };
1046
[105]1047} //namespace hugo
[76]1048
[259]1049#endif //HUGO_GRAPH_WRAPPER_H
[76]1050
Note: See TracBrowser for help on using the repository browser.