You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1581 lines
44 KiB
C++
1581 lines
44 KiB
C++
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
|
// Copyright (C) 2010 Winch Gate Property Limited
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as
|
|
// published by the Free Software Foundation, either version 3 of the
|
|
// License, or (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Affero General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
#include "std3d.h"
|
|
|
|
#include "nel/3d/scene_group.h"
|
|
#include "nel/misc/stream.h"
|
|
#include "nel/misc/matrix.h"
|
|
#include "nel/3d/scene.h"
|
|
#include "nel/3d/transform_shape.h"
|
|
#include "nel/3d/mesh_instance.h"
|
|
#include "nel/3d/shape_bank.h"
|
|
#include "nel/3d/u_instance_group.h"
|
|
#include "nel/3d/vertex_buffer.h"
|
|
#include "nel/3d/index_buffer.h"
|
|
#include "nel/3d/text_context.h"
|
|
#include "nel/3d/water_model.h"
|
|
#include "nel/3d/water_shape.h"
|
|
#include "nel/misc/polygon.h"
|
|
#include "nel/misc/path.h"
|
|
|
|
|
|
using namespace NLMISC;
|
|
using namespace std;
|
|
|
|
#ifdef DEBUG_NEW
|
|
#define new DEBUG_NEW
|
|
#endif
|
|
|
|
namespace NL3D
|
|
{
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// CInstance
|
|
// ---------------------------------------------------------------------------
|
|
|
|
// ***************************************************************************
|
|
CInstanceGroup::CInstance::CInstance ()
|
|
{
|
|
/* ***********************************************
|
|
* WARNING: This Class/Method must be thread-safe (ctor/dtor/serial): no static access for instance
|
|
* It can be loaded/called through CAsyncFileManager for instance
|
|
* ***********************************************/
|
|
|
|
DontAddToScene = false;
|
|
AvoidStaticLightPreCompute= false;
|
|
StaticLightEnabled= false;
|
|
DontCastShadow= false;
|
|
LocalAmbientId= 0xFF;
|
|
DontCastShadowForInterior= false;
|
|
DontCastShadowForExterior= false;
|
|
Visible= true;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::CInstance::serial (NLMISC::IStream& f)
|
|
{
|
|
/* ***********************************************
|
|
* WARNING: This Class/Method must be thread-safe (ctor/dtor/serial): no static access for instance
|
|
* It can be loaded/called through CAsyncFileManager for instance
|
|
* ***********************************************/
|
|
|
|
/*
|
|
Version 7:
|
|
- Visible
|
|
Version 6:
|
|
- DontCastShadowForExterior
|
|
Version 5:
|
|
- DontCastShadowForInterior
|
|
Version 4:
|
|
- LocalAmbientId.
|
|
Version 3:
|
|
- StaticLight.
|
|
Version 2:
|
|
- gameDev data.
|
|
Version 1:
|
|
- Clusters
|
|
*/
|
|
// Serial a version number
|
|
sint version=f.serialVersion (7);
|
|
|
|
|
|
// Visible
|
|
if (version >= 7)
|
|
f.serial(Visible);
|
|
|
|
// DontCastShadowForExterior
|
|
if (version >= 6)
|
|
f.serial(DontCastShadowForExterior);
|
|
else
|
|
DontCastShadowForExterior= false;
|
|
|
|
// DontCastShadowForInterior
|
|
if (version >= 5)
|
|
f.serial(DontCastShadowForInterior);
|
|
else
|
|
DontCastShadowForInterior= false;
|
|
|
|
// Serial the LocalAmbientId.
|
|
if (version >= 4)
|
|
{
|
|
f.serial(LocalAmbientId);
|
|
}
|
|
else if(f.isReading())
|
|
{
|
|
LocalAmbientId= 0xFF;
|
|
}
|
|
|
|
// Serial the StaticLight
|
|
if (version >= 3)
|
|
{
|
|
f.serial (AvoidStaticLightPreCompute);
|
|
f.serial (DontCastShadow);
|
|
f.serial (StaticLightEnabled);
|
|
f.serial (SunContribution);
|
|
nlassert(CInstanceGroup::NumStaticLightPerInstance==2);
|
|
f.serial (Light[0]);
|
|
f.serial (Light[1]);
|
|
}
|
|
else if(f.isReading())
|
|
{
|
|
AvoidStaticLightPreCompute= false;
|
|
StaticLightEnabled= false;
|
|
DontCastShadow= false;
|
|
}
|
|
|
|
// Serial the gamedev data
|
|
if (version >= 2)
|
|
{
|
|
f.serial (InstanceName);
|
|
f.serial (DontAddToScene);
|
|
}
|
|
|
|
// Serial the clusters
|
|
if (version >= 1)
|
|
f.serialCont (Clusters);
|
|
|
|
// Serial the name
|
|
f.serial (Name);
|
|
|
|
// Serial the position vector
|
|
f.serial (Pos);
|
|
|
|
// Serial the rotation vector
|
|
f.serial (Rot);
|
|
|
|
// Serial the scale vector
|
|
f.serial (Scale);
|
|
|
|
// Serial the parent location in the vector (-1 if no parent)
|
|
f.serial (nParent);
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// CInstanceGroup
|
|
// ---------------------------------------------------------------------------
|
|
|
|
// ***************************************************************************
|
|
|
|
uint CInstanceGroup::getNumInstance () const
|
|
{
|
|
return (uint)_InstancesInfos.size();
|
|
}
|
|
|
|
// ***************************************************************************
|
|
|
|
const string& CInstanceGroup::getShapeName (uint instanceNb) const
|
|
{
|
|
// Return the name of the n-th instance
|
|
return _InstancesInfos[instanceNb].Name;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
|
|
const string& CInstanceGroup::getInstanceName (uint instanceNb) const
|
|
{
|
|
// Return the name of the n-th instance
|
|
return _InstancesInfos[instanceNb].InstanceName;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
|
|
const CVector& CInstanceGroup::getInstancePos (uint instanceNb) const
|
|
{
|
|
// Return the position vector of the n-th instance
|
|
return _InstancesInfos[instanceNb].Pos;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
|
|
const CQuat& CInstanceGroup::getInstanceRot (uint instanceNb) const
|
|
{
|
|
// Return the rotation vector of the n-th instance
|
|
return _InstancesInfos[instanceNb].Rot;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
|
|
const CVector& CInstanceGroup::getInstanceScale (uint instanceNb) const
|
|
{
|
|
// Return the scale vector of the n-th instance
|
|
return _InstancesInfos[instanceNb].Scale;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
|
|
void CInstanceGroup::getInstanceMatrix(uint instanceNb,NLMISC::CMatrix &dest) const
|
|
{
|
|
dest.identity();
|
|
dest.translate(getInstancePos(instanceNb));
|
|
dest.rotate(getInstanceRot(instanceNb));
|
|
dest.scale(getInstanceScale(instanceNb));
|
|
}
|
|
|
|
|
|
|
|
// ***************************************************************************
|
|
|
|
sint32 CInstanceGroup::getInstanceParent (uint instanceNb) const
|
|
{
|
|
// Return the scale vector of the n-th instance
|
|
return _InstancesInfos[instanceNb].nParent;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
const CInstanceGroup::CInstance &CInstanceGroup::getInstance(uint instanceNb) const
|
|
{
|
|
return _InstancesInfos[instanceNb];
|
|
}
|
|
|
|
// ***************************************************************************
|
|
CInstanceGroup::CInstance &CInstanceGroup::getInstance(uint instanceNb)
|
|
{
|
|
return _InstancesInfos[instanceNb];
|
|
}
|
|
|
|
// ***************************************************************************
|
|
CTransformShape *CInstanceGroup::getTransformShape(uint instanceNb) const
|
|
{
|
|
if(instanceNb>_Instances.size())
|
|
return NULL;
|
|
return _Instances[instanceNb];
|
|
}
|
|
|
|
// ***************************************************************************
|
|
CInstanceGroup::CInstanceGroup()
|
|
{
|
|
/* ***********************************************
|
|
* WARNING: This Class/Method must be thread-safe (ctor/dtor/serial): no static access for instance
|
|
* It can be loaded/called through CAsyncFileManager for instance
|
|
* ***********************************************/
|
|
|
|
_IGSurfaceLight.setOwner(this);
|
|
_GlobalPos = CVector(0,0,0);
|
|
_Root = NULL;
|
|
_ClusterSystemForInstances = NULL;
|
|
_ParentClusterSystem = NULL;
|
|
_RealTimeSunContribution= true;
|
|
_AddToSceneState = StateNotAdded;
|
|
_TransformName = NULL;
|
|
_AddRemoveInstance = NULL;
|
|
_IGAddBeginCallback = NULL;
|
|
_UserIg= NULL;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
CInstanceGroup::~CInstanceGroup()
|
|
{
|
|
/* ***********************************************
|
|
* WARNING: This Class/Method must be thread-safe (ctor/dtor/serial): no static access for instance
|
|
* It can be loaded/called through CAsyncFileManager for instance
|
|
* ***********************************************/
|
|
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::build (const CVector &vGlobalPos, const TInstanceArray& array,
|
|
const std::vector<CCluster>& Clusters,
|
|
const std::vector<CPortal>& Portals,
|
|
const std::vector<CPointLightNamed> &pointLightList,
|
|
const CIGSurfaceLight::TRetrieverGridMap *retrieverGridMap,
|
|
float igSurfaceLightCellSize)
|
|
{
|
|
_GlobalPos = vGlobalPos;
|
|
// Copy the array
|
|
_InstancesInfos = array;
|
|
|
|
_Portals = Portals;
|
|
_ClusterInfos = Clusters;
|
|
|
|
// Link portals and clusters
|
|
uint32 i, j, k;
|
|
for (i = 0; i < _Portals.size(); ++i)
|
|
{
|
|
for (j = 0; j < _ClusterInfos.size(); ++j)
|
|
{
|
|
bool bPortalInCluster = true;
|
|
for (k = 0; k < _Portals[i]._Poly.size(); ++k)
|
|
if (!_ClusterInfos[j].isIn (_Portals[i]._Poly[k]) )
|
|
{
|
|
bPortalInCluster = false;
|
|
break;
|
|
}
|
|
if (bPortalInCluster)
|
|
{
|
|
_Portals[i].setCluster(&_ClusterInfos[j]);
|
|
_ClusterInfos[j].link (&_Portals[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Create Meta Cluster if needed
|
|
/*
|
|
CCluster clusterTemp;
|
|
bool mustAdd = false;
|
|
for (i = 0; i < _Portals.size(); ++i)
|
|
if (_Portals[i].getNbCluster() == 1)
|
|
{
|
|
mustAdd = true;
|
|
break;
|
|
}
|
|
if (mustAdd)
|
|
{
|
|
CCluster clusterTemp;
|
|
_ClusterInfos.push_back(clusterTemp);
|
|
CCluster *pMetaCluster = &_ClusterInfos[_ClusterInfos.size()-1];
|
|
pMetaCluster->setMetaCluster();
|
|
for (i = 0; i < _Portals.size(); ++i)
|
|
if (_Portals[i].getNbCluster() == 1)
|
|
{
|
|
_Portals[i].setCluster(pMetaCluster);
|
|
pMetaCluster->link(&_Portals[i]);
|
|
}
|
|
}*/
|
|
|
|
|
|
// Build the list of light. NB: sort by LightGroupName the array.
|
|
std::vector<uint> plRemap;
|
|
buildPointLightList(pointLightList, plRemap);
|
|
|
|
// Build IgSurfaceLight
|
|
// clear
|
|
_IGSurfaceLight.clear();
|
|
if(retrieverGridMap)
|
|
{
|
|
//build
|
|
_IGSurfaceLight.build(*retrieverGridMap, igSurfaceLightCellSize, plRemap);
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::build (const CVector &vGlobalPos, const TInstanceArray& array,
|
|
const std::vector<CCluster>& Clusters,
|
|
const std::vector<CPortal>& Portals)
|
|
{
|
|
// empty pointLightList
|
|
std::vector<CPointLightNamed> pointLightList;
|
|
|
|
build(vGlobalPos, array, Clusters, Portals, pointLightList);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::retrieve (CVector &vGlobalPos, TInstanceArray& array,
|
|
std::vector<CCluster>& Clusters,
|
|
std::vector<CPortal>& Portals,
|
|
std::vector<CPointLightNamed> &pointLightList) const
|
|
{
|
|
// Just copy infos. NB: light information order have change but is still valid
|
|
vGlobalPos= _GlobalPos;
|
|
array= _InstancesInfos;
|
|
|
|
Portals= _Portals;
|
|
Clusters= _ClusterInfos;
|
|
// Must reset links to all portals and clusters.
|
|
uint i;
|
|
for(i=0; i<Portals.size(); i++)
|
|
Portals[i].resetClusterLinks();
|
|
for(i=0; i<Clusters.size(); i++)
|
|
Clusters[i].resetPortalLinks();
|
|
|
|
|
|
pointLightList= getPointLightList();
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
|
|
void CInstanceGroup::serial (NLMISC::IStream& f)
|
|
{
|
|
/* ***********************************************
|
|
* WARNING: This Class/Method must be thread-safe (ctor/dtor/serial): no static access for instance
|
|
* It can be loaded/called through CAsyncFileManager for instance
|
|
* ***********************************************/
|
|
|
|
// Serial a header
|
|
f.serialCheck (NELID("TPRG"));
|
|
|
|
/*
|
|
Version 5:
|
|
_ _RealTimeSunContribution
|
|
Version 4:
|
|
_ IGSurfaceLight
|
|
Version 3:
|
|
- PointLights
|
|
*/
|
|
// Serial a version number
|
|
sint version=f.serialVersion (5);
|
|
|
|
|
|
// _RealTimeSunContribution
|
|
if (version >= 5)
|
|
{
|
|
f.serial(_RealTimeSunContribution);
|
|
}
|
|
else if(f.isReading())
|
|
{
|
|
_RealTimeSunContribution= true;
|
|
}
|
|
|
|
|
|
// Serial the IGSurfaceLight
|
|
if (version >= 4)
|
|
{
|
|
f.serial(_IGSurfaceLight);
|
|
}
|
|
else if(f.isReading())
|
|
{
|
|
_IGSurfaceLight.clear();
|
|
}
|
|
|
|
|
|
// Serial the PointLights info
|
|
if (version >= 3)
|
|
{
|
|
f.serial(_PointLightArray);
|
|
}
|
|
else if(f.isReading())
|
|
{
|
|
_PointLightArray.clear();
|
|
}
|
|
|
|
if (version >= 2)
|
|
f.serial(_GlobalPos);
|
|
|
|
if (version >= 1)
|
|
{
|
|
f.serialCont (_ClusterInfos);
|
|
f.serialCont (_Portals);
|
|
// Links
|
|
if (f.isReading())
|
|
{
|
|
uint32 i, j;
|
|
for (i = 0; i < _ClusterInfos.size(); ++i)
|
|
{
|
|
uint32 nNbPortals;
|
|
f.serial (nNbPortals);
|
|
_ClusterInfos[i]._Portals.resize (nNbPortals);
|
|
// Recreate clusters to portals links
|
|
for (j = 0; j < nNbPortals; ++j)
|
|
{
|
|
sint32 nPortalNb;
|
|
f.serial (nPortalNb);
|
|
_ClusterInfos[i]._Portals[j] = &_Portals[nPortalNb];
|
|
_Portals[nPortalNb].setCluster (&_ClusterInfos[i]);
|
|
}
|
|
}
|
|
}
|
|
else // We are writing to the stream
|
|
{
|
|
uint32 i, j;
|
|
for (i = 0; i < _ClusterInfos.size(); ++i)
|
|
{
|
|
uint32 nNbPortals = (uint32)_ClusterInfos[i]._Portals.size();
|
|
f.serial (nNbPortals);
|
|
for (j = 0; j < nNbPortals; ++j)
|
|
{
|
|
sint32 nPortalNb = (sint32)(_ClusterInfos[i]._Portals[j] - &_Portals[0]);
|
|
f.serial (nPortalNb);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Serial the array
|
|
f.serialCont (_InstancesInfos);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::createRoot (CScene& scene)
|
|
{
|
|
_Root = (CTransform*)scene.createModel (TransformId);
|
|
_Root->setDontUnfreezeChildren (true);
|
|
setPos (CVector(0,0,0));
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::setTransformNameCallback (ITransformName *pTN)
|
|
{
|
|
_TransformName = pTN;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::setAddRemoveInstanceCallback(IAddRemoveInstance *callback)
|
|
{
|
|
_AddRemoveInstance = callback;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::setIGAddBeginCallback(IIGAddBegin *callback)
|
|
{
|
|
_IGAddBeginCallback = callback;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
bool CInstanceGroup::addToScene (CScene& scene, IDriver *driver, uint selectedTexture)
|
|
{
|
|
// Init the scene lights
|
|
_PointLightArray.initAnimatedLightIndex (scene);
|
|
|
|
uint32 i, j;
|
|
|
|
// Test if portals are linked to their 2 clusters
|
|
for (i = 0; i < _Portals.size(); ++i)
|
|
for (j = 0; j < 2; ++j)
|
|
{
|
|
if (_Portals[i]._Clusters[j] == NULL)
|
|
{
|
|
nlwarning("Portal %d (name:%s) is not linked to 2 clusters. Instance Group Not Added To Scene.", i, _Portals[i].getName().c_str());
|
|
}
|
|
}
|
|
|
|
_Instances.resize (_InstancesInfos.size(), NULL);
|
|
|
|
if (_IGAddBeginCallback)
|
|
_IGAddBeginCallback->startAddingIG((uint)_InstancesInfos.size());
|
|
|
|
// Creation and positionning of the new instance
|
|
|
|
vector<CInstance>::iterator it = _InstancesInfos.begin();
|
|
|
|
// Water surface may have a callback when they are created, and this callback need their position
|
|
// Their position isn't set right now however, so must call that callback later
|
|
IWaterSurfaceAddedCallback *oldCallback = scene.getWaterCallback();
|
|
scene.setWaterCallback(NULL);
|
|
for (i = 0; i < _InstancesInfos.size(); ++i, ++it)
|
|
{
|
|
// Get the shape name
|
|
string shapeName;
|
|
getShapeName (i, shapeName);
|
|
if (!shapeName.empty ())
|
|
{
|
|
if (!_InstancesInfos[i].DontAddToScene)
|
|
{
|
|
// TMP FIX : some pacs_prim files where exported ...
|
|
if (nlstricmp(NLMISC::CFile::getExtension(shapeName), "pacs_prim") == 0)
|
|
{
|
|
nlwarning("Can't read %s (not a shape)", shapeName.c_str());
|
|
_Instances[i] = NULL;
|
|
}
|
|
else
|
|
{
|
|
_Instances[i] = scene.createInstance (shapeName);
|
|
|
|
if (_Instances[i] == NULL)
|
|
{
|
|
nlwarning("Not found '%s' file", shapeName.c_str());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
scene.setWaterCallback(oldCallback);
|
|
return addToSceneWhenAllShapesLoaded (scene, driver, selectedTexture);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
|
|
void CInstanceGroup::getShapeName (uint instanceIndex, std::string &shapeName) const
|
|
{
|
|
const CInstance &rInstanceInfo = _InstancesInfos[instanceIndex];
|
|
shapeName = rInstanceInfo.Name;
|
|
|
|
// If there is a callback added to this instance group then transform
|
|
// the name of the shape to load.
|
|
if (_TransformName != NULL && !rInstanceInfo.InstanceName.empty())
|
|
{
|
|
shapeName = _TransformName->transformName (instanceIndex, rInstanceInfo.InstanceName, rInstanceInfo.Name);
|
|
}
|
|
|
|
toLower(shapeName);
|
|
if (!shapeName.empty() && shapeName.find('.') == std::string::npos)
|
|
shapeName += ".shape";
|
|
}
|
|
|
|
// ***************************************************************************
|
|
// Private method
|
|
bool CInstanceGroup::addToSceneWhenAllShapesLoaded (CScene& scene, IDriver *driver, uint selectedTexture)
|
|
{
|
|
uint32 i, j;
|
|
|
|
vector<CInstance>::iterator it = _InstancesInfos.begin();
|
|
for (i = 0; i < _InstancesInfos.size(); ++i, ++it)
|
|
{
|
|
CInstance &rInstanceInfo = *it;
|
|
|
|
if (!rInstanceInfo.DontAddToScene)
|
|
{
|
|
if (_Instances[i])
|
|
{
|
|
_Instances[i]->setPos (rInstanceInfo.Pos);
|
|
_Instances[i]->setRotQuat (rInstanceInfo.Rot);
|
|
_Instances[i]->setScale (rInstanceInfo.Scale);
|
|
_Instances[i]->setPivot (CVector::Null);
|
|
if(rInstanceInfo.Visible)
|
|
_Instances[i]->show();
|
|
else
|
|
_Instances[i]->hide();
|
|
|
|
if (scene.getWaterCallback())
|
|
{
|
|
CWaterModel *wm = dynamic_cast<CWaterModel *>(_Instances[i]);
|
|
if (wm)
|
|
{
|
|
const CWaterShape *ws = safe_cast<const CWaterShape *>((const IShape *) wm->Shape);
|
|
scene.getWaterCallback()->waterSurfaceAdded(ws->getShape(), wm->getMatrix(), ws->isSplashEnabled(), ws->getUseSceneWaterEnvMap(0) || ws->getUseSceneWaterEnvMap(1));
|
|
}
|
|
}
|
|
// Static Light Setup
|
|
if( rInstanceInfo.StaticLightEnabled )
|
|
{
|
|
// Count lights.
|
|
uint numPointLights;
|
|
for(numPointLights= 0; numPointLights<CInstanceGroup::NumStaticLightPerInstance; numPointLights++)
|
|
{
|
|
if(rInstanceInfo.Light[numPointLights]==0xFF)
|
|
break;
|
|
}
|
|
// Max allowed.
|
|
numPointLights= min(numPointLights, (uint)NL3D_MAX_LIGHT_CONTRIBUTION);
|
|
|
|
// Get pl ptrs.
|
|
CPointLight *pls[CInstanceGroup::NumStaticLightPerInstance];
|
|
for(uint j=0; j<numPointLights;j++)
|
|
{
|
|
uint plId= rInstanceInfo.Light[j];
|
|
pls[j]= (CPointLight*)(&_PointLightArray.getPointLights()[plId]);
|
|
}
|
|
|
|
// get frozenAmbientlight.
|
|
CPointLight *frozenAmbientlight;
|
|
if(rInstanceInfo.LocalAmbientId == 0xFF)
|
|
// must take the sun one.
|
|
frozenAmbientlight= NULL;
|
|
else
|
|
// ok, take the local ambient one.
|
|
frozenAmbientlight= (CPointLight*)(&_PointLightArray.getPointLights()[rInstanceInfo.LocalAmbientId]);
|
|
|
|
// Setup the instance.
|
|
_Instances[i]->freezeStaticLightSetup(pls, numPointLights, rInstanceInfo.SunContribution, frozenAmbientlight);
|
|
}
|
|
|
|
// Driver not NULL ?
|
|
if (driver)
|
|
{
|
|
// Flush shape's texture with this driver
|
|
_Instances[i]->Shape->flushTextures (*driver, selectedTexture);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
_Instances[i] = NULL;
|
|
}
|
|
}
|
|
|
|
// Setup the hierarchy
|
|
// We just have to set the traversal HRC (Hierarchy)
|
|
if (_Root == NULL)
|
|
{
|
|
createRoot (scene);
|
|
}
|
|
it = _InstancesInfos.begin();
|
|
for (i = 0; i < _InstancesInfos.size(); ++i, ++it)
|
|
if (!_InstancesInfos[i].DontAddToScene && _Instances[i] != NULL)
|
|
{
|
|
CInstance &rInstanceInfo = *it;
|
|
if( rInstanceInfo.nParent != -1 ) // Is the instance get a parent
|
|
_Instances[rInstanceInfo.nParent]->hrcLinkSon( _Instances[i] );
|
|
else
|
|
_Root->hrcLinkSon( _Instances[i] );
|
|
}
|
|
// Attach the root of the instance group to the root of the hierarchy traversal
|
|
scene.getRoot()->hrcLinkSon( _Root );
|
|
|
|
// Cluster / Portals
|
|
// -----------------
|
|
|
|
CClipTrav *pClipTrav = &scene.getClipTrav();
|
|
_ClipTrav = pClipTrav;
|
|
|
|
// Create the MOT links (create the physical clusters)
|
|
_ClusterInstances.resize (_ClusterInfos.size());
|
|
for (i = 0; i < _ClusterInstances.size(); ++i)
|
|
{
|
|
_ClusterInstances[i] = (CCluster*)scene.createModel (ClusterId);
|
|
_ClusterInstances[i]->Group = this;
|
|
_ClusterInstances[i]->_Portals = _ClusterInfos[i]._Portals;
|
|
_ClusterInstances[i]->_LocalVolume = _ClusterInfos[i]._LocalVolume;
|
|
_ClusterInstances[i]->_LocalBBox = _ClusterInfos[i]._LocalBBox;
|
|
_ClusterInstances[i]->_Volume = _ClusterInfos[i]._Volume;
|
|
_ClusterInstances[i]->_BBox = _ClusterInfos[i]._BBox;
|
|
_ClusterInstances[i]->FatherVisible = _ClusterInfos[i].FatherVisible;
|
|
_ClusterInstances[i]->VisibleFromFather = _ClusterInfos[i].VisibleFromFather;
|
|
_ClusterInstances[i]->FatherAudible = _ClusterInfos[i].FatherAudible;
|
|
_ClusterInstances[i]->AudibleFromFather = _ClusterInfos[i].AudibleFromFather;
|
|
_ClusterInstances[i]->Name = _ClusterInfos[i].Name;
|
|
_ClusterInstances[i]->setSoundGroup(_ClusterInfos[i].getSoundGroup());
|
|
_ClusterInstances[i]->setEnvironmentFx(_ClusterInfos[i].getEnvironmentFx());
|
|
pClipTrav->registerCluster (_ClusterInstances[i]);
|
|
_ClusterInstances[i]->clipUnlinkFromAll();
|
|
}
|
|
|
|
// Relink portals with newly created clusters
|
|
for (i = 0; i < _Portals.size(); ++i)
|
|
for (j = 0; j < 2; ++j)
|
|
{
|
|
if (_Portals[i]._Clusters[j])
|
|
{
|
|
sint32 nClusterNb;
|
|
nClusterNb = (sint32)(_Portals[i]._Clusters[j] - &_ClusterInfos[0]);
|
|
_Portals[i]._Clusters[j] = _ClusterInstances[nClusterNb];
|
|
}
|
|
}
|
|
|
|
// Link shapes to clusters
|
|
for (i = 0; i < _Instances.size(); ++i)
|
|
if (_Instances[i] != NULL && !_InstancesInfos[i].DontAddToScene)
|
|
{
|
|
if (!_InstancesInfos[i].Clusters.empty())
|
|
{
|
|
_Instances[i]->clipUnlinkFromAll();
|
|
for (j = 0; j < _InstancesInfos[i].Clusters.size(); ++j)
|
|
{
|
|
uint32 clusterInst = _InstancesInfos[i].Clusters[j];
|
|
if (clusterInst < _ClusterInstances.size())
|
|
_ClusterInstances[clusterInst]->clipAddChild( _Instances[i] );
|
|
else
|
|
nlwarning("IG: BUG: Cluster infos size %u, indexing %u", (uint32)_ClusterInstances.size(), clusterInst);
|
|
}
|
|
// For the first time we have to set all the instances to NOT move (and not be rebinded)
|
|
_Instances[i]->freeze();
|
|
_Instances[i]->setClusterSystem (this);
|
|
}
|
|
else
|
|
{
|
|
// These instances are not attached to a cluster at this level so we cannot freeze them
|
|
// Moreover we must set their clustersystem they will be tested against
|
|
_Instances[i]->setClusterSystem (_ClusterSystemForInstances);
|
|
}
|
|
}
|
|
_Root->freeze();
|
|
|
|
// HRC OBS like
|
|
for (i = 0; i < _ClusterInstances.size(); ++i)
|
|
{
|
|
_ClusterInstances[i]->setWorldMatrix (_Root->getMatrix());
|
|
|
|
for (j = 0; j < _ClusterInstances[i]->getNbPortals(); ++j)
|
|
{
|
|
CPortal *pPortal = _ClusterInstances[i]->getPortal(j);
|
|
pPortal->setWorldMatrix (_Root->getMatrix());
|
|
}
|
|
|
|
// Re affect the cluster to the accelerator if not the root
|
|
if (!_ClusterInstances[i]->isRoot())
|
|
{
|
|
_ClipTrav->unregisterCluster(_ClusterInstances[i]);
|
|
_ClipTrav->registerCluster (_ClusterInstances[i]);
|
|
}
|
|
}
|
|
|
|
|
|
// Link the instance group to the parent
|
|
linkToParent (scene.getGlobalInstanceGroup());
|
|
|
|
// Attach the clusters to the root of the instance group
|
|
for (i = 0; i < _ClusterInstances.size(); ++i)
|
|
_Root->hrcLinkSon( _ClusterInstances[i] );
|
|
|
|
|
|
// Default: freezeHRC all instances.
|
|
freezeHRC();
|
|
|
|
|
|
// Register the instanceGroup for light animation
|
|
// -----------------
|
|
// If some PointLight to animate
|
|
if(!_PointLightArray.getPointLights().empty())
|
|
scene.addInstanceGroupForLightAnimation(this);
|
|
|
|
_AddToSceneState = StateAdded;
|
|
|
|
if (_AddRemoveInstance)
|
|
_AddRemoveInstance->instanceGroupAdded();
|
|
return true;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
bool CInstanceGroup::addToSceneAsync (CScene& scene, IDriver *driver, uint selectedTexture)
|
|
{
|
|
// Init the scene lights
|
|
_PointLightArray.initAnimatedLightIndex (scene);
|
|
|
|
uint32 i;
|
|
|
|
_AddToSceneState = StateAdding;
|
|
_AddToSceneTempScene = &scene;
|
|
_AddToSceneTempDriver = driver;
|
|
_AddToSceneTempSelectTexture = selectedTexture;
|
|
|
|
_Instances.resize (_InstancesInfos.size(), NULL);
|
|
|
|
if (_IGAddBeginCallback)
|
|
_IGAddBeginCallback->startAddingIG((uint)_InstancesInfos.size());
|
|
|
|
// Creation and positionning of the new instance
|
|
|
|
vector<CInstance>::iterator it = _InstancesInfos.begin();
|
|
set<string> allShapesToLoad;
|
|
_AddToSceneSignal = false;
|
|
bool loadAsyncStarted = false;
|
|
for (i = 0; i < _InstancesInfos.size(); ++i, ++it)
|
|
{
|
|
CInstance &rInstanceInfo = *it;
|
|
if (!rInstanceInfo.DontAddToScene)
|
|
{
|
|
string shapeName = rInstanceInfo.Name;
|
|
if (_TransformName != NULL && !rInstanceInfo.InstanceName.empty())
|
|
{
|
|
shapeName = _TransformName->transformName (i, rInstanceInfo.InstanceName, rInstanceInfo.Name);
|
|
}
|
|
|
|
toLower(shapeName);
|
|
|
|
if (!shapeName.empty() && shapeName.find('.') == std::string::npos)
|
|
shapeName += ".shape";
|
|
|
|
|
|
if (allShapesToLoad.find(shapeName) == allShapesToLoad.end())
|
|
{
|
|
allShapesToLoad.insert (shapeName);
|
|
if (scene.getShapeBank()->getPresentState(shapeName) != CShapeBank::Present)
|
|
{
|
|
// Load it from file asynchronously
|
|
scene.getShapeBank()->loadAsync (shapeName, scene.getDriver(), rInstanceInfo.Pos, &_AddToSceneSignal, selectedTexture);
|
|
loadAsyncStarted = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (!loadAsyncStarted)
|
|
_AddToSceneSignal = true;
|
|
else
|
|
_AddToSceneSignal = false;
|
|
//CAsyncFileManager::getInstance().signal (&_AddToSceneSignal);
|
|
|
|
return true;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::stopAddToSceneAsync ()
|
|
{
|
|
if (_AddToSceneState != StateAdding)
|
|
return;
|
|
vector<CInstance>::iterator it = _InstancesInfos.begin();
|
|
CAsyncFileManager::getInstance().cancelSignal (&_AddToSceneSignal);
|
|
for (uint32 i = 0; i < _InstancesInfos.size(); ++i, ++it)
|
|
{
|
|
CInstance &rInstanceInfo = *it;
|
|
if (!rInstanceInfo.DontAddToScene)
|
|
{
|
|
string shapeName;
|
|
|
|
|
|
bool getShapeName = true;
|
|
|
|
if (_TransformName != NULL && !rInstanceInfo.InstanceName.empty())
|
|
{
|
|
shapeName = _TransformName->transformName (i, rInstanceInfo.InstanceName, rInstanceInfo.Name);
|
|
if (shapeName != rInstanceInfo.Name)
|
|
getShapeName = false;
|
|
}
|
|
|
|
|
|
if (getShapeName)
|
|
{
|
|
if (rInstanceInfo.Name.find('.') == std::string::npos)
|
|
shapeName = rInstanceInfo.Name + ".shape";
|
|
else // extension has already been added
|
|
shapeName = rInstanceInfo.Name;
|
|
}
|
|
|
|
toLower(shapeName);
|
|
_AddToSceneTempScene->getShapeBank()->cancelLoadAsync (shapeName);
|
|
}
|
|
}
|
|
_AddToSceneState = StateNotAdded;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
CInstanceGroup::TState CInstanceGroup::getAddToSceneState ()
|
|
{
|
|
// If we are adding but we have finished loading shapes (all shapes are here)
|
|
if (_AddToSceneState == StateAdding)
|
|
{
|
|
if (_AddToSceneSignal)
|
|
{
|
|
addToScene (*_AddToSceneTempScene, _AddToSceneTempDriver, _AddToSceneTempSelectTexture);
|
|
}
|
|
}
|
|
return _AddToSceneState;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
// Search in the hierarchy of ig the most low level (child) ig that contains the clusters that
|
|
// are flagged to be visible from father or which father is visible
|
|
bool CInstanceGroup::linkToParent (CInstanceGroup *pFather)
|
|
{
|
|
uint32 i, j;
|
|
bool ret;
|
|
/*
|
|
for (i = 0; i < pFather->_ClusterInstances.size(); ++i)
|
|
{
|
|
for(j = 0; j < pFather->_ClusterInstances[i]->Children.size(); ++j)
|
|
{
|
|
if (linkToParent(pFather->_ClusterInstances[i]->Children[j]->Group))
|
|
return true;
|
|
}
|
|
}
|
|
*/
|
|
ret = false;
|
|
if (this != pFather)
|
|
{
|
|
for (j = 0; j < this->_ClusterInstances.size(); ++j)
|
|
{
|
|
if ((this->_ClusterInstances[j]->FatherVisible) ||
|
|
(this->_ClusterInstances[j]->VisibleFromFather))
|
|
{
|
|
for (i = 0; i < pFather->_ClusterInstances .size(); ++i)
|
|
{
|
|
// If my cluster j is in the cluster i of the father
|
|
if (pFather->_ClusterInstances[i]->isIn(this->_ClusterInstances[j]->getBBox()))
|
|
{
|
|
if (this->_ClusterInstances[j]->Father != pFather->_ClusterInstances[i]) // and not already son of the father cluster ?
|
|
{
|
|
// unlink from parent
|
|
this->_ClusterInstances[j]->unlinkFromParent();
|
|
|
|
// relink to the new father found
|
|
pFather->_ClusterInstances[i]->Children.push_back(this->_ClusterInstances[j]);
|
|
this->_ClusterInstances[j]->Father = pFather->_ClusterInstances[i];
|
|
}
|
|
ret = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// store new parent
|
|
if(ret)
|
|
_ParentClusterSystem= pFather;
|
|
|
|
return ret;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
bool CInstanceGroup::removeFromScene (CScene& scene)
|
|
{
|
|
uint32 i, j, k;
|
|
|
|
// Remove shapes
|
|
for (i = 0; i < _Instances.size(); ++i)
|
|
{
|
|
CTransformShape *pTShape = _Instances[i];
|
|
if(pTShape)
|
|
{
|
|
// For security, unfreeze any StaticLightSetup setuped.
|
|
pTShape->unfreezeStaticLightSetup();
|
|
// delete the instance
|
|
scene.deleteInstance (pTShape);
|
|
_Instances[i] = NULL;
|
|
}
|
|
}
|
|
|
|
// Relink portals with old clusters
|
|
for (i = 0; i < _Portals.size(); ++i)
|
|
for (k = 0; k < 2; ++k)
|
|
{
|
|
if (_Portals[i]._Clusters[k])
|
|
{
|
|
for (j = 0; j < _ClusterInstances.size(); ++j)
|
|
if( _Portals[i]._Clusters[k] == _ClusterInstances[j] )
|
|
break;
|
|
|
|
nlassert (j!=_ClusterInstances.size());
|
|
_Portals[i]._Clusters[k] = &_ClusterInfos[j];
|
|
}
|
|
}
|
|
|
|
// Remove clusters
|
|
CClipTrav *pClipTrav = &scene.getClipTrav();
|
|
for (i = 0; i < _ClusterInstances.size(); ++i)
|
|
{
|
|
pClipTrav->unregisterCluster (_ClusterInstances[i]);
|
|
scene.deleteModel (_ClusterInstances[i]);
|
|
}
|
|
|
|
scene.deleteModel (_Root);
|
|
_Root = NULL;
|
|
|
|
|
|
// UnRegister the instanceGroup for light animation
|
|
// -----------------
|
|
// If some PointLight to animate
|
|
if(!_PointLightArray.getPointLights().empty())
|
|
scene.removeInstanceGroupForLightAnimation(this);
|
|
|
|
if (_AddRemoveInstance)
|
|
_AddRemoveInstance->instanceGroupRemoved();
|
|
return true;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::getLights( set<string> &LightNames )
|
|
{
|
|
LightNames.clear();
|
|
for( uint32 i = 0; i < _Instances.size(); ++i )
|
|
{
|
|
CMeshInstance *pMI = dynamic_cast<CMeshInstance*>(_Instances[i]);
|
|
if( pMI != NULL )
|
|
{
|
|
uint32 nNbLM = pMI->getNbLightMap();
|
|
for( uint32 j = 0; j < nNbLM; ++j )
|
|
{
|
|
string sTmp;
|
|
pMI->getLightMapName( j, sTmp );
|
|
set<string>::iterator itSet = LightNames.find(sTmp);
|
|
if( itSet == LightNames.end() )
|
|
LightNames.insert( sTmp );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::getBlendShapes( set<string> &BlendShapeNames )
|
|
{
|
|
BlendShapeNames.clear();
|
|
for( uint32 i = 0; i < _Instances.size(); ++i )
|
|
{
|
|
CMeshBaseInstance *pMBI = dynamic_cast<CMeshBaseInstance*>(_Instances[i]);
|
|
if (pMBI != NULL)
|
|
{
|
|
uint32 nNbBS = pMBI->getNbBlendShape();
|
|
for( uint32 j = 0; j < nNbBS; ++j )
|
|
{
|
|
string sTmp;
|
|
pMBI->getBlendShapeName( j, sTmp );
|
|
set<string>::iterator itSet = BlendShapeNames.find(sTmp);
|
|
if( itSet == BlendShapeNames.end() )
|
|
BlendShapeNames.insert( sTmp );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::setBlendShapeFactor( const string &BlendShapeName, float rFactor )
|
|
{
|
|
for( uint32 i = 0; i < _Instances.size(); ++i )
|
|
{
|
|
CMeshBaseInstance *pMI = dynamic_cast<CMeshBaseInstance*>(_Instances[i]);
|
|
if( pMI != NULL )
|
|
{
|
|
pMI->setBlendShapeFactor( BlendShapeName, rFactor );
|
|
}
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::addCluster(CCluster *pCluster)
|
|
{
|
|
_ClusterInstances.push_back(pCluster);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::setClusterSystemForInstances(CInstanceGroup *pIG)
|
|
{
|
|
_ClusterSystemForInstances = pIG;
|
|
for (uint32 i = 0; i < _Instances.size(); ++i)
|
|
if (_Instances[i] && _InstancesInfos[i].Clusters.empty())
|
|
_Instances[i]->setClusterSystem (_ClusterSystemForInstances);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::getDynamicPortals (std::vector<std::string> &names)
|
|
{
|
|
for (uint32 i = 0; i < _Portals.size(); ++i)
|
|
if (!_Portals[i].getName().empty())
|
|
names.push_back (_Portals[i].getName());
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::setDynamicPortal (std::string& name, bool opened)
|
|
{
|
|
for (uint32 i = 0; i < _Portals.size(); ++i)
|
|
if (_Portals[i].getName() == name)
|
|
_Portals[i].open (opened);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
bool CInstanceGroup::getDynamicPortal (std::string& name)
|
|
{
|
|
for (uint32 i = 0; i < _Portals.size(); ++i)
|
|
if (_Portals[i].getName() == name)
|
|
return _Portals[i].isOpened ();
|
|
return false;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::setPos (const CVector &pos)
|
|
{
|
|
if (_Root != NULL)
|
|
/// \todo Make this work (precision): _Root->setPos (_GlobalPos+pos);
|
|
_Root->setPos (pos);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::setRotQuat (const CQuat &quat)
|
|
{
|
|
if (_Root != NULL)
|
|
_Root->setRotQuat (quat);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
CVector CInstanceGroup::getPos ()
|
|
{
|
|
if (_Root != NULL)
|
|
return _Root->getPos ();
|
|
else
|
|
return CVector(0.0f, 0.0f, 0.0f);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
CQuat CInstanceGroup::getRotQuat ()
|
|
{
|
|
if (_Root != NULL)
|
|
return _Root->getRotQuat ();
|
|
else
|
|
return CQuat();
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::linkRoot (CScene &/* scene */, CTransform *father)
|
|
{
|
|
if(_Root)
|
|
{
|
|
father->hrcLinkSon( _Root );
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::freezeHRC()
|
|
{
|
|
// For all instances.
|
|
for (uint i=0; i < _Instances.size(); i++)
|
|
{
|
|
if(_Instances[i])
|
|
_Instances[i]->freezeHRC();
|
|
}
|
|
// and for root.
|
|
_Root->freezeHRC();
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::unfreezeHRC()
|
|
{
|
|
// For all instances.
|
|
for (uint i=0; i < _Instances.size(); i++)
|
|
{
|
|
if(_Instances[i])
|
|
_Instances[i]->unfreezeHRC();
|
|
}
|
|
// and for root.
|
|
_Root->unfreezeHRC();
|
|
}
|
|
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::buildPointLightList(const std::vector<CPointLightNamed> &pointLightList,
|
|
std::vector<uint> &plRemap)
|
|
{
|
|
// build.
|
|
_PointLightArray.build(pointLightList, plRemap);
|
|
|
|
// remap Instance precalc lighted.
|
|
for(uint i=0; i<_InstancesInfos.size(); i++)
|
|
{
|
|
CInstance &inst= _InstancesInfos[i];
|
|
// If the instance has no precomputed lighting, skip
|
|
if(!inst.StaticLightEnabled)
|
|
continue;
|
|
|
|
// remap pointlights
|
|
for(uint l=0; l<CInstanceGroup::NumStaticLightPerInstance; l++)
|
|
{
|
|
// If NULL light, break and continue to next instance
|
|
if(inst.Light[l]== 0xFF)
|
|
break;
|
|
else
|
|
{
|
|
// Check good index.
|
|
nlassert(inst.Light[l] < _PointLightArray.getPointLights().size());
|
|
// Remap index, because of light sorting.
|
|
inst.Light[l]= uint8(plRemap[inst.Light[l]]);
|
|
}
|
|
}
|
|
|
|
// remap ambient light
|
|
if(inst.LocalAmbientId!=0xFF)
|
|
{
|
|
nlassert(inst.LocalAmbientId < _PointLightArray.getPointLights().size());
|
|
inst.LocalAmbientId= uint8(plRemap[inst.LocalAmbientId]);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::setPointLightFactor(const CScene &scene)
|
|
{
|
|
_PointLightArray.setPointLightFactor(scene);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::enableRealTimeSunContribution(bool enable)
|
|
{
|
|
_RealTimeSunContribution= enable;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
|
|
|
|
// ***************************************************************************
|
|
void CInstanceGroup::displayDebugClusters(IDriver *drv, class CTextContext *txtCtx)
|
|
{
|
|
uint opacity= 50;
|
|
CRGBA colorCluster(255, 128, 255, uint8(opacity));
|
|
// portals are drawn twice
|
|
CRGBA colorPortal(128, 255, 128, uint8(opacity/2));
|
|
|
|
CMaterial clusterMat;
|
|
CMaterial portalMat;
|
|
CMaterial lineMat;
|
|
clusterMat.initUnlit();
|
|
clusterMat.setBlend(true);
|
|
clusterMat.setBlendFunc(CMaterial::srcalpha, CMaterial::invsrcalpha);
|
|
clusterMat.setZWrite(false);
|
|
clusterMat.setDoubleSided(true);
|
|
clusterMat.setColor(colorCluster);
|
|
portalMat.initUnlit();
|
|
portalMat.setBlend(true);
|
|
portalMat.setBlendFunc(CMaterial::srcalpha, CMaterial::invsrcalpha);
|
|
portalMat.setZWrite(false);
|
|
portalMat.setDoubleSided(true);
|
|
portalMat.setColor(colorPortal);
|
|
lineMat.initUnlit();
|
|
lineMat.setBlend(true);
|
|
lineMat.setBlendFunc(CMaterial::srcalpha, CMaterial::invsrcalpha);
|
|
lineMat.setZWrite(false);
|
|
lineMat.setDoubleSided(true);
|
|
lineMat.setColor(CRGBA(0,0,0,uint8(opacity)));
|
|
|
|
|
|
// The geometry for each cluster
|
|
CVertexBuffer vb;
|
|
// too big cluster won't be rendered
|
|
const uint maxVertices= 10000;
|
|
vb.setVertexFormat(CVertexBuffer::PositionFlag);
|
|
vb.setNumVertices(maxVertices);
|
|
CIndexBuffer clusterTriangles;
|
|
CIndexBuffer clusterLines;
|
|
CIndexBuffer portalTriangles;
|
|
CIndexBuffer portalLines;
|
|
//
|
|
clusterTriangles.setFormat(NL_DEFAULT_INDEX_BUFFER_FORMAT);
|
|
clusterLines.setFormat(NL_DEFAULT_INDEX_BUFFER_FORMAT);
|
|
portalTriangles.setFormat(NL_DEFAULT_INDEX_BUFFER_FORMAT);
|
|
portalLines.setFormat(NL_DEFAULT_INDEX_BUFFER_FORMAT);
|
|
//
|
|
clusterTriangles.setNumIndexes(maxVertices*3);
|
|
clusterLines.setNumIndexes(maxVertices*2);
|
|
portalTriangles.setNumIndexes(maxVertices*3);
|
|
portalLines.setNumIndexes(maxVertices*2);
|
|
|
|
// setup identity matrix
|
|
drv->setupModelMatrix(CMatrix::Identity);
|
|
|
|
// For all clusters
|
|
uint i;
|
|
for(i=0;i<_ClusterInstances.size();i++)
|
|
{
|
|
CCluster *cluster= _ClusterInstances[i];
|
|
if(cluster)
|
|
{
|
|
uint numTotalVertices= 0;
|
|
|
|
// **** Build a set of polys representing the volume (slow but debug!)
|
|
static std::vector<CPolygon> polygons;
|
|
polygons.clear();
|
|
polygons.resize(cluster->_Volume.size());
|
|
// for each plane, build the associated polygon
|
|
uint j;
|
|
for(j=0;j<polygons.size();j++)
|
|
{
|
|
// Start with a big quad centered on bbox center
|
|
CPlane p= cluster->_Volume[j];
|
|
p.normalize();
|
|
CVector quadCenter= p.project(cluster->_BBox.getCenter());
|
|
|
|
// choose a basis on this plane
|
|
CMatrix mat;
|
|
mat.setArbitraryRotK(p.getNormal());
|
|
mat.setPos(quadCenter);
|
|
|
|
// Build the initial Big quad
|
|
CPolygon &poly= polygons[j];
|
|
poly.Vertices.resize(4);
|
|
float s= 10 * cluster->_BBox.getRadius();
|
|
poly.Vertices[0]= mat * CVector(-s,-s,0);
|
|
poly.Vertices[1]= mat * CVector(s,-s,0);
|
|
poly.Vertices[2]= mat * CVector(s,s,0);
|
|
poly.Vertices[3]= mat * CVector(-s,s,0);
|
|
|
|
// clip this poly against all the other (ie not me) planes
|
|
// This make this algo O(N2) but this is for debug....
|
|
for(uint k=0;k<cluster->_Volume.size();k++)
|
|
{
|
|
if(j!=k)
|
|
{
|
|
poly.clip(&cluster->_Volume[k], 1);
|
|
}
|
|
}
|
|
|
|
// count the number of vertices / triangles / lines to add
|
|
if(poly.Vertices.size()>=3)
|
|
{
|
|
numTotalVertices+= (uint)poly.Vertices.size();
|
|
}
|
|
}
|
|
|
|
// **** count the number of portals vertices
|
|
for(j=0;j<cluster->_Portals.size();j++)
|
|
{
|
|
numTotalVertices+= (uint)cluster->_Portals[j]->_Poly.size();
|
|
}
|
|
|
|
// **** Draw those cluster polygons, and portals
|
|
// too big clusters won't be rendered
|
|
if(numTotalVertices<=maxVertices)
|
|
{
|
|
uint iVert= 0;
|
|
uint j;
|
|
|
|
// build the cluster geometry
|
|
clusterTriangles.setNumIndexes(maxVertices*3);
|
|
clusterLines.setNumIndexes(maxVertices*2);
|
|
|
|
// Locks
|
|
CVertexBufferReadWrite vba;
|
|
vb.lock (vba);
|
|
CIndexBufferReadWrite ibaCT;
|
|
clusterTriangles.lock (ibaCT);
|
|
CIndexBufferReadWrite ibaCL;
|
|
clusterLines.lock (ibaCL);
|
|
|
|
uint numTriIndexes = 0;
|
|
uint numLineIndexes = 0;
|
|
|
|
for(j=0;j<polygons.size();j++)
|
|
{
|
|
CPolygon &poly= polygons[j];
|
|
if(poly.Vertices.size()>=3)
|
|
{
|
|
uint k;
|
|
// add the vertices
|
|
for(k=0;k<poly.Vertices.size();k++)
|
|
vba.setVertexCoord(iVert+k, poly.Vertices[k]);
|
|
|
|
// add the triangles
|
|
for(k=0;k<poly.Vertices.size()-2;k++)
|
|
{
|
|
if (numTriIndexes<clusterTriangles.capacity())
|
|
{
|
|
ibaCT.setTri(numTriIndexes, iVert+0, iVert+k+1, iVert+k+2);
|
|
numTriIndexes += 3;
|
|
}
|
|
}
|
|
|
|
// add the lines
|
|
for(k=0;k<poly.Vertices.size();k++)
|
|
{
|
|
if (numLineIndexes<clusterLines.capacity())
|
|
{
|
|
ibaCL.setLine(numLineIndexes, iVert+k, iVert+ ((k+1)%poly.Vertices.size()) );
|
|
numLineIndexes += 2;
|
|
}
|
|
}
|
|
|
|
iVert+= (uint)poly.Vertices.size();
|
|
}
|
|
}
|
|
|
|
// Unlocks
|
|
ibaCT.unlock ();
|
|
ibaCL.unlock ();
|
|
clusterTriangles.setNumIndexes(numTriIndexes);
|
|
clusterLines.setNumIndexes(numLineIndexes);
|
|
|
|
// build the portal geometry
|
|
portalTriangles.setNumIndexes(maxVertices*3);
|
|
portalLines.setNumIndexes(maxVertices*2);
|
|
|
|
// Locks
|
|
CIndexBufferReadWrite ibaPT;
|
|
portalTriangles.lock (ibaPT);
|
|
CIndexBufferReadWrite ibaPL;
|
|
portalLines.lock (ibaPL);
|
|
|
|
numTriIndexes = 0;
|
|
numLineIndexes = 0;
|
|
|
|
for(j=0;j<cluster->_Portals.size();j++)
|
|
{
|
|
std::vector<CVector> &portalVerts= cluster->_Portals[j]->_Poly;
|
|
if(portalVerts.size()>=3)
|
|
{
|
|
uint k;
|
|
// add the vertices
|
|
for(k=0;k<portalVerts.size();k++)
|
|
vba.setVertexCoord(iVert+k, portalVerts[k]);
|
|
|
|
// add the triangles
|
|
for(k=0;k<portalVerts.size()-2;k++)
|
|
{
|
|
if (numTriIndexes<clusterTriangles.capacity())
|
|
{
|
|
ibaPT.setTri(numTriIndexes, iVert+0, iVert+k+1, iVert+k+2);
|
|
numTriIndexes += 3;
|
|
}
|
|
}
|
|
|
|
// add the lines
|
|
for(k=0;k<portalVerts.size();k++)
|
|
{
|
|
if (numTriIndexes<clusterTriangles.capacity())
|
|
{
|
|
ibaPL.setLine(numLineIndexes, iVert+k, iVert+ ((k+1)%portalVerts.size()) );
|
|
numLineIndexes += 2;
|
|
}
|
|
}
|
|
|
|
iVert+= (uint)portalVerts.size();
|
|
}
|
|
}
|
|
|
|
// Unlock
|
|
ibaPT.unlock ();
|
|
ibaPL.unlock ();
|
|
portalTriangles.setNumIndexes(numTriIndexes);
|
|
portalLines.setNumIndexes(numLineIndexes);
|
|
vba.unlock ();
|
|
|
|
// render 2 pass with or without ZBuffer (for clearness)
|
|
for(uint pass=0;pass<2;pass++)
|
|
{
|
|
if(pass==0)
|
|
{
|
|
clusterMat.setZFunc(CMaterial::always);
|
|
portalMat.setZFunc(CMaterial::always);
|
|
lineMat.setZFunc(CMaterial::always);
|
|
}
|
|
else
|
|
{
|
|
clusterMat.setZFunc(CMaterial::lessequal);
|
|
portalMat.setZFunc(CMaterial::lessequal);
|
|
lineMat.setZFunc(CMaterial::lessequal);
|
|
}
|
|
|
|
drv->activeVertexBuffer(vb);
|
|
drv->activeIndexBuffer(clusterTriangles);
|
|
drv->renderTriangles (clusterMat, 0, clusterTriangles.getNumIndexes()/3);
|
|
drv->activeIndexBuffer(clusterLines);
|
|
drv->renderLines (lineMat, 0, clusterLines.getNumIndexes()/2);
|
|
drv->activeIndexBuffer(portalTriangles);
|
|
drv->renderTriangles (portalMat, 0, portalTriangles.getNumIndexes()/3);
|
|
drv->activeIndexBuffer(portalLines);
|
|
drv->renderLines (lineMat, 0, portalLines.getNumIndexes()/2);
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
// **** For all clusters, Draw the cluster name at center of the cluster
|
|
if(txtCtx)
|
|
{
|
|
CComputedString computedStr;
|
|
|
|
// bkup fontSize
|
|
uint bkFontSize;
|
|
CMatrix fontMatrix;
|
|
bkFontSize= txtCtx->getFontSize();
|
|
// to be readable
|
|
txtCtx->setFontSize(24);
|
|
|
|
// the font matrix
|
|
fontMatrix.setRot(drv->getViewMatrix().inverted());
|
|
fontMatrix.normalize(CMatrix::YZX);
|
|
fontMatrix.scale(10);
|
|
|
|
// parse all clusters
|
|
for(i=0;i<_ClusterInstances.size();i++)
|
|
{
|
|
CCluster *cluster= _ClusterInstances[i];
|
|
if(cluster)
|
|
{
|
|
fontMatrix.setPos(cluster->_BBox.getCenter());
|
|
txtCtx->computeString(cluster->Name, computedStr);
|
|
computedStr.render3D(*drv, fontMatrix);
|
|
}
|
|
}
|
|
|
|
// restore fontsize
|
|
txtCtx->setFontSize(bkFontSize);
|
|
}
|
|
|
|
}
|
|
|
|
|
|
} // NL3D
|