mirror of https://github.com/axmolengine/axmol.git
911 lines
22 KiB
C++
911 lines
22 KiB
C++
/**
|
|
Copyright 2013 BlackBerry Inc.
|
|
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
|
Copyright (c) 2019-present Axmol Engine contributors (see AUTHORS.md).
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
|
|
Original file from GamePlay3D: http://gameplay3d.org
|
|
|
|
This file was modified to fit the axmol project
|
|
*/
|
|
|
|
#include "math/Mat4.h"
|
|
|
|
#include <cmath>
|
|
#include "math/Quaternion.h"
|
|
#include "math/MathUtil.h"
|
|
#include "base/Macros.h"
|
|
|
|
NS_AX_MATH_BEGIN
|
|
|
|
#if defined(AX_DLLEXPORT) || defined(AX_DLLIMPORT)
|
|
const Mat4 Mat4::IDENTITY =
|
|
Mat4(1.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f);
|
|
|
|
const Mat4 Mat4::ZERO = Mat4(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
|
|
#endif
|
|
|
|
|
|
void Mat4::createLookAt(const Vec3& eyePosition, const Vec3& targetPosition, const Vec3& up, Mat4* dst)
|
|
{
|
|
createLookAt(eyePosition.x, eyePosition.y, eyePosition.z, targetPosition.x, targetPosition.y, targetPosition.z,
|
|
up.x, up.y, up.z, dst);
|
|
}
|
|
|
|
void Mat4::createLookAt(float eyePositionX,
|
|
float eyePositionY,
|
|
float eyePositionZ,
|
|
float targetPositionX,
|
|
float targetPositionY,
|
|
float targetPositionZ,
|
|
float upX,
|
|
float upY,
|
|
float upZ,
|
|
Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
Vec3 eye(eyePositionX, eyePositionY, eyePositionZ);
|
|
Vec3 target(targetPositionX, targetPositionY, targetPositionZ);
|
|
Vec3 up(upX, upY, upZ);
|
|
up.normalize();
|
|
|
|
Vec3 zaxis;
|
|
Vec3::subtract(eye, target, &zaxis);
|
|
zaxis.normalize();
|
|
|
|
Vec3 xaxis;
|
|
Vec3::cross(up, zaxis, &xaxis);
|
|
xaxis.normalize();
|
|
|
|
Vec3 yaxis;
|
|
Vec3::cross(zaxis, xaxis, &yaxis);
|
|
yaxis.normalize();
|
|
|
|
dst->m[0] = xaxis.x;
|
|
dst->m[1] = yaxis.x;
|
|
dst->m[2] = zaxis.x;
|
|
dst->m[3] = 0.0f;
|
|
|
|
dst->m[4] = xaxis.y;
|
|
dst->m[5] = yaxis.y;
|
|
dst->m[6] = zaxis.y;
|
|
dst->m[7] = 0.0f;
|
|
|
|
dst->m[8] = xaxis.z;
|
|
dst->m[9] = yaxis.z;
|
|
dst->m[10] = zaxis.z;
|
|
dst->m[11] = 0.0f;
|
|
|
|
dst->m[12] = -Vec3::dot(xaxis, eye);
|
|
dst->m[13] = -Vec3::dot(yaxis, eye);
|
|
dst->m[14] = -Vec3::dot(zaxis, eye);
|
|
dst->m[15] = 1.0f;
|
|
}
|
|
|
|
void Mat4::createPerspective(float fieldOfView, float aspectRatio, float zNearPlane, float zFarPlane, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
AX_ASSERT(zFarPlane != zNearPlane);
|
|
|
|
float f_n = 1.0f / (zFarPlane - zNearPlane);
|
|
float theta = MATH_DEG_TO_RAD(fieldOfView) * 0.5f;
|
|
if (std::abs(std::fmod(theta, MATH_PIOVER2)) < MATH_EPSILON)
|
|
{
|
|
AXLOGE("Invalid field of view value ({}) causes attempted calculation tan({}), which is undefined.",
|
|
fieldOfView, theta);
|
|
return;
|
|
}
|
|
float divisor = std::tan(theta);
|
|
AX_ASSERT(divisor);
|
|
float factor = 1.0f / divisor;
|
|
|
|
memset(dst->m, 0, sizeof(dst->m));
|
|
|
|
AX_ASSERT(aspectRatio);
|
|
dst->m[0] = (1.0f / aspectRatio) * factor;
|
|
dst->m[5] = factor;
|
|
dst->m[10] = (-(zFarPlane + zNearPlane)) * f_n;
|
|
dst->m[11] = -1.0f;
|
|
dst->m[14] = -2.0f * zFarPlane * zNearPlane * f_n;
|
|
|
|
// https://metashapes.com/blog/opengl-metal-projection-matrix-problem/
|
|
#ifdef AX_USE_METAL
|
|
dst->m[10] = -(zFarPlane)*f_n;
|
|
dst->m[14] = -(zFarPlane * zNearPlane) * f_n;
|
|
#endif
|
|
}
|
|
|
|
void Mat4::createOrthographic(float width, float height, float zNearPlane, float zFarPlane, Mat4* dst)
|
|
{
|
|
float halfWidth = width / 2.0f;
|
|
float halfHeight = height / 2.0f;
|
|
createOrthographicOffCenter(-halfWidth, halfWidth, -halfHeight, halfHeight, zNearPlane, zFarPlane, dst);
|
|
}
|
|
|
|
void Mat4::createOrthographicOffCenter(float left,
|
|
float right,
|
|
float bottom,
|
|
float top,
|
|
float zNearPlane,
|
|
float zFarPlane,
|
|
Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
AX_ASSERT(right != left);
|
|
AX_ASSERT(top != bottom);
|
|
AX_ASSERT(zFarPlane != zNearPlane);
|
|
|
|
memset(dst->m, 0, sizeof(dst->m));
|
|
dst->m[0] = 2 / (right - left);
|
|
dst->m[5] = 2 / (top - bottom);
|
|
dst->m[10] = 2 / (zNearPlane - zFarPlane);
|
|
|
|
dst->m[12] = (left + right) / (left - right);
|
|
dst->m[13] = (top + bottom) / (bottom - top);
|
|
dst->m[14] = (zNearPlane + zFarPlane) / (zNearPlane - zFarPlane);
|
|
dst->m[15] = 1;
|
|
|
|
//// https://metashapes.com/blog/opengl-metal-projection-matrix-problem/
|
|
#ifdef AX_USE_METAL
|
|
dst->m[10] = 1 / (zNearPlane - zFarPlane);
|
|
dst->m[14] = zNearPlane / (zNearPlane - zFarPlane);
|
|
#endif
|
|
}
|
|
|
|
void Mat4::createBillboard(const Vec3& objectPosition,
|
|
const Vec3& cameraPosition,
|
|
const Vec3& cameraUpVector,
|
|
Mat4* dst)
|
|
{
|
|
createBillboardHelper(objectPosition, cameraPosition, cameraUpVector, nullptr, dst);
|
|
}
|
|
|
|
void Mat4::createBillboard(const Vec3& objectPosition,
|
|
const Vec3& cameraPosition,
|
|
const Vec3& cameraUpVector,
|
|
const Vec3& cameraForwardVector,
|
|
Mat4* dst)
|
|
{
|
|
createBillboardHelper(objectPosition, cameraPosition, cameraUpVector, &cameraForwardVector, dst);
|
|
}
|
|
|
|
void Mat4::createBillboardHelper(const Vec3& objectPosition,
|
|
const Vec3& cameraPosition,
|
|
const Vec3& cameraUpVector,
|
|
const Vec3* cameraForwardVector,
|
|
Mat4* dst)
|
|
{
|
|
Vec3 delta(objectPosition, cameraPosition);
|
|
bool isSufficientDelta = delta.lengthSquared() > MATH_EPSILON;
|
|
|
|
dst->setIdentity();
|
|
dst->m[3] = objectPosition.x;
|
|
dst->m[7] = objectPosition.y;
|
|
dst->m[11] = objectPosition.z;
|
|
|
|
// As per the contracts for the 2 variants of createBillboard, we need
|
|
// either a safe default or a sufficient distance between object and camera.
|
|
if (cameraForwardVector || isSufficientDelta)
|
|
{
|
|
Vec3 target = isSufficientDelta ? cameraPosition : (objectPosition - *cameraForwardVector);
|
|
|
|
// A billboard is the inverse of a lookAt rotation
|
|
Mat4 lookAt;
|
|
createLookAt(objectPosition, target, cameraUpVector, &lookAt);
|
|
dst->m[0] = lookAt.m[0];
|
|
dst->m[1] = lookAt.m[4];
|
|
dst->m[2] = lookAt.m[8];
|
|
dst->m[4] = lookAt.m[1];
|
|
dst->m[5] = lookAt.m[5];
|
|
dst->m[6] = lookAt.m[9];
|
|
dst->m[8] = lookAt.m[2];
|
|
dst->m[9] = lookAt.m[6];
|
|
dst->m[10] = lookAt.m[10];
|
|
}
|
|
}
|
|
|
|
// void Mat4::createReflection(const Plane& plane, Mat4* dst)
|
|
// {
|
|
// Vec3 normal(plane.getNormal());
|
|
// float k = -2.0f * plane.getDistance();
|
|
|
|
// dst->setIdentity();
|
|
|
|
// dst->m[0] -= 2.0f * normal.x * normal.x;
|
|
// dst->m[5] -= 2.0f * normal.y * normal.y;
|
|
// dst->m[10] -= 2.0f * normal.z * normal.z;
|
|
// dst->m[1] = dst->m[4] = -2.0f * normal.x * normal.y;
|
|
// dst->m[2] = dst->m[8] = -2.0f * normal.x * normal.z;
|
|
// dst->m[6] = dst->m[9] = -2.0f * normal.y * normal.z;
|
|
|
|
// dst->m[3] = k * normal.x;
|
|
// dst->m[7] = k * normal.y;
|
|
// dst->m[11] = k * normal.z;
|
|
// }
|
|
|
|
void Mat4::createScale(const Vec3& scale, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
dst->setIdentity();
|
|
|
|
dst->m[0] = scale.x;
|
|
dst->m[5] = scale.y;
|
|
dst->m[10] = scale.z;
|
|
}
|
|
|
|
void Mat4::createScale(float xScale, float yScale, float zScale, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
dst->setIdentity();
|
|
|
|
dst->m[0] = xScale;
|
|
dst->m[5] = yScale;
|
|
dst->m[10] = zScale;
|
|
}
|
|
|
|
void Mat4::createRotation(const Quaternion& q, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
float x2 = q.x + q.x;
|
|
float y2 = q.y + q.y;
|
|
float z2 = q.z + q.z;
|
|
|
|
float xx2 = q.x * x2;
|
|
float yy2 = q.y * y2;
|
|
float zz2 = q.z * z2;
|
|
float xy2 = q.x * y2;
|
|
float xz2 = q.x * z2;
|
|
float yz2 = q.y * z2;
|
|
float wx2 = q.w * x2;
|
|
float wy2 = q.w * y2;
|
|
float wz2 = q.w * z2;
|
|
|
|
dst->m[0] = 1.0f - yy2 - zz2;
|
|
dst->m[1] = xy2 + wz2;
|
|
dst->m[2] = xz2 - wy2;
|
|
dst->m[3] = 0.0f;
|
|
|
|
dst->m[4] = xy2 - wz2;
|
|
dst->m[5] = 1.0f - xx2 - zz2;
|
|
dst->m[6] = yz2 + wx2;
|
|
dst->m[7] = 0.0f;
|
|
|
|
dst->m[8] = xz2 + wy2;
|
|
dst->m[9] = yz2 - wx2;
|
|
dst->m[10] = 1.0f - xx2 - yy2;
|
|
dst->m[11] = 0.0f;
|
|
|
|
dst->m[12] = 0.0f;
|
|
dst->m[13] = 0.0f;
|
|
dst->m[14] = 0.0f;
|
|
dst->m[15] = 1.0f;
|
|
}
|
|
|
|
void Mat4::createRotation(const Vec3& axis, float angle, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
float x = axis.x;
|
|
float y = axis.y;
|
|
float z = axis.z;
|
|
|
|
// Make sure the input axis is normalized.
|
|
float n = x * x + y * y + z * z;
|
|
if (n != 1.0f)
|
|
{
|
|
// Not normalized.
|
|
n = std::sqrt(n);
|
|
// Prevent divide too close to zero.
|
|
if (n > 0.000001f)
|
|
{
|
|
n = 1.0f / n;
|
|
x *= n;
|
|
y *= n;
|
|
z *= n;
|
|
}
|
|
}
|
|
|
|
float c = std::cos(angle);
|
|
float s = std::sin(angle);
|
|
|
|
float t = 1.0f - c;
|
|
float tx = t * x;
|
|
float ty = t * y;
|
|
float tz = t * z;
|
|
float txy = tx * y;
|
|
float txz = tx * z;
|
|
float tyz = ty * z;
|
|
float sx = s * x;
|
|
float sy = s * y;
|
|
float sz = s * z;
|
|
|
|
dst->m[0] = c + tx * x;
|
|
dst->m[1] = txy + sz;
|
|
dst->m[2] = txz - sy;
|
|
dst->m[3] = 0.0f;
|
|
|
|
dst->m[4] = txy - sz;
|
|
dst->m[5] = c + ty * y;
|
|
dst->m[6] = tyz + sx;
|
|
dst->m[7] = 0.0f;
|
|
|
|
dst->m[8] = txz + sy;
|
|
dst->m[9] = tyz - sx;
|
|
dst->m[10] = c + tz * z;
|
|
dst->m[11] = 0.0f;
|
|
|
|
dst->m[12] = 0.0f;
|
|
dst->m[13] = 0.0f;
|
|
dst->m[14] = 0.0f;
|
|
dst->m[15] = 1.0f;
|
|
}
|
|
|
|
void Mat4::createRotationX(float angle, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
dst->setIdentity();
|
|
|
|
float c = std::cos(angle);
|
|
float s = std::sin(angle);
|
|
|
|
dst->m[5] = c;
|
|
dst->m[6] = s;
|
|
dst->m[9] = -s;
|
|
dst->m[10] = c;
|
|
}
|
|
|
|
void Mat4::createRotationY(float angle, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
dst->setIdentity();
|
|
|
|
float c = std::cos(angle);
|
|
float s = std::sin(angle);
|
|
|
|
dst->m[0] = c;
|
|
dst->m[2] = -s;
|
|
dst->m[8] = s;
|
|
dst->m[10] = c;
|
|
}
|
|
|
|
void Mat4::createRotationZ(float angle, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
dst->setIdentity();
|
|
|
|
float c = std::cos(angle);
|
|
float s = std::sin(angle);
|
|
|
|
dst->m[0] = c;
|
|
dst->m[1] = s;
|
|
dst->m[4] = -s;
|
|
dst->m[5] = c;
|
|
}
|
|
|
|
void Mat4::createTranslation(const Vec3& translation, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
dst->setIdentity();
|
|
|
|
dst->m[12] = translation.x;
|
|
dst->m[13] = translation.y;
|
|
dst->m[14] = translation.z;
|
|
}
|
|
|
|
void Mat4::createTranslation(float xTranslation, float yTranslation, float zTranslation, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
dst->setIdentity();
|
|
|
|
dst->m[12] = xTranslation;
|
|
dst->m[13] = yTranslation;
|
|
dst->m[14] = zTranslation;
|
|
}
|
|
|
|
void Mat4::add(float scalar)
|
|
{
|
|
add(scalar, this);
|
|
}
|
|
|
|
void Mat4::add(float scalar, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
MathUtil::addMatrix(m, scalar, dst->m);
|
|
}
|
|
|
|
void Mat4::add(const Mat4& mat)
|
|
{
|
|
add(*this, mat, this);
|
|
}
|
|
|
|
void Mat4::add(const Mat4& m1, const Mat4& m2, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
MathUtil::addMatrix(m1.m, m2.m, dst->m);
|
|
}
|
|
|
|
bool Mat4::decompose(Vec3* scale, Quaternion* rotation, Vec3* translation) const
|
|
{
|
|
if (translation)
|
|
{
|
|
// Extract the translation.
|
|
translation->x = m[12];
|
|
translation->y = m[13];
|
|
translation->z = m[14];
|
|
}
|
|
|
|
// Nothing left to do.
|
|
if (scale == nullptr && rotation == nullptr)
|
|
return true;
|
|
|
|
// Extract the scale.
|
|
// This is simply the length of each axis (row/column) in the matrix.
|
|
Vec3 xaxis(m[0], m[1], m[2]);
|
|
float scaleX = xaxis.length();
|
|
|
|
Vec3 yaxis(m[4], m[5], m[6]);
|
|
float scaleY = yaxis.length();
|
|
|
|
Vec3 zaxis(m[8], m[9], m[10]);
|
|
float scaleZ = zaxis.length();
|
|
|
|
// Determine if we have a negative scale (true if determinant is less than zero).
|
|
// In this case, we simply negate a single axis of the scale.
|
|
float det = determinant();
|
|
if (det < 0)
|
|
scaleZ = -scaleZ;
|
|
|
|
if (scale)
|
|
{
|
|
scale->x = scaleX;
|
|
scale->y = scaleY;
|
|
scale->z = scaleZ;
|
|
}
|
|
|
|
// Nothing left to do.
|
|
if (rotation == nullptr)
|
|
return true;
|
|
|
|
// Scale too close to zero, can't decompose rotation.
|
|
if (scaleX < MATH_TOLERANCE || scaleY < MATH_TOLERANCE || std::abs(scaleZ) < MATH_TOLERANCE)
|
|
return false;
|
|
|
|
float rn;
|
|
|
|
// Factor the scale out of the matrix axes.
|
|
rn = 1.0f / scaleX;
|
|
xaxis.x *= rn;
|
|
xaxis.y *= rn;
|
|
xaxis.z *= rn;
|
|
|
|
rn = 1.0f / scaleY;
|
|
yaxis.x *= rn;
|
|
yaxis.y *= rn;
|
|
yaxis.z *= rn;
|
|
|
|
rn = 1.0f / scaleZ;
|
|
zaxis.x *= rn;
|
|
zaxis.y *= rn;
|
|
zaxis.z *= rn;
|
|
|
|
// Now calculate the rotation from the resulting matrix (axes).
|
|
float trace = xaxis.x + yaxis.y + zaxis.z + 1.0f;
|
|
|
|
if (trace > MATH_EPSILON)
|
|
{
|
|
float s = 0.5f / std::sqrt(trace);
|
|
rotation->w = 0.25f / s;
|
|
rotation->x = (yaxis.z - zaxis.y) * s;
|
|
rotation->y = (zaxis.x - xaxis.z) * s;
|
|
rotation->z = (xaxis.y - yaxis.x) * s;
|
|
}
|
|
else
|
|
{
|
|
// Note: since xaxis, yaxis, and zaxis are normalized,
|
|
// we will never divide by zero in the code below.
|
|
if (xaxis.x > yaxis.y && xaxis.x > zaxis.z)
|
|
{
|
|
float s = 0.5f / std::sqrt(1.0f + xaxis.x - yaxis.y - zaxis.z);
|
|
rotation->w = (yaxis.z - zaxis.y) * s;
|
|
rotation->x = 0.25f / s;
|
|
rotation->y = (yaxis.x + xaxis.y) * s;
|
|
rotation->z = (zaxis.x + xaxis.z) * s;
|
|
}
|
|
else if (yaxis.y > zaxis.z)
|
|
{
|
|
float s = 0.5f / std::sqrt(1.0f + yaxis.y - xaxis.x - zaxis.z);
|
|
rotation->w = (zaxis.x - xaxis.z) * s;
|
|
rotation->x = (yaxis.x + xaxis.y) * s;
|
|
rotation->y = 0.25f / s;
|
|
rotation->z = (zaxis.y + yaxis.z) * s;
|
|
}
|
|
else
|
|
{
|
|
float s = 0.5f / std::sqrt(1.0f + zaxis.z - xaxis.x - yaxis.y);
|
|
rotation->w = (xaxis.y - yaxis.x) * s;
|
|
rotation->x = (zaxis.x + xaxis.z) * s;
|
|
rotation->y = (zaxis.y + yaxis.z) * s;
|
|
rotation->z = 0.25f / s;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
float Mat4::determinant() const
|
|
{
|
|
float a0 = m[0] * m[5] - m[1] * m[4];
|
|
float a1 = m[0] * m[6] - m[2] * m[4];
|
|
float a2 = m[0] * m[7] - m[3] * m[4];
|
|
float a3 = m[1] * m[6] - m[2] * m[5];
|
|
float a4 = m[1] * m[7] - m[3] * m[5];
|
|
float a5 = m[2] * m[7] - m[3] * m[6];
|
|
float b0 = m[8] * m[13] - m[9] * m[12];
|
|
float b1 = m[8] * m[14] - m[10] * m[12];
|
|
float b2 = m[8] * m[15] - m[11] * m[12];
|
|
float b3 = m[9] * m[14] - m[10] * m[13];
|
|
float b4 = m[9] * m[15] - m[11] * m[13];
|
|
float b5 = m[10] * m[15] - m[11] * m[14];
|
|
|
|
// Calculate the determinant.
|
|
return (a0 * b5 - a1 * b4 + a2 * b3 + a3 * b2 - a4 * b1 + a5 * b0);
|
|
}
|
|
|
|
void Mat4::getScale(Vec3* scale) const
|
|
{
|
|
decompose(scale, nullptr, nullptr);
|
|
}
|
|
|
|
bool Mat4::getRotation(Quaternion* rotation) const
|
|
{
|
|
return decompose(nullptr, rotation, nullptr);
|
|
}
|
|
|
|
void Mat4::getTranslation(Vec3* translation) const
|
|
{
|
|
decompose(nullptr, nullptr, translation);
|
|
}
|
|
|
|
void Mat4::getUpVector(Vec3* dst) const
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
dst->x = m[4];
|
|
dst->y = m[5];
|
|
dst->z = m[6];
|
|
}
|
|
|
|
void Mat4::getDownVector(Vec3* dst) const
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
dst->x = -m[4];
|
|
dst->y = -m[5];
|
|
dst->z = -m[6];
|
|
}
|
|
|
|
void Mat4::getLeftVector(Vec3* dst) const
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
dst->x = -m[0];
|
|
dst->y = -m[1];
|
|
dst->z = -m[2];
|
|
}
|
|
|
|
void Mat4::getRightVector(Vec3* dst) const
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
dst->x = m[0];
|
|
dst->y = m[1];
|
|
dst->z = m[2];
|
|
}
|
|
|
|
void Mat4::getForwardVector(Vec3* dst) const
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
dst->x = -m[8];
|
|
dst->y = -m[9];
|
|
dst->z = -m[10];
|
|
}
|
|
|
|
void Mat4::getBackVector(Vec3* dst) const
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
dst->x = m[8];
|
|
dst->y = m[9];
|
|
dst->z = m[10];
|
|
}
|
|
|
|
Mat4 Mat4::getInversed() const
|
|
{
|
|
Mat4 mat(*this);
|
|
mat.inverse();
|
|
return mat;
|
|
}
|
|
|
|
bool Mat4::inverse()
|
|
{
|
|
float a0 = m[0] * m[5] - m[1] * m[4];
|
|
float a1 = m[0] * m[6] - m[2] * m[4];
|
|
float a2 = m[0] * m[7] - m[3] * m[4];
|
|
float a3 = m[1] * m[6] - m[2] * m[5];
|
|
float a4 = m[1] * m[7] - m[3] * m[5];
|
|
float a5 = m[2] * m[7] - m[3] * m[6];
|
|
float b0 = m[8] * m[13] - m[9] * m[12];
|
|
float b1 = m[8] * m[14] - m[10] * m[12];
|
|
float b2 = m[8] * m[15] - m[11] * m[12];
|
|
float b3 = m[9] * m[14] - m[10] * m[13];
|
|
float b4 = m[9] * m[15] - m[11] * m[13];
|
|
float b5 = m[10] * m[15] - m[11] * m[14];
|
|
|
|
// Calculate the determinant.
|
|
float det = a0 * b5 - a1 * b4 + a2 * b3 + a3 * b2 - a4 * b1 + a5 * b0;
|
|
|
|
// Close to zero, can't invert.
|
|
if (std::abs(det) <= MATH_TOLERANCE)
|
|
return false;
|
|
|
|
// Support the case where m == dst.
|
|
Mat4 inverse;
|
|
inverse.m[0] = m[5] * b5 - m[6] * b4 + m[7] * b3;
|
|
inverse.m[1] = -m[1] * b5 + m[2] * b4 - m[3] * b3;
|
|
inverse.m[2] = m[13] * a5 - m[14] * a4 + m[15] * a3;
|
|
inverse.m[3] = -m[9] * a5 + m[10] * a4 - m[11] * a3;
|
|
|
|
inverse.m[4] = -m[4] * b5 + m[6] * b2 - m[7] * b1;
|
|
inverse.m[5] = m[0] * b5 - m[2] * b2 + m[3] * b1;
|
|
inverse.m[6] = -m[12] * a5 + m[14] * a2 - m[15] * a1;
|
|
inverse.m[7] = m[8] * a5 - m[10] * a2 + m[11] * a1;
|
|
|
|
inverse.m[8] = m[4] * b4 - m[5] * b2 + m[7] * b0;
|
|
inverse.m[9] = -m[0] * b4 + m[1] * b2 - m[3] * b0;
|
|
inverse.m[10] = m[12] * a4 - m[13] * a2 + m[15] * a0;
|
|
inverse.m[11] = -m[8] * a4 + m[9] * a2 - m[11] * a0;
|
|
|
|
inverse.m[12] = -m[4] * b3 + m[5] * b1 - m[6] * b0;
|
|
inverse.m[13] = m[0] * b3 - m[1] * b1 + m[2] * b0;
|
|
inverse.m[14] = -m[12] * a3 + m[13] * a1 - m[14] * a0;
|
|
inverse.m[15] = m[8] * a3 - m[9] * a1 + m[10] * a0;
|
|
|
|
multiply(inverse, 1.0f / det, this);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool Mat4::isIdentity() const
|
|
{
|
|
return (memcmp(m, IDENTITY.m, sizeof(m)) == 0);
|
|
}
|
|
|
|
void Mat4::multiply(float scalar)
|
|
{
|
|
multiply(scalar, this);
|
|
}
|
|
|
|
void Mat4::multiply(float scalar, Mat4* dst) const
|
|
{
|
|
multiply(*this, scalar, dst);
|
|
}
|
|
|
|
void Mat4::multiply(const Mat4& m, float scalar, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
MathUtil::multiplyMatrix(m.m, scalar, dst->m);
|
|
}
|
|
|
|
void Mat4::multiply(const Mat4& mat)
|
|
{
|
|
multiply(*this, mat, this);
|
|
}
|
|
|
|
void Mat4::multiply(const Mat4& m1, const Mat4& m2, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
MathUtil::multiplyMatrix(m1.m, m2.m, dst->m);
|
|
}
|
|
|
|
void Mat4::negate()
|
|
{
|
|
MathUtil::negateMatrix(m, m);
|
|
}
|
|
|
|
Mat4 Mat4::getNegated() const
|
|
{
|
|
Mat4 mat(*this);
|
|
mat.negate();
|
|
return mat;
|
|
}
|
|
|
|
void Mat4::rotate(const Quaternion& q)
|
|
{
|
|
rotate(q, this);
|
|
}
|
|
|
|
void Mat4::rotate(const Quaternion& q, Mat4* dst) const
|
|
{
|
|
Mat4 r;
|
|
createRotation(q, &r);
|
|
multiply(*this, r, dst);
|
|
}
|
|
|
|
void Mat4::rotate(const Vec3& axis, float angle)
|
|
{
|
|
rotate(axis, angle, this);
|
|
}
|
|
|
|
void Mat4::rotate(const Vec3& axis, float angle, Mat4* dst) const
|
|
{
|
|
Mat4 r;
|
|
createRotation(axis, angle, &r);
|
|
multiply(*this, r, dst);
|
|
}
|
|
|
|
void Mat4::rotateX(float angle)
|
|
{
|
|
rotateX(angle, this);
|
|
}
|
|
|
|
void Mat4::rotateX(float angle, Mat4* dst) const
|
|
{
|
|
Mat4 r;
|
|
createRotationX(angle, &r);
|
|
multiply(*this, r, dst);
|
|
}
|
|
|
|
void Mat4::rotateY(float angle)
|
|
{
|
|
rotateY(angle, this);
|
|
}
|
|
|
|
void Mat4::rotateY(float angle, Mat4* dst) const
|
|
{
|
|
Mat4 r;
|
|
createRotationY(angle, &r);
|
|
multiply(*this, r, dst);
|
|
}
|
|
|
|
void Mat4::rotateZ(float angle)
|
|
{
|
|
rotateZ(angle, this);
|
|
}
|
|
|
|
void Mat4::rotateZ(float angle, Mat4* dst) const
|
|
{
|
|
Mat4 r;
|
|
createRotationZ(angle, &r);
|
|
multiply(*this, r, dst);
|
|
}
|
|
|
|
void Mat4::scale(float value)
|
|
{
|
|
scale(value, this);
|
|
}
|
|
|
|
void Mat4::scale(float value, Mat4* dst) const
|
|
{
|
|
scale(value, value, value, dst);
|
|
}
|
|
|
|
void Mat4::scale(float xScale, float yScale, float zScale)
|
|
{
|
|
scale(xScale, yScale, zScale, this);
|
|
}
|
|
|
|
void Mat4::scale(float xScale, float yScale, float zScale, Mat4* dst) const
|
|
{
|
|
Mat4 s;
|
|
createScale(xScale, yScale, zScale, &s);
|
|
multiply(*this, s, dst);
|
|
}
|
|
|
|
void Mat4::scale(const Vec3& s)
|
|
{
|
|
scale(s.x, s.y, s.z, this);
|
|
}
|
|
|
|
void Mat4::scale(const Vec3& s, Mat4* dst) const
|
|
{
|
|
scale(s.x, s.y, s.z, dst);
|
|
}
|
|
|
|
void Mat4::subtract(const Mat4& mat)
|
|
{
|
|
subtract(*this, mat, this);
|
|
}
|
|
|
|
void Mat4::subtract(const Mat4& m1, const Mat4& m2, Mat4* dst)
|
|
{
|
|
AX_ASSERT(dst);
|
|
MathUtil::subtractMatrix(m1.m, m2.m, dst->m);
|
|
}
|
|
|
|
void Mat4::transformVector(Vec3* vector) const
|
|
{
|
|
AX_ASSERT(vector);
|
|
transformVector(vector->x, vector->y, vector->z, 0.0f, vector);
|
|
}
|
|
|
|
void Mat4::transformVector(const Vec3& vector, Vec3* dst) const
|
|
{
|
|
transformVector(vector.x, vector.y, vector.z, 0.0f, dst);
|
|
}
|
|
|
|
void Mat4::transformVector(float x, float y, float z, float w, Vec3* dst) const
|
|
{
|
|
AX_ASSERT(dst);
|
|
|
|
MathUtil::transformVec4(m, x, y, z, w, reinterpret_cast<float*>(dst));
|
|
}
|
|
|
|
void Mat4::transformVector(Vec4* vector) const
|
|
{
|
|
AX_ASSERT(vector);
|
|
transformVector(*vector, vector);
|
|
}
|
|
|
|
void Mat4::transformVector(const Vec4& vector, Vec4* dst) const
|
|
{
|
|
AX_ASSERT(dst);
|
|
MathUtil::transformVec4(m, reinterpret_cast<const float*>(&vector), reinterpret_cast<float*>(dst));
|
|
}
|
|
|
|
void Mat4::translate(float x, float y, float z)
|
|
{
|
|
translate(x, y, z, this);
|
|
}
|
|
|
|
void Mat4::translate(float x, float y, float z, Mat4* dst) const
|
|
{
|
|
Mat4 t;
|
|
createTranslation(x, y, z, &t);
|
|
multiply(*this, t, dst);
|
|
}
|
|
|
|
void Mat4::translate(const Vec3& t)
|
|
{
|
|
translate(t.x, t.y, t.z, this);
|
|
}
|
|
|
|
void Mat4::translate(const Vec3& t, Mat4* dst) const
|
|
{
|
|
translate(t.x, t.y, t.z, dst);
|
|
}
|
|
|
|
void Mat4::transpose()
|
|
{
|
|
MathUtil::transposeMatrix(m, m);
|
|
}
|
|
|
|
Mat4 Mat4::getTransposed() const
|
|
{
|
|
Mat4 mat(*this);
|
|
mat.transpose();
|
|
return mat;
|
|
}
|
|
|
|
NS_AX_MATH_END
|