khanat-opennel-code/code/nel/include/nel/3d/transform.h

1010 lines
40 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/>.
#ifndef NL_TRANSFORM_H
#define NL_TRANSFORM_H
#include "nel/3d/hrc_trav.h"
#include "nel/3d/track.h"
#include "nel/3d/transformable.h"
#include "nel/3d/animated_value.h"
#include "nel/3d/channel_mixer.h"
#include "nel/misc/matrix.h"
#include "nel/misc/matrix.h"
#include "nel/misc/aabbox.h"
#include "nel/3d/light_contribution.h"
#include "nel/3d/lighting_manager.h"
#include "nel/misc/class_id.h"
#include "nel/3d/fast_ptr_list.h"
namespace NLMISC
{
class CAABBox;
};
namespace NL3D
{
using NLMISC::CRGBA;
using NLMISC::CVector;
using NLMISC::CPlane;
using NLMISC::CMatrix;
using NLMISC::CAABBox;
class CSkeletonModel;
class CInstanceGroup;
class ILogicInfo;
class CLoadBalancingGroup;
class CSkinSpecularRdrPass;
class CShadowMap;
class CMaterial;
class IDriver;
class CInstanceUser;
// ***************************************************************************
// ClassIds.
const NLMISC::CClassId TransformId=NLMISC::CClassId(0x174750cb, 0xf952024);
// ***************************************************************************
/**
* A basic node which provide an animatable matrix (ITransformable).
* May be derived for each node who want to support such a scheme (CCamera, CLight, CInstance ... )
*
* CTransform ALWAYS herit scale from fathers! (joints skeleton may not...) (nbyoyo: else, this breaks the update() system).
*
* CTransform Default tracks are identity (derived class may change this).
*
* CTransform by default IS NOT RENDERABLE. ie never inserted in renderList.
* Deriver should call setIsRenderable(true) to make the model renderable
*
* CTransform by default IS NOT ANIMDETAIL-able. ie never inserted in anim detail list.
* It is automatically inserted in anim detail list if registerToChannelMixer() is called.
* Deriver should call setIsForceAnimdetail(true) to make the model always anim-detail-ed
*
* CTransform by default IS NOT LOADBALANCE-able. ie never inserted into list for LoadBlancing.
* Deriver should call setIsLoadbalancable(true) to support this traversal.
*
* \author Lionel Berenguier
* \author Nevrax France
* \date 2000
*/
class CTransform : public ITransformable
{
public:
/// Call at the beginning of the program, to register the model
static void registerBasic();
/// get the scene which has created us
CScene *getOwnerScene() const {return _OwnerScene;}
public:
/// \name Model updating/traversing features
// @{
/** This function update the model (called by CScene::updateModels())
* Deriver Must :
* - call BaseClass::update() (eg: CTransform::update()).
* - test if something is different (eg: animation modification). Then update Model information (eg compute new Matrix).
*
* The default behavior is to update transform Matrix etc...
*/
virtual void update();
/**
* Extra init for a model. this method is called by the framework at the very end of CScene::createModel()
* Warning! if the model is a CTransformShape, then when initModel() is called, Shape and other related member/setup
* of IShape::createInstance() are not yet done (because createModel() is called at the beginning in createInstance()).
*
* Because initModel() is called at the very end, deriver could implement anything like creating other models,
* but not deleting this model...
*
* Default behavior is to do nothing.
*/
virtual void initModel();
/// Each method is called in its associated traversal
virtual void traverseHrc();
/** The base traverseClip method.
* The behavior is to:
* - test if _WorldVis is visible.
* - test if is clipped with clip() OR IF SKELETON MODEL, USE SKELETON MODEL clip!!
* - if visible and not clipped, set \c _Visible=true (else false). and
* - add the CTransform* to the ClipTrav list
* - if _Visible==true, and renderable, add it to the RenderTraversal: \c RenderTrav->addRenderModel(model);
* - always traverseSons(), to clip the sons.
*/
virtual void traverseClip();
/// call updateWorldMatrixFromFather(), then traverseAnimDetailWithoutUpdateWorldMatrix()
virtual void traverseAnimDetail();
/// no-op by default
virtual void traverseLoadBalancing();
/// traverse the lightedModel per default: recompute LightContribution is isLightable()
virtual void traverseLight();
/// no-op by default
virtual void traverseRender();
/// clip method called by traverseClip(). deafult is always visible
virtual bool clip()
{
return true;
}
/// Called at RenderTrav to profile current render. no-op per default
virtual void profileRender();
// @}
// build matching user interface object (a class derived from CInstanceUser)
virtual class UTransform *buildMatchingUserInterfaceObject ();
public:
/** Accessors for opacity/transparency
*/
void setTransparency(bool v);
void setOpacity(bool v);
// no op for non multi-lod object, else, force the opacity /
void setBypassLODOpacityFlag(bool bypass);
bool getBypassLODOpacityFlag() const{ return getStateFlag(BypassLODOpacity) != 0; }
// return a non-zero value if true
uint32 isOpaque() { return getStateFlag(IsOpaque); }
uint32 isTransparent() { return getStateFlag(IsTransparent); }
/** Set the current layer for this transform.
* Typically, this is used to sort transparent objects. Isn't used with solid objects.
* For now :
* Layer 0 is for underwater
* Layer 1 is for water surfaces
* Layer 2 is for object above water
*/
void setOrderingLayer(uint layer) { _OrderingLayer = uint8(layer); }
/// Get the ordering layer
uint getOrderingLayer() const { return _OrderingLayer; }
/// Set priority for transparent ordering. When sorting before drawing, priority is taken in account before distance, so
// an object with distance 10 and priority 1 will be displayed before any object with distance 1 and priority 0.
// IMPORTANT : priority is clamped by the number of priorities defined in the scene. By default there's only one priority of 0
// that is possible, so priority ordering doesn't actually occurs.
void setTransparencyPriority(uint8 priority) { _TransparencyPriority = priority; }
uint8 getTransparencyPriority() const { return _TransparencyPriority; }
/// Hide the object and his sons.
void hide();
/// Show the objet and his sons.
void show();
/*
* Enable / disable user clipping. If enable, the transform is not clipped into the engine.
* The user has to use show / hide to clip or not the transform.
*/
void setUserClipping(bool enable);
/// Return the user clipping state
bool getUserClipping() const;
/// herit the visibility from his father. (default behavior).
void heritVisibility();
/// Get the local visibility state.
CHrcTrav::TVisibility getVisibility() {return Visibility;}
/// Get the skeleton model. Returnr NULL in normal mode.
CSkeletonModel* getSkeletonModel () const {return _FatherSkeletonModel;}
// Get the ancestor skeleton, (father skeleton that has the least depth in the model tree), or NULL if none
CSkeletonModel *getAncestorSkeletonModel() const { return _AncestorSkeletonModel; }
/// \name Hierarchy linking
// @{
/** link son to this in Hierarchy traversal
* NB: link does nothing if the son node is HRC frozen */
void hrcLinkSon(CTransform *son);
/** unlink this from any Father in Hrc. No-op if no parent
* NB: unlink does nothing if the node is HRC frozen */
void hrcUnlink();
// get the Hrc parent if any (else NULL)
CTransform *hrcGetParent() const {return _HrcParent;}
// get Sons links. NB: indices are no more valid after an hrcUnlink()
uint hrcGetNumChildren() const {return _HrcSons.size();}
CTransform *hrcGetChild(uint index) const;
// @}
/// \name Clip linking
// @{
// Add a link from me to son for Clip Graph (no-op if son==this)
void clipAddChild(CTransform *son);
// Remove a link from me to son for Clip Graph (no-op if not found)
void clipDelChild(CTransform *son);
// unlink from all parent clip
void clipUnlinkFromAll();
// get Parents links. NB: indices are no more valid after a clipDelChild()
uint clipGetNumParents() const {return (uint)_ClipParents.size();}
CTransform *clipGetParent(uint index) const;
// get Sons links. NB: indices are no more valid after a clipDelChild()
uint clipGetNumChildren() const {return _ClipSons.size();}
CTransform *clipGetChild(uint index) const;
// @}
/// \name Derived from ITransformable.
// @{
/// Default Track Values are identity (pos,pivot= 0, scale= 1, rots=0).
virtual ITrack* getDefaultTrack (uint valueId);
/** register transform channels (in global anim mode).
* \see setChannelMixerOwnerShip
*/
virtual void registerToChannelMixer(CChannelMixer *chanMixer, const std::string &prefix);
// @}
/** This force gives this object ownership of the channel mixer it is registered to, so it will delete it when the dtor is called.
* It should be called AFTER this object has been registered to a channel mixer, because a new registration will broke the ownership.
* This is useful for automatic animations, when there's no owner of the channel mixer that could delete it.
*/
void setChannelMixerOwnerShip(bool enable = true) { setStateFlag(IsDeleteChannelMixer, enable); }
bool getChannelMixerOwnerShip() const { return getStateFlag(IsDeleteChannelMixer)!=0; }
/** freeze the preceding position of the model. Do not use, special code for cluster.
* This inform the scene that preceding position setuped by user is "frozen". ie at next render(), this
* object won't be added to the "object moving list" (useful for cluster mgt).
* The "frozen" state is disabled (at render() time) if:
* - change in position (directly or indireclty, such as animation) is performed after the freeze().
* - the "frozen" state of a father is not enabled (or disabled by a change in position of him :) ).
*/
void freeze();
void setDontUnfreezeChildren(bool val);
/** freeze the HRC so the WorldMatrix computed at next render() will be kept for long, and the model won't
* either be tested in HRC (which is still expensive, even if the worldmatrix doesn't need to be recomputed).
* The model won't either be validated. It is suposed to not change at all. Also, if it is not a son of a CCluster,
* it may be accelerated during Cliping (with CQuadGridClipManager).
*
* NB: the model won't be tested in HRC anymore.
* calling freezeHRC() on a model in a hierarchy without calling it to the root of the hierarchy
* will result in that the model won't be validated nor be HRC traversed.
* To be simplier, you should freezeHRC() all the models of a hierarchy, from base root to leaves.
*
* NB: if the hierarchy of this object must change, or if the object must moves, you must call unfreezeHRC() first,
* and you should do this for all the parents of this model.
*/
void freezeHRC();
/** see freezeHRC().
*/
void unfreezeHRC();
/** special feature for CQuadGridClipManager. return a non-zero value if true
*/
uint32 isQuadGridClipEnabled() const {return getStateFlag(QuadGridClipEnabled);}
/**
* Get the worldMatrix that is computed at last Hrc pass
*/
const CMatrix& getWorldMatrix() const {return _WorldMatrix;}
// force the world matrix (could be modified next frame after hrc traversal
void setWorldMatrix(const CMatrix &mat) { _WorldMatrix = mat;}
/** tells if the transform has been determined as visible in the hrc traversal
*/
bool isHrcVisible() const
{
return _WorldVis;
}
/** tells if the transform has been clipped in the clip traversal.
*/
bool isClipVisible() const
{
return _Visible;
}
// A RefPtr is kept on the clusterSystem, so getClusterSystem() will return NULL if the instance group has been deleted
void setClusterSystem (CInstanceGroup *pCS);
CInstanceGroup* getClusterSystem ();
/// name Lighting Behavior.
// @{
/** reset lights which influence this models. NB: the model is removed from all lights's list (except
* FrozenStaticLightSetup). Called by light rendering.
*
* NB: the model is NOT removed from LightingManager (with eraseStaticLightedModel()).
*/
void resetLighting();
/** true if the model can be lighted (such as CMeshBaseInstance)
* Default behavior is false.
* Deriver must use setIsLightable(true) method if the instance can be lighted.
* \return 0 if getUserLightable() is false, or if the model can't be lighted at all. else return a non-zero value
*/
uint32 isLightable() const {return getStateFlag(IsFinalLightable);}
/** Set the UserLightable flag. if false, isLightable() will always return false.
* Doing this, user can disable lighting on a model which may be interesting for speed.
* Default behavior is UserLightable==true.
*/
void setUserLightable(bool enable);
/** Get the UserLightable flag.
*/
bool getUserLightable() const {return getStateFlag(IsUserLightable)!=0;}
/** Freeze and set the Static Light Setup. Called by CInstanceGroup::addToScene()
* NB: it calls resetLighting() first.
* NB: nlassert(numPointLights<=NL3D_MAX_LIGHT_CONTRIBUTION)
*/
void freezeStaticLightSetup(CPointLight *pointLight[NL3D_MAX_LIGHT_CONTRIBUTION],
uint numPointLights, uint8 sunContribution, CPointLight *frozenAmbientlight);
/** unFreeze the Static Light Setup. Must be called if static pointLights are deleted.
* NB: it calls resetLighting() first.
* NB: do not need to call it if pointLights and this transform are deleted at same time.
*/
void unfreezeStaticLightSetup();
/** non-zero if the lighting Manager must take into account the bbox of the transform.
* Default behavior is false. Deriver must call setIsBigLightable() at initialisation to change it.
*/
uint32 isBigLightable() const {return getStateFlag(IsBigLightable);}
/** return true if the current light contribution of this model use a MergedPointLight
*/
bool useMergedPointLight() const {return _LightContribution.UseMergedPointLight;}
/** Return the current light contribution of this model
*/
const CLightContribution &getLightContribution() const { return _LightContribution; }
/** get the HotSpot of the model for Light computation. For models with global attenuation, this is
* the point taken for attenuation computes. NB: should return the current world position.
* NB: return also the modelRadius (only for bigLightable)
* Default to NULL.
*/
virtual void getLightHotSpotInWorld(CVector &modelPos, float &modelRadius) const {modelPos= CVector::Null; modelRadius=0;}
// @}
/** Set the LogicInfo for this transfrom, eg to retrieve statc light information, see ILogicInfo.
* Ptr is kept in CTransfrom, so should call setLogicInfo(NULL) before to clean up.
*/
void setLogicInfo(ILogicInfo *logicInfo) {_LogicInfo= logicInfo;}
/** Get the untransformed AABBox of the transform. NULL (gtCenter()= 0, gtSize()==0) by default.
*/
virtual void getAABBox(NLMISC::CAABBox &bbox) const;
/// name Load Balancing Behavior.
// @{
/** Change the load Balancing group of a model. Every models are in a special LoadBalancingGroup.
* NB: the group is created if did not exist.
* NB: if models are skinned, it is their Skeleton which drive the group
*
* By default, models lies in the "Default" group, but Skeletons for skinning and ParticlesSystems which
* are in "Skin" and "Fx" group respectively.
* The "Default" group is special because it is not balanced (ie models are only degraded from
* their distance to camera)
*/
void setLoadBalancingGroup(const std::string &group);
/** Get the load Balancing group of a model. see setLoadBalancingGroup().
*/
const std::string &getLoadBalancingGroup() const;
// @}
/// \name Skinning Behavior.
// @{
/// return non-zero if I am a skeleton. if yes, static_cast<CSkeletonModel*> may be used
uint32 isSkeleton() const {return getStateFlag(IsSkeleton);}
/// non-zero if the model is skinned onto a skeleton.
uint32 isSkinned() const {return getStateFlag(IsSkinned);}
/** Called for edition purpose (slow call O(NVertex))
* It return the BBox (local to the Bone at its bind pos) of all of the vertices
* of this mesh that are bound to this bone
* NB: the instance must be skinned to the skeleton instance, BUT the result is not modified
* by current bone world matrix (eg: no current scale influence)
* \param boneId: the id of the skeleton's bone (which the instance is skinned to)
* \param bbox: the bbox (in bone basis) filled if return is true
* \return false if no vertices are bound to
*/
virtual bool getSkinBoneBBox(NLMISC::CAABBox &/* bbox */, uint /* boneId */) {return false;}
// @}
/// name Misc
// @{
/** set the Mean color of the transform. The mean color can be used for many purpose, such as drawing
* objects if the textures are not loaded. It is used also for Lod Character.
* Default color is (255,255,255)
*/
void setMeanColor(CRGBA color);
/// see setMeanColor()
CRGBA getMeanColor() const {return _MeanColor;}
/// non-zero if the model is animDetailable (ie added to the animDetail list if visible)
uint32 isAnimDetailable() const {return getStateFlag(IsAnimDetailable);}
/// non-zero if the model is loadBalancable (ie added to the loadBalancing list if visible)
uint32 isLoadBalancable() const {return getStateFlag(IsLoadBalancable);}
/// non-zero if the model is renderable (ie something may appear on screen)
uint32 isRenderable() const {return getStateFlag(IsRenderable);}
/// non-zero if the CTransform can be casted to a CMeshBaseInstance
uint32 isMeshBaseInstance() const {return getStateFlag(IsMeshBaseInstance);}
/// non-zero if the CTransform can be casted to a CTransformShape
uint32 isTransformShape() const {return getStateFlag(IsTransformShape);}
/// non-zero if the CTransform can be casted to a CCluster
uint32 isCluster() const {return getStateFlag(IsCluster);}
/** true if the transform support fast intersection (fastIntersect() works)
* For now, supports
* - skeleton with ALL skins that are shadowSkined (w or wo MRM)
* - mesh that are not skinned (Standard/MRM/MultiLod), and that have been flagged through
* CShapeBank::buildSystemGeometryForshape()
*/
bool supportFastIntersect() const {return _SupportFastIntersect;}
/** test if the transform intersect the ray (p0, dir).
* \return false if not supported/no triangles, else true if can do the test (even if don't intersect!)
* if intersect, dist2D=0, and distZ= Depth Distance
* if don't intersect, dist2D="nearest distance to the ray", and distZ=0
* \param computeDist2D if false and don't intersect, then return dist2D=FLT_MAX, and distZ=0
*/
virtual bool fastIntersect(const NLMISC::CVector &/* p0 */, const NLMISC::CVector &/* dir */, float &/* dist2D */, float &/* distZ */, bool /* computeDist2D */) {return false;}
/// internal only: used by CMeshBase
void enableFastIntersectSupport(bool enable) {_SupportFastIntersect= enable;}
// @}
/** \name ShadowMapping
* NB: It is the deriver work to call CRenderTrav::getShadowMapManager().addShadowReceiver() or
* CRenderTrav::getShadowMapManager().addShadowCaster(), typically in traverseHRC() or in traverseRender()
* NB: when _AncestorSkeletonModel!=NULL, the shadowMapCaster should not be Added.
* NB: Deriver must also implement createShadowMap() and deleteShadowMap() protected callBacks,
* if it supports CastShadowMaping
*/
// @{
/** By default, map shadow casting is disabled. This enabled shadow for this model.
* Fails if the model don't support dynamic Map Shadow Casting (eg landscape)
* Dervier note: createShadowMap() and deleteShadowMap() is called here.
*/
void enableCastShadowMap(bool state);
/// true if the instance cast shadow. By default false
bool canCastShadowMap() const {return getStateFlag(IsFinalShadowMapCaster)!=0;}
/** By default, map shadow receiving is disabled. This enabled shadow for this model.
* Fails if the model don't support dynamic Map Shadow Receiving (eg Particle system)
*/
void enableReceiveShadowMap(bool state) {if(modelCanReceiveShadowMap()) setStateFlag(IsFinalShadowMapReceiver, state);}
/// true if the instance receive shadow. By default false
bool canReceiveShadowMap() const {return getStateFlag(IsFinalShadowMapReceiver)!=0;}
/// true if the model provide a method to support shadowMap generation
uint32 modelCanCastShadowMap() const {return getStateFlag(IsShadowMapCaster);}
/// true if the model provide a method to support shadowMap receiving
uint32 modelCanReceiveShadowMap() const {return getStateFlag(IsShadowMapReceiver);}
/** For Casters. Display the Shadow to the "Auxiliary Driver".
* This method should only write to AlphaBuffer (since RGB may be the current rendered scene!),
* with Alpha==1 when pixel is shadowed.
* The ShadowMapManager has already cleared the AlphaBuffer to black, and has already enabled alpha write only.
* The ShadowMapManager has already setuped Viewport/Scissor as its convenience.
* The extra blurring is a work of the ShadowMapManager (which blurs multiple shadows in a same pass)
* NB: you can overwrite the current driver frustum/ViewMatrix/modelMatrix without backuping it (ShadowMapManager work)
*/
virtual void generateShadowMap(const CVector &/* lightDir */) { }
/** get The shadow Map result for receveing. If NULL, nothing is displayed.
*/
virtual CShadowMap *getShadowMap() {return NULL;}
/** For receivers. get the World Instance bbox that includes the receiver.
*/
virtual void getReceiverBBox(CAABBox &bbox);
/** For receivers. Modulate the Object with a ShadowMap. The model shoud render in the scene driver a version
* of its geometry simplified, and modulate the background with shadowColor.
* \param casterPos the world position of the caster model.
* \param shadowMat a correclty setuped material with good ShadowColor, ready to be rendered.
*/
virtual void receiveShadowMap(CShadowMap * /* shadowMap */, const CVector &/* casterPos */, const CMaterial &/* shadowMat */) { }
/** For receivers. Retrieve the WorldMatrix of the model used for IDriver::render(). By default it returns getWorldMatrix().
* The exception is the Landscape and his "ZBuffer Problem" management.
*/
virtual const CMatrix &getReceiverRenderWorldMatrix() const {return getWorldMatrix();}
/// For ShadowMapManager. true if the model is rendering its ShadowMap this frame.
void setGeneratingShadowMap(bool state) {if(canCastShadowMap()) setStateFlag(IsGeneratingShadowMap, state);}
bool isGeneratingShadowMap() const {return getStateFlag(IsGeneratingShadowMap)!=0;}
/** Special For Skeleton Caster. When Skeletons cast shadows, they first compute the WorldBBox.
* The model should compute its bbox in World (best fit).
* \return false if the model don't support it (default), or if hidden in HRC!!
*/
virtual bool computeWorldBBoxForShadow(NLMISC::CAABBox &/* worldBB */) {return false;}
/** Special For Skeleton Caster. Render into the AuxDriver the mesh, within the current
* setuped Frustum/ViewMatrix.
* no-op by default, or if hidden in HRC!!
* \param rootSkeleton the skeleton which is currently rendering its shadowMap
*/
virtual void renderIntoSkeletonShadowMap(CSkeletonModel * /* rootSkeleton */, CMaterial &/* castMat */) { }
/** To limit some problems when the light direction is too on the XY axis.
* This method set an "angle" threshold for the shadow direction
* Actually, you give the minimum negative Z (not ang angle) the normalized shadow direction must have
* \param zthre possible values are in [-1,1].
* -1 force the direction to be (0,0,-1) in all case
* 0 means the z may be 0 (the direction is totaly XY), but at least the direction must go downward
* 1 means there is no restriction, the shadow direction can either be upward
* default is -0.5 (this implies a minimum angle of 30 degrees with the XY plane)
*/
void setShadowMapDirectionZThreshold(float zthre);
float getShadowMapDirectionZThreshold() const {return _ShadowMapDirectionZThreshold;}
/** To limit some problems with interior, a limit of shadow depth by caster can be given.
* This is the length in the lightDir direction where the shadow can touch receivers.
* Can be used also to have some big objects that cast shadows further.
* Default to 8.0.
*/
void setShadowMapMaxDepth(float depth);
float getShadowMapMaxDepth() const {return _ShadowMapMaxDepth;}
// @}
/** Force the transform to always be attached to the root
* As a consequence, it can't be inserted into a cluster system (even the root cluster)
* and is thus always visible when in the frustum (not clusterized)
* NB : any call to setClusterSystem will cause an assertion when the flag is set
* NB : any call to hrcUnlink will cause an assertion when the flag is set (must remain linked to the root)
*/
void setForceClipRoot(bool forceClipRoot);
bool getForceClipRoot() const { return getStateFlag(ForceClipRoot) != 0; }
// test if the model is a flare
virtual bool isFlare() const { return false; }
/// Don't use, used by CSkeletonSpawnScript to indicate that the WorldMatrix of this object is special
void setSSSWO(bool state) {setStateFlag(SSSWO, state);}
bool getSSSWO() const {return getStateFlag(SSSWO)!=0;}
// ********
private:
CHrcTrav::TVisibility Visibility;
static CTrackDefaultVector DefaultPos;
static CTrackDefaultVector DefaultPivot;
static CTrackDefaultVector DefaultRotEuler;
static CTrackDefaultQuat DefaultRotQuat;
static CTrackDefaultVector DefaultScale;
protected:
/** Constructor
* The deriver \b should do a \c TouchObs.resize(Last), to ensure he resize the BitSet correctly.
* The dervier \b should keep/declare ctor and dtor protected, to avoid user error (new and delete).
*/
CTransform();
/// Destructor
virtual ~CTransform();
/// special feature for CQuadGridClipManager. called at unfreezeHRC(). Used by CTransformShape.
virtual void unlinkFromQuadCluster() {}
/// \name Skinning Behavior.
// @{
/// Deriver must change this method if the model can be skinned. called rarely
virtual bool isSkinnable() const {return false;}
/** Deriver must change this method if isSkinnable(). called by CSkeletonModel::bindSkin()
* NB: _FatherSkeletonModel is valid when setApplySkin() is called
* The default behavior must be called: it sets the flag so isSkinned() return the good thing
*/
virtual void setApplySkin(bool state);
/** Deriver must change this method if isSkinnable(). It return the list of bone (correct skeleton index)
* used by the skins (NB: without the parents of the bone).
* NB: if an index is -1, it means that the skin bone has not been found in the skeleton (skip it)
* default is to return NULL.
*/
virtual const std::vector<sint32> *getSkinBoneUsage() const {return NULL;}
/** Deriver must change this method if isSkinnable(). It return a list of sphere relative to each bone
* of the father skeleton. Use with getSkinBoneUsage() to know to which bone this sphere apply
* NB: if a sphere radius is -1, it means that the bone is not used (for any reason...)
* default is to return NULL.
*/
virtual const std::vector<NLMISC::CBSphere> *getSkinBoneSphere() const {return NULL;}
/** Deriver must change this method if isSkinnable(). It renders the skin with current ctx of the skeletonModel
* SkeletonModel has already setuped the Light and the modelMatrix in the driver.
* If the skin is a MRM, it is the skeleton which drives the MRM level with alphaMRM: [0,1]
* default is nop
*/
virtual void renderSkin(float /* alphaMRM */) { }
/** Deriver may support SkinGrouping if isSkinnable().
* It renders the skin with current ctx of the skeletonModel, but torn in 2 pass: fillVB,a nd renderPrimitives
* Deriver may check NL3D_MESH_SKIN_MANAGER_VERTEXFORMAT and NL3D_MESH_SKIN_MANAGER_MAXVERTICES
*/
virtual bool supportSkinGrouping() const {return false;}
/** if supportSkinGrouping(), called to transform the VBuffer, and store it into dest.
* \return number of vertices added to the VBuffer, or -1 if > reaminingVertices
*/
virtual sint renderSkinGroupGeom(float /* alphaMRM */, uint /* remainingVertices */, uint8 * /* dest */) {return 0;}
/** if supportSkinGrouping(), called to render the primitives of the already skinned vertices (VB activated in the driver)
* Optionally, fill specRdrPasses with specular rdrPass to sort (used for specular grouping).
* \param baseVertex value to add to each PBlock index.
*/
virtual void renderSkinGroupPrimitives(uint /* baseVertex */, std::vector<CSkinSpecularRdrPass> &/* specularRdrPasses */, uint /* skinIndex */) { }
/// Render a specific specular renderPass returned by renderSkinGroupPrimitives
virtual void renderSkinGroupSpecularRdrPass(uint /* rdrPass */) { }
/// Special Skinning For ShadowMapping
virtual bool supportShadowSkinGrouping() const {return false;}
virtual sint renderShadowSkinGeom(uint /* remainingVertices */, uint8 * /* vbDest */) {return 0;}
virtual void renderShadowSkinPrimitives(CMaterial &/* castMat */, IDriver * /* drv */, uint /* baseVertex */) { }
/** Special use of skinning to compute intersection of a ray with it.
* \return false if not supported/no triangles, else true if can do the test (even if don't intersect!)
* if intersect, dist2D=0, and distZ= Depth Distance
* if don't intersect, dist2D="nearest distance to the ray", and distZ=0
* \param computeDist2D if false and don't intersect, then return dist2D=FLT_MAX, and distZ=0
*/
virtual bool supportIntersectSkin() const {return false;}
virtual bool intersectSkin(const CMatrix &/* toRaySpace */, float &/* dist2D */, float &/* distZ */, bool /* computeDist2D */) {return false;}
// The SkeletonModel, root of us (skinning or sticked object). NULL , if normal mode.
CSkeletonModel *_FatherSkeletonModel;
// If sticked object, id of the bone in the _FatherSkeletonModel.
uint _FatherBoneId;
// @}
/// name Lighting Behavior.
// @{
/// The contribution of all lights. This enlarge the struct only of approx 15%
CLightContribution _LightContribution;
/// non-zero if the object needs to updatelighting.
uint32 isNeedUpdateLighting() const {return getStateFlag(IsNeedUpdateLighting);}
/// non-zero if the object has a FrozenStaticLightSetup not correclty updated.
uint32 isNeedUpdateFrozenStaticLightSetup() const {return getStateFlag(IsNeedUpdateFrozenStaticLightSetup);}
/// each transform may be in a quadGird of lighted models (see CLightingManager)
CLightingManager::CQGItLightedModel _LightedModelIt;
// @}
/** get the channelMixer owned by the transform. return result of a refPtr => may be NULL.
*/
CChannelMixer *getChannelMixer() const {return _ChannelMixer;}
/// \name Model Feature initialisation.
// @{
/// Deriver must use this method with true to indicate the model support lighting.
void setIsLightable(bool val);
/** Deriver must use this method with true to indicate the model can be rendered.
* "can be rendered" means if object has to be inserted in RenderTrav list.
* eg: a mesh must be inserted in a render list, but not a light, or a NULL transform.
* The default is false.
*/
void setIsRenderable(bool val);
/// Deriver must use this method with true to indicate the model is a big lightable.
void setIsBigLightable(bool val);
/// For CSkeletonModel only.
void setIsSkeleton(bool val);
/** Deriver must use this method with true if the model must be AnimDetail-ed whatever
* registerToChannelMixer() has been called or not
*/
void setIsForceAnimDetail(bool val);
/// Deriver must use this method with true to indicate the model support loadBalancing.
void setIsLoadbalancable(bool val);
/// For CMeshBaseInstance only
void setIsMeshBaseInstance(bool val) {setStateFlag(IsMeshBaseInstance, val);}
/// For CTransformShape only.
void setIsTransformShape(bool val) {setStateFlag(IsTransformShape, val);}
/// For CCluster only.
void setIsCluster(bool val) {setStateFlag(IsCluster, val);}
/// ShadowMap
void setIsShadowMapCaster(bool val) {setStateFlag(IsShadowMapCaster, val);}
void setIsShadowMapReceiver(bool val) {setStateFlag(IsShadowMapReceiver, val);}
// @}
/// Test if obj must be displayed when sticked to an object displayed as a LOD (example: sword in hand of a character displayed as a LOD state)
bool getShowWhenLODSticked() const { return _ForceCLodSticked; }
// force to compute that transform matrix (useful if matrix needed but clipped because sticked to a clipped skeleton for example)
void forceCompute();
private:
static CTransform *creator() {return new CTransform;}
friend class CSkeletonModel;
friend class CScene;
friend class CClipTrav;
friend class CAnimDetailTrav;
friend class CRenderTrav;
// The Scene which owns us
CScene *_OwnerScene;
/// \name Hrc / Clip hierarchy.
// @{
// Hrc hierarchy. One parent and possible multiple sons
CFastPtrListNode _HrcNode;
CFastPtrList<CTransform> _HrcSons;
CTransform *_HrcParent;
NLMISC::CRefPtr<CTransform> _HrcParentUnfreeze;
/* Clip Graph. DAG (Direct Acyclic Graph)
* NB: implementation optmized for Low number of parent. clipAddChild() and clipDelChild() is in O(numParents).
*/
struct CClipNode
{
CFastPtrListNode ClipNode;
CTransform *Parent;
};
CFastPtrList<CTransform> _ClipSons;
std::vector<CClipNode*> _ClipParents;
bool clipHasParent(CTransform *parent);
void clipDelFromParent(CTransform *parent);
// @}
/// \name Model updating/traversing features
// @{
// linked list of models to update.
CTransform *_PrecModelToUpdate;
CTransform *_NextModelToUpdate;
// for CScene::createModel() and for CTransform::freezeHRC() only.
void linkToUpdateList();
void unlinkFromUpdateList();
// @}
// For anim detail.
NLMISC::CRefPtr<CChannelMixer> _ChannelMixer;
// Last date of ITransformable matrix.
uint64 _LastTransformableMatrixDate;
NLMISC::CRefPtr<CInstanceGroup> _ClusterSystem;
enum TFreezeHRCState { FreezeHRCStateDisabled=0, FreezeHRCStateRequest, FreezeHRCStateReady, FreezeHRCStateEnabled};
TFreezeHRCState _FreezeHRCState;
uint8 _OrderingLayer;
uint8 _TransparencyPriority;
// For stickObjectEx(). with forceCLod==true
bool _ForceCLodSticked : 1;
/// true if need to compute transform
bool _TransformDirty : 1;
// see supportFastIntersect. Filled by CSkeletonModel
bool _SupportFastIntersect : 1;
/// See ILogicInfo. Used for lighting. default is NULL.
ILogicInfo *_LogicInfo;
/// see setMeanColor()
CRGBA _MeanColor;
/// For Shadow Caster registration to list
std::list<CTransform*>::iterator _ItShadowCasterInScene;
/// For Shadow Casters
float _ShadowMapDirectionZThreshold;
float _ShadowMapMaxDepth;
/// \name State Flag mgt (boolean compression)
// @{
/// State Flags.
enum TState {
// Post-clipping Traversal flags. If set, then the object is inserted into traversal list.
IsAnimDetailable= 0x0001,
IsLoadBalancable= 0x0002,
IsLightable= 0x0004,
IsRenderable= 0x0008,
// Transparency Flags.
IsTransparent= 0x0010,
IsOpaque= 0x0020,
// For fast clip.
QuadGridClipEnabled= 0x0040,
// Lighting.
IsUserLightable= 0x0080,
IsFinalLightable= 0x0100, // IsLightable && IsUserLightable
IsBigLightable= 0x0200,
IsNeedUpdateLighting= 0x0400,
IsNeedUpdateFrozenStaticLightSetup=
0x0800,
// Skinning
IsSkeleton= 0x1000, // set if the model is a skeleton (faster than dynamic_cast)
IsSkinned= 0x2000, // true if the model is isSkinnable() and if currently skinned
// Misc
IsDeleteChannelMixer= 0x4000,
IsForceAnimDetail= 0x8000,
IsMeshBaseInstance= 0x10000,
IsTransformShape= 0x20000, // set if the model is a transform_shape (faster than dynamic_cast)
IsCluster= 0x40000, // set if the model is a cluster (faster than dynamic_cast)
UserClipping= 0x80000, // set if the user provide a clip method, don't call clip() in ClipTrav
// ShadowMap
IsShadowMapCaster= 0x100000, // set if the model can cast ShadowMap
IsFinalShadowMapCaster= 0x200000, // set if the model can cast ShadowMap AND the user want it
IsShadowMapReceiver= 0x400000, // set if the model can receive ShadowMap
IsFinalShadowMapReceiver= 0x800000, // set if the model can receive ShadowMap AND the user want it
IsGeneratingShadowMap= 0x1000000, // temp set if the model is asked to render its shadowMap this frame.
ForceClipRoot = 0x2000000,// Force the object to always be attached to the root
// As a consequence, it can't be inserted into a cluster system (even the root cluster)
// and is thus always visible when in the frustum
ClusterSystemAuto = 0x4000000,
SSSWO = 0x8000000, // Special for SkeletonSpawnScript. if set, the WorldMatrix is special
BypassLODOpacity = 0x10000000 // for mesh multi lod : do not use the LOD opacity / transparency, but the parent one (overwritten at traversal, else ...)
// NB: may continue on >=0x20000000
};
/// Flags for the General State of the Transform. They are both static or dynamic flags.
uint32 _StateFlags;
/// This is used to set Static or dynamic flags. val must take 0 or 1.
void setStateFlag(uint32 mask, bool val)
{
// reset the state.
_StateFlags&= ~mask;
// set the state
_StateFlags|= ( 0- ((uint32)val) ) & mask;
}
/// return a non zero-value if state is set.
uint32 getStateFlag(uint32 mask) const
{
return _StateFlags&mask;
}
// @}
protected:
/** State for renderFiltering. Default is 0xFFFFFFFF (always displayed)
* Deriver work to change this value
*/
uint32 _RenderFilterType;
protected:
/// \name Hrc Traversal
// @{
/// Hrc IN variables.
CMatrix _LocalMatrix;
CHrcTrav::TVisibility _LocalVis; // The visibility state of the node.
sint64 _LocalDate; // The update date of the LocalMatrix.
/// Hrc OUT variables.
CMatrix _WorldMatrix;
sint64 _WorldDate; // The update date of the WorldMatrix.
bool _WorldVis; // Is the node visible? (enabled?)
// Transform Specicic Hrc
bool _Frozen;
bool _DontUnfreezeChildren; // Useful when cluster system move to not test instance again
bool _ClipLinkedInSonsOfAncestorSkeletonModelGroup;
// !NULL if any skeleton is an ancestor in hierarchy. Updated at each Hrc traversal!!
CSkeletonModel *_AncestorSkeletonModel;
/// Update the world state according to the parent world state and the local states.
void updateWorld();
// according to _AncestorSkeletonModel, link clipTrav.
void updateClipTravForAncestorSkeleton();
// @}
/// \name Clip Traversal
// @{
/// date of last traverseClip()
sint64 _ClipDate;
// The index of the Observer in the _VisibleList; -1 (default) means not in
sint _IndexInVisibleList;
/// set to true is the object is visible (not clipped).
bool _Visible;
// @}
/// \name Render Traversal
// @{
// Used by CRenderTrav. see CRenderTrav::removeRenderModel() implementation
uint8 _IndexLSBInRenderList;
// @}
/// \name AnimDetail Traversal
// @{
/** For Skeleton Object Stick.
* update the wolrd matrix. no-op if skinned. no-op if no AcnestorSkeletonModel.
* use standard father WorldMatrix if !_FatherSkeletonModel else get the correct boneId
* WorldMatrix from _FatherSkeletonModel
*/
void updateWorldMatrixFromFather();
/** traverse without updatin WorldMatrixFromFather:
* - animdetail if the model channelmixer is not NULL, and if model not clipped
*/
void traverseAnimDetailWithoutUpdateWorldMatrix();
// @}
/// \name LoadBalancing Traversal
// @{
// Which group owns this model
CLoadBalancingGroup *_LoadBalancingGroup;
// @}
/// \name ShadowMap
// @{
/** To implement for ShadowCaster support. typically allocate a CShadowMap and store
* NB: the texture doesn't have to be inited at this time. Update it each frame in generateShadowMap()
* MUST call registerShadowCasterToList()
*/
virtual void createShadowMap() {}
/// To implement for ShadowCaster support. typically free the shadowMap + MUST call unregisterShadowCasterToList()
virtual void deleteShadowMap() {}
// @}
};
} // namespace NL3D
#endif // NL_TRANSFORM_H
/* End of transform.h */