You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
ryzom-core/nel/src/3d/transform.cpp

1475 lines
42 KiB
C++

// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
// Copyright (C) 2010 Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// This program 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
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
#include "std3d.h"
#include "nel/3d/transform.h"
#include "nel/3d/skeleton_model.h"
#include "nel/3d/scene.h"
#include "nel/3d/scene_group.h"
#include "nel/3d/root_model.h"
#include "nel/3d/u_transform.h"
#include "nel/misc/fast_floor.h"
#include "nel/misc/hierarchical_timer.h"
using namespace NLMISC;
using namespace std;
#ifdef DEBUG_NEW
#define new DEBUG_NEW
#endif
namespace NL3D
{
// ***************************************************************************
#define NL3D_TRANSFORM_DEFAULT_SHADOW_MAP_DEPTH 8.f
// ***************************************************************************
void CTransform::registerBasic()
{
CScene::registerModel( TransformId, 0, CTransform::creator);
}
// ***************************************************************************
CTransform::CTransform()
{
// important to reset for destructor to know if linked or not (CCluster !!)
_OwnerScene= NULL;
// Hrc/Graph hierarchy
_HrcParent= NULL;
_HrcParentUnfreeze= NULL;
_PrecModelToUpdate= NULL;
_NextModelToUpdate= NULL;
_TransformDirty= true;
Visibility= CHrcTrav::Herit;
_LastTransformableMatrixDate= 0;
_FatherSkeletonModel= NULL;
_ClusterSystem = NULL;
_FreezeHRCState= FreezeHRCStateDisabled;
_OrderingLayer = 2;
_TransparencyPriority = 0;
// No logicInfo by default
_LogicInfo= NULL;
_ForceCLodSticked= false;
// default MeanColor value
_MeanColor.set(255,255,255,255);
// Default ShadowMap direction
_ShadowMapDirectionZThreshold= -0.5f;
_ShadowMapMaxDepth= NL3D_TRANSFORM_DEFAULT_SHADOW_MAP_DEPTH;
// Setup some state.
/*
Default are:
IsAnimDetailable= 0
IsLoadBalancable= 0
IsLightable= 0
IsRenderable= 0
IsTransparent= 0
IsOpaque= 1
QuadGridClipEnabled= 0.
IsUserLightable= 1 // default, the model may be lighted.
IsFinalLightable= 0
IsNeedUpdateLighting= 0
ISNeedUpdateFrozenStaticLightSetup= 0
IsSkeleton= 0
IsTransformShape=0
IsCluster= 0
IsMeshBaseInstance= 0
IsDeleteChannelMixer = 0;
*/
_StateFlags= IsOpaque | IsUserLightable;
// By default, always allow rendering of Transform Models.
_RenderFilterType = std::numeric_limits<uint32>::max();
// By default, don't suport fast intersection detection
_SupportFastIntersect= false;
// **** HRC Init Traversal Computed Data.
_LocalVis= CHrcTrav::Herit; _LocalMatrix.identity(); _LocalDate=0;
_WorldVis= true; _WorldMatrix.identity();
// Init the _WorldDate to -1 so at first pass, _LocalDate>_WorldDate, and so
// the model will be processed and so it'll may be inserted in LightingManager (for example)
_WorldDate=-1;
_Frozen = false;
_DontUnfreezeChildren = false;
_AncestorSkeletonModel= NULL;
_ClipLinkedInSonsOfAncestorSkeletonModelGroup= false;
// **** Clip Init Traversal Computed Data.
_ClipDate= 0;
_Visible=false;
_IndexInVisibleList= -1;
// **** AnimDetail Init Traversal Computed Data.
// none
// **** LoadBalancing Init Traversal Computed Data.
_LoadBalancingGroup= NULL;
}
// ***************************************************************************
CTransform::~CTransform()
{
// If still binded to a father skeleton
if( _FatherSkeletonModel )
{
/* If skinned, cannot detach me from skeleton here because detachSkeletonSon()
use some virtual calls of transform: setApplySkin().
Hence, It is the deriver job to detach himself from the skeleton.
NB: test isSkinned(), not isSkinnable(), since isSkinned() is not virtual ....
This means that if a Mesh isSkinnable(), but never skinned, it is not asserted here.
*/
if( isSkinned() )
{
nlstop;
}
else
// Can detach Me. Important for UTransform sticked
_FatherSkeletonModel->detachSkeletonSon(this);
}
// resetLighting, removing me from PointLight Transform list.
// NB: not done for FrozenStaticLightSetup, because those lights don't owns me.
resetLighting();
// Must also remove me from the lightingManager.
// must test getOwnerScene() because of CCluster usage out of CScene (thanks to mat!! :) )
if(getOwnerScene())
{
CLightTrav &lightTrav= getOwnerScene()->getLightTrav();
_LightedModelIt= lightTrav.LightingManager.eraseStaticLightedModel(_LightedModelIt);
}
if (getChannelMixerOwnerShip()) delete (CChannelMixer *) _ChannelMixer;
// ensure the model is no more linked to the UpdateList.
unlinkFromUpdateList();
// I must remove me from _VisibleList.
if(_IndexInVisibleList>=0)
{
CClipTrav &clipTrav= getOwnerScene()->getClipTrav();
nlassert(_IndexInVisibleList < (sint)clipTrav._CurrentNumVisibleModels );
// Mark NULL. NB: faster than a CRefPtr.
clipTrav._VisibleList[_IndexInVisibleList]= NULL;
_IndexInVisibleList= -1;
}
// remove me from parents in Hrc and Clip
setStateFlag(ForceClipRoot, false); // ensure that not 'glued' to the root so that the following call will succeed
hrcUnlink();
clipUnlinkFromAll();
// remove mys sons.
while(hrcGetNumChildren())
{
hrcGetChild(0)->hrcUnlink();
}
while(clipGetNumChildren())
{
clipDelChild(clipGetChild(0));
}
nlassert(_HrcSons.empty());
nlassert(_HrcParent==NULL);
nlassert(_ClipSons.empty());
nlassert(_ClipParents.empty());
}
// ***************************************************************************
void CTransform::initModel()
{
// assign me to the default group
_LoadBalancingGroup= getOwnerScene()->getLoadBalancingTrav().getDefaultGroup();
}
// ***************************************************************************
void CTransform::hide()
{
// Optim: do nothing if already set
if(Visibility!= CHrcTrav::Hide)
{
_TransformDirty= true;
Visibility= CHrcTrav::Hide;
// If skinned, then must inform skeleton parent that it must recompute skin render/animDetail lists
if(isSkinned())
{
nlassert(_FatherSkeletonModel);
_FatherSkeletonModel->dirtSkinRenderLists();
}
}
}
// ***************************************************************************
void CTransform::setTransparency(bool v)
{
bool bTmp = getStateFlag(IsTransparent) == 0 ? false : true;
if (bTmp != v)
{
setStateFlag(IsTransparent, v);
if(isSkinned())
{
nlassert(_FatherSkeletonModel);
_FatherSkeletonModel->dirtSkinRenderLists();
}
}
}
// ***************************************************************************
void CTransform::setBypassLODOpacityFlag(bool bypass)
{
setStateFlag(BypassLODOpacity, bypass);
}
// ***************************************************************************
void CTransform::setOpacity(bool v)
{
bool bTmp = getStateFlag(IsOpaque) == 0 ? false : true;
if (bTmp != v)
{
setStateFlag(IsOpaque, v);
if(isSkinned())
{
nlassert(_FatherSkeletonModel);
_FatherSkeletonModel->dirtSkinRenderLists();
}
}
}
// ***************************************************************************
void CTransform::show()
{
// Optim: do nothing if already set
if(Visibility!= CHrcTrav::Show)
{
_TransformDirty= true;
Visibility= CHrcTrav::Show;
// If skinned, then must inform skeleton parent that it must recompute skin render/animDetail lists
if(isSkinned())
{
nlassert(_FatherSkeletonModel);
_FatherSkeletonModel->dirtSkinRenderLists();
}
}
}
// ***************************************************************************
void CTransform::heritVisibility()
{
// Optim: do nothing if already set
if(Visibility!= CHrcTrav::Herit)
{
_TransformDirty= true;
Visibility= CHrcTrav::Herit;
// If skinned, then must inform skeleton parent that it must recompute skin render/animDetail lists
if(isSkinned())
{
nlassert(_FatherSkeletonModel);
_FatherSkeletonModel->dirtSkinRenderLists();
}
}
}
// ***************************************************************************
CTrackDefaultVector CTransform::DefaultPos( CVector::Null );
CTrackDefaultVector CTransform::DefaultRotEuler( CVector::Null );
CTrackDefaultQuat CTransform::DefaultRotQuat( NLMISC::CQuat::Identity );
CTrackDefaultVector CTransform::DefaultScale( CVector(1,1,1) );
CTrackDefaultVector CTransform::DefaultPivot( CVector::Null );
ITrack* CTransform::getDefaultTrack (uint valueId)
{
// Cyril: prefer do it here in CTransform, because of CCamera, CLight etc... (which may not need a default value too!!)
// what value ?
switch (valueId)
{
case PosValue: return &DefaultPos;
case RotEulerValue: return &DefaultRotEuler;
case RotQuatValue: return &DefaultRotQuat;
case ScaleValue: return &DefaultScale;
case PivotValue: return &DefaultPivot;
}
// No, only ITrnasformable values!
nlstop;
// Deriver note: else call BaseClass::getDefaultTrack(valueId);
return NULL;
}
// ***************************************************************************
void CTransform::registerToChannelMixer(CChannelMixer *chanMixer, const std::string &prefix)
{
if (getChannelMixerOwnerShip() && chanMixer != _ChannelMixer)
{
delete _ChannelMixer;
setChannelMixerOwnerShip(false);
}
// Hey!! we are animated!!
_ChannelMixer= chanMixer;
// Update flag, if we must be inserted in AnimDetail
setStateFlag(IsAnimDetailable, _ChannelMixer || getStateFlag(IsForceAnimDetail) );
// If skinned, then must inform skeleton parent that it must recompute skin render/animDetail lists
if(isSkinned())
{
nlassert(_FatherSkeletonModel);
_FatherSkeletonModel->dirtSkinRenderLists();
}
// For CTransfom, channels are not detailled.
addValue(chanMixer, PosValue, OwnerBit, prefix, false);
addValue(chanMixer, RotEulerValue, OwnerBit, prefix, false);
addValue(chanMixer, RotQuatValue, OwnerBit, prefix, false);
addValue(chanMixer, ScaleValue, OwnerBit, prefix, false);
addValue(chanMixer, PivotValue, OwnerBit, prefix, false);
// Deriver note: if necessary, call BaseClass::registerToChannelMixer(chanMixer, prefix);
}
// ***************************************************************************
void CTransform::freeze()
{
// First, update the model
// _Frozen state is disabled here (in CTransform::update()).
update();
// Then flag the frozen state.
_Frozen= true;
}
// ***************************************************************************
void CTransform::setDontUnfreezeChildren(bool val)
{
_DontUnfreezeChildren = val;
}
// ***************************************************************************
void CTransform::freezeHRC()
{
// if disabled, say we are ready to validate our worldMatrix for long.
if(_FreezeHRCState==FreezeHRCStateDisabled)
{
_FreezeHRCState= FreezeHRCStateRequest;
setStateFlag(QuadGridClipEnabled, true);
/* If the transform is not frozen (ie staticaly inserted in a cluster),
We must be sure it will be tested against QuadGridClipManager at next ClipTrav pass.
=> must make this object a "moving object" at next render=> dirt _LocalMatrixDate.
*/
if(!_Frozen)
{
_TransformDirty= true;
}
}
}
// ***************************************************************************
void CTransform::unfreezeHRC()
{
// if this model is no HRC frozen disabled
if(_FreezeHRCState!=FreezeHRCStateDisabled)
{
// if model correctly frozen.
if(_FreezeHRCState == CTransform::FreezeHRCStateEnabled )
{
// Should not be linked : can't link after a freezeHRC
nlassert (_HrcParent == NULL);
// Set as unfreeze else, hrcLinkSon doesn't work
_FreezeHRCState= FreezeHRCStateDisabled;
// Link this model to the previous HRC parent.
if (_HrcParentUnfreeze)
_HrcParentUnfreeze->hrcLinkSon( this );
else
getOwnerScene()->getRoot()->hrcLinkSon( this );
// Link this object to the validateList.
linkToUpdateList();
// if lightable()
if( isLightable() )
{
CLightTrav &lightTrav= getOwnerScene()->getLightTrav();
// Lighting: must remove the object from the quadGrid.
// NB: works if _LightedModelIt==NULL. result is that _LightedModelIt= NULL.
_LightedModelIt= lightTrav.LightingManager.eraseStaticLightedModel(_LightedModelIt);
}
}
else
_FreezeHRCState= FreezeHRCStateDisabled;
// unlink me from any QuadCluster, and disable QuadCluster
unlinkFromQuadCluster();
setStateFlag(QuadGridClipEnabled, false);
}
}
// ***************************************************************************
void CTransform::update()
{
// test if the matrix has been changed in ITransformable.
if(ITransformable::compareMatrixDate(_LastTransformableMatrixDate))
{
_LastTransformableMatrixDate= ITransformable::getMatrixDate();
_TransformDirty= true;
}
// update the freezeHRC state.
if(_FreezeHRCState != CTransform::FreezeHRCStateDisabled)
{
// if the model request to be frozen in HRC
if(_FreezeHRCState == CTransform::FreezeHRCStateRequest )
{
// Wait for next Hrc traversal to compute good _WorldMatrix for this model and his sons.
// Also, next Hrc traversal will insert the model in the LightingManager quadGrid (if lightable)
_FreezeHRCState = CTransform::FreezeHRCStateReady;
}
// if the model is ready to be frozen in HRC, then do it!!
else if( _FreezeHRCState == CTransform::FreezeHRCStateReady )
{
// Unlink this model.
hrcUnlink();
// unLink this object from the validateList. NB: the list will still be correclty parsed.
unlinkFromUpdateList();
// if lightable, the model is inserted in a quadgrid to update his lighting only when
// dynamicLights touch him (since himself is static).
if( isLightable() )
{
CLightTrav &lightTrav= getOwnerScene()->getLightTrav();
// Lighting: must reinsert the object from the quadGrid.
// NB: works if _LightedModelIt==NULL. result is that _LightedModelIt= NULL.
_LightedModelIt= lightTrav.LightingManager.eraseStaticLightedModel(_LightedModelIt);
// insert in the quadgrid.
_LightedModelIt= lightTrav.LightingManager.insertStaticLightedModel(this);
}
// Now this model won't be tested for validation nor for worldMatrix update. End!!
_FreezeHRCState = CTransform::FreezeHRCStateEnabled;
}
}
// update _LocalMatrix
if(_TransformDirty)
{
// update the local matrix.
_LocalMatrix= getMatrix();
_LocalVis= Visibility;
// update the date of the local matrix.
_LocalDate= getOwnerScene()->getHrcTrav().CurrentDate;
// The transform has been modified. Hence, it is no more frozen.
_Frozen= false;
// ok!
_TransformDirty= false;
}
}
// ***************************************************************************
void CTransform::getAABBox(NLMISC::CAABBox &bbox) const
{
bbox.setCenter(CVector::Null);
bbox.setHalfSize(CVector::Null);
}
// ***************************************************************************
void CTransform::setLoadBalancingGroup(const std::string &group)
{
// Get the traversal.
CLoadBalancingTrav &trav= getOwnerScene()->getLoadBalancingTrav();
// get the group from trav (create if needed), and set it.
_LoadBalancingGroup= trav.getOrCreateGroup(group);
}
// ***************************************************************************
const std::string &CTransform::getLoadBalancingGroup() const
{
// get the group name
return _LoadBalancingGroup->Name;
}
// ***************************************************************************
void CTransform::setMeanColor(CRGBA color)
{
// if the color is different from prec
if(color!=_MeanColor)
{
// change it.
_MeanColor= color;
}
}
// ***************************************************************************
void CTransform::setIsLightable(bool val)
{
setStateFlag(IsLightable, val);
// update IsFinalLightable
setStateFlag(IsFinalLightable, (getStateFlag(IsLightable) && getStateFlag(IsUserLightable)) );
}
// ***************************************************************************
void CTransform::setUserLightable(bool enable)
{
setStateFlag(IsUserLightable, enable);
// update IsFinalLightable
setStateFlag(IsFinalLightable, (getStateFlag(IsLightable) && getStateFlag(IsUserLightable)) );
}
// ***************************************************************************
void CTransform::setIsRenderable(bool val)
{
setStateFlag(IsRenderable, val);
}
// ***************************************************************************
void CTransform::setIsBigLightable(bool val)
{
setStateFlag(IsBigLightable, val);
}
// ***************************************************************************
void CTransform::setIsSkeleton(bool val)
{
setStateFlag(IsSkeleton, val);
}
// ***************************************************************************
void CTransform::setApplySkin(bool state)
{
setStateFlag(IsSkinned, state);
}
// ***************************************************************************
void CTransform::setIsForceAnimDetail(bool val)
{
setStateFlag(IsForceAnimDetail, val );
// Update flag, if we must be inserted in AnimDetail
setStateFlag(IsAnimDetailable, _ChannelMixer || getStateFlag(IsForceAnimDetail) );
// If skinned, then must inform skeleton parent that it must recompute skin render/animDetail lists
if(isSkinned())
{
nlassert(_FatherSkeletonModel);
_FatherSkeletonModel->dirtSkinRenderLists();
}
}
// ***************************************************************************
void CTransform::setIsLoadbalancable(bool val)
{
setStateFlag(IsLoadBalancable, val );
}
// ***************************************************************************
void CTransform::linkToUpdateList()
{
if(!_OwnerScene)
return;
// If the model is not already inserted.
if( ! (_PrecModelToUpdate!=NULL || _OwnerScene->_UpdateModelList==this) )
{
// insert it.
_NextModelToUpdate= _OwnerScene->_UpdateModelList;
_PrecModelToUpdate= NULL;
if(_NextModelToUpdate)
_NextModelToUpdate->_PrecModelToUpdate= this;
_OwnerScene->_UpdateModelList= this;
}
}
// ***************************************************************************
void CTransform::unlinkFromUpdateList()
{
if(!_OwnerScene)
return;
// If the model is inserted.
if( _PrecModelToUpdate!=NULL || _OwnerScene->_UpdateModelList==this )
{
// update prec.
if(_PrecModelToUpdate)
_PrecModelToUpdate->_NextModelToUpdate= _NextModelToUpdate;
else
_OwnerScene->_UpdateModelList= _NextModelToUpdate;
// update next.
if(_NextModelToUpdate)
_NextModelToUpdate->_PrecModelToUpdate= _PrecModelToUpdate;
// End.
_PrecModelToUpdate= NULL;
_NextModelToUpdate= NULL;
}
}
// ***************************************************************************
// ***************************************************************************
// Hrc Trav
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
void CTransform::updateWorld()
{
const CMatrix *pFatherWM;
bool visFather;
// If not root case, link to Fahter.
if(_HrcParent)
{
pFatherWM= &(_HrcParent->_WorldMatrix);
visFather= _HrcParent->_WorldVis;
// if _HrcParent is not frozen (for any reason), disable us!
if (!_HrcParent->_Frozen && !_HrcParent->_DontUnfreezeChildren)
_Frozen= false;
// herit _AncestorSkeletonModel
if (_HrcParent->_AncestorSkeletonModel)
// If my father has an _AncestorSkeletonModel, get it.
_AncestorSkeletonModel= _HrcParent->_AncestorSkeletonModel;
else
// else I have an ancestor skel model if I am sticked/binded directly to a skeleton model.
_AncestorSkeletonModel= _FatherSkeletonModel;
}
// else, default!!
else
{
pFatherWM= &(CMatrix::Identity);
visFather= true;
// at the root of the hierarchy, we have no parent, hence no FatherSkeletonModel nor _AncestorSkeletonModel.
_AncestorSkeletonModel= NULL;
// NB: Root is Frozen by essence :), so don't modify the frozen state here.
}
// Combine matrix
if(_LocalDate>_WorldDate || (_HrcParent && _HrcParent->_WorldDate>_WorldDate) )
{
// Must recompute the world matrix. ONLY IF I AM NOT SKINNED/STICKED TO A SKELETON in the hierarchy!
if( _AncestorSkeletonModel==NULL )
{
_WorldMatrix= *pFatherWM * _LocalMatrix;
_WorldDate= getOwnerScene()->getHrcTrav().CurrentDate;
// Add the model to the moving object list, only if I am a transform shape
if (!_Frozen && isTransformShape() && !getStateFlag(ForceClipRoot))
getOwnerScene()->getHrcTrav()._MovingObjects.push_back (static_cast<CTransformShape*>(this));
}
}
// Update dynamic lighting.
/*
If the model is not frozen in StaticLight, then must update lighting each frame.
Even if the object doesn't move, a new dynamic light may enter in its aera. Hence we must test
it in the light quadrid. StaticLight-ed Objects don't need it because they are inserted in a special quadgrid,
where dynamics lights touch all StaticLight-ed object to force their computing
NB: not done if _AncestorSkeletonModel!=NULL. no need because in this case,
result is driven by the _LightContribution of the _AncestorSkeletonModel.
*/
if( !_LightContribution.FrozenStaticLightSetup && _AncestorSkeletonModel==NULL )
{
// if the model is lightable reset lighting
if( isLightable() )
resetLighting();
}
// Combine visibility.
switch(_LocalVis)
{
case CHrcTrav::Herit: _WorldVis= visFather; break;
case CHrcTrav::Hide: _WorldVis= false; break;
case CHrcTrav::Show: _WorldVis= true; break;
default: break;
}
// If I have an ancestor Skeleton Model, I must be binded in ClipTrav to the SonsOfAncestorSkeletonModelGroup
updateClipTravForAncestorSkeleton();
}
// ***************************************************************************
void CTransform::updateClipTravForAncestorSkeleton()
{
// If I have an ancestor Skeleton Model, I must be binded in ClipTrav to the SonsOfAncestorSkeletonModelGroup
if(_AncestorSkeletonModel && !_ClipLinkedInSonsOfAncestorSkeletonModelGroup)
{
// must unlink from ALL olds models.
clipUnlinkFromAll();
// And link to SonsOfAncestorSkeletonModelGroup.
getOwnerScene()->SonsOfAncestorSkeletonModelGroup->clipAddChild(this);
// update the flag.
_ClipLinkedInSonsOfAncestorSkeletonModelGroup= true;
}
// else I must be binded to the standard Root.
if(!_AncestorSkeletonModel && _ClipLinkedInSonsOfAncestorSkeletonModelGroup)
{
// verify first I am really still linked to the SonsOfAncestorSkeletonModelGroup.
// This test is important, because link may have changed for any reason (portals, clipManager....).
if( clipGetNumParents() == 1 && clipGetParent(0)==getOwnerScene()->SonsOfAncestorSkeletonModelGroup )
{
// must unlink from ALL olds models.
clipUnlinkFromAll();
// and now, link to std root.
getOwnerScene()->getRoot()->clipAddChild(this);
}
// update the flag
_ClipLinkedInSonsOfAncestorSkeletonModelGroup= false;
}
}
// ***************************************************************************
void CTransform::traverseHrc()
{
// Recompute the matrix, according to _HrcParent matrix mode, and local matrix.
updateWorld();
// Traverse the Hrc sons.
uint num= hrcGetNumChildren();
for(uint i=0;i<num;i++)
hrcGetChild(i)->traverseHrc();
}
// ***************************************************************************
// ***************************************************************************
// Clip Trav
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
void CTransform::setClusterSystem(CInstanceGroup *pCS)
{
if (pCS != NULL)
{
nlassert(!getStateFlag(ForceClipRoot)); // the transform must be linked to the root, and have not cluster system when this flag is set
}
// Special case for the "AutoClusterSystem" when pCS==-1
if(pCS==(CInstanceGroup*)-1)
{
_ClusterSystem = NULL;
setStateFlag(ClusterSystemAuto, true);
}
else
{
_ClusterSystem = pCS;
setStateFlag(ClusterSystemAuto, false);
}
}
// ***************************************************************************
CInstanceGroup* CTransform::getClusterSystem ()
{
if(getStateFlag(ClusterSystemAuto))
return (CInstanceGroup*)-1;
else
return _ClusterSystem;
}
// ***************************************************************************
void CTransform::traverseClip()
{
// disable H_AUTO, because slowdown when lot of models (eg 1000-2000 tested in forest)
//H_AUTO( NL3D_TransformClip );
CScene *scene= getOwnerScene();
CClipTrav &clipTrav= scene->getClipTrav();
if ((_ClipDate == clipTrav.CurrentDate) && _Visible)
return;
_ClipDate = clipTrav.CurrentDate;
// clip: update Visible flag.
_Visible= false;
// if at least visible.
if(_WorldVis)
{
// If linked to a SkeletonModel anywhere in the hierarchy, don't clip, and use skeleton model clip result.
// This works because we are sons of a special node which is not in the clip traversal, and
// which is traversed at end of the traversal.
if( _AncestorSkeletonModel!=NULL )
{
_Visible= _AncestorSkeletonModel->isClipVisible();
// Special test: if we are sticked to a skeletonModel, and if we are still visible, maybe we don't have to
if(_Visible && _FatherSkeletonModel)
{
// if our skeletonModel father is displayed with a Lod, maybe we are not to be displayed
if(_FatherSkeletonModel->isDisplayedAsLodCharacter())
{
// We are visible only if we where sticked to the skeleton with forceCLod==true.
// This is also true if we are actually a skeletonModel
if(!_ForceCLodSticked)
// otherWise we are not visible. eg: this is the case of skins and some sticked object
_Visible= false;
}
}
}
// else, clip.
else
{
// If the instance is not filtered
if(scene->getFilterRenderFlags() & _RenderFilterType)
{
// User cliping enabled ?
if (_StateFlags & UserClipping)
_Visible= true;
else
_Visible= clip();
}
}
}
// if visible, add to list.
if(_Visible)
{
// add this model to the visibility list.
clipTrav.addVisibleModel(this);
// Has not an ancestor skeleton model?
if( _AncestorSkeletonModel==NULL )
{
// If needed, insert the model in the lighted list.
// don't insert if has an ancestorSkeletonModel, because in this case, result is driven by
// the _LightContribution of the _AncestorSkeletonModel.
if( isLightable() )
scene->getLightTrav().addLightedModel(this);
// If needed, insert the model in the animDetail list.
// don't insert if has an ancestoreSkeletonModel, because in this case, this ancestore will
// animDetail through the hierarchy...
if( isAnimDetailable() )
scene->getAnimDetailTrav().addVisibleModel(this);
}
// If needed, Add it to the loadBalancing trav
if( isLoadBalancable() )
scene->getLoadBalancingTrav().addVisibleModel(this);
// If needed, insert the model in the render list.
if( isRenderable() )
scene->getRenderTrav().addRenderModel(this);
}
// Traverse the Clip sons.
uint num= clipGetNumChildren();
for(uint i=0;i<num;i++)
clipGetChild(i)->traverseClip();
}
// ***************************************************************************
// ***************************************************************************
// AnimDetail Trav
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
void CTransform::updateWorldMatrixFromFather()
{
// If I am not skinned, and If I have a skeleton ancestor
if(!isSkinned() && _AncestorSkeletonModel )
{
// Compute the HRC _WorldMatrix.
// if I am not sticked.
if(!_FatherSkeletonModel)
{
// get the normal father worldMatrix in Hrc.
CTransform *fatherTransform= hrcGetParent();
// if exist
if(fatherTransform)
{
const CMatrix &parentWM= fatherTransform->_WorldMatrix;
// combine worldMatrix
_WorldMatrix= parentWM * _LocalMatrix;
}
else
_WorldMatrix= _LocalMatrix;
}
else
{
// get the worldMatrix of the bone if I am sticked (standard stick)
if(!getStateFlag(SSSWO))
{
const CMatrix &parentWM= _FatherSkeletonModel->Bones[_FatherBoneId].getWorldMatrix();
// combine worldMatrix
_WorldMatrix= parentWM * _LocalMatrix;
}
// Special SkeletonSpawnScript stick
else
{
// The parent matrix must be computed from a special matrix given to the skeleton model
CMatrix parentWM;
parentWM.setRot(CVector::I, _FatherSkeletonModel->getSSSWODir(), CVector::K);
parentWM.normalize(CMatrix::YZX);
parentWM.setPos(_FatherSkeletonModel->getSSSWOPos());
// combine worldMatrix
_WorldMatrix= parentWM * _LocalMatrix;
}
}
}
}
// ***************************************************************************
void CTransform::traverseAnimDetailWithoutUpdateWorldMatrix()
{
// AnimDetail behavior: animate only if not clipped.
// NB: no need to test because of VisibilityList use.
// test if the refptr is NULL or not (RefPtr).
CChannelMixer *chanmix= _ChannelMixer;
if(chanmix)
{
// eval detail!!
chanmix->eval(true, getOwnerScene()->getAnimDetailTrav().CurrentDate);
}
}
// ***************************************************************************
void CTransform::traverseAnimDetail()
{
// First, test if I must update my worldMatrix because of the ancestorSkeleton scheme
updateWorldMatrixFromFather();
// eval channelMixer.
traverseAnimDetailWithoutUpdateWorldMatrix();
// NB: if want to add something, do it in traverseAnimDetailWithoutUpdateWorldMatrix(), because
// CSkeletonModel doesn't call CTransform::traverseAnimDetail()
}
// ***************************************************************************
// ***************************************************************************
// LoadBalancing
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
void CTransform::traverseLoadBalancing()
{
// noop
}
// ***************************************************************************
// ***************************************************************************
// Lighting.
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
void CTransform::resetLighting()
{
// if the model is already isNeedUpdateLighting, his light setup is reseted.
// so no need to reset again
if(isNeedUpdateLighting())
return;
// For all light not in FrozenStaticLightSetup, remove me from their list
uint startLight= 0;
if(_LightContribution.FrozenStaticLightSetup)
{
startLight= _LightContribution.NumFrozenStaticLight;
}
// for all light in the list, remove me from their list.
for(uint i=startLight; i<NL3D_MAX_LIGHT_CONTRIBUTION; i++)
{
CPointLight *pl= _LightContribution.PointLight[i];
// if end of list, break.
if(!pl)
break;
else
{
// remove me from this light.
pl->removeLightedModel(_LightContribution.TransformIterator[i]);
}
}
// empty the list.
if(startLight<NL3D_MAX_LIGHT_CONTRIBUTION)
_LightContribution.PointLight[startLight]= NULL;
// the model needs to update his lighting.
setStateFlag(IsNeedUpdateLighting, true);
}
// ***************************************************************************
void CTransform::freezeStaticLightSetup(CPointLight *pointLight[NL3D_MAX_LIGHT_CONTRIBUTION],
uint numPointLights, uint8 sunContribution, CPointLight *frozenAmbientlight)
{
nlassert(numPointLights <= NL3D_MAX_LIGHT_CONTRIBUTION);
// resetLighting() first.
resetLighting();
// Enable StaticLightSetup.
_LightContribution.FrozenStaticLightSetup= true;
_LightContribution.NumFrozenStaticLight= uint8(numPointLights);
_LightContribution.SunContribution= sunContribution;
// setup the FrozenAmbientLight
_LightContribution.FrozenAmbientLight= frozenAmbientlight;
// Setup other pointLights
uint i;
for(i=0;i<numPointLights;i++)
{
// set the light
_LightContribution.PointLight[i]= pointLight[i];
// Enable at max.
_LightContribution.Factor[i]= 255;
// Compute static AttFactor Later because don't have WorlPosition of the model here!!
setStateFlag(IsNeedUpdateFrozenStaticLightSetup, true);
// Do NOT set the iterator, because it is a staticLight.
}
// End the list
if(i<NL3D_MAX_LIGHT_CONTRIBUTION)
_LightContribution.PointLight[i]= NULL;
}
// ***************************************************************************
void CTransform::unfreezeStaticLightSetup()
{
// resetLighting() first.
resetLighting();
// Disable StaticLightSetup.
_LightContribution.FrozenStaticLightSetup= false;
_LightContribution.NumFrozenStaticLight= 0;
// End the list
_LightContribution.PointLight[0]= NULL;
// No more FrozenAmbientLight
_LightContribution.FrozenAmbientLight= NULL;
// Don't need to update StaticLightSetup since no more exist.
setStateFlag(IsNeedUpdateFrozenStaticLightSetup, false);
}
// ***************************************************************************
void CTransform::traverseLight()
{
// if the model do not need to update his lighting, just skip.
if(!isNeedUpdateLighting())
return;
// If a freezeStaticLightSetup() has been called on this model recently.
if(isNeedUpdateFrozenStaticLightSetup())
{
// Now, the correct matrix is computed.
// get the untransformed bbox from the model.
CAABBox bbox;
getAABBox(bbox);
// get transformed center pos of bbox
CVector worldModelPos= getWorldMatrix() * bbox.getCenter();
// So we can compute AttFactor for each static light influencing this static object
uint numPointLights= _LightContribution.NumFrozenStaticLight;
for(uint i=0;i<numPointLights;i++)
{
const CPointLight *pl= _LightContribution.PointLight[i];
// don't worry about the precision of floor, because of *255.
float distToModel= (pl->getPosition() - worldModelPos).norm();
sint attFactor= NLMISC::OptFastFloor( 255 * pl->computeLinearAttenuation(worldModelPos, distToModel) );
_LightContribution.AttFactor[i]= (uint8)attFactor;
}
// clean.
setStateFlag(CTransform::IsNeedUpdateFrozenStaticLightSetup, false);
}
// see CTransform::clip(), here I am Lightable(), and I have no _AncestorSkeletonModel
// So I am sure that I really need to recompute my ModelLightContributions.
CScene *scene= getOwnerScene();
scene->getLightTrav().LightingManager.computeModelLightContributions(scene->getSunAmbient(), this,
_LightContribution, _LogicInfo);
// done!
setStateFlag(CTransform::IsNeedUpdateLighting, false);
}
// ***************************************************************************
// ***************************************************************************
// Rendering
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
void CTransform::traverseRender()
{
// no-op
}
// ***************************************************************************
void CTransform::profileRender()
{
// no-op
}
// ***************************************************************************
// ***************************************************************************
// Hrc Linking
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
void CTransform::hrcLinkSon(CTransform *son)
{
if(!son)
return;
// If not unfrozen, can't link
if (son->_FreezeHRCState != CTransform::FreezeHRCStateDisabled)
return;
// no-op if already me.
if(son->_HrcParent==this)
return;
// unlink from anyone
son->hrcUnlink();
// link son to me
_HrcSons.insert(son, &son->_HrcNode);
// link me to son
son->_HrcParent= this;
// Backup parent
son->_HrcParentUnfreeze= this;
// my son should recompute his worldMatrix!
son->_WorldDate= -1;
}
// ***************************************************************************
void CTransform::hrcUnlink()
{
// no-op if already NULL
if(_HrcParent==NULL)
return;
// if ForceClipRoot flag is set, then the fx can't be linked elsewhere in the hierarchy
nlassert(!getStateFlag(ForceClipRoot));
// unlink my parent from me.
_HrcNode.unlink();
// unlink me from parent
_HrcParent= NULL;
_HrcParentUnfreeze= NULL;
// I should recompute my worldMatrix (well not useful since not linked, but still do it...)
_WorldDate= -1;
}
// ***************************************************************************
CTransform *CTransform::hrcGetChild(uint index) const
{
nlassert(index < _HrcSons.size());
return (const_cast<CTransform*>(this))->_HrcSons.begin()[index];
}
// ***************************************************************************
// ***************************************************************************
// Clip Linking
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
void CTransform::clipAddChild(CTransform *son)
{
if(!son)
return;
// if already linked, no-op.
if(son->clipHasParent(this))
return;
// add a new parent entry for our son.
CClipNode *clipNode= new CClipNode;
son->_ClipParents.push_back(clipNode);
// link the son to us
clipNode->Parent= this;
// link us to the son
_ClipSons.insert(son, &clipNode->ClipNode);
}
// ***************************************************************************
void CTransform::clipDelChild(CTransform *son)
{
if(!son)
return;
// try to remove from me from my parent
son->clipDelFromParent(this);
}
// ***************************************************************************
void CTransform::clipUnlinkFromAll()
{
// unlink from all parent clip
while( clipGetNumParents() )
{
clipDelFromParent( clipGetParent(0) );
}
}
// ***************************************************************************
CTransform *CTransform::clipGetParent(uint index) const
{
nlassert(index < _ClipParents.size());
return _ClipParents[index]->Parent;
}
// ***************************************************************************
CTransform *CTransform::clipGetChild(uint index) const
{
nlassert(index < _ClipSons.size());
return (const_cast<CTransform*>(this))->_ClipSons.begin()[index];
}
// ***************************************************************************
bool CTransform::clipHasParent(CTransform *parent)
{
// search O(n) for all parents
for(uint i=0;i<_ClipParents.size();i++)
{
if(_ClipParents[i]->Parent==parent)
return true;
}
return false;
}
// ***************************************************************************
void CTransform::clipDelFromParent(CTransform *parent)
{
// search O(n) for all Parents
uint numParents= (uint)_ClipParents.size();
for(uint i=0;i<numParents;i++)
{
if(_ClipParents[i]->Parent==parent)
{
// found! remove me from my parent list
_ClipParents[i]->ClipNode.unlink();
// remove this parent entry. swap with last
swap(_ClipParents[i], _ClipParents[numParents-1]);
// and delete last.
delete _ClipParents[numParents-1];
_ClipParents.resize(numParents-1);
break;
}
}
}
// ***************************************************************************
void CTransform::setUserClipping(bool enable)
{
setStateFlag (UserClipping, enable);
}
// ***************************************************************************
bool CTransform::getUserClipping() const
{
return getStateFlag(UserClipping) != 0;
}
// ***************************************************************************
// ***************************************************************************
// ShadowMap
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
void CTransform::getReceiverBBox(CAABBox &bbox)
{
bbox.setCenter(CVector::Null);
bbox.setHalfSize(CVector::Null);
}
// ***************************************************************************
void CTransform::enableCastShadowMap(bool state)
{
bool precState= canCastShadowMap();
if(modelCanCastShadowMap())
setStateFlag(IsFinalShadowMapCaster, state);
else
setStateFlag(IsFinalShadowMapCaster, false);
// if just enabled, create the shadowMap
if(canCastShadowMap() && !precState)
{
createShadowMap();
// The user must have created it.
nlassert(getShadowMap());
}
// if just disabled, free ressource
else if(!canCastShadowMap() && precState)
{
deleteShadowMap();
}
}
// ***************************************************************************
void CTransform::forceCompute()
{
// if father is a skeleton, force to compute the bone we are sticked to
if (_FatherSkeletonModel)
{
_FatherSkeletonModel->forceComputeBone(_FatherBoneId);
}
else
{
// force to compute the father
if (_HrcParent)
{
_HrcParent->forceCompute();
}
}
// compute
update();
updateWorldMatrixFromFather();
}
// ***************************************************************************
void CTransform::setForceClipRoot(bool forceClipRoot)
{
if (forceClipRoot == (getStateFlag(ForceClipRoot) != 0)) return;
if (forceClipRoot)
{
// unlink from previous father and link to the root
hrcUnlink();
if (_OwnerScene)
{
_OwnerScene->getRoot()->hrcLinkSon(this);
}
setClusterSystem(NULL);
}
setStateFlag(ForceClipRoot, forceClipRoot);
}
// ***************************************************************************
UTransform *CTransform::buildMatchingUserInterfaceObject()
{
return new UTransform(this);
}
// ***************************************************************************
void CTransform::setShadowMapDirectionZThreshold(float zthre)
{
clamp(zthre, -1.f, 1.f);
_ShadowMapDirectionZThreshold= zthre;
}
// ***************************************************************************
void CTransform::setShadowMapMaxDepth(float depth)
{
depth= max(0.f, depth);
_ShadowMapMaxDepth= depth;
}
}