-
Notifications
You must be signed in to change notification settings - Fork 400
/
Copy pathRayIntersection.cs
103 lines (82 loc) · 3.54 KB
/
RayIntersection.cs
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
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace g3
{
public class RayIntersection
{
private RayIntersection()
{
}
// basic ray-sphere intersection
public static bool Sphere(Vector3f vOrigin, Vector3f vDirection, Vector3f vCenter, float fRadius, out float fRayT)
{
bool bHit = SphereSigned(ref vOrigin, ref vDirection, ref vCenter, fRadius, out fRayT);
fRayT = Math.Abs(fRayT);
return bHit;
}
public static bool SphereSigned(ref Vector3f vOrigin, ref Vector3f vDirection, ref Vector3f vCenter, float fRadius, out float fRayT)
{
fRayT = 0.0f;
Vector3f m = vOrigin - vCenter;
float b = m.Dot(vDirection);
float c = m.Dot(m) - fRadius * fRadius;
// Exit if r’s origin outside s (c > 0) and r pointing away from s (b > 0)
if (c > 0.0f && b > 0.0f)
return false;
float discr = b * b - c;
// A negative discriminant corresponds to ray missing sphere
if (discr < 0.0f)
return false;
// Ray now found to intersect sphere, compute smallest t value of intersection
fRayT = -b - (float)Math.Sqrt(discr);
return true;
}
public static bool SphereSigned(ref Vector3d vOrigin, ref Vector3d vDirection, ref Vector3d vCenter, double fRadius, out double fRayT)
{
fRayT = 0.0;
Vector3d m = vOrigin - vCenter;
double b = m.Dot(ref vDirection);
double c = m.Dot(m) - fRadius * fRadius;
// Exit if r’s origin outside s (c > 0) and r pointing away from s (b > 0)
if (c > 0.0f && b > 0.0f)
return false;
double discr = b * b - c;
// A negative discriminant corresponds to ray missing sphere
if (discr < 0.0)
return false;
// Ray now found to intersect sphere, compute smallest t value of intersection
fRayT = -b - Math.Sqrt(discr);
return true;
}
public static bool InfiniteCylinder(Vector3f vOrigin, Vector3f vDirection, Vector3f vCylOrigin, Vector3f vCylAxis, float fRadius, out float fRayT)
{
bool bHit = InfiniteCylinderSigned(vOrigin, vDirection, vCylOrigin, vCylAxis, fRadius, out fRayT);
fRayT = Math.Abs(fRayT);
return bHit;
}
public static bool InfiniteCylinderSigned(Vector3f vOrigin, Vector3f vDirection, Vector3f vCylOrigin, Vector3f vCylAxis, float fRadius, out float fRayT)
{
// [RMS] ugh this is shit...not even sure it works in general, but works for a ray inside cylinder
fRayT = 0.0f;
Vector3f AB = vCylAxis;
Vector3f AO = (vOrigin - vCylOrigin);
if (AO.DistanceSquared(AO.Dot(AB) * AB) > fRadius * fRadius)
return false;
Vector3f AOxAB = AO.Cross(AB);
Vector3f VxAB = vDirection.Cross(AB);
float ab2 = AB.Dot(AB);
float a = VxAB.Dot(VxAB);
float b = 2 * VxAB.Dot(AOxAB);
float c = AOxAB.Dot(AOxAB) - (fRadius * fRadius * ab2);
double discrim = b * b - 4 * a * c;
if (discrim <= 0)
return false;
discrim = Math.Sqrt(discrim);
fRayT = (-b - (float)discrim) / (2 * a);
//float t1 = (-b + (float)discrim) / (2 * a);
return true;
}
}
}