7#include <unordered_map>
8#include <unordered_set>
76using std::unordered_map;
77using std::unordered_set;
137constexpr int32_t EntityRenderer::BATCHRENDERER_MAX;
138constexpr int32_t EntityRenderer::INSTANCEDRENDERING_OBJECTS_MAX;
161 delete contextIdx.bbEffectColorMuls;
162 delete contextIdx.bbEffectColorAdds;
163 delete contextIdx.bbMvMatrices;
167 delete batchRenderer;
179 auto created =
false;
181 contexts[i].vboInstancedRenderingIds = vboManaged->getVBOIds();
193 batchRenderer->dispose();
194 batchRenderer->release();
204 if (batchRenderer->acquire())
return batchRenderer;
210 batchRenderer->initialize();
212 if (batchRenderer->acquire())
return batchRenderer;
215 Console::println(
string(
"EntityRenderer::acquireTrianglesBatchRenderer()::failed"));
230 auto elementsIssued = 0;
233 queueElement->engine =
engine;
234 queueElement->rendering.renderPass = renderPass;
236 queueElement->rendering.renderTypes = renderTypes;
237 for (
auto i = 0; i < objects.size(); i++) {
238 queueElement->objects.push_back(objects[i]);
240 auto queueElementToSubmit = queueElement;
243 queueElement->engine =
engine;
244 queueElement->rendering.renderPass = renderPass;
246 queueElement->rendering.renderTypes = renderTypes;
251 if (queueElement->objects.empty() ==
false) {
257 while (
true ==
true) {
258 auto elementsProcessed = 0;
259 for (
auto engineThread:
Engine::engineThreads) elementsProcessed+= engineThread->getProcessedElements();
260 if (elementsProcessed == elementsIssued) {
272 engineThread->transparentRenderFacesPool->reset();
282 if (transparentRenderFaces.size() > 0) {
303 for (
auto transparentRenderFace: transparentRenderFaces) {
347 auto object3DNode = transparentRenderFaces[0]->object3DNode;
348 auto object3D =
static_cast<Object3D*
>(object3DNode->object);
351 if (object3DNode->mesh->skinning ==
true) {
354 modelViewMatrix.
set(*object3DNode->nodeTransformationsMatrix).
multiply(object3D->getTransformationsMatrix());
357 auto model = object3DNode->object->getModel();
358 auto& facesEntities = object3DNode->node->getFacesEntities();
361 auto& effectColorAdd = object3D->getEffectColorAdd();
362 auto& effectColorMul = object3D->getEffectColorMul();
364 auto textureCoordinates =
false;
366 for (
auto i = 0; i < transparentRenderFaces.size(); i++) {
367 auto transparentRenderFace = transparentRenderFaces[i];
368 auto facesEntityIdx = transparentRenderFace->facesEntityIdx;
370 if (facesEntity != &facesEntities[facesEntityIdx]) {
371 facesEntity = &facesEntities[facesEntityIdx];
376 auto transparentRenderFacesNodeKey =
TransparentRenderFacesGroup::createKey(model, object3DNode, facesEntityIdx, effectColorAdd, effectColorMul, material, textureCoordinates, object3D->getShader());
381 trfNode = trfNodeIt->second;
383 if (trfNode ==
nullptr) {
386 trfNode->
set(
this, model, object3DNode, facesEntityIdx, effectColorAdd, effectColorMul, material, textureCoordinates, object3D->getShader());
390 for (
auto vertexIdx = 0; vertexIdx < 3; vertexIdx++) {
391 auto arrayIdx = transparentRenderFace->object3DNode->mesh->indices[transparentRenderFace->faceIdx * 3 + vertexIdx];
393 modelViewMatrix.
multiply((*transparentRenderFace->object3DNode->mesh->vertices)[arrayIdx]),
394 modelViewMatrix.
multiplyNoTranslation((*transparentRenderFace->object3DNode->mesh->normals)[arrayIdx]),
395 transparentRenderFace->object3DNode->textureMatricesByEntities[facesEntityIdx].multiply(
396 textureCoordinates.size() > 0?
397 Vector2(textureCoordinates[arrayIdx].getArray()):
424 auto& object3DRenderContext =
contexts[0];
433 auto currentFrontFace = -1;
434 string shaderParametersHash;
435 auto firstObject = objects[0];
437 vector<int32_t>* boundVBOBaseIds =
nullptr;
438 vector<int32_t>* boundVBOTangentBitangentIds =
nullptr;
439 vector<int32_t>* boundVBOOrigins =
nullptr;
440 auto currentLODLevel = -1;
441 int32_t boundEnvironmentMappingCubeMapTextureId = -1;
442 Vector3 boundEnvironmentMappingCubeMapPosition;
443 for (
auto object3DNodeIdx = 0; object3DNodeIdx < firstObject->object3dNodes.size(); object3DNodeIdx++) {
444 auto object3DNode = firstObject->object3dNodes[object3DNodeIdx];
446 auto& facesEntities = object3DNode->node->getFacesEntities();
448 auto facesEntityIdxCount = facesEntities.size();
449 for (
auto faceEntityIdx = 0; faceEntityIdx < facesEntityIdxCount; faceEntityIdx++) {
450 auto facesEntity = &facesEntities[faceEntityIdx];
451 auto isTextureCoordinatesAvailable = facesEntity->isTextureCoordinatesAvailable();
452 auto faces = facesEntity->getFaces().size() * firstObject->instances;
453 auto facesToRender = facesEntity->getFaces().size() * firstObject->enabledInstances;
455 auto material = facesEntity->getMaterial();
456 auto specularMaterialProperties = material !=
nullptr?material->getSpecularMaterialProperties():
nullptr;
458 auto transparentFacesEntity =
false;
460 if (specularMaterialProperties !=
nullptr) {
461 if (specularMaterialProperties->hasColorTransparency() ==
true || specularMaterialProperties->hasTextureTransparency() ==
true) transparentFacesEntity =
true;
462 if (material->isDoubleSided() ==
true ||
463 (specularMaterialProperties->hasDiffuseTextureTransparency() ==
true && specularMaterialProperties->hasDiffuseTextureMaskedTransparency() ==
true)) {
468 if (transparentFacesEntity ==
true) {
470 auto objectCount = objects.size();
471 for (
auto objectIdx = 0; objectIdx < objectCount; objectIdx++) {
472 auto object = objects[objectIdx];
473 auto _object3DNode =
object->object3dNodes[object3DNodeIdx];
477 if (collectTransparentFaces ==
true) {
479 (_object3DNode->mesh->skinning ==
true?
481 modelViewMatrix.
set(*_object3DNode->nodeTransformationsMatrix).
multiply(object->getTransformationsMatrix())
482 ).multiply(cameraMatrix),
483 object->object3dNodes[object3DNodeIdx],
495 bool materialUpdateOnly =
false;
496 auto objectCount = objects.size();
497 for (
auto objectIdx = 0; objectIdx < objectCount; objectIdx++) {
498 auto object = objects[objectIdx];
499 auto _object3DNode =
object->object3dNodes[object3DNodeIdx];
501 if (object->effectColorMul.getAlpha() < 1.0f - Math::EPSILON ||
502 object->effectColorAdd.getAlpha() < -Math::EPSILON) {
504 if (collectTransparentFaces ==
true) {
506 (_object3DNode->mesh->skinning ==
true ?
508 modelViewMatrix.
set(*_object3DNode->nodeTransformationsMatrix).
multiply(object->getTransformationsMatrix())
509 ).multiply(cameraMatrix),
520 auto distanceSquared = objectCamFromAxis.
set(object->getBoundingBoxTransformed()->computeClosestPointInBoundingBox(camera->getLookFrom())).
sub(camera->getLookFrom()).
computeLengthSquared();
521 auto distanceShader =
object->getDistanceShader().empty() ==
true?
false:distanceSquared >= Math::square(object->getDistanceShaderDistance());
523 distanceShader ==
false?
525 object->getDistanceShader();
532 materialUpdateOnly =
false;
536 if (materialUpdateOnly ==
false ||
checkMaterialChangable(_object3DNode, faceEntityIdx, renderTypes) ==
true) {
537 setupMaterial(contextIdx, _object3DNode, faceEntityIdx, renderTypes, materialUpdateOnly, materialKey);
539 materialUpdateOnly =
true;
548 auto currentVBOBaseIds = _object3DNode->renderer->vboBaseIds;
549 if (boundVBOBaseIds != currentVBOBaseIds) {
550 boundVBOBaseIds = currentVBOBaseIds;
552 if (isTextureCoordinatesAvailable ==
true &&
564 auto currentVBOLods = _object3DNode->renderer->vboLods;
565 if (currentVBOLods !=
nullptr) {
568 if (currentVBOLods->size() >= 3 && distanceSquared >= Math::square(_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD3Distance())) {
571 if (currentVBOLods->size() >= 2 && distanceSquared >= Math::square(_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD2Distance())) {
574 if (currentVBOLods->size() >= 1 && distanceSquared >= Math::square(_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD1Distance())) {
583 faces = (_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD3Indices().size() / 3) * firstObject->instances;
584 facesToRender = (_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD3Indices().size() / 3) * firstObject->enabledInstances;
587 faces = (_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD2Indices().size() / 3) * firstObject->instances;
588 facesToRender = (_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD2Indices().size() / 3) * firstObject->enabledInstances;
591 faces = (_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD1Indices().size() / 3) * firstObject->instances;
592 facesToRender = (_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD1Indices().size() / 3) * firstObject->enabledInstances;
597 currentLODLevel = lodLevel;
600 auto currentVBONormalMappingIds = _object3DNode->renderer->vboNormalMappingIds;
609 auto currentVBOOrigins = _object3DNode->renderer->vboOrigins;
610 if (currentVBOOrigins !=
nullptr && currentVBOOrigins != boundVBOOrigins) {
615 _object3DNode->mesh->skinning ==
true?
617 modelViewMatrix.
set(*_object3DNode->nodeTransformationsMatrix).
multiply(object->getTransformationsMatrix())
622 if (objectFrontFace != currentFrontFace) {
624 currentFrontFace = objectFrontFace;
634 shadowMapping !=
nullptr) {
635 shadowMapping->startObjectTransformations(
637 _object3DNode->mesh->skinning ==
true ?
639 modelViewMatrix.
set(*_object3DNode->nodeTransformationsMatrix).
multiply(object->getTransformationsMatrix())
651 if (object->getReflectionEnvironmentMappingId().empty() ==
false) {
652 auto environmentMappingEntityCandidate =
engine->
getEntity(object->getReflectionEnvironmentMappingId());
653 if (environmentMappingEntityCandidate !=
nullptr) {
655 environmentMappingEntity =
static_cast<EnvironmentMapping*
>(environmentMappingEntityCandidate);
659 if (entityHierarchyEnvironmentMappingEntity !=
nullptr) environmentMappingEntity =
static_cast<EnvironmentMapping*
>(entityHierarchyEnvironmentMappingEntity);
662 if (environmentMappingEntity !=
nullptr) {
663 Vector3 environmentMappingTranslation;
664 object->getTransformationsMatrix().getTranslation(environmentMappingTranslation);
666 Vector3 environmentMappingCubeMapPosition =
object->hasReflectionEnvironmentMappingPosition() ==
true?
object->getReflectionEnvironmentMappingPosition():environmentMappingTranslation;
667 if (environmentMappingCubeMapTextureId != boundEnvironmentMappingCubeMapTextureId || environmentMappingCubeMapPosition.
equals(boundEnvironmentMappingCubeMapPosition) ==
false) {
668 boundEnvironmentMappingCubeMapTextureId = environmentMappingCubeMapTextureId;
669 boundEnvironmentMappingCubeMapPosition = environmentMappingCubeMapPosition;
677 if (environmentMappingEntity ==
nullptr && boundEnvironmentMappingCubeMapTextureId != -1) {
681 boundEnvironmentMappingCubeMapTextureId = -1;
687 shadowMapping !=
nullptr) {
688 shadowMapping->endObjectTransformations();
693 if (specularMaterialProperties !=
nullptr) {
694 if (material->isDoubleSided() ==
true ||
695 (specularMaterialProperties->hasDiffuseTextureTransparency() ==
true && specularMaterialProperties->hasDiffuseTextureMaskedTransparency() ==
true)) {
710 auto& object3DRenderContext =
contexts[threadIdx];
711 auto contextIdx = threadIdx;
722 auto cullingMode = 1;
725 auto firstObject = objects[0];
728 for (
auto object3DNodeIdx = 0; object3DNodeIdx < firstObject->object3dNodes.size(); object3DNodeIdx++) {
729 auto object3DNode = firstObject->object3dNodes[object3DNodeIdx];
731 auto& facesEntities = object3DNode->node->getFacesEntities();
733 auto facesEntityIdxCount = facesEntities.size();
734 for (
auto faceEntityIdx = 0; faceEntityIdx < facesEntityIdxCount; faceEntityIdx++) {
735 auto facesEntity = &facesEntities[faceEntityIdx];
736 auto isTextureCoordinatesAvailable = facesEntity->isTextureCoordinatesAvailable();
737 auto faces = facesEntity->getFaces().size() * firstObject->instances;
738 auto facesToRender = facesEntity->getFaces().size() * firstObject->enabledInstances;
740 auto material = facesEntity->getMaterial();
741 auto specularMaterialProperties = material !=
nullptr?material->getSpecularMaterialProperties():
nullptr;
743 auto transparentFacesEntity =
false;
745 if (specularMaterialProperties !=
nullptr) {
746 if (specularMaterialProperties->hasColorTransparency() ==
true || specularMaterialProperties->hasTextureTransparency() ==
true) transparentFacesEntity =
true;
749 if (transparentFacesEntity ==
true) {
751 auto objectCount = objects.size();
752 for (
auto objectIdx = 0; objectIdx < objectCount; objectIdx++) {
753 auto object = objects[objectIdx];
754 auto _object3DNode =
object->object3dNodes[object3DNodeIdx];
758 if (collectTransparentFaces ==
true) {
760 (_object3DNode->mesh->skinning ==
true?
762 modelViewMatrixTemp.
set(*_object3DNode->nodeTransformationsMatrix).
multiply(object->getTransformationsMatrix())
763 ).multiply(cameraMatrix),
764 object->object3dNodes[object3DNodeIdx],
776 if (material->isDoubleSided() ==
true ||
777 (specularMaterialProperties->hasDiffuseTextureTransparency() ==
true && specularMaterialProperties->hasDiffuseTextureMaskedTransparency() ==
true)) {
778 if (cullingMode != 0) {
783 if (cullingMode != 1) {
789 if (cullingMode != 1) {
796 object3DRenderContext.objectsToRender = objects;
798 auto hadFrontFaceSetup =
false;
799 auto hadShaderSetup =
false;
806 FloatBuffer fbEffectColorMuls = object3DRenderContext.bbEffectColorMuls->asFloatBuffer();
807 FloatBuffer fbEffectColorAdds = object3DRenderContext.bbEffectColorAdds->asFloatBuffer();
808 FloatBuffer fbMvMatrices = object3DRenderContext.bbMvMatrices->asFloatBuffer();
811 string shaderParametersHash;
812 bool materialUpdateOnly =
false;
813 vector<int32_t>* boundVBOBaseIds =
nullptr;
814 vector<int32_t>* boundVBOTangentBitangentIds =
nullptr;
815 vector<int32_t>* boundVBOOrigins =
nullptr;
816 auto currentLODLevel = -1;
817 int32_t boundEnvironmentMappingCubeMapTextureId = -1;
818 Vector3 boundEnvironmentMappingCubeMapPosition;
819 auto objectCount = object3DRenderContext.objectsToRender.size();
822 auto textureMatrix = object3DRenderContext.objectsToRender[0]->object3dNodes[object3DNodeIdx]->textureMatricesByEntities[faceEntityIdx];
825 for (
auto objectIdx = 0; objectIdx < objectCount; objectIdx++) {
826 auto object = object3DRenderContext.objectsToRender[objectIdx];
827 auto _object3DNode =
object->object3dNodes[object3DNodeIdx];
830 if (object->effectColorMul.getAlpha() < 1.0f - Math::EPSILON ||
831 object->effectColorAdd.getAlpha() < -Math::EPSILON) {
833 if (collectTransparentFaces ==
true) {
835 (_object3DNode->mesh->skinning ==
true ?
837 modelViewMatrixTemp.
set(*_object3DNode->nodeTransformationsMatrix).
multiply(object->getTransformationsMatrix())
838 ).multiply(cameraMatrix),
850 object3DRenderContext.objectsNotRendered.push_back(
object);
855 if (_object3DNode->textureMatricesByEntities[faceEntityIdx].equals(textureMatrix) ==
false) {
856 object3DRenderContext.objectsNotRendered.push_back(
object);
862 auto distanceSquared = objectCamFromAxis.
set(object->getBoundingBoxTransformed()->computeClosestPointInBoundingBox(camera->getLookFrom())).
sub(camera->getLookFrom()).
computeLengthSquared();
863 auto distanceShader =
object->getDistanceShader().empty() ==
true?
false:distanceSquared >= Math::square(object->getDistanceShaderDistance());
865 distanceShader ==
false?
867 object->getDistanceShader();
868 if (hadShaderSetup ==
false) {
878 hadShaderSetup =
true;
881 object3DRenderContext.objectsNotRendered.push_back(
object);
886 auto materialKeyCurrent = materialKey;
887 if (materialUpdateOnly ==
false ||
checkMaterialChangable(_object3DNode, faceEntityIdx, renderTypes) ==
true) {
888 setupMaterial(contextIdx, _object3DNode, faceEntityIdx, renderTypes, materialUpdateOnly, materialKeyCurrent, materialKey);
890 if (materialUpdateOnly ==
false) {
891 materialKey = materialKeyCurrent;
892 materialUpdateOnly =
true;
897 if (materialKey != materialKeyCurrent) {
898 object3DRenderContext.objectsNotRendered.push_back(
object);
903 if (shaderParametersHash.empty() ==
true) {
909 object3DRenderContext.objectsNotRendered.push_back(
object);
913 auto currentVBOBaseIds = _object3DNode->renderer->vboBaseIds;
914 if (boundVBOBaseIds ==
nullptr) {
915 boundVBOBaseIds = currentVBOBaseIds;
917 if (isTextureCoordinatesAvailable ==
true &&
932 if (boundVBOBaseIds != currentVBOBaseIds) {
933 object3DRenderContext.objectsNotRendered.push_back(
object);
936 auto currentVBOLods = _object3DNode->renderer->vboLods;
937 if (currentVBOLods !=
nullptr) {
940 if (currentVBOLods->size() >= 3 && distanceSquared >= Math::square(_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD3Distance())) {
943 if (currentVBOLods->size() >= 2 && distanceSquared >= Math::square(_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD2Distance())) {
946 if (currentVBOLods->size() >= 1 && distanceSquared >= Math::square(_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD1Distance())) {
949 if (currentLODLevel != -1 && lodLevel != currentLODLevel) {
950 object3DRenderContext.objectsNotRendered.push_back(
object);
957 faces = (_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD3Indices().size() / 3) * firstObject->instances;
958 facesToRender = (_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD3Indices().size() / 3) * firstObject->enabledInstances;
961 faces = (_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD2Indices().size() / 3) * firstObject->instances;
962 facesToRender = (_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD2Indices().size() / 3) * firstObject->enabledInstances;
965 faces = (_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD1Indices().size() / 3) * firstObject->instances;
966 facesToRender = (_object3DNode->node->getFacesEntities()[faceEntityIdx].getLOD1Indices().size() / 3) * firstObject->enabledInstances;
971 currentLODLevel = lodLevel;
974 auto currentVBONormalMappingIds = _object3DNode->renderer->vboNormalMappingIds;
978 if (boundVBOTangentBitangentIds ==
nullptr) {
984 boundVBOTangentBitangentIds = currentVBONormalMappingIds;
987 if (currentVBONormalMappingIds != boundVBOTangentBitangentIds) {
988 object3DRenderContext.objectsNotRendered.push_back(
object);
994 auto currentVBOOrigins = _object3DNode->renderer->vboOrigins;
995 if (currentVBOOrigins !=
nullptr) {
997 if (boundVBOOrigins ==
nullptr) {
1000 boundVBOOrigins = currentVBOOrigins;
1003 if (currentVBOOrigins != boundVBOOrigins) {
1004 object3DRenderContext.objectsNotRendered.push_back(
object);
1010 modelViewMatrix.
set(
1011 _object3DNode->mesh->skinning ==
true?
1013 modelViewMatrixTemp.
set(*_object3DNode->nodeTransformationsMatrix).
1014 multiply(object->getTransformationsMatrix())
1020 if (cullingMode == 1) {
1021 if (hadFrontFaceSetup ==
false) {
1022 hadFrontFaceSetup =
true;
1023 if (objectFrontFace != frontFace) {
1024 frontFace = objectFrontFace;
1028 if (objectFrontFace != frontFace) {
1029 object3DRenderContext.objectsNotRendered.push_back(
object);
1035 if (object->getReflectionEnvironmentMappingId().empty() ==
false) {
1038 auto environmentMappingEntityCandidate =
engine->
getEntity(object->getReflectionEnvironmentMappingId());
1039 if (environmentMappingEntityCandidate !=
nullptr) {
1041 environmentMappingEntity =
static_cast<EnvironmentMapping*
>(environmentMappingEntityCandidate);
1045 if (entityHierarchyEnvironmentMappingEntity !=
nullptr) environmentMappingEntity =
static_cast<EnvironmentMapping*
>(entityHierarchyEnvironmentMappingEntity);
1049 if (environmentMappingEntity !=
nullptr) {
1050 Vector3 environmentMappingTranslation;
1051 object->getTransformationsMatrix().getTranslation(environmentMappingTranslation);
1052 auto environmentMappingCubeMapTextureId = environmentMappingEntity->
getCubeMapTextureId();
1053 Vector3 environmentMappingCubeMapPosition =
object->hasReflectionEnvironmentMappingPosition() ==
true?
object->getReflectionEnvironmentMappingPosition():environmentMappingTranslation;
1054 if (boundEnvironmentMappingCubeMapTextureId == -1) {
1055 boundEnvironmentMappingCubeMapTextureId = environmentMappingCubeMapTextureId;
1056 boundEnvironmentMappingCubeMapPosition = environmentMappingCubeMapPosition;
1062 if (boundEnvironmentMappingCubeMapTextureId != environmentMappingCubeMapTextureId || environmentMappingCubeMapPosition.
equals(boundEnvironmentMappingCubeMapPosition) ==
false) {
1063 object3DRenderContext.objectsNotRendered.push_back(
object);
1068 if (boundEnvironmentMappingCubeMapTextureId != -1) {
1069 object3DRenderContext.objectsNotRendered.push_back(
object);
1074 fbEffectColorMuls.
put(object->effectColorMul.getArray());
1075 fbEffectColorAdds.
put(object->effectColorAdd.getArray());
1083 auto objectsToRenderIssue = fbMvMatrices.
getPosition() / 16;
1084 if (objectsToRenderIssue > 0) {
1113 object3DRenderContext.objectsToRender = object3DRenderContext.objectsNotRendered;
1114 object3DRenderContext.objectsNotRendered.clear();
1117 if (boundEnvironmentMappingCubeMapTextureId != -1) {
1122 }
while (object3DRenderContext.objectsToRender.size() > 0);
1130 if (cullingMode != 1) {
1139 object3DRenderContext.objectsToRender.clear();
1140 object3DRenderContext.objectsNotRendered.clear();
1146 auto material = facesEntities[facesEntityIdx].getMaterial();
1148 if (material ==
nullptr) material = Material::getDefaultMaterial();
1149 auto specularMaterialProperties = material->getSpecularMaterialProperties();
1150 auto pbrMaterialProperties = material->getPBRMaterialProperties();
1153 materialKey = material->getId();
1159 if (updateOnly ==
false) {
1164 rendererMaterial.
ambient = specularMaterialProperties->getAmbientColor().getArray();
1165 rendererMaterial.diffuse = specularMaterialProperties->getDiffuseColor().getArray();
1166 rendererMaterial.specular = specularMaterialProperties->getSpecularColor().getArray();
1167 rendererMaterial.emission = specularMaterialProperties->getEmissionColor().getArray();
1168 rendererMaterial.shininess = specularMaterialProperties->getShininess();
1169 rendererMaterial.reflection = specularMaterialProperties->getReflection();
1170 rendererMaterial.diffuseTextureMaskedTransparency = specularMaterialProperties->hasDiffuseTextureMaskedTransparency() ==
true?1:0;
1171 rendererMaterial.diffuseTextureMaskedTransparencyThreshold = specularMaterialProperties->getDiffuseTextureMaskedTransparencyThreshold();
1193 if (pbrMaterialProperties ==
nullptr) {
1195 rendererMaterial.metallicFactor = 1.0f;
1196 rendererMaterial.roughnessFactor = 1.0f;
1197 rendererMaterial.normalScale = 1.0f;
1198 rendererMaterial.exposure = 1.0f;
1199 rendererMaterial.baseColorTextureMaskedTransparency = 0;
1200 rendererMaterial.baseColorTextureMaskedTransparencyThreshold = 0.0f;
1202 rendererMaterial.baseColorFactor = pbrMaterialProperties->getBaseColorFactor().getArray();
1203 rendererMaterial.metallicFactor = pbrMaterialProperties->getMetallicFactor();
1204 rendererMaterial.roughnessFactor = pbrMaterialProperties->getRoughnessFactor();
1205 rendererMaterial.normalScale = pbrMaterialProperties->getNormalScale();
1206 rendererMaterial.exposure = pbrMaterialProperties->getExposure();
1207 rendererMaterial.baseColorTextureMaskedTransparency = pbrMaterialProperties->hasBaseColorTextureMaskedTransparency() ==
true?1:0;
1208 rendererMaterial.baseColorTextureMaskedTransparencyThreshold = pbrMaterialProperties->getBaseColorTextureMaskedTransparencyThreshold();
1230 auto diffuseTexture = specularMaterialProperties->getDiffuseTexture();
1232 rendererMaterial.diffuseTextureMaskedTransparencyThreshold = specularMaterialProperties->getDiffuseTextureMaskedTransparencyThreshold();
1233 rendererMaterial.textureAtlasSize = specularMaterialProperties->getTextureAtlasSize();
1234 rendererMaterial.textureAtlasPixelDimension = { 0.0f, 0.0f };
1235 if (rendererMaterial.textureAtlasSize > 1 && diffuseTexture !=
nullptr) {
1236 rendererMaterial.textureAtlasPixelDimension[0] = 1.0f / diffuseTexture->getTextureWidth();
1237 rendererMaterial.textureAtlasPixelDimension[1] = 1.0f / diffuseTexture->getTextureHeight();
1241 specularMaterialProperties->hasDiffuseTextureMaskedTransparency() ==
true) {
1242 auto diffuseTextureId =
1247 materialKey.append((
const char*)&diffuseTextureId,
sizeof(diffuseTextureId));
1248 if (updateOnly ==
false || currentMaterialKey.empty() ==
true) {
1256 if (pbrMaterialProperties ==
nullptr) {
1258 rendererMaterial.baseColorTextureMaskedTransparencyThreshold = 0.0f;
1260 rendererMaterial.baseColorTextureMaskedTransparency = pbrMaterialProperties->hasBaseColorTextureMaskedTransparency() ==
true?1:0;
1261 rendererMaterial.baseColorTextureMaskedTransparencyThreshold = pbrMaterialProperties->getBaseColorTextureMaskedTransparencyThreshold();
1308 if (pses.size() == 0)
return;
1311 struct PseParameters {
1312 const Color4* effectColorAdd;
1313 const Color4* effectColorMul;
1314 int32_t textureIndex;
1316 int32_t textureHorizontalSprites;
1317 int32_t textureVerticalSprites;
1320 unordered_map<void*, PseParameters> rendererPseParameters;
1339 unordered_map<int, int> textureIndices;
1343 for (
auto entity: pses) {
1344 if (entity->getRenderPass() != renderPass)
continue;
1346 if (ppse !=
nullptr) {
1347 auto textureIndexIt = textureIndices.find(ppse->getTextureId());
1348 int textureIndex = textureIndexIt == textureIndices.end()?-1:textureIndexIt->second;
1349 if (textureIndex == -1) textureIndices[ppse->getTextureId()] = textureIndex = textureIndices.size();
1350 rendererPseParameters[ppse] = {
1352 .effectColorMul = &ppse->getEffectColorMul(),
1353 .textureIndex = textureIndex,
1354 .textureId = ppse->getTextureId(),
1355 .textureHorizontalSprites = ppse->getTextureHorizontalSprites(),
1356 .textureVerticalSprites = ppse->getTextureVerticalSprites(),
1357 .pointSize = ppse->getPointSize()
1362 if (fpse !=
nullptr) {
1363 auto textureIndexIt = textureIndices.find(fpse->getTextureId());
1364 int textureIndex = textureIndexIt == textureIndices.end()?-1:textureIndexIt->second;
1365 if (textureIndex == -1) textureIndices[fpse->getTextureId()] = textureIndex = textureIndices.size();
1366 rendererPseParameters[fpse] = {
1368 .effectColorMul = &fpse->getEffectColorMul(),
1369 .textureIndex = textureIndex,
1370 .textureId = fpse->getTextureId(),
1371 .textureHorizontalSprites = fpse->getTextureHorizontalSprites(),
1372 .textureVerticalSprites = fpse->getTextureVerticalSprites(),
1373 .pointSize = fpse->getPointSize()
1383 auto pseParameters = &rendererPseParameters.find(points[0]->particleSystem)->second;
1384 auto currentPpse =
static_cast<void*
>(points[0]->particleSystem);
1387 auto point = points[i];
1388 if (point->particleSystem != (
void*)currentPpse) {
1389 pseParameters = &rendererPseParameters.find(point->particleSystem)->second;
1390 currentPpse = point->particleSystem;
1394 pseParameters->textureIndex,
1395 pseParameters->pointSize,
1396 *pseParameters->effectColorMul,
1397 *pseParameters->effectColorAdd,
1398 pseParameters->textureHorizontalSprites,
1399 pseParameters->textureVerticalSprites
1404 auto point = points[i];
1405 if (point->particleSystem != (
void*)currentPpse) {
1406 pseParameters = &rendererPseParameters.find(point->particleSystem)->second;
1407 currentPpse = point->particleSystem;
1411 pseParameters->textureIndex,
1412 pseParameters->pointSize,
1413 *pseParameters->effectColorMul,
1414 *pseParameters->effectColorAdd,
1415 pseParameters->textureHorizontalSprites,
1416 pseParameters->textureVerticalSprites
1422 array<int32_t, 16> textureIds;
1424 for (
auto& textureIt: textureIndices) {
1425 if (textureIt.second >= 16)
continue;
1426 textureIds[textureIt.second] = textureIt.first;
1448 if (objects.size() == 0)
return;
1464 for (
auto object: objects) {
1465 if (object->getRenderPass() != renderPass)
continue;
static STATIC_DLL_IMPEXT Queue< EngineThreadQueueElement > * engineThreadsQueue
static STATIC_DLL_IMPEXT vector< EngineThread * > engineThreads
array< Light, LIGHTS_MAX > lights
ShadowMapping * getShadowMapping()
static LinesShader * getLinesShader()
static STATIC_DLL_IMPEXT EngineThreadQueueElementPool engineThreadQueueElementPool
static int getThreadCount()
Entity * getEntity(const string &id)
Returns a entity by given id.
static Engine * getInstance()
Returns engine instance.
static ParticlesShader * getParticlesShader()
static VBOManager * getVBOManager()
static constexpr int ENGINETHREADSQUEUE_RENDER_DISPATCH_COUNT
Entity hierarchy to be used with engine class.
const string & getShaderParametersHash() const
@ ENTITYTYPE_ENVIRONMENTMAPPING
@ ENTITYTYPE_ENTITYHIERARCHY
Environment mapping entity.
int32_t getCubeMapTextureId()
Fog particle system entity to be used with engine class.
Object 3D to be used with engine class.
Object 3D to be used with engine class.
Point particle system entity to be used with engine class.
Color 4 base definition class.
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
Node faces entity A node can have multiple entities containing faces and a applied material.
bool isTextureCoordinatesAvailable() const
const Material * getMaterial() const
Representation of a 3d model.
const vector< TextureCoordinate > & getTextureCoordinates() const
const vector< FacesEntity > & getFacesEntities() const
Represents specular material properties.
Represents specular material properties.
Class representing texture UV coordinates data.
Interface to lighting shader program.
void setParameters(int contextIdx, int32_t textureId, float lineWidth)
Set parameters.
VBOManager_VBOManaged * addVBO(const string &vboId, int32_t ids, bool useGPUMemory, bool shared, bool &created)
Adds a VBO to manager or retrieve VBO if existing.
void removeVBO(const string &vboId)
Removes a VBO from manager.
Particles shader program.
void setParameters(int contextIdx, const array< int32_t, 16 > &textureIds)
Set parameters.
virtual void setTextureUnit(int contextIdx, int32_t textureUnit)=0
Sets up texture unit.
virtual void bindCubeMapTexture(int contextIdx, int32_t textureId)=0
Binds a cube map texture with given id or unbinds when using ID_NONE.
virtual void onUpdateProjectionMatrix(int contextIdx)=0
Update projection matrix event.
Renderer_SpecularMaterial & getSpecularMaterial(int contextIdx)
Get specular material.
virtual void bindEffectColorMulsBufferObject(int contextIdx, int32_t bufferObjectId, int32_t divisor)=0
Bind effect color muls buffer object.
array< float, 4 > & getEffectColorMul(int contextIdx)
Get effect color mul.
virtual void drawInstancedIndexedTrianglesFromBufferObjects(int contextIdx, int32_t triangles, int32_t trianglesOffset, int32_t instances)=0
Draw instanced indexed triangles from buffer objects.
virtual void enableCulling(int contextIdx)=0
Enable culling.
virtual void drawLinesFromBufferObjects(int contextIdx, int32_t points, int32_t pointsOffset)=0
Draw lines from buffer objects.
virtual void onUpdateCameraMatrix(int contextIdx)=0
Update camera matrix event.
virtual bool isSupportingMultithreadedRendering()=0
virtual void bindTangentsBufferObject(int contextIdx, int32_t bufferObjectId)=0
Bind tangents buffer object.
int32_t LIGHTING_SPECULAR
virtual void onUpdateShader(int contextIdx)=0
On update shader.
virtual bool isSupportingIntegerProgramAttributes()=0
virtual void unbindBufferObjects(int contextIdx)=0
Unbind buffer objects.
virtual void enableBlending()=0
Enables blending.
virtual void disableBlending()=0
Disables blending.
virtual void onUpdateTextureMatrix(int contextIdx)=0
Update texture matrix for active texture unit event.
virtual void bindTexture(int contextIdx, int32_t textureId)=0
Binds a texture with given id or unbinds when using ID_NONE.
Matrix4x4 & getCameraMatrix()
virtual void bindModelMatricesBufferObject(int contextIdx, int32_t bufferObjectId)=0
Bind model matrices buffer object.
const string & getShader(int contextIdx)
Get shader.
virtual void bindIndicesBufferObject(int contextIdx, int32_t bufferObjectId)=0
Bind indices buffer object.
virtual void setFrontFace(int contextIdx, int32_t frontFace)=0
Set up clock wise or counter clock wise faces as front face.
virtual bool isInstancedRenderingAvailable()=0
Checks if instanced rendering is available.
void setShader(int contextIdx, const string &id)
Set shader.
Renderer_PBRMaterial & getPBRMaterial(int contextIdx)
Get PBR material.
virtual void onUpdateModelViewMatrix(int contextIdx)=0
Update model view matrix event.
virtual void bindBitangentsBufferObject(int contextIdx, int32_t bufferObjectId)=0
Bind bitangents buffer object.
virtual void onUpdateShaderParameters(int contextIdx)=0
On update shader parameters.
Matrix4x4 & getModelViewMatrix()
const EntityShaderParameters & getShaderParameters(int contextIdx)
Get shader parameters.
int32_t getLighting(int contextIdx)
Get current lighting model.
virtual void disableCulling(int contextIdx)=0
Disable culling.
virtual void drawIndexedTrianglesFromBufferObjects(int contextIdx, int32_t triangles, int32_t trianglesOffset)=0
Draw indexed triangles from buffer objects.
int32_t CONTEXTINDEX_DEFAULT
virtual bool isNormalMappingAvailable()=0
Matrix2D3x3 & getTextureMatrix(int contextIdx)
Get texture matrix.
virtual bool isSpecularMappingAvailable()=0
virtual void uploadBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, FloatBuffer *data)=0
Uploads buffer data to buffer object.
virtual void bindTextureCoordinatesBufferObject(int contextIdx, int32_t bufferObjectId)=0
Bind texture coordinates buffer object.
virtual void onUpdateEffect(int contextIdx)=0
Update material.
void setEnvironmentMappingCubeMapPosition(int contextIdx, array< float, 3 > &position)
Set environment mapping cube map position.
virtual void onUpdateMaterial(int contextIdx)=0
On update material.
virtual void bindColorsBufferObject(int contextIdx, int32_t bufferObjectId)=0
Bind colors buffer object.
array< float, 4 > & getEffectColorAdd(int contextIdx)
Get effect color add.
virtual void bindVerticesBufferObject(int contextIdx, int32_t bufferObjectId)=0
Bind vertices buffer object.
void setShaderParameters(int contextIdx, const EntityShaderParameters ¶meters)
Set shader parameters.
virtual void bindEffectColorAddsBufferObject(int contextIdx, int32_t bufferObjectId, int32_t divisor)=0
Bind effect color adds buffer object.
virtual void bindNormalsBufferObject(int contextIdx, int32_t bufferObjectId)=0
Bind normals buffer object.
virtual void bindOriginsBufferObject(int contextIdx, int32_t bufferObjectId)=0
Bind origins buffer object.
Batch renderer for points.
void render(int contextIdx)
Render.
void addPoint(const TransparentRenderPoint *point, int textureIndex, float pointSize, const Color4 &effectColorMul, const Color4 &effectColorAdd, int textureHorizontalSprites, int textureVerticalSprites)
Adds a transparent render point to this transparent render points.
void clear()
Clears this batch vbo renderer.
void addPointNoInteger(const TransparentRenderPoint *point, int textureIndex, float pointSize, const Color4 &effectColorMul, const Color4 &effectColorAdd, int textureHorizontalSprites, int textureVerticalSprites)
Adds a transparent render point to this transparent render points.
Batch renderer for transparent triangles.
Entity renderer transparent render faces node pool.
static constexpr int32_t RENDERTYPE_TEXTUREARRAYS_DIFFUSEMASKEDTRANSPARENCY
static constexpr int32_t INSTANCEDRENDERING_OBJECTS_MAX
void prepareTransparentFaces(const vector< TransparentRenderFace * > &transparentRenderFaces)
Renders transparent faces TODO: guess this should be optimized regarding GL commands skinned mesh is ...
void render(Entity::RenderPass renderPass, const vector< Object3D * > &objects, bool renderTransparentFaces, int32_t renderTypes)
Renders all given objects.
static constexpr int32_t RENDERTYPE_TEXTUREARRAYS
EntityRenderer_TransparentRenderFacesGroupPool * transparentRenderFacesGroupPool
static constexpr int32_t RENDERTYPE_NORMALS
void renderObjectsOfSameTypeInstanced(int threadIdx, const vector< Object3D * > &objects, bool collectTransparentFaces, int32_t renderTypes, TransparentRenderFacesPool *transparentRenderFacesPool)
Renders multiple objects of same type(with same model) using instancing.
BatchRendererTriangles * acquireTrianglesBatchRenderer()
static constexpr int32_t RENDERTYPE_TEXTURES_DIFFUSEMASKEDTRANSPARENCY
TransparentRenderFacesPool * transparentRenderFacesPool
BatchRendererPoints * psePointBatchRenderer
bool checkMaterialChangable(Object3DNode *object3DNode, int32_t facesEntityIdx, int32_t renderTypes)
Checks if a material could change when having multiple objects but same model.
static constexpr int32_t RENDERTYPE_TEXTURES
static constexpr int32_t RENDERTYPE_MATERIALS
static constexpr int32_t RENDERTYPE_SHADOWMAPPING
unordered_map< uint8_t, unordered_map< Model *, vector< Object3D * > > > objectsByShadersAndModels
void releaseTransparentFacesGroups()
Release transparent faces groups.
vector< Object3DRenderContext > contexts
void renderFunction(int threadIdx, Entity::RenderPass renderPass, const vector< Object3D * > &objects, unordered_map< uint8_t, unordered_map< Model *, vector< Object3D * > > > &objectsByShadersAndModels, bool renderTransparentFaces, int renderTypes, TransparentRenderFacesPool *transparentRenderFacesPool)
Render function.
static constexpr int32_t RENDERTYPE_EFFECTCOLORS
void setupMaterial(int contextIdx, Object3DNode *object3DNode, int32_t facesEntityIdx, int32_t renderTypes, bool updateOnly, string &materialKey, const string ¤tMaterialKey=string())
Set ups a material for rendering.
void renderTransparentFacesGroups(int contextIdx)
Render transparent faces groups.
vector< TransparentRenderFace * > nodeTransparentRenderFaces
void clearMaterial(int contextIdx)
Clear material for rendering.
unordered_map< string, TransparentRenderFacesGroup * > transparentRenderFacesGroups
static constexpr int32_t BATCHRENDERER_MAX
void reset()
Resets the object 3d renderer.
~EntityRenderer()
Destructor.
RenderTransparentRenderPointsPool * renderTransparentRenderPointsPool
vector< BatchRendererTriangles * > trianglesBatchRenderers
void renderObjectsOfSameTypeNonInstanced(const vector< Object3D * > &objects, bool collectTransparentFaces, int32_t renderTypes)
Renders multiple objects of same type(with same model) not using instancing.
void renderTransparentFaces()
Renders collected transparent faces.
Object 3D node mesh specifically for rendering.
Object 3D node VBO renderer.
Object 3d node specifically for rendering.
vector< int32_t > pbrMaterialBaseColorTextureIdsByEntities
vector< int32_t > specularMaterialDiffuseTextureIdsByEntities
vector< int32_t > specularMaterialDynamicDiffuseTextureIdsByEntities
static void setupTextures(Renderer *renderer, int contextIdx, Object3DNode *object3DNode, int32_t facesEntityIdx)
Set up textures for given object3d node and faces entity.
vector< int32_t > pbrMaterialMetallicRoughnessTextureIdsByEntities
vector< int32_t > specularMaterialNormalTextureIdsByEntities
vector< int32_t > specularMaterialSpecularTextureIdsByEntities
vector< int32_t > pbrMaterialNormalTextureIdsByEntities
static constexpr int32_t TEXTUREID_NONE
Buffers used to transfer data between main memory to graphics board memory.
Render transparent render points pool.
void sort()
Sort transparent render points.
const vector< TransparentRenderPoint * > & getTransparentRenderPoints()
void merge(TransparentRenderPointsPool *pool2, const Matrix4x4 &cameraMatrix)
Merge another pool into this pool.
const int32_t getTransparentRenderPointsCount()
Transparent render faces group.
Object3DNode * object3DNode
void addVertex(const Vector3 &vertex, const Vector3 &normal, const Vector2 &textureCoordinate)
Adds a vertex to this transparent render faces group.
static const string createKey(Model *model, Object3DNode *object3DNode, int32_t facesEntityIdx, const Color4 &effectColorAdd, const Color4 &effectColorMul, const Material *material, bool textureCoordinates, const string &shader)
Creates a key for given transparent render faces group attributes.
void set(EntityRenderer *object3DRenderer, Model *model, Object3DNode *object3DNode, int32_t facesEntityIdx, const Color4 &effectColorAdd, const Color4 &effectColorMul, const Material *material, bool textureCoordinates, const string &shader)
Set transparent render faces group.
Transparent render faces pool.
void createTransparentRenderFaces(Matrix4x4 &modelViewMatrix, Object3DNode *object3DNode, int32_t facesEntityIdx, int32_t faceIdx)
Creates an array of transparent render faces from.
void merge(TransparentRenderFacesPool *srcTransparentRenderFacesPool)
Merges given transparent render faces pool into this pool.
vector< TransparentRenderFace * > & getTransparentRenderFaces()
Transparent render points pool.
Matrix2D3x3 & identity()
Setup identity matrix.
Matrix2D3x3 & set(float r0c0, float r1c0, float r2c0, float r0c1, float r1c1, float r2c1, float r0c2, float r1c2, float r2c2)
Set up matrix by values.
Simple class to determine if a transform is negative.
bool isNegative(Matrix4x4 &matrix)
Check if matrix is negative.
Matrix4x4 & identity()
Setup identity matrix.
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.
Vector3 multiplyNoTranslation(const Vector3 &v) const
Multiplies a vector3 with this matrix ignoring translation.
Vector3 multiply(const Vector3 &v) const
Multiplies a vector3 with this matrix into destination vector.
array< float, 16 > & getArray() const
Returns array data.
bool equals(const Vector3 &v) const
Compares this vector with given vector.
Vector3 & set(float x, float y, float z)
Set up vector.
float computeLengthSquared() const
Vector3 & sub(const Vector3 &v)
Subtracts a vector.
array< float, 3 > & getArray() const
virtual int32_t getPosition()
FloatBuffer * put(float value)
Put a float value into float buffer.
T allocate()
Allocate a new element from pool.
void release(T element)
Release element in pool for being reused.
void reset()
Reset this pool.
Lighting shader constants.
Particle emitter interface.
array< float, 4 > baseColorFactor
int baseColorTextureMaskedTransparency
array< float, 4 > ambient
int diffuseTextureMaskedTransparency
Entity renderer parameters.
Transparent face to be rendered.
static bool compare(TransparentRenderFace *face1, TransparentRenderFace *face2)
Compare.
Transparent point to be rendered.