-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathpolygon.go
139 lines (120 loc) · 2.87 KB
/
polygon.go
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
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
package GoPolygons
import (
"math"
)
type Polygon struct {
Points []Point
ContainRect Rect
}
func NewPolygon(pts []Point) *Polygon {
left := pts[0].X
right := pts[0].X
top := pts[0].Y
bottom := pts[0].Y
for i := 0; i < len((pts)); i++ {
left = math.Min(left, pts[i].X)
right = math.Max(right, pts[i].X)
top = math.Min(top, pts[i].Y)
bottom = math.Max(bottom, pts[i].Y)
}
return &Polygon{Points: pts, ContainRect: NewRect(left, top, right, bottom)}
}
//Ray method to determine whether the point inside the polygon
func (pgn *Polygon) ptInPolygonRayCasting(pt Point) bool {
px := pt.X
py := pt.Y
flag := false
ptCount := len(pgn.Points)
i := 0
j := ptCount - 1
for i < ptCount {
sx := pgn.Points[i].X
sy := pgn.Points[i].Y
tx := pgn.Points[j].X
ty := pgn.Points[j].Y
// Point coincides with the polygon vertices
if (sx == px && sy == py) || (tx == px && ty == py) {
return true
}
// Determine whether the two end points of the line on both sides of the ray
if (sy < py && ty >= py) || (sy >= py && ty < py) {
// X-ray coordinates on the line with the same Y coordinate of the point
x := sx + (py-sy)*(tx-sx)/(ty-sy)
// Point in polygon edge
if x == px {
return true
}
// Rays pass through the polygon boundary
if x > px {
flag = !flag
}
}
j = i
i++
}
// When the number of rays passing through the polygon boundary is odd, the point in the polygon
return flag
}
func (pgn *Polygon) PtInPolygon(pt Point) bool {
return pgn.ptInPolygonRayCasting(pt)
}
func (pgn *Polygon) ContainPolygon(cpgn Polygon) (ret bool) {
if pgn.ContainRect.IsCross(cpgn.ContainRect) { //Envelope box must intersect
ret = true
PointCount := len(cpgn.Points)
for i := 0; i < PointCount; i++ {
if !pgn.PtInPolygon(cpgn.Points[i]) {
ret = false
break
}
}
}
return ret
}
//Polygons intersect, false disjoint, true intersection
func (pgn *Polygon) IntersectPolygon(cpgn Polygon) bool {
if !pgn.ContainRect.IsCross(cpgn.ContainRect) { //Envelope box must intersect
return false
}
//Positive points included
for i := 0; i < len(cpgn.Points); i++ {
if pgn.PtInPolygon(cpgn.Points[i]) {
return true
}
}
//Anti-contained point
for i := 0; i < len(pgn.Points); i++ {
if cpgn.PtInPolygon(pgn.Points[i]) {
return true
}
}
//Determine whether the line intersect
ptPgnCount := len(pgn.Points)
ptCPgnCount := len(cpgn.Points)
i := 0
j := ptPgnCount - 1
for i < ptPgnCount {
sx := pgn.Points[i].X
sy := pgn.Points[i].Y
tx := pgn.Points[j].X
ty := pgn.Points[j].Y
line := NewLine(sx, sy, tx, ty)
k := 0
l := ptCPgnCount - 1
for k < ptCPgnCount {
csx := cpgn.Points[k].X
csy := cpgn.Points[k].Y
ctx := cpgn.Points[l].X
cty := cpgn.Points[l].Y
cline := NewLine(csx, csy, ctx, cty)
if line.IsLineSegmentCross(cline) {
return true
}
l = k
k++
}
j = i
i++
}
return false
}