TDME2 1.9.121
SceneConnector.cpp
Go to the documentation of this file.
2
3#if defined(_WIN32) && defined(_MSC_VER)
4 #pragma warning(disable:4503)
5#endif
6
7#include <map>
8#include <string>
9#include <vector>
10
11#include <tdme/tdme.h>
12#include <tdme/audio/Audio.h>
13#include <tdme/audio/Sound.h>
58#include <tdme/engine/Engine.h>
59#include <tdme/engine/Entity.h>
63#include <tdme/engine/Light.h>
74#include <tdme/math/Math.h>
75#include <tdme/math/Vector3.h>
76#include <tdme/math/Vector4.h>
84
85using std::map;
86using std::string;
87using std::to_string;
88using std::vector;
89
91
154using tdme::math::Math;
164
165Model* SceneConnector::emptyModel = nullptr;
166float SceneConnector::renderGroupsPartitionWidth = 64.0f;
167float SceneConnector::renderGroupsPartitionHeight = 64.0f;
168float SceneConnector::renderGroupsPartitionDepth = 64.0f;
169int SceneConnector::renderGroupsReduceBy = 1;
170int SceneConnector::renderGroupsLODLevels = 3;
171float SceneConnector::renderGroupsLOD2MinDistance = 25.0;
172float SceneConnector::renderGroupsLOD3MinDistance = 50.0;
173int SceneConnector::renderGroupsLOD2ReduceBy = 4;
174int SceneConnector::renderGroupsLOD3ReduceBy = 16;
175bool SceneConnector::enableEarlyZRejection = false;
176
177void SceneConnector::setLights(Engine* engine, Scene* scene, const Vector3& translation)
178{
179 for (auto i = 0; i < Engine::LIGHTS_MAX; i++) {
180 if (i >= scene->getLightCount()) {
181 engine->getLightAt(i)->setEnabled(false);
182 continue;
183 }
184 engine->getLightAt(i)->setAmbient(Color4(scene->getLightAt(i)->getAmbient()));
185 engine->getLightAt(i)->setDiffuse(Color4(scene->getLightAt(i)->getDiffuse()));
186 engine->getLightAt(i)->setSpecular(Color4(scene->getLightAt(i)->getSpecular()));
187 engine->getLightAt(i)->setSpotDirection(scene->getLightAt(i)->getSpotDirection());
188 engine->getLightAt(i)->setSpotExponent(scene->getLightAt(i)->getSpotExponent());
189 engine->getLightAt(i)->setSpotCutOff(scene->getLightAt(i)->getSpotCutOff());
193 engine->getLightAt(i)->setEnabled(scene->getLightAt(i)->isEnabled());
194 engine->getLightAt(i)->setPosition(
195 Vector4(
196 scene->getLightAt(i)->getPosition().getX() + translation.getX(),
197 scene->getLightAt(i)->getPosition().getY() + translation.getY(),
198 scene->getLightAt(i)->getPosition().getZ() + translation.getZ(),
199 scene->getLightAt(i)->getPosition().getW()
200 )
201 );
202 }
203}
204
205Entity* SceneConnector::createParticleSystem(PrototypeParticleSystem* particleSystem, const string& id, bool enableDynamicShadows)
206{
207 ParticleEmitter* engineEmitter = nullptr;
208 {
209 auto v = particleSystem->getEmitter();
210 if (v == PrototypeParticleSystem_Emitter::NONE) {
211 return nullptr;
212 } else
213 if (v == PrototypeParticleSystem_Emitter::POINT_PARTICLE_EMITTER) {
214 auto emitter = particleSystem->getPointParticleEmitter();
215 engineEmitter = new PointParticleEmitter(emitter->getCount(), emitter->getLifeTime(), emitter->getLifeTimeRnd(), emitter->getMass(), emitter->getMassRnd(), emitter->getPosition(), emitter->getVelocity(), emitter->getVelocityRnd(), emitter->getColorStart(), emitter->getColorEnd());
216 } else
217 if (v == PrototypeParticleSystem_Emitter::BOUNDINGBOX_PARTICLE_EMITTER) {
218 auto emitter = particleSystem->getBoundingBoxParticleEmitters();
219 engineEmitter = new BoundingBoxParticleEmitter(emitter->getCount(), emitter->getLifeTime(), emitter->getLifeTimeRnd(), emitter->getMass(), emitter->getMassRnd(), new OrientedBoundingBox(emitter->getObbCenter(), emitter->getObbAxis0(), emitter->getObbAxis1(), emitter->getObbAxis2(), emitter->getObbHalfextension()), emitter->getVelocity(), emitter->getVelocityRnd(), emitter->getColorStart(), emitter->getColorEnd());
220 } else
221 if (v == PrototypeParticleSystem_Emitter::CIRCLE_PARTICLE_EMITTER) {
222 auto emitter = particleSystem->getCircleParticleEmitter();
223 engineEmitter = new CircleParticleEmitter(emitter->getCount(), emitter->getLifeTime(), emitter->getLifeTimeRnd(), emitter->getAxis0(), emitter->getAxis1(), emitter->getCenter(), emitter->getRadius(), emitter->getMass(), emitter->getMassRnd(), emitter->getVelocity(), emitter->getVelocityRnd(), emitter->getColorStart(), emitter->getColorEnd());
224 } else
225 if (v == PrototypeParticleSystem_Emitter::CIRCLE_PARTICLE_EMITTER_PLANE_VELOCITY) {
226 auto emitter = particleSystem->getCircleParticleEmitterPlaneVelocity();
227 engineEmitter = new CircleParticleEmitterPlaneVelocity(emitter->getCount(), emitter->getLifeTime(), emitter->getLifeTimeRnd(), emitter->getAxis0(), emitter->getAxis1(), emitter->getCenter(), emitter->getRadius(), emitter->getMass(), emitter->getMassRnd(), emitter->getVelocity(), emitter->getVelocityRnd(), emitter->getColorStart(), emitter->getColorEnd());
228 } else
229 if (v == PrototypeParticleSystem_Emitter::SPHERE_PARTICLE_EMITTER) {
230 auto emitter = particleSystem->getSphereParticleEmitter();
231 engineEmitter = new SphereParticleEmitter(emitter->getCount(), emitter->getLifeTime(), emitter->getLifeTimeRnd(), emitter->getMass(), emitter->getMassRnd(), new Sphere(emitter->getCenter(), emitter->getRadius()), emitter->getVelocity(), emitter->getVelocityRnd(), emitter->getColorStart(), emitter->getColorEnd());
232 } else {
233 Console::println(
234 string(
235 "SceneConnector::createParticleSystem(): unknown particle system emitter '" +
236 particleSystem->getEmitter()->getName() +
237 "'"
238 )
239 );
240 return nullptr;
241 }
242 }
243
244 {
245 {
246 auto v = particleSystem->getType();
247 if (v == PrototypeParticleSystem_Type::NONE) {
248 return nullptr;
249 } else
250 if (v == PrototypeParticleSystem_Type::OBJECT_PARTICLE_SYSTEM) {
251 auto objectParticleSystem = particleSystem->getObjectParticleSystem();
252 if (objectParticleSystem->getModel() == nullptr) return nullptr;
253
254 return new ObjectParticleSystem(
255 id,
256 objectParticleSystem->getModel(),
257 objectParticleSystem->getScale(),
258 objectParticleSystem->isAutoEmit(),
259 enableDynamicShadows,
260 enableDynamicShadows,
261 objectParticleSystem->getMaxCount(),
262 engineEmitter
263 );
264 } else
265 if (v == PrototypeParticleSystem_Type::POINT_PARTICLE_SYSTEM) {
266 auto pointParticleSystem = particleSystem->getPointParticleSystem();
267 return new PointsParticleSystem(
268 id,
269 engineEmitter,
270 pointParticleSystem->getMaxPoints(),
271 pointParticleSystem->getPointSize(),
272 pointParticleSystem->isAutoEmit(),
273 pointParticleSystem->getTexture(),
274 pointParticleSystem->getTextureHorizontalSprites(),
275 pointParticleSystem->getTextureVerticalSprites(),
276 pointParticleSystem->getTextureSpritesFPS()
277 );
278 } else
279 if (v == PrototypeParticleSystem_Type::FOG_PARTICLE_SYSTEM) {
280 auto fogParticleSystem = particleSystem->getFogParticleSystem();
281 return new FogParticleSystem(
282 id,
283 engineEmitter,
284 fogParticleSystem->getMaxPoints(),
285 fogParticleSystem->getPointSize(),
286 fogParticleSystem->getTexture(),
287 fogParticleSystem->getTextureHorizontalSprites(),
288 fogParticleSystem->getTextureVerticalSprites(),
289 fogParticleSystem->getTextureSpritesFPS()
290 );
291 } else {
292 Console::println(
293 string(
294 "SceneConnector::createParticleSystem(): unknown particle system type '" +
295 particleSystem->getType()->getName() +
296 "'"
297 )
298 );
299 return nullptr;
300 }
301 }
302 }
303
304}
305
306Entity* SceneConnector::createEmpty(const string& id, const Transformations& transformations) {
307 if (emptyModel == nullptr) {
308 emptyModel = ModelReader::read("resources/engine/models", "empty.tm");
309 }
310 auto entity = new Object3D(
311 id,
313 );
314 entity->fromTransformations(transformations);
315 return entity;
316}
317
318Entity* SceneConnector::createEntity(Prototype* prototype, const string& id, const Transformations& transformations, int instances, Entity* parentEntity) {
319 Entity* entity = nullptr;
320
321 // objects
322 if (prototype->getModel() != nullptr) {
323 auto imposterLOD = prototype->getImposterLOD();
324 auto lodLevel2 = prototype->getLODLevel2();
325 auto lodLevel3 = prototype->getLODLevel3();
326 // with LOD
327 if (imposterLOD != nullptr) {
328 entity = new LODObject3DImposter(
329 id,
330 prototype->getModel(),
331 imposterLOD->getModels(),
332 imposterLOD->getMinDistance()
333 );
334 entity->setParentEntity(parentEntity);
335 auto imposterLodObject = dynamic_cast<LODObject3DImposter*>(entity);
336 imposterLodObject->setEffectColorAddLOD2(imposterLOD->getColorAdd());
337 imposterLodObject->setEffectColorMulLOD2(imposterLOD->getColorMul());
338 if (prototype->getShader() == "water" || prototype->getShader() == "pbr-water") imposterLodObject->setRenderPass(Entity::RENDERPASS_WATER);
339 imposterLodObject->setShader(prototype->getShader());
340 imposterLodObject->setDistanceShader(prototype->getDistanceShader());
341 imposterLodObject->setDistanceShaderDistance(prototype->getDistanceShaderDistance());
342 auto shaderParametersDefault = Engine::getShaderParameterDefaults(prototype->getShader());
343 auto distanceShaderParametersDefault = Engine::getShaderParameterDefaults(prototype->getDistanceShader());
344 for (auto& parameterIt: shaderParametersDefault) {
345 auto& parameterName = parameterIt.first;
346 auto parameterValue = prototype->getShaderParameters().getShaderParameter(parameterName);
347 imposterLodObject->setShaderParameter(parameterName, parameterValue);
348 }
349 for (auto& parameterIt: distanceShaderParametersDefault) {
350 auto& parameterName = parameterIt.first;
351 auto parameterValue = prototype->getDistanceShaderParameters().getShaderParameter(parameterName);
352 imposterLodObject->setDistanceShaderParameter(parameterName, parameterValue);
353 }
354 } else
355 if (lodLevel2 != nullptr) {
356 entity = new LODObject3D(
357 id,
358 prototype->getModel(),
359 lodLevel2->getType(),
360 lodLevel2->getMinDistance(),
361 lodLevel2->getModel(),
362 lodLevel3 != nullptr?lodLevel3->getType():LODObject3D::LODLEVELTYPE_NONE,
363 lodLevel3 != nullptr?lodLevel3->getMinDistance():0.0f,
364 lodLevel3 != nullptr?lodLevel3->getModel():nullptr
365 );
366 entity->setParentEntity(parentEntity);
367 auto lodObject = dynamic_cast<LODObject3D*>(entity);
368 lodObject->setEffectColorAddLOD2(lodLevel2->getColorAdd());
369 lodObject->setEffectColorMulLOD2(lodLevel2->getColorMul());
370 if (lodLevel3 != nullptr) {
371 lodObject->setEffectColorAddLOD3(lodLevel3->getColorAdd());
372 lodObject->setEffectColorMulLOD3(lodLevel3->getColorMul());
373 }
374 if (prototype->getShader() == "water" || prototype->getShader() == "pbr-water") lodObject->setRenderPass(Entity::RENDERPASS_WATER);
375 lodObject->setShader(prototype->getShader());
376 lodObject->setDistanceShader(prototype->getDistanceShader());
377 lodObject->setDistanceShaderDistance(prototype->getDistanceShaderDistance());
378 auto shaderParametersDefault = Engine::getShaderParameterDefaults(prototype->getShader());
379 auto distanceShaderParametersDefault = Engine::getShaderParameterDefaults(prototype->getDistanceShader());
380 for (auto& parameterIt: shaderParametersDefault) {
381 auto& parameterName = parameterIt.first;
382 auto parameterValue = prototype->getShaderParameters().getShaderParameter(parameterName);
383 lodObject->setShaderParameter(parameterName, parameterValue);
384 }
385 for (auto& parameterIt: distanceShaderParametersDefault) {
386 auto& parameterName = parameterIt.first;
387 auto parameterValue = prototype->getDistanceShaderParameters().getShaderParameter(parameterName);
388 lodObject->setDistanceShaderParameter(parameterName, parameterValue);
389 }
390 } else {
391 // single
392 entity = new Object3D(
393 id,
394 prototype->getModel(),
395 instances
396 );
397 entity->setParentEntity(parentEntity);
398 auto object = dynamic_cast<Object3D*>(entity);
399 if (prototype->getShader() == "water" || prototype->getShader() == "pbr-water") object->setRenderPass(Entity::RENDERPASS_WATER);
400 object->setShader(prototype->getShader());
401 object->setDistanceShader(prototype->getDistanceShader());
402 object->setDistanceShaderDistance(prototype->getDistanceShaderDistance());
403 auto shaderParametersDefault = Engine::getShaderParameterDefaults(prototype->getShader());
404 auto distanceShaderParametersDefault = Engine::getShaderParameterDefaults(prototype->getDistanceShader());
405 for (auto& parameterIt: shaderParametersDefault) {
406 auto& parameterName = parameterIt.first;
407 auto parameterValue = prototype->getShaderParameters().getShaderParameter(parameterName);
408 object->setShaderParameter(parameterName, parameterValue);
409 }
410 for (auto& parameterIt: distanceShaderParametersDefault) {
411 auto& parameterName = parameterIt.first;
412 auto parameterValue = prototype->getDistanceShaderParameters().getShaderParameter(parameterName);
413 object->setDistanceShaderParameter(parameterName, parameterValue);
414 }
415 if (enableEarlyZRejection == true && prototype->isTerrainMesh() == true) {
416 object->setEnableEarlyZRejection(true);
417 }
418 }
419 } else
420 // particle system
421 if (prototype->getType() == Prototype_Type::PARTICLESYSTEM) {
422 vector<ParticleSystemEntity*> particleSystems;
423 for (auto i = 0; i < prototype->getParticleSystemsCount(); i++) {
424 auto particleSystem = createParticleSystem(
425 prototype->getParticleSystemAt(i),
426 id + (i == 0?"":"." + to_string(i)),
427 true
428 );
429 if (particleSystem != nullptr) {
430 particleSystems.push_back(dynamic_cast<ParticleSystemEntity*>(particleSystem));
431 }
432 }
433 if (particleSystems.size() == 1) {
434 entity = dynamic_cast<Entity*>(particleSystems[0]);
435 entity->setParentEntity(parentEntity);
436 } else
437 if (particleSystems.size() > 1) {
438 entity = new ParticleSystemGroup(
439 id,
440 true,
441 true,
442 true,
443 particleSystems
444 );
445 entity->setParentEntity(parentEntity);
446 }
447 } else
448 // trigger/environment mapping
449 if (prototype->getType() == Prototype_Type::TRIGGER ||
450 prototype->getType() == Prototype_Type::ENVIRONMENTMAPPING) {
451 // bounding volumes
452 auto entityBoundingVolumesHierarchy = new EntityHierarchy(id);
453 for (auto i = 0; i < prototype->getBoundingVolumeCount(); i++) {
454 auto entityBoundingVolume = prototype->getBoundingVolume(i);
455 if (entityBoundingVolume->getModel() != nullptr) {
456 auto bvObject = new Object3D("tdme.prototype.bv." + to_string(i), entityBoundingVolume->getModel());
457 bvObject->setRenderPass(Entity::RENDERPASS_POST_POSTPROCESSING);
458 entityBoundingVolumesHierarchy->addEntity(bvObject);
459 }
460 }
461 if (prototype->getType() == Prototype_Type::ENVIRONMENTMAPPING &&
462 prototype->getBoundingVolumeCount() == 1 &&
463 dynamic_cast<OrientedBoundingBox*>(prototype->getBoundingVolume(0)->getBoundingVolume()) != nullptr) {
464 BoundingBox aabb(dynamic_cast<OrientedBoundingBox*>(prototype->getBoundingVolume(0)->getBoundingVolume()));
465 auto environmentMapping = new EnvironmentMapping("environmentmapping", Engine::getEnvironmentMappingWidth(), Engine::getEnvironmentMappingHeight(), aabb);
466 environmentMapping->setRenderPassMask(prototype->getEnvironmentMapRenderPassMask());
467 environmentMapping->setTimeRenderUpdateFrequency(prototype->getEnvironmentMapTimeRenderUpdateFrequency());
468 entityBoundingVolumesHierarchy->addEntity(environmentMapping);
469 }
470 entityBoundingVolumesHierarchy->update();
471 if (entityBoundingVolumesHierarchy->getEntities().size() == 0) {
472 entityBoundingVolumesHierarchy->dispose();
473 delete entityBoundingVolumesHierarchy;
474 } else {
475 entity = entityBoundingVolumesHierarchy;
476 entity->setParentEntity(parentEntity);
477 }
478 }
479
480 //
481 if (entity != nullptr) {
482 if (prototype->isTerrainMesh() == true) {
484 }
485 entity->setContributesShadows(prototype->isContributesShadows());
486 entity->setReceivesShadows(prototype->isReceivesShadows());
487 entity->fromTransformations(transformations);
488 }
489
490 // done
491 return entity;
492}
493
494Entity* SceneConnector::createEntity(SceneEntity* sceneEntity, const Vector3& translation, int instances, Entity* parentEntity) {
495 Transformations transformations;
496 transformations.fromTransformations(sceneEntity->getTransformations());
497 if (translation.equals(Vector3()) == false) {
498 transformations.setTranslation(transformations.getTranslation().clone().add(translation));
499 transformations.update();
500 }
501 return createEntity(sceneEntity->getPrototype(), sceneEntity->getId(), transformations, instances, parentEntity);
502}
503
504void SceneConnector::addScene(Engine* engine, Scene* scene, bool addEmpties, bool addTrigger, bool addEnvironmentMapping, bool pickable, bool enable, const Vector3& translation, ProgressCallback* progressCallback)
505{
506 if (progressCallback != nullptr) progressCallback->progress(0.0f);
507 // TODO: progress callbacks for terrain
508
509 // scene library
510 auto sceneLibrary = scene->getLibrary();
511
512 // terrain
513 {
514 auto prototype = sceneLibrary->getTerrainPrototype();
515 if (prototype != nullptr) {
516 //
517 auto terrain = prototype->getTerrain();
518 auto width = terrain->getWidth();
519 auto depth = terrain->getDepth();
520 // terrain
521 BoundingBox terrainBoundingBox;
522 vector<Model*> terrainModels;
523 Terrain2::createTerrainModels(width, depth, 0.0f, terrain->getHeightVector(), terrainBoundingBox, terrainModels/*, true*/); // TODO: finish LOD
524 if (terrainModels.empty() == false) {
525 auto idx = 0;
526 for (auto terrainModel: terrainModels) {
527 auto terrainObject3D = new Object3D("tdme.terrain." + to_string(idx++), terrainModel);
528 terrainObject3D->setRenderPass(Entity::RENDERPASS_TERRAIN);
529 terrainObject3D->setShader("terrain");
530 terrainObject3D->setDistanceShader("terrain");
531 terrainObject3D->setContributesShadows(true);
532 terrainObject3D->setReceivesShadows(true);
533 terrainObject3D->setPickable(pickable);
534 terrainObject3D->setEnabled(enable);
535 terrainObject3D->setTranslation(translation);
536 terrainObject3D->update();
537 engine->addEntity(terrainObject3D);
538 }
539 }
540 // water
541 {
542 auto idx = 0;
543 auto waterPositionMapsIndices = terrain->getWaterPositionMapsIndices();
544 for (auto waterPositionMapIdx: waterPositionMapsIndices) {
545 vector<Model*> waterModels;
546 Terrain2::createWaterModels(
547 terrainBoundingBox,
548 prototype->getTerrain()->getWaterPositionMap(waterPositionMapIdx),
549 prototype->getTerrain()->getWaterPositionMapHeight(waterPositionMapIdx),
550 waterPositionMapIdx,
551 waterModels
552 );
553 for (auto waterModel: waterModels) {
554 auto waterObject3D = new Object3D("tdme.water." + to_string(idx++), waterModel);
555 waterObject3D->setRenderPass(Entity::RENDERPASS_WATER);
556 waterObject3D->setShader("water");
557 waterObject3D->setDistanceShader("water");
558 waterObject3D->setContributesShadows(false);
559 waterObject3D->setReceivesShadows(false);
560 waterObject3D->setReflectionEnvironmentMappingId("sky_environment_mapping");
561 waterObject3D->setReflectionEnvironmentMappingPosition(
562 Terrain2::computeWaterReflectionEnvironmentMappingPosition(
563 terrain->getWaterPositionMap(waterPositionMapIdx),
564 terrain->getWaterPositionMapHeight(waterPositionMapIdx)
565 )
566 );
567 waterObject3D->setPickable(pickable);
568 waterObject3D->setEnabled(enable);
569 waterObject3D->setTranslation(translation);
570 waterObject3D->update();
571 engine->addEntity(waterObject3D);
572 }
573 }
574 }
575 // foliage
576 {
577 //
578 auto& foliageMaps = terrain->getFoliageMaps();
579
580 //
581 auto foliageRenderGroupIdx = 0;
582 auto partitionIdx = 0;
583 map<int, int> prototypeEntityIdx;
584 for (auto& foliageMapPartition: foliageMaps) {
585 auto partitionPrototypeInstanceCount = 0;
586 for (auto& foliageMapPartitionIt: foliageMapPartition) {
587 partitionPrototypeInstanceCount+= foliageMapPartitionIt.second.size();
588 }
589 if (partitionPrototypeInstanceCount > 0) {
590 unordered_map<string, Object3DRenderGroup*> object3DRenderGroupByShaderParameters;
591 for (auto& foliageMapPartitionIt: foliageMapPartition) {
592 auto prototypeIdx = foliageMapPartitionIt.first;
593 auto& transformationsVector = foliageMapPartitionIt.second;
594 if (transformationsVector.empty() == true) continue;
595 auto foliagePrototype = prototype->getTerrain()->getFoliagePrototype(prototypeIdx);
596 if (foliagePrototype->isRenderGroups() == false) {
597 for (auto& transformations: transformationsVector) {
598 auto entity = createEntity(foliagePrototype, "tdme.foliage." + to_string(prototypeIdx) + "." + to_string(prototypeEntityIdx[prototypeIdx]++), transformations);
599 if (entity == nullptr) continue;
600 entity->setTranslation(entity->getTranslation().clone().add(translation));
601 entity->setPickable(pickable);
602 entity->setEnabled(enable);
603 entity->update();
604 engine->addEntity(entity);
605 }
606 } else {
607 Object3DRenderGroup* foliagePartitionObject3DRenderGroup = nullptr;
608 auto contributesShadows = foliagePrototype->isContributesShadows();
609 auto receivesShadows = foliagePrototype->isReceivesShadows();
610 auto hash = foliagePrototype->getShaderParameters().getShaderParametersHash() + "|" + foliagePrototype->getDistanceShaderParameters().getShaderParametersHash() + "|" + to_string(contributesShadows) + "|" + to_string(receivesShadows);
611 auto foliagePartitionObject3DRenderGroupIt = object3DRenderGroupByShaderParameters.find(hash);
612 if (foliagePartitionObject3DRenderGroupIt != object3DRenderGroupByShaderParameters.end()) {
613 foliagePartitionObject3DRenderGroup = foliagePartitionObject3DRenderGroupIt->second;
614 }
615 if (foliagePartitionObject3DRenderGroup == nullptr) {
616 foliagePartitionObject3DRenderGroup =
618 "tdme.fo3rg." + to_string(foliageRenderGroupIdx++),
624 );
625 foliagePartitionObject3DRenderGroup->setContributesShadows(contributesShadows);
626 foliagePartitionObject3DRenderGroup->setReceivesShadows(receivesShadows);
627 foliagePartitionObject3DRenderGroup->setShader(foliagePrototype->getShader());
628 foliagePartitionObject3DRenderGroup->setDistanceShader(foliagePrototype->getDistanceShader());
629 foliagePartitionObject3DRenderGroup->setDistanceShaderDistance(foliagePrototype->getDistanceShaderDistance());
630 auto shaderParametersDefault = Engine::getShaderParameterDefaults(foliagePrototype->getShader());
631 auto distanceShaderParametersDefault = Engine::getShaderParameterDefaults(foliagePrototype->getDistanceShader());
632 for (auto& parameterIt: shaderParametersDefault) {
633 auto& parameterName = parameterIt.first;
634 auto parameterValue = foliagePrototype->getShaderParameters().getShaderParameter(parameterName);
635 foliagePartitionObject3DRenderGroup->setShaderParameter(parameterName, parameterValue);
636 }
637 for (auto& parameterIt: distanceShaderParametersDefault) {
638 auto& parameterName = parameterIt.first;
639 auto parameterValue = foliagePrototype->getDistanceShaderParameters().getShaderParameter(parameterName);
640 foliagePartitionObject3DRenderGroup->setDistanceShaderParameter(parameterName, parameterValue);
641 }
642 foliagePartitionObject3DRenderGroup->setPickable(false);
643 foliagePartitionObject3DRenderGroup->setEnabled(enable);
644 object3DRenderGroupByShaderParameters[hash] = foliagePartitionObject3DRenderGroup;
645 }
646
647 //
648 auto objectIdx = -1;
649 for (auto& transformations: transformationsVector) {
650 objectIdx++;
651 if (objectIdx % renderGroupsReduceBy != 0) continue;
652 foliagePartitionObject3DRenderGroup->addObject(foliagePrototype->getModel(), transformations);
653 }
654 }
655 }
656 for (auto& object3DRenderGroupByShaderParametersIt: object3DRenderGroupByShaderParameters) {
657 auto foliagePartitionObject3DRenderGroup = object3DRenderGroupByShaderParametersIt.second;
658 foliagePartitionObject3DRenderGroup->updateRenderGroup();
659 foliagePartitionObject3DRenderGroup->setTranslation(translation);
660 foliagePartitionObject3DRenderGroup->update();
661 engine->addEntity(foliagePartitionObject3DRenderGroup);
662 }
663 }
664 partitionIdx++;
665 }
666 }
667 }
668 }
669
670 // scene entities
671 map<string, map<string, map<Model*, vector<Transformations*>>>> renderGroupEntitiesByShaderPartitionModel;
672 map<Model*, Prototype*> renderGroupSceneEditorEntities;
673 auto progressStepCurrent = 0;
674 for (auto i = 0; i < scene->getEntityCount(); i++) {
675 auto sceneEntity = scene->getEntityAt(i);
676
677 if (progressCallback != nullptr && progressStepCurrent % 1000 == 0) progressCallback->progress(0.0f + static_cast<float>(progressStepCurrent) / static_cast<float>(scene->getEntityCount()) * 0.5f);
678 progressStepCurrent++;
679
680 if (addEmpties == false && sceneEntity->getPrototype()->getType() == Prototype_Type::EMPTY) continue;
681 if (addTrigger == false && sceneEntity->getPrototype()->getType() == Prototype_Type::TRIGGER) continue;
682
683 if (sceneEntity->getPrototype()->isRenderGroups() == true) {
684 auto minX = sceneEntity->getTransformations().getTranslation().getX();
685 auto minY = sceneEntity->getTransformations().getTranslation().getY();
686 auto minZ = sceneEntity->getTransformations().getTranslation().getZ();
687 auto partitionX = (int)(minX / renderGroupsPartitionWidth);
688 auto partitionY = (int)(minY / renderGroupsPartitionHeight);
689 auto partitionZ = (int)(minZ / renderGroupsPartitionDepth);
690 renderGroupSceneEditorEntities[sceneEntity->getPrototype()->getModel()] = sceneEntity->getPrototype();
691 renderGroupEntitiesByShaderPartitionModel[sceneEntity->getPrototype()->getShader() + "." + sceneEntity->getPrototype()->getDistanceShader() + "." + to_string(static_cast<int>(sceneEntity->getPrototype()->getDistanceShaderDistance() / 10.0f))][to_string(partitionX) + "," + to_string(partitionY) + "," + to_string(partitionZ)][sceneEntity->getPrototype()->getModel()].push_back(&sceneEntity->getTransformations());
692 } else {
693 Entity* entity = createEntity(sceneEntity);
694 if (entity == nullptr) continue;
695
696 entity->setTranslation(entity->getTranslation().clone().add(translation));
697 entity->setPickable(pickable);
698 entity->setContributesShadows(sceneEntity->getPrototype()->isContributesShadows());
699 entity->setReceivesShadows(sceneEntity->getPrototype()->isReceivesShadows());
700 if (sceneEntity->getPrototype()->getType() == Prototype_Type::EMPTY) {
701 entity->setScale(Vector3(Math::sign(entity->getScale().getX()), Math::sign(entity->getScale().getY()), Math::sign(entity->getScale().getZ())));
702 }
703 if (sceneEntity->getPrototype()->getType()->hasNonEditScaleDownMode() == true) {
704 entity->setScale(
705 sceneEntity->getPrototype()->getType()->getNonEditScaleDownModeDimension().
706 clone().
707 scale(
708 Vector3(
709 1.0f / (sceneEntity->getTransformations().getScale().getX() * entity->getBoundingBox()->getDimensions().getX()),
710 1.0f / (sceneEntity->getTransformations().getScale().getY() * entity->getBoundingBox()->getDimensions().getY()),
711 1.0f / (sceneEntity->getTransformations().getScale().getZ() * entity->getBoundingBox()->getDimensions().getZ())
712 )
713 )
714 );
715 }
716 entity->update();
717 entity->setEnabled(enable);
718
719 auto object = dynamic_cast<Object3D*>(entity);
720 if (object != nullptr) object->setReflectionEnvironmentMappingId(sceneEntity->getReflectionEnvironmentMappingId());
721
722 engine->addEntity(entity);
723 }
724 }
725
726 // do render groups
727 {
728 auto idx = 0;
729 progressStepCurrent = 0;
730 auto progressStepMax = 0;
731 if (progressCallback != nullptr) {
732 for (auto& itShader: renderGroupEntitiesByShaderPartitionModel) {
733 for (auto& itPartition: itShader.second) {
734 for (auto& itModel: itPartition.second) {
735 progressStepMax++;
736 }
737 }
738 }
739 }
740 for (auto& itShader: renderGroupEntitiesByShaderPartitionModel) {
741 Console::println("SceneConnector::addLevel(): adding render group: " + itShader.first);
742 for (auto& itPartition: itShader.second) {
743 map<string, Object3DRenderGroup*> object3DRenderGroupsByShaderParameters;
744 for (auto& itModel: itPartition.second) {
745 if (progressCallback != nullptr) {
746 progressCallback->progress(0.5f + static_cast<float>(progressStepCurrent) / static_cast<float>(progressStepMax) * 0.5f);
747 }
748 progressStepCurrent++;
749 auto prototype = renderGroupSceneEditorEntities[itModel.first];
750 auto contributesShadows = prototype->isContributesShadows();
751 auto receivesShadows = prototype->isReceivesShadows();
752 auto hash = prototype->getShaderParameters().getShaderParametersHash() + "|" + prototype->getDistanceShaderParameters().getShaderParametersHash() + "|" + to_string(contributesShadows) + "|" + to_string(receivesShadows);
753 if (object3DRenderGroupsByShaderParameters.find(hash) == object3DRenderGroupsByShaderParameters.end()) {
754 auto object3DRenderNode =
756 "tdme.o3rg." + to_string(idx++),
762 );
763 object3DRenderNode->setContributesShadows(contributesShadows);
764 object3DRenderNode->setReceivesShadows(receivesShadows);
765 object3DRenderNode->setShader(prototype->getShader());
766 object3DRenderNode->setDistanceShader(prototype->getDistanceShader());
767 object3DRenderNode->setDistanceShaderDistance(prototype->getDistanceShaderDistance());
768 auto shaderParametersDefault = Engine::getShaderParameterDefaults(prototype->getShader());
769 auto distanceShaderParametersDefault = Engine::getShaderParameterDefaults(prototype->getDistanceShader());
770 for (auto& parameterIt: shaderParametersDefault) {
771 auto& parameterName = parameterIt.first;
772 auto parameterValue = prototype->getShaderParameters().getShaderParameter(parameterName);
773 object3DRenderNode->setShaderParameter(parameterName, parameterValue);
774 }
775 for (auto& parameterIt: distanceShaderParametersDefault) {
776 auto& parameterName = parameterIt.first;
777 auto parameterValue = prototype->getDistanceShaderParameters().getShaderParameter(parameterName);
778 object3DRenderNode->setDistanceShaderParameter(parameterName, parameterValue);
779 }
780 object3DRenderGroupsByShaderParameters[hash] = object3DRenderNode;
781 }
782 auto object3DRenderNode = object3DRenderGroupsByShaderParameters[hash];
783 auto objectIdx = -1;
784 for (auto transformation: itModel.second) {
785 objectIdx++;
786 if (objectIdx % renderGroupsReduceBy != 0) continue;
787 object3DRenderNode->addObject(prototype->getModel(), *transformation);
788 }
789 }
790 for (auto& object3DRenderGroupsByShaderParametersIt: object3DRenderGroupsByShaderParameters) {
791 auto object3DRenderNode = object3DRenderGroupsByShaderParametersIt.second;
792 object3DRenderNode->updateRenderGroup();
793 engine->addEntity(object3DRenderNode);
794 }
795 }
796 }
797 }
798
799 //
800 if (progressCallback != nullptr) {
801 progressCallback->progress(1.0f);
802 delete progressCallback;
803 }
804}
805
806Body* SceneConnector::createBody(World* world, Prototype* prototype, const string& id, const Transformations& transformations, uint16_t collisionTypeId, int index, PrototypePhysics_BodyType* overrideType) {
807 if (prototype->getType() == Prototype_Type::EMPTY) return nullptr;
808
809 auto physicsType = overrideType != nullptr?overrideType:prototype->getPhysics()->getType();
810 if (prototype->getType() == Prototype_Type::TRIGGER) {
811 vector<BoundingVolume*> boundingVolumes;
812 for (auto j = 0; j < prototype->getBoundingVolumeCount(); j++) {
813 auto entityBv = prototype->getBoundingVolume(j);
814 if (index == -1 || index == j) boundingVolumes.push_back(entityBv->getBoundingVolume());
815 }
816 if (boundingVolumes.size() == 0) return nullptr;
817 return world->addCollisionBody(
818 id,
819 true,
820 collisionTypeId == 0?RIGIDBODY_TYPEID_TRIGGER:collisionTypeId,
821 transformations,
822 boundingVolumes
823 );
824 } else
825 if (prototype->getType() == Prototype_Type::MODEL &&
826 prototype->isTerrainMesh() == true) {
827 Object3DModel terrainModel(prototype->getModel());
828 auto terrainMesh = new TerrainMesh(&terrainModel, transformations);
829 if (physicsType == PrototypePhysics_BodyType::COLLISION_BODY) {
830 return world->addCollisionBody(
831 id,
832 true,
833 collisionTypeId == 0?RIGIDBODY_TYPEID_COLLISION:collisionTypeId,
835 {terrainMesh}
836 );
837 } else
838 if (physicsType == PrototypePhysics_BodyType::STATIC_RIGIDBODY) {
839 return world->addStaticRigidBody(
840 id,
841 true,
842 collisionTypeId == 0?RIGIDBODY_TYPEID_STATIC:collisionTypeId,
844 prototype->getPhysics()->getFriction(),
845 {terrainMesh}
846 );
847 } else
848 if (physicsType == PrototypePhysics_BodyType::DYNAMIC_RIGIDBODY) {
849 return world->addRigidBody(
850 id,
851 true,
852 collisionTypeId == 0?RIGIDBODY_TYPEID_DYNAMIC:collisionTypeId,
854 prototype->getPhysics()->getRestitution(),
855 prototype->getPhysics()->getFriction(),
856 prototype->getPhysics()->getMass(),
857 prototype->getPhysics()->getInertiaTensor(),
858 {terrainMesh}
859 );
860 }
861 } else {
862 vector<BoundingVolume*> boundingVolumes;
863 for (auto j = 0; j < prototype->getBoundingVolumeCount(); j++) {
864 auto entityBv = prototype->getBoundingVolume(j);
865 if (index == -1 || index == j) boundingVolumes.push_back(entityBv->getBoundingVolume());
866 }
867 if (boundingVolumes.size() == 0) return nullptr;
868 if (physicsType == PrototypePhysics_BodyType::COLLISION_BODY) {
869 return world->addCollisionBody(
870 id,
871 true,
872 collisionTypeId == 0?RIGIDBODY_TYPEID_COLLISION:collisionTypeId,
873 transformations,
874 boundingVolumes
875 );
876 } else
877 if (physicsType == PrototypePhysics_BodyType::STATIC_RIGIDBODY) {
878 return world->addStaticRigidBody(
879 id,
880 true,
881 collisionTypeId == 0?RIGIDBODY_TYPEID_STATIC:collisionTypeId,
882 transformations,
883 prototype->getPhysics()->getFriction(),
884 boundingVolumes
885 );
886 } else
887 if (physicsType == PrototypePhysics_BodyType::DYNAMIC_RIGIDBODY) {
888 return world->addRigidBody(
889 id,
890 true,
891 collisionTypeId == 0?RIGIDBODY_TYPEID_DYNAMIC:collisionTypeId,
892 transformations,
893 prototype->getPhysics()->getRestitution(),
894 prototype->getPhysics()->getFriction(),
895 prototype->getPhysics()->getMass(),
896 prototype->getPhysics()->getInertiaTensor(),
897 boundingVolumes
898 );
899 }
900 }
901 return nullptr;
902}
903
904Body* SceneConnector::createBody(World* world, SceneEntity* sceneEntity, const Vector3& translation, uint16_t collisionTypeId, int index, PrototypePhysics_BodyType* overrideType) {
905 Transformations transformations;
906 transformations.fromTransformations(sceneEntity->getTransformations());
907 if (translation.equals(Vector3()) == false) {
908 transformations.setTranslation(transformations.getTranslation().clone().add(translation));
909 transformations.update();
910 }
911 return createBody(world, sceneEntity->getPrototype(), sceneEntity->getId(), transformations, collisionTypeId, index, overrideType);
912}
913
914void SceneConnector::addScene(World* world, Scene* scene, bool enable, const Vector3& translation, ProgressCallback* progressCallback)
915{
916 if (progressCallback != nullptr) progressCallback->progress(0.0f);
917 auto progressStepCurrent = 0;
918
919 // scene library
920 auto sceneLibrary = scene->getLibrary();
921
922 // terrain + foliage
923 {
924 auto prototype = sceneLibrary->getTerrainPrototype();
925 if (prototype != nullptr) {
926 //
927 auto terrain = prototype->getTerrain();
928 auto width = terrain->getWidth();
929 auto depth = terrain->getDepth();
930 auto terrainHeightVectorVerticesPerX = static_cast<int>(Math::ceil(width / Terrain2::STEP_SIZE));
931 auto terreinHeightVectorVerticesPerZ = static_cast<int>(Math::ceil(depth / Terrain2::STEP_SIZE));
932 // terrain
933 auto minHeight = terrain->getHeightVector()[0];
934 auto maxHeight = terrain->getHeightVector()[0];
935 for (auto heightValue: terrain->getHeightVector()) {
936 if (heightValue < minHeight) minHeight = heightValue;
937 if (heightValue > maxHeight) maxHeight = heightValue;
938 }
939 {
940 Transformations transformations;
941 transformations.setTranslation(Vector3(width / 2.0f, (minHeight + maxHeight) / 2.0f, depth / 2.0f));
942 transformations.update();
943 auto rigidBody = world->addStaticRigidBody(
944 "tdme.terrain",
945 true,
947 transformations,
948 0.5f,
949 {
950 new HeightMap(
951 terrainHeightVectorVerticesPerX,
952 terreinHeightVectorVerticesPerZ,
953 minHeight,
954 maxHeight,
955 terrain->getHeightVector().data()
956 )
957 }
958 );
959 rigidBody->setEnabled(enable);
960 }
961 // single foliage
962 {
963 //
964 auto& foliageMaps = terrain->getFoliageMaps();
965
966 //
967 map<int, int> prototypeBodyIdx;
968 for (auto& foliageMapPartition: foliageMaps) {
969 for (auto& foliageMapPartitionIt: foliageMapPartition) {
970 auto prototypeIdx = foliageMapPartitionIt.first;
971 auto& transformationsVector = foliageMapPartitionIt.second;
972 if (transformationsVector.empty() == true) continue;
973 auto foliagePrototype = prototype->getTerrain()->getFoliagePrototype(prototypeIdx);
974 if (foliagePrototype->isRenderGroups() == true) continue;
975 for (auto& foliageTransformations: transformationsVector) {
976 auto body = createBody(world, foliagePrototype, "tdme.foliage." + to_string(prototypeIdx) + "." + to_string(prototypeBodyIdx[prototypeIdx]++), foliageTransformations);
977 if (body == nullptr) continue;
978 if (translation.equals(Vector3()) == false) {
979 auto transformations = foliageTransformations;
980 transformations.setTranslation(transformations.getTranslation().clone().add(translation));
981 transformations.update();
982 body->fromTransformations(transformations);
983 }
984 body->setEnabled(enable);
985 }
986 }
987 }
988 }
989 }
990 }
991
992 //
993 for (auto i = 0; i < scene->getEntityCount(); i++) {
994 auto sceneEntity = scene->getEntityAt(i);
995
996 //
997 if (progressCallback != nullptr && progressStepCurrent % 1000 == 0) progressCallback->progress(0.0f + static_cast<float>(progressStepCurrent) / static_cast<float>(scene->getEntityCount()) * 1.0f);
998 progressStepCurrent++;
999
1000 //
1001 auto rigidBody = createBody(world, sceneEntity);
1002 if (rigidBody == nullptr) continue;
1003 if (translation.equals(Vector3()) == false) {
1004 auto transformations = sceneEntity->getTransformations();
1005 transformations.setTranslation(transformations.getTranslation().clone().add(translation));
1006 transformations.update();
1007 rigidBody->fromTransformations(transformations);
1008 }
1009 rigidBody->setEnabled(enable);
1010 }
1011
1012 //
1013 if (progressCallback != nullptr) {
1014 progressCallback->progress(1.0f);
1015 delete progressCallback;
1016 }
1017}
1018
1020{
1021 // terrain + water + foliage render groups + render groups
1022 {
1023 auto idx = 0;
1024 Entity* entity = nullptr;
1025 while ((entity = engine->getEntity("tdme.terrain." + to_string(idx++))) != nullptr) {
1026 entity->setEnabled(false);
1027 }
1028 }
1029 {
1030 auto idx = 0;
1031 Entity* entity = nullptr;
1032 while ((entity = engine->getEntity("tdme.water." + to_string(idx++))) != nullptr) {
1033 entity->setEnabled(false);
1034 }
1035 }
1036 {
1037 auto idx = 0;
1038 Entity* entity = nullptr;
1039 while ((entity = engine->getEntity("tdme.fo3rg." + to_string(idx++))) != nullptr) {
1040 entity->setEnabled(false);
1041 }
1042 }
1043 {
1044 auto idx = 0;
1045 Entity* entity = nullptr;
1046 while ((entity = engine->getEntity("tdme.o3rg." + to_string(idx++))) != nullptr) {
1047 entity->setEnabled(false);
1048 }
1049 }
1050
1051 // scene library
1052 auto sceneLibrary = scene->getLibrary();
1053
1054 // single foliage
1055 {
1056 auto prototype = sceneLibrary->getTerrainPrototype();
1057 if (prototype != nullptr) {
1058 //
1059 auto terrain = prototype->getTerrain();
1060 //
1061 for (auto prototypeIdx: terrain->getFoliagePrototypeIndices()) {
1062 auto foliagePrototype = prototype->getTerrain()->getFoliagePrototype(prototypeIdx);
1063 if (foliagePrototype->isRenderGroups() == true) continue;
1064 for (auto& entityId: terrain->getFoliagePrototypeEntityIds(prototypeIdx)) {
1065 auto entity = engine->getEntity(entityId);
1066 if (entity != nullptr) entity->setEnabled(false);
1067 }
1068 }
1069 }
1070 }
1071
1072 // scene entities
1073 for (auto i = 0; i < scene->getEntityCount(); i++) {
1074 auto sceneEntity = scene->getEntityAt(i);
1075 auto entity = engine->getEntity(sceneEntity->getId());
1076 if (entity == nullptr)
1077 continue;
1078
1079 entity->setEnabled(false);
1080 }
1081}
1082
1084{
1085 // terrain
1086 {
1087 auto body = world->getBody("tdme.terrain");
1088 if (body != nullptr) body->setEnabled(false);
1089 }
1090
1091 // scene library
1092 auto sceneLibrary = scene->getLibrary();
1093
1094 // single foliage
1095 {
1096 auto prototype = sceneLibrary->getTerrainPrototype();
1097 if (prototype != nullptr) {
1098 //
1099 auto terrain = prototype->getTerrain();
1100 //
1101 for (auto prototypeIdx: terrain->getFoliagePrototypeIndices()) {
1102 auto foliagePrototype = prototype->getTerrain()->getFoliagePrototype(prototypeIdx);
1103 if (foliagePrototype->isRenderGroups() == true) continue;
1104 for (auto& bodyId: terrain->getFoliagePrototypeEntityIds(prototypeIdx)) {
1105 auto body = world->getBody(bodyId);
1106 if (body != nullptr) body->setEnabled(false);
1107 }
1108 }
1109 }
1110 }
1111
1112 // scene entities
1113 for (auto i = 0; i < scene->getEntityCount(); i++) {
1114 auto sceneEntity = scene->getEntityAt(i);
1115 auto body = world->getBody(sceneEntity->getId());
1116 if (body == nullptr) continue;
1117 body->setEnabled(false);
1118 }
1119}
1120
1121void SceneConnector::enableScene(Engine* engine, Scene* scene, const Vector3& translation)
1122{
1123 // terrain + water + foliage render groups + render groups
1124 {
1125 auto idx = 0;
1126 Entity* entity = nullptr;
1127 while ((entity = engine->getEntity("tdme.terrain." + to_string(idx++))) != nullptr) {
1128 entity->setEnabled(true);
1129 }
1130 }
1131 {
1132 auto idx = 0;
1133 Entity* entity = nullptr;
1134 while ((entity = engine->getEntity("tdme.water." + to_string(idx++))) != nullptr) {
1135 entity->setEnabled(true);
1136 }
1137 }
1138 {
1139 auto idx = 0;
1140 Entity* entity = nullptr;
1141 while ((entity = engine->getEntity("tdme.fo3rg." + to_string(idx++))) != nullptr) {
1142 entity->setEnabled(true);
1143 }
1144 }
1145 {
1146 auto idx = 0;
1147 Entity* entity = nullptr;
1148 while ((entity = engine->getEntity("tdme.o3rg." + to_string(idx++))) != nullptr) {
1149 entity->setEnabled(true);
1150 }
1151 }
1152
1153 // scene library
1154 auto sceneLibrary = scene->getLibrary();
1155
1156 // single foliage
1157 {
1158 auto prototype = sceneLibrary->getTerrainPrototype();
1159 if (prototype != nullptr) {
1160 //
1161 auto terrain = prototype->getTerrain();
1162 //
1163 for (auto prototypeIdx: terrain->getFoliagePrototypeIndices()) {
1164 auto foliagePrototype = prototype->getTerrain()->getFoliagePrototype(prototypeIdx);
1165 if (foliagePrototype->isRenderGroups() == true) continue;
1166 for (auto& entityId: terrain->getFoliagePrototypeEntityIds(prototypeIdx)) {
1167 auto entity = engine->getEntity(entityId);
1168 if (entity != nullptr) entity->setEnabled(true);
1169 }
1170 }
1171 }
1172 }
1173
1174 // scene entities
1175 for (auto i = 0; i < scene->getEntityCount(); i++) {
1176 auto sceneEntity = scene->getEntityAt(i);
1177 auto entity = engine->getEntity(sceneEntity->getId());
1178 if (entity == nullptr)
1179 continue;
1180
1181 entity->fromTransformations(sceneEntity->getTransformations());
1182 entity->setTranslation(entity->getTranslation().clone().add(translation));
1183 if (sceneEntity->getPrototype()->getType() == Prototype_Type::EMPTY) {
1184 entity->setScale(Vector3(Math::sign(entity->getScale().getX()), Math::sign(entity->getScale().getY()), Math::sign(entity->getScale().getZ())));
1185 }
1186 entity->update();
1187 entity->setEnabled(true);
1188 }
1189}
1190
1191void SceneConnector::enableScene(World* world, Scene* scene, const Vector3& translation)
1192{
1193 // terrain
1194 {
1195 auto body = world->getBody("tdme.terrain");
1196 if (body != nullptr) {
1197 body->setEnabled(true);
1198 // TODO: translate
1199 }
1200 }
1201
1202 // scene library
1203 auto sceneLibrary = scene->getLibrary();
1204
1205 // single foliage
1206 {
1207 auto prototype = sceneLibrary->getTerrainPrototype();
1208 if (prototype != nullptr) {
1209 //
1210 auto terrain = prototype->getTerrain();
1211 //
1212 for (auto prototypeIdx: terrain->getFoliagePrototypeIndices()) {
1213 auto foliagePrototype = prototype->getTerrain()->getFoliagePrototype(prototypeIdx);
1214 if (foliagePrototype->isRenderGroups() == true) continue;
1215 for (auto& bodyId: terrain->getFoliagePrototypeEntityIds(prototypeIdx)) {
1216 auto body = world->getBody(bodyId);
1217 if (body != nullptr) body->setEnabled(true);
1218 }
1219 }
1220 }
1221 }
1222
1223 // scene entities
1224 Transformations transformations;
1225 for (auto i = 0; i < scene->getEntityCount(); i++) {
1226 auto sceneEntity = scene->getEntityAt(i);
1227 auto rigidBody = world->getBody(sceneEntity->getId());
1228 if (rigidBody == nullptr) continue;
1229 transformations.fromTransformations(sceneEntity->getTransformations());
1230 transformations.setTranslation(transformations.getTranslation().clone().add(translation));
1231 transformations.update();
1232 rigidBody->fromTransformations(transformations);
1233 rigidBody->setEnabled(true);
1234 }
1235}
1236
1238 {
1239 auto idx = 0;
1240 Entity* entity = nullptr;
1241 while ((entity = engine->getEntity("tdme.terrain." + to_string(idx++))) != nullptr) {
1242 Model* model = nullptr;
1243 if (entity->getEntityType() == Entity::ENTITYTYPE_OBJECT3D) model = static_cast<Object3D*>(entity)->getModel();
1244 engine->removeEntity(entity->getId());
1245 if (model != nullptr) delete model;
1246 }
1247 }
1248 {
1249 auto idx = 0;
1250 Entity* entity = nullptr;
1251 while ((entity = engine->getEntity("tdme.water." + to_string(idx++))) != nullptr) {
1252 Model* model = nullptr;
1253 if (entity->getEntityType() == Entity::ENTITYTYPE_OBJECT3D) model = static_cast<Object3D*>(entity)->getModel();
1254 engine->removeEntity(entity->getId());
1255 if (model != nullptr) delete model;
1256 }
1257 }
1258 //
1259 engine->reset();
1260}
1261
1262void SceneConnector::addSounds(Audio* audio, Prototype* prototype, const string& id, const int poolSize) {
1263 for (auto soundDefinition: prototype->getSounds()) {
1264 if (soundDefinition->getFileName().length() > 0) {
1265 for (auto poolIdx = 0; poolIdx < poolSize; poolIdx++) {
1266 string pathName = PrototypeReader::getResourcePathName(
1267 Tools::getPathName(prototype->getFileName()),
1268 soundDefinition->getFileName()
1269 );
1270 string fileName = Tools::getFileName(soundDefinition->getFileName());
1271 auto sound = new Sound(
1272 id + "." + soundDefinition->getId() + (poolSize > 1?"." + to_string(poolIdx):""),
1273 pathName,
1274 fileName
1275 );
1276 sound->setGain(soundDefinition->getGain());
1277 sound->setPitch(soundDefinition->getPitch());
1278 sound->setLooping(soundDefinition->isLooping());
1279 sound->setFixed(soundDefinition->isFixed());
1280 audio->addEntity(sound);
1281 }
1282 }
1283 }
1284}
1285
Interface to audio module.
Definition: Audio.h:30
void addEntity(AudioEntity *entity)
Adds a audio entity.
Definition: Audio.cpp:57
Sound audio entity implementation.
Definition: Sound.h:20
Engine main class.
Definition: Engine.h:122
bool removeEntity(const string &id)
Removes an entity.
Definition: Engine.cpp:451
Light * getLightAt(int32_t idx)
Returns light at idx (0 <= idx < 8)
Definition: Engine.h:950
static const map< string, ShaderParameter > getShaderParameterDefaults(const string &shaderId)
Returns parameter defaults of shader with given id.
Definition: Engine.cpp:2134
void addEntity(Entity *entity)
Adds an entity by id.
Definition: Engine.cpp:364
Entity * getEntity(const string &id)
Returns a entity by given id.
Definition: Engine.h:981
static constexpr int LIGHTS_MAX
Definition: Engine.h:173
void reset()
Removes all entities and caches.
Definition: Engine.cpp:652
static int32_t getEnvironmentMappingWidth()
Definition: Engine.h:678
static int32_t getEnvironmentMappingHeight()
Definition: Engine.h:685
Entity hierarchy to be used with engine class.
const ShaderParameter getShaderParameter(const string &parameterName) const
Returns shader parameter for given parameter name, if the value does not exist, the default will be r...
TDME engine entity.
Definition: Entity.h:31
virtual void setScale(const Vector3 &scale)=0
Set scale.
@ RENDERPASS_POST_POSTPROCESSING
Definition: Entity.h:58
virtual const Vector3 & getTranslation() const =0
virtual void setRenderPass(RenderPass renderPass)=0
Set render pass.
virtual const Vector3 & getScale() const =0
virtual const string & getId()=0
virtual void setTranslation(const Vector3 &translation)=0
Set translation.
virtual EntityType getEntityType()=0
virtual void fromTransformations(const Transformations &transformations)=0
Set up this transformations from given transformations.
virtual void update()=0
Update transformations.
virtual BoundingBox * getBoundingBox()=0
virtual void setContributesShadows(bool contributesShadows)=0
Enable/disable contributes shadows.
virtual void setPickable(bool pickable)=0
Set this object pickable.
virtual void setReceivesShadows(bool receivesShadows)=0
Enable/disable receives shadows.
virtual void setEnabled(bool enabled)=0
Enable/disable rendering.
virtual void setParentEntity(Entity *entity)=0
Set parent entity, needs to be called before adding to engine.
Environment mapping entity.
Fog particle system entity to be used with engine class.
LOD object 3D + imposter to be used with engine class.
void setEffectColorAddLOD2(const Color4 &effectColorAddLOD2)
Set effect color add for LOD2 level.
LOD object 3D to be used with engine class.
Definition: LODObject3D.h:47
void setEffectColorAddLOD2(const Color4 &effectColorAddLOD2)
Set effect color add for LOD2 level.
Definition: LODObject3D.h:267
Light representation.
Definition: Light.h:32
void setEnabled(bool enabled)
Set enabled.
Definition: Light.h:82
void setSpecular(const Color4 &specular)
Set specular light component.
Definition: Light.h:127
void setConstantAttenuation(float constantAttenuation)
Set up constant attenuation.
Definition: Light.h:202
void setSpotDirection(const Vector3 &spotDirection)
Set spot direction.
Definition: Light.h:157
void setAmbient(const Color4 &ambient)
Set ambient light component.
Definition: Light.h:97
void setQuadraticAttenuation(float quadraticAttenuation)
Set up quadratic attenuation.
Definition: Light.h:232
void setSpotCutOff(float spotCutOff)
Set spot cut off.
Definition: Light.h:187
void setPosition(const Vector4 &position)
Set light position.
Definition: Light.h:142
void setDiffuse(const Color4 &diffuse)
Set diffuse light component.
Definition: Light.h:112
void setLinearAttenuation(float linearAttenuation)
Set up linear attenuation.
Definition: Light.h:217
void setSpotExponent(float spotExponent)
Set up spot exponent.
Definition: Light.h:172
Object 3D render group for static objects that might be animated by shaders.
void setShader(const string &id)
Set shader id.
void setReceivesShadows(bool receivesShadows) override
Enable/disable receives shadows.
void addObject(Model *model, const Transformations &transformations)
Adds a instance to this render group.
void setPickable(bool pickable) override
Set this object pickable.
void setShaderParameter(const string &parameterName, const ShaderParameter &parameterValue)
Set shader parameter for given parameter name.
void setDistanceShader(const string &id)
Set distance shader id.
void setEnabled(bool enabled) override
Enable/disable rendering.
void setDistanceShaderDistance(float distanceShaderDistance)
Set distance shader distance.
void setContributesShadows(bool contributesShadows) override
Enable/disable contributes shadows.
void setDistanceShaderParameter(const string &parameterName, const ShaderParameter &parameterValue)
Set distance shader parameter for given parameter name.
Object 3D to be used with engine class.
Definition: Object3D.h:60
void setRenderPass(RenderPass renderPass) override
Set render pass.
Definition: Object3D.h:339
void setReflectionEnvironmentMappingId(const string &reflectionEnvironmentMappingId)
Definition: Object3D.h:408
Object particle system entity to be used with engine class.
Particle system group, which combines several particle systems into a group, to be used with engine c...
Point particle system entity to be used with engine class.
Scene engine/physics connector.
static STATIC_DLL_IMPEXT int renderGroupsLODLevels
static Entity * createEntity(Prototype *prototype, const string &id, const Transformations &transformations, int instances=1, Entity *parentEntity=nullptr)
Create engine entity.
static Body * createBody(World *world, Prototype *prototype, const string &id, const Transformations &transformations, uint16_t collisionTypeId=0, int index=-1, PrototypePhysics_BodyType *overrideType=nullptr)
Create rigid body.
static void addScene(Engine *engine, Scene *scene, bool addEmpties, bool addTrigger, bool addEnvironmentMapping, bool pickable, bool enable=true, const Vector3 &translation=Vector3(0.0f, 0.0f, 0.0f), ProgressCallback *progressCallback=nullptr)
Add scene to engine.
static STATIC_DLL_IMPEXT int renderGroupsReduceBy
static STATIC_DLL_IMPEXT Model * emptyModel
static constexpr int32_t RIGIDBODY_TYPEID_COLLISION
static STATIC_DLL_IMPEXT float renderGroupsPartitionHeight
static void enableScene(Engine *engine, Scene *scene, const Vector3 &translation=Vector3(0.0f, 0.0f, 0.0f))
Enable disabled scene in engine.
static constexpr int32_t RIGIDBODY_TYPEID_DYNAMIC
static STATIC_DLL_IMPEXT int renderGroupsLOD3ReduceBy
static STATIC_DLL_IMPEXT bool enableEarlyZRejection
static STATIC_DLL_IMPEXT int renderGroupsLOD2ReduceBy
static void disableScene(Engine *engine, Scene *scene)
Disable scene in engine.
static Entity * createEmpty(const string &id, const Transformations &transformations)
Create engine entity.
static STATIC_DLL_IMPEXT float renderGroupsPartitionDepth
static Entity * createParticleSystem(PrototypeParticleSystem *particleSystem, const string &id, bool enableDynamicShadows=true)
Create particle system.
static constexpr int32_t RIGIDBODY_TYPEID_TRIGGER
static void addSounds(Audio *audio, Prototype *prototype, const string &id, const int poolSize=1)
Add scene entity sounds into given audio instance associated with given id.
static void resetEngine(Engine *engine, Scene *scene)
Reset engine regarding given scene.
static STATIC_DLL_IMPEXT float renderGroupsPartitionWidth
static STATIC_DLL_IMPEXT float renderGroupsLOD3MinDistance
static STATIC_DLL_IMPEXT float renderGroupsLOD2MinDistance
static constexpr int32_t RIGIDBODY_TYPEID_STATIC
Transformations which contain scale, rotations and translation.
void setTranslation(const Vector3 &translation)
Set translation.
virtual void fromTransformations(const Transformations &transformations)
Set up this transformations from given transformations.
virtual void update()
Computes transformation matrix.
const Vector3 & getTranslation() const
Color 4 base definition class.
Definition: Color4Base.h:19
Color 4 definition.
Definition: Color4.h:20
Representation of a 3d model.
Definition: Model.h:32
Dynamic rigid/static rigid/collision body class.
Definition: Body.h:43
void setEnabled(bool enabled)
Set up if rigid body is enabled.
Definition: Body.cpp:205
Dynamic physics world class.
Definition: World.h:38
Body * addRigidBody(const string &id, bool enabled, uint16_t collisionTypeId, const Transformations &transformations, float restitution, float friction, float mass, const Vector3 &inertiaTensor, vector< BoundingVolume * > boundingVolumes)
Add a rigid body.
Definition: World.cpp:86
Body * addStaticRigidBody(const string &id, bool enabled, uint16_t collisionTypeId, const Transformations &transformations, float friction, vector< BoundingVolume * > boundingVolumes)
Add a static rigid body.
Definition: World.cpp:110
Body * getBody(const string &id)
Returns body identified by id.
Definition: World.cpp:122
Body * addCollisionBody(const string &id, bool enabled, uint16_t collisionTypeId, const Transformations &transformations, vector< BoundingVolume * > boundingVolumes)
Add a collision body.
Definition: World.cpp:99
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
Definition: BoundingBox.h:25
const Vector3 & getDimensions() const
Definition: BoundingBox.h:127
Convex mesh physics primitive.
Definition: ConvexMesh.h:33
Height map physics primitive.
Definition: HeightMap.h:18
Oriented bounding box physics primitive.
Sphere physics primitive.
Definition: Sphere.h:18
Terrain mesh physics primitive.
Definition: TerrainMesh.h:28
Base property model class.
Definition: BaseProperty.h:16
Prototype audio definition.
Prototype LOD level definition.
PrototypeParticleSystem_PointParticleEmitter * getPointParticleEmitter()
PrototypeParticleSystem_CircleParticleEmitterPlaneVelocity * getCircleParticleEmitterPlaneVelocity()
PrototypeParticleSystem_PointParticleSystem * getPointParticleSystem()
PrototypeParticleSystem_CircleParticleEmitter * getCircleParticleEmitter()
PrototypeParticleSystem_BoundingBoxParticleEmitter * getBoundingBoxParticleEmitters()
PrototypeParticleSystem_ObjectParticleSystem * getObjectParticleSystem()
PrototypeParticleSystem_SphereParticleEmitter * getSphereParticleEmitter()
PrototypeParticleSystem_FogParticleSystem * getFogParticleSystem()
Prototype physics body definitions.
PrototypePhysics_BodyType * getType() const
Prototype * getFoliagePrototype(int prototypeIdx)
Get foliage prototype by given index.
Prototype definition.
Definition: Prototype.h:49
PrototypeLODLevel * getLODLevel2()
Definition: Prototype.h:251
const vector< PrototypeAudio * > & getSounds()
Definition: Prototype.h:479
PrototypeImposterLOD * getImposterLOD()
Definition: Prototype.h:283
const string & getShader()
Get shader.
Definition: Prototype.h:398
PrototypeLODLevel * getLODLevel3()
Definition: Prototype.h:264
float getDistanceShaderDistance()
Get distance shader distance.
Definition: Prototype.h:432
bool isTerrainMesh()
Is terrain mesh.
Definition: Prototype.h:366
const string & getDistanceShader()
Get distance shader.
Definition: Prototype.h:415
PrototypePhysics * getPhysics()
Definition: Prototype.h:244
PrototypeParticleSystem * getParticleSystemAt(int idx)
Get particle system at given index.
Definition: Prototype.h:327
const EntityShaderParameters & getDistanceShaderParameters()
Get distance shader parameters.
Definition: Prototype.h:464
int64_t getEnvironmentMapTimeRenderUpdateFrequency()
Definition: Prototype.h:546
PrototypeBoundingVolume * getBoundingVolume(int idx)
Get bounding volume at given index.
Definition: Prototype.h:223
PrototypeTerrain * getTerrain()
Definition: Prototype.h:561
const EntityShaderParameters & getShaderParameters()
Get shader parameters.
Definition: Prototype.h:448
Scene entity definition.
Definition: SceneEntity.h:24
Transformations & getTransformations()
Definition: SceneEntity.h:78
Scene prototype library definition.
Definition: SceneLibrary.h:27
Prototype * getTerrainPrototype()
Get a terrain prototype.
Scene light definition.
Definition: SceneLight.h:21
const Color4 & getAmbient() const
Definition: SceneLight.h:69
float getQuadraticAttenuation() const
Definition: SceneLight.h:204
const Color4 & getSpecular() const
Definition: SceneLight.h:99
float getLinearAttenuation() const
Definition: SceneLight.h:189
const Vector3 & getSpotDirection() const
Definition: SceneLight.h:129
const Color4 & getDiffuse() const
Definition: SceneLight.h:84
float getConstantAttenuation() const
Definition: SceneLight.h:174
const Vector4 & getPosition() const
Definition: SceneLight.h:114
Scene definition.
Definition: Scene.h:41
SceneLibrary * getLibrary()
Definition: Scene.h:168
SceneLight * getLightAt(int i)
Get light at given index.
Definition: Scene.h:140
SceneEntity * getEntityAt(int idx)
Returns entity at given index.
Definition: Scene.h:287
Standard math functions.
Definition: Math.h:21
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
bool equals(const Vector3 &v) const
Compares this vector with given vector.
Definition: Vector3.h:381
Vector3 clone() const
Clones the vector.
Definition: Vector3.h:372
Vector3 & add(const Vector3 &v)
Adds a vector.
Definition: Vector3.h:301
3D vector 4 class
Definition: Vector4.h:19
float getY() const
Definition: Vector4.h:149
float getX() const
Definition: Vector4.h:132
float getZ() const
Definition: Vector4.h:166
float getW() const
Definition: Vector4.h:183
Console class.
Definition: Console.h:26
const string & getName() const
Definition: Enum.h:30
Model tools functions class.
Definition: ModelTools.h:38
Mutable string class.
Definition: MutableString.h:16
Helper class to create models from physics primitive bounding volumes.
Definition: Primitives.h:34
String tools class.
Definition: StringTools.h:20
Terrain 2 utility.
Definition: Terrain2.h:33
Particle system entity interface.
virtual void progress(float value)=0
Perform action.