Welcome to mirror list, hosted at ThFree Co, Russian Federation.

rect_intersect.hpp « geometry - github.com/mapsme/omim.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: 86cb903335f44339a6709734a117b50adcca2993 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
#pragma once

#include "rect2d.hpp"
#include "point2d.hpp"

namespace m2
{
  namespace detail
  {
    // bit masks for code kinds
    static const int LEFT = 1;
    static const int RIGHT = 2;
    static const int BOT = 4;
    static const int TOP = 8;

    // return accumulated bit mask for point-rect interaction
    template <class T>
    int vcode(m2::Rect<T> const & r, m2::Point<T> const & p)
    {
      return ((p.x < r.minX() ? LEFT : 0)  +
              (p.x > r.maxX() ? RIGHT : 0) +
              (p.y < r.minY() ? BOT : 0)   +
              (p.y > r.maxY() ? TOP : 0));
    }
  }

  template <class T>
  bool Intersect(m2::Rect<T> const & r, m2::Point<T> & p1, m2::Point<T> & p2)
  {
    int code[2] = { detail::vcode(r, p1), detail::vcode(r, p2) };

    // do while one of the point is out of rect
    while (code[0] || code[1])
    {
      if (code[0] & code[1])
      {
        // both point area on the one side of rect
        return false;
      }

      // choose point with non-zero code
      m2::Point<T> * pp;
      int i;
      if (code[0])
      {
        i = 0;
        pp = &p1;
      }
      else
      {
        i = 1;
        pp = &p2;
      }

      // added points compare to avoid NAN numbers
      if (code[i] & detail::LEFT)
      {
        if (p1 == p2) return false;
        pp->y += (p1.y - p2.y) * (r.minX() - pp->x) / (p1.x - p2.x);
        pp->x = r.minX();
      }
      else if (code[i] & detail::RIGHT)
      {
        if (p1 == p2) return false;
        pp->y += (p1.y - p2.y) * (r.maxX() - pp->x) / (p1.x - p2.x);
        pp->x = r.maxX();
      }

      if (code[i] & detail::BOT)
      {
        if (p1 == p2) return false;
        pp->x += (p1.x - p2.x) * (r.minY() - pp->y) / (p1.y - p2.y);
        pp->y = r.minY();
      }
      else if (code[i] & detail::TOP)
      {
        if (p1 == p2) return false;
        pp->x += (p1.x - p2.x) * (r.maxY() - pp->y) / (p1.y - p2.y);
        pp->y = r.maxY();
      }

      // update code with new point
      code[i] = detail::vcode(r, *pp);
    }

    // both codes are equal to zero => points area inside rect
    return true;
  }
}