1 module gfm.opengl.matrixstack;
2 
3 import std.c.stdlib : malloc, free;
4 
5 import gfm.opengl.opengl;
6 
7 import gfm.math.vector,
8        gfm.math.matrix;
9 
10 /// A matrix stack designed to replace fixed-pipeline matrix stacks.
11 /// This stack always expose both the top element and its inverse.
12 final class MatrixStack(int R, T) if (R == 3 || R == 4)
13 {
14     public
15     {
16         alias Matrix!(T, R, R) matrix_t; /// Type of matrices in the stack. Can be 3x3 or 4x4.
17 
18         /// Creates a matrix stack.
19         /// The stack is initialized with one element, an identity matrix.
20         this(size_t depth = 32) nothrow
21         {
22             assert(depth > 0);
23             size_t memNeeded = matrix_t.sizeof * depth * 2;
24             void* data = malloc(memNeeded * 2);
25             _matrices = cast(matrix_t*)data;
26             _invMatrices = cast(matrix_t*)(data + memNeeded);
27             _top = 0;
28             _depth = depth;
29             loadIdentity();
30         }
31 
32         /// Releases the matrix stack memory.
33         ~this()
34         {
35             if (_matrices !is null)
36             {
37                 ensureNotInGC("MatrixStack");
38                 free(_matrices);
39                 _matrices = null;
40             }
41         }
42         deprecated("Use .destroy instead") void close(){}
43 
44         /// Replacement for $(D glLoadIdentity).
45         void loadIdentity() pure nothrow
46         {
47             _matrices[_top] = matrix_t.identity();
48             _invMatrices[_top] = matrix_t.identity();
49         }
50 
51         /// Replacement for $(D glPushMatrix).
52         void push() pure nothrow
53         {
54             if(_top + 1 >= _depth)
55                 assert(false, "Matrix stack is full");
56 
57             _matrices[_top + 1] = _matrices[_top];
58             _invMatrices[_top + 1] = _invMatrices[_top];
59             ++_top;
60         }
61 
62         /// Replacement for $(D glPopMatrix).
63         void pop() pure nothrow
64         {
65             if (_top <= 0)
66                 assert(false, "Matrix stack is empty");
67 
68             --_top;
69         }
70 
71         /// Returns: Top matrix.
72         /// Replaces $(D glLoadMatrix).
73         matrix_t top() pure const nothrow
74         {
75             return _matrices[_top];
76         }
77 
78         /// Returns: Inverse of top matrix.
79         matrix_t invTop() pure const nothrow
80         {
81             return _invMatrices[_top];
82         }
83 
84         /// Sets top matrix.
85         /// Replaces $(D glLoadMatrix).
86         void setTop(matrix_t m) pure nothrow
87         {
88             _matrices[_top] = m;
89             _invMatrices[_top] = m.inverse();
90         }
91 
92         /// Replacement for $(D glMultMatrix).
93         void mult(matrix_t m) pure nothrow
94         {
95             mult(m, m.inverse());
96         }
97 
98         /// Replacement for $(D glMultMatrix), with provided inverse.
99         void mult(matrix_t m, matrix_t invM) pure nothrow
100         {
101             _matrices[_top] = _matrices[_top] * m;
102             _invMatrices[_top] = invM *_invMatrices[_top];
103         }
104 
105         /// Replacement for $(D glTranslate).
106         void translate(Vector!(T, R-1) v) pure nothrow
107         {
108             mult(matrix_t.translation(v), matrix_t.translation(-v));
109         }
110 
111         /// Replacement for $(D glScale).
112         void scale(Vector!(T, R-1) v) pure nothrow
113         {
114             mult(matrix_t.scaling(v), matrix_t.scaling(1 / v));
115         }
116 
117         static if (R == 4)
118         {
119             /// Replacement for $(D glRotate).
120             /// Warning: Angle is given in radians, unlike the original API.
121             void rotate(T angle, Vector!(T, 3) axis) pure nothrow
122             {
123                 matrix_t rot = matrix_t.rotation(angle, axis);
124                 mult(rot, rot.transposed()); // inversing a rotation matrix is tranposing
125             }
126 
127             /// Replacement for $(D gluPerspective).
128             /// Warning: FOV is given in radians, unlike the original API.
129             void perspective(T FOVInRadians, T aspect, T zNear, T zFar) pure nothrow
130             {
131                 mult(matrix_t.perspective(FOVInRadians, aspect, zNear, zFar));
132             }
133 
134             /// Replacement for $(D glOrtho).
135             void ortho(T left, T right, T bottom, T top, T near, T far) pure nothrow
136             {
137                 mult(matrix_t.orthographic(left, right, bottom, top, near, far));
138             }
139 
140             /// Replacement for $(D gluLookAt).
141             void lookAt(vec3!T eye, vec3!T target, vec3!T up) pure nothrow
142             {
143                 mult(matrix_t.lookAt(eye, target, up));
144             }
145         }
146     }
147 
148     private
149     {
150         size_t _top; // index of top matrix
151         size_t _depth;
152         matrix_t* _matrices;
153         matrix_t* _invMatrices;
154     }
155 }
156 
157 unittest
158 {
159     auto s = new MatrixStack!(4, double)();
160     scope(exit) destroy(s);
161     s.loadIdentity();
162     s.push();
163     s.pop();
164     s.translate(vec3d(4, 5, 6));
165     s.scale(vec3d(0.5));
166 
167 
168     auto t = new MatrixStack!(3, float)();
169     scope(exit) destroy(t);
170     t.loadIdentity();
171     t.push();
172     t.pop();
173     t.translate(vec2f(-4, 5));
174     t.scale(vec2f(0.5));
175 }