TDME2 1.9.121
Object3DNodeMesh.cpp
Go to the documentation of this file.
2
3#include <map>
4#include <string>
5
6#include <tdme/tdme.h>
21#include <tdme/math/Math.h>
22#include <tdme/math/Matrix4x4.h>
23#include <tdme/math/Vector3.h>
28
29using std::map;
30using std::string;
31
52
53Object3DNodeMesh::Object3DNodeMesh(Object3DNodeRenderer* object3DNodeRenderer, Engine::AnimationProcessingTarget animationProcessingTarget, Node* node, const vector<map<string, Matrix4x4*>*>& transformationMatrices, const vector<map<string, Matrix4x4*>*>& skinningMatrices, int instances)
54{
55 //
56 this->instances = instances;
57 this->object3DNodeRenderer = object3DNodeRenderer;
58 this->node = node;
59 // node data
60 auto& nodeVertices = node->getVertices();
61 auto& nodeNormals = node->getNormals();
62 auto& nodeTextureCoordinates = node->getTextureCoordinates();
63 auto& nodeTangents = node->getTangents();
64 auto& nodeBitangents = node->getBitangents();
65 // determine face count
67 // animation processing target
68 this->animationProcessingTarget = animationProcessingTarget;
69 // transformations for skinned meshes
70 auto skinning = node->getSkinning();
71 this->skinning = skinning != nullptr;
73 if (skinning != nullptr) {
74 jointsSkinningMatrices.resize(instances);
75 for (auto i = 0; i < instances; i++) {
76 for (auto& joint: skinning->getJoints()) {
77 jointsSkinningMatrices[i].push_back(skinningMatrices[i]->find(joint.getNodeId())->second);
78 }
79 }
80 }
81 // set up transformed vertices, normals and friends
82 if (instances > 1 || (skinning != nullptr && animationProcessingTarget == Engine::AnimationProcessingTarget::CPU) ||
83 animationProcessingTarget == Engine::AnimationProcessingTarget::CPU_NORENDERING) {
84 // transformed mesh vertices
85 transformedVertices.resize(nodeVertices.size() * instances);
87 {
88 auto idx = 0;
89 for (auto i = 0; i < instances; i++)
90 for (auto j = 0; j < nodeVertices.size(); j++) {
91 transformedVertices[idx++].set(nodeVertices[j]);
92 }
93 }
94 // transformed mesh normals
95 transformedNormals.resize(nodeNormals.size() * instances);
97 {
98 auto idx = 0;
99 for (auto i = 0; i < instances; i++)
100 for (auto j = 0; j < nodeNormals.size(); j++) {
101 transformedNormals[idx++].set(nodeNormals[j]);
102 }
103 }
104 if (instances > 1) {
105 // transformed mesh texture coordinates
106 transformedTextureCoordinates.resize(nodeTextureCoordinates.size() * instances);
108 {
109 auto idx = 0;
110 for (auto i = 0; i < instances; i++)
111 for (auto j = 0; j < nodeTextureCoordinates.size(); j++) {
112 transformedTextureCoordinates[idx++].set(nodeTextureCoordinates[j]);
113 }
114 }
115 } else {
116 textureCoordinates = &nodeTextureCoordinates;
117 }
118 // transformed mesh tangents
119 if (nodeTangents.size() > 0) {
120 transformedTangents.resize(nodeTangents.size() * instances);
122 {
123 auto idx = 0;
124 for (auto i = 0; i < instances; i++)
125 for (auto j = 0; j < nodeTangents.size(); j++) {
126 transformedTangents[idx++].set(nodeTangents[j]);
127 }
128 }
129 }
130 // transformed mesh bitangents
131 if (nodeBitangents.size() > 0) {
132 transformedBitangents.resize(nodeBitangents.size() * instances);
134 {
135 auto idx = 0;
136 for (auto i = 0; i < instances; i++)
137 for (auto j = 0; j < nodeBitangents.size(); j++) {
138 transformedBitangents[idx++].set(nodeBitangents[j]);
139 }
140 }
141 }
142 } else {
143 // no transformations on CPU, we can use model data
144 vertices = &nodeVertices;
145 normals = &nodeNormals;
146 textureCoordinates = &nodeTextureCoordinates;
147 if (nodeTangents.size() > 0) {
148 tangents = &nodeTangents;
149 }
150 if (nodeBitangents.size() > 0) {
151 bitangents = &nodeBitangents;
152 }
153 }
154
155 // indices
156 auto indicesCount = 0;
157 for (auto& facesEntity : node->getFacesEntities()) {
158 indicesCount += 3 * facesEntity.getFaces().size();
159 }
160 indices.resize(instances * indicesCount);
161 {
162 auto j = 0;
163 for (auto& facesEntity : node->getFacesEntities()) {
164 for (auto i = 0; i < instances; i++) {
165 for (auto& face : facesEntity.getFaces())
166 for (auto& vertexIndex : face.getVertexIndices()) {
167 indices[j++] = nodeVertices.size() * i + vertexIndex;
168 }
169 }
170 }
171 }
172
173 //
174 recreatedBuffers = false;
175 // node transformations matrix
176 if (animationProcessingTarget == Engine::AnimationProcessingTarget::CPU ||
177 animationProcessingTarget == Engine::AnimationProcessingTarget::CPU_NORENDERING ||
178 animationProcessingTarget == Engine::AnimationProcessingTarget::GPU) {
179 // node transformations matrix
180 cNodeTransformationsMatrix = transformationMatrices[0]->find(node->getId())->second;
181 }
182 // skinning
183 if ((skinning != nullptr &&
184 (animationProcessingTarget == Engine::AnimationProcessingTarget::CPU || animationProcessingTarget == Engine::AnimationProcessingTarget::CPU_NORENDERING))) {
185 // skinning computation caches if computing skinning on CPU
186 if (animationProcessingTarget == Engine::AnimationProcessingTarget::CPU || animationProcessingTarget == Engine::AnimationProcessingTarget::CPU_NORENDERING) {
188 cSkinningJointWeight.resize(nodeVertices.size());
190 for (auto i = 0; i < instances; i++) cSkinningJointTransformationsMatrices[i].resize(nodeVertices.size());
191 // compute joint weight caches
192 auto& joints = skinning->getJoints();
193 auto& weights = skinning->getWeights();
194 auto& jointsWeights = skinning->getVerticesJointsWeights();
195 for (auto vertexIndex = 0; vertexIndex < nodeVertices.size(); vertexIndex++) {
196 auto vertexJointWeights = jointsWeights[vertexIndex].size();
197 if (vertexJointWeights > cSkinningMaxVertexWeights) cSkinningMaxVertexWeights = vertexJointWeights;
198 cSkinningJointWeight[vertexIndex].resize(vertexJointWeights);
199 for (auto i = 0; i < instances; i++) cSkinningJointTransformationsMatrices[i][vertexIndex].resize(vertexJointWeights);
200 {
201 auto jointWeightIdx = 0;
202 for (auto& jointWeight : jointsWeights[vertexIndex]) {
203 auto& joint = joints[jointWeight.getJointIndex()];
204 cSkinningJointWeight[vertexIndex][jointWeightIdx] = weights[jointWeight.getWeightIndex()];
205 // next
206 jointWeightIdx++;
207 }
208 }
209 for (auto i = 0; i < instances; i++) {
210 auto jointWeightIdx = 0;
211 for (auto& jointWeight : jointsWeights[vertexIndex]) {
212 auto& joint = joints[jointWeight.getJointIndex()];
213 auto skinningMatrixIt = skinningMatrices[i]->find(joint.getNodeId());
214 cSkinningJointTransformationsMatrices[i][vertexIndex][jointWeightIdx] = skinningMatrixIt->second;
215 // next
216 jointWeightIdx++;
217 }
218 }
219 }
220 }
221 }
223}
224
226{
227 // transformations for skinned meshes
228 auto skinning = node->getSkinning();
229 if (skinning != nullptr) {
230 // compute skinning on CPU if required
231 if (animationProcessingTarget == Engine::AnimationProcessingTarget::GPU) {
232 Engine::getSkinningShader()->computeSkinning(contextIdx, object3DBase, this);
233 } else
234 if (animationProcessingTarget == Engine::AnimationProcessingTarget::CPU || animationProcessingTarget == Engine::AnimationProcessingTarget::CPU_NORENDERING) {
235 auto& nodeVertices = node->getVertices();
236 auto& nodeNormals = node->getNormals();
237 auto& nodeTangent = node->getTangents();
238 auto& nodeBitangent = node->getBitangents();
239 auto& jointsWeights = skinning->getVerticesJointsWeights();
240 const Vector3* vertex;
241 Vector3* transformedVertex;
242 const Vector3* normal;
243 Vector3* transformedNormal;
244 const Vector3* tangent;
245 Vector3* transformedTangent;
246 const Vector3* bitangent;
247 Vector3* transformedBitangent;
248 float totalWeights;
249 float weightNormalized;
250 auto j = 0;
251 Matrix4x4 transformationsMatrix; // TODO: try to avoid multiplying matrix with at each vertex
252 auto currentInstance = object3DBase->getCurrentInstance();
253 for (auto i = 0; i < instances; i++) {
254 if (object3DBase->instanceEnabled[i] == false) continue;
255 object3DBase->setCurrentInstance(i);
256 for (auto vertexIndex = 0; vertexIndex < nodeVertices.size(); vertexIndex++) {
257 // do vertices
258 vertex = &nodeVertices[vertexIndex];
259 transformedVertex = &transformedVertices[nodeVertices.size() * j + vertexIndex].set(0.0f, 0.0f, 0.0f);
260 normal = &nodeNormals[vertexIndex];
261 transformedNormal = &transformedNormals[nodeVertices.size() * j + vertexIndex].set(0.0f, 0.0f, 0.0f);
262 tangent = tangents != nullptr?&nodeTangent[vertexIndex]:nullptr;
263 transformedTangent = tangents != nullptr?&transformedTangents[nodeVertices.size() * j + vertexIndex].set(0.0f, 0.0f, 0.0f):nullptr;
264 bitangent = bitangents != nullptr?&nodeBitangent[vertexIndex]:nullptr;
265 transformedBitangent = bitangents != nullptr?&transformedBitangents[nodeVertices.size() * j + vertexIndex].set(0.0f, 0.0f, 0.0f):nullptr;
266 // compute every influence on vertex and vertex normals
267 totalWeights = 0.0f;
268 for (auto vertexJointWeightIdx = 0; vertexJointWeightIdx < jointsWeights[vertexIndex].size(); vertexJointWeightIdx++) {
269 auto weight = cSkinningJointWeight[vertexIndex][vertexJointWeightIdx];
270 // skinning transformation matrix
271 transformationsMatrix.set(*cSkinningJointTransformationsMatrices[i][vertexIndex][vertexJointWeightIdx]).multiply(object3DBase->getTransformationsMatrix());
272 // vertex
273 transformedVertex->add(transformationsMatrix.multiply(*vertex).scale(weight));
274 // normals
275 transformedNormal->add(transformationsMatrix.multiplyNoTranslation(*normal).scale(weight));
276 // tangent
277 if (tangent != nullptr && transformedTangent != nullptr) {
278 transformedTangent->add(transformationsMatrix.multiplyNoTranslation(*tangent).scale(weight));
279 }
280 // bitangent
281 if (bitangent != nullptr && transformedBitangent != nullptr) {
282 transformedBitangent->add(transformationsMatrix.multiplyNoTranslation(*bitangent).scale(weight));
283 }
284 //
285 totalWeights += weight;
286 }
287 // scale to full weight
288 if (Math::abs(totalWeights - 1.0f) > Math::EPSILON) {
289 weightNormalized = 1.0f / totalWeights;
290 // vertex
291 transformedVertex->scale(weightNormalized);
292 // normals
293 transformedNormal->scale(weightNormalized);
294 // tangent
295 if (transformedTangent != nullptr) {
296 transformedTangent->scale(weightNormalized);
297 }
298 // bitangent
299 if (transformedBitangent != nullptr) {
300 transformedBitangent->scale(weightNormalized);
301 }
302 }
303 // normalize normal
304 transformedNormal->normalize();
305 }
306 j++;
307 }
308 object3DBase->setCurrentInstance(currentInstance);
309 // recreate buffers
311 }
312 } else
313 if (animationProcessingTarget == Engine::AnimationProcessingTarget::CPU_NORENDERING) {
314 auto& nodeVertices = node->getVertices();
315 auto& nodeNormals = node->getNormals();
316 // transformations for non skinned rendering
317 // vertices
318 for (auto vertexIndex = 0; vertexIndex < nodeVertices.size(); vertexIndex++) {
319 transformedVertices[vertexIndex].set(cNodeTransformationsMatrix->multiply(nodeVertices[vertexIndex]));
320 }
321 // normals
322 for (auto normalIndex = 0; normalIndex < nodeNormals.size(); normalIndex++) {
323 transformedNormals[normalIndex].set(cNodeTransformationsMatrix->multiplyNoTranslation(nodeNormals[normalIndex]).normalize());
324 }
325 // TODO: tangents, bitangents, but actually it is only in use for computing bounding volumes, so I am not in a hurry
326 // recreate buffers
328 }
329}
330
332{
333 recreatedBuffers = true;
334}
335
337{
338 if (recreatedBuffers == true) {
339 recreatedBuffers = false;
340 return true;
341 } else {
342 return false;
343 }
344}
345
346void Object3DNodeMesh::setupVertexIndicesBuffer(Renderer *renderer, int contextIdx, int32_t vboId) {
347 // upload
348 if (renderer->isUsingShortIndices() == true) {
349 if (instances * indices.size() > 65535) {
350 Console::println(
351 "Object3DNodeMesh::setupVertexIndicesBuffer(): " +
352 node->getModel()->getName() + ":" +
353 node->getName() + ":" +
354 "more than 2^16-1 indices: " +
355 to_string(indices.size())
356 );
357 }
358 auto sbIndices = ObjectBuffer::getByteBuffer(contextIdx, instances * faceCount * 3 * sizeof(uint16_t))->asShortBuffer();
359 // create face vertex indices, will never be changed in engine
360 for (auto i = 0; i < instances; i++)
361 for (auto index: indices) {
362 sbIndices.put(index);
363 }
364 // done, upload
365 renderer->uploadIndicesBufferObject(contextIdx, vboId, sbIndices.getPosition() * sizeof(uint16_t), &sbIndices);
366 } else {
367 auto ibIndices = ObjectBuffer::getByteBuffer(contextIdx, instances * faceCount * 3 * sizeof(uint32_t))->asIntBuffer();
368 // create face vertex indices, will never be changed in engine
369 for (auto i = 0; i < instances; i++)
370 for (auto index: indices) {
371 ibIndices.put(index);
372 }
373 // done, upload
374 renderer->uploadIndicesBufferObject(contextIdx, vboId, ibIndices.getPosition() * sizeof(uint32_t), &ibIndices);
375 }
376}
377
378
379void Object3DNodeMesh::setupTextureCoordinatesBuffer(Renderer* renderer, int contextIdx, int32_t vboId)
380{
381 if (textureCoordinates->size() == 0) return;
382 // create texture coordinates buffer, will never be changed in engine
383 auto fbTextureCoordinates = ObjectBuffer::getByteBuffer(contextIdx, textureCoordinates->size() * 2 * sizeof(float))->asFloatBuffer();
384 // construct texture coordinates byte buffer as this will not change usually
385 for (auto& textureCoordinate: *textureCoordinates) {
386 fbTextureCoordinates.put(textureCoordinate.getArray());
387 }
388 // done, upload
389 renderer->uploadBufferObject(contextIdx, vboId, fbTextureCoordinates.getPosition() * sizeof(float), &fbTextureCoordinates);
390}
391
392void Object3DNodeMesh::setupVerticesBuffer(Renderer* renderer, int contextIdx, int32_t vboId)
393{
394 auto fbVertices = ObjectBuffer::getByteBuffer(contextIdx, vertices->size() * 3 * sizeof(float))->asFloatBuffer();
395 // create vertices buffers
396 for (auto& vertex: *vertices) {
397 fbVertices.put(vertex.getArray());
398 }
399 // done, upload
400 renderer->uploadBufferObject(contextIdx, vboId, fbVertices.getPosition() * sizeof(float), &fbVertices);
401}
402
403void Object3DNodeMesh::setupNormalsBuffer(Renderer* renderer, int contextIdx, int32_t vboId)
404{
405 auto fbNormals = ObjectBuffer::getByteBuffer(contextIdx, normals->size() * 3 * sizeof(float))->asFloatBuffer();
406 // create normals buffers
407 for (auto& normal: *normals) {
408 fbNormals.put(normal.getArray());
409 }
410 // done, upload
411 renderer->uploadBufferObject(contextIdx, vboId, fbNormals.getPosition() * sizeof(float), &fbNormals);
412}
413
414void Object3DNodeMesh::setupTangentsBuffer(Renderer* renderer, int contextIdx, int32_t vboId)
415{
416 // check if we have tangents
417 if (tangents == nullptr) return;
418 auto fbTangents = ObjectBuffer::getByteBuffer(contextIdx, tangents->size() * 3 * sizeof(float))->asFloatBuffer();
419 // create tangents buffers
420 for (auto& tangent: *tangents) {
421 fbTangents.put(tangent.getArray());
422 }
423 // done, upload
424 renderer->uploadBufferObject(contextIdx, vboId, fbTangents.getPosition() * sizeof(float), &fbTangents);
425}
426
427void Object3DNodeMesh::setupBitangentsBuffer(Renderer* renderer, int contextIdx, int32_t vboId)
428{
429 // check if we have bitangents
430 if (bitangents == nullptr) return;
431 auto fbBitangents = ObjectBuffer::getByteBuffer(contextIdx, bitangents->size() * 3 * sizeof(float))->asFloatBuffer();
432 // create bitangents buffers
433 for (auto& bitangent: *bitangents) {
434 fbBitangents.put(bitangent.getArray());
435 }
436 // done, upload
437 renderer->uploadBufferObject(contextIdx, vboId, fbBitangents.getPosition() * sizeof(float), &fbBitangents);
438}
439
440void Object3DNodeMesh::setupOriginsBuffer(Renderer* renderer, int contextIdx, int32_t vboId) {
441 // check if we have origins
442 auto& origins = node->getOrigins();
443 if (origins.size() == 0) return;
444 // create origins buffer, will never be changed in engine
445 auto fbOrigins = ObjectBuffer::getByteBuffer(contextIdx, origins.size() * 3 * sizeof(float))->asFloatBuffer();
446 // construct origins buffer
447 for (auto& origin: origins) {
448 fbOrigins.put(origin.getArray());
449 }
450 // done, upload
451 renderer->uploadBufferObject(contextIdx, vboId, fbOrigins.getPosition() * sizeof(float), &fbOrigins);
452}
453
454void Object3DNodeMesh::setupLodBuffer(Renderer* renderer, int contextIdx, int32_t vboId, int lodLevel) {
455 // TODO: we only support faces entities 0 lod indices for terrain now
456 const vector<int32_t>* indices { nullptr };
457 switch (lodLevel) {
458 case 1: indices = &node->getFacesEntities()[0].getLOD1Indices(); break;
459 case 2: indices = &node->getFacesEntities()[0].getLOD2Indices(); break;
460 case 3: indices = &node->getFacesEntities()[0].getLOD3Indices(); break;
461 default:
462 Console::println(
463 "Object3DNodeMesh::setupLodBuffer(): " +
464 node->getModel()->getName() + ":" +
465 node->getName() + ":" +
466 "no valid lod level: " + to_string(lodLevel)
467 );
468 return;
469 }
470 if (indices->empty() == true) {
471 Console::println(
472 "Object3DNodeMesh::setupLodBuffer(): " +
473 node->getModel()->getName() + ":" +
474 node->getName() + ":" +
475 "no indices"
476 );
477 return;
478 }
479 if (renderer->isUsingShortIndices() == true) {
480 if (instances * indices->size() > 65535) {
481 Console::println(
482 "Object3DNodeMesh::setupLodBuffer(): " +
483 node->getModel()->getName() + ":" +
484 node->getName() + ":" +
485 "more than 2^16-1 indices: " +
486 to_string(indices->size())
487 );
488 }
489 // create indices buffer, will never be changed in engine
490 auto sbIndices = ObjectBuffer::getByteBuffer(contextIdx, indices->size() * sizeof(uint16_t))->asShortBuffer();
491 // construct indices buffer
492 for (auto index: *indices) {
493 sbIndices.put(index);
494 }
495 renderer->uploadIndicesBufferObject(contextIdx, vboId, sbIndices.getPosition() * sizeof(uint16_t), &sbIndices);
496 } else {
497 // create indices buffer, will never be changed in engine
498 auto ibIndices = ObjectBuffer::getByteBuffer(contextIdx, indices->size() * sizeof(uint32_t))->asIntBuffer();
499 // construct indices buffer
500 for (auto index: *indices) {
501 ibIndices.put(index);
502 }
503 renderer->uploadIndicesBufferObject(contextIdx, vboId, ibIndices.getPosition() * sizeof(uint32_t), &ibIndices);
504 }
505}
static SkinningShader * getSkinningShader()
Definition: Engine.h:435
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
Definition: Face.h:19
Node faces entity A node can have multiple entities containing faces and a applied material.
Definition: FacesEntity.h:28
Joint / Bone.
Definition: Joint.h:19
const string & getName()
Definition: Model.h:126
Model node.
Definition: Node.h:31
int32_t getFaceCount() const
Definition: Node.cpp:109
Model * getModel()
Definition: Node.h:70
const vector< Vector3 > & getTangents() const
Definition: Node.h:199
const vector< Vector3 > & getOrigins() const
Definition: Node.h:276
const vector< Vector3 > & getBitangents() const
Definition: Node.h:212
Skinning * getSkinning()
Definition: Node.h:238
const vector< TextureCoordinate > & getTextureCoordinates() const
Definition: Node.h:186
const string & getId()
Returns id.
Definition: Node.h:85
const vector< Vector3 > & getNormals() const
Definition: Node.h:173
const string & getName()
Definition: Node.h:92
const vector< FacesEntity > & getFacesEntities() const
Definition: Node.h:256
const vector< Vector3 > & getVertices() const
Definition: Node.h:151
Skinning definition for nodes.
Definition: Skinning.h:27
Class representing texture UV coordinates data.
virtual void uploadIndicesBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, ShortBuffer *data)=0
Uploads buffer data to buffer object.
virtual void uploadBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, FloatBuffer *data)=0
Uploads buffer data to buffer object.
const Matrix4x4 & getTransformationsMatrix() const
Definition: Object3DBase.h:279
void setCurrentInstance(int currentInstance)
Set current instance.
Definition: Object3DBase.h:151
Object 3D node mesh specifically for rendering.
void setupOriginsBuffer(Renderer *renderer, int contextIdx, int32_t vboId)
Set up render node object origins data buffer.
void setupNormalsBuffer(Renderer *renderer, int contextIdx, int32_t vboId)
Set up normals buffer.
void computeTransformations(int contextIdx, Object3DBase *object3DBase)
Computes mesh transformations.
void setupVertexIndicesBuffer(Renderer *renderer, int contextIdx, int32_t vboId)
Set up vertex indices buffer.
Engine::AnimationProcessingTarget animationProcessingTarget
vector< map< string, Matrix4x4 * > * > skinningMatrices
void setupLodBuffer(Renderer *renderer, int contextIdx, int32_t vboId, int lodLevel)
Set up render node object lod data buffer.
void setupTextureCoordinatesBuffer(Renderer *renderer, int contextIdx, int32_t vboId)
Set up texture coordinates buffer.
void recreateBuffers()
Recreates node float buffers.
void setupTangentsBuffer(Renderer *renderer, int contextIdx, int32_t vboId)
Set up tangents buffer.
const vector< TextureCoordinate > * textureCoordinates
void setupVerticesBuffer(Renderer *renderer, int contextIdx, int32_t vboId)
Set up vertices buffer.
vector< vector< Matrix4x4 * > > jointsSkinningMatrices
void setupBitangentsBuffer(Renderer *renderer, int contextIdx, int32_t vboId)
Set up bitangents buffer.
vector< vector< vector< Matrix4x4 * > > > cSkinningJointTransformationsMatrices
Buffers used to transfer data between main memory to graphics board memory.
Definition: ObjectBuffer.h:23
static ByteBuffer * getByteBuffer(int contextIdx, int32_t bytes)
Get byte buffer for given context.
Interface to compute shader skinning shader program.
void computeSkinning(int contextIdx, Object3DBase *object3DBase, Object3DNodeMesh *object3DNodeMesh)
Compute skinning.
Standard math functions.
Definition: Math.h:21
4x4 3D Matrix class
Definition: Matrix4x4.h:24
Matrix4x4 & set(float r0c0, float r1c0, float r2c0, float r3c0, float r0c1, float r1c1, float r2c1, float r3c1, float r0c2, float r1c2, float r2c2, float r3c2, float r0c3, float r1c3, float r2c3, float r3c3)
Set up matrix by values.
Definition: Matrix4x4.h:95
Vector3 multiplyNoTranslation(const Vector3 &v) const
Multiplies a vector3 with this matrix ignoring translation.
Definition: Matrix4x4.h:364
Vector3 multiply(const Vector3 &v) const
Multiplies a vector3 with this matrix into destination vector.
Definition: Matrix4x4.h:351
3D vector 3 class
Definition: Vector3.h:22
Vector3 & normalize()
Normalize the vector.
Definition: Vector3.h:288
Vector3 & add(const Vector3 &v)
Adds a vector.
Definition: Vector3.h:301
Vector3 & scale(float scale)
Scale this vector.
Definition: Vector3.h:349
Byte buffer class.
Definition: ByteBuffer.h:24
ShortBuffer asShortBuffer()
Definition: ByteBuffer.h:50
FloatBuffer asFloatBuffer()
Definition: ByteBuffer.h:36
Console class.
Definition: Console.h:26
Float buffer class.
Definition: FloatBuffer.h:18
FloatBuffer * put(float value)
Put a float value into float buffer.
Definition: FloatBuffer.h:52
IntBuffer * put(int32_t value)
Puts a value into buffer at its current position.
Definition: IntBuffer.h:59
Short buffer class.
Definition: ShortBuffer.h:14
ShortBuffer * put(int16_t value)
Put a value into current position.
Definition: ShortBuffer.h:58