| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814 |
- /* global QUnit */
- import { Matrix3 } from '../../../../src/math/Matrix3.js';
- import { Matrix4 } from '../../../../src/math/Matrix4.js';
- import { Vector3 } from '../../../../src/math/Vector3.js';
- import { Euler } from '../../../../src/math/Euler.js';
- import { Quaternion } from '../../../../src/math/Quaternion.js';
- import * as MathUtils from '../../../../src/math/MathUtils.js';
- import { eps } from '../../utils/math-constants.js';
- function matrixEquals4( a, b, tolerance ) {
- tolerance = tolerance || 0.0001;
- if ( a.elements.length != b.elements.length ) {
- return false;
- }
- for ( let i = 0, il = a.elements.length; i < il; i ++ ) {
- const delta = a.elements[ i ] - b.elements[ i ];
- if ( delta > tolerance ) {
- return false;
- }
- }
- return true;
- }
- // from Euler.js
- function eulerEquals( a, b, tolerance ) {
- tolerance = tolerance || 0.0001;
- const diff = Math.abs( a.x - b.x ) + Math.abs( a.y - b.y ) + Math.abs( a.z - b.z );
- return ( diff < tolerance );
- }
- export default QUnit.module( 'Maths', () => {
- QUnit.module( 'Matrix4', () => {
- // INSTANCING
- QUnit.test( 'Instancing', ( assert ) => {
- const a = new Matrix4();
- assert.ok( a.determinant() == 1, 'Passed!' );
- const b = new Matrix4().set( 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 );
- assert.ok( b.elements[ 0 ] == 0 );
- assert.ok( b.elements[ 1 ] == 4 );
- assert.ok( b.elements[ 2 ] == 8 );
- assert.ok( b.elements[ 3 ] == 12 );
- assert.ok( b.elements[ 4 ] == 1 );
- assert.ok( b.elements[ 5 ] == 5 );
- assert.ok( b.elements[ 6 ] == 9 );
- assert.ok( b.elements[ 7 ] == 13 );
- assert.ok( b.elements[ 8 ] == 2 );
- assert.ok( b.elements[ 9 ] == 6 );
- assert.ok( b.elements[ 10 ] == 10 );
- assert.ok( b.elements[ 11 ] == 14 );
- assert.ok( b.elements[ 12 ] == 3 );
- assert.ok( b.elements[ 13 ] == 7 );
- assert.ok( b.elements[ 14 ] == 11 );
- assert.ok( b.elements[ 15 ] == 15 );
- assert.ok( ! matrixEquals4( a, b ), 'Passed!' );
- const c = new Matrix4( 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 );
- assert.ok( c.elements[ 0 ] == 0 );
- assert.ok( c.elements[ 1 ] == 4 );
- assert.ok( c.elements[ 2 ] == 8 );
- assert.ok( c.elements[ 3 ] == 12 );
- assert.ok( c.elements[ 4 ] == 1 );
- assert.ok( c.elements[ 5 ] == 5 );
- assert.ok( c.elements[ 6 ] == 9 );
- assert.ok( c.elements[ 7 ] == 13 );
- assert.ok( c.elements[ 8 ] == 2 );
- assert.ok( c.elements[ 9 ] == 6 );
- assert.ok( c.elements[ 10 ] == 10 );
- assert.ok( c.elements[ 11 ] == 14 );
- assert.ok( c.elements[ 12 ] == 3 );
- assert.ok( c.elements[ 13 ] == 7 );
- assert.ok( c.elements[ 14 ] == 11 );
- assert.ok( c.elements[ 15 ] == 15 );
- assert.ok( ! matrixEquals4( a, c ), 'Passed!' );
- } );
- // PUBLIC STUFF
- QUnit.test( 'isMatrix4', ( assert ) => {
- const a = new Matrix4();
- assert.ok( a.isMatrix4 === true, 'Passed!' );
- const b = new Vector3();
- assert.ok( ! b.isMatrix4, 'Passed!' );
- } );
- QUnit.test( 'set', ( assert ) => {
- const b = new Matrix4();
- assert.ok( b.determinant() == 1, 'Passed!' );
- b.set( 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 );
- assert.ok( b.elements[ 0 ] == 0 );
- assert.ok( b.elements[ 1 ] == 4 );
- assert.ok( b.elements[ 2 ] == 8 );
- assert.ok( b.elements[ 3 ] == 12 );
- assert.ok( b.elements[ 4 ] == 1 );
- assert.ok( b.elements[ 5 ] == 5 );
- assert.ok( b.elements[ 6 ] == 9 );
- assert.ok( b.elements[ 7 ] == 13 );
- assert.ok( b.elements[ 8 ] == 2 );
- assert.ok( b.elements[ 9 ] == 6 );
- assert.ok( b.elements[ 10 ] == 10 );
- assert.ok( b.elements[ 11 ] == 14 );
- assert.ok( b.elements[ 12 ] == 3 );
- assert.ok( b.elements[ 13 ] == 7 );
- assert.ok( b.elements[ 14 ] == 11 );
- assert.ok( b.elements[ 15 ] == 15 );
- } );
- QUnit.test( 'identity', ( assert ) => {
- const b = new Matrix4().set( 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 );
- assert.ok( b.elements[ 0 ] == 0 );
- assert.ok( b.elements[ 1 ] == 4 );
- assert.ok( b.elements[ 2 ] == 8 );
- assert.ok( b.elements[ 3 ] == 12 );
- assert.ok( b.elements[ 4 ] == 1 );
- assert.ok( b.elements[ 5 ] == 5 );
- assert.ok( b.elements[ 6 ] == 9 );
- assert.ok( b.elements[ 7 ] == 13 );
- assert.ok( b.elements[ 8 ] == 2 );
- assert.ok( b.elements[ 9 ] == 6 );
- assert.ok( b.elements[ 10 ] == 10 );
- assert.ok( b.elements[ 11 ] == 14 );
- assert.ok( b.elements[ 12 ] == 3 );
- assert.ok( b.elements[ 13 ] == 7 );
- assert.ok( b.elements[ 14 ] == 11 );
- assert.ok( b.elements[ 15 ] == 15 );
- const a = new Matrix4();
- assert.ok( ! matrixEquals4( a, b ), 'Passed!' );
- b.identity();
- assert.ok( matrixEquals4( a, b ), 'Passed!' );
- } );
- QUnit.test( 'clone', ( assert ) => {
- const a = new Matrix4().set( 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 );
- const b = a.clone();
- assert.ok( matrixEquals4( a, b ), 'Passed!' );
- // ensure that it is a true copy
- a.elements[ 0 ] = 2;
- assert.ok( ! matrixEquals4( a, b ), 'Passed!' );
- } );
- QUnit.test( 'copy', ( assert ) => {
- const a = new Matrix4().set( 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 );
- const b = new Matrix4().copy( a );
- assert.ok( matrixEquals4( a, b ), 'Passed!' );
- // ensure that it is a true copy
- a.elements[ 0 ] = 2;
- assert.ok( ! matrixEquals4( a, b ), 'Passed!' );
- } );
- QUnit.test( 'setFromMatrix3', ( assert ) => {
- const a = new Matrix3().set(
- 0, 1, 2,
- 3, 4, 5,
- 6, 7, 8
- );
- const b = new Matrix4();
- const c = new Matrix4().set(
- 0, 1, 2, 0,
- 3, 4, 5, 0,
- 6, 7, 8, 0,
- 0, 0, 0, 1
- );
- b.setFromMatrix3( a );
- assert.ok( b.equals( c ) );
- } );
- QUnit.test( 'copyPosition', ( assert ) => {
- const a = new Matrix4().set( 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16 );
- const b = new Matrix4().set( 1, 2, 3, 0, 5, 6, 7, 0, 9, 10, 11, 0, 13, 14, 15, 16 );
- assert.notOk( matrixEquals4( a, b ), 'a and b initially not equal' );
- b.copyPosition( a );
- assert.ok( matrixEquals4( a, b ), 'a and b equal after copyPosition()' );
- } );
- QUnit.test( 'makeBasis/extractBasis', ( assert ) => {
- const identityBasis = [ new Vector3( 1, 0, 0 ), new Vector3( 0, 1, 0 ), new Vector3( 0, 0, 1 ) ];
- const a = new Matrix4().makeBasis( identityBasis[ 0 ], identityBasis[ 1 ], identityBasis[ 2 ] );
- const identity = new Matrix4();
- assert.ok( matrixEquals4( a, identity ), 'Passed!' );
- const testBases = [[ new Vector3( 0, 1, 0 ), new Vector3( - 1, 0, 0 ), new Vector3( 0, 0, 1 ) ]];
- for ( let i = 0; i < testBases.length; i ++ ) {
- const testBasis = testBases[ i ];
- const b = new Matrix4().makeBasis( testBasis[ 0 ], testBasis[ 1 ], testBasis[ 2 ] );
- const outBasis = [ new Vector3(), new Vector3(), new Vector3() ];
- b.extractBasis( outBasis[ 0 ], outBasis[ 1 ], outBasis[ 2 ] );
- // check what goes in, is what comes out.
- for ( let j = 0; j < outBasis.length; j ++ ) {
- assert.ok( outBasis[ j ].equals( testBasis[ j ] ), 'Passed!' );
- }
- // get the basis out the hard war
- for ( let j = 0; j < identityBasis.length; j ++ ) {
- outBasis[ j ].copy( identityBasis[ j ] );
- outBasis[ j ].applyMatrix4( b );
- }
- // did the multiply method of basis extraction work?
- for ( let j = 0; j < outBasis.length; j ++ ) {
- assert.ok( outBasis[ j ].equals( testBasis[ j ] ), 'Passed!' );
- }
- }
- } );
- QUnit.test( 'makeRotationFromEuler/extractRotation', ( assert ) => {
- const testValues = [
- new Euler( 0, 0, 0, 'XYZ' ),
- new Euler( 1, 0, 0, 'XYZ' ),
- new Euler( 0, 1, 0, 'ZYX' ),
- new Euler( 0, 0, 0.5, 'YZX' ),
- new Euler( 0, 0, - 0.5, 'YZX' )
- ];
- for ( let i = 0; i < testValues.length; i ++ ) {
- const v = testValues[ i ];
- const m = new Matrix4().makeRotationFromEuler( v );
- const v2 = new Euler().setFromRotationMatrix( m, v.order );
- const m2 = new Matrix4().makeRotationFromEuler( v2 );
- assert.ok( matrixEquals4( m, m2, eps ), 'makeRotationFromEuler #' + i + ': original and Euler-derived matrices are equal' );
- assert.ok( eulerEquals( v, v2, eps ), 'makeRotationFromEuler #' + i + ': original and matrix-derived Eulers are equal' );
- const m3 = new Matrix4().extractRotation( m2 );
- const v3 = new Euler().setFromRotationMatrix( m3, v.order );
- assert.ok( matrixEquals4( m, m3, eps ), 'extractRotation #' + i + ': original and extracted matrices are equal' );
- assert.ok( eulerEquals( v, v3, eps ), 'extractRotation #' + i + ': original and extracted Eulers are equal' );
- }
- } );
- QUnit.todo( 'makeRotationFromQuaternion', ( assert ) => {
- // makeRotationFromQuaternion( q )
- assert.ok( false, 'everything\'s gonna be alright' );
- } );
- QUnit.test( 'lookAt', ( assert ) => {
- const a = new Matrix4();
- const expected = new Matrix4().identity();
- const eye = new Vector3( 0, 0, 0 );
- const target = new Vector3( 0, 1, - 1 );
- const up = new Vector3( 0, 1, 0 );
- a.lookAt( eye, target, up );
- const rotation = new Euler().setFromRotationMatrix( a );
- assert.numEqual( rotation.x * ( 180 / Math.PI ), 45, 'Check the rotation' );
- // eye and target are in the same position
- eye.copy( target );
- a.lookAt( eye, target, up );
- assert.ok( matrixEquals4( a, expected ), 'Check the result for eye == target' );
- // up and z are parallel
- eye.set( 0, 1, 0 );
- target.set( 0, 0, 0 );
- a.lookAt( eye, target, up );
- expected.set(
- 1, 0, 0, 0,
- 0, 0.0001, 1, 0,
- 0, - 1, 0.0001, 0,
- 0, 0, 0, 1
- );
- assert.ok( matrixEquals4( a, expected ), 'Check the result for when up and z are parallel' );
- } );
- QUnit.test( 'multiply', ( assert ) => {
- const lhs = new Matrix4().set( 2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53 );
- const rhs = new Matrix4().set( 59, 61, 67, 71, 73, 79, 83, 89, 97, 101, 103, 107, 109, 113, 127, 131 );
- lhs.multiply( rhs );
- assert.ok( lhs.elements[ 0 ] == 1585 );
- assert.ok( lhs.elements[ 1 ] == 5318 );
- assert.ok( lhs.elements[ 2 ] == 10514 );
- assert.ok( lhs.elements[ 3 ] == 15894 );
- assert.ok( lhs.elements[ 4 ] == 1655 );
- assert.ok( lhs.elements[ 5 ] == 5562 );
- assert.ok( lhs.elements[ 6 ] == 11006 );
- assert.ok( lhs.elements[ 7 ] == 16634 );
- assert.ok( lhs.elements[ 8 ] == 1787 );
- assert.ok( lhs.elements[ 9 ] == 5980 );
- assert.ok( lhs.elements[ 10 ] == 11840 );
- assert.ok( lhs.elements[ 11 ] == 17888 );
- assert.ok( lhs.elements[ 12 ] == 1861 );
- assert.ok( lhs.elements[ 13 ] == 6246 );
- assert.ok( lhs.elements[ 14 ] == 12378 );
- assert.ok( lhs.elements[ 15 ] == 18710 );
- } );
- QUnit.test( 'premultiply', ( assert ) => {
- const lhs = new Matrix4().set( 2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53 );
- const rhs = new Matrix4().set( 59, 61, 67, 71, 73, 79, 83, 89, 97, 101, 103, 107, 109, 113, 127, 131 );
- rhs.premultiply( lhs );
- assert.ok( rhs.elements[ 0 ] == 1585 );
- assert.ok( rhs.elements[ 1 ] == 5318 );
- assert.ok( rhs.elements[ 2 ] == 10514 );
- assert.ok( rhs.elements[ 3 ] == 15894 );
- assert.ok( rhs.elements[ 4 ] == 1655 );
- assert.ok( rhs.elements[ 5 ] == 5562 );
- assert.ok( rhs.elements[ 6 ] == 11006 );
- assert.ok( rhs.elements[ 7 ] == 16634 );
- assert.ok( rhs.elements[ 8 ] == 1787 );
- assert.ok( rhs.elements[ 9 ] == 5980 );
- assert.ok( rhs.elements[ 10 ] == 11840 );
- assert.ok( rhs.elements[ 11 ] == 17888 );
- assert.ok( rhs.elements[ 12 ] == 1861 );
- assert.ok( rhs.elements[ 13 ] == 6246 );
- assert.ok( rhs.elements[ 14 ] == 12378 );
- assert.ok( rhs.elements[ 15 ] == 18710 );
- } );
- QUnit.test( 'multiplyMatrices', ( assert ) => {
- // Reference:
- //
- // #!/usr/bin/env python
- // from __future__ import print_function
- // import numpy as np
- // print(
- // np.dot(
- // np.reshape([2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53], (4, 4)),
- // np.reshape([59, 61, 67, 71, 73, 79, 83, 89, 97, 101, 103, 107, 109, 113, 127, 131], (4, 4))
- // )
- // )
- //
- // [[ 1585 1655 1787 1861]
- // [ 5318 5562 5980 6246]
- // [10514 11006 11840 12378]
- // [15894 16634 17888 18710]]
- const lhs = new Matrix4().set( 2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53 );
- const rhs = new Matrix4().set( 59, 61, 67, 71, 73, 79, 83, 89, 97, 101, 103, 107, 109, 113, 127, 131 );
- const ans = new Matrix4();
- ans.multiplyMatrices( lhs, rhs );
- assert.ok( ans.elements[ 0 ] == 1585 );
- assert.ok( ans.elements[ 1 ] == 5318 );
- assert.ok( ans.elements[ 2 ] == 10514 );
- assert.ok( ans.elements[ 3 ] == 15894 );
- assert.ok( ans.elements[ 4 ] == 1655 );
- assert.ok( ans.elements[ 5 ] == 5562 );
- assert.ok( ans.elements[ 6 ] == 11006 );
- assert.ok( ans.elements[ 7 ] == 16634 );
- assert.ok( ans.elements[ 8 ] == 1787 );
- assert.ok( ans.elements[ 9 ] == 5980 );
- assert.ok( ans.elements[ 10 ] == 11840 );
- assert.ok( ans.elements[ 11 ] == 17888 );
- assert.ok( ans.elements[ 12 ] == 1861 );
- assert.ok( ans.elements[ 13 ] == 6246 );
- assert.ok( ans.elements[ 14 ] == 12378 );
- assert.ok( ans.elements[ 15 ] == 18710 );
- } );
- QUnit.test( 'multiplyScalar', ( assert ) => {
- const b = new Matrix4().set( 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 );
- assert.ok( b.elements[ 0 ] == 0 );
- assert.ok( b.elements[ 1 ] == 4 );
- assert.ok( b.elements[ 2 ] == 8 );
- assert.ok( b.elements[ 3 ] == 12 );
- assert.ok( b.elements[ 4 ] == 1 );
- assert.ok( b.elements[ 5 ] == 5 );
- assert.ok( b.elements[ 6 ] == 9 );
- assert.ok( b.elements[ 7 ] == 13 );
- assert.ok( b.elements[ 8 ] == 2 );
- assert.ok( b.elements[ 9 ] == 6 );
- assert.ok( b.elements[ 10 ] == 10 );
- assert.ok( b.elements[ 11 ] == 14 );
- assert.ok( b.elements[ 12 ] == 3 );
- assert.ok( b.elements[ 13 ] == 7 );
- assert.ok( b.elements[ 14 ] == 11 );
- assert.ok( b.elements[ 15 ] == 15 );
- b.multiplyScalar( 2 );
- assert.ok( b.elements[ 0 ] == 0 * 2 );
- assert.ok( b.elements[ 1 ] == 4 * 2 );
- assert.ok( b.elements[ 2 ] == 8 * 2 );
- assert.ok( b.elements[ 3 ] == 12 * 2 );
- assert.ok( b.elements[ 4 ] == 1 * 2 );
- assert.ok( b.elements[ 5 ] == 5 * 2 );
- assert.ok( b.elements[ 6 ] == 9 * 2 );
- assert.ok( b.elements[ 7 ] == 13 * 2 );
- assert.ok( b.elements[ 8 ] == 2 * 2 );
- assert.ok( b.elements[ 9 ] == 6 * 2 );
- assert.ok( b.elements[ 10 ] == 10 * 2 );
- assert.ok( b.elements[ 11 ] == 14 * 2 );
- assert.ok( b.elements[ 12 ] == 3 * 2 );
- assert.ok( b.elements[ 13 ] == 7 * 2 );
- assert.ok( b.elements[ 14 ] == 11 * 2 );
- assert.ok( b.elements[ 15 ] == 15 * 2 );
- } );
- QUnit.test( 'determinant', ( assert ) => {
- const a = new Matrix4();
- assert.ok( a.determinant() == 1, 'Passed!' );
- a.elements[ 0 ] = 2;
- assert.ok( a.determinant() == 2, 'Passed!' );
- a.elements[ 0 ] = 0;
- assert.ok( a.determinant() == 0, 'Passed!' );
- // calculated via http://www.euclideanspace.com/maths/algebra/matrix/functions/determinant/fourD/index.htm
- a.set( 2, 3, 4, 5, - 1, - 21, - 3, - 4, 6, 7, 8, 10, - 8, - 9, - 10, - 12 );
- assert.ok( a.determinant() == 76, 'Passed!' );
- } );
- QUnit.test( 'transpose', ( assert ) => {
- const a = new Matrix4();
- let b = a.clone().transpose();
- assert.ok( matrixEquals4( a, b ), 'Passed!' );
- b = new Matrix4().set( 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 );
- const c = b.clone().transpose();
- assert.ok( ! matrixEquals4( b, c ), 'Passed!' );
- c.transpose();
- assert.ok( matrixEquals4( b, c ), 'Passed!' );
- } );
- QUnit.test( 'setPosition', ( assert ) => {
- const a = new Matrix4().set( 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 );
- const b = new Vector3( - 1, - 2, - 3 );
- const c = new Matrix4().set( 0, 1, 2, - 1, 4, 5, 6, - 2, 8, 9, 10, - 3, 12, 13, 14, 15 );
- a.setPosition( b );
- assert.ok( matrixEquals4( a, c ), 'Passed!' );
- const d = new Matrix4().set( 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 );
- const e = new Matrix4().set( 0, 1, 2, - 1, 4, 5, 6, - 2, 8, 9, 10, - 3, 12, 13, 14, 15 );
- d.setPosition( - 1, - 2, - 3 );
- assert.ok( matrixEquals4( d, e ), 'Passed!' );
- } );
- QUnit.test( 'invert', ( assert ) => {
- const zero = new Matrix4().set( 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 );
- const identity = new Matrix4();
- const a = new Matrix4();
- const b = new Matrix4().set( 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 );
- a.copy( b ).invert();
- assert.ok( matrixEquals4( a, zero ), 'Passed!' );
- const testMatrices = [
- new Matrix4().makeRotationX( 0.3 ),
- new Matrix4().makeRotationX( - 0.3 ),
- new Matrix4().makeRotationY( 0.3 ),
- new Matrix4().makeRotationY( - 0.3 ),
- new Matrix4().makeRotationZ( 0.3 ),
- new Matrix4().makeRotationZ( - 0.3 ),
- new Matrix4().makeScale( 1, 2, 3 ),
- new Matrix4().makeScale( 1 / 8, 1 / 2, 1 / 3 ),
- new Matrix4().makePerspective( - 1, 1, 1, - 1, 1, 1000 ),
- new Matrix4().makePerspective( - 16, 16, 9, - 9, 0.1, 10000 ),
- new Matrix4().makeTranslation( 1, 2, 3 )
- ];
- for ( let i = 0, il = testMatrices.length; i < il; i ++ ) {
- const m = testMatrices[ i ];
- const mInverse = new Matrix4().copy( m ).invert();
- const mSelfInverse = m.clone();
- mSelfInverse.copy( mSelfInverse ).invert();
- // self-inverse should the same as inverse
- assert.ok( matrixEquals4( mSelfInverse, mInverse ), 'Passed!' );
- // the determinant of the inverse should be the reciprocal
- assert.ok( Math.abs( m.determinant() * mInverse.determinant() - 1 ) < 0.0001, 'Passed!' );
- const mProduct = new Matrix4().multiplyMatrices( m, mInverse );
- // the determinant of the identity matrix is 1
- assert.ok( Math.abs( mProduct.determinant() - 1 ) < 0.0001, 'Passed!' );
- assert.ok( matrixEquals4( mProduct, identity ), 'Passed!' );
- }
- } );
- QUnit.test( 'scale', ( assert ) => {
- const a = new Matrix4().set( 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16 );
- const b = new Vector3( 2, 3, 4 );
- const c = new Matrix4().set( 2, 6, 12, 4, 10, 18, 28, 8, 18, 30, 44, 12, 26, 42, 60, 16 );
- a.scale( b );
- assert.ok( matrixEquals4( a, c ), 'Passed!' );
- } );
- QUnit.test( 'getMaxScaleOnAxis', ( assert ) => {
- const a = new Matrix4().set( 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16 );
- const expected = Math.sqrt( 3 * 3 + 7 * 7 + 11 * 11 );
- assert.ok( Math.abs( a.getMaxScaleOnAxis() - expected ) <= eps, 'Check result' );
- } );
- QUnit.test( 'makeTranslation', ( assert ) => {
- const a = new Matrix4();
- const b = new Vector3( 2, 3, 4 );
- const c = new Matrix4().set( 1, 0, 0, 2, 0, 1, 0, 3, 0, 0, 1, 4, 0, 0, 0, 1 );
- a.makeTranslation( b.x, b.y, b.z );
- assert.ok( matrixEquals4( a, c ), 'Passed!' );
- a.makeTranslation( b );
- assert.ok( matrixEquals4( a, c ), 'Passed!' );
- } );
- QUnit.test( 'makeRotationX', ( assert ) => {
- const a = new Matrix4();
- const b = Math.sqrt( 3 ) / 2;
- const c = new Matrix4().set( 1, 0, 0, 0, 0, b, - 0.5, 0, 0, 0.5, b, 0, 0, 0, 0, 1 );
- a.makeRotationX( Math.PI / 6 );
- assert.ok( matrixEquals4( a, c ), 'Passed!' );
- } );
- QUnit.test( 'makeRotationY', ( assert ) => {
- const a = new Matrix4();
- const b = Math.sqrt( 3 ) / 2;
- const c = new Matrix4().set( b, 0, 0.5, 0, 0, 1, 0, 0, - 0.5, 0, b, 0, 0, 0, 0, 1 );
- a.makeRotationY( Math.PI / 6 );
- assert.ok( matrixEquals4( a, c ), 'Passed!' );
- } );
- QUnit.test( 'makeRotationZ', ( assert ) => {
- const a = new Matrix4();
- const b = Math.sqrt( 3 ) / 2;
- const c = new Matrix4().set( b, - 0.5, 0, 0, 0.5, b, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1 );
- a.makeRotationZ( Math.PI / 6 );
- assert.ok( matrixEquals4( a, c ), 'Passed!' );
- } );
- QUnit.test( 'makeRotationAxis', ( assert ) => {
- const axis = new Vector3( 1.5, 0.0, 1.0 ).normalize();
- const radians = MathUtils.degToRad( 45 );
- const a = new Matrix4().makeRotationAxis( axis, radians );
- const expected = new Matrix4().set(
- 0.9098790095958609, - 0.39223227027636803, 0.13518148560620882, 0,
- 0.39223227027636803, 0.7071067811865476, - 0.588348405414552, 0,
- 0.13518148560620882, 0.588348405414552, 0.7972277715906868, 0,
- 0, 0, 0, 1
- );
- assert.ok( matrixEquals4( a, expected ), 'Check numeric result' );
- } );
- QUnit.test( 'makeScale', ( assert ) => {
- const a = new Matrix4();
- const c = new Matrix4().set( 2, 0, 0, 0, 0, 3, 0, 0, 0, 0, 4, 0, 0, 0, 0, 1 );
- a.makeScale( 2, 3, 4 );
- assert.ok( matrixEquals4( a, c ), 'Passed!' );
- } );
- QUnit.test( 'makeShear', ( assert ) => {
- const a = new Matrix4();
- const c = new Matrix4().set( 1, 3, 5, 0, 1, 1, 6, 0, 2, 4, 1, 0, 0, 0, 0, 1 );
- a.makeShear( 1, 2, 3, 4, 5, 6 );
- assert.ok( matrixEquals4( a, c ), 'Passed!' );
- } );
- QUnit.test( 'compose/decompose', ( assert ) => {
- const tValues = [
- new Vector3(),
- new Vector3( 3, 0, 0 ),
- new Vector3( 0, 4, 0 ),
- new Vector3( 0, 0, 5 ),
- new Vector3( - 6, 0, 0 ),
- new Vector3( 0, - 7, 0 ),
- new Vector3( 0, 0, - 8 ),
- new Vector3( - 2, 5, - 9 ),
- new Vector3( - 2, - 5, - 9 )
- ];
- const sValues = [
- new Vector3( 1, 1, 1 ),
- new Vector3( 2, 2, 2 ),
- new Vector3( 1, - 1, 1 ),
- new Vector3( - 1, 1, 1 ),
- new Vector3( 1, 1, - 1 ),
- new Vector3( 2, - 2, 1 ),
- new Vector3( - 1, 2, - 2 ),
- new Vector3( - 1, - 1, - 1 ),
- new Vector3( - 2, - 2, - 2 )
- ];
- const rValues = [
- new Quaternion(),
- new Quaternion().setFromEuler( new Euler( 1, 1, 0 ) ),
- new Quaternion().setFromEuler( new Euler( 1, - 1, 1 ) ),
- new Quaternion( 0, 0.9238795292366128, 0, 0.38268342717215614 )
- ];
- for ( let ti = 0; ti < tValues.length; ti ++ ) {
- for ( let si = 0; si < sValues.length; si ++ ) {
- for ( let ri = 0; ri < rValues.length; ri ++ ) {
- const t = tValues[ ti ];
- const s = sValues[ si ];
- const r = rValues[ ri ];
- const m = new Matrix4().compose( t, r, s );
- const t2 = new Vector3();
- const r2 = new Quaternion();
- const s2 = new Vector3();
- m.decompose( t2, r2, s2 );
- const m2 = new Matrix4().compose( t2, r2, s2 );
- /*
- // debug code
- const matrixIsSame = matrixEquals4( m, m2 );
- if ( ! matrixIsSame ) {
- console.log( t, s, r );
- console.log( t2, s2, r2 );
- console.log( m, m2 );
- }
- */
- assert.ok( matrixEquals4( m, m2 ), 'Passed!' );
- }
- }
- }
- } );
- QUnit.test( 'makePerspective', ( assert ) => {
- const a = new Matrix4().makePerspective( - 1, 1, - 1, 1, 1, 100 );
- const expected = new Matrix4().set(
- 1, 0, 0, 0,
- 0, - 1, 0, 0,
- 0, 0, - 101 / 99, - 200 / 99,
- 0, 0, - 1, 0
- );
- assert.ok( matrixEquals4( a, expected ), 'Check result' );
- } );
- QUnit.test( 'makeOrthographic', ( assert ) => {
- const a = new Matrix4().makeOrthographic( - 1, 1, - 1, 1, 1, 100 );
- const expected = new Matrix4().set(
- 1, 0, 0, 0,
- 0, - 1, 0, 0,
- 0, 0, - 2 / 99, - 101 / 99,
- 0, 0, 0, 1
- );
- assert.ok( matrixEquals4( a, expected ), 'Check result' );
- } );
- QUnit.test( 'equals', ( assert ) => {
- const a = new Matrix4().set( 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16 );
- const b = new Matrix4().set( 0, - 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16 );
- assert.notOk( a.equals( b ), 'Check that a does not equal b' );
- assert.notOk( b.equals( a ), 'Check that b does not equal a' );
- a.copy( b );
- assert.ok( a.equals( b ), 'Check that a equals b after copy()' );
- assert.ok( b.equals( a ), 'Check that b equals a after copy()' );
- } );
- QUnit.test( 'fromArray', ( assert ) => {
- const a = new Matrix4();
- const b = new Matrix4().set( 1, 5, 9, 13, 2, 6, 10, 14, 3, 7, 11, 15, 4, 8, 12, 16 );
- a.fromArray( [ 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16 ] );
- assert.ok( a.equals( b ), 'Passed' );
- } );
- QUnit.test( 'toArray', ( assert ) => {
- const a = new Matrix4().set( 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16 );
- const noOffset = [ 1, 5, 9, 13, 2, 6, 10, 14, 3, 7, 11, 15, 4, 8, 12, 16 ];
- const withOffset = [ undefined, 1, 5, 9, 13, 2, 6, 10, 14, 3, 7, 11, 15, 4, 8, 12, 16 ];
- let array = a.toArray();
- assert.deepEqual( array, noOffset, 'No array, no offset' );
- array = [];
- a.toArray( array );
- assert.deepEqual( array, noOffset, 'With array, no offset' );
- array = [];
- a.toArray( array, 1 );
- assert.deepEqual( array, withOffset, 'With array, with offset' );
- } );
- } );
- } );
|