mirror of
https://port.numenaute.org/aleajactaest/khanat-opennel-code.git
synced 2024-12-02 08:35:59 +00:00
2346 lines
66 KiB
C++
2346 lines
66 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/misc/hierarchical_timer.h"
|
|
#include "nel/3d/skeleton_model.h"
|
|
#include "nel/3d/hrc_trav.h"
|
|
#include "nel/3d/clip_trav.h"
|
|
#include "nel/3d/anim_detail_trav.h"
|
|
#include "nel/3d/render_trav.h"
|
|
#include "nel/3d/skeleton_shape.h"
|
|
#include "nel/3d/scene.h"
|
|
#include "nel/3d/lod_character_manager.h"
|
|
#include "nel/3d/lod_character_shape.h"
|
|
#include "nel/misc/rgba.h"
|
|
#include "nel/misc/aabbox.h"
|
|
#include "nel/3d/vertex_stream_manager.h"
|
|
#include "nel/3d/mesh_base_instance.h"
|
|
#include "nel/3d/async_texture_manager.h"
|
|
|
|
|
|
using namespace std;
|
|
using namespace NLMISC;
|
|
|
|
namespace NL3D
|
|
{
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::registerBasic()
|
|
{
|
|
CScene::registerModel(SkeletonModelId, TransformShapeId, CSkeletonModel::creator);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
CTrackDefaultString CSkeletonModel::_DefaultSpawnScript;
|
|
|
|
|
|
// ***************************************************************************
|
|
IAnimatedValue *CSkeletonModel::getValue (uint valueId)
|
|
{
|
|
// what value ?
|
|
switch (valueId)
|
|
{
|
|
case SpawnScriptValue: return &_SpawnScript;
|
|
}
|
|
|
|
return CTransformShape::getValue(valueId);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
const char *CSkeletonModel::getValueName (uint valueId) const
|
|
{
|
|
// what value ?
|
|
switch (valueId)
|
|
{
|
|
case SpawnScriptValue: return getSpawnScriptValueName();
|
|
}
|
|
|
|
return CTransformShape::getValueName(valueId);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
ITrack *CSkeletonModel::getDefaultTrack (uint valueId)
|
|
{
|
|
// what value ?
|
|
switch (valueId)
|
|
{
|
|
case SpawnScriptValue: return &_DefaultSpawnScript;
|
|
}
|
|
|
|
return CTransformShape::getDefaultTrack(valueId);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::registerToChannelMixer(CChannelMixer *chanMixer, const std::string &prefix)
|
|
{
|
|
/* add the Spawn Script value. The animation is evaluated at detail time, as the script evaluation.
|
|
This seems dangerous (create and delete models at evalDetail time) but works because:
|
|
- deletedModels() in a current CScene::render() are delayed to end of render()
|
|
and are "temp removed" from the render trav
|
|
- createdModels() in CSkeletonSpawnScript are delayed to the end of CScene::render()
|
|
- if a skeleton is not visible, or in LOD form, then its sticked SpawnedModels are not visible too,
|
|
whether or not they are too old regarding the animation time
|
|
*/
|
|
_SpawnScriptChannelId= addValue(chanMixer, SpawnScriptValue, OwnerBit, prefix, true);
|
|
|
|
// add standard
|
|
CTransformShape::registerToChannelMixer(chanMixer, prefix);
|
|
|
|
// Add any bones.
|
|
for(uint i=0;i<Bones.size();i++)
|
|
{
|
|
// append bonename.
|
|
Bones[i].registerToChannelMixer(chanMixer, prefix + Bones[i].getBoneName() + ".");
|
|
}
|
|
|
|
}
|
|
|
|
// ***************************************************************************
|
|
CSkeletonModel::CSkeletonModel()
|
|
{
|
|
IAnimatable::resize(AnimValueLast);
|
|
_DisplayedAsLodCharacter= false;
|
|
_LodCharacterDistance= 0;
|
|
_OOLodCharacterDistance= 0;
|
|
_IsEnableLOD=true;
|
|
|
|
_DefaultMRMSetup= true;
|
|
|
|
_SkinToRenderDirty= false;
|
|
|
|
_CLodVertexAlphaDirty= true;
|
|
|
|
_LodEmit= CRGBA::Black;
|
|
|
|
// Inform the transform that I am a skeleton
|
|
CTransform::setIsSkeleton(true);
|
|
|
|
// By default, no skins, hence, impossible to have transparent pass. But opaque pass is always possible
|
|
// because of CLod rendering
|
|
setOpacity(true);
|
|
setTransparency(false);
|
|
|
|
// AnimDetail behavior: Must be traversed in AnimDetail, even if no channel mixer registered
|
|
CTransform::setIsForceAnimDetail(true);
|
|
|
|
// LoadBalancing behavior. true because directly act on skins to draw all their MRM level
|
|
CTransform::setIsLoadbalancable(true);
|
|
|
|
// Lighting behavior. Lightable because skins/stickedObjects may surely need its LightContribution
|
|
CTransform::setIsLightable(true);
|
|
|
|
// Render behavior. Always renderable, because either render the skeleton as a CLod, or render skins
|
|
CTransform::setIsRenderable(true);
|
|
|
|
// build a bug-free level detail
|
|
buildDefaultLevelDetail();
|
|
|
|
// RenderFilter: We are a skeleton
|
|
_RenderFilterType= UScene::FilterSkeleton;
|
|
|
|
_AnimCtrlUsage= 0;
|
|
|
|
// ShadowMap
|
|
CTransform::setIsShadowMapCaster(true);
|
|
_ShadowMap= NULL;
|
|
|
|
// SpawnScript
|
|
_SSSWOPos= CVector::Null;
|
|
_SSSWODir= CVector::J;
|
|
_SpawnScriptChannelId= -1;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
CSkeletonModel::~CSkeletonModel()
|
|
{
|
|
// if initModel() called
|
|
if(getOwnerScene())
|
|
{
|
|
// release the _SpawnScriptEvaluator (delete any instance sticked)
|
|
_SpawnScriptEvaluator.release(getOwnerScene());
|
|
|
|
// remove from scene
|
|
getOwnerScene()->eraseSkeletonModelToList(_ItSkeletonInScene);
|
|
}
|
|
|
|
|
|
// detach skeleton sons from skins.
|
|
while(_Skins.begin()!=_Skins.end())
|
|
{
|
|
detachSkeletonSon(*_Skins.begin());
|
|
}
|
|
|
|
// detach skeleton sons from sticked objects.
|
|
while(_StickedObjects.begin()!=_StickedObjects.end())
|
|
{
|
|
detachSkeletonSon(*_StickedObjects.begin());
|
|
}
|
|
|
|
// Free Lod instance
|
|
setLodCharacterShape(-1);
|
|
|
|
// delete the shadowMap
|
|
deleteShadowMap();
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::initModel()
|
|
{
|
|
// Link this skeleton to the CScene.
|
|
_ItSkeletonInScene= getOwnerScene()->appendSkeletonModelToList(this);
|
|
|
|
// Call base class
|
|
CTransformShape::initModel();
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::initBoneUsages()
|
|
{
|
|
// reset all to 0.
|
|
_BoneUsage.resize(Bones.size());
|
|
for(uint i=0; i<_BoneUsage.size(); i++)
|
|
{
|
|
_BoneUsage[i].Usage= 0;
|
|
_BoneUsage[i].ForcedUsage= 0;
|
|
_BoneUsage[i].CLodForcedUsage= 0;
|
|
_BoneUsage[i].MustCompute= 0;
|
|
_BoneUsage[i].ValidBoneSkinMatrix= 0;
|
|
}
|
|
// reserve space for bone to compute
|
|
_BoneToCompute.reserve(Bones.size());
|
|
|
|
_BoneToComputeDirty= false;
|
|
_CurLod= 0;
|
|
_CurLodInterp= 1.f;
|
|
// Default is 0.5 meters.
|
|
_LodInterpMultiplier= 1.f / 0.5f;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::incBoneUsage(uint i, TBoneUsageType boneUsageType)
|
|
{
|
|
nlassert(i<_BoneUsage.size());
|
|
|
|
// Get ptr on according refCount
|
|
uint16 *usagePtr;
|
|
if(boneUsageType == UsageNormal)
|
|
usagePtr= &_BoneUsage[i].Usage;
|
|
else if(boneUsageType == UsageForced)
|
|
usagePtr= &_BoneUsage[i].ForcedUsage;
|
|
else
|
|
usagePtr= &_BoneUsage[i].CLodForcedUsage;
|
|
|
|
// If the bone was not used before, must update MustCompute.
|
|
if(*usagePtr==0)
|
|
_BoneToComputeDirty= true;
|
|
|
|
// Inc the refCount of the bone.
|
|
nlassert(*usagePtr<65535);
|
|
(*usagePtr)++;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::decBoneUsage(uint i, TBoneUsageType boneUsageType)
|
|
{
|
|
nlassert(i<_BoneUsage.size());
|
|
|
|
// Get ptr on according refCount
|
|
uint16 *usagePtr;
|
|
if(boneUsageType == UsageNormal)
|
|
usagePtr= &_BoneUsage[i].Usage;
|
|
else if(boneUsageType == UsageForced)
|
|
usagePtr= &_BoneUsage[i].ForcedUsage;
|
|
else
|
|
usagePtr= &_BoneUsage[i].CLodForcedUsage;
|
|
|
|
// If the bone was used before (and now won't be more), must update MustCompute.
|
|
if(*usagePtr==1)
|
|
_BoneToComputeDirty= true;
|
|
|
|
// Inc the refCount of the bone.
|
|
nlassert(*usagePtr>0);
|
|
(*usagePtr)--;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::flagBoneAndParents(uint32 boneId, std::vector<bool> &boneUsage) const
|
|
{
|
|
nlassert( boneUsage.size()==Bones.size() );
|
|
nlassert( boneId<Bones.size() );
|
|
|
|
// Flag this bone.
|
|
boneUsage[boneId]= true;
|
|
|
|
// if has father, flag it (recurs).
|
|
sint fatherId= Bones[boneId].getFatherId();
|
|
if(fatherId>=0)
|
|
flagBoneAndParents(fatherId, boneUsage);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::incForcedBoneUsageAndParents(uint i, bool forceCLod)
|
|
{
|
|
// inc forced.
|
|
incBoneUsage(i, forceCLod?UsageCLodForced:UsageForced );
|
|
|
|
// recurs to father
|
|
sint fatherId= Bones[i].getFatherId();
|
|
// if not a root bone...
|
|
if(fatherId>=0)
|
|
incForcedBoneUsageAndParents(fatherId, forceCLod);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::decForcedBoneUsageAndParents(uint i, bool forceCLod)
|
|
{
|
|
// dec forced
|
|
decBoneUsage(i, forceCLod?UsageCLodForced:UsageForced);
|
|
|
|
// recurs to father
|
|
sint fatherId= Bones[i].getFatherId();
|
|
// if not a root bone...
|
|
if(fatherId>=0)
|
|
decForcedBoneUsageAndParents(fatherId, forceCLod);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::updateBoneToCompute()
|
|
{
|
|
// If already computed, skip
|
|
if(!_BoneToComputeDirty)
|
|
return;
|
|
|
|
// get the channelMixer owned by CTransform.
|
|
CChannelMixer *chanMixer= getChannelMixer();
|
|
|
|
// Get Lod infos from skeletonShape
|
|
CSkeletonShape *skeShape= (CSkeletonShape*)(IShape*)Shape;
|
|
const CSkeletonShape::CLod &lod= skeShape->getLod(_CurLod);
|
|
|
|
// reset _BoneToCompute
|
|
_BoneToCompute.clear();
|
|
|
|
// For all bones
|
|
for(uint i=0; i<_BoneUsage.size(); i++)
|
|
{
|
|
// If we are in CLod mode
|
|
if(isDisplayedAsLodCharacter())
|
|
// don't compute the bone
|
|
_BoneUsage[i].MustCompute= 0;
|
|
else
|
|
{
|
|
// set MustCompute to non 0 if (Usage && Lod) || ForcedUsage;
|
|
_BoneUsage[i].MustCompute= (_BoneUsage[i].Usage & lod.ActiveBones[i]) | _BoneUsage[i].ForcedUsage;
|
|
}
|
|
// if CLodForcedUsage for the bone, it must be computed, whatever _DisplayedAsLodCharacter state
|
|
_BoneUsage[i].MustCompute|= _BoneUsage[i].CLodForcedUsage;
|
|
|
|
// If the bone must be computed (if !0)
|
|
if(_BoneUsage[i].MustCompute)
|
|
{
|
|
// lodEnable the channels of this bone
|
|
if(chanMixer)
|
|
Bones[i].lodEnableChannels(chanMixer, true);
|
|
|
|
// This bone is computed => take his valid boneSkinMatrix.
|
|
_BoneUsage[i].ValidBoneSkinMatrix= i;
|
|
|
|
// Append to the list to compute.
|
|
//-------
|
|
CBoneCompute bc;
|
|
bc.Bone= &Bones[i];
|
|
sint fatherId= Bones[i].getFatherId();
|
|
// if a root bone...
|
|
if(fatherId==-1)
|
|
bc.Father= NULL;
|
|
else
|
|
bc.Father= &Bones[fatherId];
|
|
// MustInterpolate??
|
|
bc.MustInterpolate= false;
|
|
const CSkeletonShape::CLod *lodNext= NULL;
|
|
// if a lod exist after current lod, and if lod interpolation enabled
|
|
if( _CurLod < skeShape->getNumLods()-1 && _LodInterpMultiplier>0 )
|
|
{
|
|
// get next lod.
|
|
lodNext= &skeShape->getLod(_CurLod+1);
|
|
// Lod interpolation on this bone ?? only if at next lod, the bone is disabled.
|
|
// And only if it is not enabed because of a "Forced reason"
|
|
// Must also have a father, esle can't interpolate.
|
|
if(lodNext->ActiveBones[i]==0 && _BoneUsage[i].ForcedUsage==0 && _BoneUsage[i].CLodForcedUsage==0
|
|
&& bc.Father)
|
|
bc.MustInterpolate= true;
|
|
}
|
|
// append
|
|
_BoneToCompute.push_back(bc);
|
|
}
|
|
else
|
|
{
|
|
// lodDisable the channels of this bone
|
|
if(chanMixer)
|
|
Bones[i].lodEnableChannels(chanMixer, false);
|
|
|
|
// This bone is not computed => take the valid boneSkinMatrix of his father
|
|
sint fatherId= Bones[i].getFatherId();
|
|
if(fatherId<0)
|
|
// just take me, even if not computed.
|
|
_BoneUsage[i].ValidBoneSkinMatrix= i;
|
|
else
|
|
// NB: father ValidBoneSkinMatrix already computed because of the hierarchy order of Bones array.
|
|
_BoneUsage[i].ValidBoneSkinMatrix= _BoneUsage[fatherId].ValidBoneSkinMatrix;
|
|
}
|
|
}
|
|
|
|
// Enable SpawnScript animation only if we are not in CLod
|
|
if(_SpawnScriptChannelId>=0 && chanMixer)
|
|
chanMixer->lodEnableChannel(_SpawnScriptChannelId, !isDisplayedAsLodCharacter());
|
|
|
|
// computed
|
|
_BoneToComputeDirty= false;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
bool CSkeletonModel::isBoneComputed(uint boneId) const
|
|
{
|
|
if(boneId>=_BoneUsage.size())
|
|
return false;
|
|
else
|
|
return _BoneUsage[boneId].MustCompute!=0 && isClipVisible();
|
|
}
|
|
|
|
|
|
// struct used by CSkeletonModel::forceComputeBone only
|
|
struct CForceComputeBoneInfo
|
|
{
|
|
CTransform *Transform;
|
|
uint StickBoneID; // if the transform is a skeleton, gives the bone to which child of interest is sticked
|
|
};
|
|
|
|
// ***************************************************************************
|
|
bool CSkeletonModel::forceComputeBone(uint boneId)
|
|
{
|
|
if(boneId >= _BoneUsage.size()) return false;
|
|
// build list of ancestor, then must build
|
|
std::vector<CForceComputeBoneInfo> ancestors;
|
|
// count the number of ancestors (to avoid unwanted alloc with vector)
|
|
uint numAncestors = 1;
|
|
CTransform *currTransform = this;
|
|
for(;;)
|
|
{
|
|
currTransform = currTransform->_HrcParent ? currTransform->_HrcParent : currTransform->_FatherSkeletonModel; // find father transform (maybe a skeleton or a std transform)
|
|
if (!currTransform) break; // root reached ?
|
|
++ numAncestors;
|
|
}
|
|
ancestors.reserve(numAncestors);
|
|
// build list of ancestor
|
|
currTransform = this;
|
|
uint currStickBone = boneId;
|
|
for(;;)
|
|
{
|
|
// if curr transform is a skeleton, animate all bone from stick bone to the root bone
|
|
if (currTransform->isSkeleton())
|
|
{
|
|
if (_ChannelMixer)
|
|
{
|
|
CSkeletonModel *skel = static_cast<CSkeletonModel *>(currTransform);
|
|
nlassert(boneId < skel->_BoneUsage.size());
|
|
nlassert(currStickBone < skel->Bones.size());
|
|
sint currBoneIndex = currStickBone;
|
|
// force channel mixer to eval for that bone
|
|
while (currBoneIndex != -1)
|
|
{
|
|
nlassert((uint) currBoneIndex < skel->Bones.size());
|
|
skel->Bones[currBoneIndex].forceAnimate(*_ChannelMixer);
|
|
currBoneIndex = skel->Bones[currBoneIndex].getFatherId();
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// update stickBone ID (if father is a skeleton)
|
|
currStickBone = _FatherBoneId;
|
|
}
|
|
CForceComputeBoneInfo fcbi;
|
|
fcbi.StickBoneID = currStickBone;
|
|
fcbi.Transform = currTransform;
|
|
ancestors.push_back(fcbi);
|
|
currTransform = currTransform->_HrcParent ? currTransform->_HrcParent : currTransform->_FatherSkeletonModel; // find father transform (maybe a skeleton or a std transform)
|
|
if (!currTransform) break; // root reached ?
|
|
}
|
|
// bones must be recomputed from father bone to sons, so must traverse bones until root is reached to retrieve correct ordering
|
|
CBone *OrderedBone[MaxNumBones];
|
|
//
|
|
const CMatrix *parentWorldMatrix = &CMatrix::Identity;
|
|
for(std::vector<CForceComputeBoneInfo>::reverse_iterator it = ancestors.rbegin(); it != ancestors.rend(); ++it)
|
|
{
|
|
// update world matrix (NB : the call to getmatrix will update the local matrix)
|
|
it->Transform->setWorldMatrix(*parentWorldMatrix * it->Transform->getMatrix());
|
|
if (it->Transform->isSkeleton())
|
|
{
|
|
CSkeletonModel *skel = static_cast<CSkeletonModel *>(it->Transform);
|
|
// reorder bones
|
|
uint numBones = 0;
|
|
nlassert(it->StickBoneID < skel->Bones.size());
|
|
sint currBoneIndex = it->StickBoneID;
|
|
nlassert(currBoneIndex != -1);
|
|
do
|
|
{
|
|
nlassert(numBones < MaxNumBones);
|
|
nlassert((uint) currBoneIndex < skel->Bones.size());
|
|
OrderedBone[numBones] = &skel->Bones[currBoneIndex];
|
|
currBoneIndex = OrderedBone[numBones]->getFatherId();
|
|
++ numBones;
|
|
}
|
|
while (currBoneIndex != -1);
|
|
const CMatrix &modelWorldMatrix = it->Transform->getWorldMatrix();
|
|
// recompute bones
|
|
CBone *fatherBone = NULL;
|
|
while (numBones--)
|
|
{
|
|
OrderedBone[numBones]->compute(fatherBone, modelWorldMatrix, NULL);
|
|
fatherBone = OrderedBone[numBones];
|
|
}
|
|
parentWorldMatrix = &(OrderedBone[0]->getWorldMatrix());
|
|
}
|
|
else
|
|
{
|
|
parentWorldMatrix = &it->Transform->getWorldMatrix();
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
|
|
// ***************************************************************************
|
|
const NLMISC::CMatrix &CSkeletonModel::getActiveBoneSkinMatrix(uint boneId) const
|
|
{
|
|
// Get me or first father with MustCompute==true.
|
|
uint validBoneId= _BoneUsage[boneId].ValidBoneSkinMatrix;
|
|
// return his WorldMatrix.
|
|
return Bones[validBoneId].getBoneSkinMatrix();
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
bool CSkeletonModel::bindSkin(CTransform *mi)
|
|
{
|
|
nlassert(mi);
|
|
if( !mi->isSkinnable() )
|
|
return false;
|
|
|
|
// try to detach this object from any skeleton first (possibly me).
|
|
if(mi->_FatherSkeletonModel)
|
|
mi->_FatherSkeletonModel->detachSkeletonSon(mi);
|
|
|
|
// Then Add me.
|
|
_Skins.insert(mi);
|
|
|
|
// advert skin transform it is skinned.
|
|
mi->_FatherSkeletonModel= this;
|
|
// setApplySkin() use _FatherSkeletonModel to computeBonesId() and to update current skeleton bone usage.
|
|
mi->setApplySkin(true);
|
|
|
|
|
|
// Unlink the Skin from Hrc and clip, because SkeletonModel now does the job for him.
|
|
// First ensure that the transform is not frozen (unlink from some quadGrids etc...)
|
|
mi->unfreezeHRC();
|
|
// then never re-parse in validateList/Hrc/Clip
|
|
mi->unlinkFromUpdateList();
|
|
mi->hrcUnlink();
|
|
// ClipTrav is a graph, so must unlink from ALL olds models.
|
|
mi->clipUnlinkFromAll();
|
|
// Ensure flag is correct
|
|
mi->_ClipLinkedInSonsOfAncestorSkeletonModelGroup= false;
|
|
|
|
|
|
// must recompute lod vertex alpha when LodCharacter used
|
|
dirtLodVertexAlpha();
|
|
// must recompute list of skins.
|
|
dirtSkinRenderLists();
|
|
|
|
// Ok, skinned
|
|
return true;
|
|
}
|
|
// ***************************************************************************
|
|
void CSkeletonModel::stickObject(CTransform *mi, uint boneId)
|
|
{
|
|
// by default don't force display of "mi" if the skeleton become in CLod state
|
|
stickObjectEx(mi, boneId, false);
|
|
}
|
|
// ***************************************************************************
|
|
void CSkeletonModel::stickObjectEx(CTransform *mi, uint boneId, bool forceCLod)
|
|
{
|
|
nlassert(mi);
|
|
|
|
// if "mi" is a skeleton, forceCLod must be true, for correct animation purpose
|
|
if(dynamic_cast<CSkeletonModel*>(mi))
|
|
forceCLod= true;
|
|
|
|
// try to detach this object from any skeleton first (possibly me).
|
|
if(mi->_FatherSkeletonModel)
|
|
mi->_FatherSkeletonModel->detachSkeletonSon(mi);
|
|
|
|
// Then Add me.
|
|
_StickedObjects.insert(mi);
|
|
// increment the refCount usage of the bone
|
|
incForcedBoneUsageAndParents(boneId, forceCLod);
|
|
|
|
// advert transform of its sticked state.
|
|
mi->_FatherSkeletonModel= this;
|
|
mi->_FatherBoneId= boneId;
|
|
// advert him if it is "ForceCLod" sticked
|
|
mi->_ForceCLodSticked= forceCLod;
|
|
|
|
// link correctly Hrc only. ClipTrav grah updated in Hrc traversal.
|
|
hrcLinkSon( mi );
|
|
|
|
// must recompute lod vertex alpha when LodCharacter used
|
|
dirtLodVertexAlpha();
|
|
}
|
|
// ***************************************************************************
|
|
void CSkeletonModel::detachSkeletonSon(CTransform *tr)
|
|
{
|
|
nlassert(tr);
|
|
|
|
// If the instance is not binded/sticked to the skeleton, exit.
|
|
if(tr->_FatherSkeletonModel!=this)
|
|
return;
|
|
|
|
// try to erase from StickObject.
|
|
_StickedObjects.erase(tr);
|
|
// try to erase from Skins.
|
|
_Skins.erase(tr);
|
|
|
|
// If the instance is not skinned, then it is sticked
|
|
bool wasSkinned= tr->isSkinned()!=0;
|
|
if( !wasSkinned )
|
|
{
|
|
// Then decrement Bone Usage RefCount. Decrement from CLodForcedUsage if was sticked with forceCLod==true
|
|
decForcedBoneUsageAndParents(tr->_FatherBoneId, tr->_ForceCLodSticked);
|
|
}
|
|
else
|
|
{
|
|
// it is skinned, advert the skinning is no more OK.
|
|
// setApplySkin() use _FatherSkeletonModel to update current skeleton bone usage.
|
|
tr->setApplySkin(false);
|
|
}
|
|
|
|
// advert transform it is no more sticked/skinned.
|
|
tr->_FatherSkeletonModel= NULL;
|
|
tr->_ForceCLodSticked= false;
|
|
|
|
// link correctly Hrc / Clip / UpdateList...
|
|
getOwnerScene()->getRoot()->hrcLinkSon( tr );
|
|
if( !wasSkinned )
|
|
{
|
|
// No-op. ClipTrav graph/UpdateList updated in Hrc traversal.
|
|
}
|
|
else
|
|
{
|
|
// Skin case: must do the Job here.
|
|
// Update ClipTrav here.
|
|
getOwnerScene()->getRoot()->clipAddChild(tr);
|
|
// Must re-add to the update list.
|
|
tr->linkToUpdateList();
|
|
}
|
|
|
|
|
|
// must recompute lod vertex alpha when LodCharacter used
|
|
dirtLodVertexAlpha();
|
|
// must recompute list of skins if was skinned
|
|
if( wasSkinned )
|
|
dirtSkinRenderLists();
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
sint32 CSkeletonModel::getBoneIdByName(const std::string &name) const
|
|
{
|
|
CSkeletonShape *shp= safe_cast<CSkeletonShape*>((IShape*)Shape);
|
|
return shp->getBoneIdByName(name);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::setInterpolationDistance(float dist)
|
|
{
|
|
dist= std::max(0.f, dist);
|
|
// disable interpolation?
|
|
if(dist==0)
|
|
_LodInterpMultiplier= 0.f;
|
|
else
|
|
_LodInterpMultiplier= 1.f / dist;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
float CSkeletonModel::getInterpolationDistance() const
|
|
{
|
|
if(_LodInterpMultiplier==0)
|
|
return 0.f;
|
|
else
|
|
return 1.f / _LodInterpMultiplier;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::traverseAnimDetail()
|
|
{
|
|
CSkeletonShape *skeShape= ((CSkeletonShape*)(IShape*)Shape);
|
|
|
|
/* NB: If "this" skeleton has an AncestorSkeletonModel displayed but "this" skeleton is clipped,
|
|
it will be still animDetailed.
|
|
So its possible sticked sons will be displayed with correct WorldMatrix (if not themselves clipped).
|
|
*/
|
|
|
|
/* If the Root Skeleton Model (ie me or my AncestorSM) is asked to render a ShadowMap, BUT I am
|
|
in CLod Form (and visible in HRC else won't be rendered in shadowMap...), then temporarly
|
|
Avoid CLod!! To really compute the bones for this frame only.
|
|
*/
|
|
bool tempAvoidCLod= false;
|
|
bool genShadow;
|
|
if(_AncestorSkeletonModel)
|
|
genShadow= _AncestorSkeletonModel->isGeneratingShadowMap();
|
|
else
|
|
genShadow= isGeneratingShadowMap();
|
|
// do the test.
|
|
if(genShadow && isDisplayedAsLodCharacter() && isHrcVisible() )
|
|
{
|
|
tempAvoidCLod= true;
|
|
// Disable it just the time of this AnimDetail
|
|
setDisplayLodCharacterFlag(false);
|
|
}
|
|
|
|
|
|
// Update Lod, and animate.
|
|
//===============
|
|
|
|
/*
|
|
CTransformShape::traverseAnimDetail() is torn in 2 here because
|
|
channels may be enabled/disabled by updateBoneToCompute()
|
|
*/
|
|
|
|
// First update Skeleton WorldMatrix (case where the skeleton is sticked).
|
|
CTransform::updateWorldMatrixFromFather();
|
|
// get dist from camera.
|
|
float dist= (getWorldMatrix().getPos() - getOwnerScene()->getClipTrav().CamPos).norm();
|
|
// Use dist to get current lod to use for this skeleton
|
|
uint newLod= skeShape->getLodForDistance( dist );
|
|
if(!_IsEnableLOD) newLod = 0;
|
|
if(newLod != _CurLod)
|
|
{
|
|
// set new lod to use.
|
|
_CurLod= newLod;
|
|
// dirt the skeleton.
|
|
_BoneToComputeDirty= true;
|
|
}
|
|
|
|
// If needed, let's know which bone has to be computed, and enable / disable (lod) channels in channelMixer.
|
|
updateBoneToCompute();
|
|
|
|
// Animate skeleton.
|
|
CTransformShape::traverseAnimDetailWithoutUpdateWorldMatrix();
|
|
|
|
// If in normal mode, must update the SpawnScript
|
|
if(!isDisplayedAsLodCharacter())
|
|
{
|
|
_SpawnScriptEvaluator.evaluate(this);
|
|
}
|
|
|
|
// Prepare Lod Bone interpolation.
|
|
//===============
|
|
|
|
float lodBoneInterp;
|
|
const CSkeletonShape::CLod *lodNext= NULL;
|
|
// if a lod exist after current lod, and if lod interpolation enabled
|
|
if( _CurLod < skeShape->getNumLods()-1 && _LodInterpMultiplier>0 && _IsEnableLOD)
|
|
{
|
|
// get next lod.
|
|
lodNext= &skeShape->getLod(_CurLod+1);
|
|
// get interp value to next.
|
|
lodBoneInterp= (lodNext->Distance - dist) * _LodInterpMultiplier;
|
|
NLMISC::clamp(lodBoneInterp, 0.f, 1.f);
|
|
// if still 1, keep cur matrix => disable interpolation
|
|
if(lodBoneInterp==1.f)
|
|
lodNext=NULL;
|
|
}
|
|
// else, no interpolation
|
|
else
|
|
{
|
|
lodBoneInterp=1.f;
|
|
}
|
|
_CurLodInterp= lodBoneInterp;
|
|
|
|
|
|
// Compute bones
|
|
//===============
|
|
|
|
|
|
// test if bones must be updated. either if animation change or if BoneUsage change.
|
|
// Retrieve the WorldMatrix of the current CTransformShape.
|
|
const CMatrix &modelWorldMatrix= getWorldMatrix();
|
|
|
|
// must test / update the hierarchy of Bones.
|
|
// Since they are orderd in depth-first order, we are sure that parent are computed before sons.
|
|
uint numBoneToCompute= (uint)_BoneToCompute.size();
|
|
CSkeletonModel::CBoneCompute *pBoneCompute= numBoneToCompute? &_BoneToCompute[0] : NULL;
|
|
// traverse only bones which need to be computed
|
|
for(;numBoneToCompute>0;numBoneToCompute--, pBoneCompute++)
|
|
{
|
|
// compute the bone with his father, if any
|
|
pBoneCompute->Bone->compute( pBoneCompute->Father, modelWorldMatrix, this);
|
|
|
|
// Lod interpolation on this bone .. only if interp is enabled now, and if bone wants it
|
|
if(lodNext && pBoneCompute->MustInterpolate)
|
|
{
|
|
// interpolate with my father matrix.
|
|
const CMatrix &fatherMatrix= pBoneCompute->Father->getBoneSkinMatrix();
|
|
pBoneCompute->Bone->interpolateBoneSkinMatrix(fatherMatrix, lodBoneInterp);
|
|
}
|
|
}
|
|
|
|
IAnimatable::clearFlag(CSkeletonModel::OwnerBit);
|
|
|
|
// Sticked Objects:
|
|
// they will update their WorldMatrix after, because of the AnimDetail traverse scheme:
|
|
// traverse visible Clip models, and if skeleton, traverse Hrc sons.
|
|
|
|
|
|
// Restore the Initial CLod flag if needed (see above)
|
|
if(tempAvoidCLod)
|
|
{
|
|
setDisplayLodCharacterFlag(true);
|
|
}
|
|
|
|
|
|
// Update Animated Skins.
|
|
//===============
|
|
for(uint i=0;i<_AnimDetailSkins.size();i++)
|
|
{
|
|
// traverse it. NB: updateWorldMatrixFromFather() is called but no-op because isSkinned()
|
|
_AnimDetailSkins[i]->traverseAnimDetail();
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::computeAllBones(const CMatrix &modelWorldMatrix)
|
|
{
|
|
// must test / update the hierarchy of Bones.
|
|
// Since they are orderd in depth-first order, we are sure that parent are computed before sons.
|
|
for(uint i=0;i<Bones.size();i++)
|
|
{
|
|
sint fatherId= Bones[i].getFatherId();
|
|
// if a root bone...
|
|
if(fatherId==-1)
|
|
// Compute root bone worldMatrix. Do not allow special AnimCtrl
|
|
Bones[i].compute( NULL, modelWorldMatrix, NULL);
|
|
else
|
|
// Compute bone worldMatrix. Do not allow special AnimCtrl
|
|
Bones[i].compute( &Bones[fatherId], modelWorldMatrix, NULL);
|
|
}
|
|
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::setLodCharacterDistance(float dist)
|
|
{
|
|
_LodCharacterDistance= max(dist, 0.f);
|
|
if(_LodCharacterDistance>0)
|
|
_OOLodCharacterDistance= 1.0f/_LodCharacterDistance;
|
|
else
|
|
_OOLodCharacterDistance= 0;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::setLodCharacterShape(sint shapeId)
|
|
{
|
|
// get a ptr on the scene which owns us, and so on the lodManager.
|
|
CScene *scene= getOwnerScene();
|
|
CLodCharacterManager *mngr= scene->getLodCharacterManager();
|
|
|
|
// if mngr not setuped, noop (lod not possible).
|
|
if(!mngr)
|
|
return;
|
|
|
|
// If a shape was setup, free the instance
|
|
if(_CLodInstance.ShapeId>=0)
|
|
{
|
|
mngr->releaseInstance(_CLodInstance);
|
|
_CLodInstance.ShapeId= -1;
|
|
}
|
|
|
|
// assign
|
|
_CLodInstance.ShapeId= shapeId;
|
|
|
|
// if a real shape is setuped, alloc an instance
|
|
if(_CLodInstance.ShapeId>=0)
|
|
{
|
|
mngr->initInstance(_CLodInstance);
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::computeLodTexture()
|
|
{
|
|
// is lod setuped
|
|
if(_CLodInstance.ShapeId<0)
|
|
return;
|
|
|
|
// get a ptr on the scene which owns us, and so on the lodManager.
|
|
CScene *scene= getOwnerScene();
|
|
CLodCharacterManager *mngr= scene->getLodCharacterManager();
|
|
// mngr must be setuped since shape Id is >-1
|
|
nlassert(mngr);
|
|
/* Get the asyncTextureManager. This is a Hack. We use the AsyncTextureManager to store very low version of Textures
|
|
(kept in DXTC1 format for minimum memory overhead).
|
|
HENCE Lod Texture can work only with Async Textured instances!!
|
|
*/
|
|
CAsyncTextureManager *asyncMngr= scene->getAsyncTextureManager();
|
|
// if not setuped, cancel
|
|
if(!asyncMngr)
|
|
return;
|
|
|
|
|
|
// **** start process. If cannot (TextureId==no more texture space), just quit.
|
|
if(!mngr->startTextureCompute(_CLodInstance))
|
|
return;
|
|
uint maxNumBmpToReset= 0;
|
|
|
|
// **** For all skins which have a LodTexture setuped
|
|
ItTransformSet it= _Skins.begin();
|
|
for(;it!=_Skins.end();it++)
|
|
{
|
|
// the skin should be a meshBaseInstance setuped to asyncTexturing
|
|
CMeshBaseInstance *mbi= dynamic_cast<CMeshBaseInstance*>(*it);
|
|
if(mbi && mbi->getAsyncTextureMode() && mbi->Shape)
|
|
{
|
|
CMeshBase *mb= (CMeshBase*)(IShape*)(mbi->Shape);
|
|
// get the LodTexture info of this shape.
|
|
const CLodCharacterTexture *lodText= mb->getLodCharacterTexture();
|
|
// if setuped
|
|
if(lodText)
|
|
{
|
|
// Ok, compute influence of this instance on the Lod.
|
|
|
|
// ---- Build all bmps of the instance with help of the asyncTextureManager
|
|
uint numMats= (uint)mbi->Materials.size();
|
|
// 256 materials possibles for the lod Manager
|
|
numMats= min(numMats, 256U);
|
|
// for endTexturecompute
|
|
maxNumBmpToReset= max(maxNumBmpToReset, numMats);
|
|
// process each materials
|
|
for(uint i=0;i<numMats;i++)
|
|
{
|
|
// get the manager bitmap to write to
|
|
CLodCharacterTmpBitmap &dstBmp= mngr->getTmpBitmap(uint8(i));
|
|
|
|
// if the material stage 0 is not textured, or has not a valid async id, build the bitmap with a color.
|
|
sint asyncTextId= mbi->getAsyncTextureId(i,0);
|
|
const CBitmap *coarseBitmap= NULL;
|
|
if(asyncTextId!=-1)
|
|
{
|
|
// get it from async manager
|
|
coarseBitmap= asyncMngr->getCoarseBitmap(asyncTextId);
|
|
}
|
|
|
|
// So if we have no bmp here, build with material color, else build a texture
|
|
if(!coarseBitmap)
|
|
{
|
|
dstBmp.build(mbi->Materials[i].getDiffuse());
|
|
}
|
|
else
|
|
{
|
|
dstBmp.build(*coarseBitmap);
|
|
}
|
|
}
|
|
|
|
// ---- add the lodTextureInfo to the current texture computed
|
|
mngr->addTextureCompute(_CLodInstance, *lodText);
|
|
}
|
|
}
|
|
}
|
|
|
|
// **** compile the process
|
|
mngr->endTextureCompute(_CLodInstance, maxNumBmpToReset);
|
|
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::setLodCharacterAnimId(uint animId)
|
|
{
|
|
_CLodInstance.AnimId= animId;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::setLodCharacterAnimTime(TGlobalAnimationTime time)
|
|
{
|
|
_CLodInstance.AnimTime= time;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::setLodCharacterWrapMode(bool wrapMode)
|
|
{
|
|
_CLodInstance.WrapMode= wrapMode;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
float CSkeletonModel::computeDisplayLodCharacterPriority() const
|
|
{
|
|
// if enabled
|
|
if(_LodCharacterDistance>0 && _CLodInstance.ShapeId>=0)
|
|
{
|
|
CVector globalPos;
|
|
|
|
// Get object position, test visibility;
|
|
// If has a skeleton ancestor, take his world position instead, because ours is invalid.
|
|
if( _AncestorSkeletonModel != NULL)
|
|
{
|
|
// if the ancestore is clipped, quit
|
|
if( !_AncestorSkeletonModel->isClipVisible() )
|
|
return 0;
|
|
// take ancestor world position
|
|
globalPos= _AncestorSkeletonModel->getWorldMatrix().getPos();
|
|
}
|
|
else
|
|
{
|
|
// if the skeleton is clipped, quit
|
|
if( !isClipVisible() )
|
|
return 0;
|
|
// take our world position
|
|
globalPos= getWorldMatrix().getPos();
|
|
}
|
|
|
|
// compute distance from camera.
|
|
float dist= (getOwnerScene()->getClipTrav().CamPos - globalPos).norm();
|
|
|
|
// compute priority
|
|
return dist*_OOLodCharacterDistance;
|
|
}
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::setDisplayLodCharacterFlag(bool displayCLod)
|
|
{
|
|
// if enabled
|
|
if(_LodCharacterDistance>0 && _CLodInstance.ShapeId>=0)
|
|
{
|
|
// If the flag has changed since last frame, must recompute bone Usage.
|
|
if(_DisplayedAsLodCharacter != displayCLod)
|
|
_BoneToComputeDirty= true;
|
|
|
|
// set new state
|
|
_DisplayedAsLodCharacter= displayCLod;
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::traverseRender()
|
|
{
|
|
H_AUTO( NL3D_Skeleton_Render );
|
|
|
|
// render as CLod, or render Skins.
|
|
if(isDisplayedAsLodCharacter())
|
|
renderCLod();
|
|
else
|
|
renderSkins();
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::computeCLodVertexAlpha(CLodCharacterManager *mngr)
|
|
{
|
|
// if shape id set.
|
|
if(_CLodInstance.ShapeId<0)
|
|
return;
|
|
// get the lod shape,a nd check exist in the manager
|
|
const CLodCharacterShape *lodShape= mngr->getShape(_CLodInstance.ShapeId);
|
|
if(lodShape)
|
|
{
|
|
// start process.
|
|
//-----------------
|
|
lodShape->startBoneAlpha(_CLodInstance.VertexAlphas);
|
|
|
|
// build an Id map, from Skeleton Ids to the lodShapes ids. (because may be differents)
|
|
static vector<sint> boneMap;
|
|
// reset to -1 (ie not found)
|
|
boneMap.clear();
|
|
boneMap.resize(Bones.size(), -1);
|
|
uint i;
|
|
// for all skeletons bones.
|
|
for(i=0; i<boneMap.size(); i++)
|
|
{
|
|
boneMap[i]= lodShape->getBoneIdByName(Bones[i].getBoneName());;
|
|
}
|
|
|
|
// Parse all skins
|
|
//-----------------
|
|
ItTransformSet it;
|
|
for(it= _Skins.begin(); it!=_Skins.end(); it++)
|
|
{
|
|
CTransform *skin= *it;
|
|
|
|
// get array of bone used for this skin.
|
|
const vector<sint32> *skinUsage= skin->getSkinBoneUsage();
|
|
// check correct skin
|
|
if(skinUsage)
|
|
{
|
|
// For all bones used
|
|
for(uint i=0; i<skinUsage->size(); i++)
|
|
{
|
|
// the id in the vector point to a bone in the skeleton. Hence use the boneMap to translate it
|
|
// in the lodShape ids.
|
|
sint idInLod= boneMap[(*skinUsage)[i]];
|
|
// only if id found in the lod shape
|
|
if(idInLod>=0)
|
|
// add color to this bone.
|
|
lodShape->addBoneAlpha(idInLod, _CLodInstance.VertexAlphas);
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
// Parse all sticked objects
|
|
//-----------------
|
|
for(it= _StickedObjects.begin(); it!=_StickedObjects.end(); it++)
|
|
{
|
|
CTransform *object= *it;
|
|
|
|
// get on which bone this object is linked.
|
|
// use the boneMap to translate id to lodShape id.
|
|
sint idInLod= boneMap[object->_FatherBoneId];
|
|
|
|
// only if id found in the lod shape
|
|
if(idInLod>=0)
|
|
// add color to this bone.
|
|
lodShape->addBoneAlpha(idInLod, _CLodInstance.VertexAlphas);
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::updateSkinRenderLists()
|
|
{
|
|
// If need to update array of skins to compute
|
|
if(_SkinToRenderDirty)
|
|
{
|
|
uint i;
|
|
|
|
_SkinToRenderDirty= false;
|
|
|
|
// Reset the LevelDetail.
|
|
_LevelDetail.MinFaceUsed= 0;
|
|
_LevelDetail.MaxFaceUsed= 0;
|
|
// If must follow default MRM setup from skins.
|
|
if(_DefaultMRMSetup)
|
|
{
|
|
_LevelDetail.DistanceCoarsest= 0;
|
|
_LevelDetail.DistanceMiddle= 0;
|
|
_LevelDetail.DistanceFinest= 0;
|
|
}
|
|
|
|
// reset Bone Sphere of skeleton.
|
|
static std::vector<bool> sphereEmpty;
|
|
sphereEmpty.clear();
|
|
sphereEmpty.resize(Bones.size(), true);
|
|
for(i=0;i<Bones.size();i++)
|
|
{
|
|
// Default sphere is centered on the bone pos, and has 0 radius.
|
|
Bones[i]._MaxSphere.Center= CVector::Null;
|
|
Bones[i]._MaxSphere.Radius= 0;
|
|
}
|
|
|
|
// Parse to count new size of the arrays, and to build MRM info, and bone Max sphere
|
|
uint opaqueSize= 0;
|
|
uint transparentSize= 0;
|
|
uint animDetailSize= 0;
|
|
ItTransformSet it;
|
|
// also test if can support fast intersection
|
|
_SupportFastIntersect= !_Skins.empty();
|
|
for(it= _Skins.begin();it!=_Skins.end();it++)
|
|
{
|
|
CTransform *skin= *it;
|
|
|
|
// If the skin is hidden, don't add it to any list!
|
|
if(skin->getVisibility()==CHrcTrav::Hide)
|
|
continue;
|
|
|
|
// if transparent, then must fill in transparent list.
|
|
if(skin->isTransparent())
|
|
transparentSize++;
|
|
// else may fill in opaquelist. NB: for optimisation, don't add in opaqueList
|
|
// if added to the transperent list (all materials are rendered)
|
|
else if(skin->isOpaque())
|
|
opaqueSize++;
|
|
|
|
// if animDetailable, then must fill list
|
|
if(skin->isAnimDetailable())
|
|
animDetailSize++;
|
|
|
|
// if the skin support MRM, then must update levelDetal number of faces
|
|
CTransformShape *trShape= dynamic_cast<CTransformShape*>(skin);
|
|
if(trShape)
|
|
{
|
|
const CMRMLevelDetail *skinLevelDetail= trShape->getMRMLevelDetail();
|
|
if(skinLevelDetail)
|
|
{
|
|
// Add Faces to the Skeleton level detail
|
|
_LevelDetail.MinFaceUsed+= skinLevelDetail->MinFaceUsed;
|
|
_LevelDetail.MaxFaceUsed+= skinLevelDetail->MaxFaceUsed;
|
|
// MRM Max skin setup.
|
|
if(_DefaultMRMSetup)
|
|
{
|
|
// Get the maximum distance setup (ie the one which degrades the less)
|
|
_LevelDetail.DistanceCoarsest= max(_LevelDetail.DistanceCoarsest, skinLevelDetail->DistanceCoarsest);
|
|
_LevelDetail.DistanceMiddle= max(_LevelDetail.DistanceMiddle, skinLevelDetail->DistanceMiddle);
|
|
_LevelDetail.DistanceFinest= max(_LevelDetail.DistanceFinest, skinLevelDetail->DistanceFinest);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Enlarge Bone BBox
|
|
const std::vector<sint32> *boneUsage= skin->getSkinBoneUsage();
|
|
const std::vector<NLMISC::CBSphere> *boneSphere= skin->getSkinBoneSphere();
|
|
if(boneUsage && boneSphere)
|
|
{
|
|
nlassert(boneUsage->size()==boneSphere->size());
|
|
for(i=0;i<boneUsage->size();i++)
|
|
{
|
|
const CBSphere &sphere= (*boneSphere)[i];
|
|
sint boneId= (*boneUsage)[i];
|
|
nlassert(boneId<(sint)Bones.size());
|
|
// if valid boneId, and sphere not empty (ie not -1 radius)
|
|
if(boneId>-1 && sphere.Radius>=0)
|
|
{
|
|
if(sphereEmpty[boneId])
|
|
{
|
|
sphereEmpty[boneId]= false;
|
|
Bones[boneId]._MaxSphere= sphere;
|
|
}
|
|
else
|
|
{
|
|
Bones[boneId]._MaxSphere.setUnion(Bones[boneId]._MaxSphere, sphere);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Whole skeleton model Support Fast intersection only if all
|
|
// displayed skins support skin intersection
|
|
_SupportFastIntersect= _SupportFastIntersect && skin->supportIntersectSkin();
|
|
}
|
|
|
|
// MRM Max skin setup.
|
|
if(_DefaultMRMSetup)
|
|
{
|
|
// compile LevelDetail.
|
|
if(_LevelDetail.MaxFaceUsed==0)
|
|
// build a bug-free level detail
|
|
buildDefaultLevelDetail();
|
|
else
|
|
_LevelDetail.compileDistanceSetup();
|
|
}
|
|
|
|
// alloc array.
|
|
_OpaqueSkins.clear();
|
|
_TransparentSkins.clear();
|
|
_AnimDetailSkins.clear();
|
|
_OpaqueSkins.resize(opaqueSize);
|
|
_TransparentSkins.resize(transparentSize);
|
|
_AnimDetailSkins.resize(animDetailSize);
|
|
|
|
// ReParse, to fill array.
|
|
uint opaqueId= 0;
|
|
uint transparentId= 0;
|
|
uint animDetailId= 0;
|
|
for(it= _Skins.begin();it!=_Skins.end();it++)
|
|
{
|
|
CTransform *skin= *it;
|
|
|
|
// If the skin is hidden, don't add it to any list!
|
|
if(skin->getVisibility()==CHrcTrav::Hide)
|
|
continue;
|
|
|
|
// if transparent, then must fill in transparent list.
|
|
if(skin->isTransparent())
|
|
{
|
|
nlassert(transparentId<transparentSize);
|
|
_TransparentSkins[transparentId++]= skin;
|
|
}
|
|
// else may fill in opaquelist. NB: for optimisation, don't add in opaqueList
|
|
// if added to the transperent list (all materials are rendered)
|
|
else if(skin->isOpaque())
|
|
{
|
|
nlassert(opaqueId<opaqueSize);
|
|
_OpaqueSkins[opaqueId++]= skin;
|
|
}
|
|
|
|
// if animDetailable, then must fill list
|
|
if(skin->isAnimDetailable())
|
|
{
|
|
nlassert(animDetailId<animDetailSize);
|
|
_AnimDetailSkins[animDetailId++]= skin;
|
|
}
|
|
}
|
|
|
|
// set the Transparency to the skeleton only if has at least one transparent skin
|
|
setTransparency( transparentSize>0 );
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::buildDefaultLevelDetail()
|
|
{
|
|
// Avoid divide by zero.
|
|
_LevelDetail.MinFaceUsed= 0;
|
|
_LevelDetail.MaxFaceUsed= 0;
|
|
_LevelDetail.DistanceFinest= 1;
|
|
_LevelDetail.DistanceMiddle= 2;
|
|
_LevelDetail.DistanceCoarsest= 3;
|
|
_LevelDetail.compileDistanceSetup();
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::renderCLod()
|
|
{
|
|
CRenderTrav &renderTrav= getOwnerScene()->getRenderTrav();
|
|
IDriver *drv= renderTrav.getDriver();
|
|
CScene *scene= getOwnerScene();
|
|
|
|
// Transparent pass? quit
|
|
if(!renderTrav.isCurrentPassOpaque())
|
|
return;
|
|
|
|
// the lod manager. no op if not here
|
|
CLodCharacterManager *mngr= scene->getLodCharacterManager();
|
|
if(!mngr)
|
|
return;
|
|
|
|
// Get global lighting on the instance. Suppose SunAmbient only.
|
|
//=================
|
|
const CLightContribution *lightContrib;
|
|
|
|
// the std case is to take my model lightContribution
|
|
if(_AncestorSkeletonModel==NULL)
|
|
lightContrib= &getSkeletonLightContribution();
|
|
// but if skinned/sticked (directly or not) to a skeleton, take its.
|
|
else
|
|
lightContrib= &_AncestorSkeletonModel->getSkeletonLightContribution();
|
|
|
|
// compute his main light contribution result. Try first with sun
|
|
CRGBA mainAmbient= scene->getSunAmbient();
|
|
CRGBA mainDiffuse= scene->getSunDiffuse();
|
|
// modulate sun contribution
|
|
mainDiffuse.modulateFromuiRGBOnly(mainDiffuse, lightContrib->SunContribution );
|
|
CVector mainLightDir= scene->getSunDirection();
|
|
// Add ambient with Lod Emit
|
|
mainAmbient.addRGBOnly(mainAmbient, _LodEmit);
|
|
|
|
|
|
/* During night, and in the buildings, it may be better to use one of the other Points lights
|
|
Test only with the first pointLight, for faster compute, even if It may fail in some cases.
|
|
*/
|
|
CPointLight *mainPL= lightContrib->PointLight[0];
|
|
if(mainPL)
|
|
{
|
|
CRGBA plDiffuse;
|
|
// get the diffuse of the pointLight, attenuated from distance and importance.
|
|
plDiffuse.modulateFromuiRGBOnly(mainPL->getDiffuse(), lightContrib->AttFactor[0]);
|
|
plDiffuse.A = 255;
|
|
// compare the 2 diffuse
|
|
uint d0= mainDiffuse.R + mainDiffuse.G + mainDiffuse.B;
|
|
uint d1= plDiffuse.R + plDiffuse.G + plDiffuse.B;
|
|
// if the pointLight is lighter, take it.
|
|
if(d1>d0)
|
|
{
|
|
// leave ambient, but take diffuse and pointLight fake Direction
|
|
mainDiffuse= plDiffuse;
|
|
mainLightDir= getWorldMatrix().getPos() - mainPL->getPosition();
|
|
mainLightDir.normalize();
|
|
}
|
|
}
|
|
|
|
|
|
// compute colors of the lods.
|
|
//=================
|
|
// NB: even if texturing is sufficient, still important for AlphaTest.
|
|
|
|
// If must recompute alpha because of change of skin added/deleted
|
|
if(_CLodVertexAlphaDirty)
|
|
{
|
|
// recompute vertex alpha
|
|
computeCLodVertexAlpha(mngr);
|
|
// set _CLodVertexAlphaDirty to false.
|
|
_CLodVertexAlphaDirty= false;
|
|
}
|
|
|
|
// render the Lod in the LodManager.
|
|
//=================
|
|
// render must have been intialized
|
|
nlassert(mngr->isRendering());
|
|
|
|
|
|
// add the instance to the manager.
|
|
if(!mngr->addRenderCharacterKey(_CLodInstance, getWorldMatrix(),
|
|
mainAmbient, mainDiffuse, mainLightDir) )
|
|
{
|
|
// If failed to add it because no more vertex space in the manager, retry.
|
|
|
|
// close vertexBlock, compile render
|
|
mngr->endRender();
|
|
// and restart.
|
|
mngr->beginRender(drv, renderTrav.CamPos);
|
|
|
|
// retry. but no-op if refail.
|
|
mngr->addRenderCharacterKey(_CLodInstance, getWorldMatrix(),
|
|
mainAmbient, mainDiffuse, mainLightDir);
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::renderSkins()
|
|
{
|
|
// Render skins according to the pass.
|
|
CRenderTrav &rdrTrav= getOwnerScene()->getRenderTrav();
|
|
// get a ptr on the driver
|
|
IDriver *drv= rdrTrav.getDriver();
|
|
nlassert(drv);
|
|
|
|
|
|
// Compute the levelOfDetail
|
|
float alphaMRM= _LevelDetail.getLevelDetailFromPolyCount(getNumTrianglesAfterLoadBalancing());
|
|
|
|
// force normalisation of normals..
|
|
bool bkupNorm= drv->isForceNormalize();
|
|
drv->forceNormalize(true);
|
|
|
|
|
|
// rdr good pass
|
|
if(rdrTrav.isCurrentPassOpaque())
|
|
{
|
|
// Compute in Pass Opaque only the light contribution.
|
|
// Easier for skeleton: suppose lightable, no local attenuation
|
|
|
|
// the std case is to take my model lightContribution
|
|
if(_AncestorSkeletonModel==NULL)
|
|
setupCurrentLightContribution(&_LightContribution, false);
|
|
// but if sticked (directly or not) to a skeleton, take its.
|
|
else
|
|
setupCurrentLightContribution(&_AncestorSkeletonModel->_LightContribution, false);
|
|
|
|
|
|
// Activate Driver setup: light and modelMatrix
|
|
changeLightSetup( &rdrTrav );
|
|
rdrTrav.getDriver()->setupModelMatrix(getWorldMatrix());
|
|
|
|
|
|
// Render all totaly opaque skins.
|
|
renderSkinList(_OpaqueSkins, alphaMRM);
|
|
}
|
|
else
|
|
{
|
|
// NB: must have some transparent skins, since thee skeletonModel is traversed in the transparent pass.
|
|
|
|
// Activate Driver setup: light and modelMatrix
|
|
changeLightSetup( &rdrTrav );
|
|
rdrTrav.getDriver()->setupModelMatrix(getWorldMatrix());
|
|
|
|
|
|
// render all opaque/transparent skins
|
|
renderSkinList(_TransparentSkins, alphaMRM);
|
|
}
|
|
|
|
|
|
// bkup force normalisation.
|
|
drv->forceNormalize(bkupNorm);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::renderSkinList(NLMISC::CObjectVector<CTransform*, false> &skinList, float alphaMRM)
|
|
{
|
|
CRenderTrav &rdrTrav= getOwnerScene()->getRenderTrav();
|
|
|
|
// if the SkinManager is not possible at all, just rendered the std way.
|
|
if( !rdrTrav.getMeshSkinManager() )
|
|
{
|
|
for(uint i=0;i<skinList.size();i++)
|
|
{
|
|
skinList[i]->renderSkin(alphaMRM);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// get the meshSkinManager
|
|
CVertexStreamManager &meshSkinManager= *rdrTrav.getMeshSkinManager();
|
|
|
|
// array (rarely allocated) of skins with grouping support
|
|
static std::vector<CTransform*> skinsToGroup;
|
|
static std::vector<uint> baseVertices;
|
|
skinsToGroup.clear();
|
|
baseVertices.clear();
|
|
|
|
// get the maxVertices the manager support
|
|
uint maxVertices= meshSkinManager.getMaxVertices();
|
|
uint vertexSize= meshSkinManager.getVertexSize();
|
|
|
|
// render any skins which do not support SkinGrouping, and fill array of skins to group
|
|
for(uint i=0;i<skinList.size();i++)
|
|
{
|
|
// If don't support, or if too big to fit in the manager, just renderSkin()
|
|
if(!skinList[i]->supportSkinGrouping())
|
|
{
|
|
H_AUTO( NL3D_Skin_NotGrouped );
|
|
skinList[i]->renderSkin(alphaMRM);
|
|
}
|
|
else
|
|
{
|
|
skinsToGroup.push_back(skinList[i]);
|
|
}
|
|
}
|
|
|
|
H_AUTO( NL3D_Skin_Grouped );
|
|
|
|
// For each skin, have an index which gives the decal of the vertices in the buffer
|
|
baseVertices.resize(skinsToGroup.size());
|
|
|
|
// while there is skin to render in group
|
|
uint skinId= 0;
|
|
while(skinId<skinsToGroup.size())
|
|
{
|
|
// space left in the manager
|
|
uint remainingVertices= maxVertices;
|
|
uint currentBaseVertex= 0;
|
|
|
|
|
|
// First pass, fill The VB.
|
|
//------------
|
|
// lock buffer
|
|
uint8 *vbDest= meshSkinManager.lock();
|
|
|
|
// For all skins until the buffer is full
|
|
uint startSkinId= skinId;
|
|
while(skinId<skinsToGroup.size())
|
|
{
|
|
// if success to fill the AGP
|
|
sint numVerticesAdded= skinsToGroup[skinId]->renderSkinGroupGeom(alphaMRM, remainingVertices,
|
|
vbDest + vertexSize*currentBaseVertex );
|
|
// -1 means that this skin can't render because no space left for her. Then stop for this block
|
|
if(numVerticesAdded==-1)
|
|
break;
|
|
// Else ok, get the currentBaseVertex for this skin
|
|
baseVertices[skinId]= currentBaseVertex;
|
|
// and jump to the next place
|
|
currentBaseVertex+= numVerticesAdded;
|
|
remainingVertices-= numVerticesAdded;
|
|
|
|
// go to the next skin
|
|
skinId++;
|
|
}
|
|
|
|
// release buffer. ATI: release only vertices used.
|
|
meshSkinManager.unlock(currentBaseVertex);
|
|
|
|
// Second pass, render the primitives.
|
|
//------------
|
|
meshSkinManager.activate();
|
|
|
|
/* Render any primitives that are not specular. Group specular ones into specularRdrPasses.
|
|
NB: this speed a lot (specular setup is heavy)!
|
|
*/
|
|
static std::vector<CSkinSpecularRdrPass> specularRdrPasses;
|
|
specularRdrPasses.clear();
|
|
for(uint i=startSkinId;i<skinId;i++)
|
|
{
|
|
// render the skin in the current buffer
|
|
skinsToGroup[i]->renderSkinGroupPrimitives(baseVertices[i], specularRdrPasses, i);
|
|
}
|
|
|
|
// If any skin Specular rdrPass to render
|
|
if(!specularRdrPasses.empty())
|
|
{
|
|
// Sort by Specular Map. HTimerInfo: take 0.0% time
|
|
sort(specularRdrPasses.begin(), specularRdrPasses.end());
|
|
|
|
// Batch Specular! HTimerInfo: take 0.2%
|
|
rdrTrav.getDriver()->startSpecularBatch();
|
|
|
|
// Render all of them
|
|
for(uint i=0;i<specularRdrPasses.size();i++)
|
|
{
|
|
CSkinSpecularRdrPass &specRdrPass= specularRdrPasses[i];
|
|
// render the associated skin in the current buffer
|
|
skinsToGroup[specRdrPass.SkinIndex]->renderSkinGroupSpecularRdrPass(specRdrPass.RdrPassIndex);
|
|
}
|
|
|
|
// End Batch Specular! HTimerInfo: take 0.0%
|
|
rdrTrav.getDriver()->endSpecularBatch();
|
|
}
|
|
|
|
|
|
// End of this block, swap to the next buffer. HTimerInfo: take 0.0%
|
|
meshSkinManager.swapVBHard();
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
float CSkeletonModel::getNumTriangles (float distance)
|
|
{
|
|
// If the skeleton is displayed as a CLod suppose 0 triangles.
|
|
if( isDisplayedAsLodCharacter() )
|
|
return 0;
|
|
else
|
|
// NB: this is an approximation, but this is continious.
|
|
return _LevelDetail.getNumTriangles(distance);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::changeMRMDistanceSetup(float distanceFinest, float distanceMiddle, float distanceCoarsest)
|
|
{
|
|
// check input.
|
|
if(distanceFinest<0) return;
|
|
if(distanceMiddle<=distanceFinest) return;
|
|
if(distanceCoarsest<=distanceMiddle) return;
|
|
|
|
// Change.
|
|
_LevelDetail.DistanceFinest= distanceFinest;
|
|
_LevelDetail.DistanceMiddle= distanceMiddle;
|
|
_LevelDetail.DistanceCoarsest= distanceCoarsest;
|
|
|
|
// compile
|
|
_LevelDetail.compileDistanceSetup();
|
|
|
|
// Never more use MAX skin setup.
|
|
_DefaultMRMSetup= false;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::resetDefaultMRMDistanceSetup()
|
|
{
|
|
_DefaultMRMSetup= true;
|
|
|
|
// Must use Skins linked to know the MRM setup.
|
|
dirtSkinRenderLists();
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
bool CSkeletonModel::computeRenderedBBox(NLMISC::CAABBox &bbox, bool computeInWorld)
|
|
{
|
|
// reset bbox
|
|
CAABBox tmpBBox;
|
|
tmpBBox.setCenter(CVector::Null);
|
|
tmpBBox.setHalfSize(CVector::Null);
|
|
bool empty= true;
|
|
|
|
// Not visible => empty bbox
|
|
if(!isClipVisible())
|
|
return false;
|
|
|
|
// For all bones
|
|
uint i;
|
|
for(i=0;i<Bones.size();i++)
|
|
{
|
|
if(isBoneComputed(i))
|
|
{
|
|
CVector pos;
|
|
if(computeInWorld)
|
|
pos= Bones[i].getWorldMatrix().getPos();
|
|
else
|
|
pos= Bones[i].getLocalSkeletonMatrix().getPos();
|
|
if(empty)
|
|
{
|
|
empty= false;
|
|
tmpBBox.setCenter(pos);
|
|
}
|
|
else
|
|
tmpBBox.extend(pos);
|
|
}
|
|
}
|
|
|
|
// End!
|
|
if(!empty)
|
|
{
|
|
bbox= tmpBBox;
|
|
return true;
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::getWorldMaxBoneSpheres(std::vector<NLMISC::CBSphere> &dest) const
|
|
{
|
|
dest.clear();
|
|
// Not visible => empty bbox
|
|
if(!isClipVisible())
|
|
return;
|
|
dest.resize(_BoneToCompute.size());
|
|
for(uint i=0;i<_BoneToCompute.size();i++)
|
|
{
|
|
CBone *bone= _BoneToCompute[i].Bone;
|
|
bone->_MaxSphere.applyTransform(bone->getWorldMatrix(), dest[i]);
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
bool CSkeletonModel::computeRenderedBBoxWithBoneSphere(NLMISC::CAABBox &bbox, bool computeInWorld)
|
|
{
|
|
// Not visible => empty bbox
|
|
if(!isClipVisible())
|
|
return false;
|
|
|
|
if(_BoneToCompute.empty())
|
|
return false;
|
|
|
|
if (_Skins.empty())
|
|
return false;
|
|
|
|
updateSkinRenderLists();
|
|
|
|
// **** Compute The BBox with Bones of the skeleton
|
|
CVector minBB(0.f, 0.f, 0.f), maxBB(0.f, 0.f, 0.f);
|
|
for(uint i=0;i<_BoneToCompute.size();i++)
|
|
{
|
|
CBone *bone= _BoneToCompute[i].Bone;
|
|
// compute the world / local sphere
|
|
const CMatrix &boneMat = computeInWorld ? bone->getWorldMatrix() : bone->getLocalSkeletonMatrix();
|
|
CBSphere sphere;
|
|
bone->_MaxSphere.applyTransform(boneMat, sphere);
|
|
// compute bone min max bounding cube.
|
|
CVector minBone, maxBone;
|
|
minBone= maxBone= sphere.Center;
|
|
float r= sphere.Radius;
|
|
|
|
|
|
minBone.x-= r;
|
|
minBone.y-= r;
|
|
minBone.z-= r;
|
|
maxBone.x+= r;
|
|
maxBone.y+= r;
|
|
maxBone.z+= r;
|
|
|
|
|
|
// set or extend
|
|
if(i==0)
|
|
{
|
|
minBB= minBone;
|
|
maxBB= maxBone;
|
|
}
|
|
else
|
|
{
|
|
minBB.minof(minBB, minBone);
|
|
maxBB.maxof(maxBB, maxBone);
|
|
}
|
|
}
|
|
|
|
// build the bbox
|
|
bbox.setMinMax(minBB, maxBB);
|
|
return true;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
bool CSkeletonModel::computeCurrentBBox(NLMISC::CAABBox &bbox, bool forceCompute /* = false*/, bool computeInWorld)
|
|
{
|
|
// animate all bones channels (detail only channels). don't bother cur lod state.
|
|
CChannelMixer *chanmix= getChannelMixer();
|
|
if (chanmix)
|
|
{
|
|
// Force detail evaluation.
|
|
chanmix->resetEvalDetailDate();
|
|
chanmix->eval(true, 0);
|
|
chanmix->resetEvalDetailDate();
|
|
}
|
|
// compute all skeleton bones
|
|
computeAllBones(CMatrix::Identity);
|
|
|
|
// reset bbox
|
|
CAABBox tmpBBox;
|
|
tmpBBox.setCenter(CVector::Null);
|
|
tmpBBox.setHalfSize(CVector::Null);
|
|
bool empty= true;
|
|
|
|
// For all bones
|
|
uint i;
|
|
for(i=0;i<Bones.size();i++)
|
|
{
|
|
// Is the bone used ?? (whatever bone lod, or CLod state)
|
|
uint16 mustCompute = forceCompute ? 1 : _BoneUsage[i].Usage | _BoneUsage[i].ForcedUsage | _BoneUsage[i].CLodForcedUsage;
|
|
|
|
// If the bone is used.
|
|
if(mustCompute)
|
|
{
|
|
CVector pos;
|
|
if(computeInWorld)
|
|
pos= Bones[i].getWorldMatrix().getPos();
|
|
else
|
|
pos= Bones[i].getLocalSkeletonMatrix().getPos();
|
|
if(empty)
|
|
{
|
|
empty= false;
|
|
tmpBBox.setCenter(pos);
|
|
}
|
|
else
|
|
tmpBBox.extend(pos);
|
|
}
|
|
}
|
|
|
|
// End!
|
|
if(!empty)
|
|
{
|
|
bbox= tmpBBox;
|
|
return true;
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::getLightHotSpotInWorld(CVector &modelPos, float &modelRadius) const
|
|
{
|
|
// If the bone 0 is computed (pelvis), then return its worldMatrix
|
|
if(isBoneComputed(0))
|
|
{
|
|
modelPos= Bones[0].getWorldMatrix().getPos();
|
|
}
|
|
else
|
|
{
|
|
/* Else return the skeleton pos. NB: this seems useless since bone 0 not computed means no Skins.
|
|
But lighting computation is still done and may use a VisualCollisionEntity.
|
|
This system cache some infos according to position. This is why we must return a position
|
|
near the skeleton (else cache crash each frame => slowdown...)
|
|
*/
|
|
modelPos= _WorldMatrix.getPos();
|
|
}
|
|
|
|
// Consider Skeletons as not big lightable
|
|
modelRadius= 0;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::setBoneAnimCtrl(uint boneId, IAnimCtrl *ctrl)
|
|
{
|
|
if(boneId>=Bones.size())
|
|
return;
|
|
|
|
CBone &bone= Bones[boneId];
|
|
|
|
// Update refCount
|
|
if(ctrl && !bone._AnimCtrl)
|
|
_AnimCtrlUsage++;
|
|
else if(!ctrl && bone._AnimCtrl)
|
|
_AnimCtrlUsage--;
|
|
|
|
// set
|
|
bone._AnimCtrl= ctrl;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
IAnimCtrl *CSkeletonModel::getBoneAnimCtrl(uint boneId) const
|
|
{
|
|
if(boneId>=Bones.size())
|
|
return NULL;
|
|
|
|
return Bones[boneId]._AnimCtrl;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
bool CSkeletonModel::fastIntersect(const NLMISC::CVector &p0, const NLMISC::CVector &dir, float &dist2D, float &distZ, bool computeDist2D)
|
|
{
|
|
if(!_SupportFastIntersect)
|
|
return false;
|
|
|
|
// no intersection by default
|
|
dist2D= FLT_MAX;
|
|
distZ= FLT_MAX;
|
|
|
|
// The skinning must be done in final RaySpace.
|
|
CMatrix toRaySpace;
|
|
// compute the ray matrix
|
|
CVector dirn= dir;
|
|
if(dirn.isNull())
|
|
dirn= CVector::K;
|
|
dirn.normalize();
|
|
toRaySpace.setArbitraryRotK(dirn);
|
|
toRaySpace.setPos(p0);
|
|
// The skinning must be done in ray space: (RayMat-1)*skelWorldMatrix;
|
|
toRaySpace.invert();
|
|
toRaySpace*= getWorldMatrix();
|
|
|
|
// displayed as a CLod?
|
|
if(isDisplayedAsLodCharacter())
|
|
{
|
|
// must do the test with the CLod, because Bones are not animated at all (hence skinning would be false)
|
|
CLodCharacterManager *mngr= getOwnerScene()->getLodCharacterManager();
|
|
if(!mngr)
|
|
return false;
|
|
|
|
// test the instance
|
|
if(!mngr->fastIntersect(_CLodInstance, toRaySpace, dist2D, distZ, computeDist2D))
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
// For all skins
|
|
ItTransformSet it;
|
|
for(it= _Skins.begin();it!=_Skins.end();it++)
|
|
{
|
|
CTransform *skin= *it;
|
|
|
|
// If the skin is hidden, don't test intersection!
|
|
if(skin->getVisibility()==CHrcTrav::Hide)
|
|
continue;
|
|
|
|
if(!skin->supportIntersectSkin())
|
|
continue;
|
|
|
|
// compute intersection with this skin
|
|
float skinDist2D, skinDistZ;
|
|
if(skin->intersectSkin(toRaySpace, skinDist2D, skinDistZ, computeDist2D))
|
|
{
|
|
// true intersection found?
|
|
if(skinDist2D==0)
|
|
{
|
|
dist2D= 0;
|
|
distZ= min(distZ, skinDistZ);
|
|
}
|
|
// else lower the distance to the skins?
|
|
else if(dist2D>0)
|
|
{
|
|
dist2D= min(dist2D, skinDist2D);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// no intersection found? set Z to 0 (to be clean)
|
|
if(dist2D>0)
|
|
distZ= 0;
|
|
|
|
return true;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::remapSkinBones(const std::vector<string> &bonesName, std::vector<sint32> &bonesId, std::vector<uint> &remap)
|
|
{
|
|
// Resize boneId to the good size.
|
|
bonesId.resize(bonesName.size());
|
|
remap.resize(bonesName.size());
|
|
|
|
// **** For each bones, compute remap
|
|
for (uint bone=0; bone<remap.size(); bone++)
|
|
{
|
|
// Look for the bone
|
|
sint32 boneId = getBoneIdByName (bonesName[bone]);
|
|
|
|
// Setup the _BoneId.
|
|
bonesId[bone]= boneId;
|
|
|
|
// Bones found ?
|
|
if (boneId != -1)
|
|
{
|
|
// Set the bone id
|
|
remap[bone] = (uint32)boneId;
|
|
}
|
|
else
|
|
{
|
|
// Put id 0
|
|
remap[bone] = 0;
|
|
|
|
// Error
|
|
nlwarning ("Bone %s not found in the skeleton.", bonesName[bone].c_str());
|
|
}
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
// ShadowMap
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::generateShadowMap(const CVector &lightDir)
|
|
{
|
|
H_AUTO( NL3D_Skeleton_GenerateShadow );
|
|
|
|
// get the driver for Texture Render
|
|
CScene *scene= getOwnerScene();
|
|
CRenderTrav &renderTrav= scene->getRenderTrav();
|
|
IDriver *driver= renderTrav.getAuxDriver();
|
|
|
|
if(!Shape)
|
|
return;
|
|
|
|
// update ShadowMap data if needed.
|
|
// ****
|
|
updateShadowMap(driver);
|
|
|
|
// compute the ProjectionMatrix.
|
|
// ****
|
|
|
|
// Compute the BBox in World, with bounding Box of Bones, and with BoundingBox of sticked Objects
|
|
CAABBox bbWorld;
|
|
computeWorldBBoxForShadow(bbWorld);
|
|
|
|
|
|
// Here the bbox is defined in world, hence must remove the World Pos.
|
|
CMatrix localPosMatrix;
|
|
localPosMatrix.setPos(-getWorldMatrix().getPos());
|
|
|
|
// setup cameraMatrix with BBox and Enlarge For 1 pixel
|
|
CMatrix cameraMatrix;
|
|
_ShadowMap->buildCasterCameraMatrix(lightDir, localPosMatrix, bbWorld, cameraMatrix);
|
|
|
|
|
|
// Render.
|
|
// ****
|
|
// setup the orhtogonal frustum and viewMatrix to include all the object.
|
|
driver->setFrustum(0,1,0,1,0,1,false);
|
|
driver->setupViewMatrix(cameraMatrix.inverted());
|
|
|
|
// render the Skinned meshs, and also the Sticked Objects/Skeletons
|
|
CMaterial &castMat= renderTrav.getShadowMapManager().getCasterShadowMaterial();
|
|
renderIntoSkeletonShadowMap(this, castMat);
|
|
|
|
// Infos.
|
|
// ****
|
|
|
|
// Compute the BackPoint: the first point to be shadowed.
|
|
CVector backPoint= bbWorld.getCenter();
|
|
// get the 3/4 bottom of the shape
|
|
backPoint.z-= bbWorld.getHalfSize().z/2;
|
|
// Use the 3/4 bottom of the BBox minus the light direction in XY.
|
|
CVector ldir= lightDir;
|
|
ldir.z= 0;
|
|
ldir.normalize();
|
|
// NB: This way seems to works quite well, even if the worldBBox is changing every frame.
|
|
float lenXY= (CVector(bbWorld.getHalfSize().x, bbWorld.getHalfSize().y, 0)).norm();
|
|
backPoint-= ldir*lenXY;
|
|
// localPos.
|
|
backPoint-= getWorldMatrix().getPos();
|
|
|
|
// Compute LocalProjectionMatrix and other infos from cameraMatrix and backPoint?
|
|
_ShadowMap->buildProjectionInfos(cameraMatrix, backPoint, getShadowMapMaxDepth());
|
|
}
|
|
|
|
// ***************************************************************************
|
|
CShadowMap *CSkeletonModel::getShadowMap()
|
|
{
|
|
return _ShadowMap;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::createShadowMap()
|
|
{
|
|
// create the shadowMap
|
|
if(!_ShadowMap)
|
|
{
|
|
_ShadowMap= new CShadowMap(&getOwnerScene()->getRenderTrav().getShadowMapManager());
|
|
getOwnerScene()->registerShadowCasterToList(this);
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::deleteShadowMap()
|
|
{
|
|
if(_ShadowMap)
|
|
{
|
|
delete _ShadowMap;
|
|
_ShadowMap= NULL;
|
|
getOwnerScene()->unregisterShadowCasterToList(this);
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::updateShadowMap(IDriver * /* driver */)
|
|
{
|
|
nlassert(_ShadowMap);
|
|
|
|
// create/update texture
|
|
if(_ShadowMap->getTextureSize()!=getOwnerScene()->getShadowMapTextureSize())
|
|
{
|
|
_ShadowMap->initTexture(getOwnerScene()->getShadowMapTextureSize());
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::renderShadowSkins(CMaterial &castMat)
|
|
{
|
|
H_AUTO( NL3D_Skin_RenderShadow );
|
|
|
|
CRenderTrav &rdrTrav= getOwnerScene()->getRenderTrav();
|
|
// Render Shadow in auxiliary driver.
|
|
IDriver *driver= rdrTrav.getAuxDriver();
|
|
|
|
// if the SkinManager is not possible at all, just rendered the std way
|
|
if( !rdrTrav.getShadowMeshSkinManager() )
|
|
{
|
|
// can occurs?????
|
|
// ABORT!! ... avoid Mesh Shadowing (free shadowMap)? Replace with a dummy Shadow?
|
|
// For now, no-op...
|
|
}
|
|
else
|
|
{
|
|
uint i;
|
|
|
|
// get the meshSkinManager
|
|
CVertexStreamManager &meshSkinManager= *rdrTrav.getShadowMeshSkinManager();
|
|
|
|
// array (rarely allocated) of skins with grouping support
|
|
static std::vector<CTransform*> skinsToGroup;
|
|
static std::vector<uint> baseVertices;
|
|
skinsToGroup.clear();
|
|
baseVertices.clear();
|
|
|
|
// get the maxVertices the manager support
|
|
uint maxVertices= meshSkinManager.getMaxVertices();
|
|
uint vertexSize= meshSkinManager.getVertexSize();
|
|
|
|
// fill array of skins to group (suppose all support else won't be rendered)
|
|
for(i=0;i<_OpaqueSkins.size();i++)
|
|
{
|
|
if(_OpaqueSkins[i]->supportShadowSkinGrouping())
|
|
skinsToGroup.push_back(_OpaqueSkins[i]);
|
|
}
|
|
for(i=0;i<_TransparentSkins.size();i++)
|
|
{
|
|
if(_TransparentSkins[i]->supportShadowSkinGrouping())
|
|
skinsToGroup.push_back(_TransparentSkins[i]);
|
|
}
|
|
|
|
// For each skin, have an index which gives the decal of the vertices in the buffer
|
|
baseVertices.resize(skinsToGroup.size());
|
|
|
|
// while there is skin to render in group
|
|
uint skinId= 0;
|
|
while(skinId<skinsToGroup.size())
|
|
{
|
|
// space left in the manager
|
|
uint remainingVertices= maxVertices;
|
|
uint currentBaseVertex= 0;
|
|
|
|
// First pass, fill The VB.
|
|
//------------
|
|
// lock buffer
|
|
uint8 *vbDest= meshSkinManager.lock();
|
|
|
|
// For all skins until the buffer is full
|
|
uint startSkinId= skinId;
|
|
while(skinId<skinsToGroup.size())
|
|
{
|
|
// if success to fill the AGP
|
|
sint numVerticesAdded= skinsToGroup[skinId]->renderShadowSkinGeom(remainingVertices,
|
|
vbDest + vertexSize*currentBaseVertex );
|
|
// -1 means that this skin can't render because no space left for her. Then stop for this block
|
|
if(numVerticesAdded==-1)
|
|
break;
|
|
// Else ok, get the currentBaseVertex for this skin
|
|
baseVertices[skinId]= currentBaseVertex;
|
|
// and jump to the next place
|
|
currentBaseVertex+= numVerticesAdded;
|
|
remainingVertices-= numVerticesAdded;
|
|
|
|
// go to the next skin
|
|
skinId++;
|
|
}
|
|
|
|
// release buffer. ATI: release only vertices used.
|
|
meshSkinManager.unlock(currentBaseVertex);
|
|
|
|
// Second pass, render the primitives.
|
|
//------------
|
|
meshSkinManager.activate();
|
|
|
|
// Render any primitives
|
|
for(uint i=startSkinId;i<skinId;i++)
|
|
{
|
|
// render the skin in the current buffer
|
|
skinsToGroup[i]->renderShadowSkinPrimitives(castMat, driver, baseVertices[i]);
|
|
}
|
|
|
|
// End of this block, swap to the next buffer
|
|
meshSkinManager.swapVBHard();
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
bool CSkeletonModel::computeWorldBBoxForShadow(NLMISC::CAABBox &worldBB)
|
|
{
|
|
uint i;
|
|
|
|
// If even not visible, no-op
|
|
if(!isHrcVisible() || !Shape)
|
|
return false;
|
|
|
|
// **** Compute The BBox with Bones of the skeleton
|
|
CVector minBB(0.f, 0.f, 0.f), maxBB(0.f, 0.f, 0.f);
|
|
for(i=0;i<_BoneToCompute.size();i++)
|
|
{
|
|
CBone *bone= _BoneToCompute[i].Bone;
|
|
// compute the world sphere
|
|
const CMatrix &worldMat= bone->getWorldMatrix();
|
|
CBSphere worldSphere;
|
|
bone->_MaxSphere.applyTransform(worldMat, worldSphere);
|
|
// compute bone min max bounding cube.
|
|
CVector minBone, maxBone;
|
|
minBone= maxBone= worldSphere.Center;
|
|
float r= worldSphere.Radius;
|
|
minBone.x-= r;
|
|
minBone.y-= r;
|
|
minBone.z-= r;
|
|
maxBone.x+= r;
|
|
maxBone.y+= r;
|
|
maxBone.z+= r;
|
|
// set or extend
|
|
if(i==0)
|
|
{
|
|
minBB= minBone;
|
|
maxBB= maxBone;
|
|
}
|
|
else
|
|
{
|
|
minBB.minof(minBB, minBone);
|
|
maxBB.maxof(maxBB, maxBone);
|
|
}
|
|
}
|
|
// build the bbox
|
|
worldBB.setMinMax(minBB, maxBB);
|
|
/*
|
|
// Fake Version. Faster (-0.2 ms for 8 compute each frame) but false.
|
|
for(i=0;i<_BoneToCompute.size();i++)
|
|
{
|
|
CBone *bone= _BoneToCompute[i].Bone;
|
|
const CMatrix &worldMat= bone->getWorldMatrix();
|
|
if(i==0)
|
|
worldBB.setCenter(worldMat.getPos());
|
|
else
|
|
worldBB.extend(worldMat.getPos());
|
|
}
|
|
worldBB.setHalfSize(worldBB.getHalfSize() *1.5f);
|
|
*/
|
|
|
|
|
|
// **** Add to this bbox the ones of the Sticked objects.
|
|
ItTransformSet it;
|
|
for(it= _StickedObjects.begin();it!=_StickedObjects.end();it++)
|
|
{
|
|
CTransform *stickModel= *it;
|
|
// Do the same for this son (NB: recurs, may be a skeleton too!!)
|
|
CAABBox stickBB;
|
|
if(stickModel->computeWorldBBoxForShadow(stickBB))
|
|
{
|
|
// Make union of the 2
|
|
worldBB= CAABBox::computeAABBoxUnion(worldBB, stickBB);
|
|
}
|
|
}
|
|
|
|
// Done!
|
|
return true;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CSkeletonModel::renderIntoSkeletonShadowMap(CSkeletonModel *rootSkeleton, CMaterial &castMat)
|
|
{
|
|
// If even not visible, no-op
|
|
if(!isHrcVisible() || !Shape)
|
|
return;
|
|
|
|
// render into aux Driver
|
|
IDriver *driver= getOwnerScene()->getRenderTrav().getAuxDriver();
|
|
|
|
// **** Render the Skeleton Skins
|
|
// The model Matrix is special here. It must be the Skeleton World Matrix, minus The Root Skeleton pos.
|
|
CMatrix localPosMatrix;
|
|
localPosMatrix.setRot( getWorldMatrix() );
|
|
// NB: if this==rootSkeleton, then the final pos will be CVector::Null
|
|
localPosMatrix.setPos( getWorldMatrix().getPos() - rootSkeleton->getWorldMatrix().getPos() );
|
|
driver->setupModelMatrix(localPosMatrix);
|
|
|
|
// render the skins.
|
|
renderShadowSkins(castMat);
|
|
|
|
// **** Render The Sticked Objects.
|
|
ItTransformSet it;
|
|
for(it= _StickedObjects.begin();it!=_StickedObjects.end();it++)
|
|
{
|
|
CTransform *stickModel= *it;
|
|
stickModel->renderIntoSkeletonShadowMap(rootSkeleton, castMat);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
} // NL3D
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|