qgenericmatrix.h 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402
  1. /****************************************************************************
  2. **
  3. ** Copyright (C) 2016 The Qt Company Ltd.
  4. ** Contact: https://www.qt.io/licensing/
  5. **
  6. ** This file is part of the QtGui module of the Qt Toolkit.
  7. **
  8. ** $QT_BEGIN_LICENSE:LGPL$
  9. ** Commercial License Usage
  10. ** Licensees holding valid commercial Qt licenses may use this file in
  11. ** accordance with the commercial license agreement provided with the
  12. ** Software or, alternatively, in accordance with the terms contained in
  13. ** a written agreement between you and The Qt Company. For licensing terms
  14. ** and conditions see https://www.qt.io/terms-conditions. For further
  15. ** information use the contact form at https://www.qt.io/contact-us.
  16. **
  17. ** GNU Lesser General Public License Usage
  18. ** Alternatively, this file may be used under the terms of the GNU Lesser
  19. ** General Public License version 3 as published by the Free Software
  20. ** Foundation and appearing in the file LICENSE.LGPL3 included in the
  21. ** packaging of this file. Please review the following information to
  22. ** ensure the GNU Lesser General Public License version 3 requirements
  23. ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
  24. **
  25. ** GNU General Public License Usage
  26. ** Alternatively, this file may be used under the terms of the GNU
  27. ** General Public License version 2.0 or (at your option) the GNU General
  28. ** Public license version 3 or any later version approved by the KDE Free
  29. ** Qt Foundation. The licenses are as published by the Free Software
  30. ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
  31. ** included in the packaging of this file. Please review the following
  32. ** information to ensure the GNU General Public License requirements will
  33. ** be met: https://www.gnu.org/licenses/gpl-2.0.html and
  34. ** https://www.gnu.org/licenses/gpl-3.0.html.
  35. **
  36. ** $QT_END_LICENSE$
  37. **
  38. ****************************************************************************/
  39. #ifndef QGENERICMATRIX_H
  40. #define QGENERICMATRIX_H
  41. #include <QtCore/qmetatype.h>
  42. #include <QtCore/qdebug.h>
  43. #include <QtCore/qdatastream.h>
  44. QT_BEGIN_NAMESPACE
  45. template <int N, int M, typename T>
  46. class QGenericMatrix
  47. {
  48. public:
  49. QGenericMatrix();
  50. explicit QGenericMatrix(Qt::Initialization) {}
  51. explicit QGenericMatrix(const T *values);
  52. const T& operator()(int row, int column) const;
  53. T& operator()(int row, int column);
  54. bool isIdentity() const;
  55. void setToIdentity();
  56. void fill(T value);
  57. QGenericMatrix<M, N, T> transposed() const Q_REQUIRED_RESULT;
  58. QGenericMatrix<N, M, T>& operator+=(const QGenericMatrix<N, M, T>& other);
  59. QGenericMatrix<N, M, T>& operator-=(const QGenericMatrix<N, M, T>& other);
  60. QGenericMatrix<N, M, T>& operator*=(T factor);
  61. QGenericMatrix<N, M, T>& operator/=(T divisor);
  62. bool operator==(const QGenericMatrix<N, M, T>& other) const;
  63. bool operator!=(const QGenericMatrix<N, M, T>& other) const;
  64. void copyDataTo(T *values) const;
  65. T *data() { return *m; }
  66. const T *data() const { return *m; }
  67. const T *constData() const { return *m; }
  68. #if !defined(Q_NO_TEMPLATE_FRIENDS)
  69. template<int NN, int MM, typename TT>
  70. friend QGenericMatrix<NN, MM, TT> operator+(const QGenericMatrix<NN, MM, TT>& m1, const QGenericMatrix<NN, MM, TT>& m2);
  71. template<int NN, int MM, typename TT>
  72. friend QGenericMatrix<NN, MM, TT> operator-(const QGenericMatrix<NN, MM, TT>& m1, const QGenericMatrix<NN, MM, TT>& m2);
  73. template<int NN, int M1, int M2, typename TT>
  74. friend QGenericMatrix<M1, M2, TT> operator*(const QGenericMatrix<NN, M2, TT>& m1, const QGenericMatrix<M1, NN, TT>& m2);
  75. template<int NN, int MM, typename TT>
  76. friend QGenericMatrix<NN, MM, TT> operator-(const QGenericMatrix<NN, MM, TT>& matrix);
  77. template<int NN, int MM, typename TT>
  78. friend QGenericMatrix<NN, MM, TT> operator*(TT factor, const QGenericMatrix<NN, MM, TT>& matrix);
  79. template<int NN, int MM, typename TT>
  80. friend QGenericMatrix<NN, MM, TT> operator*(const QGenericMatrix<NN, MM, TT>& matrix, TT factor);
  81. template<int NN, int MM, typename TT>
  82. friend QGenericMatrix<NN, MM, TT> operator/(const QGenericMatrix<NN, MM, TT>& matrix, TT divisor);
  83. private:
  84. #endif
  85. T m[N][M]; // Column-major order to match OpenGL.
  86. #if !defined(Q_NO_TEMPLATE_FRIENDS)
  87. template <int NN, int MM, typename TT>
  88. friend class QGenericMatrix;
  89. #endif
  90. };
  91. template <int N, int M, typename T>
  92. class QTypeInfo<QGenericMatrix<N, M, T> >
  93. : public QTypeInfoMerger<QGenericMatrix<N, M, T>, T>
  94. {
  95. #if QT_VERSION < QT_VERSION_CHECK(6,0,0)
  96. public:
  97. enum {
  98. isStatic = true,
  99. }; // at least Q_RELOCATABLE_TYPE, for BC during Qt 5
  100. #endif
  101. };
  102. template <int N, int M, typename T>
  103. Q_INLINE_TEMPLATE QGenericMatrix<N, M, T>::QGenericMatrix()
  104. {
  105. setToIdentity();
  106. }
  107. template <int N, int M, typename T>
  108. Q_OUTOFLINE_TEMPLATE QGenericMatrix<N, M, T>::QGenericMatrix(const T *values)
  109. {
  110. for (int col = 0; col < N; ++col)
  111. for (int row = 0; row < M; ++row)
  112. m[col][row] = values[row * N + col];
  113. }
  114. template <int N, int M, typename T>
  115. Q_INLINE_TEMPLATE const T& QGenericMatrix<N, M, T>::operator()(int row, int column) const
  116. {
  117. Q_ASSERT(row >= 0 && row < M && column >= 0 && column < N);
  118. return m[column][row];
  119. }
  120. template <int N, int M, typename T>
  121. Q_INLINE_TEMPLATE T& QGenericMatrix<N, M, T>::operator()(int row, int column)
  122. {
  123. Q_ASSERT(row >= 0 && row < M && column >= 0 && column < N);
  124. return m[column][row];
  125. }
  126. template <int N, int M, typename T>
  127. Q_OUTOFLINE_TEMPLATE bool QGenericMatrix<N, M, T>::isIdentity() const
  128. {
  129. for (int col = 0; col < N; ++col) {
  130. for (int row = 0; row < M; ++row) {
  131. if (row == col) {
  132. if (m[col][row] != 1.0f)
  133. return false;
  134. } else {
  135. if (m[col][row] != 0.0f)
  136. return false;
  137. }
  138. }
  139. }
  140. return true;
  141. }
  142. template <int N, int M, typename T>
  143. Q_OUTOFLINE_TEMPLATE void QGenericMatrix<N, M, T>::setToIdentity()
  144. {
  145. for (int col = 0; col < N; ++col) {
  146. for (int row = 0; row < M; ++row) {
  147. if (row == col)
  148. m[col][row] = 1.0f;
  149. else
  150. m[col][row] = 0.0f;
  151. }
  152. }
  153. }
  154. template <int N, int M, typename T>
  155. Q_OUTOFLINE_TEMPLATE void QGenericMatrix<N, M, T>::fill(T value)
  156. {
  157. for (int col = 0; col < N; ++col)
  158. for (int row = 0; row < M; ++row)
  159. m[col][row] = value;
  160. }
  161. template <int N, int M, typename T>
  162. Q_OUTOFLINE_TEMPLATE QGenericMatrix<M, N, T> QGenericMatrix<N, M, T>::transposed() const
  163. {
  164. QGenericMatrix<M, N, T> result(Qt::Uninitialized);
  165. for (int row = 0; row < M; ++row)
  166. for (int col = 0; col < N; ++col)
  167. result.m[row][col] = m[col][row];
  168. return result;
  169. }
  170. template <int N, int M, typename T>
  171. Q_OUTOFLINE_TEMPLATE QGenericMatrix<N, M, T>& QGenericMatrix<N, M, T>::operator+=(const QGenericMatrix<N, M, T>& other)
  172. {
  173. for (int row = 0; row < M; ++row)
  174. for (int col = 0; col < N; ++col)
  175. m[col][row] += other.m[col][row];
  176. return *this;
  177. }
  178. template <int N, int M, typename T>
  179. Q_OUTOFLINE_TEMPLATE QGenericMatrix<N, M, T>& QGenericMatrix<N, M, T>::operator-=(const QGenericMatrix<N, M, T>& other)
  180. {
  181. for (int row = 0; row < M; ++row)
  182. for (int col = 0; col < N; ++col)
  183. m[col][row] -= other.m[col][row];
  184. return *this;
  185. }
  186. template <int N, int M, typename T>
  187. Q_OUTOFLINE_TEMPLATE QGenericMatrix<N, M, T>& QGenericMatrix<N, M, T>::operator*=(T factor)
  188. {
  189. for (int row = 0; row < M; ++row)
  190. for (int col = 0; col < N; ++col)
  191. m[col][row] *= factor;
  192. return *this;
  193. }
  194. template <int N, int M, typename T>
  195. Q_OUTOFLINE_TEMPLATE bool QGenericMatrix<N, M, T>::operator==(const QGenericMatrix<N, M, T>& other) const
  196. {
  197. for (int row = 0; row < M; ++row)
  198. for (int col = 0; col < N; ++col) {
  199. if (m[col][row] != other.m[col][row])
  200. return false;
  201. }
  202. return true;
  203. }
  204. template <int N, int M, typename T>
  205. Q_OUTOFLINE_TEMPLATE bool QGenericMatrix<N, M, T>::operator!=(const QGenericMatrix<N, M, T>& other) const
  206. {
  207. for (int row = 0; row < M; ++row)
  208. for (int col = 0; col < N; ++col) {
  209. if (m[col][row] != other.m[col][row])
  210. return true;
  211. }
  212. return false;
  213. }
  214. template <int N, int M, typename T>
  215. Q_OUTOFLINE_TEMPLATE QGenericMatrix<N, M, T>& QGenericMatrix<N, M, T>::operator/=(T divisor)
  216. {
  217. for (int row = 0; row < M; ++row)
  218. for (int col = 0; col < N; ++col)
  219. m[col][row] /= divisor;
  220. return *this;
  221. }
  222. template <int N, int M, typename T>
  223. Q_OUTOFLINE_TEMPLATE QGenericMatrix<N, M, T> operator+(const QGenericMatrix<N, M, T>& m1, const QGenericMatrix<N, M, T>& m2)
  224. {
  225. QGenericMatrix<N, M, T> result(Qt::Uninitialized);
  226. for (int row = 0; row < M; ++row)
  227. for (int col = 0; col < N; ++col)
  228. result.m[col][row] = m1.m[col][row] + m2.m[col][row];
  229. return result;
  230. }
  231. template <int N, int M, typename T>
  232. Q_OUTOFLINE_TEMPLATE QGenericMatrix<N, M, T> operator-(const QGenericMatrix<N, M, T>& m1, const QGenericMatrix<N, M, T>& m2)
  233. {
  234. QGenericMatrix<N, M, T> result(Qt::Uninitialized);
  235. for (int row = 0; row < M; ++row)
  236. for (int col = 0; col < N; ++col)
  237. result.m[col][row] = m1.m[col][row] - m2.m[col][row];
  238. return result;
  239. }
  240. template <int N, int M1, int M2, typename T>
  241. Q_OUTOFLINE_TEMPLATE QGenericMatrix<M1, M2, T> operator*(const QGenericMatrix<N, M2, T>& m1, const QGenericMatrix<M1, N, T>& m2)
  242. {
  243. QGenericMatrix<M1, M2, T> result(Qt::Uninitialized);
  244. for (int row = 0; row < M2; ++row) {
  245. for (int col = 0; col < M1; ++col) {
  246. T sum(0.0f);
  247. for (int j = 0; j < N; ++j)
  248. sum += m1.m[j][row] * m2.m[col][j];
  249. result.m[col][row] = sum;
  250. }
  251. }
  252. return result;
  253. }
  254. template <int N, int M, typename T>
  255. Q_OUTOFLINE_TEMPLATE QGenericMatrix<N, M, T> operator-(const QGenericMatrix<N, M, T>& matrix)
  256. {
  257. QGenericMatrix<N, M, T> result(Qt::Uninitialized);
  258. for (int row = 0; row < M; ++row)
  259. for (int col = 0; col < N; ++col)
  260. result.m[col][row] = -matrix.m[col][row];
  261. return result;
  262. }
  263. template <int N, int M, typename T>
  264. Q_OUTOFLINE_TEMPLATE QGenericMatrix<N, M, T> operator*(T factor, const QGenericMatrix<N, M, T>& matrix)
  265. {
  266. QGenericMatrix<N, M, T> result(Qt::Uninitialized);
  267. for (int row = 0; row < M; ++row)
  268. for (int col = 0; col < N; ++col)
  269. result.m[col][row] = matrix.m[col][row] * factor;
  270. return result;
  271. }
  272. template <int N, int M, typename T>
  273. Q_OUTOFLINE_TEMPLATE QGenericMatrix<N, M, T> operator*(const QGenericMatrix<N, M, T>& matrix, T factor)
  274. {
  275. QGenericMatrix<N, M, T> result(Qt::Uninitialized);
  276. for (int row = 0; row < M; ++row)
  277. for (int col = 0; col < N; ++col)
  278. result.m[col][row] = matrix.m[col][row] * factor;
  279. return result;
  280. }
  281. template <int N, int M, typename T>
  282. Q_OUTOFLINE_TEMPLATE QGenericMatrix<N, M, T> operator/(const QGenericMatrix<N, M, T>& matrix, T divisor)
  283. {
  284. QGenericMatrix<N, M, T> result(Qt::Uninitialized);
  285. for (int row = 0; row < M; ++row)
  286. for (int col = 0; col < N; ++col)
  287. result.m[col][row] = matrix.m[col][row] / divisor;
  288. return result;
  289. }
  290. template <int N, int M, typename T>
  291. Q_OUTOFLINE_TEMPLATE void QGenericMatrix<N, M, T>::copyDataTo(T *values) const
  292. {
  293. for (int col = 0; col < N; ++col)
  294. for (int row = 0; row < M; ++row)
  295. values[row * N + col] = T(m[col][row]);
  296. }
  297. // Define aliases for the useful variants of QGenericMatrix.
  298. typedef QGenericMatrix<2, 2, float> QMatrix2x2;
  299. typedef QGenericMatrix<2, 3, float> QMatrix2x3;
  300. typedef QGenericMatrix<2, 4, float> QMatrix2x4;
  301. typedef QGenericMatrix<3, 2, float> QMatrix3x2;
  302. typedef QGenericMatrix<3, 3, float> QMatrix3x3;
  303. typedef QGenericMatrix<3, 4, float> QMatrix3x4;
  304. typedef QGenericMatrix<4, 2, float> QMatrix4x2;
  305. typedef QGenericMatrix<4, 3, float> QMatrix4x3;
  306. #ifndef QT_NO_DEBUG_STREAM
  307. template <int N, int M, typename T>
  308. QDebug operator<<(QDebug dbg, const QGenericMatrix<N, M, T> &m)
  309. {
  310. QDebugStateSaver saver(dbg);
  311. dbg.nospace() << "QGenericMatrix<" << N << ", " << M
  312. << ", " << QTypeInfo<T>::name()
  313. << ">(" << endl << qSetFieldWidth(10);
  314. for (int row = 0; row < M; ++row) {
  315. for (int col = 0; col < N; ++col)
  316. dbg << m(row, col);
  317. dbg << endl;
  318. }
  319. dbg << qSetFieldWidth(0) << ')';
  320. return dbg;
  321. }
  322. #endif
  323. #ifndef QT_NO_DATASTREAM
  324. template <int N, int M, typename T>
  325. QDataStream &operator<<(QDataStream &stream, const QGenericMatrix<N, M, T> &matrix)
  326. {
  327. for (int row = 0; row < M; ++row)
  328. for (int col = 0; col < N; ++col)
  329. stream << double(matrix(row, col));
  330. return stream;
  331. }
  332. template <int N, int M, typename T>
  333. QDataStream &operator>>(QDataStream &stream, QGenericMatrix<N, M, T> &matrix)
  334. {
  335. double x;
  336. for (int row = 0; row < M; ++row) {
  337. for (int col = 0; col < N; ++col) {
  338. stream >> x;
  339. matrix(row, col) = T(x);
  340. }
  341. }
  342. return stream;
  343. }
  344. #endif
  345. QT_END_NAMESPACE
  346. Q_DECLARE_METATYPE(QMatrix2x2)
  347. Q_DECLARE_METATYPE(QMatrix2x3)
  348. Q_DECLARE_METATYPE(QMatrix2x4)
  349. Q_DECLARE_METATYPE(QMatrix3x2)
  350. Q_DECLARE_METATYPE(QMatrix3x3)
  351. Q_DECLARE_METATYPE(QMatrix3x4)
  352. Q_DECLARE_METATYPE(QMatrix4x2)
  353. Q_DECLARE_METATYPE(QMatrix4x3)
  354. #endif