forked from mirror/DotRecast
329 lines
9.5 KiB
C#
329 lines
9.5 KiB
C#
/*
|
|
recast4j copyright (c) 2015-2019 Piotr Piastucki piotr@jtilia.org
|
|
|
|
This software is provided 'as-is', without any express or implied
|
|
warranty. In no event will the authors be held liable for any damages
|
|
arising from the use of this software.
|
|
Permission is granted to anyone to use this software for any purpose,
|
|
including commercial applications, and to alter it and redistribute it
|
|
freely, subject to the following restrictions:
|
|
1. The origin of this software must not be misrepresented; you must not
|
|
claim that you wrote the original software. If you use this software
|
|
in a product, an acknowledgment in the product documentation would be
|
|
appreciated but is not required.
|
|
2. Altered source versions must be plainly marked as such, and must not be
|
|
misrepresented as being the original software.
|
|
3. This notice may not be removed or altered from any source distribution.
|
|
*/
|
|
|
|
using System;
|
|
using System.Runtime.CompilerServices;
|
|
using System.Security.Permissions;
|
|
|
|
namespace DotRecast.Core
|
|
{
|
|
public struct Vector3f
|
|
{
|
|
public float x;
|
|
public float y;
|
|
public float z;
|
|
|
|
public static Vector3f Zero { get; } = new Vector3f(0, 0, 0);
|
|
public static Vector3f Up { get; } = new Vector3f(0, 1, 0);
|
|
|
|
public static Vector3f Of(float[] f)
|
|
{
|
|
return Of(f, 0);
|
|
}
|
|
|
|
public static Vector3f Of(float[] f, int idx)
|
|
{
|
|
return Of(f[idx + 0], f[idx + 1], f[idx + 2]);
|
|
}
|
|
|
|
public static Vector3f Of(float x, float y, float z)
|
|
{
|
|
return new Vector3f(x, y, z);
|
|
}
|
|
|
|
public Vector3f(float x, float y, float z)
|
|
{
|
|
this.x = x;
|
|
this.y = y;
|
|
this.z = z;
|
|
}
|
|
|
|
public Vector3f(float f)
|
|
{
|
|
x = f;
|
|
y = f;
|
|
z = f;
|
|
}
|
|
|
|
|
|
public Vector3f(float[] f)
|
|
{
|
|
x = f[0];
|
|
y = f[1];
|
|
z = f[2];
|
|
}
|
|
|
|
public float this[int index]
|
|
{
|
|
get => GetElement(index);
|
|
set => SetElement(index, value);
|
|
}
|
|
|
|
public float GetElement(int index)
|
|
{
|
|
switch (index)
|
|
{
|
|
case 0: return x;
|
|
case 1: return y;
|
|
case 2: return z;
|
|
default: throw new IndexOutOfRangeException($"{index}");
|
|
}
|
|
}
|
|
|
|
public void SetElement(int index, float value)
|
|
{
|
|
switch (index)
|
|
{
|
|
case 0:
|
|
x = value;
|
|
break;
|
|
case 1:
|
|
y = value;
|
|
break;
|
|
case 2:
|
|
z = value;
|
|
break;
|
|
|
|
default: throw new IndexOutOfRangeException($"{index}-{value}");
|
|
}
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public void Set(float a, float b, float c)
|
|
{
|
|
x = a;
|
|
y = b;
|
|
z = c;
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public void Set(float[] @in)
|
|
{
|
|
Set(@in, 0);
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public void Set(float[] @in, int i)
|
|
{
|
|
x = @in[i];
|
|
y = @in[i + 1];
|
|
z = @in[i + 2];
|
|
}
|
|
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public readonly float Length()
|
|
{
|
|
return (float)Math.Sqrt(x * x + y * y + z * z);
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public readonly Vector3f Subtract(Vector3f right)
|
|
{
|
|
return new Vector3f(
|
|
x - right.x,
|
|
y - right.y,
|
|
z - right.z
|
|
);
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public readonly Vector3f Add(Vector3f v2)
|
|
{
|
|
return new Vector3f(
|
|
x + v2.x,
|
|
y + v2.y,
|
|
z + v2.z
|
|
);
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public readonly Vector3f Scale(float scale)
|
|
{
|
|
return new Vector3f(
|
|
x * scale,
|
|
y * scale,
|
|
z * scale
|
|
);
|
|
}
|
|
|
|
|
|
/// Derives the dot product of two vectors on the xz-plane. (@p u . @p v)
|
|
/// @param[in] u A vector [(x, y, z)]
|
|
/// @param[in] v A vector [(x, y, z)]
|
|
/// @return The dot product on the xz-plane.
|
|
///
|
|
/// The vectors are projected onto the xz-plane, so the y-values are
|
|
/// ignored.
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public readonly float Dot2D(Vector3f v)
|
|
{
|
|
return x * v.x + z * v.z;
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public readonly float Dot2D(float[] v, int vi)
|
|
{
|
|
return x * v[vi] + z * v[vi + 2];
|
|
}
|
|
|
|
public override bool Equals(object obj)
|
|
{
|
|
if (!(obj is Vector3f))
|
|
return false;
|
|
|
|
return Equals((Vector3f)obj);
|
|
}
|
|
|
|
public bool Equals(Vector3f other)
|
|
{
|
|
return x.Equals(other.x) &&
|
|
y.Equals(other.y) &&
|
|
z.Equals(other.z);
|
|
}
|
|
|
|
|
|
public override int GetHashCode()
|
|
{
|
|
int hash = x.GetHashCode();
|
|
hash = RcHashCodes.CombineHashCodes(hash, y.GetHashCode());
|
|
hash = RcHashCodes.CombineHashCodes(hash, z.GetHashCode());
|
|
return hash;
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public void Normalize()
|
|
{
|
|
float d = (float)(1.0f / Math.Sqrt(RcMath.Sqr(x) + RcMath.Sqr(y) + RcMath.Sqr(z)));
|
|
if (d != 0)
|
|
{
|
|
x *= d;
|
|
y *= d;
|
|
z *= d;
|
|
}
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static bool operator ==(Vector3f left, Vector3f right)
|
|
{
|
|
return left.Equals(right);
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static bool operator !=(Vector3f left, Vector3f right)
|
|
{
|
|
return !left.Equals(right);
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static Vector3f operator -(Vector3f left, Vector3f right)
|
|
{
|
|
return left.Subtract(right);
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static Vector3f operator +(Vector3f left, Vector3f right)
|
|
{
|
|
return left.Add(right);
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static Vector3f operator *(Vector3f left, Vector3f right)
|
|
{
|
|
return new Vector3f(
|
|
left.x * right.x,
|
|
left.y * right.y,
|
|
left.z * right.z
|
|
);
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static Vector3f operator *(Vector3f left, float right)
|
|
{
|
|
return left * new Vector3f(right);
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static Vector3f operator *(float left, Vector3f right)
|
|
{
|
|
return right * left;
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static Vector3f Cross(Vector3f v1, Vector3f v2)
|
|
{
|
|
return new Vector3f(
|
|
(v1.y * v2.z) - (v1.z * v2.y),
|
|
(v1.z * v2.x) - (v1.x * v2.z),
|
|
(v1.x * v2.y) - (v1.y * v2.x)
|
|
);
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static Vector3f Lerp(Vector3f v1, Vector3f v2, float t)
|
|
{
|
|
return new Vector3f(
|
|
v1.x + (v2.x - v1.x) * t,
|
|
v1.y + (v2.y - v1.y) * t,
|
|
v1.z + (v2.z - v1.z) * t
|
|
);
|
|
}
|
|
|
|
/// Returns the distance between two points.
|
|
/// @param[in] v1 A point. [(x, y, z)]
|
|
/// @param[in] v2 A point. [(x, y, z)]
|
|
/// @return The distance between the two points.
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static float Distance(Vector3f v1, Vector3f v2)
|
|
{
|
|
float dx = v2.x - v1.x;
|
|
float dy = v2.y - v1.y;
|
|
float dz = v2.z - v1.z;
|
|
return (float)Math.Sqrt(dx * dx + dy * dy + dz * dz);
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static float Dot(Vector3f v1, Vector3f v2)
|
|
{
|
|
return (v1.x * v2.x) + (v1.y * v2.y)
|
|
+ (v1.z * v2.z);
|
|
}
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static float PerpXZ(Vector3f a, Vector3f b)
|
|
{
|
|
return (a.x * b.z) - (a.z * b.x);
|
|
}
|
|
|
|
/// Performs a scaled vector addition. (@p v1 + (@p v2 * @p s))
|
|
/// @param[out] dest The result vector. [(x, y, z)]
|
|
/// @param[in] v1 The base vector. [(x, y, z)]
|
|
/// @param[in] v2 The vector to scale and add to @p v1. [(x, y, z)]
|
|
/// @param[in] s The amount to scale @p v2 by before adding to @p v1.
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static Vector3f Mad(Vector3f v1, Vector3f v2, float s)
|
|
{
|
|
return new Vector3f()
|
|
{
|
|
x = v1.x + (v2.x * s),
|
|
y = v1.y + (v2.y * s),
|
|
z = v1.z + (v2.z * s),
|
|
};
|
|
}
|
|
}
|
|
} |