TDME2 1.9.121
ShadowMap.cpp
Go to the documentation of this file.
2
3#include <vector>
4
5#include <tdme/tdme.h>
11#include <tdme/engine/Camera.h>
12#include <tdme/engine/Engine.h>
13#include <tdme/engine/Entity.h>
17#include <tdme/engine/Light.h>
25#include <tdme/math/Math.h>
26#include <tdme/math/Matrix4x4.h>
27#include <tdme/math/Vector3.h>
28
29using std::vector;
30
53
54ShadowMap::ShadowMap(ShadowMapping* shadowMapping, int32_t width, int32_t height)
55{
56 this->shadowMapping = shadowMapping;
60 if (shadowMapping->renderer->getRendererType() == Renderer::RENDERERTYPE_VULKAN) {
62 0.5f, 0.0f, 0.0f, 0.0f,
63 0.0f, -0.5f, 0.0f, 0.0f,
64 0.0f, 0.0f, 0.5f, 0.0f,
65 0.5f, 0.5f, 0.5f, 1.0f
66 );
67 } else {
69 0.5f, 0.0f, 0.0f, 0.0f,
70 0.0f, 0.5f, 0.0f, 0.0f,
71 0.0f, 0.0f, 0.5f, 0.0f,
72 0.5f, 0.5f, 0.5f, 1.0f
73 );
74 }
76}
77
79 delete lightCamera;
80 delete frameBuffer;
81}
82
84{
86}
87
88void ShadowMap::reshape(int32_t width, int32_t height)
89{
90}
91
93{
95}
96
98{
100}
101
103{
104 return lightCamera;
105}
106
108{
109 // use default context
110 auto contextIdx = shadowMapping->renderer->CONTEXTINDEX_DEFAULT;
111
112 //
113 auto camera = shadowMapping->engine->getCamera();
114
115 //
116 auto lightDirection = light->getSpotDirection().clone().normalize();
117
118 // directional lights
119 if (light->isDirectional() == true) {
120 // try to determine light position
121 // left
122 auto left = camera->getModelViewProjectionInvertedMatrix().multiply(
123 Vector4(
124 (2.0f * 0.0f) - 1.0f,
125 1.0f - (2.0f * 0.5f),
126 2.0f * 0.997f - 1.0f,
127 1.0f
128 )
129 );
130 left.scale(1.0f / left.getW());
131
132 // right
133 auto right = camera->getModelViewProjectionInvertedMatrix().multiply(
134 Vector4(
135 (2.0f * 1.0f) - 1.0f,
136 1.0f - (2.0f * 0.5f),
137 2.0f * 0.997f - 1.0f,
138 1.0f
139 )
140 );
141 right.scale(1.0f / right.getW());
142
143 // center
144 auto center4 = camera->getModelViewProjectionInvertedMatrix().multiply(
145 Vector4(
146 (2.0f * 0.5f) - 1.0f,
147 1.0f - (2.0f * 1.0f),
148 2.0f * 0.5f - 1.0f,
149 1.0f
150 )
151 );
152 center4.scale(1.0f / center4.getW());
153
154 // so we get some contraints for the shadow map camera, TODO: improve me
155 Vector3 center(Vector3(center4.getX(), center4.getY(), center4.getZ()));
156 auto width = Vector3(right.getX(), right.getY(), right.getZ()).sub(Vector3(left.getX(), left.getY(), left.getZ())).computeLength() * shadowMapping->engine->getShadowMapLightEyeDistanceScale();
157
158 // light camera
159 // compute camera from view of light
160 auto lightLookFrom = center.clone().sub(lightDirection.clone().scale(width * 0.5f));
161
162 // set up light camera from view of light
163 Vector3 lightCameraUpVector;
164 Vector3 lightCameraSideVector;
167 lightCamera->setZNear(camera->getZNear());
168 lightCamera->setZFar(250.0f);
169 lightCamera->setLookFrom(lightLookFrom);
170 } else {
171 auto lightPosition = light->getPosition();
172 lightPosition.scale(1.0f / lightPosition.getW());
173 auto lightPosition3 = Vector3(lightPosition.getX(), lightPosition.getY(), lightPosition.getZ());
175 lightCamera->setZNear(camera->getZNear());
176 lightCamera->setZFar(150.0f);
177 lightCamera->setLookFrom(lightPosition3);
178 }
179
180 //
181 lightCamera->setForwardVector(lightDirection);
182 lightCamera->setSideVector(Vector3(1.0f, 0.0f, 0.0f));
183 // TODO: fix cross product NaN if side vector == forward vector
184 auto lightCameraUpVector = Vector3::computeCrossProduct(lightCamera->getForwardVector(), lightCamera->getSideVector());
185 lightCamera->setUpVector(lightCameraUpVector);
186 auto lightCameraSideVector = Vector3::computeCrossProduct(lightCamera->getForwardVector(), lightCamera->getUpVector());
187 lightCamera->setSideVector(lightCameraSideVector);
188 lightCamera->setUpVector(lightCameraUpVector);
191
192 // clear visible objects
193 visibleObjects.clear();
194
195 // determine visible objects and objects that should generate a shadow
197 switch (entity->getEntityType()) {
199 {
200 auto org = static_cast<Object3DRenderGroup*>(entity);
201 auto orgEntity = org->getEntity();
202 if (orgEntity != nullptr) {
203 if (orgEntity->isContributesShadows() == false) continue;
204 switch(orgEntity->getEntityType()) {
206 {
207 auto object = static_cast<Object3D*>(orgEntity);
208 visibleObjects.push_back(object);
209 }
210 break;
212 {
213 auto lodObject = static_cast<LODObject3D*>(orgEntity);
214 if (lodObject->isContributesShadows() == false) continue;
215 auto object = lodObject->getLODObject();
216 if (object != nullptr) {
217 visibleObjects.push_back(object);
218 }
219 }
220 break;
221 default:
222 break;
223 }
224 }
225 }
226 break;
228 {
229 auto object = static_cast<Object3D*>(entity);
230 if (object->isContributesShadows() == false) continue;
231 visibleObjects.push_back(object);
232 }
233 break;
235 {
236 auto lodObject = static_cast<LODObject3D*>(entity);
237 if (lodObject->isContributesShadows() == false) continue;
238 auto object = lodObject->getLODObject();
239 if (object != nullptr) visibleObjects.push_back(object);
240 }
241 break;
243 {
244 auto object = static_cast<ImposterObject3D*>(entity);
245 if (object->isContributesShadows() == false) continue;
246 visibleObjects.push_back(object->getBillboardObject());
247 }
248 break;
250 {
251 auto lodObjectImposter = static_cast<LODObject3DImposter*>(entity);
252 if (lodObjectImposter->isContributesShadows() == false) continue;
253 auto object = lodObjectImposter->getLODObject();
254 if (object != nullptr) visibleObjects.push_back(object);
255 }
256 break;
258 {
259 auto opse = static_cast<ObjectParticleSystem*>(entity);
260 if (opse->isContributesShadows() == false) continue;
261 for (auto object: opse->getEnabledObjects()) visibleObjects.push_back(object);
262 }
263 break;
265 {
266 auto psg = static_cast<ParticleSystemGroup*>(entity);
267 for (auto ps: psg->getParticleSystems()) {
268 if (ps->getEntityType() != Entity::ENTITYTYPE_OBJECTPARTICLESYSTEM) continue;
269 auto opse = static_cast<ObjectParticleSystem*>(ps);
270 if (opse->isContributesShadows() == false) continue;
271 for (auto object: opse->getEnabledObjects()) visibleObjects.push_back(object);
272 }
273 }
274 break;
276 {
277 auto eh = static_cast<EntityHierarchy*>(entity);
278 if (eh->isContributesShadows() == false) continue;
279 for (auto entity: eh->getEntities()) {
280 if (entity->getEntityType() != Entity::ENTITYTYPE_OBJECT3D) continue;
281 auto object = static_cast<Object3D*>(entity);
282 if (object->isEnabled() == false) continue;
283 visibleObjects.push_back(object);
284 }
285 }
286 break;
287 default:
288 break;
289 }
290 }
291
292 // bind frame buffer
294 // clear depth buffer
296 // generate shadow map texture matrix
298 // only draw opaque face entities as shadows will not be produced from transparent objects
299 for (auto i = 0; i < Entity::RENDERPASS_MAX; i++) {
300 auto renderPass = static_cast<Entity::RenderPass>(Math::pow(2, i));
302 renderPass,
304 false,
305 EntityRenderer::RENDERTYPE_TEXTUREARRAYS_DIFFUSEMASKEDTRANSPARENCY |
306 EntityRenderer::RENDERTYPE_TEXTURES_DIFFUSEMASKEDTRANSPARENCY
307 );
308 }
309}
310
312{
313 // matrices
314 auto modelViewMatrix = shadowMapping->renderer->getModelViewMatrix();
315 auto projectionMatrix = shadowMapping->renderer->getProjectionMatrix();
316 // compute shadow texture matrix
317 depthBiasMVPMatrix.set(modelViewMatrix).multiply(projectionMatrix).multiply(biasMatrix);
318}
319
321{
323}
void setFrustumMode(FrustumMode frustumMode)
Set frustum mode.
Definition: Camera.h:101
const Vector3 & getForwardVector() const
Definition: Camera.h:183
const Vector3 & getSideVector() const
Definition: Camera.h:198
void setSideVector(const Vector3 &sideVector)
Set side vector.
Definition: Camera.h:206
void update(int contextIdx, int32_t width, int32_t height)
Sets up camera while resizing the view port.
Definition: Camera.cpp:154
Frustum * getFrustum()
Definition: Camera.h:271
void setUpVector(const Vector3 &upVector)
Set up vector.
Definition: Camera.h:176
void setCameraMode(CameraMode cameraMode)
Set camera mode.
Definition: Camera.h:86
void setLookFrom(const Vector3 &lookFrom)
Set look from.
Definition: Camera.h:221
void setOrthographicFrustumScale(float orthographicFrustumScale)
Set orthographic frustum scale.
Definition: Camera.h:116
const Vector3 & getUpVector() const
Definition: Camera.h:168
void setForwardVector(const Vector3 &forwardVector)
Set forward vector.
Definition: Camera.h:191
void setZNear(float zNear)
Set z near.
Definition: Camera.h:146
@ FRUSTUMMODE_ORTHOGRAPHIC
Definition: Camera.h:24
void setZFar(float zFar)
Set z far.
Definition: Camera.h:161
Engine main class.
Definition: Engine.h:122
Partition * getPartition()
Definition: Engine.h:914
float getShadowMapLightEyeDistanceScale()
Definition: Engine.h:624
Camera * getCamera()
Definition: Engine.h:907
Entity hierarchy to be used with engine class.
TDME engine entity.
Definition: Entity.h:31
static constexpr int RENDERPASS_MAX
Definition: Entity.h:56
@ ENTITYTYPE_LODOBJECT3DIMPOSTER
Definition: Entity.h:66
@ ENTITYTYPE_OBJECT3DRENDERGROUP
Definition: Entity.h:68
@ ENTITYTYPE_IMPOSTEROBJECT3D
Definition: Entity.h:63
@ ENTITYTYPE_OBJECTPARTICLESYSTEM
Definition: Entity.h:70
@ ENTITYTYPE_PARTICLESYSTEMGROUP
Definition: Entity.h:71
@ ENTITYTYPE_ENTITYHIERARCHY
Definition: Entity.h:61
Frame buffer class.
Definition: FrameBuffer.h:21
void initialize()
Initialize the frame buffer.
Definition: FrameBuffer.cpp:33
void enableFrameBuffer()
Enables this frame buffer to be rendered.
Definition: FrameBuffer.cpp:78
static constexpr int32_t FRAMEBUFFER_DEPTHBUFFER
Definition: FrameBuffer.h:24
void dispose()
Disposes this frame buffer.
Definition: FrameBuffer.cpp:67
void bindDepthBufferTexture(int contextIdx)
Bind depth texture.
Definition: FrameBuffer.cpp:92
void update()
Setups frustum, should be called if frustum did change.
Definition: Frustum.cpp:30
Imposter object 3d to be used with engine class.
LOD object 3D + imposter to be used with engine class.
LOD object 3D to be used with engine class.
Definition: LODObject3D.h:47
Light representation.
Definition: Light.h:32
bool isDirectional() const
Returns if light is directional light like sun, moon lights.
Definition: Light.h:308
const Vector3 & getSpotDirection() const
Definition: Light.h:149
const Vector4 & getPosition() const
Definition: Light.h:134
Object 3D render group for static objects that might be animated by shaders.
Object 3D to be used with engine class.
Definition: Object3D.h:60
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...
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
Definition: BoundingBox.h:25
Line segment helper functions.
Definition: LineSegment.h:17
virtual void clear(int32_t mask)=0
Clear render buffer with given mask.
void render(Entity::RenderPass renderPass, const vector< Object3D * > &objects, bool renderTransparentFaces, int32_t renderTypes)
Renders all given objects.
void reshape(int32_t width, int32_t height)
Reshape frame buffer.
Definition: ShadowMap.cpp:88
void updateDepthBiasMVPMatrix(int contextIdx)
Set up shadow texture matrix computed and stored before.
Definition: ShadowMap.cpp:320
void dispose()
Disposes this shadow map.
Definition: ShadowMap.cpp:92
void computeDepthBiasMVPMatrix()
Computes shadow texture matrix and stores it.
Definition: ShadowMap.cpp:311
void createShadowMap(Light *light)
Create shadow map.
Definition: ShadowMap.cpp:107
void bindDepthBufferTexture(int contextIdx)
Binds frame buffer depth texture.
Definition: ShadowMap.cpp:97
void updateDepthBiasMVPMatrix(int contextIdx, Matrix4x4 &depthBiasMVPMatrix)
Update depth bias mvp matrix with given matrix.
Standard math functions.
Definition: Math.h:21
4x4 3D Matrix class
Definition: Matrix4x4.h:24
Matrix4x4 & identity()
Setup identity matrix.
Definition: Matrix4x4.h:326
Matrix4x4 & set(float r0c0, float r1c0, float r2c0, float r3c0, float r0c1, float r1c1, float r2c1, float r3c1, float r0c2, float r1c2, float r2c2, float r3c2, float r0c3, float r1c3, float r2c3, float r3c3)
Set up matrix by values.
Definition: Matrix4x4.h:95
Vector3 multiply(const Vector3 &v) const
Multiplies a vector3 with this matrix into destination vector.
Definition: Matrix4x4.h:351
3D vector 3 class
Definition: Vector3.h:22
float computeLength() const
Definition: Vector3.h:202
Vector3 & normalize()
Normalize the vector.
Definition: Vector3.h:288
Vector3 clone() const
Clones the vector.
Definition: Vector3.h:372
Vector3 & sub(const Vector3 &v)
Subtracts a vector.
Definition: Vector3.h:325
Vector3 & scale(float scale)
Scale this vector.
Definition: Vector3.h:349
3D vector 4 class
Definition: Vector4.h:19
Vector4 & scale(float scale)
Scale this vector.
Definition: Vector4.h:229
Partition interface.
Definition: Partition.h:19
virtual const vector< Entity * > & getVisibleEntities(Frustum *frustum)=0
Get visible entities.