mirror of
https://github.com/aap/librw.git
synced 2025-01-22 08:41:27 +00:00
1238 lines
20 KiB
C++
1238 lines
20 KiB
C++
#include "math/math.h"
|
|
|
|
/*
|
|
* Vec3
|
|
*/
|
|
|
|
std::ostream&
|
|
operator<<(std::ostream& of, const Vec3 &v)
|
|
{
|
|
v.print(of);
|
|
return of;
|
|
}
|
|
|
|
|
|
Vec3::Vec3(void)
|
|
: x(0.0f), y(0.0f), z(0.0f)
|
|
{
|
|
}
|
|
|
|
Vec3::Vec3(float x, float y, float z)
|
|
: x(x), y(y), z(z)
|
|
{
|
|
}
|
|
|
|
Vec3::Vec3(float *v)
|
|
: x(v[0]), y(v[1]), z(v[2])
|
|
{
|
|
}
|
|
|
|
float*
|
|
Vec3::ptr(void)
|
|
{
|
|
return &x;
|
|
}
|
|
|
|
Vec3
|
|
Vec3::operator-(void) const
|
|
{
|
|
return Vec3(-x, -y, -z);
|
|
}
|
|
|
|
Vec3
|
|
Vec3::operator+(const Vec3 &rhs) const
|
|
{
|
|
return Vec3(this->x+rhs.x, this->y+rhs.y, this->z+rhs.z);
|
|
}
|
|
|
|
Vec3
|
|
Vec3::operator-(const Vec3 &rhs) const
|
|
{
|
|
return Vec3(this->x-rhs.x, this->y-rhs.y, this->z-rhs.z);
|
|
}
|
|
|
|
Vec3
|
|
Vec3::operator*(float rhs) const
|
|
{
|
|
return Vec3(this->x*rhs, this->y*rhs, this->z*rhs);
|
|
}
|
|
|
|
Vec3
|
|
Vec3::operator/(float rhs) const
|
|
{
|
|
return Vec3(this->x/rhs, this->y/rhs, this->z/rhs);
|
|
}
|
|
|
|
Vec3&
|
|
Vec3::operator+=(const Vec3 &rhs)
|
|
{
|
|
*this = *this + rhs;
|
|
return *this;
|
|
}
|
|
|
|
Vec3&
|
|
Vec3::operator-=(const Vec3 &rhs)
|
|
{
|
|
*this = *this - rhs;
|
|
return *this;
|
|
}
|
|
|
|
Vec3&
|
|
Vec3::operator*=(float rhs)
|
|
{
|
|
*this = *this * rhs;
|
|
return *this;
|
|
}
|
|
|
|
Vec3&
|
|
Vec3::operator/=(float rhs)
|
|
{
|
|
*this = *this / rhs;
|
|
return *this;
|
|
}
|
|
|
|
bool
|
|
Vec3::operator==(const Vec3 &rhs) const
|
|
{
|
|
return (this->x == rhs.x) &&
|
|
(this->y == rhs.y) &&
|
|
(this->z == rhs.z);
|
|
}
|
|
|
|
bool
|
|
Vec3::operator!=(const Vec3 &rhs) const
|
|
{
|
|
return (this->x != rhs.x) ||
|
|
(this->y != rhs.y) ||
|
|
(this->z != rhs.z);
|
|
}
|
|
|
|
float
|
|
Vec3::norm(void) const
|
|
{
|
|
return sqrt(normsq());
|
|
}
|
|
|
|
float
|
|
Vec3::normsq(void) const
|
|
{
|
|
return x*x + y*y + z*z;
|
|
}
|
|
|
|
Vec3
|
|
Vec3::normalized(void) const
|
|
{
|
|
return Vec3(*this)/norm();
|
|
}
|
|
|
|
float
|
|
Vec3::dot(const Vec3 &rhs) const
|
|
{
|
|
return this->x*rhs.x + this->y*rhs.y + this->z*rhs.z;
|
|
}
|
|
|
|
Vec3
|
|
Vec3::cross(const Vec3 &rhs) const
|
|
{
|
|
return Vec3(this->y*rhs.z - this->z*rhs.y,
|
|
this->z*rhs.x - this->x*rhs.z,
|
|
this->x*rhs.y - this->y*rhs.x);
|
|
}
|
|
|
|
void
|
|
Vec3::print(std::ostream &of) const
|
|
{
|
|
of << "V3(" << x << ", " << y << ", " << z << ")";
|
|
}
|
|
|
|
/*
|
|
* Vec4
|
|
*/
|
|
|
|
std::ostream&
|
|
operator<<(std::ostream& of, const Vec4 &v)
|
|
{
|
|
v.print(of);
|
|
return of;
|
|
}
|
|
|
|
|
|
Vec4::Vec4(void)
|
|
: x(0.0f), y(0.0f), z(0.0f), w(0.0f)
|
|
{
|
|
}
|
|
|
|
Vec4::Vec4(float x, float y, float z, float w)
|
|
: x(x), y(y), z(z), w(w)
|
|
{
|
|
}
|
|
|
|
Vec4::Vec4(float *v)
|
|
: x(v[0]), y(v[1]), z(v[2]), w(v[3])
|
|
{
|
|
}
|
|
|
|
float*
|
|
Vec4::ptr(void)
|
|
{
|
|
return &x;
|
|
}
|
|
|
|
Vec4
|
|
Vec4::operator-(void) const
|
|
{
|
|
return Vec4(-x, -y, -z, -w);
|
|
}
|
|
|
|
Vec4
|
|
Vec4::operator+(const Vec4 &rhs) const
|
|
{
|
|
return Vec4(this->x+rhs.x, this->y+rhs.y, this->z+rhs.z, this->w+rhs.w);
|
|
}
|
|
|
|
Vec4
|
|
Vec4::operator-(const Vec4 &rhs) const
|
|
{
|
|
return Vec4(this->x-rhs.x, this->y-rhs.y, this->z-rhs.z, this->w-rhs.w);
|
|
}
|
|
|
|
Vec4
|
|
Vec4::operator*(float rhs) const
|
|
{
|
|
return Vec4(this->x*rhs, this->y*rhs, this->z*rhs, this->w*rhs);
|
|
}
|
|
|
|
Vec4
|
|
Vec4::operator/(float rhs) const
|
|
{
|
|
return Vec4(this->x/rhs, this->y/rhs, this->z/rhs, this->w/rhs);
|
|
}
|
|
|
|
Vec4&
|
|
Vec4::operator+=(const Vec4 &rhs)
|
|
{
|
|
*this = *this + rhs;
|
|
return *this;
|
|
}
|
|
|
|
Vec4&
|
|
Vec4::operator-=(const Vec4 &rhs)
|
|
{
|
|
*this = *this - rhs;
|
|
return *this;
|
|
}
|
|
|
|
Vec4&
|
|
Vec4::operator*=(float rhs)
|
|
{
|
|
*this = *this * rhs;
|
|
return *this;
|
|
}
|
|
|
|
Vec4&
|
|
Vec4::operator/=(float rhs)
|
|
{
|
|
*this = *this / rhs;
|
|
return *this;
|
|
}
|
|
|
|
bool
|
|
Vec4::operator==(const Vec4 &rhs) const
|
|
{
|
|
return (this->x == rhs.x) &&
|
|
(this->y == rhs.y) &&
|
|
(this->z == rhs.z) &&
|
|
(this->w == rhs.w);
|
|
}
|
|
|
|
bool
|
|
Vec4::operator!=(const Vec4 &rhs) const
|
|
{
|
|
return (this->x != rhs.x) ||
|
|
(this->y != rhs.y) ||
|
|
(this->z != rhs.z) ||
|
|
(this->w != rhs.w);
|
|
}
|
|
|
|
float
|
|
Vec4::norm(void) const
|
|
{
|
|
return sqrt(normsq());
|
|
}
|
|
|
|
float
|
|
Vec4::normsq(void) const
|
|
{
|
|
return x*x + y*y + z*z + w*w;
|
|
}
|
|
|
|
Vec4
|
|
Vec4::normalized(void) const
|
|
{
|
|
return Vec4(*this)/norm();
|
|
}
|
|
|
|
float
|
|
Vec4::dot(const Vec4 &rhs) const
|
|
{
|
|
return this->x*rhs.x + this->y*rhs.y + this->z*rhs.z + this->w*rhs.w;
|
|
}
|
|
|
|
void
|
|
Vec4::print(std::ostream &of) const
|
|
{
|
|
of << "V4(" << x << ", " << y << ", " << z << ", " << w << ")";
|
|
}
|
|
|
|
/*
|
|
* Quat
|
|
*/
|
|
|
|
std::ostream&
|
|
operator<<(std::ostream& of, const Quat &v)
|
|
{
|
|
v.print(of);
|
|
return of;
|
|
}
|
|
|
|
|
|
Quat::Quat(void)
|
|
: w(0.0f), x(0.0f), y(0.0f), z(0.0f)
|
|
{
|
|
}
|
|
|
|
Quat::Quat(float w)
|
|
: w(w), x(0.0f), y(0.0f), z(0.0f)
|
|
{
|
|
}
|
|
|
|
Quat::Quat(float x, float y, float z)
|
|
: w(0.0f), x(x), y(y), z(z)
|
|
{
|
|
}
|
|
|
|
Quat::Quat(float w, float x, float y, float z)
|
|
: w(w), x(x), y(y), z(z)
|
|
{
|
|
}
|
|
|
|
float*
|
|
Quat::ptr(void)
|
|
{
|
|
return &w;
|
|
}
|
|
|
|
Quat
|
|
Quat::operator-(void) const
|
|
{
|
|
return Quat(-w, -x, -y, -z);
|
|
}
|
|
|
|
Quat
|
|
Quat::operator+(const Quat &rhs) const
|
|
{
|
|
return Quat(this->w+rhs.w, this->x+rhs.x, this->y+rhs.y, this->z+rhs.z);
|
|
}
|
|
|
|
Quat
|
|
Quat::operator-(const Quat &rhs) const
|
|
{
|
|
return Quat(this->w-rhs.w, this->x-rhs.x, this->y-rhs.y, this->z-rhs.z);
|
|
}
|
|
|
|
Quat
|
|
Quat::operator*(const Quat &rhs) const
|
|
{
|
|
return Quat(
|
|
this->w*rhs.w - this->x*rhs.x - this->y*rhs.y - this->z*rhs.z,
|
|
this->w*rhs.x + this->x*rhs.w + this->y*rhs.z - this->z*rhs.y,
|
|
this->w*rhs.y + this->y*rhs.w + this->z*rhs.x - this->x*rhs.z,
|
|
this->w*rhs.z + this->z*rhs.w + this->x*rhs.y - this->y*rhs.x);
|
|
}
|
|
|
|
Quat
|
|
Quat::operator*(float rhs) const
|
|
{
|
|
return Quat(this->w*rhs, this->x*rhs, this->y*rhs, this->z*rhs);
|
|
}
|
|
|
|
Quat
|
|
Quat::operator/(float rhs) const
|
|
{
|
|
return Quat(this->w/rhs, this->x/rhs, this->y/rhs, this->z/rhs);
|
|
}
|
|
|
|
Quat&
|
|
Quat::operator+=(const Quat &rhs)
|
|
{
|
|
*this = *this + rhs;
|
|
return *this;
|
|
}
|
|
|
|
Quat&
|
|
Quat::operator-=(const Quat &rhs)
|
|
{
|
|
*this = *this - rhs;
|
|
return *this;
|
|
}
|
|
|
|
Quat&
|
|
Quat::operator*=(const Quat &rhs)
|
|
{
|
|
*this = *this * rhs;
|
|
return *this;
|
|
}
|
|
|
|
Quat&
|
|
Quat::operator*=(float rhs)
|
|
{
|
|
*this = *this * rhs;
|
|
return *this;
|
|
}
|
|
|
|
Quat&
|
|
Quat::operator/=(float rhs)
|
|
{
|
|
*this = *this / rhs;
|
|
return *this;
|
|
}
|
|
|
|
bool
|
|
Quat::operator==(const Quat &rhs) const
|
|
{
|
|
return (this->w == rhs.w) &&
|
|
(this->x == rhs.x) &&
|
|
(this->y == rhs.y) &&
|
|
(this->z == rhs.z);
|
|
}
|
|
|
|
bool
|
|
Quat::operator!=(const Quat &rhs) const
|
|
{
|
|
return (this->w != rhs.w) ||
|
|
(this->x != rhs.x) ||
|
|
(this->y != rhs.y) ||
|
|
(this->z != rhs.z);
|
|
}
|
|
|
|
Quat
|
|
Quat::inv(void) const
|
|
{
|
|
return K() / N();
|
|
}
|
|
|
|
Quat
|
|
Quat::K(void) const
|
|
{
|
|
return Quat(w, -x, -y, -z);
|
|
}
|
|
|
|
Quat
|
|
Quat::S(void) const
|
|
{
|
|
return Quat(w);
|
|
}
|
|
|
|
Quat
|
|
Quat::V(void) const
|
|
{
|
|
return Quat(x, y, z);
|
|
}
|
|
|
|
float
|
|
Quat::T(void) const
|
|
{
|
|
return sqrt(N());
|
|
}
|
|
|
|
float
|
|
Quat::N(void) const
|
|
{
|
|
return w*w + x*x + y*y + z*z;
|
|
}
|
|
|
|
Quat
|
|
Quat::U(void) const
|
|
{
|
|
return Quat(*this)/T();
|
|
}
|
|
|
|
Quat
|
|
Quat::wedge(const Quat &rhs) const
|
|
{
|
|
return Quat(0.0f,
|
|
this->y*rhs.z - this->z*rhs.y,
|
|
this->z*rhs.x - this->x*rhs.z,
|
|
this->x*rhs.y - this->y*rhs.x);
|
|
}
|
|
|
|
float
|
|
Quat::inner(const Quat &rhs) const
|
|
{
|
|
return this->w*rhs.w + this->x*rhs.x + this->y*rhs.y + this->z*rhs.z;
|
|
}
|
|
|
|
Quat
|
|
Quat::lerp(const Quat &q2, float t) const
|
|
{
|
|
Quat q1 = *this;
|
|
float cos = q1.inner(q2);
|
|
if(cos < 0)
|
|
q1 = -q1;
|
|
return (q1*(1.0f - t) + q2*t).U();
|
|
}
|
|
|
|
Quat
|
|
Quat::slerp(const Quat &q2, float t) const
|
|
{
|
|
Quat q1 = *this;
|
|
float cos = q1.inner(q2);
|
|
if(cos < 0){
|
|
cos = -cos;
|
|
q1 = -q1;
|
|
}
|
|
float phi = acos(cos);
|
|
if(phi > 0.00001){
|
|
float s = sin(phi);
|
|
q1 = q1*sin((1.0f-t)*phi)/s + q2*sin(t*phi)/s;
|
|
}
|
|
return q1;
|
|
}
|
|
|
|
void
|
|
Quat::print(std::ostream &of) const
|
|
{
|
|
of << "Q(" << w << ", " << x << ", " << y << ", " << z << ")";
|
|
}
|
|
|
|
/*
|
|
* Quat
|
|
*/
|
|
|
|
std::ostream&
|
|
operator<<(std::ostream& of, const DQuat &v)
|
|
{
|
|
v.print(of);
|
|
return of;
|
|
}
|
|
|
|
|
|
DQuat::DQuat(void)
|
|
: q1(), q2()
|
|
{
|
|
}
|
|
|
|
DQuat::DQuat(const Quat &q1, const Quat &q2)
|
|
: q1(q1), q2(q2)
|
|
{
|
|
}
|
|
|
|
DQuat
|
|
DQuat::operator-(void) const
|
|
{
|
|
return DQuat(-q1, -q2);
|
|
}
|
|
|
|
DQuat
|
|
DQuat::operator+(const DQuat &rhs) const
|
|
{
|
|
return DQuat(this->q1+rhs.q1, this->q2+rhs.q2);
|
|
}
|
|
|
|
DQuat
|
|
DQuat::operator-(const DQuat &rhs) const
|
|
{
|
|
return DQuat(this->q1-rhs.q1, this->q2-rhs.q2);
|
|
}
|
|
|
|
DQuat
|
|
DQuat::operator*(const DQuat &rhs) const
|
|
{
|
|
return DQuat(this->q1*rhs.q1, this->q1*rhs.q2 + this->q2*rhs.q1);
|
|
}
|
|
|
|
DQuat
|
|
DQuat::operator*(float rhs) const
|
|
{
|
|
return DQuat(this->q1*rhs, this->q2*rhs);
|
|
}
|
|
|
|
DQuat
|
|
DQuat::operator/(float rhs) const
|
|
{
|
|
return DQuat(this->q1/rhs, this->q2/rhs);
|
|
}
|
|
|
|
DQuat&
|
|
DQuat::operator+=(const DQuat &rhs)
|
|
{
|
|
*this = *this + rhs;
|
|
return *this;
|
|
}
|
|
|
|
DQuat&
|
|
DQuat::operator-=(const DQuat &rhs)
|
|
{
|
|
*this = *this - rhs;
|
|
return *this;
|
|
}
|
|
|
|
DQuat&
|
|
DQuat::operator*=(const DQuat &rhs)
|
|
{
|
|
*this = *this * rhs;
|
|
return *this;
|
|
}
|
|
|
|
DQuat&
|
|
DQuat::operator*=(float rhs)
|
|
{
|
|
*this = *this * rhs;
|
|
return *this;
|
|
}
|
|
|
|
DQuat&
|
|
DQuat::operator/=(float rhs)
|
|
{
|
|
*this = *this / rhs;
|
|
return *this;
|
|
}
|
|
|
|
bool
|
|
DQuat::operator==(const DQuat &rhs) const
|
|
{
|
|
return (this->q1 == rhs.q1) &&
|
|
(this->q2 == rhs.q2);
|
|
}
|
|
|
|
bool
|
|
DQuat::operator!=(const DQuat &rhs) const
|
|
{
|
|
return (this->q1 != rhs.q1) ||
|
|
(this->q2 != rhs.q2);
|
|
}
|
|
|
|
DQuat
|
|
DQuat::K(void) const
|
|
{
|
|
return DQuat(q1.K(), -q2.K());
|
|
}
|
|
|
|
void
|
|
DQuat::print(std::ostream &of) const
|
|
{
|
|
of << "DQ(" << q1 << ", " << q2 << ")";
|
|
}
|
|
|
|
/*
|
|
* Mat3
|
|
*/
|
|
|
|
std::ostream&
|
|
operator<<(std::ostream& of, const Mat3 &v)
|
|
{
|
|
v.print(of);
|
|
return of;
|
|
}
|
|
|
|
Mat3::Mat3(void)
|
|
{
|
|
for(int i = 0; i < 3*3; i++)
|
|
cr[i] = 0.0f;
|
|
}
|
|
|
|
Mat3::Mat3(float f)
|
|
{
|
|
for(int i = 0; i < 3; i++)
|
|
for(int j = 0; j < 3; j++)
|
|
e[i][j] = (i == j) ? f : 0.0f;
|
|
}
|
|
|
|
Mat3::Mat3(float *f)
|
|
{
|
|
for(int i = 0; i < 3*3; i++)
|
|
cr[i] = f[i];
|
|
}
|
|
|
|
Mat3::Mat3(float e00, float e10, float e20,
|
|
float e01, float e11, float e21,
|
|
float e02, float e12, float e22)
|
|
{
|
|
e[0][0] = e00; e[1][0] = e10; e[2][0] = e20;
|
|
e[0][1] = e01; e[1][1] = e11; e[2][1] = e21;
|
|
e[0][2] = e02; e[1][2] = e12; e[2][2] = e22;
|
|
}
|
|
|
|
float*
|
|
Mat3::ptr(void)
|
|
{
|
|
return &e[0][0];
|
|
}
|
|
|
|
Mat3
|
|
Mat3::rotation(float theta, const Vec3 &v)
|
|
{
|
|
Mat3 m(1.0f);
|
|
float c = cos(theta);
|
|
float s = sin(theta);
|
|
m.e[0][0] = v.x*v.x*(1-c) + c;
|
|
m.e[1][0] = v.x*v.y*(1-c) - v.z*s;
|
|
m.e[2][0] = v.x*v.z*(1-c) + v.y*s;
|
|
|
|
m.e[0][1] = v.y*v.x*(1-c) + v.z*s;
|
|
m.e[1][1] = v.y*v.y*(1-c) + c;
|
|
m.e[2][1] = v.y*v.z*(1-c) - v.x*s;
|
|
|
|
m.e[0][2] = v.z*v.x*(1-c) - v.y*s;
|
|
m.e[1][2] = v.z*v.y*(1-c) + v.x*s;
|
|
m.e[2][2] = v.z*v.z*(1-c) + c;
|
|
return m;
|
|
}
|
|
|
|
Mat3
|
|
Mat3::scale(const Vec3 &v)
|
|
{
|
|
Mat3 m(1.0f);
|
|
m.e[0][0] = v.x;
|
|
m.e[1][1] = v.y;
|
|
m.e[2][2] = v.z;
|
|
return m;
|
|
}
|
|
|
|
Mat3
|
|
Mat3::transpose(void) const
|
|
{
|
|
float e[3][3];
|
|
for(int i = 0; i < 3; i++)
|
|
for(int j = 0; j < 3; j++){
|
|
e[j][i] = this->e[i][j];
|
|
e[i][j] = this->e[j][i];
|
|
}
|
|
return Mat3(&e[0][0]);
|
|
}
|
|
|
|
Mat3
|
|
Mat3::operator+(const Mat3 &rhs) const
|
|
{
|
|
float e[9];
|
|
for(int i = 0; i < 3*3; i++)
|
|
e[i] = this->cr[i] + rhs.cr[i];
|
|
return Mat3(e);
|
|
}
|
|
|
|
Mat3
|
|
Mat3::operator-(const Mat3 &rhs) const
|
|
{
|
|
float e[9];
|
|
for(int i = 0; i < 3*3; i++)
|
|
e[i] = this->cr[i] - rhs.cr[i];
|
|
return Mat3(e);
|
|
}
|
|
|
|
Mat3
|
|
Mat3::operator*(float rhs) const
|
|
{
|
|
float e[9];
|
|
for(int i = 0; i < 3*3; i++)
|
|
e[i] = this->cr[i]*rhs;
|
|
return Mat3(e);
|
|
}
|
|
|
|
Mat3
|
|
Mat3::operator/(float rhs) const
|
|
{
|
|
float e[9];
|
|
for(int i = 0; i < 3*3; i++)
|
|
e[i] = this->cr[i]/rhs;
|
|
return Mat3(e);
|
|
}
|
|
|
|
Mat3
|
|
Mat3::operator*(const Mat3 &rhs) const
|
|
{
|
|
float e[3][3];
|
|
for(int i = 0; i < 3; i++)
|
|
for(int j = 0; j < 3; j++)
|
|
e[i][j] = this->e[0][j]*rhs.e[i][0] +
|
|
this->e[1][j]*rhs.e[i][1] +
|
|
this->e[2][j]*rhs.e[i][2];
|
|
return Mat3(&e[0][0]);
|
|
}
|
|
|
|
Vec3
|
|
Mat3::operator*(const Vec3 &rhs) const
|
|
{
|
|
float e[3];
|
|
for(int i = 0; i < 3; i++)
|
|
e[i] = this->e[0][i]*rhs.x +
|
|
this->e[1][i]*rhs.y +
|
|
this->e[2][i]*rhs.z;
|
|
return Vec3(e);
|
|
}
|
|
|
|
Mat3&
|
|
Mat3::operator+=(const Mat3 &rhs)
|
|
{
|
|
*this = *this + rhs;
|
|
return *this;
|
|
}
|
|
|
|
Mat3&
|
|
Mat3::operator-=(const Mat3 &rhs)
|
|
{
|
|
*this = *this - rhs;
|
|
return *this;
|
|
}
|
|
|
|
Mat3&
|
|
Mat3::operator*=(float rhs)
|
|
{
|
|
*this = *this * rhs;
|
|
return *this;
|
|
}
|
|
|
|
Mat3&
|
|
Mat3::operator/=(float rhs)
|
|
{
|
|
*this = *this / rhs;
|
|
return *this;
|
|
}
|
|
|
|
Mat3&
|
|
Mat3::operator*=(const Mat3 &rhs)
|
|
{
|
|
*this = *this * rhs;
|
|
return *this;
|
|
}
|
|
|
|
void
|
|
Mat3::print(std::ostream &of) const
|
|
{
|
|
#define CM << ", " <<
|
|
of << "M3(" << e[0][0] CM e[1][0] CM e[2][0] << std::endl;
|
|
of << " " << e[0][1] CM e[1][1] CM e[2][1] << std::endl;
|
|
of << " " << e[0][2] CM e[1][2] CM e[2][2] << ")";
|
|
#undef CM
|
|
}
|
|
|
|
/*
|
|
* Mat3
|
|
*/
|
|
|
|
std::ostream&
|
|
operator<<(std::ostream& of, const Mat4 &v)
|
|
{
|
|
v.print(of);
|
|
return of;
|
|
}
|
|
|
|
Mat4::Mat4(void)
|
|
{
|
|
for(int i = 0; i < 4*4; i++)
|
|
cr[i] = 0.0f;
|
|
}
|
|
|
|
Mat4::Mat4(float f)
|
|
{
|
|
for(int i = 0; i < 4; i++)
|
|
for(int j = 0; j < 4; j++)
|
|
e[i][j] = (i == j) ? f : 0.0f;
|
|
}
|
|
|
|
Mat4::Mat4(float *f)
|
|
{
|
|
for(int i = 0; i < 4*4; i++)
|
|
cr[i] = f[i];
|
|
}
|
|
|
|
Mat4::Mat4(float e00, float e10, float e20, float e30,
|
|
float e01, float e11, float e21, float e31,
|
|
float e02, float e12, float e22, float e32,
|
|
float e03, float e13, float e23, float e33)
|
|
{
|
|
e[0][0] = e00; e[1][0] = e10; e[2][0] = e20; e[3][0] = e30;
|
|
e[0][1] = e01; e[1][1] = e11; e[2][1] = e21; e[3][1] = e31;
|
|
e[0][2] = e02; e[1][2] = e12; e[2][2] = e22; e[3][2] = e32;
|
|
e[0][3] = e03; e[1][3] = e13; e[2][3] = e23; e[3][3] = e33;
|
|
}
|
|
|
|
float*
|
|
Mat4::ptr(void)
|
|
{
|
|
return &e[0][0];
|
|
}
|
|
|
|
Mat4
|
|
Mat4::perspective(float fov, float aspect, float n, float f)
|
|
{
|
|
float r = n*tan(fov*3.14159f/360.0f);
|
|
float t = r/aspect;
|
|
return frustum(-r, r, -t, t, n, f);
|
|
}
|
|
|
|
Mat4
|
|
Mat4::frustum(float l, float r, float b, float t,
|
|
float n, float f)
|
|
{
|
|
Mat4 m(1.0f);
|
|
m.e[0][0] = (2.0f*n)/(r-l);
|
|
m.e[1][1] = (2.0f*n)/(t-b);
|
|
m.e[2][0] = (r+l)/(r-l);
|
|
m.e[2][1] = (t+b)/(t-b);
|
|
// TODO:
|
|
// m.e[2][2] = -(f+n)/(f-n);
|
|
// m.e[2][3] = -1.0f;
|
|
// m.e[3][2] = -2.0f*f*n/(f-n);
|
|
m.e[2][2] = (f+n)/(f-n);
|
|
m.e[2][3] = 1.0f;
|
|
m.e[3][2] = -1.0f*f*n/(f-n);
|
|
m.e[3][3] = 0.0f;
|
|
return m;
|
|
}
|
|
|
|
Mat4
|
|
Mat4::ortho(float l, float r, float b, float t,
|
|
float n, float f)
|
|
{
|
|
Mat4 m(1.0f);
|
|
m.e[0][0] = 2.0f/(r-l);
|
|
m.e[3][0] = -(r+l)/(r-l);
|
|
m.e[1][1] = 2.0f/(t-b);
|
|
m.e[3][1] = -(t+b)/(t-b);
|
|
m.e[2][2] = -2.0f/(f-n);
|
|
m.e[3][2] = -(f+n)/(f-n);
|
|
return m;
|
|
}
|
|
|
|
Mat4
|
|
Mat4::lookat(const Vec3 &pos, const Vec3 &target, const Vec3 &up)
|
|
{
|
|
Vec3 forward = (target - pos).normalized();
|
|
Vec3 side = forward.cross(up).normalized();
|
|
Vec3 nup = side.cross(forward);
|
|
Mat4 m(1.0f);
|
|
m.e[0][0] = side.x;
|
|
m.e[1][0] = side.y;
|
|
m.e[2][0] = side.z;
|
|
m.e[0][1] = nup.x;
|
|
m.e[1][1] = nup.y;
|
|
m.e[2][1] = nup.z;
|
|
m.e[0][2] = -forward.x;
|
|
m.e[1][2] = -forward.y;
|
|
m.e[2][2] = -forward.z;
|
|
m = m*Mat4::translation(-pos);
|
|
// TODO:
|
|
m.e[0][2] *= -1.0f;
|
|
m.e[1][2] *= -1.0f;
|
|
m.e[2][2] *= -1.0f;
|
|
m.e[3][2] *= -1.0f;
|
|
return m;
|
|
}
|
|
|
|
Mat4
|
|
Mat4::translation(const Vec3 &v)
|
|
{
|
|
Mat4 m(1.0f);
|
|
m.e[3][0] = v.x;
|
|
m.e[3][1] = v.y;
|
|
m.e[3][2] = v.z;
|
|
return m;
|
|
}
|
|
|
|
Mat4
|
|
Mat4::rotation(float theta, const Vec3 &v)
|
|
{
|
|
Mat4 m(1.0f);
|
|
float c = cos(theta);
|
|
float s = sin(theta);
|
|
m.e[0][0] = v.x*v.x*(1-c) + c;
|
|
m.e[1][0] = v.x*v.y*(1-c) - v.z*s;
|
|
m.e[2][0] = v.x*v.z*(1-c) + v.y*s;
|
|
|
|
m.e[0][1] = v.y*v.x*(1-c) + v.z*s;
|
|
m.e[1][1] = v.y*v.y*(1-c) + c;
|
|
m.e[2][1] = v.y*v.z*(1-c) - v.x*s;
|
|
|
|
m.e[0][2] = v.z*v.x*(1-c) - v.y*s;
|
|
m.e[1][2] = v.z*v.y*(1-c) + v.x*s;
|
|
m.e[2][2] = v.z*v.z*(1-c) + c;
|
|
return m;
|
|
}
|
|
|
|
Mat4
|
|
Mat4::scale(const Vec3 &v)
|
|
{
|
|
Mat4 m(1.0f);
|
|
m.e[0][0] = v.x;
|
|
m.e[1][1] = v.y;
|
|
m.e[2][2] = v.z;
|
|
return m;
|
|
}
|
|
|
|
Mat4
|
|
Mat4::transpose(void) const
|
|
{
|
|
float e[4][4];
|
|
for(int i = 0; i < 4; i++)
|
|
for(int j = 0; j < 4; j++){
|
|
e[j][i] = this->e[i][j];
|
|
e[i][j] = this->e[j][i];
|
|
}
|
|
return Mat4(&e[0][0]);
|
|
}
|
|
|
|
Mat4
|
|
Mat4::operator+(const Mat4 &rhs) const
|
|
{
|
|
float e[16];
|
|
for(int i = 0; i < 4*4; i++)
|
|
e[i] = this->cr[i] + rhs.cr[i];
|
|
return Mat4(e);
|
|
}
|
|
|
|
Mat4
|
|
Mat4::operator-(const Mat4 &rhs) const
|
|
{
|
|
float e[16];
|
|
for(int i = 0; i < 4*4; i++)
|
|
e[i] = this->cr[i] - rhs.cr[i];
|
|
return Mat4(e);
|
|
}
|
|
|
|
Mat4
|
|
Mat4::operator*(float rhs) const
|
|
{
|
|
float e[16];
|
|
for(int i = 0; i < 4*4; i++)
|
|
e[i] = this->cr[i]*rhs;
|
|
return Mat4(e);
|
|
}
|
|
|
|
Mat4
|
|
Mat4::operator/(float rhs) const
|
|
{
|
|
float e[16];
|
|
for(int i = 0; i < 4*4; i++)
|
|
e[i] = this->cr[i]/rhs;
|
|
return Mat4(e);
|
|
}
|
|
|
|
Mat4
|
|
Mat4::operator*(const Mat4 &rhs) const
|
|
{
|
|
float e[4][4];
|
|
for(int i = 0; i < 4; i++)
|
|
for(int j = 0; j < 4; j++)
|
|
e[i][j] = this->e[0][j]*rhs.e[i][0] +
|
|
this->e[1][j]*rhs.e[i][1] +
|
|
this->e[2][j]*rhs.e[i][2] +
|
|
this->e[3][j]*rhs.e[i][3];
|
|
return Mat4(&e[0][0]);
|
|
}
|
|
|
|
Vec4
|
|
Mat4::operator*(const Vec4 &rhs) const
|
|
{
|
|
float e[4];
|
|
for(int i = 0; i < 4; i++)
|
|
e[i] = this->e[0][i]*rhs.x +
|
|
this->e[1][i]*rhs.y +
|
|
this->e[2][i]*rhs.z +
|
|
this->e[3][i]*rhs.w;
|
|
return Vec4(e);
|
|
}
|
|
|
|
Mat4&
|
|
Mat4::operator+=(const Mat4 &rhs)
|
|
{
|
|
*this = *this + rhs;
|
|
return *this;
|
|
}
|
|
|
|
Mat4&
|
|
Mat4::operator-=(const Mat4 &rhs)
|
|
{
|
|
*this = *this - rhs;
|
|
return *this;
|
|
}
|
|
|
|
Mat4&
|
|
Mat4::operator*=(float rhs)
|
|
{
|
|
*this = *this * rhs;
|
|
return *this;
|
|
}
|
|
|
|
Mat4&
|
|
Mat4::operator/=(float rhs)
|
|
{
|
|
*this = *this / rhs;
|
|
return *this;
|
|
}
|
|
|
|
Mat4&
|
|
Mat4::operator*=(const Mat4 &rhs)
|
|
{
|
|
*this = *this * rhs;
|
|
return *this;
|
|
}
|
|
|
|
void
|
|
Mat4::print(std::ostream &of) const
|
|
{
|
|
#define CM << ", " <<
|
|
of << "M4(" << e[0][0] CM e[1][0] CM e[2][0] CM e[3][0] << std::endl;
|
|
of << " " << e[0][1] CM e[1][1] CM e[2][1] CM e[3][1] << std::endl;
|
|
of << " " << e[0][2] CM e[1][2] CM e[2][2] CM e[3][2] << std::endl;
|
|
of << " " << e[0][3] CM e[1][3] CM e[2][3] CM e[3][3] << ")";
|
|
#undef CM
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Vec3::Vec3(const Vec4 &v)
|
|
: x(v.x), y(v.y), z(v.z)
|
|
{
|
|
}
|
|
|
|
Vec3::Vec3(const Quat &q)
|
|
: x(q.x), y(q.y), z(q.z)
|
|
{
|
|
}
|
|
|
|
Vec4::Vec4(const Vec3 &v, float w)
|
|
: x(v.x), y(v.y), z(v.z), w(w)
|
|
{
|
|
}
|
|
|
|
Vec4::Vec4(const Quat &q)
|
|
: x(q.x), y(q.y), z(q.z), w(q.w)
|
|
{
|
|
}
|
|
|
|
Quat::Quat(const Vec3 &v)
|
|
: w(0), x(v.x), y(v.y), z(v.z)
|
|
{
|
|
}
|
|
|
|
Quat::Quat(float w, const Vec3 &v)
|
|
: w(w), x(v.x), y(v.y), z(v.z)
|
|
{
|
|
}
|
|
|
|
Quat::Quat(const Vec4 &v)
|
|
: w(v.w), x(v.x), y(v.y), z(v.z)
|
|
{
|
|
}
|
|
|
|
Quat::Quat(const Mat3 &m)
|
|
{
|
|
float trace, s, q[4];
|
|
int i, j, k;
|
|
|
|
int nxt[3] = {1, 2, 0};
|
|
|
|
trace = m.e[0][0] + m.e[1][1] + m.e[2][2];
|
|
if(trace > 0.0f){
|
|
s = sqrt(trace + 1.0f);
|
|
this->w = s / 2.0f;
|
|
s = 0.5f / s;
|
|
this->x = (m.e[2][1] - m.e[1][2]) * s;
|
|
this->y = (m.e[0][2] - m.e[2][0]) * s;
|
|
this->z = (m.e[1][0] - m.e[0][1]) * s;
|
|
}else{
|
|
i = 0;
|
|
if(m.e[1][1] > m.e[0][0]) i = 1;
|
|
if(m.e[2][2] > m.e[i][i]) i = 2;
|
|
j = nxt[i];
|
|
k = nxt[j];
|
|
s = sqrt((m.e[i][i] - m.e[j][j] - m.e[k][k]) + 1.0);
|
|
q[i] = s*0.5f;
|
|
if(q[i] != 0.0f) s = 0.5f / s;
|
|
q[3] = (m.e[k][j] - m.e[j][k]) * s;
|
|
q[j] = (m.e[j][i] + m.e[i][j]) * s;
|
|
q[k] = (m.e[k][i] + m.e[i][k]) * s;
|
|
this->w = q[3];
|
|
this->x = q[0];
|
|
this->y = q[1];
|
|
this->z = q[2];
|
|
}
|
|
}
|
|
|
|
Mat3::Mat3(const Mat4 &m)
|
|
{
|
|
for(int i = 0; i < 3; i++)
|
|
for(int j = 0; j < 3; j++)
|
|
this->e[i][j] = m.e[i][j];
|
|
}
|
|
|
|
Mat4::Mat4(const Mat3 &m)
|
|
{
|
|
for(int i = 0; i < 3; i++)
|
|
for(int j = 0; j < 3; j++)
|
|
this->e[i][j] = m.e[i][j];
|
|
this->e[0][3] = 0.0f;
|
|
this->e[1][3] = 0.0f;
|
|
this->e[2][3] = 0.0f;
|
|
this->e[3][3] = 1.0f;
|
|
this->e[3][2] = 0.0f;
|
|
this->e[3][1] = 0.0f;
|
|
this->e[3][0] = 0.0f;
|
|
}
|
|
|
|
Mat3
|
|
Mat3::rotation(const Quat &v)
|
|
{
|
|
Mat3 m(1.0f);
|
|
m.e[0][0] = v.w*v.w + v.x*v.x - v.y*v.y - v.z*v.z;
|
|
m.e[1][0] = 2*v.x*v.y - 2*v.w*v.z;
|
|
m.e[2][0] = 2*v.w*v.y + 2*v.x*v.z;
|
|
m.e[0][1] = 2*v.w*v.z + 2*v.x*v.y;
|
|
m.e[1][1] = v.w*v.w - v.x*v.x + v.y*v.y - v.z*v.z;
|
|
m.e[2][1] = 2*v.y*v.z - 2*v.w*v.x;
|
|
m.e[0][2] = 2*v.x*v.z - 2*v.w*v.y;
|
|
m.e[1][2] = 2*v.w*v.x + 2*v.y*v.z;
|
|
m.e[2][2] = v.w*v.w - v.x*v.x - v.y*v.y + v.z*v.z;
|
|
return m;
|
|
}
|
|
|
|
Mat4
|
|
Mat4::rotation(const Quat &q)
|
|
{
|
|
Mat4 m(1.0f);
|
|
m.e[0][0] = q.w*q.w + q.x*q.x - q.y*q.y - q.z*q.z;
|
|
m.e[1][0] = 2*q.x*q.y - 2*q.w*q.z;
|
|
m.e[2][0] = 2*q.w*q.y + 2*q.x*q.z;
|
|
m.e[0][1] = 2*q.w*q.z + 2*q.x*q.y;
|
|
m.e[1][1] = q.w*q.w - q.x*q.x + q.y*q.y - q.z*q.z;
|
|
m.e[2][1] = 2*q.y*q.z - 2*q.w*q.x;
|
|
m.e[0][2] = 2*q.x*q.z - 2*q.w*q.y;
|
|
m.e[1][2] = 2*q.w*q.x + 2*q.y*q.z;
|
|
m.e[2][2] = q.w*q.w - q.x*q.x - q.y*q.y + q.z*q.z;
|
|
return m;
|
|
}
|
|
|
|
Mat4
|
|
Mat4::transrot(const DQuat &q)
|
|
{
|
|
const Quat &q1 = q.q1;
|
|
const Quat &q2 = q.q2;
|
|
Mat4 m(1.0f);
|
|
m.e[0][0] = q1.w*q1.w + q1.x*q1.x - q1.y*q1.y - q1.z*q1.z;
|
|
m.e[1][0] = 2*q1.x*q1.y - 2*q1.w*q1.z;
|
|
m.e[2][0] = 2*q1.w*q1.y + 2*q1.x*q1.z;
|
|
m.e[0][1] = 2*q1.w*q1.z + 2*q1.x*q1.y;
|
|
m.e[1][1] = q1.w*q1.w - q1.x*q1.x + q1.y*q1.y - q1.z*q1.z;
|
|
m.e[2][1] = 2*q1.y*q1.z - 2*q1.w*q1.x;
|
|
m.e[0][2] = 2*q1.x*q1.z - 2*q1.w*q1.y;
|
|
m.e[1][2] = 2*q1.w*q1.x + 2*q1.y*q1.z;
|
|
m.e[2][2] = q1.w*q1.w - q1.x*q1.x - q1.y*q1.y + q1.z*q1.z;
|
|
m.e[3][0] = 2*(q1.w*q2.x - q2.w*q1.x + q1.y*q2.z - q1.z*q2.y);
|
|
m.e[3][1] = 2*(q1.w*q2.y - q2.w*q1.y + q1.z*q2.x - q1.x*q2.z);
|
|
m.e[3][2] = 2*(q1.w*q2.z - q2.w*q1.z + q1.x*q2.y - q1.y*q2.x);
|
|
return m;
|
|
}
|
|
|