Vectors, Matrices, Quaternions - garsue/libgdx GitHub Wiki

Introduction

Libgdx has several linear algebra classes for dealing with common tasks in physics and applied math. These include:

A full explanation of these concepts is beyond the current scope, but the above links may provide a starting point for further understanding. What follows is an overview of their use and implementation in Libgdx.


Vectors

A vector is an array of numbers used to describe a concept with a direction and magnitude such as position, velocity, or acceleration. As such vectors are typically used to describe the physical properties of motion of a body through space. Libgdx has vector classes for two (Vector2) (code) and 3-dimensional (Vector3) (code) spaces. Each contains common operations for working with vector quantities such as addition, subtraction, normalization, and the cross and dot products. There are also methods for linear and spherical linear interpolation between two vectors.

Method Chaining

A pattern also found elsewhere in Libgdx is the use of method chaining for convenience and reduced typing. Each operation which modifies a vector returns a reference to that vector to continue chaining operations in the same line call. The following example creates a unit vector pointing along the direction from a point (x1, y1, z1) to another point (x2, y2, z2):

Vector3 vec = new Vector3( x2, y2, z2 ).sub( x1, y1, z1 ).nor();

A new Vector3 is instantiated with the second point coordinates, the first point is subtracted from this, and the result is normalized. This is of course equivalent to:

Vector3 vec = new Vector3( x2, y2, z2 );  // new vector at (x2, y2, z2)
vec.sub( x1, y1, z1 );                    // subtract point (x1, y1, z1)
vec.nor();                                // normalize result

Temporary Vectors

The tmp() functions return a reference to a static vector useful for temporary calculations where instantiation of a new vector would be unnecessary. New object instantiation can be costly and is especially discouraged on mobile platforms such as Android, where too much object instantiation can provoke unnecessary garbage collection and interrupt the user experience. Use care with this temporary vector and never keep a reference to it as it is static and therefore shared across all vector instances.


Matrices

A matrix is a two-dimensional array of numbers. Matrices are used in graphics to perform transformations on and projections of vertices in 3D space for display on 2D screens. As with OpenGL, Libgdx stores matrices in column-major order. Matrices come in 3x3 (Matrix3) (code) and 4x4 (Matrix4) (code) varieties with convenient methods for moving values between the two. Libgdx includes many common operations for working with matrices such as building translation and scaling transformations, building rotations from Euler angles, axis-angle pairs or quaternions, building projections, and performing multiplication with other matrices and vectors.

Probably the most common use for matrices in Libgdx is through the view and projection matrices of the Camera(code) class which are used to control how geometry is rendered to the screen. Cameras, which come in OrthographicCamera (code) and PerspectiveCamera (code) varieties, provide a convenient and intuitive way to control the view in a rendered scene through a position and viewing direction, yet underneath are a simple group of 4x4 matrices which are used to tell OpenGL how to process geometry for render.

Method Chaining

As with vectors, operations on matrices in Libgdx return a reference to the modified matrix to allow chaining of operations in a single line call. The following creates a new 4x4 model-view matrix useful for an object with position (x, y, z) and rotation described by an axis-angle pair:

Matrix4 mat = new Matrix4().setToRotation( axis, angle ).trn( x, y, z );

This is of course equivalent to:

Matrix4 mat = new Matrix4();       // new identity matrix
mat.setToRotation( axis, angle );  // set rotation from axis-angle pair
mat.trn( x, y, z );                // translate by x, y, z

Native Methods

The matrix classes have a number of their operations available in static methods backed by fast native code. While member syntax is often easier to read and more convenient to write, these static methods should be used in areas where performance is a concern. The following example uses one of these methods to perform a multiplication between two 4x4 matrices:

Matrix4 matA;
Matrix4 matB;
Matrix4.mul( matA.val, matB.val );     // the result is stored in matA

Notice the use of .val to access the underlying float array which backs each matrix. The native methods work directly on these arrays. The above is functionally equivalent to the member syntax:

matA.mul( matB );

Quaternions

Quaternions are four-dimensional number systems which extend complex numbers. They have many esoteric uses in higher mathematics and number theory. Specifically in the context Libgdx the use of unit-quaternions can be useful for describing rotations in three-dimensional space, as they provide for simpler composition, numerical stability, and the avoidance of gimbal lock making them preferable often to other methods of rotational representation which may variously fall short in these areas.

Quaternions can be constructed by supplying their four components explicitly or by passing an axis-angle pair. Note that while a quaternion is often described as the combination of a vector and a scalar, it is not merely an axis-angle pair. Libgdx also provide methods for converting between quaternions and the various other rotational representations such as Euler angles, rotation matrices, and axis-angle pairs.

⚠️ **GitHub.com Fallback** ⚠️