2006-07-18 23:21:48 +08:00
/* -*-c++-*- OpenSceneGraph - Copyright (C) 1998-2006 Robert Osfield
2005-06-14 21:16:58 +08:00
*
* This library is open source and may be redistributed and / or modified under
* the terms of the OpenSceneGraph Public License ( OSGPL ) version 0.0 or
* ( at your option ) any later version . The full license is in LICENSE file
* included with this distribution , and on the openscenegraph . org website .
*
* This library is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* OpenSceneGraph Public License for more details .
*/
2006-11-27 22:52:07 +08:00
# include <osg/Camera>
2005-11-01 18:42:54 +08:00
# include <osg/Notify>
2005-06-14 21:16:58 +08:00
using namespace osg ;
2006-11-27 22:52:07 +08:00
Camera : : Camera ( ) :
2006-09-19 04:54:48 +08:00
_view ( 0 ) ,
2007-02-22 03:06:43 +08:00
_allowEventFocus ( true ) ,
2005-06-14 21:16:58 +08:00
_clearColor ( osg : : Vec4 ( 0.0f , 0.0f , 0.0f , 1.0f ) ) ,
2005-06-15 04:51:35 +08:00
_clearMask ( GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT ) ,
2006-02-21 05:05:23 +08:00
_transformOrder ( PRE_MULTIPLY ) ,
2005-06-15 04:51:35 +08:00
_renderOrder ( POST_RENDER ) ,
2007-05-22 17:32:38 +08:00
_projectionResizePolicy ( HORIZONTAL ) ,
2006-09-04 21:15:08 +08:00
_renderOrderNum ( 0 ) ,
2005-08-17 18:12:49 +08:00
_drawBuffer ( GL_NONE ) ,
_readBuffer ( GL_NONE ) ,
2005-11-01 18:42:54 +08:00
_renderTargetImplementation ( FRAME_BUFFER ) ,
_renderTargetFallback ( FRAME_BUFFER )
2005-06-14 21:16:58 +08:00
{
2005-07-20 00:30:55 +08:00
setStateSet ( new StateSet ) ;
2005-06-14 21:16:58 +08:00
}
2006-11-27 22:52:07 +08:00
Camera : : Camera ( const Camera & camera , const CopyOp & copyop ) :
2005-06-14 21:16:58 +08:00
Transform ( camera , copyop ) ,
2005-06-15 04:51:35 +08:00
CullSettings ( camera ) ,
2006-09-19 04:54:48 +08:00
_view ( camera . _view ) ,
2007-02-22 03:06:43 +08:00
_allowEventFocus ( camera . _allowEventFocus ) ,
2005-06-14 21:16:58 +08:00
_clearColor ( camera . _clearColor ) ,
2007-06-09 18:06:38 +08:00
_displaySettings ( camera . _displaySettings ) ,
2005-06-14 21:16:58 +08:00
_clearMask ( camera . _clearMask ) ,
2005-07-20 00:30:55 +08:00
_colorMask ( camera . _colorMask ) ,
2005-06-14 21:16:58 +08:00
_viewport ( camera . _viewport ) ,
2005-06-15 18:59:10 +08:00
_transformOrder ( camera . _transformOrder ) ,
2007-05-22 17:32:38 +08:00
_projectionResizePolicy ( camera . _projectionResizePolicy ) ,
2005-06-14 21:16:58 +08:00
_projectionMatrix ( camera . _projectionMatrix ) ,
2005-06-15 04:51:35 +08:00
_viewMatrix ( camera . _viewMatrix ) ,
_renderOrder ( camera . _renderOrder ) ,
2006-09-04 21:15:08 +08:00
_renderOrderNum ( camera . _renderOrderNum ) ,
2005-08-17 18:12:49 +08:00
_drawBuffer ( camera . _drawBuffer ) ,
_readBuffer ( camera . _readBuffer ) ,
2005-06-15 04:51:35 +08:00
_renderTargetImplementation ( camera . _renderTargetImplementation ) ,
2005-11-01 18:42:54 +08:00
_renderTargetFallback ( camera . _renderTargetFallback ) ,
2005-07-25 04:31:21 +08:00
_bufferAttachmentMap ( camera . _bufferAttachmentMap ) ,
2007-06-02 03:45:24 +08:00
_preDrawCallback ( camera . _preDrawCallback ) ,
2005-07-25 04:31:21 +08:00
_postDrawCallback ( camera . _postDrawCallback )
2005-07-20 00:30:55 +08:00
{
2007-01-02 02:20:10 +08:00
// need to copy/share graphics context?
2005-06-14 21:16:58 +08:00
}
2006-11-27 22:52:07 +08:00
Camera : : ~ Camera ( )
2005-06-14 21:16:58 +08:00
{
2007-02-05 21:44:16 +08:00
setCameraThread ( 0 ) ;
2007-01-02 02:20:10 +08:00
if ( _graphicsContext . valid ( ) ) _graphicsContext - > removeCamera ( this ) ;
2005-06-14 21:16:58 +08:00
}
2007-01-02 02:20:10 +08:00
void Camera : : setGraphicsContext ( GraphicsContext * context )
{
if ( _graphicsContext = = context ) return ;
if ( _graphicsContext . valid ( ) ) _graphicsContext - > removeCamera ( this ) ;
_graphicsContext = context ;
if ( _graphicsContext . valid ( ) ) _graphicsContext - > addCamera ( this ) ;
}
2006-11-27 22:52:07 +08:00
bool Camera : : isRenderToTextureCamera ( ) const
2005-12-07 19:36:56 +08:00
{
return ( ! _bufferAttachmentMap . empty ( ) ) ;
}
2006-11-27 22:52:07 +08:00
void Camera : : setRenderTargetImplementation ( RenderTargetImplementation impl )
2005-11-01 18:42:54 +08:00
{
_renderTargetImplementation = impl ;
if ( impl < FRAME_BUFFER ) _renderTargetFallback = ( RenderTargetImplementation ) ( impl + 1 ) ;
else _renderTargetFallback = impl ;
}
2006-11-27 22:52:07 +08:00
void Camera : : setRenderTargetImplementation ( RenderTargetImplementation impl , RenderTargetImplementation fallback )
2005-11-01 18:42:54 +08:00
{
2005-11-04 20:08:16 +08:00
if ( impl < fallback | | ( impl = = FRAME_BUFFER & & fallback = = FRAME_BUFFER ) )
2005-11-01 18:42:54 +08:00
{
_renderTargetImplementation = impl ;
_renderTargetFallback = fallback ;
}
2005-11-04 20:08:16 +08:00
else
2005-11-01 18:42:54 +08:00
{
2006-11-27 22:52:07 +08:00
osg : : notify ( osg : : NOTICE ) < < " Warning: Camera::setRenderTargetImplementation(impl,fallback) must have a lower rated fallback than the main target implementation. " < < std : : endl ;
2005-11-01 18:42:54 +08:00
setRenderTargetImplementation ( impl ) ;
}
}
2006-11-27 22:52:07 +08:00
void Camera : : setColorMask ( osg : : ColorMask * colorMask )
2005-07-20 00:30:55 +08:00
{
if ( _colorMask = = colorMask ) return ;
osg : : StateSet * stateset = getOrCreateStateSet ( ) ;
if ( _colorMask . valid ( ) & & stateset )
{
stateset - > removeAttribute ( _colorMask . get ( ) ) ;
}
_colorMask = colorMask ;
if ( _colorMask . valid ( ) & & stateset )
{
stateset - > setAttribute ( _colorMask . get ( ) ) ;
}
}
2006-11-27 22:52:07 +08:00
void Camera : : setColorMask ( bool red , bool green , bool blue , bool alpha )
2005-07-20 00:30:55 +08:00
{
if ( ! _colorMask ) setColorMask ( new osg : : ColorMask ) ;
if ( _colorMask . valid ( ) ) _colorMask - > setMask ( red , green , blue , alpha ) ;
}
2006-11-27 22:52:07 +08:00
void Camera : : setViewport ( osg : : Viewport * viewport )
2005-07-20 00:30:55 +08:00
{
if ( _viewport = = viewport ) return ;
osg : : StateSet * stateset = getOrCreateStateSet ( ) ;
if ( _viewport . valid ( ) & & stateset )
{
stateset - > removeAttribute ( _viewport . get ( ) ) ;
}
_viewport = viewport ;
if ( _viewport . valid ( ) & & stateset )
{
stateset - > setAttribute ( _viewport . get ( ) ) ;
}
}
2006-11-27 22:52:07 +08:00
void Camera : : setViewport ( int x , int y , int width , int height )
2005-07-20 00:30:55 +08:00
{
if ( ! _viewport ) setViewport ( new osg : : Viewport ) ;
if ( _viewport . valid ( ) ) _viewport - > setViewport ( x , y , width , height ) ;
}
2006-11-27 22:52:07 +08:00
Matrixd Camera : : getInverseViewMatrix ( ) const
2005-06-14 21:16:58 +08:00
{
Matrixd inverse ;
inverse . invert ( _viewMatrix ) ;
return inverse ;
}
2006-11-27 22:52:07 +08:00
void Camera : : setProjectionMatrixAsOrtho ( double left , double right ,
2005-06-14 21:16:58 +08:00
double bottom , double top ,
double zNear , double zFar )
{
setProjectionMatrix ( osg : : Matrixd : : ortho ( left , right ,
bottom , top ,
zNear , zFar ) ) ;
}
2006-11-27 22:52:07 +08:00
void Camera : : setProjectionMatrixAsOrtho2D ( double left , double right ,
2005-06-14 21:16:58 +08:00
double bottom , double top )
{
setProjectionMatrix ( osg : : Matrixd : : ortho2D ( left , right ,
bottom , top ) ) ;
}
2006-11-27 22:52:07 +08:00
void Camera : : setProjectionMatrixAsFrustum ( double left , double right ,
2005-06-14 21:16:58 +08:00
double bottom , double top ,
double zNear , double zFar )
{
setProjectionMatrix ( osg : : Matrixd : : frustum ( left , right ,
bottom , top ,
zNear , zFar ) ) ;
}
2006-11-27 22:52:07 +08:00
void Camera : : setProjectionMatrixAsPerspective ( double fovy , double aspectRatio ,
2005-06-14 21:16:58 +08:00
double zNear , double zFar )
{
setProjectionMatrix ( osg : : Matrixd : : perspective ( fovy , aspectRatio ,
zNear , zFar ) ) ;
}
2006-11-27 22:52:07 +08:00
bool Camera : : getProjectionMatrixAsOrtho ( double & left , double & right ,
2005-06-14 21:16:58 +08:00
double & bottom , double & top ,
double & zNear , double & zFar )
{
return _projectionMatrix . getOrtho ( left , right ,
bottom , top ,
zNear , zFar ) ;
}
2006-11-27 22:52:07 +08:00
bool Camera : : getProjectionMatrixAsFrustum ( double & left , double & right ,
2005-06-14 21:16:58 +08:00
double & bottom , double & top ,
double & zNear , double & zFar )
{
return _projectionMatrix . getFrustum ( left , right ,
bottom , top ,
zNear , zFar ) ;
}
2006-11-27 22:52:07 +08:00
bool Camera : : getProjectionMatrixAsPerspective ( double & fovy , double & aspectRatio ,
2005-06-14 21:16:58 +08:00
double & zNear , double & zFar )
{
return _projectionMatrix . getPerspective ( fovy , aspectRatio , zNear , zFar ) ;
}
2006-11-27 22:52:07 +08:00
void Camera : : setViewMatrixAsLookAt ( const Vec3 & eye , const Vec3 & center , const Vec3 & up )
2005-06-14 21:16:58 +08:00
{
setViewMatrix ( osg : : Matrixd : : lookAt ( eye , center , up ) ) ;
}
2006-11-27 22:52:07 +08:00
void Camera : : getViewMatrixAsLookAt ( Vec3 & eye , Vec3 & center , Vec3 & up , float lookDistance )
2005-06-14 21:16:58 +08:00
{
_viewMatrix . getLookAt ( eye , center , up , lookDistance ) ;
}
2005-06-15 04:51:35 +08:00
2006-11-27 22:52:07 +08:00
void Camera : : attach ( BufferComponent buffer , GLenum internalFormat )
2005-06-15 04:51:35 +08:00
{
_bufferAttachmentMap [ buffer ] . _internalFormat = internalFormat ;
}
2006-11-27 22:52:07 +08:00
void Camera : : attach ( BufferComponent buffer , osg : : Texture * texture , unsigned int level , unsigned int face , bool mipMapGeneration )
2005-06-15 04:51:35 +08:00
{
_bufferAttachmentMap [ buffer ] . _texture = texture ;
2005-06-16 04:06:10 +08:00
_bufferAttachmentMap [ buffer ] . _level = level ;
2005-06-15 04:51:35 +08:00
_bufferAttachmentMap [ buffer ] . _face = face ;
2005-07-26 00:12:24 +08:00
_bufferAttachmentMap [ buffer ] . _mipMapGeneration = mipMapGeneration ;
2005-06-15 04:51:35 +08:00
}
2006-11-27 22:52:07 +08:00
void Camera : : attach ( BufferComponent buffer , osg : : Image * image )
2005-06-15 04:51:35 +08:00
{
_bufferAttachmentMap [ buffer ] . _image = image ;
}
2006-11-27 22:52:07 +08:00
void Camera : : detach ( BufferComponent buffer )
2005-06-15 04:51:35 +08:00
{
_bufferAttachmentMap . erase ( buffer ) ;
}
2007-01-04 22:11:51 +08:00
void Camera : : resizeGLObjectBuffers ( unsigned int maxSize )
{
2007-08-02 19:02:47 +08:00
if ( _renderingCache . valid ( ) )
2007-07-28 18:28:40 +08:00
{
2007-08-02 19:02:47 +08:00
const_cast < Camera * > ( this ) - > _renderingCache - > resizeGLObjectBuffers ( maxSize ) ;
2007-07-28 18:28:40 +08:00
}
2007-01-04 22:11:51 +08:00
Transform : : resizeGLObjectBuffers ( maxSize ) ;
}
2006-11-27 22:52:07 +08:00
void Camera : : releaseGLObjects ( osg : : State * state ) const
2005-12-08 18:06:57 +08:00
{
2007-08-02 19:02:47 +08:00
if ( _renderingCache . valid ( ) )
2007-07-28 18:28:40 +08:00
{
2007-08-02 19:02:47 +08:00
const_cast < Camera * > ( this ) - > _renderingCache - > releaseGLObjects ( state ) ;
2007-07-28 18:28:40 +08:00
}
2005-12-08 18:06:57 +08:00
Transform : : releaseGLObjects ( state ) ;
}
2005-06-15 04:51:35 +08:00
2006-11-27 22:52:07 +08:00
bool Camera : : computeLocalToWorldMatrix ( Matrix & matrix , NodeVisitor * ) const
2005-06-14 21:16:58 +08:00
{
if ( _referenceFrame = = RELATIVE_RF )
{
2006-02-21 05:05:23 +08:00
if ( _transformOrder = = PRE_MULTIPLY )
2005-06-15 18:59:10 +08:00
{
matrix . preMult ( _viewMatrix ) ;
}
else
{
matrix . postMult ( _viewMatrix ) ;
}
2005-06-14 21:16:58 +08:00
}
else // absolute
{
matrix = _viewMatrix ;
}
return true ;
}
2006-11-27 22:52:07 +08:00
bool Camera : : computeWorldToLocalMatrix ( Matrix & matrix , NodeVisitor * ) const
2005-06-14 21:16:58 +08:00
{
const Matrixd & inverse = getInverseViewMatrix ( ) ;
if ( _referenceFrame = = RELATIVE_RF )
{
2006-02-21 05:05:23 +08:00
if ( _transformOrder = = PRE_MULTIPLY )
2005-06-15 18:59:10 +08:00
{
// note doing inverse so pre becomes post.
matrix . postMult ( inverse ) ;
}
else
{
// note doing inverse so post becomes pre.
matrix . preMult ( inverse ) ;
}
2005-06-14 21:16:58 +08:00
}
else // absolute
{
matrix = inverse ;
}
return true ;
}
2007-02-05 21:44:16 +08:00
void Camera : : createCameraThread ( )
{
if ( ! _cameraThread )
{
2007-07-12 23:54:45 +08:00
setCameraThread ( new OperationThread ) ;
2007-02-05 21:44:16 +08:00
}
}
2007-07-12 23:54:45 +08:00
void Camera : : setCameraThread ( OperationThread * gt )
2007-02-05 21:44:16 +08:00
{
if ( _cameraThread = = gt ) return ;
if ( _cameraThread . valid ( ) )
{
// need to kill the thread in some way...
_cameraThread - > cancel ( ) ;
_cameraThread - > setParent ( 0 ) ;
}
_cameraThread = gt ;
if ( _cameraThread . valid ( ) )
{
_cameraThread - > setParent ( this ) ;
}
}