2024-09-24 07:11:35 +00:00
|
|
|
using System.Numerics;
|
|
|
|
|
|
|
|
namespace KeepersCompound.Lightmapper;
|
|
|
|
|
2024-10-27 07:53:58 +00:00
|
|
|
public static class Utils
|
|
|
|
{
|
|
|
|
// Expects Hue and Saturation are 0-1, Brightness 0-255
|
|
|
|
// https://en.wikipedia.org/wiki/HSL_and_HSV#HSV_to_RGB
|
2024-12-09 18:25:33 +00:00
|
|
|
// This is actually not accurate to *real* hsb to rgb. But I'm replicating the inaccuracies of Dark :(
|
|
|
|
// See issue#1
|
2024-10-27 07:53:58 +00:00
|
|
|
public static Vector3 HsbToRgb(float hue, float saturation, float brightness)
|
|
|
|
{
|
2024-12-09 18:25:33 +00:00
|
|
|
hue *= 3;
|
|
|
|
var color = hue switch
|
2024-10-27 07:53:58 +00:00
|
|
|
{
|
2024-12-09 18:25:33 +00:00
|
|
|
< 1 => new Vector3(1f - hue, hue, 0),
|
2024-12-26 21:15:53 +00:00
|
|
|
< 2 => new Vector3(0, 2f - hue, hue - 1f),
|
|
|
|
_ => new Vector3(hue - 2f, 0, 3f - hue),
|
2024-10-27 07:53:58 +00:00
|
|
|
};
|
2024-12-09 18:25:33 +00:00
|
|
|
|
|
|
|
color *= saturation;
|
|
|
|
color += Vector3.One * (1.0f - saturation);
|
|
|
|
color *= brightness;
|
|
|
|
|
|
|
|
return color;
|
2024-10-27 07:53:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-24 07:11:35 +00:00
|
|
|
public static class MathUtils
|
|
|
|
{
|
2024-09-25 16:39:44 +00:00
|
|
|
public const float Epsilon = 0.001f;
|
|
|
|
|
2024-09-24 07:55:05 +00:00
|
|
|
public readonly struct Aabb
|
|
|
|
{
|
|
|
|
public readonly Vector3 Min;
|
|
|
|
public readonly Vector3 Max;
|
|
|
|
|
|
|
|
public Aabb(Vector3[] points)
|
|
|
|
{
|
|
|
|
Min = new Vector3(float.MaxValue, float.MaxValue, float.MaxValue);
|
|
|
|
Max = new Vector3(float.MinValue, float.MinValue, float.MinValue);
|
|
|
|
|
|
|
|
foreach (var p in points)
|
|
|
|
{
|
|
|
|
Min = Vector3.Min(Min, p);
|
|
|
|
Max = Vector3.Max(Max, p);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public readonly struct Sphere
|
|
|
|
{
|
|
|
|
public readonly Vector3 Position;
|
|
|
|
public readonly float Radius;
|
|
|
|
|
|
|
|
public Sphere(Vector3 position, float radius)
|
|
|
|
{
|
|
|
|
Position = position;
|
|
|
|
Radius = radius;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static Vector3 ClosestPoint(Aabb aabb, Vector3 point)
|
|
|
|
{
|
|
|
|
return Vector3.Min(aabb.Max, Vector3.Max(aabb.Min, point));
|
|
|
|
}
|
|
|
|
|
|
|
|
public static bool Intersects(Sphere sphere, Aabb aabb)
|
|
|
|
{
|
|
|
|
var closestPoint = ClosestPoint(aabb, sphere.Position);
|
|
|
|
var d2 = (sphere.Position - closestPoint).LengthSquared();
|
|
|
|
var r2 = sphere.Radius * sphere.Radius;
|
|
|
|
return d2 < r2;
|
|
|
|
}
|
|
|
|
|
2024-10-27 08:14:47 +00:00
|
|
|
// Should automagically handle max float radii
|
2024-10-06 09:55:56 +00:00
|
|
|
public static bool Intersects(Sphere sphere, Sphere other)
|
|
|
|
{
|
|
|
|
var rsum = sphere.Radius + other.Radius;
|
|
|
|
return (sphere.Position - other.Position).Length() <= rsum;
|
|
|
|
}
|
|
|
|
|
2025-01-04 23:08:18 +00:00
|
|
|
public static bool Intersects(Aabb aabb, Vector3 point)
|
|
|
|
{
|
|
|
|
var min = aabb.Min;
|
|
|
|
var max = aabb.Max;
|
|
|
|
return point.X >= min.X && point.Y >= min.Y && point.Z >= min.Z &&
|
|
|
|
point.X <= max.X && point.Y <= max.Y && point.Z <= max.Z;
|
|
|
|
}
|
|
|
|
|
2024-09-24 07:11:35 +00:00
|
|
|
public static float DistanceFromPlane(Plane plane, Vector3 point)
|
|
|
|
{
|
2025-01-05 14:49:02 +00:00
|
|
|
return (Vector3.Dot(plane.Normal, point) + plane.D) / plane.Normal.Length();
|
|
|
|
}
|
|
|
|
|
|
|
|
public static bool IsCoplanar(Plane p0, Plane p1)
|
|
|
|
{
|
|
|
|
var m = p0.D / p1.D;
|
|
|
|
var n0 = p0.Normal;
|
|
|
|
var n1 = p1.Normal * m;
|
|
|
|
return Math.Abs(n0.X - n1.X) < Epsilon && Math.Abs(n0.Y - n1.Y) < Epsilon && Math.Abs(n0.Z - n1.Z) < Epsilon;
|
2024-09-24 07:11:35 +00:00
|
|
|
}
|
2024-09-24 20:13:02 +00:00
|
|
|
|
2024-09-25 17:25:54 +00:00
|
|
|
public record PlanePointMapper
|
|
|
|
{
|
2024-12-09 12:04:32 +00:00
|
|
|
public Vector3 Normal { get; }
|
2024-09-25 17:25:54 +00:00
|
|
|
Vector3 _origin;
|
|
|
|
Vector3 _xAxis;
|
|
|
|
Vector3 _yAxis;
|
|
|
|
|
|
|
|
public PlanePointMapper(Vector3 normal, Vector3 p0, Vector3 p1)
|
|
|
|
{
|
2024-12-09 12:04:32 +00:00
|
|
|
Normal = normal;
|
2024-09-25 17:25:54 +00:00
|
|
|
_origin = p0;
|
|
|
|
_xAxis = p1 - _origin;
|
|
|
|
_yAxis = Vector3.Cross(normal, _xAxis);
|
|
|
|
|
|
|
|
_xAxis = Vector3.Normalize(_xAxis);
|
|
|
|
_yAxis = Vector3.Normalize(_yAxis);
|
|
|
|
}
|
|
|
|
|
|
|
|
public Vector2 MapTo2d(Vector3 point)
|
|
|
|
{
|
|
|
|
var offset = point - _origin;
|
|
|
|
var x = Vector3.Dot(offset, _xAxis);
|
|
|
|
var y = Vector3.Dot(offset, _yAxis);
|
|
|
|
return new Vector2(x, y);
|
|
|
|
}
|
|
|
|
|
|
|
|
public Vector2[] MapTo2d(Vector3[] points)
|
|
|
|
{
|
|
|
|
var points2d = new Vector2[points.Length];
|
|
|
|
for (var i = 0; i < points.Length; i++)
|
|
|
|
{
|
|
|
|
points2d[i] = MapTo2d(points[i]);
|
|
|
|
}
|
|
|
|
return points2d;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Vector3 MapTo3d(Vector2 point)
|
|
|
|
{
|
|
|
|
return _origin + point.X * _xAxis + point.Y * _yAxis;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Vector3[] MapTo3d(Vector2[] points)
|
|
|
|
{
|
|
|
|
var points3d = new Vector3[points.Length];
|
|
|
|
for (var i = 0; i < points.Length; i++)
|
|
|
|
{
|
|
|
|
points3d[i] = MapTo3d(points[i]);
|
|
|
|
}
|
|
|
|
return points3d;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static Vector2 ClipPointToPoly2d(Vector2 point, Vector2[] vertices)
|
|
|
|
{
|
|
|
|
var vertexCount = vertices.Length;
|
|
|
|
for (var i = 0; i < vertexCount; i++)
|
2024-09-24 20:13:02 +00:00
|
|
|
{
|
2024-09-25 17:25:54 +00:00
|
|
|
var a = vertices[i];
|
|
|
|
var b = vertices[(i + 1) % vertexCount];
|
2024-09-24 20:13:02 +00:00
|
|
|
var segment = b - a;
|
2024-09-25 17:25:54 +00:00
|
|
|
var offset = point - a;
|
2024-09-24 20:13:02 +00:00
|
|
|
var norm = Vector2.Normalize(new Vector2(-segment.Y, segment.X));
|
|
|
|
var side = Vector2.Dot(norm, offset);
|
2024-09-25 16:39:44 +00:00
|
|
|
if (side >= -Epsilon)
|
2024-09-24 20:13:02 +00:00
|
|
|
{
|
2024-09-25 16:43:31 +00:00
|
|
|
// We apply epsilon so that we push slightly into the poly. If we only
|
|
|
|
// push to the edge then Embree sometimes misses casts. The reason
|
|
|
|
// it's 2 epsilon is so Side == -Epsilon still gets pushed in properly
|
2024-09-25 17:25:54 +00:00
|
|
|
point -= norm * (side + 2 * Epsilon);
|
2024-09-24 20:13:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-25 17:25:54 +00:00
|
|
|
return point;
|
2024-09-24 20:13:02 +00:00
|
|
|
}
|
2024-09-24 07:11:35 +00:00
|
|
|
}
|