khanat-opennel-code/code/nel/include/nel/3d/ps_ribbon.h
2010-11-02 13:42:25 +01:00

284 lines
9.9 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_PS_RIBBON_H
#define NL_PS_RIBBON_H
#include "nel/3d/ps_particle_basic.h"
#include "nel/3d/ps_ribbon_base.h"
#include "nel/3d/vertex_buffer.h"
#include "nel/3d/vertex_buffer.h"
#include "nel/3d/index_buffer.h"
#include "nel/misc/vector.h"
namespace NL3D
{
/** 'Ribbon' particle : a shape is extruded while it follows the particle.
* It replaces the old ribbon version. It has more limitations : no 2d rotations and it has no propagation of size and color.
* It doesn't take as much memory, and displays better (the length is not dependent on framerate as it was the case before)
*/
class CPSRibbon : public CPSRibbonBase,
public CPSColoredParticle,
public CPSSizedParticle,
public CPSMaterial,
public CPSTexturedParticleNoAnim
{
public:
// Orientation mode
enum TOrientation { FollowPath = 0, FollowPathXY, Identity, OrientationCount };
///\name Object
///@{
/** ctor.
* \param nbSegmentInTail the number of segment the ribbon has in its tail
* \param shape pointer to a shape that will be extruded along the ribbon. It must have a unit size
* and be located in the x-y plane. This will be copied
* \param nbPointsInShape : the number of points in the shape
*/
CPSRibbon();
/// dtor
~CPSRibbon();
/// serialisation. Derivers must override this, and call their parent version
virtual void serial(NLMISC::IStream &f) throw(NLMISC::EStream);
//
NLMISC_DECLARE_CLASS(CPSRibbon);
///@}
///\name Behaviour
///@{
/** (de)activate color fading
* when its done, colors fades to black along the ribbon.
*/
virtual void setColorFading(bool onOff = true)
{
_ColorFading = onOff;
touch();
}
/** Test whether color fading is activated.
*/
virtual bool getColorFading(void) const
{
return _ColorFading;
}
// Set orienation of slices
void setOrientation(TOrientation orientation) { _Orientation = orientation; }
TOrientation getOrientation() const { return _Orientation; }
//void setPersistAfterDeath(bool persit = true);
/** return true if the ribbon light persist after death
* \see _PersistAfterDeath()
*/
//bool getPersistAfterDeath(void) const { return _DyingRibbons != NULL; }
///@}
/// inherited from CPSParticle
virtual void step(TPSProcessPass pass);
/// return true if there are transparent faces in the object
virtual bool hasTransparentFaces(void);
/// return true if there are Opaque faces in the object
virtual bool hasOpaqueFaces(void);
virtual uint32 getNumWantedTris() const;
/// set a texture
void setTexture(NLMISC::CSmartPtr<ITexture> tex) { _Tex = tex; touch(); }
/// Set texture factors.
void setTexFactor(float uFactor = 1.f, float vFactor = 1.f)
{
_UFactor = uFactor;
_VFactor = vFactor;
touch();
}
// get the u-factor for texture mapping
float getUFactor(void) const { return _UFactor; }
// get the v-factor for texture mapping
float getVFactor(void) const { return _VFactor; }
/// get the texture used
ITexture *getTexture(void) { return _Tex; }
const ITexture *getTexture(void) const { return _Tex; }
/** set a new shape for the ribbon
* \param points A list of points that define the shape of the slice of the ribbon
* These points must be located in the x-y plane (z can be used for effects).
* \param nbPointsInShape The number of points in the shape
* \param braceMode In this mode, the ribbon mesh is a brace. The number of vertices must be even. Each pair of vertices defines a segment whose V tex coord goes from 0 to 1
* When set to false, the list of vertices define a closed loop shape
*/
void setShape(const NLMISC::CVector *shape, uint32 nbPointsInShape, bool braceMode = false);
// See if shape is in brace mode (rather then in closed loop mode)
bool getBraceMode() const { return _BraceMode; }
/// get the number of vertice in the shape used for ribbons
uint32 getNbVerticesInShape(void) const { return (uint32)_Shape.size(); }
/** Get a copy of the shape used for ribbon
* \param dest a table of cvector that has the right size, it will be filled with vertices
* \see getNbVerticesInShape()
*/
void getShape(NLMISC::CVector *shape) const;
///\name Predefined shapes
///@{
// TODO : replace that ugly stuff with std::vectors
/// Predefined shape : a regular losange shape
static const NLMISC::CVector Losange[];
/// number of vertices in the losange
static const uint NbVerticesInLosange;
/// Predefined shape : height sides
static const NLMISC::CVector HeightSides[];
/// number of vertices in the height side (must be 8 ... :) )
static const uint NbVerticesInHeightSide;
/// Predifined shape : pentagram
static const NLMISC::CVector Pentagram[];
static const uint NbVerticesInPentagram;
/// Predifined shape : triangle
static const NLMISC::CVector Triangle[];
static const uint NbVerticesInTriangle;
// Simple segments
static const NLMISC::CVector SimpleSegmentX[];
static const uint NbVerticesInSimpleSegmentX;
static const NLMISC::CVector SimpleSegmentY[];
static const uint NbVerticesInSimpleSegmentY;
static const NLMISC::CVector SimpleSegmentZ[];
static const uint NbVerticesInSimpleSegmentZ;
// Simple brace
static const NLMISC::CVector SimpleBrace[];
static const uint NbVerticesInSimpleBrace;
///@}
/// from CPSParticle : return true if there are lightable faces in the object
virtual bool hasLightableFaces() { return false; }
// from CPSParticle
virtual bool supportGlobalColorLighting() const { return true; }
// from CPSLocatedBindable
virtual void enumTexs(std::vector<NLMISC::CSmartPtr<ITexture> > &dest, IDriver &drv);
// from CPSParticle
virtual void setZBias(float value) { CPSMaterial::setZBias(value); }
virtual float getZBias() const { return CPSMaterial::getZBias(); }
protected:
/// interface to derived classes
// the number of dying ribbons that are present
//uint32 _NbDyingRibbons;
// a counter to tell how much frame is left for each ribbon
//std::vector<uint32> _DyingRibbonsLifeLeft;
/// inherited from CPSLocatedBindable
virtual void newElement(const CPSEmitterInfo &info) ;
/// inherited from CPSLocatedBindable
virtual void deleteElement(uint32 index);
/// inherited from CPSLocatedBindable
virtual void resize(uint32 size);
/// From CPSSizedParticle
virtual CPSLocated *getSizeOwner(void) { return _Owner; }
/// From CPSColoredParticle
virtual CPSLocated *getColorOwner(void) { return _Owner; }
private:
/// update the material and the vb so that they match the color scheme. Inherited from CPSColoredParticle
virtual void updateMatAndVbForColor(void);
/// display a set of ribbons
void displayRibbons(uint32 nbRibbons, uint32 srcStep);
/**\name Vertex buffers & their corresponding index buffers. We keep a map of pretextured vertex buffer (with or without colors).
* Vb for ribbons that have the same size are shared.
*/
//@{
/** a struct containing a vertex buffer and the matching primitive block
* The number of slice is encoded in the upper word of the vb index (the int used to lookup in the map)
* The number of vertices per slices is encoded in the lower word
*/
class CVBnPB
{
public:
CVertexBuffer VB;
CIndexBuffer PB;
public:
CVBnPB()
{
NL_SET_IB_NAME(PB, "CPSRibbon::CVBnPB::PB");
VB.setName("CPSRibbon::CVBnPB::VB");
}
};
typedef CHashMap<uint, CVBnPB> TVBMap;
//
static TVBMap _VBMaps[16]; // 4 bits defines the display mode :
// - color / no color
// - faded / not faded
// - textured / not textured
// - closed loop shape / brace shape
/// get a vertex buffer and a primitive suited for the current ribbon
CVBnPB &getVBnPB();
/// get the number of ribbons contained in a vb for a given length. (e.g the number of ribbons that can be batched)
uint getNumRibbonsInVB() const;
//@}
CSmartPtr<ITexture> _Tex;
CPSVector<NLMISC::CVector>::V _Shape;
float _UFactor, _VFactor;
TOrientation _Orientation;
bool _BraceMode : 1; // the ribbon mesh is shaped like a brace
bool _ColorFading : 1;
bool _GlobalColor : 1; // to see whether the system uses global color
bool _Lighted : 1;
bool _ForceLighted : 1;
bool _Touch : 1; // we use this to see if we must setup the material again
void touch() { _Touch = true; }
void updateMaterial();
void updateTexturedMaterial();
void updateUntexturedMaterial();
void setupGlobalColor();
void setupTexturedGlobalColor();
void setupUntexturedGlobalColor();
void setupTextureMatrix();
/// Get the number of vertices in each slices (depends on whether the ribbon is textured or not)
uint getNumVerticesInSlice() const;
};
} // NL3D
#endif // NL_PS_RIBBON_H
/* End of ps_ribbon.h */