2020-09-14 20:17:07 +00:00
# include "stereo.hpp"
2020-09-19 18:55:06 +00:00
# include "stringops.hpp"
2020-09-14 20:17:07 +00:00
# include <osg/io_utils>
# include <osg/ViewportIndexed>
# include <osgUtil/CullVisitor>
# include <osgViewer/Viewer>
# include <iostream>
# include <components/debug/debuglog.hpp>
2020-09-19 18:55:06 +00:00
2020-09-14 20:17:07 +00:00
# include <components/sceneutil/statesetupdater.hpp>
# include <components/sceneutil/visitor.hpp>
2020-09-19 18:55:06 +00:00
# include <components/settings/settings.hpp>
2020-09-14 20:17:07 +00:00
namespace Misc
{
Pose Pose : : operator + ( const Pose & rhs )
{
Pose pose = * this ;
pose . position + = this - > orientation * rhs . position ;
pose . orientation = rhs . orientation * this - > orientation ;
return pose ;
}
const Pose & Pose : : operator + = ( const Pose & rhs )
{
* this = * this + rhs ;
return * this ;
}
Pose Pose : : operator * ( float scalar )
{
Pose pose = * this ;
pose . position * = scalar ;
return pose ;
}
const Pose & Pose : : operator * = ( float scalar )
{
* this = * this * scalar ;
return * this ;
}
Pose Pose : : operator / ( float scalar )
{
Pose pose = * this ;
pose . position / = scalar ;
return pose ;
}
const Pose & Pose : : operator / = ( float scalar )
{
* this = * this / scalar ;
return * this ;
}
bool Pose : : operator = = ( const Pose & rhs ) const
{
return position = = rhs . position & & orientation = = rhs . orientation ;
}
osg : : Matrix Pose : : viewMatrix ( bool useGLConventions )
{
if ( useGLConventions )
{
// When applied as an offset to an existing view matrix,
// that view matrix will already convert points to a camera space
// with opengl conventions. So we need to convert offsets to opengl
// conventions.
float y = position . y ( ) ;
float z = position . z ( ) ;
position . y ( ) = z ;
position . z ( ) = - y ;
y = orientation . y ( ) ;
z = orientation . z ( ) ;
orientation . y ( ) = z ;
orientation . z ( ) = - y ;
osg : : Matrix viewMatrix ;
viewMatrix . setTrans ( - position ) ;
viewMatrix . postMultRotate ( orientation . conj ( ) ) ;
return viewMatrix ;
}
else
{
osg : : Vec3d forward = orientation * osg : : Vec3d ( 0 , 1 , 0 ) ;
osg : : Vec3d up = orientation * osg : : Vec3d ( 0 , 0 , 1 ) ;
osg : : Matrix viewMatrix ;
viewMatrix . makeLookAt ( position , position + forward , up ) ;
return viewMatrix ;
}
}
bool FieldOfView : : operator = = ( const FieldOfView & rhs ) const
{
return angleDown = = rhs . angleDown
& & angleUp = = rhs . angleUp
& & angleLeft = = rhs . angleLeft
& & angleRight = = rhs . angleRight ;
}
// near and far named with an underscore because of windows' headers galaxy brain defines.
osg : : Matrix FieldOfView : : perspectiveMatrix ( float near_ , float far_ )
{
const float tanLeft = tanf ( angleLeft ) ;
const float tanRight = tanf ( angleRight ) ;
const float tanDown = tanf ( angleDown ) ;
const float tanUp = tanf ( angleUp ) ;
const float tanWidth = tanRight - tanLeft ;
const float tanHeight = tanUp - tanDown ;
const float offset = near_ ;
float matrix [ 16 ] = { } ;
matrix [ 0 ] = 2 / tanWidth ;
matrix [ 4 ] = 0 ;
matrix [ 8 ] = ( tanRight + tanLeft ) / tanWidth ;
matrix [ 12 ] = 0 ;
matrix [ 1 ] = 0 ;
matrix [ 5 ] = 2 / tanHeight ;
matrix [ 9 ] = ( tanUp + tanDown ) / tanHeight ;
matrix [ 13 ] = 0 ;
if ( far_ < = near_ ) {
matrix [ 2 ] = 0 ;
matrix [ 6 ] = 0 ;
matrix [ 10 ] = - 1 ;
matrix [ 14 ] = - ( near_ + offset ) ;
}
else {
matrix [ 2 ] = 0 ;
matrix [ 6 ] = 0 ;
matrix [ 10 ] = - ( far_ + offset ) / ( far_ - near_ ) ;
matrix [ 14 ] = - ( far_ * ( near_ + offset ) ) / ( far_ - near_ ) ;
}
matrix [ 3 ] = 0 ;
matrix [ 7 ] = 0 ;
matrix [ 11 ] = - 1 ;
matrix [ 15 ] = 0 ;
return osg : : Matrix ( matrix ) ;
}
bool View : : operator = = ( const View & rhs ) const
{
return pose = = rhs . pose & & fov = = rhs . fov ;
}
std : : ostream & operator < < (
std : : ostream & os ,
const Pose & pose )
{
os < < " position= " < < pose . position < < " , orientation= " < < pose . orientation ;
return os ;
}
std : : ostream & operator < < (
std : : ostream & os ,
const FieldOfView & fov )
{
os < < " left= " < < fov . angleLeft < < " , right= " < < fov . angleRight < < " , down= " < < fov . angleDown < < " , up= " < < fov . angleUp ;
return os ;
}
std : : ostream & operator < < (
std : : ostream & os ,
const View & view )
{
os < < " pose=< " < < view . pose < < " >, fov=< " < < view . fov < < " > " ;
return os ;
}
2020-09-14 21:09:44 +00:00
// Update stereo view/projection during update
2020-09-14 20:17:07 +00:00
class StereoUpdateCallback : public osg : : Callback
{
public :
2020-09-14 21:09:44 +00:00
StereoUpdateCallback ( StereoView * stereoView ) : stereoView ( stereoView ) { }
2020-09-14 20:17:07 +00:00
bool run ( osg : : Object * object , osg : : Object * data ) override
{
auto b = traverse ( object , data ) ;
2020-09-14 21:09:44 +00:00
stereoView - > update ( ) ;
2020-09-14 20:17:07 +00:00
return b ;
}
2020-09-14 21:09:44 +00:00
StereoView * stereoView ;
2020-09-14 20:17:07 +00:00
} ;
2020-09-14 21:09:44 +00:00
// Update states during cull
class StereoStatesetUpdateCallback : public SceneUtil : : StateSetUpdater
2020-09-14 20:17:07 +00:00
{
public :
2020-09-14 21:09:44 +00:00
StereoStatesetUpdateCallback ( StereoView * view )
2020-09-14 20:17:07 +00:00
: stereoView ( view )
{
}
protected :
virtual void setDefaults ( osg : : StateSet * stateset )
{
auto stereoViewMatrixUniform = new osg : : Uniform ( osg : : Uniform : : FLOAT_MAT4 , " stereoViewMatrices " , 2 ) ;
stateset - > addUniform ( stereoViewMatrixUniform , osg : : StateAttribute : : OVERRIDE ) ;
auto stereoViewProjectionsUniform = new osg : : Uniform ( osg : : Uniform : : FLOAT_MAT4 , " stereoViewProjections " , 2 ) ;
stateset - > addUniform ( stereoViewProjectionsUniform ) ;
auto geometryPassthroughUniform = new osg : : Uniform ( " geometryPassthrough " , false ) ;
stateset - > addUniform ( geometryPassthroughUniform ) ;
}
virtual void apply ( osg : : StateSet * stateset , osg : : NodeVisitor * /*nv*/ )
{
2020-09-14 21:09:44 +00:00
stereoView - > updateStateset ( stateset ) ;
2020-09-14 20:17:07 +00:00
}
private :
StereoView * stereoView ;
} ;
2020-09-19 18:55:06 +00:00
StereoView : : StereoView ( osgViewer : : Viewer * viewer , Technique technique , osg : : Node : : NodeMask geometryShaderMask , osg : : Node : : NodeMask noShaderMask )
2020-09-14 20:17:07 +00:00
: osg : : Group ( )
, mViewer ( viewer )
, mMainCamera ( mViewer - > getCamera ( ) )
, mRoot ( viewer - > getSceneData ( ) - > asGroup ( ) )
2020-09-19 18:55:06 +00:00
, mTechnique ( technique )
2020-09-14 20:17:07 +00:00
, mGeometryShaderMask ( geometryShaderMask )
2020-09-19 18:55:06 +00:00
, mNoShaderMask ( noShaderMask )
2020-12-08 20:37:27 +00:00
, mMasterConfig ( new SharedShadowMapConfig )
, mSlaveConfig ( new SharedShadowMapConfig )
, mSharedShadowMaps ( Settings : : Manager : : getBool ( " shared shadow maps " , " Stereo " ) )
2020-09-14 20:17:07 +00:00
{
2020-09-19 18:55:06 +00:00
if ( technique = = Technique : : None )
// Do nothing
return ;
2020-12-08 20:37:27 +00:00
mMasterConfig - > _id = " STEREO " ;
mMasterConfig - > _master = true ;
mSlaveConfig - > _id = " STEREO " ;
mSlaveConfig - > _master = false ;
2020-09-14 20:17:07 +00:00
SceneUtil : : FindByNameVisitor findScene ( " Scene Root " ) ;
mRoot - > accept ( findScene ) ;
mScene = findScene . mFoundNode ;
if ( ! mScene )
throw std : : logic_error ( " Couldn't find scene root " ) ;
2020-09-19 18:55:06 +00:00
setName ( " Stereo Root " ) ;
2020-09-14 20:17:07 +00:00
mRoot - > setDataVariance ( osg : : Object : : STATIC ) ;
setDataVariance ( osg : : Object : : STATIC ) ;
mLeftCamera - > setReferenceFrame ( osg : : Transform : : ABSOLUTE_RF ) ;
mLeftCamera - > setProjectionResizePolicy ( osg : : Camera : : FIXED ) ;
mLeftCamera - > setProjectionMatrix ( osg : : Matrix : : identity ( ) ) ;
mLeftCamera - > setViewMatrix ( osg : : Matrix : : identity ( ) ) ;
mLeftCamera - > setName ( " Stereo Left " ) ;
mLeftCamera - > setDataVariance ( osg : : Object : : STATIC ) ;
mRightCamera - > setReferenceFrame ( osg : : Transform : : ABSOLUTE_RF ) ;
mRightCamera - > setProjectionResizePolicy ( osg : : Camera : : FIXED ) ;
mRightCamera - > setProjectionMatrix ( osg : : Matrix : : identity ( ) ) ;
mRightCamera - > setViewMatrix ( osg : : Matrix : : identity ( ) ) ;
mRightCamera - > setName ( " Stereo Right " ) ;
mRightCamera - > setDataVariance ( osg : : Object : : STATIC ) ;
2020-09-19 18:55:06 +00:00
// Update stereo statesets/matrices, but after the main camera updates.
auto mainCameraCB = mMainCamera - > getUpdateCallback ( ) ;
mMainCamera - > removeUpdateCallback ( mainCameraCB ) ;
mMainCamera - > addUpdateCallback ( new StereoUpdateCallback ( this ) ) ;
mMainCamera - > addUpdateCallback ( mainCameraCB ) ;
// Do a blank double buffering of camera statesets on update. Actual state updates are performed in StereoView::Update()
mLeftCamera - > setUpdateCallback ( new SceneUtil : : StateSetUpdater ( ) ) ;
mRightCamera - > setUpdateCallback ( new SceneUtil : : StateSetUpdater ( ) ) ;
if ( mTechnique = = Technique : : GeometryShader_IndexedViewports )
{
setupGeometryShaderIndexedViewportTechnique ( ) ;
}
else
{
setupBruteForceTechnique ( ) ;
}
}
void StereoView : : setupBruteForceTechnique ( )
{
mLeftCamera - > setRenderOrder ( osg : : Camera : : NESTED_RENDER ) ;
mLeftCamera - > setClearColor ( mMainCamera - > getClearColor ( ) ) ;
mLeftCamera - > setClearMask ( GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT ) ;
mLeftCamera - > setCullMask ( mMainCamera - > getCullMask ( ) ) ;
mRightCamera - > setRenderOrder ( osg : : Camera : : NESTED_RENDER ) ;
mRightCamera - > setClearColor ( mMainCamera - > getClearColor ( ) ) ;
mRightCamera - > setClearMask ( GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT ) ;
mRightCamera - > setCullMask ( mMainCamera - > getCullMask ( ) ) ;
2020-12-08 20:37:27 +00:00
if ( mSharedShadowMaps )
{
mLeftCamera - > setUserData ( mMasterConfig ) ;
mRightCamera - > setUserData ( mSlaveConfig ) ;
}
2020-09-19 18:55:06 +00:00
// Slave cameras must have their viewports defined immediately
auto width = mMainCamera - > getViewport ( ) - > width ( ) ;
auto height = mMainCamera - > getViewport ( ) - > height ( ) ;
mLeftCamera - > setViewport ( 0 , 0 , width / 2 , height ) ;
mRightCamera - > setViewport ( width / 2 , 0 , width / 2 , height ) ;
mViewer - > stopThreading ( ) ;
mViewer - > addSlave ( mLeftCamera , true ) ;
mViewer - > addSlave ( mRightCamera , true ) ;
mRightCamera - > setGraphicsContext ( mViewer - > getCamera ( ) - > getGraphicsContext ( ) ) ;
mLeftCamera - > setGraphicsContext ( mViewer - > getCamera ( ) - > getGraphicsContext ( ) ) ;
mViewer - > getCamera ( ) - > setGraphicsContext ( nullptr ) ;
mViewer - > realize ( ) ;
}
void StereoView : : setupGeometryShaderIndexedViewportTechnique ( )
{
mLeftCamera - > setRenderOrder ( osg : : Camera : : NESTED_RENDER ) ;
mLeftCamera - > setClearMask ( GL_NONE ) ;
mLeftCamera - > setCullMask ( mNoShaderMask ) ;
mRightCamera - > setRenderOrder ( osg : : Camera : : NESTED_RENDER ) ;
mRightCamera - > setClearMask ( GL_NONE ) ;
mRightCamera - > setCullMask ( mNoShaderMask ) ;
mMainCamera - > setCullMask ( mGeometryShaderMask ) ;
2020-09-14 20:17:07 +00:00
addChild ( mStereoGeometryShaderRoot ) ;
mStereoGeometryShaderRoot - > addChild ( mRoot ) ;
addChild ( mStereoBruteForceRoot ) ;
mStereoBruteForceRoot - > addChild ( mLeftCamera ) ;
mLeftCamera - > addChild ( mScene ) ; // Use scene directly to avoid redundant shadow computation.
mStereoBruteForceRoot - > addChild ( mRightCamera ) ;
mRightCamera - > addChild ( mScene ) ;
2020-09-14 21:09:44 +00:00
addCullCallback ( new StereoStatesetUpdateCallback ( this ) ) ;
2020-09-19 18:55:06 +00:00
// Inject self as the root of the scene graph
mViewer - > setSceneData ( this ) ;
2020-09-14 20:17:07 +00:00
}
2020-09-14 21:09:44 +00:00
void StereoView : : update ( )
2020-09-14 20:17:07 +00:00
{
auto viewMatrix = mViewer - > getCamera ( ) - > getViewMatrix ( ) ;
auto projectionMatrix = mViewer - > getCamera ( ) - > getProjectionMatrix ( ) ;
View left { } ;
View right { } ;
double near = 1.f ;
double far = 10000.f ;
if ( ! cb )
{
Log ( Debug : : Error ) < < " No update view callback. Stereo rendering will not work. " ;
}
cb - > updateView ( left , right , near , far ) ;
osg : : Vec3d leftEye = left . pose . position ;
osg : : Vec3d rightEye = right . pose . position ;
osg : : Matrix leftViewOffset = left . pose . viewMatrix ( true ) ;
osg : : Matrix rightViewOffset = right . pose . viewMatrix ( true ) ;
osg : : Matrix leftViewMatrix = viewMatrix * leftViewOffset ;
osg : : Matrix rightViewMatrix = viewMatrix * rightViewOffset ;
osg : : Matrix leftProjectionMatrix = left . fov . perspectiveMatrix ( near , far ) ;
osg : : Matrix rightProjectionMatrix = right . fov . perspectiveMatrix ( near , far ) ;
2020-09-14 21:09:44 +00:00
mRightCamera - > setViewMatrix ( rightViewMatrix ) ;
mLeftCamera - > setViewMatrix ( leftViewMatrix ) ;
mRightCamera - > setProjectionMatrix ( rightProjectionMatrix ) ;
mLeftCamera - > setProjectionMatrix ( leftProjectionMatrix ) ;
2020-09-14 20:17:07 +00:00
2020-09-19 18:55:06 +00:00
auto width = mMainCamera - > getViewport ( ) - > width ( ) ;
auto height = mMainCamera - > getViewport ( ) - > height ( ) ;
2020-12-08 20:37:27 +00:00
// To correctly cull when drawing stereo using the geometry shader, the main camera must
// draw a fake view+perspective that includes the full frustums of both the left and right eyes.
// This frustum will be computed as a perspective frustum from a position P slightly behind the eyes L and R
// where it creates the minimum frustum encompassing both eyes' frustums.
// NOTE: I make an assumption that the eyes lie in a horizontal plane relative to the base view,
// and lie mirrored around the Y axis (straight ahead).
// Re-think this if that turns out to be a bad assumption.
View frustumView ;
// Compute Frustum angles. A simple min/max.
/* Example values for reference:
Left :
angleLeft - 0.767549932 float
angleRight 0.620896876 float
angleDown - 0.837898076 float
angleUp 0.726982594 float
Right :
angleLeft - 0.620896876 float
angleRight 0.767549932 float
angleDown - 0.837898076 float
angleUp 0.726982594 float
*/
frustumView . fov . angleLeft = std : : min ( left . fov . angleLeft , right . fov . angleLeft ) ;
frustumView . fov . angleRight = std : : max ( left . fov . angleRight , right . fov . angleRight ) ;
frustumView . fov . angleDown = std : : min ( left . fov . angleDown , right . fov . angleDown ) ;
frustumView . fov . angleUp = std : : max ( left . fov . angleUp , right . fov . angleUp ) ;
// Check that the case works for this approach
auto maxAngle = std : : max ( frustumView . fov . angleRight - frustumView . fov . angleLeft , frustumView . fov . angleUp - frustumView . fov . angleDown ) ;
if ( maxAngle > osg : : PI )
2020-09-14 20:17:07 +00:00
{
2020-12-08 20:37:27 +00:00
Log ( Debug : : Error ) < < " Total FOV exceeds 180 degrees. Case cannot be culled in single-pass VR. Disabling culling to cope. Consider switching to dual-pass VR. " ;
mMainCamera - > setCullingActive ( false ) ;
return ;
// TODO: An explicit frustum projection could cope, so implement that later. Guarantee you there will be VR headsets with total horizontal fov > 180 in the future. Maybe already.
}
// Use the law of sines on the triangle spanning PLR to determine P
double angleLeft = std : : abs ( frustumView . fov . angleLeft ) ;
double angleRight = std : : abs ( frustumView . fov . angleRight ) ;
double lengthRL = ( rightEye - leftEye ) . length ( ) ;
double ratioRL = lengthRL / std : : sin ( osg : : PI - angleLeft - angleRight ) ;
double lengthLP = ratioRL * std : : sin ( angleRight ) ;
2020-09-19 18:55:06 +00:00
2020-12-08 20:37:27 +00:00
osg : : Vec3d directionLP = osg : : Vec3 ( std : : cos ( - angleLeft ) , std : : sin ( - angleLeft ) , 0 ) ;
osg : : Vec3d LP = directionLP * lengthLP ;
frustumView . pose . position = leftEye + LP ;
//frustumView.pose.position.x() += 1000;
2020-09-19 18:55:06 +00:00
2020-12-08 20:37:27 +00:00
// Base view position is 0.0, by definition.
// The length of the vector P is therefore the required offset to near/far.
auto nearFarOffset = frustumView . pose . position . length ( ) ;
2020-09-19 18:55:06 +00:00
2020-12-08 20:37:27 +00:00
// Generate the frustum matrices
auto frustumViewMatrix = viewMatrix * frustumView . pose . viewMatrix ( true ) ;
auto frustumProjectionMatrix = frustumView . fov . perspectiveMatrix ( near + nearFarOffset , far + nearFarOffset ) ;
2020-09-19 18:55:06 +00:00
2020-12-08 20:37:27 +00:00
if ( mTechnique = = Technique : : GeometryShader_IndexedViewports )
{
2020-09-19 18:55:06 +00:00
// Update camera with frustum matrices
mMainCamera - > setViewMatrix ( frustumViewMatrix ) ;
mMainCamera - > setProjectionMatrix ( frustumProjectionMatrix ) ;
mLeftCamera - > getOrCreateStateSet ( ) - > setAttribute ( new osg : : ViewportIndexed ( 0 , 0 , 0 , width / 2 , height ) , osg : : StateAttribute : : OVERRIDE ) ;
mRightCamera - > getOrCreateStateSet ( ) - > setAttribute ( new osg : : ViewportIndexed ( 0 , width / 2 , 0 , width / 2 , height ) , osg : : StateAttribute : : OVERRIDE ) ;
2020-09-14 20:17:07 +00:00
}
2020-09-19 18:55:06 +00:00
else
{
mLeftCamera - > setClearColor ( mMainCamera - > getClearColor ( ) ) ;
mRightCamera - > setClearColor ( mMainCamera - > getClearColor ( ) ) ;
2020-09-14 20:17:07 +00:00
2020-09-19 18:55:06 +00:00
mLeftCamera - > setViewport ( 0 , 0 , width / 2 , height ) ;
mRightCamera - > setViewport ( width / 2 , 0 , width / 2 , height ) ;
2020-12-08 20:37:27 +00:00
if ( mMasterConfig - > _projection = = nullptr )
mMasterConfig - > _projection = new osg : : RefMatrix ;
if ( mMasterConfig - > _modelView = = nullptr )
mMasterConfig - > _modelView = new osg : : RefMatrix ;
if ( mSharedShadowMaps )
{
mMasterConfig - > _referenceFrame = mMainCamera - > getReferenceFrame ( ) ;
mMasterConfig - > _modelView - > set ( frustumViewMatrix ) ;
mMasterConfig - > _projection - > set ( projectionMatrix ) ;
}
2020-09-19 18:55:06 +00:00
}
2020-09-14 21:09:44 +00:00
}
2020-09-14 20:17:07 +00:00
2020-09-14 21:09:44 +00:00
void StereoView : : updateStateset ( osg : : StateSet * stateset )
{
// Manage viewports in update to automatically catch window/resolution changes.
auto width = mMainCamera - > getViewport ( ) - > width ( ) ;
auto height = mMainCamera - > getViewport ( ) - > height ( ) ;
stateset - > setAttribute ( new osg : : ViewportIndexed ( 0 , 0 , 0 , width / 2 , height ) ) ;
stateset - > setAttribute ( new osg : : ViewportIndexed ( 1 , width / 2 , 0 , width / 2 , height ) ) ;
2020-09-14 20:17:07 +00:00
2020-09-14 21:09:44 +00:00
// Update stereo uniforms
auto frustumViewMatrixInverse = osg : : Matrix : : inverse ( mMainCamera - > getViewMatrix ( ) ) ;
//auto frustumViewProjectionMatrixInverse = osg::Matrix::inverse(mMainCamera->getProjectionMatrix()) * osg::Matrix::inverse(mMainCamera->getViewMatrix());
2020-09-14 20:17:07 +00:00
auto * stereoViewMatrixUniform = stateset - > getUniform ( " stereoViewMatrices " ) ;
auto * stereoViewProjectionsUniform = stateset - > getUniform ( " stereoViewProjections " ) ;
2020-09-14 21:09:44 +00:00
stereoViewMatrixUniform - > setElement ( 0 , frustumViewMatrixInverse * mLeftCamera - > getViewMatrix ( ) ) ;
stereoViewMatrixUniform - > setElement ( 1 , frustumViewMatrixInverse * mRightCamera - > getViewMatrix ( ) ) ;
stereoViewProjectionsUniform - > setElement ( 0 , frustumViewMatrixInverse * mLeftCamera - > getViewMatrix ( ) * mLeftCamera - > getProjectionMatrix ( ) ) ;
stereoViewProjectionsUniform - > setElement ( 1 , frustumViewMatrixInverse * mRightCamera - > getViewMatrix ( ) * mRightCamera - > getProjectionMatrix ( ) ) ;
2020-09-14 20:17:07 +00:00
}
2020-12-08 20:37:27 +00:00
void StereoView : : setUpdateViewCallback ( std : : shared_ptr < UpdateViewCallback > cb_ )
2020-09-14 20:17:07 +00:00
{
2020-12-08 20:37:27 +00:00
cb = cb_ ;
2020-09-14 20:17:07 +00:00
}
void disableStereoForCamera ( osg : : Camera * camera )
{
auto * viewport = camera - > getViewport ( ) ;
camera - > getOrCreateStateSet ( ) - > setAttribute ( new osg : : ViewportIndexed ( 0 , viewport - > x ( ) , viewport - > y ( ) , viewport - > width ( ) , viewport - > height ( ) ) , osg : : StateAttribute : : OVERRIDE ) ;
camera - > getOrCreateStateSet ( ) - > addUniform ( new osg : : Uniform ( " geometryPassthrough " , true ) , osg : : StateAttribute : : OVERRIDE ) ;
}
void enableStereoForCamera ( osg : : Camera * camera , bool horizontalSplit )
{
auto * viewport = camera - > getViewport ( ) ;
auto x1 = viewport - > x ( ) ;
auto y1 = viewport - > y ( ) ;
auto width = viewport - > width ( ) ;
auto height = viewport - > height ( ) ;
auto x2 = x1 ;
auto y2 = y1 ;
if ( horizontalSplit )
{
width / = 2 ;
x2 + = width ;
}
else
{
height / = 2 ;
y2 + = height ;
}
camera - > getOrCreateStateSet ( ) - > setAttribute ( new osg : : ViewportIndexed ( 0 , x1 , y1 , width , height ) ) ;
camera - > getOrCreateStateSet ( ) - > setAttribute ( new osg : : ViewportIndexed ( 1 , x2 , y2 , width , height ) ) ;
camera - > getOrCreateStateSet ( ) - > addUniform ( new osg : : Uniform ( " geometryPassthrough " , false ) ) ;
}
2020-09-19 18:55:06 +00:00
StereoView : : Technique getStereoTechnique ( void )
{
auto stereoMethodString = Settings : : Manager : : getString ( " stereo method " , " Stereo " ) ;
auto stereoMethodStringLowerCase = Misc : : StringUtils : : lowerCase ( stereoMethodString ) ;
if ( stereoMethodStringLowerCase = = " geometryshader " )
{
return Misc : : StereoView : : Technique : : GeometryShader_IndexedViewports ;
}
if ( stereoMethodStringLowerCase = = " bruteforce " )
{
return Misc : : StereoView : : Technique : : BruteForce ;
}
Log ( Debug : : Warning ) < < " Unknown stereo technique \" " < < stereoMethodString < < " \" , defaulting to BruteForce " ;
return StereoView : : Technique : : BruteForce ;
}
2020-09-14 20:17:07 +00:00
void StereoView : : DefaultUpdateViewCallback : : updateView ( View & left , View & right , double & near , double & far )
{
left . pose . position = osg : : Vec3 ( - 2.2 , 0 , 0 ) ;
right . pose . position = osg : : Vec3 ( 2.2 , 0 , 0 ) ;
left . fov = { - 0.767549932 , 0.620896876 , - 0.837898076 , 0.726982594 } ;
right . fov = { - 0.620896876 , 0.767549932 , - 0.837898076 , 0.726982594 } ;
near = 1 ;
2020-09-19 18:55:06 +00:00
far = 6656 ;
2020-09-14 20:17:07 +00:00
}
}