2008-05-22 11:51:37 +00:00
|
|
|
// Copyright (C) 2002-2008 Nikolaus Gebhardt
|
2007-05-20 18:03:49 +00:00
|
|
|
// This file is part of the "Irrlicht Engine".
|
|
|
|
// For conditions of distribution and use, see copyright notice in irrlicht.h
|
|
|
|
|
|
|
|
#ifndef __S_VIEW_FRUSTUM_H_INCLUDED__
|
|
|
|
#define __S_VIEW_FRUSTUM_H_INCLUDED__
|
|
|
|
|
|
|
|
#include "plane3d.h"
|
|
|
|
#include "vector3d.h"
|
|
|
|
#include "aabbox3d.h"
|
|
|
|
#include "matrix4.h"
|
|
|
|
#include "IVideoDriver.h"
|
|
|
|
|
|
|
|
namespace irr
|
|
|
|
{
|
|
|
|
namespace scene
|
|
|
|
{
|
|
|
|
|
|
|
|
//! Defines the view frustum. That's the space visible by the camera.
|
|
|
|
/** The view frustum is enclosed by 6 planes. These six planes share
|
|
|
|
four points. A bounding box around these four points is also stored in
|
|
|
|
this structure.
|
|
|
|
*/
|
|
|
|
struct SViewFrustum
|
|
|
|
{
|
|
|
|
enum VFPLANES
|
|
|
|
{
|
|
|
|
//! Far plane of the frustum. That is the plane farest away from the eye.
|
|
|
|
VF_FAR_PLANE = 0,
|
|
|
|
//! Near plane of the frustum. That is the plane nearest to the eye.
|
|
|
|
VF_NEAR_PLANE,
|
|
|
|
//! Left plane of the frustum.
|
|
|
|
VF_LEFT_PLANE,
|
|
|
|
//! Right plane of the frustum.
|
|
|
|
VF_RIGHT_PLANE,
|
|
|
|
//! Bottom plane of the frustum.
|
|
|
|
VF_BOTTOM_PLANE,
|
|
|
|
//! Top plane of the frustum.
|
|
|
|
VF_TOP_PLANE,
|
|
|
|
|
|
|
|
//! Amount of planes enclosing the view frustum. Should be 6.
|
|
|
|
VF_PLANE_COUNT
|
|
|
|
};
|
|
|
|
|
|
|
|
//! Hold a copy of important transform matrices
|
|
|
|
enum E_TRANSFORMATION_STATE_3
|
|
|
|
{
|
|
|
|
ETS_VIEW_PROJECTION_3 = video::ETS_PROJECTION + 1,
|
|
|
|
ETS_VIEW_MODEL_INVERSE_3,
|
|
|
|
ETS_CURRENT_3,
|
|
|
|
ETS_COUNT_3
|
|
|
|
};
|
|
|
|
|
|
|
|
//! Default Constructor
|
2007-09-17 16:09:50 +00:00
|
|
|
SViewFrustum() {}
|
2007-05-20 18:03:49 +00:00
|
|
|
|
|
|
|
//! Copy Constructor
|
|
|
|
SViewFrustum(const SViewFrustum& other);
|
|
|
|
|
2008-05-22 11:51:37 +00:00
|
|
|
//! This constructor creates a view frustum based on a projection and/or view matrix.
|
2007-05-20 18:03:49 +00:00
|
|
|
SViewFrustum(const core::matrix4& mat);
|
|
|
|
|
2008-05-22 11:51:37 +00:00
|
|
|
//! This constructor creates a view frustum based on a projection and/or view matrix.
|
2007-09-18 14:01:28 +00:00
|
|
|
inline void setFrom(const core::matrix4& mat);
|
2007-05-20 18:03:49 +00:00
|
|
|
|
|
|
|
//! transforms the frustum by the matrix
|
2008-05-22 11:51:37 +00:00
|
|
|
/** \param mat: Matrix by which the view frustum is transformed.*/
|
2007-05-20 18:03:49 +00:00
|
|
|
void transform(const core::matrix4& mat);
|
|
|
|
|
2008-05-22 11:51:37 +00:00
|
|
|
//! returns the point which is on the far left upper corner inside the the view frustum.
|
2007-05-20 18:03:49 +00:00
|
|
|
core::vector3df getFarLeftUp() const;
|
|
|
|
|
2008-05-22 11:51:37 +00:00
|
|
|
//! returns the point which is on the far left bottom corner inside the the view frustum.
|
2007-05-20 18:03:49 +00:00
|
|
|
core::vector3df getFarLeftDown() const;
|
|
|
|
|
2008-05-22 11:51:37 +00:00
|
|
|
//! returns the point which is on the far right top corner inside the the view frustum.
|
2007-05-20 18:03:49 +00:00
|
|
|
core::vector3df getFarRightUp() const;
|
|
|
|
|
2008-05-22 11:51:37 +00:00
|
|
|
//! returns the point which is on the far right bottom corner inside the the view frustum.
|
2007-05-20 18:03:49 +00:00
|
|
|
core::vector3df getFarRightDown() const;
|
|
|
|
|
|
|
|
//! returns a bounding box enclosing the whole view frustum
|
|
|
|
const core::aabbox3d<f32> &getBoundingBox() const;
|
|
|
|
|
|
|
|
//! recalculates the bounding box member based on the planes
|
|
|
|
inline void recalculateBoundingBox();
|
|
|
|
|
2007-12-02 23:57:20 +00:00
|
|
|
//! update the given state's matrix
|
2007-05-20 18:03:49 +00:00
|
|
|
void setTransformState( video::E_TRANSFORMATION_STATE state);
|
|
|
|
|
|
|
|
//! the position of the camera
|
|
|
|
core::vector3df cameraPosition;
|
|
|
|
|
|
|
|
//! all planes enclosing the view frustum.
|
|
|
|
core::plane3d<f32> planes[VF_PLANE_COUNT];
|
|
|
|
|
|
|
|
//! bounding box around the view frustum
|
|
|
|
core::aabbox3d<f32> boundingBox;
|
|
|
|
|
|
|
|
//! Hold a copy of important transform matrices
|
|
|
|
core::matrix4 Matrices[ETS_COUNT_3];
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
inline SViewFrustum::SViewFrustum(const SViewFrustum& other)
|
|
|
|
{
|
|
|
|
cameraPosition=other.cameraPosition;
|
|
|
|
boundingBox=other.boundingBox;
|
|
|
|
|
|
|
|
u32 i;
|
|
|
|
for (i=0; i<VF_PLANE_COUNT; ++i)
|
|
|
|
planes[i]=other.planes[i];
|
|
|
|
|
|
|
|
for (i=0; i<VF_PLANE_COUNT; ++i)
|
|
|
|
Matrices[i]=other.Matrices[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
inline SViewFrustum::SViewFrustum(const core::matrix4& mat)
|
|
|
|
{
|
|
|
|
setFrom ( mat );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline void SViewFrustum::transform(const core::matrix4& mat)
|
|
|
|
{
|
|
|
|
for (u32 i=0; i<VF_PLANE_COUNT; ++i)
|
|
|
|
mat.transformPlane(planes[i]);
|
|
|
|
|
|
|
|
mat.transformVect(cameraPosition);
|
|
|
|
recalculateBoundingBox();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline core::vector3df SViewFrustum::getFarLeftUp() const
|
|
|
|
{
|
|
|
|
core::vector3df p;
|
|
|
|
planes[scene::SViewFrustum::VF_FAR_PLANE].getIntersectionWithPlanes(
|
|
|
|
planes[scene::SViewFrustum::VF_TOP_PLANE],
|
|
|
|
planes[scene::SViewFrustum::VF_LEFT_PLANE], p);
|
|
|
|
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline core::vector3df SViewFrustum::getFarLeftDown() const
|
|
|
|
{
|
|
|
|
core::vector3df p;
|
|
|
|
planes[scene::SViewFrustum::VF_FAR_PLANE].getIntersectionWithPlanes(
|
|
|
|
planes[scene::SViewFrustum::VF_BOTTOM_PLANE],
|
|
|
|
planes[scene::SViewFrustum::VF_LEFT_PLANE], p);
|
|
|
|
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline core::vector3df SViewFrustum::getFarRightUp() const
|
|
|
|
{
|
|
|
|
core::vector3df p;
|
|
|
|
planes[scene::SViewFrustum::VF_FAR_PLANE].getIntersectionWithPlanes(
|
|
|
|
planes[scene::SViewFrustum::VF_TOP_PLANE],
|
|
|
|
planes[scene::SViewFrustum::VF_RIGHT_PLANE], p);
|
|
|
|
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline core::vector3df SViewFrustum::getFarRightDown() const
|
|
|
|
{
|
|
|
|
core::vector3df p;
|
|
|
|
planes[scene::SViewFrustum::VF_FAR_PLANE].getIntersectionWithPlanes(
|
|
|
|
planes[scene::SViewFrustum::VF_BOTTOM_PLANE],
|
|
|
|
planes[scene::SViewFrustum::VF_RIGHT_PLANE], p);
|
|
|
|
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline const core::aabbox3d<f32> &SViewFrustum::getBoundingBox() const
|
|
|
|
{
|
|
|
|
return boundingBox;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void SViewFrustum::recalculateBoundingBox()
|
|
|
|
{
|
|
|
|
boundingBox.reset ( cameraPosition );
|
|
|
|
|
|
|
|
boundingBox.addInternalPoint(getFarLeftUp());
|
|
|
|
boundingBox.addInternalPoint(getFarRightUp());
|
|
|
|
boundingBox.addInternalPoint(getFarLeftDown());
|
|
|
|
boundingBox.addInternalPoint(getFarRightDown());
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
//! This constructor creates a view frustum based on a projection
|
|
|
|
//! and/or view matrix.
|
|
|
|
inline void SViewFrustum::setFrom(const core::matrix4& mat)
|
|
|
|
{
|
|
|
|
// left clipping plane
|
|
|
|
planes[SViewFrustum::VF_LEFT_PLANE].Normal.X = -(mat(0,3) + mat(0,0));
|
|
|
|
planes[SViewFrustum::VF_LEFT_PLANE].Normal.Y = -(mat(1,3) + mat(1,0));
|
|
|
|
planes[SViewFrustum::VF_LEFT_PLANE].Normal.Z = -(mat(2,3) + mat(2,0));
|
2008-05-22 11:51:37 +00:00
|
|
|
planes[SViewFrustum::VF_LEFT_PLANE].D = -(mat(3,3) + mat(3,0));
|
2007-05-20 18:03:49 +00:00
|
|
|
|
|
|
|
// right clipping plane
|
|
|
|
planes[SViewFrustum::VF_RIGHT_PLANE].Normal.X = -(mat(0,3) - mat(0,0));
|
|
|
|
planes[SViewFrustum::VF_RIGHT_PLANE].Normal.Y = -(mat(1,3) - mat(1,0));
|
|
|
|
planes[SViewFrustum::VF_RIGHT_PLANE].Normal.Z = -(mat(2,3) - mat(2,0));
|
|
|
|
planes[SViewFrustum::VF_RIGHT_PLANE].D = -(mat(3,3) - mat(3,0));
|
|
|
|
|
|
|
|
// top clipping plane
|
|
|
|
planes[SViewFrustum::VF_TOP_PLANE].Normal.X = -(mat(0,3) - mat(0,1));
|
|
|
|
planes[SViewFrustum::VF_TOP_PLANE].Normal.Y = -(mat(1,3) - mat(1,1));
|
|
|
|
planes[SViewFrustum::VF_TOP_PLANE].Normal.Z = -(mat(2,3) - mat(2,1));
|
|
|
|
planes[SViewFrustum::VF_TOP_PLANE].D = -(mat(3,3) - mat(3,1));
|
|
|
|
|
|
|
|
// bottom clipping plane
|
|
|
|
planes[SViewFrustum::VF_BOTTOM_PLANE].Normal.X = -(mat(0,3) + mat(0,1));
|
|
|
|
planes[SViewFrustum::VF_BOTTOM_PLANE].Normal.Y = -(mat(1,3) + mat(1,1));
|
|
|
|
planes[SViewFrustum::VF_BOTTOM_PLANE].Normal.Z = -(mat(2,3) + mat(2,1));
|
|
|
|
planes[SViewFrustum::VF_BOTTOM_PLANE].D = -(mat(3,3) + mat(3,1));
|
|
|
|
|
|
|
|
// near clipping plane
|
|
|
|
planes[SViewFrustum::VF_NEAR_PLANE].Normal.X = -mat(0,2);
|
|
|
|
planes[SViewFrustum::VF_NEAR_PLANE].Normal.Y = -mat(1,2);
|
|
|
|
planes[SViewFrustum::VF_NEAR_PLANE].Normal.Z = -mat(2,2);
|
|
|
|
planes[SViewFrustum::VF_NEAR_PLANE].D = -mat(3,2);
|
|
|
|
|
|
|
|
// far clipping plane
|
|
|
|
planes[SViewFrustum::VF_FAR_PLANE].Normal.X = -(mat(0,3) - mat(0,2));
|
|
|
|
planes[SViewFrustum::VF_FAR_PLANE].Normal.Y = -(mat(1,3) - mat(1,2));
|
|
|
|
planes[SViewFrustum::VF_FAR_PLANE].Normal.Z = -(mat(2,3) - mat(2,2));
|
|
|
|
planes[SViewFrustum::VF_FAR_PLANE].D = -(mat(3,3) - mat(3,2));
|
|
|
|
// normalize normals
|
|
|
|
|
|
|
|
for (s32 i=0; i<6; ++i)
|
|
|
|
{
|
2008-04-29 11:34:54 +00:00
|
|
|
const f32 len = core::reciprocal_squareroot(
|
|
|
|
planes[i].Normal.getLengthSQ() );
|
2007-05-20 18:03:49 +00:00
|
|
|
planes[i].Normal *= len;
|
|
|
|
planes[i].D *= len;
|
|
|
|
}
|
|
|
|
|
|
|
|
// make bounding box
|
|
|
|
recalculateBoundingBox();
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
inline void SViewFrustum::setFrom(const core::matrix4& mat)
|
|
|
|
{
|
|
|
|
// left clipping plane
|
2008-05-22 11:51:37 +00:00
|
|
|
planes[VF_LEFT_PLANE].Normal.X = mat[3 ] + mat[0];
|
|
|
|
planes[VF_LEFT_PLANE].Normal.Y = mat[7 ] + mat[4];
|
|
|
|
planes[VF_LEFT_PLANE].Normal.Z = mat[11] + mat[8];
|
|
|
|
planes[VF_LEFT_PLANE].D = mat[15] + mat[12];
|
2007-05-20 18:03:49 +00:00
|
|
|
|
|
|
|
// right clipping plane
|
|
|
|
planes[VF_RIGHT_PLANE].Normal.X = mat[3 ] - mat[0];
|
|
|
|
planes[VF_RIGHT_PLANE].Normal.Y = mat[7 ] - mat[4];
|
|
|
|
planes[VF_RIGHT_PLANE].Normal.Z = mat[11] - mat[8];
|
|
|
|
planes[VF_RIGHT_PLANE].D = mat[15] - mat[12];
|
|
|
|
|
|
|
|
// top clipping plane
|
|
|
|
planes[VF_TOP_PLANE].Normal.X = mat[3 ] - mat[1];
|
|
|
|
planes[VF_TOP_PLANE].Normal.Y = mat[7 ] - mat[5];
|
|
|
|
planes[VF_TOP_PLANE].Normal.Z = mat[11] - mat[9];
|
|
|
|
planes[VF_TOP_PLANE].D = mat[15] - mat[13];
|
|
|
|
|
|
|
|
// bottom clipping plane
|
|
|
|
planes[VF_BOTTOM_PLANE].Normal.X = mat[3 ] + mat[1];
|
|
|
|
planes[VF_BOTTOM_PLANE].Normal.Y = mat[7 ] + mat[5];
|
|
|
|
planes[VF_BOTTOM_PLANE].Normal.Z = mat[11] + mat[9];
|
|
|
|
planes[VF_BOTTOM_PLANE].D = mat[15] + mat[13];
|
|
|
|
|
|
|
|
// far clipping plane
|
|
|
|
planes[VF_FAR_PLANE].Normal.X = mat[3 ] - mat[2];
|
|
|
|
planes[VF_FAR_PLANE].Normal.Y = mat[7 ] - mat[6];
|
|
|
|
planes[VF_FAR_PLANE].Normal.Z = mat[11] - mat[10];
|
|
|
|
planes[VF_FAR_PLANE].D = mat[15] - mat[14];
|
|
|
|
|
|
|
|
// near clipping plane
|
|
|
|
planes[VF_NEAR_PLANE].Normal.X = mat[2];
|
|
|
|
planes[VF_NEAR_PLANE].Normal.Y = mat[6];
|
|
|
|
planes[VF_NEAR_PLANE].Normal.Z = mat[10];
|
|
|
|
planes[VF_NEAR_PLANE].D = mat[14];
|
|
|
|
|
|
|
|
// normalize normals
|
|
|
|
u32 i;
|
|
|
|
for ( i=0; i != VF_PLANE_COUNT; ++i)
|
|
|
|
{
|
2008-04-29 11:34:54 +00:00
|
|
|
const f32 len = -core::reciprocal_squareroot(
|
|
|
|
planes[i].Normal.getLengthSQ());
|
2007-05-20 18:03:49 +00:00
|
|
|
planes[i].Normal *= len;
|
|
|
|
planes[i].D *= len;
|
|
|
|
}
|
|
|
|
|
|
|
|
// make bounding box
|
|
|
|
recalculateBoundingBox();
|
|
|
|
}
|
|
|
|
|
2007-12-02 23:57:20 +00:00
|
|
|
inline void SViewFrustum::setTransformState(video::E_TRANSFORMATION_STATE state)
|
2007-05-20 18:03:49 +00:00
|
|
|
{
|
|
|
|
switch ( state )
|
|
|
|
{
|
|
|
|
case video::ETS_VIEW:
|
2008-04-29 11:34:54 +00:00
|
|
|
Matrices[ETS_VIEW_PROJECTION_3].setbyproduct_nocheck(
|
|
|
|
Matrices[video::ETS_PROJECTION],
|
|
|
|
Matrices[video::ETS_VIEW]);
|
2007-12-02 23:57:20 +00:00
|
|
|
Matrices[ETS_VIEW_MODEL_INVERSE_3] = Matrices[video::ETS_VIEW];
|
2007-05-20 18:03:49 +00:00
|
|
|
Matrices[ETS_VIEW_MODEL_INVERSE_3].makeInverse();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case video::ETS_WORLD:
|
2008-04-29 11:34:54 +00:00
|
|
|
Matrices[ETS_CURRENT_3].setbyproduct(
|
|
|
|
Matrices[ETS_VIEW_PROJECTION_3 ],
|
2007-12-02 23:57:20 +00:00
|
|
|
Matrices[video::ETS_WORLD]);
|
2007-05-20 18:03:49 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} // end namespace scene
|
|
|
|
} // end namespace irr
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|