lemon/bits/edge_set_extender.h
author deba
Wed, 01 Mar 2006 13:19:28 +0000
changeset 1993 2115143eceea
parent 1979 c2992fd74dad
child 1996 5dc13b93f8b4
permissions -rw-r--r--
utility, invalid and traits moved to bits
deba@1979
     1
/* -*- C++ -*-
deba@1979
     2
 *
deba@1979
     3
 * This file is a part of LEMON, a generic C++ optimization library
deba@1979
     4
 *
deba@1979
     5
 * Copyright (C) 2003-2006
deba@1979
     6
 * Egervary Jeno Kombinatorikus Optimalizalasi Kutatocsoport
deba@1979
     7
 * (Egervary Research Group on Combinatorial Optimization, EGRES).
deba@1979
     8
 *
deba@1979
     9
 * Permission to use, modify and distribute this software is granted
deba@1979
    10
 * provided that this copyright notice appears in all copies. For
deba@1979
    11
 * precise terms see the accompanying LICENSE file.
deba@1979
    12
 *
deba@1979
    13
 * This software is provided "AS IS" with no warranty of any kind,
deba@1979
    14
 * express or implied, and with no claim as to its suitability for any
deba@1979
    15
 * purpose.
deba@1979
    16
 *
deba@1979
    17
 */
deba@1979
    18
deba@1979
    19
deba@1979
    20
namespace lemon {
deba@1979
    21
deba@1979
    22
  template <typename Base>
deba@1979
    23
  class EdgeSetExtender : public Base {
deba@1979
    24
  public:
deba@1979
    25
deba@1979
    26
    typedef Base Parent;
deba@1979
    27
    typedef EdgeSetExtender Graph;
deba@1979
    28
deba@1979
    29
    // Base extensions
deba@1979
    30
deba@1979
    31
    typedef typename Parent::Node Node;
deba@1979
    32
    typedef typename Parent::Edge Edge;
deba@1979
    33
deba@1979
    34
    int maxId(Node) const {
deba@1979
    35
      return Parent::maxNodeId();
deba@1979
    36
    }
deba@1979
    37
deba@1979
    38
    int maxId(Edge) const {
deba@1979
    39
      return Parent::maxEdgeId();
deba@1979
    40
    }
deba@1979
    41
deba@1979
    42
    Node fromId(int id, Node) const {
deba@1979
    43
      return Parent::nodeFromId(id);
deba@1979
    44
    }
deba@1979
    45
deba@1979
    46
    Edge fromId(int id, Edge) const {
deba@1979
    47
      return Parent::edgeFromId(id);
deba@1979
    48
    }
deba@1979
    49
deba@1979
    50
    Node oppositeNode(const Node &n, const Edge &e) const {
deba@1979
    51
      if (n == Parent::source(e))
deba@1979
    52
	return Parent::target(e);
deba@1979
    53
      else if(n==Parent::target(e))
deba@1979
    54
	return Parent::source(e);
deba@1979
    55
      else
deba@1979
    56
	return INVALID;
deba@1979
    57
    }
deba@1979
    58
deba@1979
    59
deba@1979
    60
    // Alteration notifier extensions
deba@1979
    61
deba@1979
    62
    /// The edge observer registry.
deba@1979
    63
    typedef AlterationNotifier<Edge> EdgeNotifier;
deba@1979
    64
deba@1979
    65
  protected:
deba@1979
    66
deba@1979
    67
    mutable EdgeNotifier edge_notifier;
deba@1979
    68
deba@1979
    69
  public:
deba@1979
    70
deba@1991
    71
    using Parent::getNotifier;
deba@1991
    72
deba@1979
    73
    /// \brief Gives back the edge alteration notifier.
deba@1979
    74
    ///
deba@1979
    75
    /// Gives back the edge alteration notifier.
deba@1979
    76
    EdgeNotifier& getNotifier(Edge) const {
deba@1979
    77
      return edge_notifier;
deba@1979
    78
    }
deba@1979
    79
deba@1979
    80
    // Iterable extensions
deba@1979
    81
deba@1979
    82
    class NodeIt : public Node { 
deba@1979
    83
      const Graph* graph;
deba@1979
    84
    public:
deba@1979
    85
deba@1979
    86
      NodeIt() {}
deba@1979
    87
deba@1979
    88
      NodeIt(Invalid i) : Node(i) { }
deba@1979
    89
deba@1979
    90
      explicit NodeIt(const Graph& _graph) : graph(&_graph) {
deba@1979
    91
	_graph.first(*static_cast<Node*>(this));
deba@1979
    92
      }
deba@1979
    93
deba@1979
    94
      NodeIt(const Graph& _graph, const Node& node) 
deba@1979
    95
	: Node(node), graph(&_graph) {}
deba@1979
    96
deba@1979
    97
      NodeIt& operator++() { 
deba@1979
    98
	graph->next(*this);
deba@1979
    99
	return *this; 
deba@1979
   100
      }
deba@1979
   101
deba@1979
   102
    };
deba@1979
   103
deba@1979
   104
deba@1979
   105
    class EdgeIt : public Edge { 
deba@1979
   106
      const Graph* graph;
deba@1979
   107
    public:
deba@1979
   108
deba@1979
   109
      EdgeIt() { }
deba@1979
   110
deba@1979
   111
      EdgeIt(Invalid i) : Edge(i) { }
deba@1979
   112
deba@1979
   113
      explicit EdgeIt(const Graph& _graph) : graph(&_graph) {
deba@1979
   114
	_graph.first(*static_cast<Edge*>(this));
deba@1979
   115
      }
deba@1979
   116
deba@1979
   117
      EdgeIt(const Graph& _graph, const Edge& e) : 
deba@1979
   118
	Edge(e), graph(&_graph) { }
deba@1979
   119
deba@1979
   120
      EdgeIt& operator++() { 
deba@1979
   121
	graph->next(*this);
deba@1979
   122
	return *this; 
deba@1979
   123
      }
deba@1979
   124
deba@1979
   125
    };
deba@1979
   126
deba@1979
   127
deba@1979
   128
    class OutEdgeIt : public Edge { 
deba@1979
   129
      const Graph* graph;
deba@1979
   130
    public:
deba@1979
   131
deba@1979
   132
      OutEdgeIt() { }
deba@1979
   133
deba@1979
   134
      OutEdgeIt(Invalid i) : Edge(i) { }
deba@1979
   135
deba@1979
   136
      OutEdgeIt(const Graph& _graph, const Node& node) 
deba@1979
   137
	: graph(&_graph) {
deba@1979
   138
	_graph.firstOut(*this, node);
deba@1979
   139
      }
deba@1979
   140
deba@1979
   141
      OutEdgeIt(const Graph& _graph, const Edge& edge) 
deba@1979
   142
	: Edge(edge), graph(&_graph) {}
deba@1979
   143
deba@1979
   144
      OutEdgeIt& operator++() { 
deba@1979
   145
	graph->nextOut(*this);
deba@1979
   146
	return *this; 
deba@1979
   147
      }
deba@1979
   148
deba@1979
   149
    };
deba@1979
   150
deba@1979
   151
deba@1979
   152
    class InEdgeIt : public Edge { 
deba@1979
   153
      const Graph* graph;
deba@1979
   154
    public:
deba@1979
   155
deba@1979
   156
      InEdgeIt() { }
deba@1979
   157
deba@1979
   158
      InEdgeIt(Invalid i) : Edge(i) { }
deba@1979
   159
deba@1979
   160
      InEdgeIt(const Graph& _graph, const Node& node) 
deba@1979
   161
	: graph(&_graph) {
deba@1979
   162
	_graph.firstIn(*this, node);
deba@1979
   163
      }
deba@1979
   164
deba@1979
   165
      InEdgeIt(const Graph& _graph, const Edge& edge) : 
deba@1979
   166
	Edge(edge), graph(&_graph) {}
deba@1979
   167
deba@1979
   168
      InEdgeIt& operator++() { 
deba@1979
   169
	graph->nextIn(*this);
deba@1979
   170
	return *this; 
deba@1979
   171
      }
deba@1979
   172
deba@1979
   173
    };
deba@1979
   174
deba@1979
   175
    /// \brief Base node of the iterator
deba@1979
   176
    ///
deba@1979
   177
    /// Returns the base node (ie. the source in this case) of the iterator
deba@1979
   178
    Node baseNode(const OutEdgeIt &e) const {
deba@1979
   179
      return Parent::source((Edge)e);
deba@1979
   180
    }
deba@1979
   181
    /// \brief Running node of the iterator
deba@1979
   182
    ///
deba@1979
   183
    /// Returns the running node (ie. the target in this case) of the
deba@1979
   184
    /// iterator
deba@1979
   185
    Node runningNode(const OutEdgeIt &e) const {
deba@1979
   186
      return Parent::target((Edge)e);
deba@1979
   187
    }
deba@1979
   188
deba@1979
   189
    /// \brief Base node of the iterator
deba@1979
   190
    ///
deba@1979
   191
    /// Returns the base node (ie. the target in this case) of the iterator
deba@1979
   192
    Node baseNode(const InEdgeIt &e) const {
deba@1979
   193
      return Parent::target((Edge)e);
deba@1979
   194
    }
deba@1979
   195
    /// \brief Running node of the iterator
deba@1979
   196
    ///
deba@1979
   197
    /// Returns the running node (ie. the source in this case) of the
deba@1979
   198
    /// iterator
deba@1979
   199
    Node runningNode(const InEdgeIt &e) const {
deba@1979
   200
      return Parent::source((Edge)e);
deba@1979
   201
    }
deba@1979
   202
deba@1979
   203
    using Parent::first;
deba@1979
   204
deba@1979
   205
    // Mappable extension
deba@1979
   206
    
deba@1979
   207
    template <typename _Value>
deba@1979
   208
    class EdgeMap 
deba@1979
   209
      : public IterableMapExtender<DefaultMap<Graph, Edge, _Value> > {
deba@1979
   210
    public:
deba@1979
   211
      typedef EdgeSetExtender Graph;
deba@1979
   212
      typedef IterableMapExtender<DefaultMap<Graph, Edge, _Value> > Parent;
deba@1979
   213
deba@1979
   214
      EdgeMap(const Graph& _g) 
deba@1979
   215
	: Parent(_g) {}
deba@1979
   216
      EdgeMap(const Graph& _g, const _Value& _v) 
deba@1979
   217
	: Parent(_g, _v) {}
deba@1979
   218
deba@1979
   219
      EdgeMap& operator=(const EdgeMap& cmap) {
deba@1979
   220
	return operator=<EdgeMap>(cmap);
deba@1979
   221
      }
deba@1979
   222
deba@1979
   223
      template <typename CMap>
deba@1979
   224
      EdgeMap& operator=(const CMap& cmap) {
deba@1979
   225
	checkConcept<concept::ReadMap<Edge, _Value>, CMap>();
deba@1979
   226
	const typename Parent::Graph* graph = Parent::getGraph();
deba@1979
   227
	Edge it;
deba@1979
   228
	for (graph->first(it); it != INVALID; graph->next(it)) {
deba@1979
   229
	  Parent::set(it, cmap[it]);
deba@1979
   230
	}
deba@1979
   231
	return *this;
deba@1979
   232
      }
deba@1979
   233
    };
deba@1979
   234
deba@1979
   235
deba@1979
   236
    // Alteration extension
deba@1979
   237
deba@1979
   238
    Edge addEdge(const Node& from, const Node& to) {
deba@1979
   239
      Edge edge = Parent::addEdge(from, to);
deba@1979
   240
      getNotifier(Edge()).add(edge);
deba@1979
   241
      return edge;
deba@1979
   242
    }
deba@1979
   243
    
deba@1979
   244
    void clear() {
deba@1979
   245
      getNotifier(Edge()).clear();
deba@1979
   246
      Parent::clear();
deba@1979
   247
    }
deba@1979
   248
deba@1979
   249
    void erase(const Edge& edge) {
deba@1979
   250
      getNotifier(Edge()).erase(edge);
deba@1979
   251
      Parent::erase(edge);
deba@1979
   252
    }
deba@1979
   253
deba@1979
   254
deba@1979
   255
    ~EdgeSetExtender() {
deba@1979
   256
      edge_notifier.clear();
deba@1979
   257
    }
deba@1979
   258
deba@1979
   259
  };
deba@1979
   260
deba@1979
   261
deba@1979
   262
  template <typename Base>
deba@1979
   263
  class UEdgeSetExtender : public Base {
deba@1979
   264
deba@1979
   265
  public:
deba@1979
   266
deba@1979
   267
    typedef Base Parent;
deba@1979
   268
    typedef UEdgeSetExtender Graph;
deba@1979
   269
deba@1979
   270
    typedef typename Parent::Node Node;
deba@1979
   271
    typedef typename Parent::Edge Edge;
deba@1979
   272
    typedef typename Parent::UEdge UEdge;
deba@1979
   273
deba@1979
   274
deba@1979
   275
    int maxId(Node) const {
deba@1979
   276
      return Parent::maxNodeId();
deba@1979
   277
    }
deba@1979
   278
deba@1979
   279
    int maxId(Edge) const {
deba@1979
   280
      return Parent::maxEdgeId();
deba@1979
   281
    }
deba@1979
   282
deba@1979
   283
    int maxId(UEdge) const {
deba@1979
   284
      return Parent::maxUEdgeId();
deba@1979
   285
    }
deba@1979
   286
deba@1979
   287
    Node fromId(int id, Node) const {
deba@1979
   288
      return Parent::nodeFromId(id);
deba@1979
   289
    }
deba@1979
   290
deba@1979
   291
    Edge fromId(int id, Edge) const {
deba@1979
   292
      return Parent::edgeFromId(id);
deba@1979
   293
    }
deba@1979
   294
deba@1979
   295
    UEdge fromId(int id, UEdge) const {
deba@1979
   296
      return Parent::uEdgeFromId(id);
deba@1979
   297
    }
deba@1979
   298
deba@1979
   299
    Node oppositeNode(const Node &n, const UEdge &e) const {
deba@1979
   300
      if( n == Parent::source(e))
deba@1979
   301
	return Parent::target(e);
deba@1979
   302
      else if( n == Parent::target(e))
deba@1979
   303
	return Parent::source(e);
deba@1979
   304
      else
deba@1979
   305
	return INVALID;
deba@1979
   306
    }
deba@1979
   307
deba@1979
   308
    Edge oppositeEdge(const Edge &e) const {
deba@1979
   309
      return Parent::direct(e, !Parent::direction(e));
deba@1979
   310
    }
deba@1979
   311
deba@1979
   312
    using Parent::direct;
deba@1979
   313
    Edge direct(const UEdge &ue, const Node &s) const {
deba@1979
   314
      return Parent::direct(ue, Parent::source(ue) == s);
deba@1979
   315
    }
deba@1979
   316
deba@1979
   317
    typedef AlterationNotifier<Edge> EdgeNotifier;
deba@1979
   318
    typedef AlterationNotifier<UEdge> UEdgeNotifier;
deba@1979
   319
deba@1979
   320
deba@1979
   321
  protected:
deba@1979
   322
deba@1979
   323
    mutable EdgeNotifier edge_notifier;
deba@1979
   324
    mutable UEdgeNotifier uedge_notifier;
deba@1979
   325
deba@1979
   326
  public:
deba@1991
   327
deba@1991
   328
    using Parent::getNotifier;
deba@1979
   329
    
deba@1979
   330
    EdgeNotifier& getNotifier(Edge) const {
deba@1979
   331
      return edge_notifier;
deba@1979
   332
    }
deba@1979
   333
deba@1979
   334
    UEdgeNotifier& getNotifier(UEdge) const {
deba@1979
   335
      return uedge_notifier;
deba@1979
   336
    }
deba@1979
   337
deba@1979
   338
deba@1979
   339
    class NodeIt : public Node { 
deba@1979
   340
      const Graph* graph;
deba@1979
   341
    public:
deba@1979
   342
deba@1979
   343
      NodeIt() {}
deba@1979
   344
deba@1979
   345
      NodeIt(Invalid i) : Node(i) { }
deba@1979
   346
deba@1979
   347
      explicit NodeIt(const Graph& _graph) : graph(&_graph) {
deba@1979
   348
	_graph.first(*static_cast<Node*>(this));
deba@1979
   349
      }
deba@1979
   350
deba@1979
   351
      NodeIt(const Graph& _graph, const Node& node) 
deba@1979
   352
	: Node(node), graph(&_graph) {}
deba@1979
   353
deba@1979
   354
      NodeIt& operator++() { 
deba@1979
   355
	graph->next(*this);
deba@1979
   356
	return *this; 
deba@1979
   357
      }
deba@1979
   358
deba@1979
   359
    };
deba@1979
   360
deba@1979
   361
deba@1979
   362
    class EdgeIt : public Edge { 
deba@1979
   363
      const Graph* graph;
deba@1979
   364
    public:
deba@1979
   365
deba@1979
   366
      EdgeIt() { }
deba@1979
   367
deba@1979
   368
      EdgeIt(Invalid i) : Edge(i) { }
deba@1979
   369
deba@1979
   370
      explicit EdgeIt(const Graph& _graph) : graph(&_graph) {
deba@1979
   371
	_graph.first(*static_cast<Edge*>(this));
deba@1979
   372
      }
deba@1979
   373
deba@1979
   374
      EdgeIt(const Graph& _graph, const Edge& e) : 
deba@1979
   375
	Edge(e), graph(&_graph) { }
deba@1979
   376
deba@1979
   377
      EdgeIt& operator++() { 
deba@1979
   378
	graph->next(*this);
deba@1979
   379
	return *this; 
deba@1979
   380
      }
deba@1979
   381
deba@1979
   382
    };
deba@1979
   383
deba@1979
   384
deba@1979
   385
    class OutEdgeIt : public Edge { 
deba@1979
   386
      const Graph* graph;
deba@1979
   387
    public:
deba@1979
   388
deba@1979
   389
      OutEdgeIt() { }
deba@1979
   390
deba@1979
   391
      OutEdgeIt(Invalid i) : Edge(i) { }
deba@1979
   392
deba@1979
   393
      OutEdgeIt(const Graph& _graph, const Node& node) 
deba@1979
   394
	: graph(&_graph) {
deba@1979
   395
	_graph.firstOut(*this, node);
deba@1979
   396
      }
deba@1979
   397
deba@1979
   398
      OutEdgeIt(const Graph& _graph, const Edge& edge) 
deba@1979
   399
	: Edge(edge), graph(&_graph) {}
deba@1979
   400
deba@1979
   401
      OutEdgeIt& operator++() { 
deba@1979
   402
	graph->nextOut(*this);
deba@1979
   403
	return *this; 
deba@1979
   404
      }
deba@1979
   405
deba@1979
   406
    };
deba@1979
   407
deba@1979
   408
deba@1979
   409
    class InEdgeIt : public Edge { 
deba@1979
   410
      const Graph* graph;
deba@1979
   411
    public:
deba@1979
   412
deba@1979
   413
      InEdgeIt() { }
deba@1979
   414
deba@1979
   415
      InEdgeIt(Invalid i) : Edge(i) { }
deba@1979
   416
deba@1979
   417
      InEdgeIt(const Graph& _graph, const Node& node) 
deba@1979
   418
	: graph(&_graph) {
deba@1979
   419
	_graph.firstIn(*this, node);
deba@1979
   420
      }
deba@1979
   421
deba@1979
   422
      InEdgeIt(const Graph& _graph, const Edge& edge) : 
deba@1979
   423
	Edge(edge), graph(&_graph) {}
deba@1979
   424
deba@1979
   425
      InEdgeIt& operator++() { 
deba@1979
   426
	graph->nextIn(*this);
deba@1979
   427
	return *this; 
deba@1979
   428
      }
deba@1979
   429
deba@1979
   430
    };
deba@1979
   431
deba@1979
   432
deba@1979
   433
    class UEdgeIt : public Parent::UEdge { 
deba@1979
   434
      const Graph* graph;
deba@1979
   435
    public:
deba@1979
   436
deba@1979
   437
      UEdgeIt() { }
deba@1979
   438
deba@1979
   439
      UEdgeIt(Invalid i) : UEdge(i) { }
deba@1979
   440
deba@1979
   441
      explicit UEdgeIt(const Graph& _graph) : graph(&_graph) {
deba@1979
   442
	_graph.first(*static_cast<UEdge*>(this));
deba@1979
   443
      }
deba@1979
   444
deba@1979
   445
      UEdgeIt(const Graph& _graph, const UEdge& e) : 
deba@1979
   446
	UEdge(e), graph(&_graph) { }
deba@1979
   447
deba@1979
   448
      UEdgeIt& operator++() { 
deba@1979
   449
	graph->next(*this);
deba@1979
   450
	return *this; 
deba@1979
   451
      }
deba@1979
   452
deba@1979
   453
    };
deba@1979
   454
deba@1979
   455
    class IncEdgeIt : public Parent::UEdge {
deba@1979
   456
      friend class UEdgeSetExtender;
deba@1979
   457
      const Graph* graph;
deba@1979
   458
      bool direction;
deba@1979
   459
    public:
deba@1979
   460
deba@1979
   461
      IncEdgeIt() { }
deba@1979
   462
deba@1979
   463
      IncEdgeIt(Invalid i) : UEdge(i), direction(false) { }
deba@1979
   464
deba@1979
   465
      IncEdgeIt(const Graph& _graph, const Node &n) : graph(&_graph) {
deba@1979
   466
	_graph.firstInc(*this, direction, n);
deba@1979
   467
      }
deba@1979
   468
deba@1979
   469
      IncEdgeIt(const Graph& _graph, const UEdge &ue, const Node &n)
deba@1979
   470
	: graph(&_graph), UEdge(ue) {
deba@1979
   471
	direction = (_graph.source(ue) == n);
deba@1979
   472
      }
deba@1979
   473
deba@1979
   474
      IncEdgeIt& operator++() {
deba@1979
   475
	graph->nextInc(*this, direction);
deba@1979
   476
	return *this; 
deba@1979
   477
      }
deba@1979
   478
    };
deba@1979
   479
deba@1979
   480
    /// \brief Base node of the iterator
deba@1979
   481
    ///
deba@1979
   482
    /// Returns the base node (ie. the source in this case) of the iterator
deba@1979
   483
    Node baseNode(const OutEdgeIt &e) const {
deba@1979
   484
      return Parent::source((Edge)e);
deba@1979
   485
    }
deba@1979
   486
    /// \brief Running node of the iterator
deba@1979
   487
    ///
deba@1979
   488
    /// Returns the running node (ie. the target in this case) of the
deba@1979
   489
    /// iterator
deba@1979
   490
    Node runningNode(const OutEdgeIt &e) const {
deba@1979
   491
      return Parent::target((Edge)e);
deba@1979
   492
    }
deba@1979
   493
deba@1979
   494
    /// \brief Base node of the iterator
deba@1979
   495
    ///
deba@1979
   496
    /// Returns the base node (ie. the target in this case) of the iterator
deba@1979
   497
    Node baseNode(const InEdgeIt &e) const {
deba@1979
   498
      return Parent::target((Edge)e);
deba@1979
   499
    }
deba@1979
   500
    /// \brief Running node of the iterator
deba@1979
   501
    ///
deba@1979
   502
    /// Returns the running node (ie. the source in this case) of the
deba@1979
   503
    /// iterator
deba@1979
   504
    Node runningNode(const InEdgeIt &e) const {
deba@1979
   505
      return Parent::source((Edge)e);
deba@1979
   506
    }
deba@1979
   507
deba@1979
   508
    /// Base node of the iterator
deba@1979
   509
    ///
deba@1979
   510
    /// Returns the base node of the iterator
deba@1979
   511
    Node baseNode(const IncEdgeIt &e) const {
deba@1979
   512
      return e.direction ? source(e) : target(e);
deba@1979
   513
    }
deba@1979
   514
    /// Running node of the iterator
deba@1979
   515
    ///
deba@1979
   516
    /// Returns the running node of the iterator
deba@1979
   517
    Node runningNode(const IncEdgeIt &e) const {
deba@1979
   518
      return e.direction ? target(e) : source(e);
deba@1979
   519
    }
deba@1979
   520
deba@1979
   521
deba@1979
   522
    template <typename _Value>
deba@1979
   523
    class EdgeMap 
deba@1979
   524
      : public IterableMapExtender<DefaultMap<Graph, Edge, _Value> > {
deba@1979
   525
    public:
deba@1979
   526
      typedef UEdgeSetExtender Graph;
deba@1979
   527
      typedef IterableMapExtender<DefaultMap<Graph, Edge, _Value> > Parent;
deba@1979
   528
deba@1979
   529
      EdgeMap(const Graph& _g) 
deba@1979
   530
	: Parent(_g) {}
deba@1979
   531
      EdgeMap(const Graph& _g, const _Value& _v) 
deba@1979
   532
	: Parent(_g, _v) {}
deba@1979
   533
deba@1979
   534
      EdgeMap& operator=(const EdgeMap& cmap) {
deba@1979
   535
	return operator=<EdgeMap>(cmap);
deba@1979
   536
      }
deba@1979
   537
deba@1979
   538
      template <typename CMap>
deba@1979
   539
      EdgeMap& operator=(const CMap& cmap) {
deba@1979
   540
	checkConcept<concept::ReadMap<Edge, _Value>, CMap>();
deba@1979
   541
	const typename Parent::Graph* graph = Parent::getGraph();
deba@1979
   542
	Edge it;
deba@1979
   543
	for (graph->first(it); it != INVALID; graph->next(it)) {
deba@1979
   544
	  Parent::set(it, cmap[it]);
deba@1979
   545
	}
deba@1979
   546
	return *this;
deba@1979
   547
      }
deba@1979
   548
    };
deba@1979
   549
deba@1979
   550
deba@1979
   551
    template <typename _Value>
deba@1979
   552
    class UEdgeMap 
deba@1979
   553
      : public IterableMapExtender<DefaultMap<Graph, UEdge, _Value> > {
deba@1979
   554
    public:
deba@1979
   555
      typedef UEdgeSetExtender Graph;
deba@1979
   556
      typedef IterableMapExtender<DefaultMap<Graph, UEdge, _Value> > Parent;
deba@1979
   557
deba@1979
   558
      UEdgeMap(const Graph& _g) 
deba@1979
   559
	: Parent(_g) {}
deba@1979
   560
      UEdgeMap(const Graph& _g, const _Value& _v) 
deba@1979
   561
	: Parent(_g, _v) {}
deba@1979
   562
deba@1979
   563
      UEdgeMap& operator=(const UEdgeMap& cmap) {
deba@1979
   564
	return operator=<UEdgeMap>(cmap);
deba@1979
   565
      }
deba@1979
   566
deba@1979
   567
      template <typename CMap>
deba@1979
   568
      UEdgeMap& operator=(const CMap& cmap) {
deba@1979
   569
	checkConcept<concept::ReadMap<UEdge, _Value>, CMap>();
deba@1979
   570
	const typename Parent::Graph* graph = Parent::getGraph();
deba@1979
   571
	UEdge it;
deba@1979
   572
	for (graph->first(it); it != INVALID; graph->next(it)) {
deba@1979
   573
	  Parent::set(it, cmap[it]);
deba@1979
   574
	}
deba@1979
   575
	return *this;
deba@1979
   576
      }
deba@1979
   577
    };
deba@1979
   578
deba@1979
   579
deba@1979
   580
    // Alteration extension
deba@1979
   581
deba@1979
   582
    UEdge addEdge(const Node& from, const Node& to) {
deba@1979
   583
      UEdge uedge = Parent::addEdge(from, to);
deba@1979
   584
      getNotifier(UEdge()).add(uedge);
deba@1979
   585
      getNotifier(Edge()).add(Parent::direct(uedge, true));
deba@1979
   586
      getNotifier(Edge()).add(Parent::direct(uedge, false));
deba@1979
   587
      return uedge;
deba@1979
   588
    }
deba@1979
   589
    
deba@1979
   590
    void clear() {
deba@1979
   591
      getNotifier(Edge()).clear();
deba@1979
   592
      getNotifier(UEdge()).clear();
deba@1979
   593
      Parent::clear();
deba@1979
   594
    }
deba@1979
   595
deba@1979
   596
    void erase(const UEdge& uedge) {
deba@1979
   597
      getNotifier(Edge()).erase(Parent::direct(uedge, true));
deba@1979
   598
      getNotifier(Edge()).erase(Parent::direct(uedge, false));
deba@1979
   599
      getNotifier(UEdge()).erase(uedge);
deba@1979
   600
      Parent::erase(uedge);
deba@1979
   601
    }
deba@1979
   602
deba@1979
   603
deba@1979
   604
    ~UEdgeSetExtender() {
deba@1979
   605
      getNotifier(Edge()).clear();
deba@1979
   606
      getNotifier(UEdge()).clear();
deba@1979
   607
    }
deba@1979
   608
    
deba@1979
   609
  };
deba@1979
   610
}