TDME2 1.9.121
LineSegment.cpp
Go to the documentation of this file.
2
3#include <tdme/tdme.h>
6#include <tdme/math/Math.h>
7#include <tdme/math/Vector3.h>
8
14
15void LineSegment::computeClosestPointOnLineSegment(const Vector3& p1, const Vector3& q1, const Vector3& p, Vector3& c) {
16 // see https://forum.unity.com/threads/how-do-i-find-the-closest-point-on-a-line.340058/
17 Vector3 lineDirection = q1.clone().sub(p1);
18 float lineLength = lineDirection.computeLength();
19 lineDirection.normalize();
20 Vector3 pSubP1 = p.clone().sub(p1);
21 float t = Vector3::computeDotProduct(pSubP1, lineDirection);
22 if (t < 0.0f) t = 0.0f;
23 if (t > lineLength) t = lineLength;
24 c.set(p1).add(lineDirection.scale(t));
25}
26
27bool LineSegment::doesLineSegmentsCollide(const Vector3& p1, const Vector3& q1, const Vector3& p2, const Vector3& q2, Vector3& p)
28{
29 Vector3 c1;
30 Vector3 c2;
31 computeClosestPointsOnLineSegments(p1, q1, p2, q2, c1, c2);
32 if (c1.sub(c2).computeLength() < Math::EPSILON) {
33 p.set(c2);
34 return true;
35 } else {
36 return false;
37 }
38}
39
40void LineSegment::computeClosestPointsOnLineSegments(const Vector3& p1, const Vector3& q1, const Vector3& p2, const Vector3& q2, Vector3& c1, Vector3& c2)
41{
42 Vector3 d1;
43 Vector3 d2;
44 Vector3 r;
45 d1.set(q1).sub(p1);
46 d2.set(q2).sub(p2);
47 r.set(p1).sub(p2);
48 auto a = Vector3::computeDotProduct(d1, d1);
49 auto e = Vector3::computeDotProduct(d2, d2);
50 auto f = Vector3::computeDotProduct(d2, r);
51 // both line segments degenerate into points?
52 if (a <= Math::EPSILON && e <= Math::EPSILON) {
53 c1 = p1;
54 c2 = p2;
55 return;
56 }
57 // first line segment degenerates into point?
58 float s;
59 float t;
60 if (a <= Math::EPSILON) {
61 s = 0.0f;
62 t = f / e;
63 t = Math::clamp(t, 0.0f, 1.0f);
64 } else {
65 auto c = Vector3::computeDotProduct(d1, r);
66 // second line segment degenerates into point?
67 if (e <= Math::EPSILON) {
68 t = 0.0f;
69 s = Math::clamp(-c / a, 0.0f, 1.0f);
70 } else {
71 // nope, use general case
72 auto b = Vector3::computeDotProduct(d1, d2);
73 auto denom = a * e - b * b;
74 if (denom != 0.0f) {
75 s = Math::clamp((b * f - c * e) / denom, 0.0f, 1.0f);
76 } else {
77 s = 0.0f;
78 }
79 t = (b * s + f) / e;
80 if (t < 0.0f) {
81 t = 0.0f;
82 s = Math::clamp(-c / a, 0.0f, 1.0f);
83 } else if (t > 1.0f) {
84 t = 1.0f;
85 s = Math::clamp((b - c) / a, 0.0f, 1.0f);
86 }
87 }
88 }
89 c1.set(p1).add(d1.scale(s));
90 c2.set(p2).add(d2.scale(t));
91}
92
93bool LineSegment::doesBoundingBoxCollideWithLineSegment(BoundingBox* boundingBox, const Vector3& p, const Vector3& q, Vector3& contactMin, Vector3& contactMax)
94{
95 Vector3 d;
96 Vector3 d1;
97 Vector3 d2;
98 auto tmin = 0.0f;
99 auto tmax = 1.0f;
100 auto minXYZ = boundingBox->getMin().getArray();
101 auto maxXYZ = boundingBox->getMax().getArray();
102 d.set(q).sub(p);
103 auto& directionXYZ = d.getArray();
104 auto& pointXYZ = p.getArray();
105 for (auto i = 0; i < 3; i++) {
106 if (Math::abs(directionXYZ[i]) < Math::EPSILON &&
107 (pointXYZ[i] <= minXYZ[i] || pointXYZ[i] >= maxXYZ[i])) {
108 return false;
109 } else {
110 auto odd = 1.0f / directionXYZ[i];
111 auto t1 = (minXYZ[i] - pointXYZ[i]) * odd;
112 auto t2 = (maxXYZ[i] - pointXYZ[i]) * odd;
113 if (t1 > t2) {
114 auto tmp = t1;
115 t1 = t2;
116 t2 = tmp;
117 }
118 tmin = Math::max(tmin, t1);
119 tmax = Math::min(tmax, t2);
120 if (tmin > tmax)
121 return false;
122
123 }
124 }
125 if (tmin > 1.0) return false;
126 // compute contact points
127 contactMin.set(p).add(d1.set(d).scale(tmin));
128 contactMax.set(p).add(d2.set(d).scale(tmax));
129 // we have a collision
130 return true;
131}
132
133bool LineSegment::doesOrientedBoundingBoxCollideWithLineSegment(OrientedBoundingBox* orientedBoundingBox, const Vector3& p, const Vector3& q, Vector3& contactMin, Vector3& contactMax)
134{
135 Vector3 d;
136 Vector3 d1;
137 Vector3 d2;
138 auto tmin = 0.0f;
139 auto tmax = 1.0f;
140 auto obbAxes = orientedBoundingBox->getAxes();
141 auto obbCenter = orientedBoundingBox->getCenter();
142 auto obbHalfExtension = orientedBoundingBox->getHalfExtension();
143 auto& obbHalfExtensionXYZ = obbHalfExtension.getArray();
144 d.set(q).sub(p);
145 for (auto i = 0; i < 3; i++) {
146 auto directionLengthOnAxis = Vector3::computeDotProduct(d, obbAxes[i]);
147 auto obbExtensionLengthOnAxis = obbHalfExtensionXYZ[i];
148 auto obbCenterLengthOnAxis = Vector3::computeDotProduct(obbCenter, obbAxes[i]);
149 auto pointLengthOnAxis = Vector3::computeDotProduct(p, obbAxes[i]);
150 if (Math::abs(directionLengthOnAxis) < Math::EPSILON &&
151 (pointLengthOnAxis <= obbCenterLengthOnAxis - obbExtensionLengthOnAxis ||
152 pointLengthOnAxis >= obbCenterLengthOnAxis + obbExtensionLengthOnAxis)) {
153 return false;
154 } else {
155 auto odd = 1.0f / directionLengthOnAxis;
156 auto t1 = (obbCenterLengthOnAxis - obbExtensionLengthOnAxis - pointLengthOnAxis) * odd;
157 auto t2 = (obbCenterLengthOnAxis + obbExtensionLengthOnAxis - pointLengthOnAxis) * odd;
158 if (t1 > t2) {
159 auto tmp = t1;
160 t1 = t2;
161 t2 = tmp;
162 }
163 tmin = Math::max(tmin, t1);
164 tmax = Math::min(tmax, t2);
165 if (tmin > tmax)
166 return false;
167
168 }
169 }
170 if (tmin > 1.0) return false;
171 // compute contact points
172 contactMin.set(p).add(d1.set(d).scale(tmin));
173 contactMax.set(p).add(d2.set(d).scale(tmax));
174 // we have a collision
175 return true;
176}
177
178bool LineSegment::doesLineSegmentCollideWithTriangle(const Vector3& p1, const Vector3& p2, const Vector3& p3, const Vector3& r1, const Vector3& r2, Vector3& contact)
179{
180 // see: https://en.wikipedia.org/wiki/M%C3%B6ller%E2%80%93Trumbore_intersection_algorithm
181 auto rayVector = r2.clone().sub(r1).normalize();
182 float a,f,u,v;
183 auto edge1 = p3.clone().sub(p1);
184 auto edge2 = p2.clone().sub(p1);
185 auto h = Vector3::computeCrossProduct(rayVector, edge2);
186 a = Vector3::computeDotProduct(edge1, h);
187 if (a > -Math::EPSILON && a < Math::EPSILON) {
188 // This ray is parallel to this triangle.
189 return false;
190 }
191 f = 1.0/a;
192 auto s = r1 - p1;
193 u = f * Vector3::computeDotProduct(s, h);
194 if (u < 0.0 || u > 1.0) {
195 return false;
196 }
197 auto q = Vector3::computeCrossProduct(s, edge1);
198 v = f * Vector3::computeDotProduct(rayVector, q);
199 if (v < 0.0 || u + v > 1.0) {
200 return false;
201 }
202 // At this stage we can compute t to find out where the intersection point is on the line.
203 float t = f * Vector3::computeDotProduct(edge2, q);
204 if (t > Math::EPSILON) {
205 // ray intersection
206 contact = r1 + rayVector * t;
207 return true;
208 } else {
209 // This means that there is a line intersection but not a ray intersection.
210 return false;
211 }
212}
213
214bool LineSegment::doesLineSegmentCollideWithPlane(const Vector3& n, float d, const Vector3& p1, const Vector3& p2, Vector3& contact) {
215 // see: https://math.stackexchange.com/questions/83990/line-and-plane-intersection-in-3d
216 auto lineDirection = p2.clone().sub(p1);
217 auto lineLength = lineDirection.computeLength();
218 lineDirection.normalize();
219 float nDotP1 = Vector3::computeDotProduct(n, p1);
220 float nDotLineDirection = Vector3::computeDotProduct(n, lineDirection);
221 auto t = ((d - nDotP1) / nDotLineDirection);
222 if (t < 0.0 || t > lineLength) return false;
223 contact.set(p1 + (lineDirection * t));
224 return true;
225}
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
Definition: BoundingBox.h:25
Line segment helper functions.
Definition: LineSegment.h:17
static bool doesLineSegmentCollideWithPlane(const Vector3 &n, float d, const Vector3 &p1, const Vector3 &p2, Vector3 &contact)
Does line segment collides with triangle.
static void computeClosestPointsOnLineSegments(const Vector3 &p1, const Vector3 &q1, const Vector3 &p2, const Vector3 &q2, Vector3 &c1, Vector3 &c2)
Computes closest points c1, c2 on line segment p1->q1, p2->q2 based on an algorithm from "Real-Time C...
Definition: LineSegment.cpp:40
static bool doesBoundingBoxCollideWithLineSegment(BoundingBox *boundingBox, const Vector3 &p, const Vector3 &q, Vector3 &contactMin, Vector3 &contactMax)
Check if segment collides with bounding box based on an algorithm from "Real-Time Collision Detection...
Definition: LineSegment.cpp:93
static bool doesOrientedBoundingBoxCollideWithLineSegment(OrientedBoundingBox *orientedBoundingBox, const Vector3 &p, const Vector3 &q, Vector3 &contactMin, Vector3 &contactMax)
Check if segment collides with oriented bounding box based on an algorithm from "Real-Time Collision ...
static bool doesLineSegmentCollideWithTriangle(const Vector3 &p1, const Vector3 &p2, const Vector3 &p3, const Vector3 &r1, const Vector3 &r2, Vector3 &contact)
Does line segment collides with triangle.
static bool doesLineSegmentsCollide(const Vector3 &p1, const Vector3 &q1, const Vector3 &p2, const Vector3 &q2, Vector3 &p)
Does line segments collide.
Definition: LineSegment.cpp:27
Oriented bounding box physics primitive.
const array< Vector3, 3 > & getAxes() const
Standard math functions.
Definition: Math.h:21
3D vector 3 class
Definition: Vector3.h:22
float computeLength() const
Definition: Vector3.h:202
Vector3 & normalize()
Normalize the vector.
Definition: Vector3.h:288
Vector3 & set(float x, float y, float z)
Set up vector.
Definition: Vector3.h:73
Vector3 clone() const
Clones the vector.
Definition: Vector3.h:372
Vector3 & sub(const Vector3 &v)
Subtracts a vector.
Definition: Vector3.h:325
Vector3 & add(const Vector3 &v)
Adds a vector.
Definition: Vector3.h:301
Vector3 & scale(float scale)
Scale this vector.
Definition: Vector3.h:349
array< float, 3 > & getArray() const
Definition: Vector3.h:171