mirror of
https://port.numenaute.org/aleajactaest/khanat-opennel-code.git
synced 2025-01-11 10:25:22 +00:00
460 lines
16 KiB
C++
460 lines
16 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_VEGETABLE_MANAGER_H
|
|
#define NL_VEGETABLE_MANAGER_H
|
|
|
|
#include "nel/misc/types_nl.h"
|
|
#include "nel/misc/matrix.h"
|
|
#include "nel/misc/rgba.h"
|
|
#include "nel/misc/block_memory.h"
|
|
#include "nel/misc/vector_2f.h"
|
|
#include "nel/misc/smart_ptr.h"
|
|
#include "nel/3d/vegetable_clip_block.h"
|
|
#include "nel/3d/vegetable_sort_block.h"
|
|
#include "nel/3d/vegetable_instance_group.h"
|
|
#include "nel/3d/vegetable_shape.h"
|
|
#include "nel/3d/vegetablevb_allocator.h"
|
|
#include "nel/3d/material.h"
|
|
#include "nel/3d/driver.h"
|
|
#include "nel/3d/vegetable_uv8.h"
|
|
|
|
|
|
namespace NL3D
|
|
{
|
|
|
|
|
|
class CVegetableBlendLayerModel;
|
|
class CScene;
|
|
class CVegetableLightEx;
|
|
|
|
|
|
// ***************************************************************************
|
|
// By default there is 20 layers.
|
|
#define NL3D_VEGETABLE_DEFAULT_NUM_BLEND_LAYER 20
|
|
// default distance is 60 meters.
|
|
#define NL3D_VEGETABLE_DEFAULT_DIST_MAX 60.f
|
|
|
|
|
|
// ***************************************************************************
|
|
/**
|
|
* Manager of vegetable. Instance Factory and rendering.
|
|
* A VegetableManager should be put into a CScene model which is Opaque (ie rendered in Opaque pass), and call
|
|
* vegetableManager::render() at this time. a good example is CLandscape.
|
|
*
|
|
* Because during render(), it uses and setup special "Vegetable Blend Layer models" to render transparents
|
|
* alpha blended vegetables. Toses models are transparent so they are drawn during the transparent pass of
|
|
* the renderTrav's CScene (so after the Opaque pass).
|
|
*
|
|
* \author Lionel Berenguier
|
|
* \author Nevrax France
|
|
* \date 2001
|
|
*/
|
|
class CVegetableManager
|
|
{
|
|
public:
|
|
/// Micro vegetation position against Water. Above water is the default.
|
|
enum TVegetableWater { AboveWater = 0, UnderWater, IntersectWater, VegetInfoLast };
|
|
|
|
public:
|
|
|
|
/**
|
|
* \param maxVertexVbHardUnlit maximum VertexCount in VBHard for Unlit (or precomputed lighted) vegetables
|
|
* \param maxVertexVbHardLighted maximum VertexCount in VBHard for Lighted vegetables
|
|
* \param nbBlendLayers for ZSort/AlphaBlend rdrPass: number of layers of vegetables rendered independently.
|
|
* \param blendLayerDistMax for ZSort/AlphaBlend rdrPass: distance of the farest layer.
|
|
*/
|
|
CVegetableManager(uint maxVertexVbHardUnlit, uint maxVertexVbHardLighted,
|
|
uint nbBlendLayers= NL3D_VEGETABLE_DEFAULT_NUM_BLEND_LAYER,
|
|
float blendLayerDistMax= NL3D_VEGETABLE_DEFAULT_DIST_MAX);
|
|
~CVegetableManager();
|
|
|
|
/** Before any render(), you must call this method (else nlassert). It creates the necessary models in the scene,
|
|
* to manage AlphaBlending correctly. Those models are deleted in the object dtor.
|
|
*/
|
|
void createVegetableBlendLayersModels(CScene *scene);
|
|
|
|
|
|
/// \name Shape management
|
|
// @{
|
|
|
|
/// Load a shape if necessary, and return a shapeId for this shape.
|
|
CVegetableShape *getVegetableShape(const std::string &shape);
|
|
|
|
// @}
|
|
|
|
|
|
/// \name instance management
|
|
// @{
|
|
|
|
/// Create a clipBlock where SortBlock will be created.
|
|
CVegetableClipBlock *createClipBlock();
|
|
/// delete such a clipBlock. all sortBlocks and so all ig must be deleted before.
|
|
void deleteClipBlock(CVegetableClipBlock *clipBlock);
|
|
|
|
/** Create a SortBlock in a clipBlock where instance group (ig) will be created.
|
|
* All AlphaBlend instances created in a SortBlock should have the same vegetWaterState: AboveWater or UnderWater.
|
|
* Each time an instance is added to the sortBlock it changes the _UnderWater state of the sortBlock.
|
|
* \param center you must give an approximate center for the sortBlock (for sorting)
|
|
* \param radius you must give an approximate radius for the sortBlock (for the system to know when you are IN
|
|
* the sortBlock, and then to sort in a better way)
|
|
*/
|
|
CVegetableSortBlock *createSortBlock(CVegetableClipBlock *clipBlock, const CVector ¢er, float radius);
|
|
/// delete such a SortBlock. all ig must be deleted before.
|
|
void deleteSortBlock(CVegetableSortBlock *sortBlock);
|
|
|
|
/** create an instance group in a sortBlock, where instances will be created.
|
|
* Instances will be frustum-clipped by the clipBlock, and sorted (for the ZSort rdrPass only) by sortBlock.
|
|
*/
|
|
CVegetableInstanceGroup *createIg(CVegetableSortBlock *sortBlock);
|
|
/** delete such an ig.
|
|
* After doing this, you must call igSortBlockOwner->updateSortBlock()
|
|
* If the sortBlock has many Igs, you can do it after deleting all your igs.
|
|
*/
|
|
void deleteIg(CVegetableInstanceGroup *ig);
|
|
|
|
// @}
|
|
|
|
|
|
/// \name Adding Instances to an Ig.
|
|
/**
|
|
* Adding instances in an Ig is a 2 step process:
|
|
* - reserve Ig Space
|
|
* - add instances to the ig
|
|
*
|
|
* More precisely:
|
|
* - For all vegetables shapes which will be created in an instance group
|
|
* - reserveIgAddInstances(shape, waterState, number of instances, vegetIgReserve)
|
|
* - call reserveIgCompile(ig, vegetIgReserve)
|
|
* - For all vegetables instances
|
|
* - addInstance(ig, ...)
|
|
*
|
|
* After this setup, you CANNOT add anymore instances to the Ig (this is a requirement for fast allocation).
|
|
*
|
|
* After adding all your instances to an Ig, you must call sortBlockOwnerOfTheIg->updateSortBlock()
|
|
* If the sortBlock has many Igs, you can do it after filling all your igs.
|
|
*/
|
|
// @{
|
|
|
|
/** reserve some instance space in an Ig.
|
|
* nothing is really done here, after doing this for all shapes of your ig, you must call
|
|
* reserveIgCompile()
|
|
* \param vegetIgReserve the object where space required for the ig is added
|
|
*/
|
|
void reserveIgAddInstances(CVegetableInstanceGroupReserve &vegetIgReserve, CVegetableShape *shape, TVegetableWater vegetWaterState, uint numInstances);
|
|
/** reserve the space in the ig.
|
|
* nlassert() if the ig is not empty.
|
|
* \see reserveIgAddInstances()
|
|
*/
|
|
void reserveIgCompile(CVegetableInstanceGroup *ig, const CVegetableInstanceGroupReserve &vegetIgReserve);
|
|
|
|
|
|
/** add an instance to an ig, enlarging the associated clipBlock bbox.
|
|
* If the shape is not lighted, then only diffuseColor is used, to setup color per vertex.
|
|
* Warning! Use OptFastFloor()! So call must be enclosed with a OptFastFloorBegin()/OptFastFloorEnd().
|
|
*
|
|
* Also, buffer must be locked.
|
|
*
|
|
* ambientColor and diffuseColor should be in [0..1] (no clamp), else uint8 will wrap...
|
|
*
|
|
* nlassert() if no sufficient space reserved in reserveIgCompile().
|
|
*
|
|
* \param dlmUV is the dynamic lightmap UV for this vegetable.
|
|
* \see reserveIgAddInstances() reserveIgCompile()
|
|
*/
|
|
void addInstance(CVegetableInstanceGroup *ig,
|
|
CVegetableShape *shape, const NLMISC::CMatrix &mat,
|
|
const NLMISC::CRGBAF &ambientColor, const NLMISC::CRGBAF &diffuseColor,
|
|
float bendFactor, float bendPhase, float bendFreqFactor, float blendDistMax,
|
|
TVegetableWater vegetWaterState, CVegetableUV8 dlmUV);
|
|
|
|
/** Setup a density multiplicator [0,1], for performance reason for instance
|
|
*/
|
|
void setGlobalDensity(float density);
|
|
|
|
/** Get density multiplicator [0,1], for performance reason for instance
|
|
*/
|
|
float getGlobalDensity() const {return _GlobalDensity;}
|
|
|
|
// @}
|
|
|
|
|
|
/// \name render
|
|
// @{
|
|
|
|
/// must give a driver to the vegetableManager, before any addInstance().
|
|
void updateDriver(IDriver *driver);
|
|
|
|
/// load a global texture used for all vegetables (lookup into CPath).
|
|
void loadTexture(const std::string &texName);
|
|
/// setup a global texture used for all vegetables (smartPtr-ized).
|
|
void loadTexture(ITexture *itex);
|
|
/// setup the directional light
|
|
void setDirectionalLight(const CRGBA &ambient, const CRGBA &diffuse, const CVector &light);
|
|
|
|
/** lock any AGP vertex buffers. Do it wisely (just one time before refine as example).
|
|
* You MUST enclose calls to addInstance() (and so CVegetable::generateInstance())
|
|
* with lockBuffers() / unlockBuffers().
|
|
*/
|
|
void lockBuffers();
|
|
/// unlock any AGP vertex buffers
|
|
void unlockBuffers();
|
|
|
|
/** render the manager into a driver, with current viewMatrix/frustum/fog setuped
|
|
* Buffers should be unlocked.
|
|
* \param textureDLM is the dynamic lightmap to use. can be NULL if don't want DLM
|
|
*/
|
|
void render(const CVector &viewCenter, const CVector &frontVector, const std::vector<CPlane> &pyramid,
|
|
ITexture *textureDLM, IDriver *driver);
|
|
|
|
// @}
|
|
|
|
|
|
/// \name Wind animation
|
|
// @{
|
|
|
|
/** set the Wind for animation.
|
|
* All thoses variables may be modified each frame without penalty.
|
|
*
|
|
* \param windDir is the direction of the wind. NB: only XY direction is kept.
|
|
* \param windFreq is the frequency for the animation (speed)
|
|
* \param windPower is the power of the wind, and is a factor (0..1) of Bend
|
|
* \param windBendMin is a value in (0..1) which indicate how much the vegetables are bended at minimum
|
|
* (for very powerfull wind)
|
|
*/
|
|
void setWind(const CVector &windDir, float windFreq, float windPower, float windBendMin);
|
|
|
|
/** set the current Time (in seconds). For Wind animation
|
|
*/
|
|
void setTime(double time);
|
|
|
|
// @}
|
|
|
|
|
|
/// \name UpdateLighting management
|
|
// @{
|
|
|
|
/** set the vegetable manager System Time (in seconds)
|
|
* This time is used for lighting update, and is independent of setTime()
|
|
*/
|
|
void setUpdateLightingTime(double time);
|
|
|
|
/** update the lighting of Igs, within a certain amount of time.
|
|
* You MUST enclose calls to updateLighting() with lockBuffers() / unlockBuffers().
|
|
*/
|
|
void updateLighting();
|
|
|
|
/** set the frequency of lighting update. If freq==1, ALL lighted igs are updated each second.
|
|
* e.g: if 1/20, then every 20 seconds, all Igs are updated.
|
|
* If you set 0, no update will be done at all (this is the default setup!!).
|
|
*/
|
|
void setUpdateLightingFrequency(float freq);
|
|
|
|
/** like updateLighting(), but update ALL vegetable
|
|
* You MUST enclose calls to updateLighting() with lockBuffers() / unlockBuffers().
|
|
*/
|
|
void updateLightingAll();
|
|
|
|
// @}
|
|
|
|
|
|
/// \name Profile
|
|
// @{
|
|
|
|
/// set to 0 the number of faces rendered
|
|
void resetNumVegetableFaceRendered();
|
|
/// get the number of faces rendered by the vegetable manager
|
|
uint getNumVegetableFaceRendered() const;
|
|
|
|
// @}
|
|
|
|
// *********************
|
|
private:
|
|
friend class CVegetableBlendLayerModel;
|
|
|
|
NLMISC::CBlockMemory<CVegetableClipBlock> _ClipBlockMemory;
|
|
NLMISC::CBlockMemory<CVegetableSortBlock> _SortBlockMemory;
|
|
NLMISC::CBlockMemory<CVegetableInstanceGroup> _InstanceGroupMemory;
|
|
|
|
// List of ClipBlock not empty. tested for clipping
|
|
CTessList<CVegetableClipBlock> _ClipBlockList;
|
|
// List of ClipBlock created, with no Ig, so not tested for clipping
|
|
CTessList<CVegetableClipBlock> _EmptyClipBlockList;
|
|
|
|
|
|
// Vegetable Shape map.
|
|
typedef std::map<std::string, CVegetableShape> TShapeMap;
|
|
typedef TShapeMap::iterator ItShapeMap;
|
|
TShapeMap _ShapeMap;
|
|
|
|
|
|
// Vertex Buffers for display. One allocator for Lighted and Unlit mode.
|
|
CVegetableVBAllocator _VBHardAllocator[CVegetableVBAllocator::VBTypeCount];
|
|
// The same, but no VBHard.
|
|
CVegetableVBAllocator _VBSoftAllocator[CVegetableVBAllocator::VBTypeCount];
|
|
// Vertex Program. One VertexProgram for each rdrPass (with / without fog)
|
|
CVertexProgram *_VertexProgram[NL3D_VEGETABLE_NRDRPASS][2];
|
|
|
|
|
|
// Material. Useful for texture and alphaTest
|
|
CMaterial _VegetableMaterial;
|
|
// Norm
|
|
CVector _DirectionalLight;
|
|
NLMISC::CRGBA _GlobalAmbient;
|
|
NLMISC::CRGBA _GlobalDiffuse;
|
|
|
|
/// Global Vegetable Density
|
|
float _GlobalDensity;
|
|
|
|
/// profile
|
|
uint _NumVegetableFaceRendered;
|
|
|
|
|
|
// return true if the ith rdrPass is 2Sided.
|
|
static bool doubleSidedRdrPass(uint rdrPass);
|
|
|
|
|
|
/// get the rdrPass and other info for a given shape.
|
|
uint getRdrPassInfoForShape(CVegetableShape *shape, TVegetableWater vegetWaterState,
|
|
bool &instanceLighted, bool &instanceDoubleSided, bool &instanceZSort,
|
|
bool &destLighted, bool &precomputeLighting);
|
|
|
|
|
|
/// Get the good allocator for the appropriate rdr pass.
|
|
CVegetableVBAllocator &getVBAllocatorForRdrPassAndVBHardMode(uint rdrPass, uint vbHardMode);
|
|
|
|
|
|
/// init the ith vertexProgram.
|
|
void initVertexProgram(uint vpType, bool fogEnabled);
|
|
|
|
|
|
/// setup the vertexProgram constants.
|
|
void setupVertexProgramConstants(IDriver *driver);
|
|
|
|
|
|
/** swap the RdrPass type (hard or soft) of the rdrPass of an instance group.
|
|
* vertices are allocated in other VBallocator, copied and freed in the old VBallocator.
|
|
*/
|
|
void swapIgRdrPassHardMode(CVegetableInstanceGroup *, uint rdrPass);
|
|
|
|
|
|
/// \name Wind animation
|
|
// @{
|
|
CVector _WindDirection;
|
|
float _WindFrequency;
|
|
float _WindPower;
|
|
float _WindBendMin;
|
|
// nb: used for wind animation
|
|
double _Time;
|
|
double _WindPrecRenderTime;
|
|
// updated at each render().
|
|
double _WindAnimTime;
|
|
|
|
// Constant LUT.
|
|
float _CosTable[NL3D_VEGETABLE_VP_LUT_SIZE];
|
|
// computed at each render().
|
|
NLMISC::CVector2f _WindTable[NL3D_VEGETABLE_VP_LUT_SIZE];
|
|
NLMISC::CVector2f _WindDeltaTable[NL3D_VEGETABLE_VP_LUT_SIZE];
|
|
|
|
|
|
// @}
|
|
|
|
|
|
/// \name Misc data to setup renderState (computed at each render())
|
|
// @{
|
|
|
|
CVector _AngleAxis;
|
|
CVector _ViewCenter;
|
|
bool _BkupFog;
|
|
// NB: the manager matrix may not be Identity, for ZBuffer precision reason.
|
|
NLMISC::CMatrix _ManagerMatrix;
|
|
|
|
// @}
|
|
|
|
|
|
/// \name CVegetableBlendLayerModel mgt.
|
|
// @{
|
|
|
|
|
|
/// For Alpha Blend rdrPass, ordering into layers.
|
|
uint _NumZSortBlendLayers;
|
|
float _ZSortLayerDistMax;
|
|
CScene *_ZSortScene;
|
|
std::vector<CVegetableBlendLayerModel*> _ZSortModelLayers;
|
|
// The same but under water
|
|
std::vector<CVegetableBlendLayerModel*> _ZSortModelLayersUW;
|
|
|
|
|
|
/// called by CVegetableBlendLayerModel.
|
|
void setupRenderStateForBlendLayerModel(IDriver *driver);
|
|
void exitRenderStateForBlendLayerModel(IDriver *driver);
|
|
|
|
// @}
|
|
|
|
|
|
/// \name UpdateLighting management
|
|
/**
|
|
* NB: we update at the precision of a shape (a dozen of vertices).
|
|
*/
|
|
// @{
|
|
|
|
// Last update time.
|
|
double _ULPrecTime;
|
|
bool _ULPrecTimeInit;
|
|
double _ULTime;
|
|
|
|
/// Frequency of update.
|
|
float _ULFrequency;
|
|
/// Current number of vertices to update. If negative, I have some advance.
|
|
float _ULNVerticesToUpdate;
|
|
/// Sum of all ig vertices to update.
|
|
uint _ULNTotalVertices;
|
|
/// the priority list of ig to update
|
|
CVegetableInstanceGroup *_ULRootIg;
|
|
/// Current instance to render in the first ig to update: rdrpass/instanceId.
|
|
uint _ULCurrentIgRdrPass;
|
|
uint _ULCurrentIgInstance;
|
|
|
|
/// update lighting according to _ULNVerticesToUpdate
|
|
void doUpdateLighting();
|
|
|
|
/** update part of the RootIg, according to _ULNVerticesToUpdate (while > 0)
|
|
* if all Ig is updated, return true and _ULCurrentIgRdrPass and _ULCurrentIgInstance is updated.
|
|
*/
|
|
bool updateLightingIGPart();
|
|
|
|
|
|
/** update part of an ig. Do not use/modify _UL*
|
|
* return number of vertices processed (nb vertices of the shape)
|
|
*/
|
|
uint updateInstanceLighting(CVegetableInstanceGroup *ig, uint rdrPassId, uint instanceId);
|
|
|
|
|
|
// @}
|
|
|
|
// last driver setupped with a call to update driver
|
|
NLMISC::CRefPtr<IDriver> _LastDriver;
|
|
|
|
|
|
};
|
|
|
|
|
|
} // NL3D
|
|
|
|
|
|
#endif // NL_VEGETABLE_MANAGER_H
|
|
|
|
/* End of vegetable_manager.h */
|