// Ryzom - 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 .
#ifndef RY_DEPOSIT_H
#define RY_DEPOSIT_H
#include "nel/ligo/primitive.h"
#include "egs_sheets/egs_static_deposit.h"
#include "game_share/time_weather_season/time_and_season.h"
#include "game_share/ecosystem.h"
#include "game_share/rm_family.h"
#include "harvest_info.h"
#include "mission_manager/ai_alias_translator.h"
#include "egs_variables.h"
#include "nel/misc/random.h"
#include "nel/misc/variable.h"
class CDeposit;
const uint16 MaxNbActiveSources = (uint16)~0;
/**
* A recent forage site prevents from extracting too much material from the same place in a short time.
* When an extraction is succesful, a timer is launched. If no more extraction is done then, the site
* is destroyed after the time is elapsed. If an extraction fails (no more stock available), the site
* will be blocked for extraction (isDepleted() will be true) during the specified time, then freed.
* Note: the timer resolution is abstract. The actual time in ticks will depend on the rate of update.
*/
class CRecentForageSite
{
public:
/// Create forage site. Precondition: dep not NULL.
CRecentForageSite( CDeposit *dep, const NLMISC::CVector& pos ) : _Deposit(dep), _Pos(pos), _TimeToLive(ForageSiteNbUpdatesToLive.get()), _LowScopeStock(ForageSiteStock.get()), _NbActiveSources(0) {}
/// Add a source
bool addActiveSource() { if ( _NbActiveSources == MaxNbActiveSources ) return false; ++_NbActiveSources; return true; }
/// Remove a source
void removeActiveSource() { nlassert(_NbActiveSources!=0); --_NbActiveSources; } // TEMP: assert
/// Return the quantity available in the deposit (or FLT_MAX if there is no quantity constraint)
float getQuantityInDeposit();
/**
* Consume 1 unit of forage site stock by a source in the (independant on the quantity extracted).
* Reset life time of forage site if successful.
* Additionally, consume n units of deposit stock (dependant on the quantity extracted).
* Return the actual quantity that is extracted (can be less that requested if the deposit is (nearly or fully) empty.
*/
float consume( float n );
/// Consume all stock at once
void depleteAll() { _LowScopeStock = 0; _TimeToLive = ForageSiteNbUpdatesToLive.get(); }
/** Update forage site. Return false if the forage site lifetime is ended and the object must be destroyed (which won't occur while there are active sources)
* This method does not need to be called every tick, but at the rate of CDeposit::lowFreqUpdate().
*/
bool lowFreqUpdate() { if ( _TimeToLive != 0 ) { --_TimeToLive; return true; } else return hasActiveSources(); }
/// Return deposit (never NULL).
const CDeposit *deposit() const { return _Deposit; }
/// Return center position
const NLMISC::CVector& pos() const { return _Pos; }
/// Test if site is depleted
bool isDepleted() const { return _LowScopeStock == 0; }
/// Test if site contains a position
bool contains( const NLMISC::CVector& pos ) const;
/// Display debug or stat info
void display( NLMISC::CLog& log ) const;
/// True if the owned deposit allo depletion risk
bool allowDepletionRisk() const;
protected:
/// Test if site has sources
bool hasActiveSources() const { return _NbActiveSources!=0; }
private:
/// Pointer to deposit (deposits don't move in memory). Never NULL.
CDeposit *_Deposit;
/// Center of site area
NLMISC::CVector _Pos;
///
uint16 _TimeToLive;
///
uint16 _LowScopeStock;
///
uint16 _NbActiveSources;
};
typedef std::list CRecentForageSites;
/**
* Type of ecosystem or terrain.
* CEcotype ojects are loaded from primitives, are used to build CDeposit objects, then erased.
*/
class CEcotypeZone : public NLLIGO::CPrimZone
{
public:
/// Init
bool build( const NLLIGO::CPrimZone* zone );
/// Return ecotype
TEcotype ecotype() const { return _Ecotype; }
private:
///
TEcotype _Ecotype;
};
typedef std::vector< CEcotypeZone* > CEcotypeZones;
/**
* Properties of auto-spawn sources of a deposit
*/
struct CAutoSpawnProperties
{
NL_INSTANCE_COUNTER_DECL(CAutoSpawnProperties);
public:
/// In game cycles, but depends of the update frequency of the deposits
uint32 SpawnPeriodGc;
/// Time before disappearance when no extraction has started (game cycles)
uint32 LifeTimeGc;
/// Time remaining since the beginning of the first extraction
uint32 ExtractionTimeGc;
/// Minimium Number of Auto-Spawned Sources that must always exist
uint32 MinimumSpawnedSources;
};
/**
* Quantity constraints
*/
struct CQuantityConstraints
{
NL_INSTANCE_COUNTER_DECL(CQuantityConstraints);
public:
/// Current quantity (shared by all materials in the deposit)
float CurrentQuantity;
/// Day of the next respawn (valid if CurrentQuantity is 0)
uint32 NextRespawnDay;
/// Initial quantity
uint16 InitialQuantity;
/// Time before the initial quantity becomes available backs, in Ryzom Days (note: 30 days for a whole season)
uint16 RespawnTimeRyzomDays;
/// Return the current quantity. If 0 and the respawn time is elapsed, unlock.
float getCurrentQuantity();
/// Consume. If the current quantity falls to 0, lock. Return the actual consumed quantity.
float consumeQuantity( float consumed );
};
namespace NLMISC
{
class IVariable;
class CWordsDictionary;
}
/**
* \author Nicolas Brigand, Alain Saffray, Olivier Cado
* \author Nevrax France
* \date 2003
*/
class CDeposit : public NLLIGO::CPrimZone, public NLMISC::CRefCount
{
public:
/// Constructor
CDeposit() : _AutoSpawnSourcePt(NULL), _QuantityConstraintsPt(NULL), _Ecotype(ECOSYSTEM::common_ecosystem), _FilterPhase(0), _KamiAnger(0.0f), _MinQuality(-1), _MaxQuality(-1), _SourceFXIndex(0), _CanProspect(false), _Enabled(false), _CurrentNbAutoSpawnedSources(0), _AllowDepletionRisk(true) {}
/// Destructor
~CDeposit();
/// Add an ecotype information
static void addEcotype( CEcotypeZone *ecotypeZone ) { _EcotypeZones.push_back( ecotypeZone ); }
/// Init deposit
bool build( const NLLIGO::CPrimZone* zone );
/// Clear all ecotype information, after having built the deposits
static void clearEcotypes();
// Update deposit (especially recent forage sites).
void lowFreqUpdate();
// Update deposit that need to auto spawn because number of harvest sources is too low
void autoSpawnUpdate();
// fill CHarvestInfos with deposit founded
///void harvestInfo( const NLMISC::CEntityId& charId, HARVEST_INFOS::CHarvestInfos& infos );
// character take MP, kami eat this fool character ?
//void characterTakeRM( const NLMISC::CEntityId& charId, uint32 depositIndexContent, uint16 quantity );
// Display deposit content
void displayContent( NLMISC::CLog * log = NLMISC::InfoLog, bool extendedInfo=false, NLMISC::CWordsDictionary *itemDictionary=NULL );
const std::string& name() const { return _Name; }
/// Return the number of MPS in the deposit
uint getContentSize() const { return _RawMaterials.size(); }
/// Return the MPS
const std::vector& getContents() const { return _RawMaterials; }
/// Return the terrain type
TEcotype ecotype() const { return _Ecotype; }
/// Return the MinQuality
sint16 minQuality() const { return _MinQuality; }
/// Return the MaxQuality or -1 if it has to be taken from the raw material information
sint16 maxQuality() const { return _MaxQuality; }
/// Return the index of the visual shape/FX of the sources from this deposit
uint16 sourceFXIndex() const { return _SourceFXIndex; }
/// Return true if the deposit is currently enabled
bool enabled() const { return _Enabled; }
/// Enable(true)/Disable(false) the deposit
void enable(bool bEnable) { _Enabled = bEnable; }
/// Return true if the deposit can be prospected
bool canProspect() const { return _CanProspect; }
/// Return true if the submitted terrain matches the deposit
bool matchEcotype( TEcotype t ) const
{ return t==_Ecotype; }
/// Return true if the submitted season matches the deposit
bool matchSeason( CRyzomTime::ESeason s ) const
{ return _SeasonFilter.empty() || (std::find( _SeasonFilter.begin(), _SeasonFilter.end(), s ) != _SeasonFilter.end()); }
/// Return true if the submitted weather matches the deposit
bool matchWeather( CRyzomTime::EWeather w ) const
{ return _WeatherFilter.empty() || (std::find( _WeatherFilter.begin(), _WeatherFilter.end(), w ) != _WeatherFilter.end()); }
/// Return true if the submitted time of day matches the deposit
bool matchTimeOfDay( CRyzomTime::ETimeOfDay t ) const
{ return _TimeOfDayFilter.empty() || (std::find( _TimeOfDayFilter.begin(), _TimeOfDayFilter.end(), t ) != _TimeOfDayFilter.end()); }
/// Return true if the deposit contains at least one RM of the specified family
bool hasFamily( RM_FAMILY::TRMFamily family ) const;
/// Return true if the deposit contains at least one RM of the specified group
bool hasGroup( RM_GROUP::TRMGroup group ) const;
/// Return true if the deposit contains at least one RM than can craft the specified item part
bool hasRMForItemPart( uint itemPartIndex ) const;
/// Return true if the deposit contains at least one RM with energy lower_eq than the specified value
bool hasLowerStatEnergy( uint8 maxStatEnergy ) const;
/// Return true if the deposit contains at least one RM with energy equalling the specifing value
bool hasExactStatEnergy( uint8 statEnergy ) const;
/// Return the kami anger level (or -1 if disabled)
float kamiAnger() const { return _KamiAnger; }
/// Set the kami anger level
void setKamiAnger( float newKamiAnger ) { _KamiAnger = newKamiAnger; }
/// Increment the kami anger level. React if a threshold is reached. Return the kami anger level (see kamiAnger())
float incKamiAnger( float delta, const std::vector& foragers );
/// Decrement the kami anger level (result clamped to 0)
float decKamiAnger( float delta ) { if ( _KamiAnger != -1.0f ) { _KamiAnger -= delta; if ( _KamiAnger < 0 ) _KamiAnger = 0; } return _KamiAnger; }
/// True if the deposit allow depletion risk for recent forage site
bool allowDepletionRisk() const {return _AllowDepletionRisk;}
/// Return the auto spawn properties (if auto-spawn is on, otherwise return NULL)
const CAutoSpawnProperties *getAutoSpawnProperties() const { return _AutoSpawnSourcePt; }
/// Return the quantuty constraints (if any, otherwise return NULL)
const CQuantityConstraints *getQuantityConstraints() const { return _QuantityConstraintsPt; }
/// Return the quantity available in the deposit (or FLT_MAX if there is no quantity constraint)
float getMaxQuantity() { return _QuantityConstraintsPt ? _QuantityConstraintsPt->getCurrentQuantity() : FLT_MAX; }
/// Consume. Return the actual consumed quantity (may be lower if there is no more to get)
float consumeQuantity( float requested ) { if ( _QuantityConstraintsPt ) return _QuantityConstraintsPt->consumeQuantity( requested ); else return requested; }
/**
* Get a random RM from the neighbourhood of the specified position.
* OptFastFloorBegin()/OptFastFloorEnd() must enclose one or more calls to this method.
*/
const CStaticDepositRawMaterial *getRandomRMAtPos( const NLMISC::CVector& pos, bool testIfSiteDepleted, bool& isDepleted );
/**
* Get the RM at the specified position (no random for map generation).
* OptFastFloorBegin()/OptFastFloorEnd() must enclose one or more calls to this method.
* \param testIfSiteDepleted Set it to false for map generation.
*/
const CStaticDepositRawMaterial *getRMAtPos( const NLMISC::CVector& pos, bool testIfSiteDepleted, bool& isDepleted );
/// Return always a forage site
CRecentForageSite *findOrCreateForageSite( const NLMISC::CVector& pos );
/// Display forage sites info
void displayRecentForageSites( NLMISC::CLog& log ) const;
/// Empty all forage sites
void depleteAllRecentForageSites();
// For auto-spawn source minimum number. Internaly used by CHarvestSource only
void decreaseAutoSpawnedSources();
void increaseAutoSpawnedSources();
protected:
/// Select the raw materials, using the specified filters and _Ecotype
void selectRMsByFilters( std::vector& exactRMCodesS, const std::vector& rmFamilyFilterS, const std::vector& itemPartsFilterS, const std::vector& craftCivS, uint minEnergy, uint maxEnergy );
std::string getSeasonStr() const;
std::string getTimeOfDayStr() const;
std::string getWeatherStr() const;
/**
* Get the ecotype zone under the position.
* This information is valid only at init time. After the deposits are built, this information
* can be found only in deposits.
* If not found, a NULL pointer is returned.
*/
static CEcotypeZone *getEcotypeZone( const NLMISC::CVector& pos );
/// auto spawn a source in the deposit, in the given bbox
void autoSpawnSource(const NLMISC::CVector &cornerMin, const NLMISC::CVector &cornerMax);
private:
/// The ecotype zones (only valid at init, cleared after deposit built)
static CEcotypeZones _EcotypeZones;
/// List of raw materials (const)
std::vector< CStaticDepositRawMaterial > _RawMaterials;
/// Type of terrain (const)
TEcotype _Ecotype;
/// Season filter (const)
std::vector _SeasonFilter;
/// Weather filter (const)
std::vector _WeatherFilter;
/// Time of day filter (const)
std::vector _TimeOfDayFilter;
/// Phase for noise value generator (deduced from season, weather, time... filters)
uint _FilterPhase;
/// Recent forage site list (per deposit, which means forage won't affect another overlapped deposit)
CRecentForageSites _RecentForageSites;
/// Deposit name
std::string _Name;
/// Not null if the deposit can spawn sources without foraging action
CAutoSpawnProperties *_AutoSpawnSourcePt;
/// From 0 to KAMI_ANGER_THRESHOLD 1 & 2, or -1 for N/A; will be persistant some day...
float _KamiAnger;
/// Min quality or -1 if there is no inferior limit
sint16 _MinQuality;
/// Max quality or -1 if it has to be taken from the raw material information
sint16 _MaxQuality;
/// Not null if the deposit has main quantity constraints
CQuantityConstraints *_QuantityConstraintsPt;
/// Index of the visual shape/FX of the sources from this deposit
uint16 _SourceFXIndex;
/// Can a player prospect on this deposit
bool _CanProspect;
/// Is this deposit enabled or disabled (for deposits that are not permanent)
bool _Enabled;
/// Is this deposit allows depletion risk. True by default
bool _AllowDepletionRisk;
// deposit id
//uint _Id;
/// Current Number of AutoSpawned Sources in this deposit
uint32 _CurrentNbAutoSpawnedSources;
};
/**
* Predicate to compare two pointed deposits using their kami anger level.
* If a value is -1, it is not considered as lower.
*/
struct CHasLowerKamiAngerPred : std::binary_function< CDeposit*, CDeposit*, bool >
{
bool operator() ( CDeposit *d1, CDeposit *d2 )
{
return (d1->kamiAnger() >= 0) && (d1->kamiAnger() < d2->kamiAnger());
}
};
/*
* Return the quantity available in the deposit (or FLT_MAX if there is no quantity constraint)
*/
inline float CRecentForageSite::getQuantityInDeposit()
{
return _Deposit->getMaxQuantity();
}
/*
* Consume 1 unit of forage site stock by a source in the (independant on the quantity extracted).
* Reset life time of forage site if successful.
* Additionally, consume n units of deposit stock (dependant on the quantity extracted).
* Return the actual quantity that is extracted (can be less that requested if the deposit is (nearly or fully) empty.
*/
inline float CRecentForageSite::consume( float n )
{
if ( _LowScopeStock != 0 )
{
--_LowScopeStock;
_TimeToLive = ForageSiteNbUpdatesToLive.get();
}
return _Deposit->consumeQuantity( n );
}
/*
* Return true if the deposit allows depletion risk
*/
inline bool CRecentForageSite::allowDepletionRisk() const
{
return _Deposit && _Deposit->allowDepletionRisk();
}
#endif // RY_DEPOSIT_H
/* End of deposit.h */