xref: /aosp_15_r20/external/libchrome/ui/gfx/geometry/matrix3_unittest.cc (revision 635a864187cb8b6c713ff48b7e790a6b21769273)
1*635a8641SAndroid Build Coastguard Worker // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2*635a8641SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*635a8641SAndroid Build Coastguard Worker // found in the LICENSE file.
4*635a8641SAndroid Build Coastguard Worker 
5*635a8641SAndroid Build Coastguard Worker #include <cmath>
6*635a8641SAndroid Build Coastguard Worker #include <limits>
7*635a8641SAndroid Build Coastguard Worker 
8*635a8641SAndroid Build Coastguard Worker #include "testing/gtest/include/gtest/gtest.h"
9*635a8641SAndroid Build Coastguard Worker #include "ui/gfx/geometry/matrix3_f.h"
10*635a8641SAndroid Build Coastguard Worker 
11*635a8641SAndroid Build Coastguard Worker namespace gfx {
12*635a8641SAndroid Build Coastguard Worker namespace {
13*635a8641SAndroid Build Coastguard Worker 
TEST(Matrix3fTest,Constructors)14*635a8641SAndroid Build Coastguard Worker TEST(Matrix3fTest, Constructors) {
15*635a8641SAndroid Build Coastguard Worker   Matrix3F zeros = Matrix3F::Zeros();
16*635a8641SAndroid Build Coastguard Worker   Matrix3F ones = Matrix3F::Ones();
17*635a8641SAndroid Build Coastguard Worker   Matrix3F identity = Matrix3F::Identity();
18*635a8641SAndroid Build Coastguard Worker 
19*635a8641SAndroid Build Coastguard Worker   Matrix3F product_ones = Matrix3F::FromOuterProduct(
20*635a8641SAndroid Build Coastguard Worker       Vector3dF(1.0f, 1.0f, 1.0f), Vector3dF(1.0f, 1.0f, 1.0f));
21*635a8641SAndroid Build Coastguard Worker   Matrix3F product_zeros = Matrix3F::FromOuterProduct(
22*635a8641SAndroid Build Coastguard Worker       Vector3dF(1.0f, 1.0f, 1.0f), Vector3dF(0.0f, 0.0f, 0.0f));
23*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(ones, product_ones);
24*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(zeros, product_zeros);
25*635a8641SAndroid Build Coastguard Worker 
26*635a8641SAndroid Build Coastguard Worker   for (int i = 0; i < 3; ++i) {
27*635a8641SAndroid Build Coastguard Worker     for (int j = 0; j < 3; ++j)
28*635a8641SAndroid Build Coastguard Worker       EXPECT_EQ(i == j ? 1.0f : 0.0f, identity.get(i, j));
29*635a8641SAndroid Build Coastguard Worker   }
30*635a8641SAndroid Build Coastguard Worker }
31*635a8641SAndroid Build Coastguard Worker 
TEST(Matrix3fTest,DataAccess)32*635a8641SAndroid Build Coastguard Worker TEST(Matrix3fTest, DataAccess) {
33*635a8641SAndroid Build Coastguard Worker   Matrix3F matrix = Matrix3F::Ones();
34*635a8641SAndroid Build Coastguard Worker   Matrix3F identity = Matrix3F::Identity();
35*635a8641SAndroid Build Coastguard Worker 
36*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(0.0f, 1.0f, 0.0f), identity.get_column(1));
37*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(0.0f, 1.0f, 0.0f), identity.get_row(1));
38*635a8641SAndroid Build Coastguard Worker   matrix.set(0.0f, 1.0f, 2.0f, 3.0f, 4.0f, 5.0f, 6.0f, 7.0f, 8.0f);
39*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(2.0f, 5.0f, 8.0f), matrix.get_column(2));
40*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(6.0f, 7.0f, 8.0f), matrix.get_row(2));
41*635a8641SAndroid Build Coastguard Worker   matrix.set_column(0, Vector3dF(0.1f, 0.2f, 0.3f));
42*635a8641SAndroid Build Coastguard Worker   matrix.set_column(0, Vector3dF(0.1f, 0.2f, 0.3f));
43*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(0.1f, 0.2f, 0.3f), matrix.get_column(0));
44*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(0.1f, 1.0f, 2.0f), matrix.get_row(0));
45*635a8641SAndroid Build Coastguard Worker 
46*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(0.1f, matrix.get(0, 0));
47*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(5.0f, matrix.get(1, 2));
48*635a8641SAndroid Build Coastguard Worker }
49*635a8641SAndroid Build Coastguard Worker 
TEST(Matrix3fTest,Determinant)50*635a8641SAndroid Build Coastguard Worker TEST(Matrix3fTest, Determinant) {
51*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(1.0f, Matrix3F::Identity().Determinant());
52*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(0.0f, Matrix3F::Zeros().Determinant());
53*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(0.0f, Matrix3F::Ones().Determinant());
54*635a8641SAndroid Build Coastguard Worker 
55*635a8641SAndroid Build Coastguard Worker   // Now for something non-trivial...
56*635a8641SAndroid Build Coastguard Worker   Matrix3F matrix = Matrix3F::Zeros();
57*635a8641SAndroid Build Coastguard Worker   matrix.set(0, 5, 6, 8, 7, 0, 1, 9, 0);
58*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(390.0f, matrix.Determinant());
59*635a8641SAndroid Build Coastguard Worker   matrix.set(2, 0, 3 * matrix.get(0, 0));
60*635a8641SAndroid Build Coastguard Worker   matrix.set(2, 1, 3 * matrix.get(0, 1));
61*635a8641SAndroid Build Coastguard Worker   matrix.set(2, 2, 3 * matrix.get(0, 2));
62*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(0, matrix.Determinant());
63*635a8641SAndroid Build Coastguard Worker 
64*635a8641SAndroid Build Coastguard Worker   matrix.set(0.57f,  0.205f,  0.942f,
65*635a8641SAndroid Build Coastguard Worker              0.314f,  0.845f,  0.826f,
66*635a8641SAndroid Build Coastguard Worker              0.131f,  0.025f,  0.962f);
67*635a8641SAndroid Build Coastguard Worker   EXPECT_NEAR(0.3149f, matrix.Determinant(), 0.0001f);
68*635a8641SAndroid Build Coastguard Worker }
69*635a8641SAndroid Build Coastguard Worker 
TEST(Matrix3fTest,Inverse)70*635a8641SAndroid Build Coastguard Worker TEST(Matrix3fTest, Inverse) {
71*635a8641SAndroid Build Coastguard Worker   Matrix3F identity = Matrix3F::Identity();
72*635a8641SAndroid Build Coastguard Worker   Matrix3F inv_identity = identity.Inverse();
73*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(identity, inv_identity);
74*635a8641SAndroid Build Coastguard Worker 
75*635a8641SAndroid Build Coastguard Worker   Matrix3F singular = Matrix3F::Zeros();
76*635a8641SAndroid Build Coastguard Worker   singular.set(1.0f, 3.0f, 4.0f,
77*635a8641SAndroid Build Coastguard Worker                2.0f, 11.0f, 5.0f,
78*635a8641SAndroid Build Coastguard Worker                0.5f, 1.5f, 2.0f);
79*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(0, singular.Determinant());
80*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Matrix3F::Zeros(), singular.Inverse());
81*635a8641SAndroid Build Coastguard Worker 
82*635a8641SAndroid Build Coastguard Worker   Matrix3F regular = Matrix3F::Zeros();
83*635a8641SAndroid Build Coastguard Worker   regular.set(0.57f,  0.205f,  0.942f,
84*635a8641SAndroid Build Coastguard Worker               0.314f,  0.845f,  0.826f,
85*635a8641SAndroid Build Coastguard Worker               0.131f,  0.025f,  0.962f);
86*635a8641SAndroid Build Coastguard Worker   Matrix3F inv_regular = regular.Inverse();
87*635a8641SAndroid Build Coastguard Worker   regular.set(2.51540616f, -0.55138018f, -1.98968043f,
88*635a8641SAndroid Build Coastguard Worker               -0.61552266f,  1.34920184f, -0.55573636f,
89*635a8641SAndroid Build Coastguard Worker               -0.32653861f,  0.04002158f,  1.32488726f);
90*635a8641SAndroid Build Coastguard Worker   EXPECT_TRUE(regular.IsNear(inv_regular, 0.00001f));
91*635a8641SAndroid Build Coastguard Worker }
92*635a8641SAndroid Build Coastguard Worker 
TEST(Matrix3fTest,Transpose)93*635a8641SAndroid Build Coastguard Worker TEST(Matrix3fTest, Transpose) {
94*635a8641SAndroid Build Coastguard Worker   Matrix3F matrix = Matrix3F::Zeros();
95*635a8641SAndroid Build Coastguard Worker 
96*635a8641SAndroid Build Coastguard Worker   matrix.set(0.0f, 1.0f, 2.0f, 3.0f, 4.0f, 5.0f, 6.0f, 7.0f, 8.0f);
97*635a8641SAndroid Build Coastguard Worker 
98*635a8641SAndroid Build Coastguard Worker   Matrix3F transpose = matrix.Transpose();
99*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(0.0f, 1.0f, 2.0f), transpose.get_column(0));
100*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(3.0f, 4.0f, 5.0f), transpose.get_column(1));
101*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(6.0f, 7.0f, 8.0f), transpose.get_column(2));
102*635a8641SAndroid Build Coastguard Worker 
103*635a8641SAndroid Build Coastguard Worker   EXPECT_TRUE(matrix.IsEqual(transpose.Transpose()));
104*635a8641SAndroid Build Coastguard Worker }
105*635a8641SAndroid Build Coastguard Worker 
TEST(Matrix3fTest,EigenvectorsIdentity)106*635a8641SAndroid Build Coastguard Worker TEST(Matrix3fTest, EigenvectorsIdentity) {
107*635a8641SAndroid Build Coastguard Worker   // This block tests the trivial case of eigenvalues of the identity matrix.
108*635a8641SAndroid Build Coastguard Worker   Matrix3F identity = Matrix3F::Identity();
109*635a8641SAndroid Build Coastguard Worker   Vector3dF eigenvals = identity.SolveEigenproblem(NULL);
110*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(1.0f, 1.0f, 1.0f), eigenvals);
111*635a8641SAndroid Build Coastguard Worker }
112*635a8641SAndroid Build Coastguard Worker 
TEST(Matrix3fTest,EigenvectorsDiagonal)113*635a8641SAndroid Build Coastguard Worker TEST(Matrix3fTest, EigenvectorsDiagonal)  {
114*635a8641SAndroid Build Coastguard Worker   // This block tests the another trivial case of eigenvalues of a diagonal
115*635a8641SAndroid Build Coastguard Worker   // matrix. Here we expect values to be sorted.
116*635a8641SAndroid Build Coastguard Worker   Matrix3F matrix = Matrix3F::Zeros();
117*635a8641SAndroid Build Coastguard Worker   matrix.set(0, 0, 1.0f);
118*635a8641SAndroid Build Coastguard Worker   matrix.set(1, 1, -2.5f);
119*635a8641SAndroid Build Coastguard Worker   matrix.set(2, 2, 3.14f);
120*635a8641SAndroid Build Coastguard Worker   Matrix3F eigenvectors = Matrix3F::Zeros();
121*635a8641SAndroid Build Coastguard Worker   Vector3dF eigenvals = matrix.SolveEigenproblem(&eigenvectors);
122*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(3.14f, 1.0f, -2.5f), eigenvals);
123*635a8641SAndroid Build Coastguard Worker 
124*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(0.0f, 0.0f, 1.0f), eigenvectors.get_column(0));
125*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(1.0f, 0.0f, 0.0f), eigenvectors.get_column(1));
126*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(0.0f, 1.0f, 0.0f), eigenvectors.get_column(2));
127*635a8641SAndroid Build Coastguard Worker }
128*635a8641SAndroid Build Coastguard Worker 
TEST(Matrix3fTest,EigenvectorsNiceNotPositive)129*635a8641SAndroid Build Coastguard Worker TEST(Matrix3fTest, EigenvectorsNiceNotPositive)  {
130*635a8641SAndroid Build Coastguard Worker   // This block tests computation of eigenvectors of a matrix where nice
131*635a8641SAndroid Build Coastguard Worker   // round values are expected.
132*635a8641SAndroid Build Coastguard Worker   Matrix3F matrix = Matrix3F::Zeros();
133*635a8641SAndroid Build Coastguard Worker   // This is not a positive-definite matrix but eigenvalues and the first
134*635a8641SAndroid Build Coastguard Worker   // eigenvector should nonetheless be computed correctly.
135*635a8641SAndroid Build Coastguard Worker   matrix.set(3, 2, 4, 2, 0, 2, 4, 2, 3);
136*635a8641SAndroid Build Coastguard Worker   Matrix3F eigenvectors = Matrix3F::Zeros();
137*635a8641SAndroid Build Coastguard Worker   Vector3dF eigenvals = matrix.SolveEigenproblem(&eigenvectors);
138*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Vector3dF(8.0f, -1.0f, -1.0f), eigenvals);
139*635a8641SAndroid Build Coastguard Worker 
140*635a8641SAndroid Build Coastguard Worker   Vector3dF expected_principal(0.66666667f, 0.33333333f, 0.66666667f);
141*635a8641SAndroid Build Coastguard Worker   EXPECT_NEAR(0.0f,
142*635a8641SAndroid Build Coastguard Worker               (expected_principal - eigenvectors.get_column(0)).Length(),
143*635a8641SAndroid Build Coastguard Worker               0.000001f);
144*635a8641SAndroid Build Coastguard Worker }
145*635a8641SAndroid Build Coastguard Worker 
TEST(Matrix3fTest,EigenvectorsPositiveDefinite)146*635a8641SAndroid Build Coastguard Worker TEST(Matrix3fTest, EigenvectorsPositiveDefinite) {
147*635a8641SAndroid Build Coastguard Worker   // This block tests computation of eigenvectors of a matrix where output
148*635a8641SAndroid Build Coastguard Worker   // is not as nice as above, but it actually meets the definition.
149*635a8641SAndroid Build Coastguard Worker   Matrix3F matrix = Matrix3F::Zeros();
150*635a8641SAndroid Build Coastguard Worker   Matrix3F eigenvectors = Matrix3F::Zeros();
151*635a8641SAndroid Build Coastguard Worker   Matrix3F expected_eigenvectors = Matrix3F::Zeros();
152*635a8641SAndroid Build Coastguard Worker   matrix.set(1, -1,  2, -1,  4,  5, 2,  5,  0);
153*635a8641SAndroid Build Coastguard Worker   Vector3dF eigenvals =  matrix.SolveEigenproblem(&eigenvectors);
154*635a8641SAndroid Build Coastguard Worker   Vector3dF expected_eigv(7.3996266f, 1.91197255f, -4.31159915f);
155*635a8641SAndroid Build Coastguard Worker   expected_eigv -= eigenvals;
156*635a8641SAndroid Build Coastguard Worker   EXPECT_NEAR(0, expected_eigv.LengthSquared(), 0.00001f);
157*635a8641SAndroid Build Coastguard Worker   expected_eigenvectors.set(0.04926317f, -0.92135662f, -0.38558414f,
158*635a8641SAndroid Build Coastguard Worker                             0.82134249f, 0.25703273f, -0.50924521f,
159*635a8641SAndroid Build Coastguard Worker                             0.56830419f, -0.2916096f, 0.76941158f);
160*635a8641SAndroid Build Coastguard Worker   EXPECT_TRUE(expected_eigenvectors.IsNear(eigenvectors, 0.00001f));
161*635a8641SAndroid Build Coastguard Worker }
162*635a8641SAndroid Build Coastguard Worker 
TEST(Matrix3fTest,Operators)163*635a8641SAndroid Build Coastguard Worker TEST(Matrix3fTest, Operators) {
164*635a8641SAndroid Build Coastguard Worker   Matrix3F matrix1 = Matrix3F::Zeros();
165*635a8641SAndroid Build Coastguard Worker   matrix1.set(1, 2, 3, 4, 5, 6, 7, 8, 9);
166*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(matrix1 + Matrix3F::Zeros(), matrix1);
167*635a8641SAndroid Build Coastguard Worker 
168*635a8641SAndroid Build Coastguard Worker   Matrix3F matrix2 = Matrix3F::Zeros();
169*635a8641SAndroid Build Coastguard Worker   matrix2.set(-1, -2, -3, -4, -5, -6, -7, -8, -9);
170*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(matrix1 + matrix2, Matrix3F::Zeros());
171*635a8641SAndroid Build Coastguard Worker 
172*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(Matrix3F::Zeros() - matrix1, matrix2);
173*635a8641SAndroid Build Coastguard Worker 
174*635a8641SAndroid Build Coastguard Worker   Matrix3F result = Matrix3F::Zeros();
175*635a8641SAndroid Build Coastguard Worker   result.set(2, 4, 6, 8, 10, 12, 14, 16, 18);
176*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(matrix1 - matrix2, result);
177*635a8641SAndroid Build Coastguard Worker   result.set(-2, -4, -6, -8, -10, -12, -14, -16, -18);
178*635a8641SAndroid Build Coastguard Worker   EXPECT_EQ(matrix2 - matrix1, result);
179*635a8641SAndroid Build Coastguard Worker }
180*635a8641SAndroid Build Coastguard Worker 
181*635a8641SAndroid Build Coastguard Worker }  // namespace
182*635a8641SAndroid Build Coastguard Worker }  // namespace gfx
183