-
Notifications
You must be signed in to change notification settings - Fork 27
/
Copy pathcollide_polygon.go
215 lines (172 loc) · 4.91 KB
/
collide_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
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
package box2d
import (
"math"
)
// Find the max separation between poly1 and poly2 using edge normals from poly1.
func FindMaxSeparation(edgeIndex *int, poly1 *PolygonShape, xf1 Transform, poly2 *PolygonShape, xf2 Transform) float64 {
count1 := poly1.Count
count2 := poly2.Count
n1s := poly1.Normals
v1s := poly1.Vertices
v2s := poly2.Vertices
xf := TransformMulT(xf2, xf1)
bestIndex := 0
maxSeparation := -math.MaxFloat64
for i := 0; i < count1; i++ {
// Get poly1 normal in frame2.
n := RotPointMul(xf.Q, n1s[i])
v1 := TransformPointMul(xf, v1s[i])
// Find deepest point for normal i.
si := math.MaxFloat64
for j := 0; j < count2; j++ {
sij := PointDot(n, PointSub(v2s[j], v1))
if sij < si {
si = sij
}
}
if si > maxSeparation {
maxSeparation = si
bestIndex = i
}
}
*edgeIndex = bestIndex
return maxSeparation
}
func FindIncidentEdge(c []ClipVertex, poly1 *PolygonShape, xf1 Transform, edge1 int, poly2 *PolygonShape, xf2 Transform) {
normals1 := poly1.Normals
count2 := poly2.Count
vertices2 := poly2.Vertices
normals2 := poly2.Normals
// Get the normal of the reference edge in poly2's frame.
normal1 := RotPointMulT(xf2.Q, RotPointMul(xf1.Q, normals1[edge1]))
// Find the incident edge on poly2.
index := 0
minDot := math.MaxFloat64
for i := 0; i < count2; i++ {
dot := PointDot(normal1, normals2[i])
if dot < minDot {
minDot = dot
index = i
}
}
// Build the clip vertices for the incident edge.
i1 := index
i2 := 0
if i1+1 < count2 {
i2 = i1 + 1
}
c[0].V = TransformPointMul(xf2, vertices2[i1])
c[0].Id.IndexA = uint8(edge1)
c[0].Id.IndexB = uint8(i1)
c[0].Id.TypeA = ContactFeatureTypeFace
c[0].Id.TypeB = ContactFeatureTypeVertex
c[1].V = TransformPointMul(xf2, vertices2[i2])
c[1].Id.IndexA = uint8(edge1)
c[1].Id.IndexB = uint8(i2)
c[1].Id.TypeA = ContactFeatureTypeFace
c[1].Id.TypeB = ContactFeatureTypeVertex
}
// Find edge normal of max separation on A - return if separating axis is found
// Find edge normal of max separation on B - return if separation axis is found
// Choose reference edge as min(minA, minB)
// Find incident edge
// Clip
// The normal points from 1 to 2
func CollidePolygons(manifold *Manifold, polyA *PolygonShape, xfA Transform, polyB *PolygonShape, xfB Transform) {
manifold.PointCount = 0
totalRadius := polyA.Radius + polyB.Radius
edgeA := 0
separationA := FindMaxSeparation(&edgeA, polyA, xfA, polyB, xfB)
if separationA > totalRadius {
return
}
edgeB := 0
separationB := FindMaxSeparation(&edgeB, polyB, xfB, polyA, xfA)
if separationB > totalRadius {
return
}
var poly1 *PolygonShape // reference polygon
var poly2 *PolygonShape // incident polygon
xf1 := Transform{}
xf2 := Transform{}
edge1 := 0 // reference edge
var flip uint8
k_tol := 0.1 * _linearSlop
if separationB > separationA+k_tol {
poly1 = polyB
poly2 = polyA
xf1 = xfB
xf2 = xfA
edge1 = edgeB
manifold.Type = ManifoldTypeFaceB
flip = 1
} else {
poly1 = polyA
poly2 = polyB
xf1 = xfA
xf2 = xfB
edge1 = edgeA
manifold.Type = ManifoldTypeFaceA
flip = 0
}
incidentEdge := make([]ClipVertex, 2)
FindIncidentEdge(incidentEdge, poly1, xf1, edge1, poly2, xf2)
count1 := poly1.Count
vertices1 := poly1.Vertices
iv1 := edge1
iv2 := 0
if edge1+1 < count1 {
iv2 = edge1 + 1
}
v11 := vertices1[iv1]
v12 := vertices1[iv2]
localTangent := PointSub(v12, v11)
localTangent.Normalize()
localNormal := PointCrossVectorScalar(localTangent, 1.0)
planePoint := PointMulScalar(0.5, PointAdd(v11, v12))
tangent := RotPointMul(xf1.Q, localTangent)
normal := PointCrossVectorScalar(tangent, 1.0)
v11 = TransformPointMul(xf1, v11)
v12 = TransformPointMul(xf1, v12)
// Face offset.
frontOffset := PointDot(normal, v11)
// Side offsets, extended by polytope skin thickness.
sideOffset1 := -PointDot(tangent, v11) + totalRadius
sideOffset2 := PointDot(tangent, v12) + totalRadius
// Clip incident edge against extruded edge1 side edges.
clipPoints1 := make([]ClipVertex, 2)
clipPoints2 := make([]ClipVertex, 2)
np := 0
// Clip to box side 1
np = ClipSegmentToLine(clipPoints1, incidentEdge, tangent.OperatorNegate(), sideOffset1, iv1)
if np < 2 {
return
}
// Clip to negative box side 1
np = ClipSegmentToLine(clipPoints2, clipPoints1, tangent, sideOffset2, iv2)
if np < 2 {
return
}
// Now clipPoints2 contains the clipped points.
manifold.LocalNormal = localNormal
manifold.LocalPoint = planePoint
pointCount := 0
for i := 0; i < len(clipPoints2); i++ {
separation := PointDot(normal, clipPoints2[i].V) - frontOffset
if separation <= totalRadius {
cp := &manifold.Points[pointCount]
cp.LocalPoint = TransformPointMulT(xf2, clipPoints2[i].V)
cp.Id = clipPoints2[i].Id
if flip != 0 {
// Swap features
cf := cp.Id
cp.Id.IndexA = cf.IndexB
cp.Id.IndexB = cf.IndexA
cp.Id.TypeA = cf.TypeB
cp.Id.TypeB = cf.TypeA
}
pointCount++
}
}
manifold.PointCount = pointCount
}