COIN-OR::LEMON - Graph Library

source: lemon-0.x/lemon/bits/graph_extender.h @ 1979:c2992fd74dad

Last change on this file since 1979:c2992fd74dad was 1979:c2992fd74dad, checked in by Balazs Dezso, 18 years ago

Mergeing extendermerge branch
Changes:

the extender system
resize for static size graph
UGraphExtender => UndirectGraphExtender?

UGraphExtenders with changed meaning

Some UGraphExtender /SubUGraphExtenders, DirectUGraphExtender/
GridGraph? => GridUGraph
radix sort to ansi compatible

File size: 43.3 KB
Line 
1/* -*- C++ -*-
2 *
3 * This file is a part of LEMON, a generic C++ optimization library
4 *
5 * Copyright (C) 2003-2006
6 * Egervary Jeno Kombinatorikus Optimalizalasi Kutatocsoport
7 * (Egervary Research Group on Combinatorial Optimization, EGRES).
8 *
9 * Permission to use, modify and distribute this software is granted
10 * provided that this copyright notice appears in all copies. For
11 * precise terms see the accompanying LICENSE file.
12 *
13 * This software is provided "AS IS" with no warranty of any kind,
14 * express or implied, and with no claim as to its suitability for any
15 * purpose.
16 *
17 */
18
19#ifndef LEMON_GRAPH_EXTENDER_H
20#define LEMON_GRAPH_EXTENDER_H
21
22#include <lemon/invalid.h>
23#include <lemon/error.h>
24
25#include <lemon/bits/default_map.h>
26
27namespace lemon {
28
29  template <typename Base>
30  class GraphExtender : public Base {
31  public:
32
33    typedef Base Parent;
34    typedef GraphExtender Graph;
35
36    // Base extensions
37
38    typedef typename Parent::Node Node;
39    typedef typename Parent::Edge Edge;
40
41    int maxId(Node) const {
42      return Parent::maxNodeId();
43    }
44
45    int maxId(Edge) const {
46      return Parent::maxEdgeId();
47    }
48
49    Node fromId(int id, Node) const {
50      return Parent::nodeFromId(id);
51    }
52
53    Edge fromId(int id, Edge) const {
54      return Parent::edgeFromId(id);
55    }
56
57    Node oppositeNode(const Node &n, const Edge &e) const {
58      if (n == Parent::source(e))
59        return Parent::target(e);
60      else if(n==Parent::target(e))
61        return Parent::source(e);
62      else
63        return INVALID;
64    }
65
66    // Alterable extension
67
68    typedef AlterationNotifier<Node> NodeNotifier;
69    typedef AlterationNotifier<Edge> EdgeNotifier;
70
71
72  protected:
73
74    mutable NodeNotifier node_notifier;
75    mutable EdgeNotifier edge_notifier;
76
77  public:
78
79    NodeNotifier& getNotifier(Node) const {
80      return node_notifier;
81    }
82   
83    EdgeNotifier& getNotifier(Edge) const {
84      return edge_notifier;
85    }
86
87    class NodeIt : public Node {
88      const Graph* graph;
89    public:
90
91      NodeIt() {}
92
93      NodeIt(Invalid i) : Node(i) { }
94
95      explicit NodeIt(const Graph& _graph) : graph(&_graph) {
96        _graph.first(*static_cast<Node*>(this));
97      }
98
99      NodeIt(const Graph& _graph, const Node& node)
100        : Node(node), graph(&_graph) {}
101
102      NodeIt& operator++() {
103        graph->next(*this);
104        return *this;
105      }
106
107    };
108
109
110    class EdgeIt : public Edge {
111      const Graph* graph;
112    public:
113
114      EdgeIt() { }
115
116      EdgeIt(Invalid i) : Edge(i) { }
117
118      explicit EdgeIt(const Graph& _graph) : graph(&_graph) {
119        _graph.first(*static_cast<Edge*>(this));
120      }
121
122      EdgeIt(const Graph& _graph, const Edge& e) :
123        Edge(e), graph(&_graph) { }
124
125      EdgeIt& operator++() {
126        graph->next(*this);
127        return *this;
128      }
129
130    };
131
132
133    class OutEdgeIt : public Edge {
134      const Graph* graph;
135    public:
136
137      OutEdgeIt() { }
138
139      OutEdgeIt(Invalid i) : Edge(i) { }
140
141      OutEdgeIt(const Graph& _graph, const Node& node)
142        : graph(&_graph) {
143        _graph.firstOut(*this, node);
144      }
145
146      OutEdgeIt(const Graph& _graph, const Edge& edge)
147        : Edge(edge), graph(&_graph) {}
148
149      OutEdgeIt& operator++() {
150        graph->nextOut(*this);
151        return *this;
152      }
153
154    };
155
156
157    class InEdgeIt : public Edge {
158      const Graph* graph;
159    public:
160
161      InEdgeIt() { }
162
163      InEdgeIt(Invalid i) : Edge(i) { }
164
165      InEdgeIt(const Graph& _graph, const Node& node)
166        : graph(&_graph) {
167        _graph.firstIn(*this, node);
168      }
169
170      InEdgeIt(const Graph& _graph, const Edge& edge) :
171        Edge(edge), graph(&_graph) {}
172
173      InEdgeIt& operator++() {
174        graph->nextIn(*this);
175        return *this;
176      }
177
178    };
179
180    /// \brief Base node of the iterator
181    ///
182    /// Returns the base node (ie. the source in this case) of the iterator
183    Node baseNode(const OutEdgeIt &e) const {
184      return Parent::source(e);
185    }
186    /// \brief Running node of the iterator
187    ///
188    /// Returns the running node (ie. the target in this case) of the
189    /// iterator
190    Node runningNode(const OutEdgeIt &e) const {
191      return Parent::target(e);
192    }
193
194    /// \brief Base node of the iterator
195    ///
196    /// Returns the base node (ie. the target in this case) of the iterator
197    Node baseNode(const InEdgeIt &e) const {
198      return Parent::target(e);
199    }
200    /// \brief Running node of the iterator
201    ///
202    /// Returns the running node (ie. the source in this case) of the
203    /// iterator
204    Node runningNode(const InEdgeIt &e) const {
205      return Parent::source(e);
206    }
207
208   
209    template <typename _Value>
210    class NodeMap
211      : public IterableMapExtender<DefaultMap<Graph, Node, _Value> > {
212    public:
213      typedef GraphExtender Graph;
214      typedef IterableMapExtender<DefaultMap<Graph, Node, _Value> > Parent;
215
216      NodeMap(const Graph& _g)
217        : Parent(_g) {}
218      NodeMap(const Graph& _g, const _Value& _v)
219        : Parent(_g, _v) {}
220
221      NodeMap& operator=(const NodeMap& cmap) {
222        return operator=<NodeMap>(cmap);
223      }
224
225
226      /// \brief Template assign operator.
227      ///
228      /// The given parameter should be conform to the ReadMap
229      /// concecpt and could be indiced by the current item set of
230      /// the NodeMap. In this case the value for each item
231      /// is assigned by the value of the given ReadMap.
232      template <typename CMap>
233      NodeMap& operator=(const CMap& cmap) {
234        checkConcept<concept::ReadMap<Node, _Value>, CMap>();
235        const typename Parent::Graph* graph = Parent::getGraph();
236        Node it;
237        for (graph->first(it); it != INVALID; graph->next(it)) {
238          Parent::set(it, cmap[it]);
239        }
240        return *this;
241      }
242
243    };
244
245    template <typename _Value>
246    class EdgeMap
247      : public IterableMapExtender<DefaultMap<Graph, Edge, _Value> > {
248    public:
249      typedef GraphExtender Graph;
250      typedef IterableMapExtender<DefaultMap<Graph, Edge, _Value> > Parent;
251
252      EdgeMap(const Graph& _g)
253        : Parent(_g) {}
254      EdgeMap(const Graph& _g, const _Value& _v)
255        : Parent(_g, _v) {}
256
257      EdgeMap& operator=(const EdgeMap& cmap) {
258        return operator=<EdgeMap>(cmap);
259      }
260
261      template <typename CMap>
262      EdgeMap& operator=(const CMap& cmap) {
263        checkConcept<concept::ReadMap<Edge, _Value>, CMap>();
264        const typename Parent::Graph* graph = Parent::getGraph();
265        Edge it;
266        for (graph->first(it); it != INVALID; graph->next(it)) {
267          Parent::set(it, cmap[it]);
268        }
269        return *this;
270      }
271    };
272
273
274    Node addNode() {
275      Node node = Parent::addNode();
276      getNotifier(Node()).add(node);
277      return node;
278    }
279   
280    Edge addEdge(const Node& from, const Node& to) {
281      Edge edge = Parent::addEdge(from, to);
282      getNotifier(Edge()).add(edge);
283      return edge;
284    }
285
286    void clear() {
287      getNotifier(Edge()).clear();
288      getNotifier(Node()).clear();
289      Parent::clear();
290    }
291
292
293    void erase(const Node& node) {
294      Edge edge;
295      Parent::firstOut(edge, node);
296      while (edge != INVALID ) {
297        erase(edge);
298        Parent::firstOut(edge, node);
299      }
300
301      Parent::firstIn(edge, node);
302      while (edge != INVALID ) {
303        erase(edge);
304        Parent::firstIn(edge, node);
305      }
306
307      getNotifier(Node()).erase(node);
308      Parent::erase(node);
309    }
310   
311    void erase(const Edge& edge) {
312      getNotifier(Edge()).erase(edge);
313      Parent::erase(edge);
314    }
315
316
317    ~GraphExtender() {
318      getNotifier(Edge()).clear();
319      getNotifier(Node()).clear();
320    }
321  };
322
323  template <typename Base>
324  class UGraphBaseExtender : public Base {
325
326  public:
327
328    typedef Base Parent;
329    typedef typename Parent::Edge UEdge;
330    typedef typename Parent::Node Node;
331
332    typedef True UndirectedTag;
333
334    class Edge : public UEdge {
335      friend class UGraphBaseExtender;
336
337    protected:
338      bool forward;
339
340      Edge(const UEdge &ue, bool _forward) :
341        UEdge(ue), forward(_forward) {}
342
343    public:
344      Edge() {}
345
346      /// Invalid edge constructor
347      Edge(Invalid i) : UEdge(i), forward(true) {}
348
349      bool operator==(const Edge &that) const {
350        return forward==that.forward && UEdge(*this)==UEdge(that);
351      }
352      bool operator!=(const Edge &that) const {
353        return forward!=that.forward || UEdge(*this)!=UEdge(that);
354      }
355      bool operator<(const Edge &that) const {
356        return forward<that.forward ||
357          (!(that.forward<forward) && UEdge(*this)<UEdge(that));
358      }
359    };
360
361
362
363    using Parent::source;
364
365    /// Source of the given Edge.
366    Node source(const Edge &e) const {
367      return e.forward ? Parent::source(e) : Parent::target(e);
368    }
369
370    using Parent::target;
371
372    /// Target of the given Edge.
373    Node target(const Edge &e) const {
374      return e.forward ? Parent::target(e) : Parent::source(e);
375    }
376
377    /// \brief Directed edge from an undirected edge.
378    ///
379    /// Returns a directed edge corresponding to the specified UEdge.
380    /// If the given bool is true the given undirected edge and the
381    /// returned edge have the same source node.
382    Edge direct(const UEdge &ue, bool d) const {
383      return Edge(ue, d);
384    }
385
386    /// Returns whether the given directed edge is same orientation as the
387    /// corresponding undirected edge.
388    ///
389    /// \todo reference to the corresponding point of the undirected graph
390    /// concept. "What does the direction of an undirected edge mean?"
391    bool direction(const Edge &e) const { return e.forward; }
392
393
394    using Parent::first;
395    using Parent::next;
396
397    void first(Edge &e) const {
398      Parent::first(e);
399      e.forward=true;
400    }
401
402    void next(Edge &e) const {
403      if( e.forward ) {
404        e.forward = false;
405      }
406      else {
407        Parent::next(e);
408        e.forward = true;
409      }
410    }
411
412    void firstOut(Edge &e, const Node &n) const {
413      Parent::firstIn(e,n);
414      if( UEdge(e) != INVALID ) {
415        e.forward = false;
416      }
417      else {
418        Parent::firstOut(e,n);
419        e.forward = true;
420      }
421    }
422    void nextOut(Edge &e) const {
423      if( ! e.forward ) {
424        Node n = Parent::target(e);
425        Parent::nextIn(e);
426        if( UEdge(e) == INVALID ) {
427          Parent::firstOut(e, n);
428          e.forward = true;
429        }
430      }
431      else {
432        Parent::nextOut(e);
433      }
434    }
435
436    void firstIn(Edge &e, const Node &n) const {
437      Parent::firstOut(e,n);
438      if( UEdge(e) != INVALID ) {
439        e.forward = false;
440      }
441      else {
442        Parent::firstIn(e,n);
443        e.forward = true;
444      }
445    }
446    void nextIn(Edge &e) const {
447      if( ! e.forward ) {
448        Node n = Parent::source(e);
449        Parent::nextOut(e);
450        if( UEdge(e) == INVALID ) {
451          Parent::firstIn(e, n);
452          e.forward = true;
453        }
454      }
455      else {
456        Parent::nextIn(e);
457      }
458    }
459
460    void firstInc(UEdge &e, bool &d, const Node &n) const {
461      d = true;
462      Parent::firstOut(e, n);
463      if (e != INVALID) return;
464      d = false;
465      Parent::firstIn(e, n);
466    }
467
468    void nextInc(UEdge &e, bool &d) const {
469      if (d) {
470        Node s = Parent::source(e);
471        Parent::nextOut(e);
472        if (e != INVALID) return;
473        d = false;
474        Parent::firstIn(e, s);
475      } else {
476        Parent::nextIn(e);
477      }
478    }
479
480    Node nodeFromId(int id) const {
481      return Parent::nodeFromId(id);
482    }
483
484    Edge edgeFromId(int id) const {
485      return direct(Parent::edgeFromId(id >> 1), bool(id & 1));
486    }
487
488    UEdge uEdgeFromId(int id) const {
489      return Parent::edgeFromId(id >> 1);
490    }
491
492    int id(const Node &n) const {
493      return Parent::id(n);
494    }
495
496    int id(const UEdge &e) const {
497      return Parent::id(e);
498    }
499
500    int id(const Edge &e) const {
501      return 2 * Parent::id(e) + int(e.forward);
502    }
503
504    int maxNodeId() const {
505      return Parent::maxNodeId();
506    }
507
508    int maxEdgeId() const {
509      return 2 * Parent::maxEdgeId() + 1;
510    }
511
512    int maxUEdgeId() const {
513      return Parent::maxEdgeId();
514    }
515
516
517    int edgeNum() const {
518      return 2 * Parent::edgeNum();
519    }
520
521    int uEdgeNum() const {
522      return Parent::edgeNum();
523    }
524
525    Edge findEdge(Node source, Node target, Edge prev) const {
526      if (prev == INVALID) {
527        UEdge edge = Parent::findEdge(source, target);
528        if (edge != INVALID) return direct(edge, true);
529        edge = Parent::findEdge(target, source);
530        if (edge != INVALID) return direct(edge, false);
531      } else if (direction(prev)) {
532        UEdge edge = Parent::findEdge(source, target, prev);
533        if (edge != INVALID) return direct(edge, true);
534        edge = Parent::findEdge(target, source);
535        if (edge != INVALID) return direct(edge, false);       
536      } else {
537        UEdge edge = Parent::findEdge(target, source, prev);
538        if (edge != INVALID) return direct(edge, false);             
539      }
540      return INVALID;
541    }
542
543    UEdge findUEdge(Node source, Node target, UEdge prev) const {
544      if (prev == INVALID) {
545        UEdge edge = Parent::findEdge(source, target);
546        if (edge != INVALID) return edge;
547        edge = Parent::findEdge(target, source);
548        if (edge != INVALID) return edge;
549      } else if (Parent::source(prev) == source) {
550        UEdge edge = Parent::findEdge(source, target, prev);
551        if (edge != INVALID) return edge;
552        edge = Parent::findEdge(target, source);
553        if (edge != INVALID) return edge;       
554      } else {
555        UEdge edge = Parent::findEdge(target, source, prev);
556        if (edge != INVALID) return edge;             
557      }
558      return INVALID;
559    }
560  };
561
562
563  template <typename Base>
564  class UGraphExtender : public Base {
565  public:
566   
567    typedef Base Parent;
568    typedef UGraphExtender Graph;
569
570    typedef typename Parent::Node Node;
571    typedef typename Parent::Edge Edge;
572    typedef typename Parent::UEdge UEdge;
573
574    // UGraph extension   
575
576    int maxId(Node) const {
577      return Parent::maxNodeId();
578    }
579
580    int maxId(Edge) const {
581      return Parent::maxEdgeId();
582    }
583
584    int maxId(UEdge) const {
585      return Parent::maxUEdgeId();
586    }
587
588    Node fromId(int id, Node) const {
589      return Parent::nodeFromId(id);
590    }
591
592    Edge fromId(int id, Edge) const {
593      return Parent::edgeFromId(id);
594    }
595
596    UEdge fromId(int id, UEdge) const {
597      return Parent::uEdgeFromId(id);
598    }
599
600    Node oppositeNode(const Node &n, const UEdge &e) const {
601      if( n == Parent::source(e))
602        return Parent::target(e);
603      else if( n == Parent::target(e))
604        return Parent::source(e);
605      else
606        return INVALID;
607    }
608
609    Edge oppositeEdge(const Edge &e) const {
610      return Parent::direct(e, !Parent::direction(e));
611    }
612
613    using Parent::direct;
614    Edge direct(const UEdge &ue, const Node &s) const {
615      return Parent::direct(ue, Parent::source(ue) == s);
616    }
617
618    // Alterable extension
619
620    typedef AlterationNotifier<Node> NodeNotifier;
621    typedef AlterationNotifier<Edge> EdgeNotifier;
622    typedef AlterationNotifier<UEdge> UEdgeNotifier;
623
624
625  protected:
626
627    mutable NodeNotifier node_notifier;
628    mutable EdgeNotifier edge_notifier;
629    mutable UEdgeNotifier uedge_notifier;
630
631  public:
632
633    NodeNotifier& getNotifier(Node) const {
634      return node_notifier;
635    }
636   
637    EdgeNotifier& getNotifier(Edge) const {
638      return edge_notifier;
639    }
640
641    UEdgeNotifier& getNotifier(UEdge) const {
642      return uedge_notifier;
643    }
644
645
646
647    class NodeIt : public Node {
648      const Graph* graph;
649    public:
650
651      NodeIt() {}
652
653      NodeIt(Invalid i) : Node(i) { }
654
655      explicit NodeIt(const Graph& _graph) : graph(&_graph) {
656        _graph.first(*static_cast<Node*>(this));
657      }
658
659      NodeIt(const Graph& _graph, const Node& node)
660        : Node(node), graph(&_graph) {}
661
662      NodeIt& operator++() {
663        graph->next(*this);
664        return *this;
665      }
666
667    };
668
669
670    class EdgeIt : public Edge {
671      const Graph* graph;
672    public:
673
674      EdgeIt() { }
675
676      EdgeIt(Invalid i) : Edge(i) { }
677
678      explicit EdgeIt(const Graph& _graph) : graph(&_graph) {
679        _graph.first(*static_cast<Edge*>(this));
680      }
681
682      EdgeIt(const Graph& _graph, const Edge& e) :
683        Edge(e), graph(&_graph) { }
684
685      EdgeIt& operator++() {
686        graph->next(*this);
687        return *this;
688      }
689
690    };
691
692
693    class OutEdgeIt : public Edge {
694      const Graph* graph;
695    public:
696
697      OutEdgeIt() { }
698
699      OutEdgeIt(Invalid i) : Edge(i) { }
700
701      OutEdgeIt(const Graph& _graph, const Node& node)
702        : graph(&_graph) {
703        _graph.firstOut(*this, node);
704      }
705
706      OutEdgeIt(const Graph& _graph, const Edge& edge)
707        : Edge(edge), graph(&_graph) {}
708
709      OutEdgeIt& operator++() {
710        graph->nextOut(*this);
711        return *this;
712      }
713
714    };
715
716
717    class InEdgeIt : public Edge {
718      const Graph* graph;
719    public:
720
721      InEdgeIt() { }
722
723      InEdgeIt(Invalid i) : Edge(i) { }
724
725      InEdgeIt(const Graph& _graph, const Node& node)
726        : graph(&_graph) {
727        _graph.firstIn(*this, node);
728      }
729
730      InEdgeIt(const Graph& _graph, const Edge& edge) :
731        Edge(edge), graph(&_graph) {}
732
733      InEdgeIt& operator++() {
734        graph->nextIn(*this);
735        return *this;
736      }
737
738    };
739
740
741    class UEdgeIt : public Parent::UEdge {
742      const Graph* graph;
743    public:
744
745      UEdgeIt() { }
746
747      UEdgeIt(Invalid i) : UEdge(i) { }
748
749      explicit UEdgeIt(const Graph& _graph) : graph(&_graph) {
750        _graph.first(*static_cast<UEdge*>(this));
751      }
752
753      UEdgeIt(const Graph& _graph, const UEdge& e) :
754        UEdge(e), graph(&_graph) { }
755
756      UEdgeIt& operator++() {
757        graph->next(*this);
758        return *this;
759      }
760
761    };
762
763    class IncEdgeIt : public Parent::UEdge {
764      friend class UGraphExtender;
765      const Graph* graph;
766      bool direction;
767    public:
768
769      IncEdgeIt() { }
770
771      IncEdgeIt(Invalid i) : UEdge(i), direction(false) { }
772
773      IncEdgeIt(const Graph& _graph, const Node &n) : graph(&_graph) {
774        _graph.firstInc(*this, direction, n);
775      }
776
777      IncEdgeIt(const Graph& _graph, const UEdge &ue, const Node &n)
778        : graph(&_graph), UEdge(ue) {
779        direction = (_graph.source(ue) == n);
780      }
781
782      IncEdgeIt& operator++() {
783        graph->nextInc(*this, direction);
784        return *this;
785      }
786    };
787
788    /// \brief Base node of the iterator
789    ///
790    /// Returns the base node (ie. the source in this case) of the iterator
791    Node baseNode(const OutEdgeIt &e) const {
792      return Parent::source((Edge)e);
793    }
794    /// \brief Running node of the iterator
795    ///
796    /// Returns the running node (ie. the target in this case) of the
797    /// iterator
798    Node runningNode(const OutEdgeIt &e) const {
799      return Parent::target((Edge)e);
800    }
801
802    /// \brief Base node of the iterator
803    ///
804    /// Returns the base node (ie. the target in this case) of the iterator
805    Node baseNode(const InEdgeIt &e) const {
806      return Parent::target((Edge)e);
807    }
808    /// \brief Running node of the iterator
809    ///
810    /// Returns the running node (ie. the source in this case) of the
811    /// iterator
812    Node runningNode(const InEdgeIt &e) const {
813      return Parent::source((Edge)e);
814    }
815
816    /// Base node of the iterator
817    ///
818    /// Returns the base node of the iterator
819    Node baseNode(const IncEdgeIt &e) const {
820      return e.direction ? source(e) : target(e);
821    }
822    /// Running node of the iterator
823    ///
824    /// Returns the running node of the iterator
825    Node runningNode(const IncEdgeIt &e) const {
826      return e.direction ? target(e) : source(e);
827    }
828
829    // Mappable extension
830
831    template <typename _Value>
832    class NodeMap
833      : public IterableMapExtender<DefaultMap<Graph, Node, _Value> > {
834    public:
835      typedef UGraphExtender Graph;
836      typedef IterableMapExtender<DefaultMap<Graph, Node, _Value> > Parent;
837
838      NodeMap(const Graph& _g)
839        : Parent(_g) {}
840      NodeMap(const Graph& _g, const _Value& _v)
841        : Parent(_g, _v) {}
842
843      NodeMap& operator=(const NodeMap& cmap) {
844        return operator=<NodeMap>(cmap);
845      }
846
847
848      /// \brief Template assign operator.
849      ///
850      /// The given parameter should be conform to the ReadMap
851      /// concecpt and could be indiced by the current item set of
852      /// the NodeMap. In this case the value for each item
853      /// is assigned by the value of the given ReadMap.
854      template <typename CMap>
855      NodeMap& operator=(const CMap& cmap) {
856        checkConcept<concept::ReadMap<Node, _Value>, CMap>();
857        const typename Parent::Graph* graph = Parent::getGraph();
858        Node it;
859        for (graph->first(it); it != INVALID; graph->next(it)) {
860          Parent::set(it, cmap[it]);
861        }
862        return *this;
863      }
864
865    };
866
867    template <typename _Value>
868    class EdgeMap
869      : public IterableMapExtender<DefaultMap<Graph, Edge, _Value> > {
870    public:
871      typedef UGraphExtender Graph;
872      typedef IterableMapExtender<DefaultMap<Graph, Edge, _Value> > Parent;
873
874      EdgeMap(const Graph& _g)
875        : Parent(_g) {}
876      EdgeMap(const Graph& _g, const _Value& _v)
877        : Parent(_g, _v) {}
878
879      EdgeMap& operator=(const EdgeMap& cmap) {
880        return operator=<EdgeMap>(cmap);
881      }
882
883      template <typename CMap>
884      EdgeMap& operator=(const CMap& cmap) {
885        checkConcept<concept::ReadMap<Edge, _Value>, CMap>();
886        const typename Parent::Graph* graph = Parent::getGraph();
887        Edge it;
888        for (graph->first(it); it != INVALID; graph->next(it)) {
889          Parent::set(it, cmap[it]);
890        }
891        return *this;
892      }
893    };
894
895
896    template <typename _Value>
897    class UEdgeMap
898      : public IterableMapExtender<DefaultMap<Graph, UEdge, _Value> > {
899    public:
900      typedef UGraphExtender Graph;
901      typedef IterableMapExtender<DefaultMap<Graph, UEdge, _Value> > Parent;
902
903      UEdgeMap(const Graph& _g)
904        : Parent(_g) {}
905      UEdgeMap(const Graph& _g, const _Value& _v)
906        : Parent(_g, _v) {}
907
908      UEdgeMap& operator=(const UEdgeMap& cmap) {
909        return operator=<UEdgeMap>(cmap);
910      }
911
912      template <typename CMap>
913      UEdgeMap& operator=(const CMap& cmap) {
914        checkConcept<concept::ReadMap<UEdge, _Value>, CMap>();
915        const typename Parent::Graph* graph = Parent::getGraph();
916        UEdge it;
917        for (graph->first(it); it != INVALID; graph->next(it)) {
918          Parent::set(it, cmap[it]);
919        }
920        return *this;
921      }
922    };
923
924    // Alteration extension
925
926    Node addNode() {
927      Node node = Parent::addNode();
928      getNotifier(Node()).add(node);
929      return node;
930    }
931
932    UEdge addEdge(const Node& from, const Node& to) {
933      UEdge uedge = Parent::addEdge(from, to);
934      getNotifier(UEdge()).add(uedge);
935      getNotifier(Edge()).add(Parent::direct(uedge, true));
936      getNotifier(Edge()).add(Parent::direct(uedge, false));
937      return uedge;
938    }
939   
940    void clear() {
941      getNotifier(Edge()).clear();
942      getNotifier(UEdge()).clear();
943      getNotifier(Node()).clear();
944      Parent::clear();
945    }
946
947    void erase(const Node& node) {
948      Edge edge;
949      Parent::firstOut(edge, node);
950      while (edge != INVALID ) {
951        erase(edge);
952        Parent::firstOut(edge, node);
953      }
954
955      Parent::firstIn(edge, node);
956      while (edge != INVALID ) {
957        erase(edge);
958        Parent::firstIn(edge, node);
959      }
960
961      getNotifier(Node()).erase(node);
962      Parent::erase(node);
963    }
964
965    void erase(const UEdge& uedge) {
966      getNotifier(Edge()).erase(Parent::direct(uedge, true));
967      getNotifier(Edge()).erase(Parent::direct(uedge, false));
968      getNotifier(UEdge()).erase(uedge);
969      Parent::erase(uedge);
970    }
971
972    ~UGraphExtender() {
973      getNotifier(Edge()).clear();
974      getNotifier(UEdge()).clear();
975      getNotifier(Node()).clear();
976    }
977
978  };
979
980
981  template <typename Base>
982  class BpUGraphBaseExtender : public Base {
983  public:
984    typedef Base Parent;
985    typedef BpUGraphBaseExtender Graph;
986
987    typedef typename Parent::Node Node;
988    typedef typename Parent::Edge UEdge;
989
990
991    using Parent::first;
992    using Parent::next;
993
994    using Parent::id;
995
996    class ANode : public Node {
997      friend class BpUGraphBaseExtender;
998    public:
999      ANode() {}
1000      ANode(const Node& node) : Node(node) {
1001        LEMON_ASSERT(Parent::aNode(node) || node == INVALID,
1002                     typename Parent::NodeSetError());
1003      }
1004      ANode(Invalid) : Node(INVALID) {}
1005    };
1006
1007    void first(ANode& node) const {
1008      Parent::firstANode(static_cast<Node&>(node));
1009    }
1010    void next(ANode& node) const {
1011      Parent::nextANode(static_cast<Node&>(node));
1012    }
1013
1014    int id(const ANode& node) const {
1015      return Parent::aNodeId(node);
1016    }
1017
1018    class BNode : public Node {
1019      friend class BpUGraphBaseExtender;
1020    public:
1021      BNode() {}
1022      BNode(const Node& node) : Node(node) {
1023        LEMON_ASSERT(Parent::bNode(node) || node == INVALID,
1024                     typename Parent::NodeSetError());
1025      }
1026      BNode(Invalid) : Node(INVALID) {}
1027    };
1028
1029    void first(BNode& node) const {
1030      Parent::firstBNode(static_cast<Node&>(node));
1031    }
1032    void next(BNode& node) const {
1033      Parent::nextBNode(static_cast<Node&>(node));
1034    }
1035 
1036    int id(const BNode& node) const {
1037      return Parent::aNodeId(node);
1038    }
1039
1040    Node source(const UEdge& edge) const {
1041      return aNode(edge);
1042    }
1043    Node target(const UEdge& edge) const {
1044      return bNode(edge);
1045    }
1046
1047    void firstInc(UEdge& edge, bool& direction, const Node& node) const {
1048      if (Parent::aNode(node)) {
1049        Parent::firstOut(edge, node);
1050        direction = true;
1051      } else {
1052        Parent::firstIn(edge, node);
1053        direction = static_cast<UEdge&>(edge) == INVALID;
1054      }
1055    }
1056    void nextInc(UEdge& edge, bool& direction) const {
1057      if (direction) {
1058        Parent::nextOut(edge);
1059      } else {
1060        Parent::nextIn(edge);
1061        if (edge == INVALID) direction = true;
1062      }
1063    }
1064
1065    int maxUEdgeId() const {
1066      return Parent::maxEdgeId();
1067    }
1068
1069    UEdge uEdgeFromId(int id) const {
1070      return Parent::edgeFromId(id);
1071    }
1072
1073    class Edge : public UEdge {
1074      friend class BpUGraphBaseExtender;
1075    protected:
1076      bool forward;
1077
1078      Edge(const UEdge& edge, bool _forward)
1079        : UEdge(edge), forward(_forward) {}
1080
1081    public:
1082      Edge() {}
1083      Edge (Invalid) : UEdge(INVALID), forward(true) {}
1084      bool operator==(const Edge& i) const {
1085        return UEdge::operator==(i) && forward == i.forward;
1086      }
1087      bool operator!=(const Edge& i) const {
1088        return UEdge::operator!=(i) || forward != i.forward;
1089      }
1090      bool operator<(const Edge& i) const {
1091        return UEdge::operator<(i) ||
1092          (!(i.forward<forward) && UEdge(*this)<UEdge(i));
1093      }
1094    };
1095
1096    void first(Edge& edge) const {
1097      Parent::first(static_cast<UEdge&>(edge));
1098      edge.forward = true;
1099    }
1100
1101    void next(Edge& edge) const {
1102      if (!edge.forward) {
1103        Parent::next(static_cast<UEdge&>(edge));
1104      }
1105      edge.forward = !edge.forward;
1106    }
1107
1108    void firstOut(Edge& edge, const Node& node) const {
1109      if (Parent::aNode(node)) {
1110        Parent::firstOut(edge, node);
1111        edge.forward = true;
1112      } else {
1113        Parent::firstIn(edge, node);
1114        edge.forward = static_cast<UEdge&>(edge) == INVALID;
1115      }
1116    }
1117    void nextOut(Edge& edge) const {
1118      if (edge.forward) {
1119        Parent::nextOut(edge);
1120      } else {
1121        Parent::nextIn(edge);
1122        edge.forward = static_cast<UEdge&>(edge) == INVALID;
1123      }
1124    }
1125
1126    void firstIn(Edge& edge, const Node& node) const {
1127      if (Parent::bNode(node)) {
1128        Parent::firstIn(edge, node);
1129        edge.forward = true;   
1130      } else {
1131        Parent::firstOut(edge, node);
1132        edge.forward = static_cast<UEdge&>(edge) == INVALID;
1133      }
1134    }
1135    void nextIn(Edge& edge) const {
1136      if (edge.forward) {
1137        Parent::nextIn(edge);
1138      } else {
1139        Parent::nextOut(edge);
1140        edge.forward = static_cast<UEdge&>(edge) == INVALID;
1141      }
1142    }
1143
1144    Node source(const Edge& edge) const {
1145      return edge.forward ? Parent::aNode(edge) : Parent::bNode(edge);
1146    }
1147    Node target(const Edge& edge) const {
1148      return edge.forward ? Parent::bNode(edge) : Parent::aNode(edge);
1149    }
1150
1151    int id(const Edge& edge) const {
1152      return (Parent::id(edge) << 1) + (edge.forward ? 0 : 1);
1153    }
1154    Edge edgeFromId(int id) const {
1155      return Edge(Parent::fromId(id >> 1, UEdge()), (id & 1) == 0);
1156    }
1157    int maxEdgeId() const {
1158      return (Parent::maxId(UEdge()) << 1) + 1;
1159    }
1160
1161    bool direction(const Edge& edge) const {
1162      return edge.forward;
1163    }
1164
1165    Edge direct(const UEdge& edge, bool direction) const {
1166      return Edge(edge, direction);
1167    }
1168  };
1169
1170  template <typename Base>
1171  class BpUGraphExtender : public Base {
1172  public:
1173    typedef Base Parent;
1174    typedef BpUGraphExtender Graph;
1175
1176    typedef typename Parent::Node Node;
1177    typedef typename Parent::BNode BNode;
1178    typedef typename Parent::ANode ANode;
1179    typedef typename Parent::Edge Edge;
1180    typedef typename Parent::UEdge UEdge;
1181
1182    Node oppositeNode(const UEdge& edge, const Node& node) const {
1183      return source(edge) == node ?
1184        target(edge) : source(edge);
1185    }
1186
1187    using Parent::direct;
1188    Edge direct(const UEdge& edge, const Node& node) const {
1189      return Edge(edge, node == Parent::source(edge));
1190    }
1191
1192    Edge oppositeEdge(const Edge& edge) const {
1193      return Parent::direct(edge, !Parent::direction(edge));
1194    }
1195
1196
1197    int maxId(Node) const {
1198      return Parent::maxNodeId();
1199    }
1200    int maxId(BNode) const {
1201      return Parent::maxBNodeId();
1202    }
1203    int maxId(ANode) const {
1204      return Parent::maxANodeId();
1205    }
1206    int maxId(Edge) const {
1207      return Parent::maxEdgeId();
1208    }
1209    int maxId(UEdge) const {
1210      return Parent::maxUEdgeId();
1211    }
1212
1213
1214    Node fromId(int id, Node) const {
1215      return Parent::nodeFromId(id);
1216    }
1217    ANode fromId(int id, ANode) const {
1218      return Parent::fromANodeId(id);
1219    }
1220    BNode fromId(int id, BNode) const {
1221      return Parent::fromBNodeId(id);
1222    }
1223    Edge fromId(int id, Edge) const {
1224      return Parent::edgeFromId(id);
1225    }
1226    UEdge fromId(int id, UEdge) const {
1227      return Parent::uEdgeFromId(id);
1228    } 
1229 
1230    typedef AlterationNotifier<Node> NodeNotifier;
1231    typedef AlterationNotifier<BNode> BNodeNotifier;
1232    typedef AlterationNotifier<ANode> ANodeNotifier;
1233    typedef AlterationNotifier<Edge> EdgeNotifier;
1234    typedef AlterationNotifier<UEdge> UEdgeNotifier;
1235
1236  protected:
1237
1238    mutable NodeNotifier nodeNotifier;
1239    mutable BNodeNotifier bNodeNotifier;
1240    mutable ANodeNotifier aNodeNotifier;
1241    mutable EdgeNotifier edgeNotifier;
1242    mutable UEdgeNotifier uEdgeNotifier;
1243
1244  public:
1245
1246    NodeNotifier& getNotifier(Node) const {
1247      return nodeNotifier;
1248    }
1249
1250    BNodeNotifier& getNotifier(BNode) const {
1251      return bNodeNotifier;
1252    }
1253
1254    ANodeNotifier& getNotifier(ANode) const {
1255      return aNodeNotifier;
1256    }
1257
1258    EdgeNotifier& getNotifier(Edge) const {
1259      return edgeNotifier;
1260    }
1261
1262    UEdgeNotifier& getNotifier(UEdge) const {
1263      return uEdgeNotifier;
1264    }
1265
1266    ~BpUGraphExtender() {
1267      getNotifier(UEdge()).clear();
1268      getNotifier(Edge()).clear();
1269      getNotifier(ANode()).clear();
1270      getNotifier(BNode()).clear();
1271      getNotifier(Node()).clear();
1272    }
1273
1274 
1275    class NodeIt : public Node {
1276      const Graph* graph;
1277    public:
1278   
1279      NodeIt() { }
1280   
1281      NodeIt(Invalid i) : Node(INVALID) { }
1282   
1283      explicit NodeIt(const Graph& _graph) : graph(&_graph) {
1284        graph->first(static_cast<Node&>(*this));
1285      }
1286
1287      NodeIt(const Graph& _graph, const Node& node)
1288        : Node(node), graph(&_graph) { }
1289   
1290      NodeIt& operator++() {
1291        graph->next(*this);
1292        return *this;
1293      }
1294
1295    };
1296
1297    class ANodeIt : public Node {
1298      friend class BpUGraphExtender;
1299      const Graph* graph;
1300    public:
1301   
1302      ANodeIt() { }
1303   
1304      ANodeIt(Invalid i) : Node(INVALID) { }
1305   
1306      explicit ANodeIt(const Graph& _graph) : graph(&_graph) {
1307        graph->firstANode(static_cast<Node&>(*this));
1308      }
1309
1310      ANodeIt(const Graph& _graph, const Node& node)
1311        : Node(node), graph(&_graph) {}
1312   
1313      ANodeIt& operator++() {
1314        graph->nextANode(*this);
1315        return *this;
1316      }
1317    };
1318
1319    class BNodeIt : public Node {
1320      friend class BpUGraphExtender;
1321      const Graph* graph;
1322    public:
1323   
1324      BNodeIt() { }
1325   
1326      BNodeIt(Invalid i) : Node(INVALID) { }
1327   
1328      explicit BNodeIt(const Graph& _graph) : graph(&_graph) {
1329        graph->firstBNode(static_cast<Node&>(*this));
1330      }
1331
1332      BNodeIt(const Graph& _graph, const Node& node)
1333        : Node(node), graph(&_graph) {}
1334   
1335      BNodeIt& operator++() {
1336        graph->nextBNode(*this);
1337        return *this;
1338      }
1339    };
1340
1341    class EdgeIt : public Edge {
1342      friend class BpUGraphExtender;
1343      const Graph* graph;
1344    public:
1345   
1346      EdgeIt() { }
1347   
1348      EdgeIt(Invalid i) : Edge(INVALID) { }
1349   
1350      explicit EdgeIt(const Graph& _graph) : graph(&_graph) {
1351        graph->first(static_cast<Edge&>(*this));
1352      }
1353
1354      EdgeIt(const Graph& _graph, const Edge& edge)
1355        : Edge(edge), graph(&_graph) { }
1356   
1357      EdgeIt& operator++() {
1358        graph->next(*this);
1359        return *this;
1360      }
1361
1362    };
1363
1364    class UEdgeIt : public UEdge {
1365      friend class BpUGraphExtender;
1366      const Graph* graph;
1367    public:
1368   
1369      UEdgeIt() { }
1370   
1371      UEdgeIt(Invalid i) : UEdge(INVALID) { }
1372   
1373      explicit UEdgeIt(const Graph& _graph) : graph(&_graph) {
1374        graph->first(static_cast<UEdge&>(*this));
1375      }
1376
1377      UEdgeIt(const Graph& _graph, const UEdge& edge)
1378        : UEdge(edge), graph(&_graph) { }
1379   
1380      UEdgeIt& operator++() {
1381        graph->next(*this);
1382        return *this;
1383      }
1384    };
1385
1386    class OutEdgeIt : public Edge {
1387      friend class BpUGraphExtender;
1388      const Graph* graph;
1389    public:
1390   
1391      OutEdgeIt() { }
1392   
1393      OutEdgeIt(Invalid i) : Edge(i) { }
1394   
1395      OutEdgeIt(const Graph& _graph, const Node& node)
1396        : graph(&_graph) {
1397        graph->firstOut(*this, node);
1398      }
1399   
1400      OutEdgeIt(const Graph& _graph, const Edge& edge)
1401        : Edge(edge), graph(&_graph) {}
1402   
1403      OutEdgeIt& operator++() {
1404        graph->nextOut(*this);
1405        return *this;
1406      }
1407
1408    };
1409
1410
1411    class InEdgeIt : public Edge {
1412      friend class BpUGraphExtender;
1413      const Graph* graph;
1414    public:
1415   
1416      InEdgeIt() { }
1417   
1418      InEdgeIt(Invalid i) : Edge(i) { }
1419   
1420      InEdgeIt(const Graph& _graph, const Node& node)
1421        : graph(&_graph) {
1422        graph->firstIn(*this, node);
1423      }
1424   
1425      InEdgeIt(const Graph& _graph, const Edge& edge) :
1426        Edge(edge), graph(&_graph) {}
1427   
1428      InEdgeIt& operator++() {
1429        graph->nextIn(*this);
1430        return *this;
1431      }
1432
1433    };
1434 
1435    /// \brief Base node of the iterator
1436    ///
1437    /// Returns the base node (ie. the source in this case) of the iterator
1438    Node baseNode(const OutEdgeIt &e) const {
1439      return Parent::source((Edge&)e);
1440    }
1441    /// \brief Running node of the iterator
1442    ///
1443    /// Returns the running node (ie. the target in this case) of the
1444    /// iterator
1445    Node runningNode(const OutEdgeIt &e) const {
1446      return Parent::target((Edge&)e);
1447    }
1448 
1449    /// \brief Base node of the iterator
1450    ///
1451    /// Returns the base node (ie. the target in this case) of the iterator
1452    Node baseNode(const InEdgeIt &e) const {
1453      return Parent::target((Edge&)e);
1454    }
1455    /// \brief Running node of the iterator
1456    ///
1457    /// Returns the running node (ie. the source in this case) of the
1458    /// iterator
1459    Node runningNode(const InEdgeIt &e) const {
1460      return Parent::source((Edge&)e);
1461    }
1462 
1463    class IncEdgeIt : public Parent::UEdge {
1464      friend class BpUGraphExtender;
1465      const Graph* graph;
1466      bool direction;
1467    public:
1468   
1469      IncEdgeIt() { }
1470   
1471      IncEdgeIt(Invalid i) : UEdge(i), direction(true) { }
1472   
1473      IncEdgeIt(const Graph& _graph, const Node &n) : graph(&_graph) {
1474        graph->firstInc(*this, direction, n);
1475      }
1476
1477      IncEdgeIt(const Graph& _graph, const UEdge &ue, const Node &n)
1478        : graph(&_graph), UEdge(ue) {
1479        direction = (graph->source(ue) == n);
1480      }
1481
1482      IncEdgeIt& operator++() {
1483        graph->nextInc(*this, direction);
1484        return *this;
1485      }
1486    };
1487 
1488
1489    /// Base node of the iterator
1490    ///
1491    /// Returns the base node of the iterator
1492    Node baseNode(const IncEdgeIt &e) const {
1493      return e.direction ? source(e) : target(e);
1494    }
1495
1496    /// Running node of the iterator
1497    ///
1498    /// Returns the running node of the iterator
1499    Node runningNode(const IncEdgeIt &e) const {
1500      return e.direction ? target(e) : source(e);
1501    }
1502
1503    template <typename _Value>
1504    class ANodeMap
1505      : public IterableMapExtender<DefaultMap<Graph, ANode, _Value> > {
1506    public:
1507      typedef BpUGraphExtender Graph;
1508      typedef IterableMapExtender<DefaultMap<Graph, ANode, _Value> >
1509      Parent;
1510   
1511      ANodeMap(const Graph& _g)
1512        : Parent(_g) {}
1513      ANodeMap(const Graph& _g, const _Value& _v)
1514        : Parent(_g, _v) {}
1515   
1516      ANodeMap& operator=(const ANodeMap& cmap) {
1517        return operator=<ANodeMap>(cmap);
1518      }
1519   
1520
1521      /// \brief Template assign operator.
1522      ///
1523      /// The given parameter should be conform to the ReadMap
1524      /// concept and could be indiced by the current item set of
1525      /// the ANodeMap. In this case the value for each item
1526      /// is assigned by the value of the given ReadMap.
1527      template <typename CMap>
1528      ANodeMap& operator=(const CMap& cmap) {
1529        checkConcept<concept::ReadMap<ANode, _Value>, CMap>();
1530        const typename Parent::Graph* graph = Parent::getGraph();
1531        ANode it;
1532        for (graph->first(it); it != INVALID; graph->next(it)) {
1533          Parent::set(it, cmap[it]);
1534        }
1535        return *this;
1536      }
1537   
1538    };
1539
1540    template <typename _Value>
1541    class BNodeMap
1542      : public IterableMapExtender<DefaultMap<Graph, BNode, _Value> > {
1543    public:
1544      typedef BpUGraphExtender Graph;
1545      typedef IterableMapExtender<DefaultMap<Graph, BNode, _Value> >
1546      Parent;
1547   
1548      BNodeMap(const Graph& _g)
1549        : Parent(_g) {}
1550      BNodeMap(const Graph& _g, const _Value& _v)
1551        : Parent(_g, _v) {}
1552   
1553      BNodeMap& operator=(const BNodeMap& cmap) {
1554        return operator=<BNodeMap>(cmap);
1555      }
1556   
1557
1558      /// \brief Template assign operator.
1559      ///
1560      /// The given parameter should be conform to the ReadMap
1561      /// concept and could be indiced by the current item set of
1562      /// the BNodeMap. In this case the value for each item
1563      /// is assigned by the value of the given ReadMap.
1564      template <typename CMap>
1565      BNodeMap& operator=(const CMap& cmap) {
1566        checkConcept<concept::ReadMap<BNode, _Value>, CMap>();
1567        const typename Parent::Graph* graph = Parent::getGraph();
1568        BNode it;
1569        for (graph->first(it); it != INVALID; graph->next(it)) {
1570          Parent::set(it, cmap[it]);
1571        }
1572        return *this;
1573      }
1574   
1575    };
1576
1577  protected:
1578
1579    template <typename _Value>
1580    class NodeMapBase : public NodeNotifier::ObserverBase {
1581    public:
1582      typedef BpUGraphExtender Graph;
1583
1584      typedef Node Key;
1585      typedef _Value Value;
1586
1587      /// The reference type of the map;
1588      typedef typename BNodeMap<_Value>::Reference Reference;
1589      /// The pointer type of the map;
1590      typedef typename BNodeMap<_Value>::Pointer Pointer;
1591     
1592      /// The const value type of the map.
1593      typedef const Value ConstValue;
1594      /// The const reference type of the map;
1595      typedef typename BNodeMap<_Value>::ConstReference ConstReference;
1596      /// The pointer type of the map;
1597      typedef typename BNodeMap<_Value>::ConstPointer ConstPointer;
1598
1599      typedef True ReferenceMapTag;
1600
1601      NodeMapBase(const Graph& _g)
1602        : graph(&_g), bNodeMap(_g), aNodeMap(_g) {
1603        NodeNotifier::ObserverBase::attach(_g.getNotifier(Node()));
1604      }
1605      NodeMapBase(const Graph& _g, const _Value& _v)
1606        : graph(&_g), bNodeMap(_g, _v),
1607          aNodeMap(_g, _v) {
1608        NodeNotifier::ObserverBase::attach(_g.getNotifier(Node()));
1609      }
1610
1611      virtual ~NodeMapBase() {     
1612        if (NodeNotifier::ObserverBase::attached()) {
1613          NodeNotifier::ObserverBase::detach();
1614        }
1615      }
1616   
1617      ConstReference operator[](const Key& node) const {
1618        if (Parent::aNode(node)) {
1619          return aNodeMap[node];
1620        } else {
1621          return bNodeMap[node];
1622        }
1623      }
1624
1625      Reference operator[](const Key& node) {
1626        if (Parent::aNode(node)) {
1627          return aNodeMap[node];
1628        } else {
1629          return bNodeMap[node];
1630        }
1631      }
1632
1633      void set(const Key& node, const Value& value) {
1634        if (Parent::aNode(node)) {
1635          aNodeMap.set(node, value);
1636        } else {
1637          bNodeMap.set(node, value);
1638        }
1639      }
1640
1641    protected:
1642     
1643      virtual void add(const Node&) {}
1644      virtual void add(const std::vector<Node>&) {}
1645      virtual void erase(const Node&) {}
1646      virtual void erase(const std::vector<Node>&) {}
1647      virtual void clear() {}
1648      virtual void build() {}
1649
1650      const Graph* getGraph() const { return graph; }
1651     
1652    private:
1653      const Graph* graph;
1654      BNodeMap<_Value> bNodeMap;
1655      ANodeMap<_Value> aNodeMap;
1656    };
1657   
1658  public:
1659
1660    template <typename _Value>
1661    class NodeMap
1662      : public IterableMapExtender<NodeMapBase<_Value> > {
1663    public:
1664      typedef BpUGraphExtender Graph;
1665      typedef IterableMapExtender< NodeMapBase<_Value> > Parent;
1666   
1667      NodeMap(const Graph& _g)
1668        : Parent(_g) {}
1669      NodeMap(const Graph& _g, const _Value& _v)
1670        : Parent(_g, _v) {}
1671   
1672      NodeMap& operator=(const NodeMap& cmap) {
1673        return operator=<NodeMap>(cmap);
1674      }
1675   
1676
1677      /// \brief Template assign operator.
1678      ///
1679      /// The given parameter should be conform to the ReadMap
1680      /// concept and could be indiced by the current item set of
1681      /// the NodeMap. In this case the value for each item
1682      /// is assigned by the value of the given ReadMap.
1683      template <typename CMap>
1684      NodeMap& operator=(const CMap& cmap) {
1685        checkConcept<concept::ReadMap<Node, _Value>, CMap>();
1686        const typename Parent::Graph* graph = Parent::getGraph();
1687        Node it;
1688        for (graph->first(it); it != INVALID; graph->next(it)) {
1689          Parent::set(it, cmap[it]);
1690        }
1691        return *this;
1692      }
1693   
1694    };
1695
1696
1697
1698    template <typename _Value>
1699    class EdgeMap
1700      : public IterableMapExtender<DefaultMap<Graph, Edge, _Value> > {
1701    public:
1702      typedef BpUGraphExtender Graph;
1703      typedef IterableMapExtender<DefaultMap<Graph, Edge, _Value> > Parent;
1704   
1705      EdgeMap(const Graph& _g)
1706        : Parent(_g) {}
1707      EdgeMap(const Graph& _g, const _Value& _v)
1708        : Parent(_g, _v) {}
1709   
1710      EdgeMap& operator=(const EdgeMap& cmap) {
1711        return operator=<EdgeMap>(cmap);
1712      }
1713   
1714      template <typename CMap>
1715      EdgeMap& operator=(const CMap& cmap) {
1716        checkConcept<concept::ReadMap<Edge, _Value>, CMap>();
1717        const typename Parent::Graph* graph = Parent::getGraph();
1718        Edge it;
1719        for (graph->first(it); it != INVALID; graph->next(it)) {
1720          Parent::set(it, cmap[it]);
1721        }
1722        return *this;
1723      }
1724    };
1725
1726    template <typename _Value>
1727    class UEdgeMap
1728      : public IterableMapExtender<DefaultMap<Graph, UEdge, _Value> > {
1729    public:
1730      typedef BpUGraphExtender Graph;
1731      typedef IterableMapExtender<DefaultMap<Graph, UEdge, _Value> >
1732      Parent;
1733   
1734      UEdgeMap(const Graph& _g)
1735        : Parent(_g) {}
1736      UEdgeMap(const Graph& _g, const _Value& _v)
1737        : Parent(_g, _v) {}
1738   
1739      UEdgeMap& operator=(const UEdgeMap& cmap) {
1740        return operator=<UEdgeMap>(cmap);
1741      }
1742   
1743      template <typename CMap>
1744      UEdgeMap& operator=(const CMap& cmap) {
1745        checkConcept<concept::ReadMap<UEdge, _Value>, CMap>();
1746        const typename Parent::Graph* graph = Parent::getGraph();
1747        UEdge it;
1748        for (graph->first(it); it != INVALID; graph->next(it)) {
1749          Parent::set(it, cmap[it]);
1750        }
1751        return *this;
1752      }
1753    };
1754
1755 
1756    Node addANode() {
1757      Node node = Parent::addANode();
1758      getNotifier(ANode()).add(node);
1759      getNotifier(Node()).add(node);
1760      return node;
1761    }
1762
1763    Node addBNode() {
1764      Node node = Parent::addBNode();
1765      getNotifier(BNode()).add(node);
1766      getNotifier(Node()).add(node);
1767      return node;
1768    }
1769 
1770    UEdge addEdge(const Node& source, const Node& target) {
1771      UEdge uedge = Parent::addEdge(source, target);
1772      getNotifier(UEdge()).add(uedge);
1773   
1774      std::vector<Edge> edges;
1775      edges.push_back(direct(uedge, true));
1776      edges.push_back(direct(uedge, false));
1777      getNotifier(Edge()).add(edges);
1778   
1779      return uedge;
1780    }
1781
1782    void clear() {
1783      getNotifier(Edge()).clear();
1784      getNotifier(UEdge()).clear();
1785      getNotifier(Node()).clear();
1786      getNotifier(BNode()).clear();
1787      getNotifier(ANode()).clear();
1788      Parent::clear();
1789    }
1790
1791    void erase(const Node& node) {
1792      UEdge uedge;
1793      bool dir;
1794      Parent::firstInc(uedge, dir, node);
1795      while (uedge != INVALID ) {
1796        erase(uedge);
1797        Parent::firstInc(uedge, dir, node);
1798      }
1799
1800      getNotifier(Node()).erase(node);
1801      Parent::erase(node);
1802    }
1803   
1804    void erase(const UEdge& uedge) {
1805      std::vector<Edge> edges;
1806      edges.push_back(direct(uedge, true));
1807      edges.push_back(direct(uedge, false));
1808      getNotifier(Edge()).erase(edges);
1809      getNotifier(UEdge()).erase(uedge);
1810      Parent::erase(uedge);
1811    }
1812
1813
1814    ~BpUGraphExtender() {
1815      getNotifier(Edge()).clear();
1816      getNotifier(UEdge()).clear();
1817      getNotifier(Node()).clear();
1818      getNotifier(BNode()).clear();
1819      getNotifier(ANode()).clear();
1820    }
1821
1822
1823  };
1824
1825}
1826
1827#endif // LEMON_UNDIR_GRAPH_EXTENDER_H
Note: See TracBrowser for help on using the repository browser.