39#include <ext/rapidjson/document.h>
78using rapidjson::Document;
79using rapidjson::Value;
83 return read(pathName, fileName,
"", progressCallback, prototypeTransformFilter);
88 if (progressCallback !=
nullptr) progressCallback->
progress(0.0f);
90 auto jsonContent = FileSystem::getInstance()->getContentAsString(pathName, fileName);
91 if (progressCallback !=
nullptr) progressCallback->
progress(0.165f);
94 jRoot.Parse(jsonContent.c_str());
95 if (progressCallback !=
nullptr) progressCallback->
progress(0.33f);
98 auto scene =
new Scene(fileName,
"");
99 scene->setApplicationRootPathName(Tools::getApplicationRootPathName(pathName));
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];
105 jSceneProperty[
"name"].GetString(),
106 jSceneProperty[
"value"].GetString()
109 if (jRoot.FindMember(
"lights") != jRoot.MemberEnd()) {
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();
118 jLight[
"ar"].GetFloat(),
119 jLight[
"ag"].GetFloat(),
120 jLight[
"ab"].GetFloat(),
121 jLight[
"aa"].GetFloat()
126 jLight[
"dr"].GetFloat(),
127 jLight[
"dg"].GetFloat(),
128 jLight[
"db"].GetFloat(),
129 jLight[
"da"].GetFloat()
134 jLight[
"sr"].GetFloat(),
135 jLight[
"sg"].GetFloat(),
136 jLight[
"sb"].GetFloat(),
137 jLight[
"sa"].GetFloat()
142 jLight[
"px"].GetFloat(),
143 jLight[
"py"].GetFloat(),
144 jLight[
"pz"].GetFloat(),
145 jLight[
"pw"].GetFloat()
148 light->setConstantAttenuation(jLight[
"ca"].GetFloat());
149 light->setLinearAttenuation(jLight[
"la"].GetFloat());
150 light->setQuadraticAttenuation(jLight[
"qa"].GetFloat());
151 light->setSpotDirection(
153 jLight[
"sdx"].GetFloat(),
154 jLight[
"sdy"].GetFloat(),
155 jLight[
"sdz"].GetFloat()
158 light->setSpotExponent(jLight[
"se"].GetFloat());
159 light->setSpotCutOff(jLight[
"sco"].GetFloat());
160 light->setEnabled(jLight[
"e"].GetBool());
165 auto progressStepCurrent = 0;
166 auto jPrototypes = jRoot[
"models"].GetArray();
167 for (
auto i = 0; i < jPrototypes.Size(); i++) {
168 auto& jPrototype = jPrototypes[i];
171 auto embedded = jPrototype.FindMember(
"e") != jPrototype.MemberEnd()?jPrototype[
"e"].GetBool():
true;
172 if (embedded ==
true) {
173 prototype = PrototypeReader::read(
174 jPrototype[
"id"].GetInt(),
176 jPrototype[
"entity"],
177 prototypeTransformFilter
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
192 Console::println(
string() +
"SceneReader::read(): An error occurred: " + exception.what());
196 if (prototype ==
nullptr) {
197 Console::println(
"SceneReader::read(): Invalid prototype = " + to_string(jPrototype[
"id"].GetInt()));
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];
205 jPrototypeProperty[
"name"].GetString(),
206 jPrototypeProperty[
"value"].GetString()
211 if (progressCallback !=
nullptr) progressCallback->
progress(0.33f +
static_cast<float>(progressStepCurrent) /
static_cast<float>(jRoot[
"models"].GetArray().Size()) * 0.33f);
212 progressStepCurrent++;
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()));
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++;
229 transformations.
setPivot(prototype->getPivot());
232 jSceneEntity[
"tx"].GetFloat(),
233 jSceneEntity[
"ty"].GetFloat(),
234 jSceneEntity[
"tz"].GetFloat()
239 jSceneEntity[
"sx"].GetFloat(),
240 jSceneEntity[
"sy"].GetFloat(),
241 jSceneEntity[
"sz"].GetFloat()
245 jSceneEntity[
"rx"].GetFloat(),
246 jSceneEntity[
"ry"].GetFloat(),
247 jSceneEntity[
"rz"].GetFloat()
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()]);
254 objectIdPrefix !=
"" ?
255 objectIdPrefix + jSceneEntity[
"id"].GetString() :
256 (jSceneEntity[
"id"].GetString()),
257 jSceneEntity.FindMember(
"descr") != jSceneEntity.MemberEnd()?jSceneEntity[
"descr"].GetString() :
"",
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()
270 sceneEntity->setReflectionEnvironmentMappingId(jSceneEntity.FindMember(
"r") != jSceneEntity.MemberEnd()?jSceneEntity[
"r"].GetString():
"");
271 scene->addEntity(sceneEntity);
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++;
276 scene->setEntityIdx(jRoot[
"objects_eidx"].GetInt());
277 scene->setFileName((pathName.empty() ==
false?pathName +
"/":
"") + fileName);
281 if (jRoot.FindMember(
"sky") != jRoot.MemberEnd()) {
282 auto& jSky = jRoot[
"sky"];
283 scene->setSkyModelFileName(jSky[
"file"].GetString());
284 scene->setSkyModelScale(
286 jSky[
"sx"].GetFloat(),
287 jSky[
"sy"].GetFloat(),
288 jSky[
"sz"].GetFloat()
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);
310 if (progressCallback !=
nullptr) {
312 delete progressCallback;
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++) {
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) {
339 if (haveName ==
true) {
340 modelName = modelNameTry;
345 if (haveName ==
false) {
348 "SceneReader::doImportFromModel(): Skipping model '" +
350 "' as no name could be created for it."
358 if (animation !=
nullptr) {
360 transformationsMatrix.
set(animationMatrices[0 % animationMatrices.size()]);
367 transformationsMatrix.
multiply(parentTransformationsMatrix);
378 prototypeMeshNode.
id = nodeId;
379 prototypeMeshNode.
name = modelName;
380 prototypeMeshNode.
node = node;
382 meshNodes.push_back(prototypeMeshNode);
387 if (progressCallback !=
nullptr) progressCallback->
progress(0.0f);
389 string modelPathName = pathName +
"/" + fileName +
"-models";
390 if (FileSystem::getInstance()->fileExists(modelPathName)) {
391 FileSystem::getInstance()->removePath(modelPathName,
true);
393 FileSystem::getInstance()->createPath(modelPathName);
395 auto sceneModel = ModelReader::read(pathName, fileName);
397 if (progressCallback !=
nullptr) progressCallback->
progress(0.1f);
399 auto upVector = sceneModel->getUpVector();
400 RotationOrder* rotationOrder = sceneModel->getRotationOrder();
403 auto scene =
new Scene(fileName,
"");
404 scene->setRotationOrder(rotationOrder);
406 auto sceneLibrary = scene->getLibrary();
409 Matrix4x4 sceneModelImportRotationMatrix;
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;
420 for (
auto& meshNode: meshNodes) {
421 auto model =
new Model(
422 meshNode.name +
".tm",
423 fileName +
"-" + meshNode.name,
428 model->setImportTransformationsMatrix(sceneModel->getImportTransformationsMatrix());
429 float importFixScale = 1.0f;
433 nodeTransformationsMatrix.
set(meshNode.transformationsMatrix);
434 nodeTransformationsMatrix.
getAxes(xAxis, yAxis, zAxis);
436 nodeTransformationsMatrix.
getScale(scale);
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)) {
446 nodeTransformationsMatrix.
setAxes(xAxis, yAxis, zAxis);
450 scale = sceneModelImportRotationMatrix.
multiply(scale);
451 rotation = sceneModelImportRotationMatrix.
multiply(rotation);
452 translation = model->getImportTransformationsMatrix().multiply(translation);
454 ModelTools::cloneNode(meshNode.node, model);
455 if (model->getSubNodes().begin() != model->getSubNodes().end()) {
456 model->getSubNodes().begin()->second->setTransformationsMatrix(
Matrix4x4().identity());
458 model->addAnimationSetup(Model::ANIMATIONSETUP_DEFAULT, 0, 0,
true);
459 ModelTools::prepareForIndexedRendering(model);
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;
470 if (height > Math::EPSILON && height < width && height < depth) {
471 importFixScale = 1.0f / height / 5.0f;
473 if (depth > Math::EPSILON) {
474 importFixScale = 1.0f / depth / 5.0f;
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);
483 auto prototypeType = Prototype_Type::MODEL;
484 if (meshNode.node->getVertices().size() == 0) {
485 prototypeType = Prototype_Type::EMPTY;
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)
497 prototype = prototypeCompare;
503 if (prototype ==
nullptr && model !=
nullptr) {
504 auto modelFileName = meshNode.
name +
".tm";
519 Prototype_Type::MODEL,
520 Tools::removeFileEnding(fileName),
521 Tools::removeFileEnding(fileName),
522 modelPathName +
"/" + modelFileName,
523 "resources/engine/models/empty.tm",
528 sceneLibrary->addPrototype(prototype);
531 if (prototypeType == Prototype_Type::EMPTY) {
532 if (emptyPrototype ==
nullptr) {
535 Prototype_Type::EMPTY,
539 "resources/engine/models/empty.tm",
541 ModelReader::read(
"resources/engine/models",
"empty.tm"),
544 sceneLibrary->addPrototype(emptyPrototype);
546 prototype = emptyPrototype;
548 Console::println(
string(
"DAEReader::readLevel(): unknown entity type. Skipping"));
558 sceneEntityTransformations.
setScale(scale);
559 sceneEntityTransformations.
update();
563 sceneEntityTransformations,
566 scene->addEntity(sceneEntity);
572 if (progressCallback !=
nullptr) progressCallback->
progress(0.9f);
578 Tools::removeFileEnding(fileName) +
".tscene",
591 if (progressCallback !=
nullptr) progressCallback->
progress(1.0f);
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.
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.
const vector< Matrix4x4 > & getTransformationsMatrices() const
Returns transformation matrices.
Representation of a 3d model.
const Matrix4x4 & getTransformationsMatrix() const
const string & getId()
Returns id.
Animation * getAnimation()
const vector< Vector3 > & getVertices() const
map< string, Node * > & getSubNodes()
Represents rotation orders of a model.
const Vector3 & getAxis0() const
int32_t getAxis0VectorIndex() const
const Vector3 & getAxis2() const
int32_t getAxis2VectorIndex() const
const Vector3 & getAxis1() const
int32_t getAxis1VectorIndex() const
bool addProperty(const string &name, const string &value)
Add a property.
void setEmbedded(bool embedded)
Set embedded.
Scene prototype library definition.
SceneLibrary * getLibrary()
static bool equals(Model *model1, Model *model2)
Compute if model 1 equals model 2.
void getAxes(Vector3 &xAxis, Vector3 &yAxis, Vector3 &zAxis) const
Get coordinate system axes.
Vector3 computeEulerAngles() const
Compute Euler angles (rotation around x, y, z axes)
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.
void getScale(Vector3 &scale) const
Get scale.
Matrix4x4 clone()
Clones this matrix.
Matrix4x4 & setAxes(const Vector3 &xAxis, const Vector3 &yAxis, const Vector3 &zAxis)
Set coordinate system axes.
void getTranslation(Vector3 &translation) const
Get translation.
Matrix4x4 & scale(float s)
Scales this matrix.
Vector3 multiply(const Vector3 &v) const
Multiplies a vector3 with this matrix into destination vector.
Vector3 & normalize()
Normalize the vector.
Vector3 & scale(float scale)
Scale this vector.
array< float, 3 > & getArray() const
File system singleton class.
std::exception Exception
Exception base class.
virtual void progress(float value)=0
Perform action.
Matrix4x4 transformationsMatrix