// NeL - MMORPG Framework
// 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 .
#include "std3d.h"
#include "nel/3d/cluster.h"
#include "nel/3d/portal.h"
#include "nel/misc/stream.h"
#include "nel/misc/string_mapper.h"
#include "nel/3d/scene.h"
#include "nel/3d/transform_shape.h"
//#include "mesh_instance.h"
#include "nel/3d/scene_group.h"
using namespace NLMISC;
using namespace std;
#ifdef DEBUG_NEW
#define new DEBUG_NEW
#endif
namespace NL3D
{
// 0.5 cm of precision
#define CLUSTERPRECISION 0.005
// ***************************************************************************
CCluster::CCluster ()
{
/* ***********************************************
* 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
* ***********************************************/
FatherVisible = VisibleFromFather = false;
FatherAudible = AudibleFromFather = false;
Father = NULL;
Group = NULL;
// map a no fx string
_EnvironmentFxId = CStringMapper::map("no fx");
// map a no soundgroup string
_SoundGroupId = CStringMapper::map("");
// I am a transform cluster
CTransform::setIsCluster(true);
// Default: not traversed
_Visited= false;
_CameraIn= false;
}
// ***************************************************************************
CCluster::~CCluster()
{
/* ***********************************************
* 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
* ***********************************************/
unlinkFromClusterTree();
}
void CCluster::setSoundGroup(const std::string &soundGroup)
{
_SoundGroupId = CStringMapper::map(soundGroup);
}
void CCluster::setSoundGroup(const NLMISC::TStringId &soundGroupId)
{
_SoundGroupId = soundGroupId;
}
const std::string &CCluster::getSoundGroup()
{
return CStringMapper::unmap(_SoundGroupId);
}
NLMISC::TStringId CCluster::getSoundGroupId()
{
return _SoundGroupId;
}
void CCluster::setEnvironmentFx(const std::string &environmentFx)
{
_EnvironmentFxId = CStringMapper::map(environmentFx);
}
void CCluster::setEnvironmentFx(const NLMISC::TStringId &environmentFxId)
{
_EnvironmentFxId = environmentFxId;
}
const std::string &CCluster::getEnvironmentFx()
{
return CStringMapper::unmap(_EnvironmentFxId);
}
NLMISC::TStringId CCluster::getEnvironmentFxId()
{
return _EnvironmentFxId;
}
// ***************************************************************************
void CCluster::unlinkFromParent()
{
/* ***********************************************
* 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
* ***********************************************/
// unlink from father sons list
if (Father)
{
Father->Children.erase(remove(Father->Children.begin(), Father->Children.end(), this), Father->Children.end());
Father = NULL;
}
}
// ***************************************************************************
void CCluster::unlinkSons()
{
/* ***********************************************
* 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
* ***********************************************/
// tells all sons that they have no more father
for(uint k = 0; k < Children.size(); ++k)
{
if (Children[k]->Father == this)
{
Children[k]->Father = NULL;
}
}
NLMISC::contReset(Children);
}
// ***************************************************************************
void CCluster::unlinkFromClusterTree()
{
/* ***********************************************
* 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
* ***********************************************/
unlinkFromParent();
unlinkSons();
}
// ***************************************************************************
void CCluster::registerBasic ()
{
CScene::registerModel (ClusterId, 0, CCluster::creator);
}
// ***************************************************************************
bool CCluster::makeVolume (const CVector& p1, const CVector& p2, const CVector& p3)
{
uint i;
// Check if the plane is not close to a plane that already define the cluster
for (i = 0; i < _LocalVolume.size(); ++i)
{
float f1 = fabsf (_LocalVolume[i]*p1);
float f2 = fabsf (_LocalVolume[i]*p2);
float f3 = fabsf (_LocalVolume[i]*p3);
if ((f1 < CLUSTERPRECISION) && (f2 < CLUSTERPRECISION) && (f3 < CLUSTERPRECISION))
return true;
}
// Check if we want to add a triangle not completely in the predefined volume
for (i = 0; i < _LocalVolume.size(); ++i)
{
float f1 = _LocalVolume[i]*p1;
float f2 = _LocalVolume[i]*p2;
float f3 = _LocalVolume[i]*p3;
if ((f1 > CLUSTERPRECISION) && (f2 > CLUSTERPRECISION) && (f3 > CLUSTERPRECISION))
return false;
}
// Build volume
CPlane p;
p.make (p1, p2, p3);
p.normalize();
_LocalVolume.push_back (p);
// Build BBox
if (_LocalVolume.size() == 1)
_LocalBBox.setCenter(p1);
else
_LocalBBox.extend(p1);
_LocalBBox.extend(p2);
_LocalBBox.extend(p3);
_Volume = _LocalVolume;
_BBox = _LocalBBox;
return true;
}
// ***************************************************************************
bool CCluster::isIn (const CVector& p)
{
for (uint i = 0; i < _Volume.size(); ++i)
if (_Volume[i]*p > CLUSTERPRECISION)
return false;
return true;
}
// ***************************************************************************
bool CCluster::isIn (const CAABBox& b)
{
for (uint i = 0; i < _Volume.size(); ++i)
{
if (!b.clipBack (_Volume[i]))
return false;
}
return true;
}
// ***************************************************************************
bool CCluster::isIn (const NLMISC::CVector& center, float size)
{
for (uint i = 0; i < _Volume.size(); ++i)
if (_Volume[i]*center > size)
return false;
return true;
}
// ***************************************************************************
bool CCluster::clipSegment (NLMISC::CVector &p0, NLMISC::CVector &p1)
{
for (uint i = 0; i < _Volume.size(); ++i)
{
if (!_Volume[i].clipSegmentBack(p0, p1))
return false;
}
return true;
}
// ***************************************************************************
void CCluster::resetPortalLinks ()
{
_Portals.clear();
}
// ***************************************************************************
void CCluster::link (CPortal* portal)
{
_Portals.push_back (portal);
}
// ***************************************************************************
void CCluster::unlink (CPortal* portal)
{
uint32 pos;
for (pos = 0; pos < _Portals.size(); ++pos)
{
if (_Portals[pos] == portal)
break;
}
if (pos < _Portals.size())
_Portals.erase (_Portals.begin()+pos);
}
// ***************************************************************************
void CCluster::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
* ***********************************************/
sint version = f.serialVersion (3);
if (version >= 1)
f.serial (Name);
f.serialCont (_LocalVolume);
f.serial (_LocalBBox);
f.serial (FatherVisible);
f.serial (VisibleFromFather);
if (f.isReading())
{
_Volume = _LocalVolume;
_BBox = _LocalBBox;
}
if (version >= 2)
{
if (f.isReading())
{
std::string soundGroup;
std::string envFxName;
f.serial(soundGroup);
_SoundGroupId = CStringMapper::map(soundGroup);
f.serial(envFxName);
if (envFxName.empty())
envFxName = "no fx";
_EnvironmentFxId = CStringMapper::map(envFxName);
}
else
{
// write the sound group
std::string soundGroup = CStringMapper::unmap(_SoundGroupId);
f.serial(soundGroup);
// write the env fx name
std::string envFxName = CStringMapper::unmap(_EnvironmentFxId);
if (envFxName == "no fx")
envFxName.clear();
f.serial(envFxName);
}
// nldebug("Cluster %s, sound group [%s]", Name.c_str(), CStringMapper::unmap(_SoundGroupId).c_str());
}
if (version >= 3)
{
f.serial(AudibleFromFather);
f.serial(FatherAudible);
}
else
{
// copy the visual property
AudibleFromFather = VisibleFromFather;
FatherAudible = FatherVisible;
}
}
// ***************************************************************************
void CCluster::setWorldMatrix (const CMatrix &WM)
{
uint32 i;
CMatrix invWM = WM;
invWM.invert();
// Transform the volume
for (i = 0; i < _LocalVolume.size(); ++i)
_Volume[i] = _LocalVolume[i] * invWM;
_BBox = NLMISC::CAABBox::transformAABBox(WM, _LocalBBox);
// Transform the bounding box
/*CVector p[8];
p[0].x = _LocalBBox.getMin().x;
p[0].y = _LocalBBox.getMin().y;
p[0].z = _LocalBBox.getMin().z;
p[1].x = _LocalBBox.getMax().x;
p[1].y = _LocalBBox.getMin().y;
p[1].z = _LocalBBox.getMin().z;
p[2].x = _LocalBBox.getMin().x;
p[2].y = _LocalBBox.getMax().y;
p[2].z = _LocalBBox.getMin().z;
p[3].x = _LocalBBox.getMax().x;
p[3].y = _LocalBBox.getMax().y;
p[3].z = _LocalBBox.getMin().z;
p[4].x = _LocalBBox.getMin().x;
p[4].y = _LocalBBox.getMin().y;
p[4].z = _LocalBBox.getMax().z;
p[5].x = _LocalBBox.getMax().x;
p[5].y = _LocalBBox.getMin().y;
p[5].z = _LocalBBox.getMax().z;
p[6].x = _LocalBBox.getMin().x;
p[6].y = _LocalBBox.getMax().y;
p[6].z = _LocalBBox.getMax().z;
p[7].x = _LocalBBox.getMax().x;
p[7].y = _LocalBBox.getMax().y;
p[7].z = _LocalBBox.getMax().z;
for (i = 0; i < 8; ++i)
p[i] = WM.mulPoint(p[i]);
CAABBox boxTemp;
boxTemp.setCenter(p[0]);
for (i = 1; i < 8; ++i)
boxTemp.extend(p[i]);
_BBox = boxTemp;*/
}
// ***************************************************************************
void CCluster::traverseHrc ()
{
CTransform::traverseHrc ();
setWorldMatrix (_WorldMatrix);
for (uint32 i = 0; i < getNbPortals(); ++i)
{
CPortal *pPortal = getPortal(i);
pPortal->setWorldMatrix (_WorldMatrix);
}
// Re affect the cluster to the accelerator if not the root
if (!isRoot())
{
Group->_ClipTrav->unregisterCluster(this);
Group->_ClipTrav->registerCluster (this);
}
}
// ***************************************************************************
bool CCluster::clip ()
{
return true;
}
// ***************************************************************************
void CCluster::traverseClip ()
{
// This is the root call called by the SceneRoot
recursTraverseClip(NULL);
}
// ***************************************************************************
void CCluster::recursTraverseClip(CTransform *caller)
{
if (_Visited)
return;
_Visited = true;
// The cluster is visible because we are in it
// So clip the models attached (with MOT links) to the cluster
uint num= clipGetNumChildren();
uint32 i;
for(i=0;itraverseClip();
// Debug visible clusters
CClipTrav &clipTrav = getOwnerScene()->getClipTrav();
if (clipTrav.getClusterVisibilityTracking())
{
clipTrav.addVisibleCluster(this);
}
// And look through portals
for (i = 0; i < getNbPortals(); ++i)
{
CPortal*pPortal = getPortal (i);
vector WorldPyrTemp = clipTrav.WorldPyramid;
bool backfaceclipped = false;
CCluster *pOtherSideCluster;
if (pPortal->getCluster(0) == this)
pOtherSideCluster = pPortal->getCluster (1);
else
pOtherSideCluster = pPortal->getCluster (0);
if (Father != NULL)
if (caller == Father) // If the caller is the father
if (VisibleFromFather)
// Backface clipping
if( !pPortal->isInFront( clipTrav.CamPos ))
backfaceclipped = true;
if (!backfaceclipped && pOtherSideCluster)
{
/* If the otherSide cluster is fully visible because the camera is IN, then don't need to clip.
This is important to landscape test, to ensure that pyramid are strictly equal from 2 paths which
come from the 2 clusters where the camera start
*/
if (pOtherSideCluster->isCameraIn() || pPortal->clipPyramid (clipTrav.CamPos, clipTrav.WorldPyramid))
{
pOtherSideCluster->recursTraverseClip(this);
}
}
clipTrav.WorldPyramid = WorldPyrTemp;
}
// Link up in hierarchy
if ((FatherVisible)&&(Father != NULL))
{
Father->recursTraverseClip(this);
}
// Link down in hierarchy
for (i = 0; i < Children.size(); ++i)
if (Children[i]->VisibleFromFather)
{
Children[i]->recursTraverseClip(this);
}
_Visited = false;
}
// ***************************************************************************
void CCluster::applyMatrix(const NLMISC::CMatrix &m)
{
uint32 i;
CMatrix invM = m;
invM.invert();
nlassert(_Volume.size() == _LocalVolume.size());
// Transform the volume
for (i = 0; i < _LocalVolume.size(); ++i)
{
_Volume[i] = _Volume[i] * invM;
_LocalVolume[i] = _LocalVolume[i] * invM;
}
// Transform the bounding boxes
_BBox = NLMISC::CAABBox::transformAABBox(m, _BBox);
_LocalBBox = NLMISC::CAABBox::transformAABBox(m, _LocalBBox);
}
// ***************************************************************************
void CCluster::cameraRayClip(const CVector &start, const CVector &end, std::vector &clusterVisited)
{
uint i;
if (_Visited)
return;
_Visited = true;
// The cluster is visible because we are in it. add it to the list of cluster (if not already inserted)
for(i=0;igetCluster(0) == this)
pOtherSideCluster = pPortal->getCluster (1);
else
pOtherSideCluster = pPortal->getCluster (0);
/*
bool backfaceclipped = false;
if (Father != NULL)
if (caller == Father) // If the caller is the father
if (VisibleFromFather)
// Backface clipping
if( !pPortal->isInFront( clipTrav.CamPos ))
backfaceclipped = true;
if (!backfaceclipped && pOtherSideCluster)*/
if (pOtherSideCluster)
{
if (pPortal->clipRay (start, end))
{
pOtherSideCluster->cameraRayClip(start, end, clusterVisited);
}
}
}
/* Link up in hierarchy. Test the Inverse Flag, cause the path is inverted here!!!
ie: if I allow the camera to go out, it MUST can re-enter (ie if I am VisibleFromFather)
*/
if ((VisibleFromFather)&&(Father != NULL))
{
Father->cameraRayClip(start, end, clusterVisited);
}
// Link down in hierarchy
for (i = 0; i < Children.size(); ++i)
{
// same remark. test FatherVisible, not VisibleFromFather
if (Children[i]->FatherVisible)
{
Children[i]->cameraRayClip(start, end, clusterVisited);
}
}
_Visited = false;
}
} // NL3D