e14b092aa7
This isn't 100% finished yet. I intend to add some better API for reading specific attributes, as well as write support (of course!)
446 lines
16 KiB
C#
446 lines
16 KiB
C#
using System;
|
|
using RobloxFiles.Enums;
|
|
|
|
namespace RobloxFiles.DataTypes
|
|
{
|
|
public class CFrame
|
|
{
|
|
private float m11 = 1, m12 = 0, m13 = 0, m14 = 0;
|
|
private float m21 = 0, m22 = 1, m23 = 0, m24 = 0;
|
|
private float m31 = 0, m32 = 0, m33 = 1, m34 = 0;
|
|
private const float m41 = 0, m42 = 0, m43 = 0, m44 = 1;
|
|
|
|
public float X => m14;
|
|
public float Y => m24;
|
|
public float Z => m34;
|
|
|
|
public Vector3 Position
|
|
{
|
|
get
|
|
{
|
|
return new Vector3(X, Y, Z);
|
|
}
|
|
set
|
|
{
|
|
m14 = value.X;
|
|
m24 = value.Y;
|
|
m34 = value.Z;
|
|
}
|
|
}
|
|
|
|
public Vector3 RightVector => new Vector3( m11, m12, m13);
|
|
public Vector3 UpVector => new Vector3( m21, m22, m23);
|
|
public Vector3 LookVector => new Vector3(-m31, -m32, -m33);
|
|
|
|
public CFrame()
|
|
{
|
|
m14 = 0;
|
|
m24 = 0;
|
|
m34 = 0;
|
|
}
|
|
|
|
public CFrame(Vector3 pos)
|
|
{
|
|
m14 = pos.X;
|
|
m24 = pos.Y;
|
|
m34 = pos.Z;
|
|
}
|
|
|
|
public CFrame(float nx = 0, float ny = 0, float nz = 0)
|
|
{
|
|
m14 = nx;
|
|
m24 = ny;
|
|
m34 = nz;
|
|
}
|
|
|
|
public CFrame(Vector3 eye, Vector3 look)
|
|
{
|
|
Vector3 zAxis = (eye - look).Unit;
|
|
Vector3 xAxis = Vector3.Up.Cross(zAxis);
|
|
Vector3 yAxis = zAxis.Cross(xAxis);
|
|
|
|
if (xAxis.Magnitude == 0)
|
|
{
|
|
if (zAxis.Y < 0)
|
|
{
|
|
xAxis = new Vector3(0, 0, -1);
|
|
yAxis = new Vector3(1, 0, 0);
|
|
zAxis = new Vector3(0, -1, 0);
|
|
}
|
|
else
|
|
{
|
|
xAxis = new Vector3(0, 0, 1);
|
|
yAxis = new Vector3(1, 0, 0);
|
|
zAxis = new Vector3(0, 1, 0);
|
|
}
|
|
}
|
|
|
|
m11 = xAxis.X; m12 = yAxis.X; m13 = zAxis.X; m14 = eye.X;
|
|
m21 = xAxis.Y; m22 = yAxis.Y; m23 = zAxis.Y; m24 = eye.Y;
|
|
m31 = xAxis.Z; m32 = yAxis.Z; m33 = zAxis.Z; m34 = eye.Z;
|
|
}
|
|
|
|
public CFrame(float nx, float ny, float nz, float i, float j, float k, float w)
|
|
{
|
|
float ii = i * i;
|
|
float jj = j * j;
|
|
float kk = k * k;
|
|
|
|
m14 = nx;
|
|
m24 = ny;
|
|
m34 = nz;
|
|
|
|
m11 = 1 - 2 * jj - 2 * kk;
|
|
m12 = 2 * (i * j - k * w);
|
|
m13 = 2 * (i * k + j * w);
|
|
|
|
m21 = 2 * (i * j + k * w);
|
|
m22 = 1 - 2 * ii - 2 * kk;
|
|
m23 = 2 * (j * k - i * w);
|
|
|
|
m31 = 2 * (i * k - j * w);
|
|
m32 = 2 * (j * k + i * w);
|
|
m33 = 1 - 2 * ii - 2 * jj;
|
|
}
|
|
|
|
public CFrame(float n14, float n24, float n34, float n11, float n12, float n13, float n21, float n22, float n23, float n31, float n32, float n33)
|
|
{
|
|
m14 = n14; m24 = n24; m34 = n34;
|
|
m11 = n11; m12 = n12; m13 = n13;
|
|
m21 = n21; m22 = n22; m23 = n23;
|
|
m31 = n31; m32 = n32; m33 = n33;
|
|
}
|
|
|
|
public CFrame(params float[] comp)
|
|
{
|
|
if (comp.Length < 12)
|
|
throw new Exception("There should be 12 floats provided to construct CFrame with an array of floats");
|
|
|
|
m14 = comp[0]; m24 = comp[1]; m34 = comp[2];
|
|
m11 = comp[3]; m12 = comp[4]; m13 = comp[5];
|
|
m21 = comp[6]; m22 = comp[7]; m23 = comp[8];
|
|
m31 = comp[9]; m32 = comp[10]; m33 = comp[11];
|
|
}
|
|
|
|
private void initFromMatrix(Vector3 pos, Vector3 vX, Vector3 vY, Vector3 vZ = null)
|
|
{
|
|
if (vZ == null)
|
|
vZ = vX.Cross(vY);
|
|
|
|
m14 = pos.X; m24 = pos.Y; m34 = pos.Z;
|
|
m11 = vX.X; m12 = vX.Y; m13 = vX.Z;
|
|
m21 = vY.X; m22 = vY.Y; m23 = vY.Z;
|
|
m31 = vZ.X; m32 = vZ.Y; m33 = vZ.Z;
|
|
}
|
|
|
|
public CFrame(Vector3 pos, Vector3 vX, Vector3 vY, Vector3 vZ = null)
|
|
{
|
|
initFromMatrix(pos, vX, vY, vZ);
|
|
}
|
|
|
|
internal CFrame(Attribute attr)
|
|
{
|
|
Vector3 pos = new Vector3(attr);
|
|
byte rawOrientId = attr.readByte();
|
|
|
|
if (rawOrientId > 0)
|
|
{
|
|
// Make sure this value is in a safe range.
|
|
int orientId = (rawOrientId - 1) % 36;
|
|
|
|
NormalId xColumn = (NormalId)(orientId / 6);
|
|
Vector3 vX = Vector3.FromNormalId(xColumn);
|
|
|
|
NormalId yColumn = (NormalId)(orientId % 6);
|
|
Vector3 vY = Vector3.FromNormalId(yColumn);
|
|
|
|
initFromMatrix(pos, vX, vY);
|
|
}
|
|
else
|
|
{
|
|
Vector3 vX = new Vector3(attr),
|
|
vY = new Vector3(attr),
|
|
vZ = new Vector3(attr);
|
|
|
|
initFromMatrix(pos, vX, vY, vZ);
|
|
}
|
|
}
|
|
|
|
public static CFrame operator +(CFrame a, Vector3 b)
|
|
{
|
|
float[] ac = a.GetComponents();
|
|
|
|
float x = ac[0], y = ac[1], z = ac[2],
|
|
m11 = ac[3], m12 = ac[4], m13 = ac[5],
|
|
m21 = ac[6], m22 = ac[7], m23 = ac[8],
|
|
m31 = ac[9], m32 = ac[10], m33 = ac[11];
|
|
|
|
return new CFrame(x + b.X, y + b.Y, z + b.Z, m11, m12, m13, m21, m22, m23, m31, m32, m33);
|
|
}
|
|
|
|
public static CFrame operator -(CFrame a, Vector3 b)
|
|
{
|
|
float[] ac = a.GetComponents();
|
|
|
|
float x = ac[0], y = ac[1], z = ac[2],
|
|
m11 = ac[3], m12 = ac[4], m13 = ac[5],
|
|
m21 = ac[6], m22 = ac[7], m23 = ac[8],
|
|
m31 = ac[9], m32 = ac[10], m33 = ac[11];
|
|
|
|
return new CFrame(x - b.X, y - b.Y, z - b.Z, m11, m12, m13, m21, m22, m23, m31, m32, m33);
|
|
}
|
|
|
|
public static Vector3 operator *(CFrame a, Vector3 b)
|
|
{
|
|
float[] ac = a.GetComponents();
|
|
|
|
float x = ac[0], y = ac[1], z = ac[2],
|
|
m11 = ac[3], m12 = ac[4], m13 = ac[5],
|
|
m21 = ac[6], m22 = ac[7], m23 = ac[8],
|
|
m31 = ac[9], m32 = ac[10], m33 = ac[11];
|
|
|
|
Vector3 right = new Vector3(m11, m21, m31);
|
|
Vector3 up = new Vector3(m12, m22, m32);
|
|
Vector3 back = new Vector3(m13, m23, m33);
|
|
return a.Position + b.X * right + b.Y * up + b.Z * back;
|
|
}
|
|
|
|
public static CFrame operator *(CFrame a, CFrame b)
|
|
{
|
|
float[] ac = a.GetComponents();
|
|
float[] bc = b.GetComponents();
|
|
|
|
float a14 = ac[0], a24 = ac[1], a34 = ac[2],
|
|
a11 = ac[3], a12 = ac[4], a13 = ac[5],
|
|
a21 = ac[6], a22 = ac[7], a23 = ac[8],
|
|
a31 = ac[9], a32 = ac[10], a33 = ac[11];
|
|
|
|
float b14 = bc[0], b24 = bc[1], b34 = bc[2],
|
|
b11 = bc[3], b12 = bc[4], b13 = bc[5],
|
|
b21 = bc[6], b22 = bc[7], b23 = bc[8],
|
|
b31 = bc[9], b32 = bc[10], b33 = bc[11];
|
|
|
|
float n11 = a11 * b11 + a12 * b21 + a13 * b31 + a14 * m41;
|
|
float n12 = a11 * b12 + a12 * b22 + a13 * b32 + a14 * m42;
|
|
float n13 = a11 * b13 + a12 * b23 + a13 * b33 + a14 * m43;
|
|
float n14 = a11 * b14 + a12 * b24 + a13 * b34 + a14 * m44;
|
|
|
|
float n21 = a21 * b11 + a22 * b21 + a23 * b31 + a24 * m41;
|
|
float n22 = a21 * b12 + a22 * b22 + a23 * b32 + a24 * m42;
|
|
float n23 = a21 * b13 + a22 * b23 + a23 * b33 + a24 * m43;
|
|
float n24 = a21 * b14 + a22 * b24 + a23 * b34 + a24 * m44;
|
|
|
|
float n31 = a31 * b11 + a32 * b21 + a33 * b31 + a34 * m41;
|
|
float n32 = a31 * b12 + a32 * b22 + a33 * b32 + a34 * m42;
|
|
float n33 = a31 * b13 + a32 * b23 + a33 * b33 + a34 * m43;
|
|
float n34 = a31 * b14 + a32 * b24 + a33 * b34 + a34 * m44;
|
|
|
|
float n41 = m41 * b11 + m42 * b21 + m43 * b31 + m44 * m41;
|
|
float n42 = m41 * b12 + m42 * b22 + m43 * b32 + m44 * m42;
|
|
float n43 = m41 * b13 + m42 * b23 + m43 * b33 + m44 * m43;
|
|
float n44 = m41 * b14 + m42 * b24 + m43 * b34 + m44 * m44;
|
|
|
|
return new CFrame(n14, n24, n34, n11, n12, n13, n21, n22, n23, n31, n32, n33);
|
|
}
|
|
|
|
public override string ToString()
|
|
{
|
|
return string.Join(", ", GetComponents());
|
|
}
|
|
|
|
private static Vector3 VectorAxisAngle(Vector3 vec, Vector3 axis, float theta)
|
|
{
|
|
Vector3 unit = vec.Unit;
|
|
|
|
float cosAng = (float)Math.Cos(theta);
|
|
float sinAng = (float)Math.Sin(theta);
|
|
|
|
return axis * cosAng + axis.Dot(unit) * unit *
|
|
(1 - cosAng) + unit.Cross(axis) * sinAng;
|
|
}
|
|
|
|
public CFrame Inverse()
|
|
{
|
|
float[] ac = GetComponents();
|
|
|
|
float a14 = ac[0], a24 = ac[1], a34 = ac[2],
|
|
a11 = ac[3], a12 = ac[4], a13 = ac[5],
|
|
a21 = ac[6], a22 = ac[7], a23 = ac[8],
|
|
a31 = ac[9], a32 = ac[10], a33 = ac[11];
|
|
|
|
float det = ( a11 * a22 * a33 * m44 + a11 * a23 * a34 * m42 + a11 * a24 * a32 * m43
|
|
+ a12 * a21 * a34 * m43 + a12 * a23 * a31 * m44 + a12 * a24 * a33 * m41
|
|
+ a13 * a21 * a32 * m44 + a13 * a22 * a34 * m41 + a13 * a24 * a31 * m42
|
|
+ a14 * a21 * a33 * m42 + a14 * a22 * a31 * m43 + a14 * a23 * a32 * m41
|
|
- a11 * a22 * a34 * m43 - a11 * a23 * a32 * m44 - a11 * a24 * a33 * m42
|
|
- a12 * a21 * a33 * m44 - a12 * a23 * a34 * m41 - a12 * a24 * a31 * m43
|
|
- a13 * a21 * a34 * m42 - a13 * a22 * a31 * m44 - a13 * a24 * a32 * m41
|
|
- a14 * a21 * a32 * m43 - a14 * a22 * a33 * m41 - a14 * a23 * a31 * m42 );
|
|
|
|
if (det == 0)
|
|
return this;
|
|
|
|
float b11 = (a22 * a33 * m44 + a23 * a34 * m42 + a24 * a32 * m43 - a22 * a34 * m43 - a23 * a32 * m44 - a24 * a33 * m42) / det;
|
|
float b12 = (a12 * a34 * m43 + a13 * a32 * m44 + a14 * a33 * m42 - a12 * a33 * m44 - a13 * a34 * m42 - a14 * a32 * m43) / det;
|
|
float b13 = (a12 * a23 * m44 + a13 * a24 * m42 + a14 * a22 * m43 - a12 * a24 * m43 - a13 * a22 * m44 - a14 * a23 * m42) / det;
|
|
float b14 = (a12 * a24 * a33 + a13 * a22 * a34 + a14 * a23 * a32 - a12 * a23 * a34 - a13 * a24 * a32 - a14 * a22 * a33) / det;
|
|
|
|
float b21 = (a21 * a34 * m43 + a23 * a31 * m44 + a24 * a33 * m41 - a21 * a33 * m44 - a23 * a34 * m41 - a24 * a31 * m43) / det;
|
|
float b22 = (a11 * a33 * m44 + a13 * a34 * m41 + a14 * a31 * m43 - a11 * a34 * m43 - a13 * a31 * m44 - a14 * a33 * m41) / det;
|
|
float b23 = (a11 * a24 * m43 + a13 * a21 * m44 + a14 * a23 * m41 - a11 * a23 * m44 - a13 * a24 * m41 - a14 * a21 * m43) / det;
|
|
float b24 = (a11 * a23 * a34 + a13 * a24 * a31 + a14 * a21 * a33 - a11 * a24 * a33 - a13 * a21 * a34 - a14 * a23 * a31) / det;
|
|
|
|
float b31 = (a21 * a32 * m44 + a22 * a34 * m41 + a24 * a31 * m42 - a21 * a34 * m42 - a22 * a31 * m44 - a24 * a32 * m41) / det;
|
|
float b32 = (a11 * a34 * m42 + a12 * a31 * m44 + a14 * a32 * m41 - a11 * a32 * m44 - a12 * a34 * m41 - a14 * a31 * m42) / det;
|
|
float b33 = (a11 * a22 * m44 + a12 * a24 * m41 + a14 * a21 * m42 - a11 * a24 * m42 - a12 * a21 * m44 - a14 * a22 * m41) / det;
|
|
float b34 = (a11 * a24 * a32 + a12 * a21 * a34 + a14 * a22 * a31 - a11 * a22 * a34 - a12 * a24 * a31 - a14 * a21 * a32) / det;
|
|
|
|
float b41 = (a21 * a33 * m42 + a22 * a31 * m43 + a23 * a32 * m41 - a21 * a32 * m43 - a22 * a33 * m41 - a23 * a31 * m42) / det;
|
|
float b42 = (a11 * a32 * m43 + a12 * a33 * m41 + a13 * a31 * m42 - a11 * a33 * m42 - a12 * a31 * m43 - a13 * a32 * m41) / det;
|
|
float b43 = (a11 * a23 * m42 + a12 * a21 * m43 + a13 * a22 * m41 - a11 * a22 * m43 - a12 * a23 * m41 - a13 * a21 * m42) / det;
|
|
float b44 = (a11 * a22 * a33 + a12 * a23 * a31 + a13 * a21 * a32 - a11 * a23 * a32 - a12 * a21 * a33 - a13 * a22 * a31) / det;
|
|
|
|
return new CFrame(b14, b24, b34, b11, b12, b13, b21, b22, b23, b31, b32, b33);
|
|
}
|
|
|
|
public static CFrame FromAxisAngle(Vector3 axis, float theta)
|
|
{
|
|
Vector3 r = VectorAxisAngle(axis, Vector3.Right, theta);
|
|
Vector3 u = VectorAxisAngle(axis, Vector3.Up, theta);
|
|
Vector3 b = VectorAxisAngle(axis, Vector3.Back, theta);
|
|
|
|
return new CFrame(0, 0, 0, r.X, u.X, b.X, r.Y, u.Y, b.Y, r.Z, u.Z, b.Z);
|
|
}
|
|
|
|
public static CFrame Angles(float x, float y, float z)
|
|
{
|
|
CFrame cfx = FromAxisAngle(Vector3.Right, x);
|
|
CFrame cfy = FromAxisAngle(Vector3.Up, y);
|
|
CFrame cfz = FromAxisAngle(Vector3.Back, z);
|
|
|
|
return cfx * cfy * cfz;
|
|
}
|
|
|
|
public static CFrame FromEulerAnglesXYZ(float x, float y, float z)
|
|
{
|
|
return Angles(x, y, z);
|
|
}
|
|
|
|
public CFrame Lerp(CFrame other, float t)
|
|
{
|
|
if (t == 0.0f)
|
|
{
|
|
return this;
|
|
}
|
|
else if (t == 1.0f)
|
|
{
|
|
return other;
|
|
}
|
|
else
|
|
{
|
|
Quaternion q1 = new Quaternion(this);
|
|
Quaternion q2 = new Quaternion(other);
|
|
|
|
CFrame rot = q1.Slerp(q2, t).ToCFrame();
|
|
Vector3 pos = Position.Lerp(other.Position, t);
|
|
|
|
return new CFrame(pos) * rot;
|
|
}
|
|
}
|
|
|
|
public CFrame ToWorldSpace(CFrame cf2)
|
|
{
|
|
return this * cf2;
|
|
}
|
|
|
|
public CFrame ToObjectSpace(CFrame other)
|
|
{
|
|
return Inverse() * other;
|
|
}
|
|
|
|
public Vector3 PointToWorldSpace(Vector3 v)
|
|
{
|
|
return this * v;
|
|
}
|
|
|
|
public Vector3 PointToObjectSpace(Vector3 v)
|
|
{
|
|
return Inverse() * v;
|
|
}
|
|
|
|
public Vector3 VectorToWorldSpace(Vector3 v)
|
|
{
|
|
return (this - Position) * v;
|
|
}
|
|
|
|
public Vector3 VectorToObjectSpace(Vector3 v)
|
|
{
|
|
return (this - Position).Inverse() * v;
|
|
}
|
|
|
|
public float[] GetComponents()
|
|
{
|
|
return new float[] { m14, m24, m34, m11, m12, m13, m21, m22, m23, m31, m32, m33 };
|
|
}
|
|
|
|
public float[] ToEulerAnglesXYZ()
|
|
{
|
|
float x = (float)Math.Atan2(-m23, m33);
|
|
float y = (float)Math.Asin(m13);
|
|
float z = (float)Math.Atan2(-m12, m11);
|
|
|
|
return new float[] { x, y, z };
|
|
}
|
|
|
|
public bool IsAxisAligned()
|
|
{
|
|
float[] matrix = GetComponents();
|
|
|
|
byte sum0 = 0,
|
|
sum1 = 0;
|
|
|
|
for (int i = 3; i < 12; i++)
|
|
{
|
|
float t = Math.Abs(matrix[i]);
|
|
|
|
if (t.FuzzyEquals(1))
|
|
{
|
|
// Approximately ±1
|
|
sum1++;
|
|
}
|
|
else if (t.FuzzyEquals(0))
|
|
{
|
|
// Approximately ±0
|
|
sum0++;
|
|
}
|
|
}
|
|
|
|
return (sum0 == 6 && sum1 == 3);
|
|
}
|
|
|
|
private static bool IsLegalOrientId(int orientId)
|
|
{
|
|
int xOrientId = (orientId / 6) % 3;
|
|
int yOrientId = orientId % 3;
|
|
|
|
return (xOrientId != yOrientId);
|
|
}
|
|
|
|
public int GetOrientId()
|
|
{
|
|
if (!IsAxisAligned())
|
|
return -1;
|
|
|
|
int xNormal = RightVector.ToNormalId();
|
|
int yNormal = UpVector.ToNormalId();
|
|
|
|
int orientId = (6 * xNormal) + yNormal;
|
|
|
|
if (!IsLegalOrientId(orientId))
|
|
return -1;
|
|
|
|
return orientId;
|
|
}
|
|
}
|
|
}
|