Matrix3.tests.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500
  1. /**
  2. * @author bhouston / http://exocortex.com
  3. * @author TristanVALCKE / https://github.com/Itee
  4. */
  5. /* global QUnit */
  6. import { Matrix3 } from '../../../../src/math/Matrix3';
  7. import { Matrix4 } from '../../../../src/math/Matrix4';
  8. function matrixEquals3( a, b, tolerance ) {
  9. tolerance = tolerance || 0.0001;
  10. if ( a.elements.length != b.elements.length ) {
  11. return false;
  12. }
  13. for ( var i = 0, il = a.elements.length; i < il; i ++ ) {
  14. var delta = a.elements[ i ] - b.elements[ i ];
  15. if ( delta > tolerance ) {
  16. return false;
  17. }
  18. }
  19. return true;
  20. }
  21. function toMatrix4( m3 ) {
  22. var result = new Matrix4();
  23. var re = result.elements;
  24. var me = m3.elements;
  25. re[ 0 ] = me[ 0 ];
  26. re[ 1 ] = me[ 1 ];
  27. re[ 2 ] = me[ 2 ];
  28. re[ 4 ] = me[ 3 ];
  29. re[ 5 ] = me[ 4 ];
  30. re[ 6 ] = me[ 5 ];
  31. re[ 8 ] = me[ 6 ];
  32. re[ 9 ] = me[ 7 ];
  33. re[ 10 ] = me[ 8 ];
  34. return result;
  35. }
  36. export default QUnit.module( 'Maths', () => {
  37. QUnit.module( 'Matrix3', () => {
  38. // INSTANCING
  39. QUnit.test( "Instancing", ( assert ) => {
  40. var a = new Matrix3();
  41. assert.ok( a.determinant() == 1, "Passed!" );
  42. var b = new Matrix3().set( 0, 1, 2, 3, 4, 5, 6, 7, 8 );
  43. assert.ok( b.elements[ 0 ] == 0 );
  44. assert.ok( b.elements[ 1 ] == 3 );
  45. assert.ok( b.elements[ 2 ] == 6 );
  46. assert.ok( b.elements[ 3 ] == 1 );
  47. assert.ok( b.elements[ 4 ] == 4 );
  48. assert.ok( b.elements[ 5 ] == 7 );
  49. assert.ok( b.elements[ 6 ] == 2 );
  50. assert.ok( b.elements[ 7 ] == 5 );
  51. assert.ok( b.elements[ 8 ] == 8 );
  52. assert.ok( ! matrixEquals3( a, b ), "Passed!" );
  53. } );
  54. // PUBLIC STUFF
  55. QUnit.test( "isMatrix3", ( assert ) => {
  56. var a = new Matrix3();
  57. assert.ok( a.isMatrix3 === true, "Passed!" );
  58. var b = new Matrix4();
  59. assert.ok( ! b.isMatrix3, "Passed!" );
  60. } );
  61. QUnit.test( "set", ( assert ) => {
  62. var b = new Matrix3();
  63. assert.ok( b.determinant() == 1, "Passed!" );
  64. b.set( 0, 1, 2, 3, 4, 5, 6, 7, 8 );
  65. assert.ok( b.elements[ 0 ] == 0 );
  66. assert.ok( b.elements[ 1 ] == 3 );
  67. assert.ok( b.elements[ 2 ] == 6 );
  68. assert.ok( b.elements[ 3 ] == 1 );
  69. assert.ok( b.elements[ 4 ] == 4 );
  70. assert.ok( b.elements[ 5 ] == 7 );
  71. assert.ok( b.elements[ 6 ] == 2 );
  72. assert.ok( b.elements[ 7 ] == 5 );
  73. assert.ok( b.elements[ 8 ] == 8 );
  74. } );
  75. QUnit.test( "identity", ( assert ) => {
  76. var b = new Matrix3().set( 0, 1, 2, 3, 4, 5, 6, 7, 8 );
  77. assert.ok( b.elements[ 0 ] == 0 );
  78. assert.ok( b.elements[ 1 ] == 3 );
  79. assert.ok( b.elements[ 2 ] == 6 );
  80. assert.ok( b.elements[ 3 ] == 1 );
  81. assert.ok( b.elements[ 4 ] == 4 );
  82. assert.ok( b.elements[ 5 ] == 7 );
  83. assert.ok( b.elements[ 6 ] == 2 );
  84. assert.ok( b.elements[ 7 ] == 5 );
  85. assert.ok( b.elements[ 8 ] == 8 );
  86. var a = new Matrix3();
  87. assert.ok( ! matrixEquals3( a, b ), "Passed!" );
  88. b.identity();
  89. assert.ok( matrixEquals3( a, b ), "Passed!" );
  90. } );
  91. QUnit.test( "clone", ( assert ) => {
  92. var a = new Matrix3().set( 0, 1, 2, 3, 4, 5, 6, 7, 8 );
  93. var b = a.clone();
  94. assert.ok( matrixEquals3( a, b ), "Passed!" );
  95. // ensure that it is a true copy
  96. a.elements[ 0 ] = 2;
  97. assert.ok( ! matrixEquals3( a, b ), "Passed!" );
  98. } );
  99. QUnit.test( "copy", ( assert ) => {
  100. var a = new Matrix3().set( 0, 1, 2, 3, 4, 5, 6, 7, 8 );
  101. var b = new Matrix3().copy( a );
  102. assert.ok( matrixEquals3( a, b ), "Passed!" );
  103. // ensure that it is a true copy
  104. a.elements[ 0 ] = 2;
  105. assert.ok( ! matrixEquals3( a, b ), "Passed!" );
  106. } );
  107. QUnit.test( "setFromMatrix4", ( assert ) => {
  108. var a = new Matrix4().set( 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 );
  109. var b = new Matrix3();
  110. var c = new Matrix3().set( 0, 1, 2, 4, 5, 6, 8, 9, 10 );
  111. b.setFromMatrix4( a );
  112. assert.ok( b.equals( c ) );
  113. } );
  114. QUnit.test( "multiply/premultiply", ( assert ) => {
  115. // both simply just wrap multiplyMatrices
  116. var a = new Matrix3().set( 2, 3, 5, 7, 11, 13, 17, 19, 23 );
  117. var b = new Matrix3().set( 29, 31, 37, 41, 43, 47, 53, 59, 61 );
  118. var expectedMultiply = [ 446, 1343, 2491, 486, 1457, 2701, 520, 1569, 2925 ];
  119. var expectedPremultiply = [ 904, 1182, 1556, 1131, 1489, 1967, 1399, 1845, 2435 ];
  120. a.multiply( b );
  121. assert.deepEqual( a.elements, expectedMultiply, "multiply: check result" );
  122. a.set( 2, 3, 5, 7, 11, 13, 17, 19, 23 );
  123. a.premultiply( b );
  124. assert.deepEqual( a.elements, expectedPremultiply, "premultiply: check result" );
  125. } );
  126. QUnit.test( "multiplyMatrices", ( assert ) => {
  127. // Reference:
  128. //
  129. // #!/usr/bin/env python
  130. // from __future__ import print_function
  131. // import numpy as np
  132. // print(
  133. // np.dot(
  134. // np.reshape([2, 3, 5, 7, 11, 13, 17, 19, 23], (3, 3)),
  135. // np.reshape([29, 31, 37, 41, 43, 47, 53, 59, 61], (3, 3))
  136. // )
  137. // )
  138. //
  139. // [[ 446 486 520]
  140. // [1343 1457 1569]
  141. // [2491 2701 2925]]
  142. var lhs = new Matrix3().set( 2, 3, 5, 7, 11, 13, 17, 19, 23 );
  143. var rhs = new Matrix3().set( 29, 31, 37, 41, 43, 47, 53, 59, 61 );
  144. var ans = new Matrix3();
  145. ans.multiplyMatrices( lhs, rhs );
  146. assert.ok( ans.elements[ 0 ] == 446 );
  147. assert.ok( ans.elements[ 1 ] == 1343 );
  148. assert.ok( ans.elements[ 2 ] == 2491 );
  149. assert.ok( ans.elements[ 3 ] == 486 );
  150. assert.ok( ans.elements[ 4 ] == 1457 );
  151. assert.ok( ans.elements[ 5 ] == 2701 );
  152. assert.ok( ans.elements[ 6 ] == 520 );
  153. assert.ok( ans.elements[ 7 ] == 1569 );
  154. assert.ok( ans.elements[ 8 ] == 2925 );
  155. } );
  156. QUnit.test( "multiplyScalar", ( assert ) => {
  157. var b = new Matrix3().set( 0, 1, 2, 3, 4, 5, 6, 7, 8 );
  158. assert.ok( b.elements[ 0 ] == 0 );
  159. assert.ok( b.elements[ 1 ] == 3 );
  160. assert.ok( b.elements[ 2 ] == 6 );
  161. assert.ok( b.elements[ 3 ] == 1 );
  162. assert.ok( b.elements[ 4 ] == 4 );
  163. assert.ok( b.elements[ 5 ] == 7 );
  164. assert.ok( b.elements[ 6 ] == 2 );
  165. assert.ok( b.elements[ 7 ] == 5 );
  166. assert.ok( b.elements[ 8 ] == 8 );
  167. b.multiplyScalar( 2 );
  168. assert.ok( b.elements[ 0 ] == 0 * 2 );
  169. assert.ok( b.elements[ 1 ] == 3 * 2 );
  170. assert.ok( b.elements[ 2 ] == 6 * 2 );
  171. assert.ok( b.elements[ 3 ] == 1 * 2 );
  172. assert.ok( b.elements[ 4 ] == 4 * 2 );
  173. assert.ok( b.elements[ 5 ] == 7 * 2 );
  174. assert.ok( b.elements[ 6 ] == 2 * 2 );
  175. assert.ok( b.elements[ 7 ] == 5 * 2 );
  176. assert.ok( b.elements[ 8 ] == 8 * 2 );
  177. } );
  178. QUnit.test( "determinant", ( assert ) => {
  179. var a = new Matrix3();
  180. assert.ok( a.determinant() == 1, "Passed!" );
  181. a.elements[ 0 ] = 2;
  182. assert.ok( a.determinant() == 2, "Passed!" );
  183. a.elements[ 0 ] = 0;
  184. assert.ok( a.determinant() == 0, "Passed!" );
  185. // calculated via http://www.euclideanspace.com/maths/algebra/matrix/functions/determinant/threeD/index.htm
  186. a.set( 2, 3, 4, 5, 13, 7, 8, 9, 11 );
  187. assert.ok( a.determinant() == - 73, "Passed!" );
  188. } );
  189. QUnit.test( "getInverse", ( assert ) => {
  190. var zero = new Matrix3().set( 0, 0, 0, 0, 0, 0, 0, 0, 0 );
  191. var identity4 = new Matrix4();
  192. var a = new Matrix3().set( 0, 0, 0, 0, 0, 0, 0, 0, 0 );
  193. var b = new Matrix3();
  194. b.getInverse( a );
  195. assert.ok( matrixEquals3( b, zero ), "Matrix a is zero matrix" );
  196. var testMatrices = [
  197. new Matrix4().makeRotationX( 0.3 ),
  198. new Matrix4().makeRotationX( - 0.3 ),
  199. new Matrix4().makeRotationY( 0.3 ),
  200. new Matrix4().makeRotationY( - 0.3 ),
  201. new Matrix4().makeRotationZ( 0.3 ),
  202. new Matrix4().makeRotationZ( - 0.3 ),
  203. new Matrix4().makeScale( 1, 2, 3 ),
  204. new Matrix4().makeScale( 1 / 8, 1 / 2, 1 / 3 )
  205. ];
  206. for ( var i = 0, il = testMatrices.length; i < il; i ++ ) {
  207. var m = testMatrices[ i ];
  208. a.setFromMatrix4( m );
  209. var mInverse3 = b.getInverse( a );
  210. var mInverse = toMatrix4( mInverse3 );
  211. // the determinant of the inverse should be the reciprocal
  212. assert.ok( Math.abs( a.determinant() * mInverse3.determinant() - 1 ) < 0.0001, "Passed!" );
  213. assert.ok( Math.abs( m.determinant() * mInverse.determinant() - 1 ) < 0.0001, "Passed!" );
  214. var mProduct = new Matrix4().multiplyMatrices( m, mInverse );
  215. assert.ok( Math.abs( mProduct.determinant() - 1 ) < 0.0001, "Passed!" );
  216. assert.ok( matrixEquals3( mProduct, identity4 ), "Passed!" );
  217. }
  218. } );
  219. QUnit.test( "transpose", ( assert ) => {
  220. var a = new Matrix3();
  221. var b = a.clone().transpose();
  222. assert.ok( matrixEquals3( a, b ), "Passed!" );
  223. var b = new Matrix3().set( 0, 1, 2, 3, 4, 5, 6, 7, 8 );
  224. var c = b.clone().transpose();
  225. assert.ok( ! matrixEquals3( b, c ), "Passed!" );
  226. c.transpose();
  227. assert.ok( matrixEquals3( b, c ), "Passed!" );
  228. } );
  229. QUnit.test( "getNormalMatrix", ( assert ) => {
  230. var a = new Matrix3();
  231. var b = new Matrix4().set(
  232. 2, 3, 5, 7,
  233. 11, 13, 17, 19,
  234. 23, 29, 31, 37,
  235. 41, 43, 47, 57
  236. );
  237. var expected = new Matrix3().set(
  238. - 1.2857142857142856, 0.7142857142857143, 0.2857142857142857,
  239. 0.7428571428571429, - 0.7571428571428571, 0.15714285714285714,
  240. - 0.19999999999999998, 0.3, - 0.09999999999999999
  241. );
  242. a.getNormalMatrix( b );
  243. assert.ok( matrixEquals3( a, expected ), "Check resulting Matrix3" );
  244. } );
  245. QUnit.test( "transposeIntoArray", ( assert ) => {
  246. var a = new Matrix3().set( 0, 1, 2, 3, 4, 5, 6, 7, 8 );
  247. var b = [];
  248. a.transposeIntoArray( b );
  249. assert.ok( b[ 0 ] == 0 );
  250. assert.ok( b[ 1 ] == 1 );
  251. assert.ok( b[ 2 ] == 2 );
  252. assert.ok( b[ 3 ] == 3 );
  253. assert.ok( b[ 4 ] == 4 );
  254. assert.ok( b[ 5 ] == 5 );
  255. assert.ok( b[ 5 ] == 5 );
  256. assert.ok( b[ 6 ] == 6 );
  257. assert.ok( b[ 7 ] == 7 );
  258. assert.ok( b[ 8 ] == 8 );
  259. } );
  260. QUnit.test( "setUvTransform", ( assert ) => {
  261. var a = new Matrix3().set(
  262. 0.1767766952966369, 0.17677669529663687, 0.32322330470336313,
  263. - 0.17677669529663687, 0.1767766952966369, 0.5,
  264. 0, 0, 1
  265. );
  266. var b = new Matrix3();
  267. var params = {
  268. centerX: 0.5,
  269. centerY: 0.5,
  270. offsetX: 0,
  271. offsetY: 0,
  272. repeatX: 0.25,
  273. repeatY: 0.25,
  274. rotation: 0.7753981633974483
  275. };
  276. var expected = new Matrix3().set(
  277. 0.1785355940258599, 0.17500011904519763, 0.32323214346447127,
  278. - 0.17500011904519763, 0.1785355940258599, 0.4982322625096689,
  279. 0, 0, 1
  280. );
  281. a.setUvTransform(
  282. params.offsetX, params.offsetY,
  283. params.repeatX, params.repeatY,
  284. params.rotation,
  285. params.centerX, params.centerY
  286. );
  287. b.identity()
  288. .translate( - params.centerX, - params.centerY )
  289. .rotate( params.rotation )
  290. .scale( params.repeatX, params.repeatY )
  291. .translate( params.centerX, params.centerY )
  292. .translate( params.offsetX, params.offsetY );
  293. assert.ok( matrixEquals3( a, expected ), "Check direct method" );
  294. assert.ok( matrixEquals3( b, expected ), "Check indirect method" );
  295. } );
  296. QUnit.test( "scale", ( assert ) => {
  297. var a = new Matrix3().set( 1, 2, 3, 4, 5, 6, 7, 8, 9 );
  298. var expected = new Matrix3().set(
  299. 0.25, 0.5, 0.75,
  300. 1, 1.25, 1.5,
  301. 7, 8, 9
  302. );
  303. a.scale( 0.25, 0.25 );
  304. assert.ok( matrixEquals3( a, expected ), "Check scaling result" );
  305. } );
  306. QUnit.test( "rotate", ( assert ) => {
  307. var a = new Matrix3().set( 1, 2, 3, 4, 5, 6, 7, 8, 9 );
  308. var expected = new Matrix3().set(
  309. 3.5355339059327373, 4.949747468305833, 6.363961030678928,
  310. 2.121320343559643, 2.121320343559643, 2.1213203435596433,
  311. 7, 8, 9
  312. );
  313. a.rotate( Math.PI / 4 );
  314. assert.ok( matrixEquals3( a, expected ), "Check rotated result" );
  315. } );
  316. QUnit.test( "translate", ( assert ) => {
  317. var a = new Matrix3().set( 1, 2, 3, 4, 5, 6, 7, 8, 9 );
  318. var expected = new Matrix3().set( 22, 26, 30, 53, 61, 69, 7, 8, 9 );
  319. a.translate( 3, 7 );
  320. assert.ok( matrixEquals3( a, expected ), "Check translation result" );
  321. } );
  322. QUnit.test( "equals", ( assert ) => {
  323. var a = new Matrix3().set( 0, 1, 2, 3, 4, 5, 6, 7, 8 );
  324. var b = new Matrix3().set( 0, - 1, 2, 3, 4, 5, 6, 7, 8 );
  325. assert.notOk( a.equals( b ), "Check that a does not equal b" );
  326. assert.notOk( b.equals( a ), "Check that b does not equal a" );
  327. a.copy( b );
  328. assert.ok( a.equals( b ), "Check that a equals b after copy()" );
  329. assert.ok( b.equals( a ), "Check that b equals a after copy()" );
  330. } );
  331. QUnit.test( "fromArray", ( assert ) => {
  332. var b = new Matrix3();
  333. b.fromArray( [ 0, 1, 2, 3, 4, 5, 6, 7, 8 ] );
  334. assert.ok( b.elements[ 0 ] == 0 );
  335. assert.ok( b.elements[ 1 ] == 1 );
  336. assert.ok( b.elements[ 2 ] == 2 );
  337. assert.ok( b.elements[ 3 ] == 3 );
  338. assert.ok( b.elements[ 4 ] == 4 );
  339. assert.ok( b.elements[ 5 ] == 5 );
  340. assert.ok( b.elements[ 6 ] == 6 );
  341. assert.ok( b.elements[ 7 ] == 7 );
  342. assert.ok( b.elements[ 8 ] == 8 );
  343. b = new Matrix3();
  344. b.fromArray( [ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18 ], 10 );
  345. assert.ok( b.elements[ 0 ] == 10 );
  346. assert.ok( b.elements[ 1 ] == 11 );
  347. assert.ok( b.elements[ 2 ] == 12 );
  348. assert.ok( b.elements[ 3 ] == 13 );
  349. assert.ok( b.elements[ 4 ] == 14 );
  350. assert.ok( b.elements[ 5 ] == 15 );
  351. assert.ok( b.elements[ 6 ] == 16 );
  352. assert.ok( b.elements[ 7 ] == 17 );
  353. assert.ok( b.elements[ 8 ] == 18 );
  354. } );
  355. QUnit.test( "toArray", ( assert ) => {
  356. var a = new Matrix3().set( 1, 2, 3, 4, 5, 6, 7, 8, 9 );
  357. var noOffset = [ 1, 4, 7, 2, 5, 8, 3, 6, 9 ];
  358. var withOffset = [ undefined, 1, 4, 7, 2, 5, 8, 3, 6, 9 ];
  359. var array = a.toArray();
  360. assert.deepEqual( array, noOffset, "No array, no offset" );
  361. var array = [];
  362. a.toArray( array );
  363. assert.deepEqual( array, noOffset, "With array, no offset" );
  364. var array = [];
  365. a.toArray( array, 1 );
  366. assert.deepEqual( array, withOffset, "With array, with offset" );
  367. } );
  368. } );
  369. } );
粤ICP备19079148号