2012-05-29 13:31:11 +00:00
// Ryzom - MMORPG Framework <http://dev.ryzom.com/projects/ryzom/>
// 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 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 < CRecentForageSite > 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 ( uint ) _RawMaterials . size ( ) ; }
/// Return the MPS
const std : : vector < CStaticDepositRawMaterial > & 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 < TDataSetRow > & 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 < std : : string > & exactRMCodesS , const std : : vector < std : : string > & rmFamilyFilterS , const std : : vector < std : : string > & itemPartsFilterS , const std : : vector < std : : string > & 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 < CRyzomTime : : ESeason > _SeasonFilter ;
/// Weather filter (const)
std : : vector < CRyzomTime : : EWeather > _WeatherFilter ;
/// Time of day filter (const)
std : : vector < CRyzomTime : : ETimeOfDay > _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 */