#ifndef Magnum_Math_Matrix4_h #define Magnum_Math_Matrix4_h /* Copyright © 2010, 2011, 2012 Vladimír Vondruš This file is part of Magnum. Magnum is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License version 3 only, as published by the Free Software Foundation. Magnum is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License version 3 for more details. */ /** @file * @brief Class Magnum::Math::Matrix4 */ #include "Matrix3.h" #include "Vector4.h" namespace Magnum { namespace Math { /** * @brief 4x4 matrix * * @todo Rotation with Euler angles * @todo Shearing * @todo Reflection */ template class Matrix4: public Matrix { public: /** * @brief Translation matrix * @param vec Translation vector */ static Matrix4 translation(const Vector3& vec) { Matrix4 out; /* (Identity matrix) */ out.set(0, 3, vec.x()); out.set(1, 3, vec.y()); out.set(2, 3, vec.z()); return out; } /** * @brief Scaling matrix * @param vec Scaling vector */ static Matrix4 scaling(const Vector3& vec) { Matrix4 out; /* (Identity matrix) */ out.set(0, 0, vec.x()); out.set(1, 1, vec.y()); out.set(2, 2, vec.z()); return out; } /** * @brief Rotation matrix * @param angle Rotation angle (counterclockwise, in radians) * @param vec Rotation vector */ static Matrix4 rotation(T angle, const Vector3& vec) { Vector3 vn = vec.normalized(); T sine = sin(angle); T cosine = cos(angle); T oneMinusCosine = T(1) - cosine; T xx = vn.x()*vn.x(); T xy = vn.x()*vn.y(); T xz = vn.x()*vn.z(); T yy = vn.y()*vn.y(); T yz = vn.y()*vn.z(); T zz = vn.z()*vn.z(); /* Not creating identity matrix, as nearly all ones would be overwritten */ Matrix4 out(false); out.set(3, 3, T(1)); out.set(0, 0, cosine + xx*oneMinusCosine); out.set(0, 1, xy*oneMinusCosine - vn.z()*sine); out.set(0, 2, xz*oneMinusCosine + vn.y()*sine); out.set(1, 0, xy*oneMinusCosine + vn.z()*sine); out.set(1, 1, cosine + yy*oneMinusCosine); out.set(1, 2, yz*oneMinusCosine - vn.x()*sine); out.set(2, 0, xz*oneMinusCosine - vn.y()*sine); out.set(2, 1, yz*oneMinusCosine + vn.x()*sine); out.set(2, 2, cosine + zz*oneMinusCosine); return out; } /** @copydoc Matrix::Matrix(bool) */ inline Matrix4(bool identity = true): Matrix(identity) {} /** @copydoc Matrix::Matrix(const T*) */ inline Matrix4(const T* data): Matrix(data) {} /** @copydoc Matrix::Matrix(const Matrix&) */ inline Matrix4(const Matrix& other): Matrix(other) {} /** @copydoc Matrix::operator=() */ inline Matrix4& operator=(const Matrix& other) { return Matrix::operator=(other); } /** @copydoc Matrix::at(size_t) const */ inline Vector4 at(size_t col) const { return Matrix::at(col); } /** @copydoc Matrix::at(size_t, size_t) const */ inline T at(size_t row, size_t col) const { return Matrix::at(row, col); } /** @copydoc Matrix::operator*(const Matrix&) const */ inline Matrix4 operator*(const Matrix& other) const { return Matrix::operator*(other); } /** @copydoc Matrix::operator*(const Vector&) const */ inline Vector4 operator*(const Vector& other) const { return Matrix::operator*(other); } /** @copydoc Matrix::transposed() */ inline Matrix4 transposed() const { return Matrix::transposed(); } /** @copydoc Matrix::ij() */ inline Matrix3 ij(size_t skipRow, size_t skipCol) const { return Matrix::ij(skipRow, skipCol); } /** @copydoc Matrix::inverse() */ inline Matrix4 inverse() const { return Matrix::inverse(); } }; #ifndef DOXYGEN_GENERATING_OUTPUT template Corrade::Utility::Debug operator<<(Corrade::Utility::Debug debug, const Magnum::Math::Matrix4& value) { return debug << static_cast&>(value); } #endif }} #endif