lemon/dim2.h
author deba
Tue, 30 Oct 2007 20:21:10 +0000
changeset 2505 1bb471764ab8
parent 2391 14a343be7a5a
child 2553 bfced05fa852
permissions -rw-r--r--
Redesign interface of MaxMatching and UnionFindEnum
New class ExtendFindEnum

Faster MaxMatching
alpar@906
     1
/* -*- C++ -*-
alpar@906
     2
 *
alpar@1956
     3
 * This file is a part of LEMON, a generic C++ optimization library
alpar@1956
     4
 *
alpar@2391
     5
 * Copyright (C) 2003-2007
alpar@1956
     6
 * Egervary Jeno Kombinatorikus Optimalizalasi Kutatocsoport
alpar@1359
     7
 * (Egervary Research Group on Combinatorial Optimization, EGRES).
alpar@906
     8
 *
alpar@906
     9
 * Permission to use, modify and distribute this software is granted
alpar@906
    10
 * provided that this copyright notice appears in all copies. For
alpar@906
    11
 * precise terms see the accompanying LICENSE file.
alpar@906
    12
 *
alpar@906
    13
 * This software is provided "AS IS" with no warranty of any kind,
alpar@906
    14
 * express or implied, and with no claim as to its suitability for any
alpar@906
    15
 * purpose.
alpar@906
    16
 *
alpar@906
    17
 */
alpar@906
    18
alpar@2207
    19
#ifndef LEMON_DIM2_H
alpar@2207
    20
#define LEMON_DIM2_H
athos@201
    21
athos@201
    22
#include <iostream>
deba@1993
    23
#include <lemon/bits/utility.h>
athos@201
    24
klao@491
    25
///\ingroup misc
alpar@249
    26
///\file
alpar@249
    27
///\brief A simple two dimensional vector and a bounding box implementation 
alpar@249
    28
///
alpar@2207
    29
/// The class \ref lemon::dim2::Point "dim2::Point" implements
alpar@249
    30
///a two dimensional vector with the usual
alpar@249
    31
/// operations.
alpar@249
    32
///
alpar@2207
    33
/// The class \ref lemon::dim2::BoundingBox "dim2::BoundingBox"
alpar@2207
    34
/// can be used to determine
alpar@2207
    35
/// the rectangular bounding box of a set of
alpar@2207
    36
/// \ref lemon::dim2::Point "dim2::Point"'s.
alpar@458
    37
///
alpar@458
    38
///\author Attila Bernath
alpar@249
    39
alpar@249
    40
alpar@921
    41
namespace lemon {
alpar@431
    42
alpar@2207
    43
  ///Tools for handling two dimensional coordinates
alpar@2207
    44
alpar@2207
    45
  ///This namespace is a storage of several
alpar@2207
    46
  ///tools for handling two dimensional coordinates
alpar@2207
    47
  namespace dim2 {
alpar@2207
    48
alpar@431
    49
  /// \addtogroup misc
alpar@431
    50
  /// @{
alpar@431
    51
alpar@1257
    52
  /// A simple two dimensional vector (plainvector) implementation
alpar@242
    53
alpar@1257
    54
  /// A simple two dimensional vector (plainvector) implementation
alpar@458
    55
  ///with the usual vector
alpar@458
    56
  /// operators.
alpar@458
    57
  ///
athos@207
    58
  template<typename T>
alpar@2207
    59
    class Point {
athos@201
    60
athos@207
    61
    public:
athos@240
    62
alpar@987
    63
      typedef T Value;
alpar@964
    64
alpar@1974
    65
      ///First co-ordinate
alpar@1974
    66
      T x;
alpar@1974
    67
      ///Second co-ordinate
alpar@1974
    68
      T y;     
athos@207
    69
      
alpar@1257
    70
      ///Default constructor
alpar@2207
    71
      Point() {}
athos@201
    72
alpar@2157
    73
      ///Construct an instance from coordinates
alpar@2207
    74
      Point(T a, T b) : x(a), y(b) { }
athos@201
    75
deba@2217
    76
      ///The dimension of the vector.
deba@2217
    77
deba@2217
    78
      ///This class give back always 2.
deba@2217
    79
      ///
deba@2212
    80
      int size() const { return 2; }
deba@2212
    81
deba@2212
    82
      ///Subscripting operator
deba@2217
    83
deba@2217
    84
      ///\c p[0] is \c p.x and \c p[1] is \c p.y
deba@2217
    85
      ///
deba@2212
    86
      T& operator[](int idx) { return idx == 0 ? x : y; }
deba@2212
    87
deba@2212
    88
      ///Const subscripting operator
deba@2217
    89
deba@2217
    90
      ///\c p[0] is \c p.x and \c p[1] is \c p.y
deba@2217
    91
      ///
deba@2212
    92
      const T& operator[](int idx) const { return idx == 0 ? x : y; }
athos@201
    93
alpar@1049
    94
      ///Conversion constructor
alpar@2207
    95
      template<class TT> Point(const Point<TT> &p) : x(p.x), y(p.y) {}
alpar@1049
    96
alpar@2157
    97
      ///Give back the square of the norm of the vector
alpar@1257
    98
      T normSquare() const {
ladanyi@1426
    99
        return x*x+y*y;
alpar@1391
   100
      }
athos@201
   101
  
alpar@2157
   102
      ///Increment the left hand side by u
alpar@2207
   103
      Point<T>& operator +=(const Point<T>& u) {
ladanyi@1426
   104
        x += u.x;
ladanyi@1426
   105
        y += u.y;
ladanyi@1426
   106
        return *this;
alpar@1391
   107
      }
athos@201
   108
  
alpar@2157
   109
      ///Decrement the left hand side by u
alpar@2207
   110
      Point<T>& operator -=(const Point<T>& u) {
ladanyi@1426
   111
        x -= u.x;
ladanyi@1426
   112
        y -= u.y;
ladanyi@1426
   113
        return *this;
alpar@1391
   114
      }
athos@201
   115
alpar@2157
   116
      ///Multiply the left hand side with a scalar
alpar@2207
   117
      Point<T>& operator *=(const T &u) {
ladanyi@1426
   118
        x *= u;
ladanyi@1426
   119
        y *= u;
ladanyi@1426
   120
        return *this;
alpar@1391
   121
      }
athos@207
   122
alpar@2157
   123
      ///Divide the left hand side by a scalar
alpar@2207
   124
      Point<T>& operator /=(const T &u) {
ladanyi@1426
   125
        x /= u;
ladanyi@1426
   126
        y /= u;
ladanyi@1426
   127
        return *this;
alpar@1391
   128
      }
athos@201
   129
  
alpar@2157
   130
      ///Return the scalar product of two vectors
alpar@2207
   131
      T operator *(const Point<T>& u) const {
ladanyi@1426
   132
        return x*u.x+y*u.y;
alpar@1391
   133
      }
athos@201
   134
  
alpar@2157
   135
      ///Return the sum of two vectors
alpar@2207
   136
      Point<T> operator+(const Point<T> &u) const {
alpar@2207
   137
        Point<T> b=*this;
ladanyi@1426
   138
        return b+=u;
alpar@1391
   139
      }
athos@201
   140
alpar@2157
   141
      ///Return the neg of the vectors
alpar@2207
   142
      Point<T> operator-() const {
alpar@2207
   143
        Point<T> b=*this;
ladanyi@1426
   144
        b.x=-b.x; b.y=-b.y;
ladanyi@1426
   145
        return b;
alpar@1391
   146
      }
alpar@1049
   147
alpar@2157
   148
      ///Return the difference of two vectors
alpar@2207
   149
      Point<T> operator-(const Point<T> &u) const {
alpar@2207
   150
        Point<T> b=*this;
ladanyi@1426
   151
        return b-=u;
alpar@1391
   152
      }
athos@201
   153
alpar@2157
   154
      ///Return a vector multiplied by a scalar
alpar@2207
   155
      Point<T> operator*(const T &u) const {
alpar@2207
   156
        Point<T> b=*this;
ladanyi@1426
   157
        return b*=u;
alpar@1391
   158
      }
athos@201
   159
alpar@2157
   160
      ///Return a vector divided by a scalar
alpar@2207
   161
      Point<T> operator/(const T &u) const {
alpar@2207
   162
        Point<T> b=*this;
ladanyi@1426
   163
        return b/=u;
alpar@1391
   164
      }
athos@201
   165
alpar@2157
   166
      ///Test equality
alpar@2207
   167
      bool operator==(const Point<T> &u) const {
ladanyi@1426
   168
        return (x==u.x) && (y==u.y);
alpar@1391
   169
      }
athos@201
   170
alpar@2157
   171
      ///Test inequality
alpar@2207
   172
      bool operator!=(Point u) const {
ladanyi@1426
   173
        return  (x!=u.x) || (y!=u.y);
alpar@1391
   174
      }
athos@201
   175
athos@207
   176
    };
athos@201
   177
alpar@2207
   178
  ///Return an Point 
deba@1999
   179
alpar@2207
   180
  ///Return an Point
alpar@2207
   181
  ///\relates Point
deba@1999
   182
  template <typename T>
deba@2212
   183
  inline Point<T> makePoint(const T& x, const T& y) {
alpar@2207
   184
    return Point<T>(x, y);
deba@1999
   185
  }
deba@1999
   186
alpar@2157
   187
  ///Return a vector multiplied by a scalar
alpar@1083
   188
alpar@2157
   189
  ///Return a vector multiplied by a scalar
alpar@2207
   190
  ///\relates Point
alpar@2207
   191
  template<typename T> Point<T> operator*(const T &u,const Point<T> &x) {
alpar@1071
   192
    return x*u;
alpar@1391
   193
  }
alpar@1071
   194
alpar@814
   195
  ///Read a plainvector from a stream
alpar@814
   196
alpar@967
   197
  ///Read a plainvector from a stream
alpar@2207
   198
  ///\relates Point
alpar@814
   199
  ///
athos@207
   200
  template<typename T>
alpar@2207
   201
  inline std::istream& operator>>(std::istream &is, Point<T> &z) {
deba@1392
   202
    char c;
deba@1392
   203
    if (is >> c) {
deba@1392
   204
      if (c != '(') is.putback(c);
deba@1392
   205
    } else {
deba@1392
   206
      is.clear();
deba@1392
   207
    }
deba@1392
   208
    if (!(is >> z.x)) return is;
deba@1392
   209
    if (is >> c) {
deba@1392
   210
      if (c != ',') is.putback(c);
deba@1392
   211
    } else {
deba@1392
   212
      is.clear();
deba@1392
   213
    }
deba@1392
   214
    if (!(is >> z.y)) return is;
deba@1392
   215
    if (is >> c) {
deba@1392
   216
      if (c != ')') is.putback(c);
deba@1392
   217
    } else {
deba@1392
   218
      is.clear();
deba@1392
   219
    }
athos@207
   220
    return is;
athos@207
   221
  }
athos@201
   222
alpar@814
   223
  ///Write a plainvector to a stream
alpar@814
   224
alpar@967
   225
  ///Write a plainvector to a stream
alpar@2207
   226
  ///\relates Point
alpar@814
   227
  ///
athos@207
   228
  template<typename T>
alpar@2207
   229
  inline std::ostream& operator<<(std::ostream &os, const Point<T>& z)
athos@207
   230
  {
athos@240
   231
    os << "(" << z.x << ", " << z.y << ")";
athos@207
   232
    return os;
athos@207
   233
  }
athos@207
   234
alpar@1202
   235
  ///Rotate by 90 degrees
alpar@1202
   236
alpar@1202
   237
  ///Returns its parameter rotated by 90 degrees in positive direction.
alpar@2207
   238
  ///\relates Point
alpar@1202
   239
  ///
alpar@1202
   240
  template<typename T>
alpar@2207
   241
  inline Point<T> rot90(const Point<T> &z)
alpar@1202
   242
  {
alpar@2207
   243
    return Point<T>(-z.y,z.x);
alpar@1202
   244
  }
alpar@1202
   245
alpar@2157
   246
  ///Rotate by 180 degrees
alpar@2157
   247
alpar@2157
   248
  ///Returns its parameter rotated by 180 degrees.
alpar@2207
   249
  ///\relates Point
alpar@2157
   250
  ///
alpar@2157
   251
  template<typename T>
alpar@2207
   252
  inline Point<T> rot180(const Point<T> &z)
alpar@2157
   253
  {
alpar@2207
   254
    return Point<T>(-z.x,-z.y);
alpar@2157
   255
  }
alpar@2157
   256
alpar@1202
   257
  ///Rotate by 270 degrees
alpar@1202
   258
alpar@1202
   259
  ///Returns its parameter rotated by 90 degrees in negative direction.
alpar@2207
   260
  ///\relates Point
alpar@1202
   261
  ///
alpar@1202
   262
  template<typename T>
alpar@2207
   263
  inline Point<T> rot270(const Point<T> &z)
alpar@1202
   264
  {
alpar@2207
   265
    return Point<T>(z.y,-z.x);
alpar@1202
   266
  }
alpar@1202
   267
alpar@1202
   268
  
athos@244
   269
alpar@458
   270
  /// A class to calculate or store the bounding box of plainvectors.
alpar@458
   271
alpar@458
   272
  /// A class to calculate or store the bounding box of plainvectors.
alpar@458
   273
  ///
alpar@458
   274
  ///\author Attila Bernath
alpar@2451
   275
    template<typename T>
athos@244
   276
    class BoundingBox {
alpar@2207
   277
      Point<T> bottom_left, top_right;
athos@244
   278
      bool _empty;
athos@244
   279
    public:
athos@244
   280
      
ladanyi@1426
   281
      ///Default constructor: creates an empty bounding box
athos@244
   282
      BoundingBox() { _empty = true; }
athos@244
   283
alpar@2157
   284
      ///Construct an instance from one point
alpar@2207
   285
      BoundingBox(Point<T> a) { bottom_left=top_right=a; _empty = false; }
alpar@2451
   286
      
alpar@2451
   287
      ///Construct an instance from two points
alpar@2451
   288
      
alpar@2451
   289
      ///Construct an instance from two points
alpar@2451
   290
      ///\warning The coordinates of the bottom-left corner must be no more
alpar@2451
   291
      ///than those of the top-right one
alpar@2451
   292
      BoundingBox(Point<T> a,Point<T> b)
alpar@2451
   293
      {
alpar@2451
   294
	bottom_left=a;
alpar@2451
   295
	top_right=b;
alpar@2451
   296
	_empty = false;
alpar@2451
   297
      }
alpar@2451
   298
      
alpar@2451
   299
      ///Construct an instance from four numbers
athos@244
   300
alpar@2451
   301
      ///Construct an instance from four numbers
alpar@2451
   302
      ///\warning The coordinates of the bottom-left corner must be no more
alpar@2451
   303
      ///than those of the top-right one
alpar@2451
   304
      BoundingBox(T l,T b,T r,T t)
alpar@2451
   305
      {
alpar@2451
   306
	bottom_left=Point<T>(l,b);
alpar@2451
   307
	top_right=Point<T>(r,t);
alpar@2451
   308
	_empty = false;
alpar@2451
   309
      }
alpar@2451
   310
      
ladanyi@1426
   311
      ///Were any points added?
athos@244
   312
      bool empty() const {
ladanyi@1426
   313
        return _empty;
athos@244
   314
      }
alpar@2451
   315
      
alpar@2157
   316
      ///Make the BoundingBox empty
alpar@1391
   317
      void clear() {
ladanyi@1426
   318
        _empty=1;
alpar@1391
   319
      }
alpar@1391
   320
alpar@2157
   321
      ///Give back the bottom left corner
alpar@2157
   322
alpar@2157
   323
      ///Give back the bottom left corner.
alpar@2157
   324
      ///If the bounding box is empty, then the return value is not defined.
alpar@2207
   325
      Point<T> bottomLeft() const {
ladanyi@1426
   326
        return bottom_left;
alpar@1391
   327
      }
athos@244
   328
alpar@2157
   329
      ///Set the bottom left corner
alpar@2157
   330
alpar@2157
   331
      ///Set the bottom left corner.
alpar@2157
   332
      ///It should only bee used for non-empty box.
alpar@2207
   333
      void bottomLeft(Point<T> p) {
alpar@1927
   334
	bottom_left = p;
alpar@1927
   335
      }
alpar@1927
   336
alpar@2157
   337
      ///Give back the top right corner
alpar@2157
   338
alpar@2157
   339
      ///Give back the top right corner.
alpar@2157
   340
      ///If the bounding box is empty, then the return value is not defined.
alpar@2207
   341
      Point<T> topRight() const {
ladanyi@1426
   342
        return top_right;
alpar@1391
   343
      }
athos@244
   344
alpar@2157
   345
      ///Set the top right corner
alpar@2157
   346
alpar@2157
   347
      ///Set the top right corner.
alpar@2157
   348
      ///It should only bee used for non-empty box.
alpar@2207
   349
      void topRight(Point<T> p) {
alpar@1927
   350
	top_right = p;
alpar@1927
   351
      }
alpar@1927
   352
alpar@2157
   353
      ///Give back the bottom right corner
alpar@2157
   354
alpar@2157
   355
      ///Give back the bottom right corner.
alpar@2157
   356
      ///If the bounding box is empty, then the return value is not defined.
alpar@2207
   357
      Point<T> bottomRight() const {
alpar@2207
   358
        return Point<T>(top_right.x,bottom_left.y);
alpar@1391
   359
      }
alpar@1045
   360
alpar@2157
   361
      ///Set the bottom right corner
alpar@2157
   362
alpar@2157
   363
      ///Set the bottom right corner.
alpar@2157
   364
      ///It should only bee used for non-empty box.
alpar@2207
   365
      void bottomRight(Point<T> p) {
alpar@1927
   366
	top_right.x = p.x;
alpar@1927
   367
	bottom_left.y = p.y;
alpar@1927
   368
      }
alpar@2157
   369
 
alpar@2157
   370
      ///Give back the top left corner
alpar@1927
   371
alpar@2157
   372
      ///Give back the top left corner.
alpar@2157
   373
      ///If the bounding box is empty, then the return value is not defined.
alpar@2207
   374
      Point<T> topLeft() const {
alpar@2207
   375
        return Point<T>(bottom_left.x,top_right.y);
alpar@1391
   376
      }
alpar@1045
   377
alpar@2157
   378
      ///Set the top left corner
alpar@2157
   379
alpar@2157
   380
      ///Set the top left corner.
alpar@2157
   381
      ///It should only bee used for non-empty box.
alpar@2207
   382
      void topLeft(Point<T> p) {
alpar@1927
   383
	top_right.y = p.y;
alpar@1927
   384
	bottom_left.x = p.x;
alpar@1927
   385
      }
alpar@1927
   386
alpar@2157
   387
      ///Give back the bottom of the box
alpar@2157
   388
alpar@2157
   389
      ///Give back the bottom of the box.
alpar@2157
   390
      ///If the bounding box is empty, then the return value is not defined.
alpar@1045
   391
      T bottom() const {
ladanyi@1426
   392
        return bottom_left.y;
alpar@1391
   393
      }
alpar@1045
   394
alpar@2157
   395
      ///Set the bottom of the box
alpar@2157
   396
alpar@2157
   397
      ///Set the bottom of the box.
alpar@2157
   398
      ///It should only bee used for non-empty box.
alpar@1927
   399
      void bottom(T t) {
alpar@1927
   400
	bottom_left.y = t;
alpar@1927
   401
      }
alpar@1927
   402
alpar@2157
   403
      ///Give back the top of the box
alpar@2157
   404
alpar@2157
   405
      ///Give back the top of the box.
alpar@2157
   406
      ///If the bounding box is empty, then the return value is not defined.
alpar@1045
   407
      T top() const {
ladanyi@1426
   408
        return top_right.y;
alpar@1391
   409
      }
alpar@1045
   410
alpar@2157
   411
      ///Set the top of the box
alpar@2157
   412
alpar@2157
   413
      ///Set the top of the box.
alpar@2157
   414
      ///It should only bee used for non-empty box.
alpar@1927
   415
      void top(T t) {
alpar@1927
   416
	top_right.y = t;
alpar@1927
   417
      }
alpar@1927
   418
alpar@2157
   419
      ///Give back the left side of the box
alpar@2157
   420
alpar@2157
   421
      ///Give back the left side of the box.
alpar@2157
   422
      ///If the bounding box is empty, then the return value is not defined.
alpar@1045
   423
      T left() const {
ladanyi@1426
   424
        return bottom_left.x;
alpar@1391
   425
      }
alpar@2157
   426
 
alpar@2157
   427
      ///Set the left side of the box
alpar@1045
   428
alpar@2157
   429
      ///Set the left side of the box.
alpar@2157
   430
      ///It should only bee used for non-empty box
alpar@1927
   431
      void left(T t) {
alpar@1927
   432
	bottom_left.x = t;
alpar@1927
   433
      }
alpar@1927
   434
alpar@2157
   435
      /// Give back the right side of the box
alpar@2157
   436
alpar@2157
   437
      /// Give back the right side of the box.
alpar@2157
   438
      ///If the bounding box is empty, then the return value is not defined.
alpar@1045
   439
      T right() const {
ladanyi@1426
   440
        return top_right.x;
alpar@1391
   441
      }
alpar@1045
   442
alpar@2157
   443
      ///Set the right side of the box
alpar@2157
   444
alpar@2157
   445
      ///Set the right side of the box.
alpar@2157
   446
      ///It should only bee used for non-empty box
alpar@1927
   447
      void right(T t) {
alpar@1927
   448
	top_right.x = t;
alpar@1927
   449
      }
alpar@1927
   450
alpar@2157
   451
      ///Give back the height of the box
alpar@2157
   452
alpar@2157
   453
      ///Give back the height of the box.
alpar@2157
   454
      ///If the bounding box is empty, then the return value is not defined.
alpar@1102
   455
      T height() const {
ladanyi@1426
   456
        return top_right.y-bottom_left.y;
alpar@1391
   457
      }
alpar@1102
   458
alpar@2157
   459
      ///Give back the width of the box
alpar@2157
   460
alpar@2157
   461
      ///Give back the width of the box.
alpar@2157
   462
      ///If the bounding box is empty, then the return value is not defined.
alpar@1102
   463
      T width() const {
ladanyi@1426
   464
        return top_right.x-bottom_left.x;
alpar@1391
   465
      }
alpar@1102
   466
athos@244
   467
      ///Checks whether a point is inside a bounding box
alpar@2207
   468
      bool inside(const Point<T>& u){
ladanyi@1426
   469
        if (_empty)
ladanyi@1426
   470
          return false;
ladanyi@1426
   471
        else{
ladanyi@1426
   472
          return ((u.x-bottom_left.x)*(top_right.x-u.x) >= 0 &&
ladanyi@1426
   473
              (u.y-bottom_left.y)*(top_right.y-u.y) >= 0 );
ladanyi@1426
   474
        }
athos@244
   475
      }
athos@244
   476
  
athos@244
   477
      ///Increments a bounding box with a point
alpar@2207
   478
      BoundingBox& add(const Point<T>& u){
ladanyi@1426
   479
        if (_empty){
ladanyi@1426
   480
          bottom_left=top_right=u;
ladanyi@1426
   481
          _empty = false;
ladanyi@1426
   482
        }
ladanyi@1426
   483
        else{
ladanyi@1426
   484
          if (bottom_left.x > u.x) bottom_left.x = u.x;
ladanyi@1426
   485
          if (bottom_left.y > u.y) bottom_left.y = u.y;
ladanyi@1426
   486
          if (top_right.x < u.x) top_right.x = u.x;
ladanyi@1426
   487
          if (top_right.y < u.y) top_right.y = u.y;
ladanyi@1426
   488
        }
ladanyi@1426
   489
        return *this;
alpar@1391
   490
      }
alpar@2214
   491
    
alpar@2214
   492
      ///Increments a bounding to contain another bounding box
alpar@1588
   493
      BoundingBox& add(const BoundingBox &u){
ladanyi@1426
   494
        if ( !u.empty() ){
alpar@1588
   495
          this->add(u.bottomLeft());
alpar@1588
   496
	  this->add(u.topRight());
ladanyi@1426
   497
        }
ladanyi@1426
   498
        return *this;
alpar@1391
   499
      }
athos@244
   500
  
alpar@1588
   501
      ///Intersection of two bounding boxes
alpar@1588
   502
      BoundingBox operator &(const BoundingBox& u){
alpar@1588
   503
        BoundingBox b;
alpar@1588
   504
	b.bottom_left.x=std::max(this->bottom_left.x,u.bottom_left.x);
alpar@1588
   505
	b.bottom_left.y=std::max(this->bottom_left.y,u.bottom_left.y);
alpar@1588
   506
	b.top_right.x=std::min(this->top_right.x,u.top_right.x);
alpar@1588
   507
	b.top_right.y=std::min(this->top_right.y,u.top_right.y);
alpar@1588
   508
	b._empty = this->_empty || u._empty ||
alpar@1588
   509
	  b.bottom_left.x>top_right.x && b.bottom_left.y>top_right.y;
alpar@1588
   510
        return b;
alpar@1391
   511
      }
athos@244
   512
athos@244
   513
    };//class Boundingbox
athos@244
   514
athos@244
   515
alpar@2207
   516
  ///Map of x-coordinates of a dim2::Point<>-map
alpar@1317
   517
alpar@1317
   518
  ///\ingroup maps
alpar@2214
   519
  ///Map of x-coordinates of a dim2::Point<>-map
alpar@1317
   520
  ///
alpar@1317
   521
  template<class M>
alpar@1317
   522
  class XMap 
alpar@1317
   523
  {
deba@1706
   524
    M& _map;
alpar@1317
   525
  public:
deba@1420
   526
alpar@1317
   527
    typedef typename M::Value::Value Value;
alpar@1317
   528
    typedef typename M::Key Key;
alpar@1317
   529
    ///\e
deba@1706
   530
    XMap(M& map) : _map(map) {}
alpar@1317
   531
    Value operator[](Key k) const {return _map[k].x;}
alpar@1352
   532
    void set(Key k,Value v) {_map.set(k,typename M::Value(v,_map[k].y));}
alpar@1317
   533
  };
alpar@1317
   534
    
alpar@1317
   535
  ///Returns an \ref XMap class
alpar@1317
   536
alpar@1317
   537
  ///This function just returns an \ref XMap class.
alpar@1317
   538
  ///
alpar@1317
   539
  ///\ingroup maps
alpar@1317
   540
  ///\relates XMap
alpar@1317
   541
  template<class M> 
alpar@1317
   542
  inline XMap<M> xMap(M &m) 
alpar@1317
   543
  {
alpar@1317
   544
    return XMap<M>(m);
alpar@1317
   545
  }
alpar@1317
   546
deba@1420
   547
  template<class M> 
deba@1420
   548
  inline XMap<M> xMap(const M &m) 
deba@1420
   549
  {
deba@1420
   550
    return XMap<M>(m);
deba@1420
   551
  }
deba@1420
   552
alpar@1317
   553
  ///Constant (read only) version of \ref XMap
alpar@1317
   554
alpar@1317
   555
  ///\ingroup maps
alpar@2214
   556
  ///Constant (read only) version of \ref XMap
alpar@1317
   557
  ///
alpar@1317
   558
  template<class M>
alpar@1317
   559
  class ConstXMap 
alpar@1317
   560
  {
deba@1706
   561
    const M& _map;
alpar@1317
   562
  public:
deba@1420
   563
alpar@1317
   564
    typedef typename M::Value::Value Value;
alpar@1317
   565
    typedef typename M::Key Key;
alpar@1317
   566
    ///\e
alpar@1317
   567
    ConstXMap(const M &map) : _map(map) {}
alpar@1317
   568
    Value operator[](Key k) const {return _map[k].x;}
alpar@1317
   569
  };
alpar@1317
   570
    
alpar@1317
   571
  ///Returns a \ref ConstXMap class
alpar@1317
   572
alpar@1317
   573
  ///This function just returns an \ref ConstXMap class.
alpar@1317
   574
  ///
alpar@1317
   575
  ///\ingroup maps
alpar@1317
   576
  ///\relates ConstXMap
alpar@1317
   577
  template<class M> 
alpar@1317
   578
  inline ConstXMap<M> xMap(const M &m) 
alpar@1317
   579
  {
alpar@1317
   580
    return ConstXMap<M>(m);
alpar@1317
   581
  }
alpar@1317
   582
alpar@2207
   583
  ///Map of y-coordinates of a dim2::Point<>-map
alpar@1317
   584
    
alpar@1317
   585
  ///\ingroup maps
alpar@2214
   586
  ///Map of y-coordinates of a dim2::Point<>-map
alpar@1317
   587
  ///
alpar@1317
   588
  template<class M>
alpar@1317
   589
  class YMap 
alpar@1317
   590
  {
deba@1706
   591
    M& _map;
alpar@1317
   592
  public:
deba@1420
   593
alpar@1317
   594
    typedef typename M::Value::Value Value;
alpar@1317
   595
    typedef typename M::Key Key;
alpar@1317
   596
    ///\e
deba@1706
   597
    YMap(M& map) : _map(map) {}
alpar@1317
   598
    Value operator[](Key k) const {return _map[k].y;}
alpar@1352
   599
    void set(Key k,Value v) {_map.set(k,typename M::Value(_map[k].x,v));}
alpar@1317
   600
  };
alpar@1317
   601
alpar@1317
   602
  ///Returns an \ref YMap class
alpar@1317
   603
alpar@1317
   604
  ///This function just returns an \ref YMap class.
alpar@1317
   605
  ///
alpar@1317
   606
  ///\ingroup maps
alpar@1317
   607
  ///\relates YMap
alpar@1317
   608
  template<class M> 
alpar@1317
   609
  inline YMap<M> yMap(M &m) 
alpar@1317
   610
  {
alpar@1317
   611
    return YMap<M>(m);
alpar@1317
   612
  }
alpar@1317
   613
deba@1420
   614
  template<class M> 
deba@1420
   615
  inline YMap<M> yMap(const M &m) 
deba@1420
   616
  {
deba@1420
   617
    return YMap<M>(m);
deba@1420
   618
  }
deba@1420
   619
alpar@1317
   620
  ///Constant (read only) version of \ref YMap
alpar@1317
   621
alpar@1317
   622
  ///\ingroup maps
alpar@2214
   623
  ///Constant (read only) version of \ref YMap
alpar@1317
   624
  ///
alpar@1317
   625
  template<class M>
alpar@1317
   626
  class ConstYMap 
alpar@1317
   627
  {
deba@1706
   628
    const M& _map;
alpar@1317
   629
  public:
deba@1420
   630
alpar@1317
   631
    typedef typename M::Value::Value Value;
alpar@1317
   632
    typedef typename M::Key Key;
alpar@1317
   633
    ///\e
alpar@1317
   634
    ConstYMap(const M &map) : _map(map) {}
alpar@1317
   635
    Value operator[](Key k) const {return _map[k].y;}
alpar@1317
   636
  };
alpar@1317
   637
    
alpar@1317
   638
  ///Returns a \ref ConstYMap class
alpar@1317
   639
alpar@1317
   640
  ///This function just returns an \ref ConstYMap class.
alpar@1317
   641
  ///
alpar@1317
   642
  ///\ingroup maps
alpar@1317
   643
  ///\relates ConstYMap
alpar@1317
   644
  template<class M> 
alpar@1317
   645
  inline ConstYMap<M> yMap(const M &m) 
alpar@1317
   646
  {
alpar@1317
   647
    return ConstYMap<M>(m);
alpar@1317
   648
  }
alpar@1317
   649
alpar@1317
   650
alpar@2214
   651
    ///\brief Map of the \ref Point::normSquare() "normSquare()"
alpar@2214
   652
    ///of an \ref Point "Point"-map
alpar@2214
   653
    ///
alpar@2214
   654
    ///Map of the \ref Point::normSquare() "normSquare()"
alpar@2214
   655
    ///of an \ref Point "Point"-map
alpar@2214
   656
    ///\ingroup maps
alpar@2214
   657
    ///
alpar@1352
   658
  template<class M>
alpar@1352
   659
  class NormSquareMap 
alpar@1352
   660
  {
deba@1706
   661
    const M& _map;
alpar@1352
   662
  public:
deba@1420
   663
alpar@1352
   664
    typedef typename M::Value::Value Value;
alpar@1352
   665
    typedef typename M::Key Key;
alpar@1352
   666
    ///\e
alpar@1352
   667
    NormSquareMap(const M &map) : _map(map) {}
alpar@1352
   668
    Value operator[](Key k) const {return _map[k].normSquare();}
alpar@1352
   669
  };
alpar@1352
   670
    
alpar@1352
   671
  ///Returns a \ref NormSquareMap class
alpar@1352
   672
alpar@1352
   673
  ///This function just returns an \ref NormSquareMap class.
alpar@1352
   674
  ///
alpar@1352
   675
  ///\ingroup maps
alpar@1352
   676
  ///\relates NormSquareMap
alpar@1352
   677
  template<class M> 
alpar@1352
   678
  inline NormSquareMap<M> normSquareMap(const M &m) 
alpar@1352
   679
  {
alpar@1352
   680
    return NormSquareMap<M>(m);
alpar@1352
   681
  }
alpar@1352
   682
alpar@431
   683
  /// @}
athos@244
   684
alpar@2207
   685
  } //namespce dim2
alpar@2207
   686
  
alpar@921
   687
} //namespace lemon
athos@201
   688
alpar@2207
   689
#endif //LEMON_DIM2_H