This article is a continuation of WebGL 2D DrawImage. If you haven’t read that I suggest you start there.
In that last article we implemented the WebGL equivalent of Canvas 2D’s drawImage
function
including its ability to specify both the source rectangle and the destination rectangle.
What we haven’t done yet is let us rotate and/or scale it from any arbitrary point. We could do that
by adding more arguments, at a minimum we’d need to specify a center point, a rotation and an x and y scale.
Fortunately there’s a more generic and useful way. The way the Canvas 2D API does that is with a matrix stack.
The matrix stack functions of the Canvas 2D API are save
, restore
, translate
, rotate
, and scale
.
A matrix stack is pretty simple to implement. We make a stack of matrices. We make functions to multiply the top matrix of the stack using by either a translation, rotation, or scale matrix using the functions we created earlier.
Here’s the implementation
First the constructor and the save
and restore
functions
function MatrixStack() {
this.stack = [];
// since the stack is empty this will put an initial matrix in it
this.restore();
}
// Pops the top of the stack restoring the previously saved matrix
MatrixStack.prototype.restore = function() {
this.stack.pop();
// Never let the stack be totally empty
if (this.stack.length < 1) {
this.stack[0] = m4.identity();
}
};
// Pushes a copy of the current matrix on the stack
MatrixStack.prototype.save = function() {
this.stack.push(this.getCurrentMatrix());
};
We also need functions for getting and setting the top matrix
// Gets a copy of the current matrix (top of the stack)
MatrixStack.prototype.getCurrentMatrix = function() {
return this.stack[this.stack.length - 1].slice(); // makes a copy
};
// Lets us set the current matrix
MatrixStack.prototype.setCurrentMatrix = function(m) {
return this.stack[this.stack.length - 1] = m;
};
Finally we need to implement translate
, rotate
, and scale
using our
previous matrix functions.
// Translates the current matrix
MatrixStack.prototype.translate = function(x, y, z) {
var m = this.getCurrentMatrix();
this.setCurrentMatrix(m4.translate(m, x, y, z));
};
// Rotates the current matrix around Z
MatrixStack.prototype.rotateZ = function(angleInRadians) {
var m = this.getCurrentMatrix();
this.setCurrentMatrix(m4.zRotate(m, angleInRadians));
};
// Scales the current matrix
MatrixStack.prototype.scale = function(x, y, z) {
var m = this.getCurrentMatrix();
this.setCurrentMatrix(m4.scale(m, x, y, z));
};
Note we’re using the 3d matrix math functions. We could just use 0
for z
on translation and 1
for z
on scale but I find that I’m so used to using the 2d functions from Canvas 2d
that I often forget to specify z
an then the code breaks so let’s make z
optional
// Translates the current matrix
MatrixStack.prototype.translate = function(x, y, z) {
+ if (z === undefined) {
+ z = 0;
+ }
var m = this.getCurrentMatrix();
this.setCurrentMatrix(m4.translate(m, x, y, z));
};
...
// Scales the current matrix
MatrixStack.prototype.scale = function(x, y, z) {
+ if (z === undefined) {
+ z = 1;
+ }
var m = this.getCurrentMatrix();
this.setCurrentMatrix(m4.scale(m, x, y, z));
};
Using our drawImage
from the previous lesson we had these lines
// this matrix will convert from pixels to clip space
var matrix = m4.orthographic(
0, gl.canvas.clientWidth, gl.canvas.clientHeight, 0, -1, 1);
// translate our quad to dstX, dstY
matrix = m4.translate(matrix, dstX, dstY, 0);
// scale our 1 unit quad
// from 1 unit to dstWidth, dstHeight units
matrix = m4.scale(matrix, dstWidth, dstHeight, 1);
We just need to create a matrix stack
var matrixStack = new MatrixStack();
and multiply in the top matrix from our stack in
// this matrix will convert from pixels to clip space
var matrix = m4.orthographic(
0, gl.canvas.clientWidth, gl.canvas.clientHeight, 0, -1, 1);
+// The matrix stack is in pixels so it goes after the projection
+// above which converted our space from clip space to pixel space
+matrix = m4.multiply(matrix, matrixStack.getCurrentMatrix());
// translate our quad to dstX, dstY
matrix = m4.translate(matrix, dstX, dstY, 0);
// scale our 1 unit quad
// from 1 unit to dstWidth, dstHeight units
matrix = m4.scale(matrix, dstWidth, dstHeight, 1);
And now we can use it the same way we’d use it with the Canvas 2D API.
If you’re not aware of how to use the matrix stack you can think of it as moving and orientating the origin of the canvas. So for example by default in a 2D canvas the origin (0,0) is at the top left corner.
For example if we move the origin to the center of the canvas then drawing an image at 0,0 will draw it starting at the center of the canvas
Let’s take our previous example and just draw a single image
var textureInfo = loadImageAndCreateTextureInfo('resources/star.jpg');
function draw(time) {
gl.clear(gl.COLOR_BUFFER_BIT);
matrixStack.save();
matrixStack.translate(gl.canvas.width / 2, gl.canvas.height / 2);
matrixStack.rotateZ(time);
drawImage(
textureInfo.texture,
textureInfo.width,
textureInfo.height,
0, 0);
matrixStack.restore();
}
And here it is.
you can see even though we’re passing 0, 0
to drawImage
since we use
matrixStack.translate
to move the origin to the center of the canvas
the image is drawn and rotates around that center.
Let’s move the center of rotation to center of the image
matrixStack.translate(gl.canvas.width / 2, gl.canvas.height / 2);
matrixStack.rotateZ(time);
+matrixStack.translate(textureInfo.width / -2, textureInfo.height / -2);
And now it rotates around the center of the image in the center of the canvas
Let’s draw the same image at each corner rotating on different corners
matrixStack.translate(gl.canvas.width / 2, gl.canvas.height / 2);
matrixStack.rotateZ(time);
+matrixStack.save();
+{
+ matrixStack.translate(textureInfo.width / -2, textureInfo.height / -2);
+
+ drawImage(
+ textureInfo.texture,
+ textureInfo.width,
+ textureInfo.height,
+ 0, 0);
+
+}
+matrixStack.restore();
+
+matrixStack.save();
+{
+ // We're at the center of the center image so go to the top/left corner
+ matrixStack.translate(textureInfo.width / -2, textureInfo.height / -2);
+ matrixStack.rotateZ(Math.sin(time * 2.2));
+ matrixStack.scale(0.2, 0.2);
+ // Now we want the bottom/right corner of the image we're about to draw
+ matrixStack.translate(-textureInfo.width, -textureInfo.height);
+
+ drawImage(
+ textureInfo.texture,
+ textureInfo.width,
+ textureInfo.height,
+ 0, 0);
+
+}
+matrixStack.restore();
+
+matrixStack.save();
+{
+ // We're at the center of the center image so go to the top/right corner
+ matrixStack.translate(textureInfo.width / 2, textureInfo.height / -2);
+ matrixStack.rotateZ(Math.sin(time * 2.3));
+ matrixStack.scale(0.2, 0.2);
+ // Now we want the bottom/left corner of the image we're about to draw
+ matrixStack.translate(0, -textureInfo.height);
+
+ drawImage(
+ textureInfo.texture,
+ textureInfo.width,
+ textureInfo.height,
+ 0, 0);
+
+}
+matrixStack.restore();
+
+matrixStack.save();
+{
+ // We're at the center of the center image so go to the bottom/left corner
+ matrixStack.translate(textureInfo.width / -2, textureInfo.height / 2);
+ matrixStack.rotateZ(Math.sin(time * 2.4));
+ matrixStack.scale(0.2, 0.2);
+ // Now we want the top/right corner of the image we're about to draw
+ matrixStack.translate(-textureInfo.width, 0);
+
+ drawImage(
+ textureInfo.texture,
+ textureInfo.width,
+ textureInfo.height,
+ 0, 0);
+
+}
+matrixStack.restore();
+
+matrixStack.save();
+{
+ // We're at the center of the center image so go to the bottom/right corner
+ matrixStack.translate(textureInfo.width / 2, textureInfo.height / 2);
+ matrixStack.rotateZ(Math.sin(time * 2.5));
+ matrixStack.scale(0.2, 0.2);
+ // Now we want the top/left corner of the image we're about to draw
+ matrixStack.translate(0, 0); // 0,0 means this line is not really doing anything
+
+ drawImage(
+ textureInfo.texture,
+ textureInfo.width,
+ textureInfo.height,
+ 0, 0);
+
+}
+matrixStack.restore();
And here’s that
If you think of the various matrix stack functions, translate
, rotateZ
, and scale
as moving the origin then the way I think of setting the center of rotation is
where would I have to move the origin so that when I call drawImage a certain part
of the image is at the previous origin?
In other words let’s say on a 400x300 canvas I call matrixStack.translate(210, 150)
.
At that point the origin is at 210, 150 and all drawing will be relative that point.
If we call drawImage
with 0, 0
this is where the image will be drawn.
Lets say we want the center of rotation to be the bottom right. In that case
where would be have to move the origin so that when we call drawImage
the point we want to be the center of rotation is at the current origin?
For the bottom right of the texture that would be -textureWidth, -textureHeight
so now when we call drawImage
with 0, 0
the texture would be drawn here
and it’s bottom right corner as at the previous origin.
At any point whatever we did before that on the matrix stack it doesn’t matter. We did a bunch
of stuff to move or scale or rotate the origin but just before we call
drawImage
wherever the origin happens to be at the moment is irrelevant.
It’s the new origin so we just have to decide where to move that origin
relative where the texture would be drawn if we had nothing before it on the stack.
You might notice a matrix stack is very similar to a scene graph that we covered before. A scene graph had a tree of nodes and as we walked the tree we multiplied each node by its parent’s node. A matrix stack is effectively another version that same process.