TDME2 1.9.121
SceneReader.cpp
Go to the documentation of this file.
2
3#include <string>
4
5#include <tdme/tdme.h>
27#include <tdme/math/Math.h>
28#include <tdme/math/Vector3.h>
29#include <tdme/math/Vector4.h>
38
39#include <ext/rapidjson/document.h>
40
41using std::string;
42using std::to_string;
43
77
78using rapidjson::Document;
79using rapidjson::Value;
80
81Scene* SceneReader::read(const string& pathName, const string& fileName, ProgressCallback* progressCallback, PrototypeTransformFilter* prototypeTransformFilter)
82{
83 return read(pathName, fileName, "", progressCallback, prototypeTransformFilter);
84}
85
86Scene* SceneReader::read(const string& pathName, const string& fileName, const string& objectIdPrefix, ProgressCallback* progressCallback, PrototypeTransformFilter* prototypeTransformFilter)
87{
88 if (progressCallback != nullptr) progressCallback->progress(0.0f);
89
90 auto jsonContent = FileSystem::getInstance()->getContentAsString(pathName, fileName);
91 if (progressCallback != nullptr) progressCallback->progress(0.165f);
92
93 Document jRoot;
94 jRoot.Parse(jsonContent.c_str());
95 if (progressCallback != nullptr) progressCallback->progress(0.33f);
96
97 //
98 auto scene = new Scene(fileName, "");
99 scene->setApplicationRootPathName(Tools::getApplicationRootPathName(pathName));
100 // auto version = Float::parseFloat((jRoot["version"].GetString()));
101 scene->setRotationOrder(jRoot.FindMember("ro") != jRoot.MemberEnd()?RotationOrder::valueOf(jRoot["ro"].GetString()):RotationOrder::XYZ);
102 for (auto i = 0; i < jRoot["properties"].GetArray().Size(); i++) {
103 auto& jSceneProperty = jRoot["properties"].GetArray()[i];
104 scene->addProperty(
105 jSceneProperty["name"].GetString(),
106 jSceneProperty["value"].GetString()
107 );
108 }
109 if (jRoot.FindMember("lights") != jRoot.MemberEnd()) {
110 auto lightIdx = 0;
111 auto jLights = jRoot["lights"].GetArray();
112 for (auto i = 0; i < jLights.Size(); i++) {
113 auto& jLight = jLights[i];
114 if (jLight["e"].GetBool() == false) continue;
115 auto light = lightIdx < scene->getLightCount()?(scene->getLightAt(jLight.FindMember("id") != jLight.MemberEnd()?jLight["id"].GetInt():lightIdx)):scene->addLight();
116 light->setAmbient(
117 Color4(
118 jLight["ar"].GetFloat(),
119 jLight["ag"].GetFloat(),
120 jLight["ab"].GetFloat(),
121 jLight["aa"].GetFloat()
122 )
123 );
124 light->setDiffuse(
125 Color4(
126 jLight["dr"].GetFloat(),
127 jLight["dg"].GetFloat(),
128 jLight["db"].GetFloat(),
129 jLight["da"].GetFloat()
130 )
131 );
132 light->setSpecular(
133 Color4(
134 jLight["sr"].GetFloat(),
135 jLight["sg"].GetFloat(),
136 jLight["sb"].GetFloat(),
137 jLight["sa"].GetFloat()
138 )
139 );
140 light->setPosition(
141 Vector4(
142 jLight["px"].GetFloat(),
143 jLight["py"].GetFloat(),
144 jLight["pz"].GetFloat(),
145 jLight["pw"].GetFloat()
146 )
147 );
148 light->setConstantAttenuation(jLight["ca"].GetFloat());
149 light->setLinearAttenuation(jLight["la"].GetFloat());
150 light->setQuadraticAttenuation(jLight["qa"].GetFloat());
151 light->setSpotDirection(
152 Vector3(
153 jLight["sdx"].GetFloat(),
154 jLight["sdy"].GetFloat(),
155 jLight["sdz"].GetFloat()
156 )
157 );
158 light->setSpotExponent(jLight["se"].GetFloat());
159 light->setSpotCutOff(jLight["sco"].GetFloat());
160 light->setEnabled(jLight["e"].GetBool());
161 lightIdx++;
162 }
163 }
164
165 auto progressStepCurrent = 0;
166 auto jPrototypes = jRoot["models"].GetArray();
167 for (auto i = 0; i < jPrototypes.Size(); i++) {
168 auto& jPrototype = jPrototypes[i];
169 Prototype* prototype = nullptr;
170 try {
171 auto embedded = jPrototype.FindMember("e") != jPrototype.MemberEnd()?jPrototype["e"].GetBool():true;
172 if (embedded == true) {
173 prototype = PrototypeReader::read(
174 jPrototype["id"].GetInt(),
175 pathName,
176 jPrototype["entity"],
177 prototypeTransformFilter
178 );
179 prototype->setEmbedded(true);
180 } else {
181 auto externalPrototypePathName = PrototypeReader::getResourcePathName(pathName, jPrototype["pf"].GetString());
182 auto externalPrototypeFileName = FileSystem::getInstance()->getFileName(jPrototype["pf"].GetString());
183 prototype = PrototypeReader::read(
184 jPrototype["id"].GetInt(),
185 externalPrototypePathName,
186 externalPrototypeFileName,
187 prototypeTransformFilter
188 );
189 prototype->setEmbedded(false);
190 }
191 } catch (Exception& exception) {
192 Console::println(string() + "SceneReader::read(): An error occurred: " + exception.what());
193 delete scene;
194 throw exception;
195 }
196 if (prototype == nullptr) {
197 Console::println("SceneReader::read(): Invalid prototype = " + to_string(jPrototype["id"].GetInt()));
198 continue;
199 }
200 scene->getLibrary()->addPrototype(prototype);
201 if (jPrototype.FindMember("properties") != jPrototype.MemberEnd()) {
202 for (auto j = 0; j < jPrototype["properties"].GetArray().Size(); j++) {
203 auto& jPrototypeProperty = jPrototype["properties"].GetArray()[j];
204 prototype->addProperty(
205 jPrototypeProperty["name"].GetString(),
206 jPrototypeProperty["value"].GetString()
207 );
208 }
209 }
210
211 if (progressCallback != nullptr) progressCallback->progress(0.33f + static_cast<float>(progressStepCurrent) / static_cast<float>(jRoot["models"].GetArray().Size()) * 0.33f);
212 progressStepCurrent++;
213 }
214
215 auto jEntities = jRoot["objects"].GetArray();
216 for (auto i = 0; i < jEntities.Size(); i++) {
217 auto& jSceneEntity = jEntities[i];
218 auto prototype = scene->getLibrary()->getPrototype(jSceneEntity["mid"].GetInt());
219 if (prototype == nullptr) {
220 Console::println("SceneReader::read(): No prototype found with id = " + to_string(jSceneEntity["mid"].GetInt()));
221
222 if (progressCallback != nullptr && progressStepCurrent % 1000 == 0) progressCallback->progress(0.66f + static_cast<float>(progressStepCurrent) / static_cast<float>(jRoot["objects"].GetArray().Size()) * 0.33f);
223 progressStepCurrent++;
224
225 continue;
226 }
227
228 Transformations transformations;
229 transformations.setPivot(prototype->getPivot());
230 transformations.setTranslation(
231 Vector3(
232 jSceneEntity["tx"].GetFloat(),
233 jSceneEntity["ty"].GetFloat(),
234 jSceneEntity["tz"].GetFloat()
235 )
236 );
237 transformations.setScale(
238 Vector3(
239 jSceneEntity["sx"].GetFloat(),
240 jSceneEntity["sy"].GetFloat(),
241 jSceneEntity["sz"].GetFloat()
242 )
243 );
244 Vector3 rotation(
245 jSceneEntity["rx"].GetFloat(),
246 jSceneEntity["ry"].GetFloat(),
247 jSceneEntity["rz"].GetFloat()
248 );
249 transformations.addRotation(scene->getRotationOrder()->getAxis0(), rotation.getArray()[scene->getRotationOrder()->getAxis0VectorIndex()]);
250 transformations.addRotation(scene->getRotationOrder()->getAxis1(), rotation.getArray()[scene->getRotationOrder()->getAxis1VectorIndex()]);
251 transformations.addRotation(scene->getRotationOrder()->getAxis2(), rotation.getArray()[scene->getRotationOrder()->getAxis2VectorIndex()]);
252 transformations.update();
253 auto sceneEntity = new SceneEntity(
254 objectIdPrefix != "" ?
255 objectIdPrefix + jSceneEntity["id"].GetString() :
256 (jSceneEntity["id"].GetString()),
257 jSceneEntity.FindMember("descr") != jSceneEntity.MemberEnd()?jSceneEntity["descr"].GetString() : "",
258 transformations,
259 prototype
260 );
261 if (jSceneEntity.FindMember("properties") != jSceneEntity.MemberEnd()) {
262 for (auto j = 0; j < jSceneEntity["properties"].GetArray().Size(); j++) {
263 auto& jSceneEntityProperty = jSceneEntity["properties"].GetArray()[j];
264 sceneEntity->addProperty(
265 jSceneEntityProperty["name"].GetString(),
266 jSceneEntityProperty["value"].GetString()
267 );
268 }
269 }
270 sceneEntity->setReflectionEnvironmentMappingId(jSceneEntity.FindMember("r") != jSceneEntity.MemberEnd()?jSceneEntity["r"].GetString():"");
271 scene->addEntity(sceneEntity);
272
273 if (progressCallback != nullptr && progressStepCurrent % 1000 == 0) progressCallback->progress(0.66f + static_cast<float>(progressStepCurrent) / static_cast<float>(jRoot["objects"].GetArray().Size()) * 0.33f);
274 progressStepCurrent++;
275 }
276 scene->setEntityIdx(jRoot["objects_eidx"].GetInt());
277 scene->setFileName((pathName.empty() == false?pathName + "/":"") + fileName);
278 scene->update();
279
280 //
281 if (jRoot.FindMember("sky") != jRoot.MemberEnd()) {
282 auto& jSky = jRoot["sky"];
283 scene->setSkyModelFileName(jSky["file"].GetString());
284 scene->setSkyModelScale(
285 Vector3(
286 jSky["sx"].GetFloat(),
287 jSky["sy"].GetFloat(),
288 jSky["sz"].GetFloat()
289 )
290 );
291 if (scene->getSkyModelFileName().empty() == false) {
292 auto skyModelPathName = PrototypeReader::getResourcePathName(pathName, scene->getSkyModelFileName());
293 auto skyModelFileName = FileSystem::getInstance()->getFileName(scene->getSkyModelFileName());
294 scene->setSkyModelFileName(skyModelPathName + "/" + skyModelFileName);
295 try {
296 scene->setSkyModel(
297 ModelReader::read(
298 skyModelPathName,
299 skyModelFileName
300 )
301 );
302 } catch (Exception& exception) {
303 delete scene;
304 throw exception;
305 }
306 }
307 }
308
309 //
310 if (progressCallback != nullptr) {
311 progressCallback->progress(1.0f);
312 delete progressCallback;
313 }
314
315 //
316 return scene;
317}
318
319void SceneReader::determineMeshNodes(Scene* scene, Node* node, const string& parentName, const Matrix4x4& parentTransformationsMatrix, vector<PrototypeMeshNode>& meshNodes) {
320 auto sceneLibrary = scene->getLibrary();
321 auto nodeId = node->getId();
322 if (parentName.length() > 0) nodeId = parentName + "." + nodeId;
323 auto modelName = nodeId;
324 modelName = StringTools::regexReplace(modelName, "[-_]{1}[0-9]+$", "");
325 modelName = StringTools::regexReplace(modelName, "[0-9]+$", "");
326 auto haveName = sceneLibrary->getPrototypeCount() == 0;
327 if (haveName == false) {
328 for (auto i = 0; i < 10000; i++) {
329 haveName = true;
330 auto modelNameTry = modelName;
331 if (i > 0) modelNameTry+= to_string(i);
332 for (auto entityIdx = 0; entityIdx < sceneLibrary->getPrototypeCount(); entityIdx++) {
333 auto entity = sceneLibrary->getPrototypeAt(entityIdx);
334 if (entity->getName() == modelNameTry) {
335 haveName = false;
336 break;
337 }
338 }
339 if (haveName == true) {
340 modelName = modelNameTry;
341 break;
342 }
343 }
344 }
345 if (haveName == false) {
346 Console::println(
347 string(
348 "SceneReader::doImportFromModel(): Skipping model '" +
349 modelName +
350 "' as no name could be created for it."
351 )
352 );
353 return;
354 }
355 Matrix4x4 transformationsMatrix;
356 // compute animation matrix if animation setups exist
357 auto animation = node->getAnimation();
358 if (animation != nullptr) {
359 auto& animationMatrices = animation->getTransformationsMatrices();
360 transformationsMatrix.set(animationMatrices[0 % animationMatrices.size()]);
361 } else {
362 // no animation matrix, set up local transformation matrix up as node matrix
363 transformationsMatrix.set(node->getTransformationsMatrix());
364 }
365
366 // apply parent transformation matrix
367 transformationsMatrix.multiply(parentTransformationsMatrix);
368
369 // check if no mesh?
370 if (node->getVertices().size() == 0 && node->getSubNodes().size() > 0) {
371 // ok, check sub meshes
372 for (auto subNodeIt: node->getSubNodes()) {
373 determineMeshNodes(scene, subNodeIt.second, nodeId, transformationsMatrix.clone(), meshNodes);
374 }
375 } else {
376 // add to node meshes, even if empty as its a empty :D
377 PrototypeMeshNode prototypeMeshNode;
378 prototypeMeshNode.id = nodeId;
379 prototypeMeshNode.name = modelName;
380 prototypeMeshNode.node = node;
381 prototypeMeshNode.transformationsMatrix.set(transformationsMatrix);
382 meshNodes.push_back(prototypeMeshNode);
383 }
384}
385
386Scene* SceneReader::readFromModel(const string& pathName, const string& fileName, ProgressCallback* progressCallback) {
387 if (progressCallback != nullptr) progressCallback->progress(0.0f);
388
389 string modelPathName = pathName + "/" + fileName + "-models";
390 if (FileSystem::getInstance()->fileExists(modelPathName)) {
391 FileSystem::getInstance()->removePath(modelPathName, true);
392 }
393 FileSystem::getInstance()->createPath(modelPathName);
394
395 auto sceneModel = ModelReader::read(pathName, fileName);
396
397 if (progressCallback != nullptr) progressCallback->progress(0.1f);
398
399 auto upVector = sceneModel->getUpVector();
400 RotationOrder* rotationOrder = sceneModel->getRotationOrder();
401
402 //
403 auto scene = new Scene(fileName, "");
404 scene->setRotationOrder(rotationOrder);
405
406 auto sceneLibrary = scene->getLibrary();
407 auto nodeIdx = 0;
408 Prototype* emptyPrototype = nullptr;
409 Matrix4x4 sceneModelImportRotationMatrix;
410 Vector3 sceneModelScale;
411 sceneModelImportRotationMatrix.set(sceneModel->getImportTransformationsMatrix());
412 sceneModelImportRotationMatrix.getScale(sceneModelScale);
413 sceneModelImportRotationMatrix.scale(Vector3(1.0f / sceneModelScale.getX(), 1.0f / sceneModelScale.getY(), 1.0f / sceneModelScale.getZ()));
414 auto progressTotal = sceneModel->getSubNodes().size();
415 auto progressIdx = 0;
416 for (auto nodeIt: sceneModel->getSubNodes()) {
417 if (progressCallback != nullptr) progressCallback->progress(0.1f + static_cast<float>(progressIdx) / static_cast<float>(progressTotal) * 0.8f);
418 vector<PrototypeMeshNode> meshNodes;
419 determineMeshNodes(scene, nodeIt.second, "", (Matrix4x4()).identity(), meshNodes);
420 for (auto& meshNode: meshNodes) {
421 auto model = new Model(
422 meshNode.name + ".tm",
423 fileName + "-" + meshNode.name,
424 upVector,
425 rotationOrder,
426 nullptr
427 );
428 model->setImportTransformationsMatrix(sceneModel->getImportTransformationsMatrix());
429 float importFixScale = 1.0f;
430 Vector3 translation, scale;
431 Vector3 xAxis, yAxis, zAxis;
432 Matrix4x4 nodeTransformationsMatrix;
433 nodeTransformationsMatrix.set(meshNode.transformationsMatrix);
434 nodeTransformationsMatrix.getAxes(xAxis, yAxis, zAxis);
435 nodeTransformationsMatrix.getTranslation(translation);
436 nodeTransformationsMatrix.getScale(scale);
437 xAxis.normalize();
438 yAxis.normalize();
439 zAxis.normalize();
440 nodeTransformationsMatrix.setAxes(xAxis, yAxis, zAxis);
441 if ((upVector == UpVector::Y_UP && Vector3::computeDotProduct(Vector3::computeCrossProduct(xAxis, yAxis), zAxis) < 0.0f) ||
442 (upVector == UpVector::Z_UP && Vector3::computeDotProduct(Vector3::computeCrossProduct(xAxis, zAxis), yAxis) < 0.0f)) {
443 xAxis.scale(-1.0f);
444 yAxis.scale(-1.0f);
445 zAxis.scale(-1.0f);
446 nodeTransformationsMatrix.setAxes(xAxis, yAxis, zAxis);
447 scale.scale(-1.0f);
448 }
449 auto rotation = nodeTransformationsMatrix.computeEulerAngles();
450 scale = sceneModelImportRotationMatrix.multiply(scale);
451 rotation = sceneModelImportRotationMatrix.multiply(rotation);
452 translation = model->getImportTransformationsMatrix().multiply(translation);
453
454 ModelTools::cloneNode(meshNode.node, model);
455 if (model->getSubNodes().begin() != model->getSubNodes().end()) {
456 model->getSubNodes().begin()->second->setTransformationsMatrix(Matrix4x4().identity());
457 }
458 model->addAnimationSetup(Model::ANIMATIONSETUP_DEFAULT, 0, 0, true);
459 ModelTools::prepareForIndexedRendering(model);
460 // scale up model if dimension too less, this occurres with importing FBX that was exported by UE
461 // TODO: maybe make this conditional
462 {
463 auto width = model->getBoundingBox()->getDimensions().getX();
464 auto height = model->getBoundingBox()->getDimensions().getY();
465 auto depth = model->getBoundingBox()->getDimensions().getZ();
466 if (width < 0.2f && height < 0.2f && depth < 0.2f) {
467 if (width > Math::EPSILON && width < height && width < depth) {
468 importFixScale = 1.0f / width / 5.0f;
469 } else
470 if (height > Math::EPSILON && height < width && height < depth) {
471 importFixScale = 1.0f / height / 5.0f;
472 } else
473 if (depth > Math::EPSILON) {
474 importFixScale = 1.0f / depth / 5.0f;
475 }
476 }
477 model->setImportTransformationsMatrix(model->getImportTransformationsMatrix().clone().scale(importFixScale));
478 model->getBoundingBox()->getMin().scale(importFixScale);
479 model->getBoundingBox()->getMax().scale(importFixScale);
480 model->getBoundingBox()->update();
481 scale.scale(1.0f / importFixScale);
482 }
483 auto prototypeType = Prototype_Type::MODEL;
484 if (meshNode.node->getVertices().size() == 0) {
485 prototypeType = Prototype_Type::EMPTY;
486 delete model;
487 model = nullptr;
488 }
489 Prototype* prototype = nullptr;
490 if (prototypeType == Prototype_Type::MODEL && model != nullptr) {
491 for (auto i = 0; i < scene->getLibrary()->getPrototypeCount(); i++) {
492 auto prototypeCompare = scene->getLibrary()->getPrototypeAt(i);
493 if (prototypeCompare->getType() != Prototype_Type::MODEL)
494 continue;
495
496 if (ModelUtilities::equals(model, prototypeCompare->getModel()) == true) {
497 prototype = prototypeCompare;
498 delete model;
499 model = nullptr;
500 break;
501 }
502 }
503 if (prototype == nullptr && model != nullptr) {
504 auto modelFileName = meshNode.name + ".tm";
505 try {
506 TMWriter::write(
507 model,
508 modelPathName,
509 modelFileName
510 );
511 } catch (Exception& exception) {
512 delete model;
513 delete scene;
514 delete sceneModel;
515 throw exception;
516 }
517 prototype = new Prototype(
518 Prototype::ID_NONE,
519 Prototype_Type::MODEL,
520 Tools::removeFileEnding(fileName),
521 Tools::removeFileEnding(fileName),
522 modelPathName + "/" + modelFileName,
523 "resources/engine/models/empty.tm",
524 string(),
525 model,
526 Vector3(0.0f, 0.0f, 0.0f)
527 );
528 sceneLibrary->addPrototype(prototype);
529 }
530 } else
531 if (prototypeType == Prototype_Type::EMPTY) {
532 if (emptyPrototype == nullptr) {
533 emptyPrototype = new Prototype(
534 nodeIdx++,
535 Prototype_Type::EMPTY,
536 "Default Empty",
537 "Default Empty",
538 string(),
539 "resources/engine/models/empty.tm",
540 string(),
541 ModelReader::read("resources/engine/models", "empty.tm"), // TODO: exception
542 Vector3(0.0f, 0.0f, 0.0f)
543 );
544 sceneLibrary->addPrototype(emptyPrototype);
545 }
546 prototype = emptyPrototype;
547 } else {
548 Console::println(string("DAEReader::readLevel(): unknown entity type. Skipping"));
549 delete model;
550 model = nullptr;
551 continue;
552 }
553 Transformations sceneEntityTransformations;
554 sceneEntityTransformations.setTranslation(translation);
555 sceneEntityTransformations.addRotation(rotationOrder->getAxis0(), rotation.getArray()[rotationOrder->getAxis0VectorIndex()]);
556 sceneEntityTransformations.addRotation(rotationOrder->getAxis1(), rotation.getArray()[rotationOrder->getAxis1VectorIndex()]);
557 sceneEntityTransformations.addRotation(rotationOrder->getAxis2(), rotation.getArray()[rotationOrder->getAxis2VectorIndex()]);
558 sceneEntityTransformations.setScale(scale);
559 sceneEntityTransformations.update();
560 auto sceneEntity = new SceneEntity(
561 meshNode.id,
562 meshNode.id,
563 sceneEntityTransformations,
564 prototype
565 );
566 scene->addEntity(sceneEntity);
567 }
568 //
569 progressIdx++;
570 }
571
572 if (progressCallback != nullptr) progressCallback->progress(0.9f);
573
574 try {
575 // export to tscene
577 pathName,
578 Tools::removeFileEnding(fileName) + ".tscene",
579 scene
580 );
581 } catch (Exception& exception) {
582 delete scene;
583 delete sceneModel;
584 throw exception;
585 }
586
587 //
588 delete sceneModel;
589
590 //
591 if (progressCallback != nullptr) progressCallback->progress(1.0f);
592
593 //
594 return scene;
595}
Transformations which contain scale, rotations and translation.
void setTranslation(const Vector3 &translation)
Set translation.
void setScale(const Vector3 &scale)
Set scale.
virtual void update()
Computes transformation matrix.
void setPivot(const Vector3 &pivot)
Set pivot.
void addRotation(const Vector3 &axis, const float angle)
Add rotation.
static void determineMeshNodes(Scene *scene, Node *node, const string &parentName, const Matrix4x4 &parentTransformationsMatrix, vector< PrototypeMeshNode > &meshNodes)
Determine mesh nodes in node hierarchy.
static Scene * read(const string &pathName, const string &fileName, ProgressCallback *progressCallback=nullptr, PrototypeTransformFilter *prototypeTransformFilter=nullptr)
Reads a scene.
Definition: SceneReader.cpp:81
static Scene * readFromModel(const string &pathName, const string &fileName, ProgressCallback *progressCallback=nullptr)
Reads a scene.
static void write(const string &pathName, const string &fileName, Scene *scene)
Writes a scene.
Definition: SceneWriter.cpp:56
const vector< Matrix4x4 > & getTransformationsMatrices() const
Returns transformation matrices.
Definition: Animation.h:41
Color 4 definition.
Definition: Color4.h:20
Representation of a 3d model.
Definition: Model.h:32
Model node.
Definition: Node.h:31
const Matrix4x4 & getTransformationsMatrix() const
Definition: Node.h:121
const string & getId()
Returns id.
Definition: Node.h:85
Animation * getAnimation()
Definition: Node.h:225
const vector< Vector3 > & getVertices() const
Definition: Node.h:151
map< string, Node * > & getSubNodes()
Definition: Node.h:289
Represents rotation orders of a model.
Definition: RotationOrder.h:23
const Vector3 & getAxis0() const
Definition: RotationOrder.h:60
const Vector3 & getAxis2() const
Definition: RotationOrder.h:74
const Vector3 & getAxis1() const
Definition: RotationOrder.h:67
Model up vector.
Definition: UpVector.h:20
bool addProperty(const string &name, const string &value)
Add a property.
Prototype definition.
Definition: Prototype.h:49
void setEmbedded(bool embedded)
Set embedded.
Definition: Prototype.h:140
Scene entity definition.
Definition: SceneEntity.h:24
Scene prototype library definition.
Definition: SceneLibrary.h:27
Scene light definition.
Definition: SceneLight.h:21
Scene definition.
Definition: Scene.h:41
SceneLibrary * getLibrary()
Definition: Scene.h:168
static bool equals(Model *model1, Model *model2)
Compute if model 1 equals model 2.
Standard math functions.
Definition: Math.h:21
4x4 3D Matrix class
Definition: Matrix4x4.h:24
void getAxes(Vector3 &xAxis, Vector3 &yAxis, Vector3 &zAxis) const
Get coordinate system axes.
Definition: Matrix4x4.h:227
Vector3 computeEulerAngles() const
Compute Euler angles (rotation around x, y, z axes)
Definition: Matrix4x4.h:661
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
void getScale(Vector3 &scale) const
Get scale.
Definition: Matrix4x4.h:282
Matrix4x4 clone()
Clones this matrix.
Definition: Matrix4x4.h:624
Matrix4x4 & setAxes(const Vector3 &xAxis, const Vector3 &yAxis, const Vector3 &zAxis)
Set coordinate system axes.
Definition: Matrix4x4.h:240
void getTranslation(Vector3 &translation) const
Get translation.
Definition: Matrix4x4.h:261
Matrix4x4 & scale(float s)
Scales this matrix.
Definition: Matrix4x4.h:418
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
float getY() const
Definition: Vector3.h:119
float getX() const
Definition: Vector3.h:103
float getZ() const
Definition: Vector3.h:136
Vector3 & normalize()
Normalize the vector.
Definition: Vector3.h:288
Vector3 & scale(float scale)
Scale this vector.
Definition: Vector3.h:349
array< float, 3 > & getArray() const
Definition: Vector3.h:171
3D vector 4 class
Definition: Vector4.h:19
File system singleton class.
Definition: FileSystem.h:14
Console class.
Definition: Console.h:26
Float class.
Definition: Float.h:23
Model tools functions class.
Definition: ModelTools.h:38
String tools class.
Definition: StringTools.h:20
std::exception Exception
Exception base class.
Definition: Exception.h:19
virtual void progress(float value)=0
Perform action.