-
Notifications
You must be signed in to change notification settings - Fork 47
/
Copy pathcontactSet.go
executable file
·121 lines (87 loc) · 3.15 KB
/
contactSet.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
package resolv
import (
"math"
)
// Intersection represents a single point of contact against a line or surface.
type Intersection struct {
Point Vector // The point of contact.
Normal Vector // The normal of the surface contacted.
}
// IntersectionSet represents a set of intersections between the calling object and one other intersecting Shape.
// A Shape's intersection test may iterate through multiple IntersectionSets - one for each pair of intersecting objects.
type IntersectionSet struct {
Intersections []Intersection // Slice of points indicating contact between the two Shapes.
Center Vector // Center of the Contact set; this is the average of all Points contained within all contacts in the IntersectionSet.
MTV Vector // Minimum Translation Vector; this is the vector to move a Shape on to move it to contact with the other, intersecting / contacting Shape.
OtherShape IShape // The other shape involved in the contact.
}
func newIntersectionSet() IntersectionSet {
return IntersectionSet{}
}
// LeftmostPoint returns the left-most point out of the IntersectionSet's Points slice.
// If the IntersectionSet is empty, this returns a zero Vector.
func (is IntersectionSet) LeftmostPoint() Vector {
var left Vector
set := false
for _, contact := range is.Intersections {
if !set || contact.Point.X < left.X {
left = contact.Point
set = true
}
}
return left
}
// RightmostPoint returns the right-most point out of the IntersectionSet's Points slice.
// If the IntersectionSet is empty, this returns a zero Vector.
func (is IntersectionSet) RightmostPoint() Vector {
var right Vector
set := false
for _, contact := range is.Intersections {
if !set || contact.Point.X > right.X {
right = contact.Point
set = true
}
}
return right
}
// TopmostPoint returns the top-most point out of the IntersectionSet's Points slice. I
// f the IntersectionSet is empty, this returns a zero Vector.
func (is IntersectionSet) TopmostPoint() Vector {
var top Vector
set := false
for _, contact := range is.Intersections {
if !set || contact.Point.Y < top.Y {
top = contact.Point
set = true
}
}
return top
}
// BottommostPoint returns the bottom-most point out of the IntersectionSet's Points slice.
// If the IntersectionSet is empty, this returns a zero Vector.
func (is IntersectionSet) BottommostPoint() Vector {
var bottom Vector
set := false
for _, contact := range is.Intersections {
if !set || contact.Point.Y > bottom.Y {
bottom = contact.Point
set = true
}
}
return bottom
}
// IsEmpty returns if the IntersectionSet is empty (and so contains no points of iontersection). This should never actually be true.
func (is IntersectionSet) IsEmpty() bool {
return len(is.Intersections) == 0
}
// Distance returns the distance between all of the intersection points when projected against an axis.
func (is IntersectionSet) Distance(alongAxis Vector) float64 {
alongAxis = alongAxis.Unit()
top, bottom := math.MaxFloat64, -math.MaxFloat64
for _, c := range is.Intersections {
d := alongAxis.Dot(c.Point)
top = min(top, d)
bottom = max(bottom, d)
}
return bottom - top
}