// 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/seg_remanence_shape.h"
#include "nel/3d/seg_remanence.h"
#include "nel/3d/driver.h"
#include "nel/3d/scene.h"
//
#include "nel/misc/bsphere.h"
#ifdef DEBUG_NEW
#define new DEBUG_NEW
#endif
namespace NL3D
{
//===========================================================
CSegRemanenceShape::CSegRemanenceShape() : _GeomTouched(true),
_MatTouched(true),
_TextureShifting(true),
_NumSlices(8),
_SliceTime(0.05f),
_RollUpRatio(1.f),
_AnimatedMat(NULL)
{
_BBox.setCenter(NLMISC::CVector::Null);
_BBox.setHalfSize(NLMISC::CVector(3, 3, 3));
setNumCorners(2);
}
//===========================================================
void CSegRemanenceShape::serial(NLMISC::IStream &f)
{
// version 2 : added default tracks
// version 1 : rollup ratio
// version 0 : base version
sint ver = f.serialVersion(2);
f.serial(_NumSlices);
f.serial(_SliceTime);
f.serialCont(_Corners);
f.serial(_Mat);
f.serial(_BBox);
f.serial(_TextureShifting);
f.serialPtr(_AnimatedMat);
if (f.isReading())
{
_GeomTouched = true;
_MatTouched = true;
}
if (ver >= 1)
{
f.serial(_RollUpRatio);
}
if (ver >= 2)
{
f.serial(_DefaultPos);
f.serial(_DefaultRotQuat);
f.serial(_DefaultScale);
}
}
//===========================================================
void CSegRemanenceShape::setSliceTime(float sliceTime)
{
nlassert(sliceTime > 0);
_SliceTime = sliceTime;
}
//===========================================================
void CSegRemanenceShape::setCorner(uint corner, const NLMISC::CVector &value)
{
nlassert(corner < _Corners.size());
_Corners[corner] = value;
}
//===========================================================
void CSegRemanenceShape::setNumSlices(uint32 numSlices)
{
nlassert(numSlices >= 2);
_NumSlices = numSlices;
_GeomTouched = true;
}
//===========================================================
NLMISC::CVector CSegRemanenceShape::getCorner(uint corner) const
{
nlassert(corner < _Corners.size());
return _Corners[corner];
}
//===========================================================
void CSegRemanenceShape::setNumCorners(uint numCorners)
{
nlassert(numCorners >= 2);
_Corners.resize(numCorners);
std::fill(_Corners.begin(), _Corners.end(), NLMISC::CVector::Null);
_GeomTouched = true;
}
//===========================================================
void CSegRemanenceShape::render(IDriver *drv, CTransformShape *trans, bool opaquePass)
{
if ((!opaquePass && _Mat.getBlend())
|| (opaquePass && !_Mat.getBlend())
)
{
CSegRemanence *sr = NLMISC::safe_cast(trans);
#ifndef DEBUG_SEG_REMANENCE_DISPLAY
if (!sr->isStarted()) return;
#endif
setupMaterial();
//
sr->render(drv, _Mat);
}
}
//===========================================================
void CSegRemanenceShape::flushTextures(IDriver &driver, uint selectedTexture)
{
_Mat.flushTextures(driver, selectedTexture);
}
//===========================================================
CTransformShape *CSegRemanenceShape::createInstance(CScene &scene)
{
CSegRemanence *sr = NLMISC::safe_cast(scene.createModel(NL3D::SegRemanenceShapeId) );
sr->Shape = this;
CAnimatedMaterial *aniMat = NULL;
if (_AnimatedMat)
{
aniMat = new CAnimatedMaterial(_AnimatedMat);
aniMat->setMaterial(&_Mat);
}
sr->setAnimatedMaterial(aniMat);
sr->setupFromShape();
// SegRemanence are added to the "Fx" Load Balancing Group.
sr->setLoadBalancingGroup("Fx");
sr->ITransformable::setPos( _DefaultPos.getDefaultValue() );
sr->ITransformable::setRotQuat( _DefaultRotQuat.getDefaultValue() );
sr->ITransformable::setScale( _DefaultScale.getDefaultValue() );
sr->setSliceTime(_SliceTime);
return sr;
}
//===========================================================
float CSegRemanenceShape::getNumTriangles(float distance)
{
return (float) (_NumSlices * 2);
}
//===========================================================
void CSegRemanenceShape::setBBox(const NLMISC::CAABBox &bbox)
{
_BBox = bbox;
}
//===========================================================
void CSegRemanenceShape::setMaterial(const CMaterial &mat)
{
_Mat = mat;
_MatTouched = true;
}
//===========================================================
void CSegRemanenceShape::setTextureShifting(bool on /*=true*/)
{
_TextureShifting = on;
_MatTouched = true;
}
//===========================================================
void CSegRemanenceShape::setRollupRatio(float ratio)
{
nlassert(ratio > 0);
_RollUpRatio = ratio;
}
//===========================================================
void CSegRemanenceShape::setupMaterial()
{
if (!_MatTouched) return;
_Mat.enableUserTexMat(0);
if (_Mat.getTexture(0))
{
_Mat.getTexture(0)->setWrapS(ITexture::Clamp);
_Mat.getTexture(0)->setWrapT(ITexture::Clamp);
}
_Mat.setDoubleSided(true);
_Mat.setLighting(false); // lighting not supported (the vb has no normals anyway..)
_MatTouched = false;
}
//===========================================================
void CSegRemanenceShape::setAnimatedMaterial(const std::string &name)
{
nlassert(!name.empty());
nlassert(_AnimatedMat == NULL);
_AnimatedMat = new CMaterialBase;
_AnimatedMat->Name = name;
}
//===========================================================
CSegRemanenceShape::CSegRemanenceShape(const CSegRemanenceShape &other) : IShape(other), _AnimatedMat(NULL)
{
copyFromOther(other);
}
//===========================================================
CSegRemanenceShape &CSegRemanenceShape::operator = (const CSegRemanenceShape &other)
{
if (&other != this)
{
copyFromOther(other);
(IShape &) *this = (IShape &) other; // copy base part
}
return *this;
}
//===========================================================
CSegRemanenceShape::~CSegRemanenceShape()
{
delete _AnimatedMat;
}
//===========================================================
void CSegRemanenceShape::copyFromOther(const CSegRemanenceShape &other)
{
if (&other == this) return;
CMaterialBase *otherAnimatedMat = other._AnimatedMat != NULL ? new CMaterialBase(*other._AnimatedMat)
: NULL;
delete _AnimatedMat;
_AnimatedMat = otherAnimatedMat;
_GeomTouched = other._GeomTouched;
_MatTouched = other._MatTouched;
_TextureShifting = other._TextureShifting;
_NumSlices = other._NumSlices;
_SliceTime = other._SliceTime;
_Corners = other._Corners;
_Mat = other._Mat;
_BBox = other._BBox;
_RollUpRatio = other._RollUpRatio;
}
//===========================================================
bool CSegRemanenceShape::clip(const std::vector &pyramid, const CMatrix &worldMatrix)
{
// Speed Clip: clip just the sphere.
NLMISC::CBSphere localSphere(_BBox.getCenter(), _BBox.getRadius());
NLMISC::CBSphere worldSphere;
// transform the sphere in WorldMatrix (with nearly good scale info).
localSphere.applyTransform(worldMatrix, worldSphere);
// if out of only plane, entirely out.
for(sint i=0;i<(sint)pyramid.size();i++)
{
// We are sure that pyramid has normalized plane normals.
// if SpherMax OUT return false.
float d= pyramid[i]*worldSphere.Center;
if(d>worldSphere.Radius)
return false;
}
return true;
}
}