// 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 .
#include "stdpch.h"
#include "deposit.h"
#include "player_manager/character.h"
#include "player_manager/player_manager.h"
#include "player_manager/player.h"
#include "zone_manager.h"
#include "entities_game_service.h"
#include "egs_globals.h"
#include "nel/misc/noise_value.h"
#include "nel/misc/variable.h"
#include "nel/misc/words_dictionary.h"
#include "game_share/time_weather_season/time_date_season_manager.h"
#include "game_share/people.h"
#include "egs_sheets/egs_sheets.h"
#include "phrase_manager/fg_prospection_phrase.h"
#include "phrase_manager/phrase_utilities_functions.h"
#include "game_share/send_chat.h"
#include "game_share/multi_target.h"
#include "phrase_manager/s_effect.h"
#include "projectile_stats.h"
using namespace std;
using namespace NLMISC;
using namespace NLLIGO;
using namespace NLNET;
using namespace RM_FAMILY;
const float SmallDepositAreaThreshold = 64.0f*64.0f;
const float SmallDepositAreaReference = 16.0f*32.0f;
// The ecotype zones (only valid at init, cleared after deposit built)
CEcotypeZones CDeposit::_EcotypeZones;
NL_INSTANCE_COUNTER_IMPL(CAutoSpawnProperties);
NL_INSTANCE_COUNTER_IMPL(CQuantityConstraints);
// Verbose deposits debug variable
//bool VerboseDeposits = false;
// Total number of CRecentForageSite objects (debug info)
uint32 TotalNbRecentForageSites = 0;
/// Export deposit contents report
bool ExportDepositContents = false;
/// Verbose the items parsed when filtering deposits
bool VerboseDepositFiltering = false;
void cbChangeDepositUpdateFrequency( NLMISC::IVariable& v )
{
/*const TGameCycle expectedTTL = 6000; // 10 min
if ( DepositUpdateFrequency.get() != 0 )
ForageSiteNbUpdatesToLive.set( (uint16)(expectedTTL / DepositUpdateFrequency.get()) );*/
nlinfo( "TODO: change ForageSiteNbUpdatesToLive according to DepositUpdateFrequency" );
}
/*
* Utility functions
*/
bool malformed( const char *field, const string& name )
{
nlwarning( "FG: Malformed primitive %s, missing field '%s'", name.c_str(), field );
return false;
}
/*
* Get the enums of the families (using RM_FAMILY::toFamily(), assumes the string are ended by _value)
* where value is the identifier number (see .typ).
* Removes duplicates (with a warning).
*/
void convertRMFamiliesNames( const string& name, const vector& src, vector& dest )
{
for ( uint i=0; i!=src.size(); ++i )
{
if ( ! src[i].empty() )
{
string::size_type p = src[i].find_last_of( '_' );
if ( p != string::npos )
{
TRMFamily family = toFamily( src[i].substr( p + 1 ) );
if ( find( dest.begin(), dest.end(), family ) == dest.end() )
dest.push_back( family );
else
nlwarning( "FG: Deposit %s: %s found twice", name.c_str(), src[i].c_str() );
}
else
{
nlwarning( "FG: Deposit %s: %s: not found", name.c_str(), src[i].c_str() );
}
}
}
}
/*
* Get the indices of the item parts (assumes the strings are ended by _index).
* Ex: blade_0.
* Removes duplicates (with a warning).
*/
void convertItemPartsNames( const string& name, const vector& src, vector& dest )
{
for ( uint i=0; i!=src.size(); ++i )
{
if ( ! src[i].empty() )
{
string::size_type p = src[i].find_last_of( '_' );
if ( p != string::npos )
{
uint itemPart;
NLMISC::fromString(src[i].substr( p + 1 ), itemPart);
if ( find( dest.begin(), dest.end(), itemPart ) == dest.end() )
dest.push_back( itemPart );
else
nlwarning( "FG: Deposit %s: %s found twice", name.c_str(), src[i].c_str() );
}
else
{
nlwarning( "FG: Deposit %s: %s: not found", name.c_str(), src[i].c_str() );
}
}
}
}
/*
* Get the civ enum.
*/
void convertCraftCivNames( const vector& src, vector& dest )
{
dest.resize( src.size() );
for ( uint i=0; i!=src.size(); ++i )
{
dest[i] = ITEM_ORIGIN::stringToEnum( src[i] );
}
}
/*
*
*/
inline void makeFullSItemCode( string& code )
{
if ( code.find( ".sitem" ) == string::npos )
{
code += ".sitem";
}
}
/*
* Return always a forage site
*/
CRecentForageSite *CDeposit::findOrCreateForageSite( const NLMISC::CVector& pos )
{
// Search in existing ones
for ( CRecentForageSites::iterator ihs=_RecentForageSites.begin(); ihs!=_RecentForageSites.end(); ++ihs )
{
if ( (*ihs).contains( pos ) )
{
return &(*ihs);
}
}
// Not found, create one
++TotalNbRecentForageSites;
CRecentForageSite newForageSite( this, pos );
_RecentForageSites.push_back( newForageSite );
return &_RecentForageSites.back(); // returning the pointer is valid because an element in std::list is never invalidated
}
/*
*
*/
bool CRecentForageSite::contains( const NLMISC::CVector& pos ) const
{
float r = ForageSiteRadius.get();
return ((pos-_Pos).sqrnorm() < r*r);
}
/*
* Display debug or stat info
*/
void CRecentForageSite::display( NLMISC::CLog& log ) const
{
log.displayNL( "Forage site at %s: %u sources, stock %u, %u lowfreq updates remaining",
_Pos.asString().c_str(), _NbActiveSources, _LowScopeStock, _TimeToLive );
}
//-----------------------------------------------------------------------------
// Parse primitive file for one ecotype
//-----------------------------------------------------------------------------
bool CEcotypeZone::build( const NLLIGO::CPrimZone* zone )
{
*( (NLLIGO::CPrimZone*)this ) = *zone;
// Read primitive name
string name;
if ( ! zone->getPropertyByName( "name", name ) ) return malformed( "name", name );
// Read ecotype
string ecotypeS;
if ( ! zone->getPropertyByName( "ecotype", ecotypeS ) ) return malformed( "ecotype", name );
_Ecotype = ECOSYSTEM::stringToEcosystem( ecotypeS ); // needs a case-unsensitive comparison, because the
if ( _Ecotype == ECOSYSTEM::unknown ) // World Editor lists names of files in CVS (in deposit_system/ecotypes)
nlwarning( "%s: Ecotype %s unknown", name.c_str(), ecotypeS.c_str() );
return true;
}
/*
* Destructor
*/
CDeposit::~CDeposit()
{
if ( _AutoSpawnSourcePt )
delete _AutoSpawnSourcePt;
_AutoSpawnSourcePt = NULL;
if ( _QuantityConstraintsPt )
delete _QuantityConstraintsPt;
_QuantityConstraintsPt = NULL;
// Avoid any bug, remove the deposit from zone manager
CZoneManager::getInstance().unregisterDepositToAutoSpawnUpdate(this);
}
/*
* 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.
*/
CEcotypeZone *CDeposit::getEcotypeZone( const NLMISC::CVector& pos )
{
// The ecotypes must not be overlapped: only the first one found is returned
for ( CEcotypeZones::iterator it=_EcotypeZones.begin(); it!=_EcotypeZones.end(); ++it )
{
CEcotypeZone *ecotypeZone = (*it);
if ( ecotypeZone->contains( pos ) )
{
return ecotypeZone;
}
}
return NULL;
}
/*
* Clear ecotype information, after having built the deposits
*/
void CDeposit::clearEcotypes()
{
for ( CEcotypeZones::iterator iez=_EcotypeZones.begin(); iez!=_EcotypeZones.end(); ++iez )
{
delete (*iez);
}
_EcotypeZones.clear();
}
struct TCompareStaticItemPtrBySheetId : public std::binary_function
{
bool operator() ( const CStaticItem* p1, const CStaticItem* p2 )
{
return (p1->SheetId < p2->SheetId);
}
};
//-----------------------------------------------------------------------------
// Parse primitive file for one deposit
//-----------------------------------------------------------------------------
bool CDeposit::build( const NLLIGO::CPrimZone* zone )
{
if ( IsRingShard )
return false;
//_Id =id;
*( (NLLIGO::CPrimZone*)this ) = *zone;
// Read primitive name
string name;
if ( ! zone->getPropertyByName( "name", name ) ) return malformed( "name", name );
_Name = name;
// Read exact raw material codes to add
vector *exactRMCodesS = NULL;
if ( ! (zone->getPropertyByName( "exact_mp_item", exactRMCodesS ) && exactRMCodesS) ) return malformed( "exact_mp_item", name );
// Read raw material family filter
vector *rmFamilyFilterS = NULL;
if ( ! (zone->getPropertyByName( "mps", rmFamilyFilterS ) && rmFamilyFilterS) ) return malformed( "mps", name );
// Read item part / craft civ filter
vector *itemPartsFilterS = NULL, *craftCivS = NULL;
if ( ! (zone->getPropertyByName( "item_parts", itemPartsFilterS ) && itemPartsFilterS) ) return malformed( "item_parts", name );
if ( ! (zone->getPropertyByName( "craft_civ", craftCivS ) && craftCivS) ) return malformed( "craft_civ", name );
// Read stat+quality filters
string minEnergyS, maxEnergyS, minQualityS, maxQualityS;
if ( ! zone->getPropertyByName( "deposit_statquality_min", minEnergyS ) ) return malformed( "deposit_statquality_min", name );
if ( ! zone->getPropertyByName( "deposit_statquality_max", maxEnergyS ) ) return malformed( "deposit_statquality_max", name );
if ( ! zone->getPropertyByName( "deposit_min_quality_250", minQualityS ) ) return malformed( "deposit_min_quality_250", name );
if ( ! zone->getPropertyByName( "deposit_max_quality_250", maxQualityS ) ) return malformed( "deposit_max_quality_250", name );
NLMISC::fromString(minQualityS, _MinQuality);
NLMISC::fromString(maxQualityS, _MaxQuality);
// Read quantity constraints
string qttyLimitS, qttyRespawnTimeS;
if ( ! zone->getPropertyByName( "deposit_quantity_limit", qttyLimitS ) ) return malformed( "deposit_quantity_limit", name );
if ( ! zone->getPropertyByName( "deposit_quantity_respawn_time_ryzomdays", qttyRespawnTimeS ) ) return malformed( "deposit_quantity_respawn_time_ryzomdays", name );
sint qttyLimit;
NLMISC::fromString(qttyLimitS, qttyLimit);
if ( qttyLimit > -1 )
{
sint qttyRespawnTime;
NLMISC::fromString(qttyRespawnTimeS, qttyRespawnTime);
if ( (qttyLimit == 0) || (qttyLimit > 0xFFFF) || (qttyRespawnTime < 1) || (qttyRespawnTime > 0xFFFF) )
nlwarning( "Invalid limit or respawn time too high in %s", name.c_str() );
else
{
_QuantityConstraintsPt = new CQuantityConstraints();
_QuantityConstraintsPt->CurrentQuantity = (float)qttyLimit;
_QuantityConstraintsPt->InitialQuantity = (uint16)qttyLimit;
_QuantityConstraintsPt->RespawnTimeRyzomDays = qttyRespawnTime;
}
}
// Get ecotype
CEcotypeZone *ecotypeZone = getEcotypeZone( getBarycentre() );
_Ecotype = ecotypeZone ? ecotypeZone->ecotype() : ECOSYSTEM::common_ecosystem;
if ( ! ecotypeZone )
nlwarning( "FG: Deposit %s has no ecotype", name.c_str() );
_FilterPhase = 0;
// Read season(s)
string s1s, s2s, s3s, s4s;
if ( ! zone->getPropertyByName( "while_season_spring", s1s ) ) return malformed( "while_season_spring", name );
if ( ! zone->getPropertyByName( "while_season_summer", s2s ) ) return malformed( "while_season_summer", name );
if ( ! zone->getPropertyByName( "while_season_automn", s3s ) ) return malformed( "while_season_automn", name );
if ( ! zone->getPropertyByName( "while_season_winter", s4s ) ) return malformed( "while_season_winter", name );
bool s1 = (s1s=="true"), s2 = (s2s=="true"), s3 = (s3s=="true"), s4 = (s4s=="true");
if ( ! (s1 && s2 && s3 && s4) )
{
if ( s1 ) { _SeasonFilter.push_back( EGSPD::CSeason::Spring ); _FilterPhase |= 0x1; }
if ( s2 ) { _SeasonFilter.push_back( EGSPD::CSeason::Summer ); _FilterPhase |= 0x2; }
if ( s3 ) { _SeasonFilter.push_back( EGSPD::CSeason::Autumn ); _FilterPhase |= 0x4; }
if ( s4 ) { _SeasonFilter.push_back( EGSPD::CSeason::Winter ); _FilterPhase |= 0x8; }
}
// Read weather
string w1s, w2s, w3s, w4s;
if ( ! zone->getPropertyByName( "while_weather_0_best", w1s ) ) return malformed( "while_weather_0_best", name );
if ( ! zone->getPropertyByName( "while_weather_1_good", w2s ) ) return malformed( "while_weather_1_good", name );
if ( ! zone->getPropertyByName( "while_weather_2_bad", w3s ) ) return malformed( "while_weather_2_bad", name );
if ( ! zone->getPropertyByName( "while_weather_3_worst", w4s ) ) return malformed( "while_weather_3_worst", name );
bool w1 = (w1s=="true"), w2 = (w2s=="true"), w3 = (w3s=="true"), w4 = (w4s=="true");
if ( ! (w1 && w2 && w3 && w4) )
{
if ( w1 ) { _WeatherFilter.push_back( CRyzomTime::best ); _FilterPhase |= 0x10; }
if ( w2 ) { _WeatherFilter.push_back( CRyzomTime::good ); _FilterPhase |= 0x20; }
if ( w3 ) { _WeatherFilter.push_back( CRyzomTime::bad ); _FilterPhase |= 0x40; }
if ( w4 ) { _WeatherFilter.push_back( CRyzomTime::worst ); _FilterPhase |= 0x80; }
}
// Read time of day
string t1s, t2s;
if ( ! zone->getPropertyByName( "while_its_day", t1s ) ) return malformed ( "while_its_day", name );
if ( ! zone->getPropertyByName( "while_its_night", t2s ) ) return malformed ( "while_its_night", name );
bool t1 = (t1s=="true"), t2 = (t2s=="true");
if ( ! (t1 && t2) )
{
if ( t1 )
{
_TimeOfDayFilter.push_back( CRyzomTime::dawn );
_TimeOfDayFilter.push_back( CRyzomTime::day );
_TimeOfDayFilter.push_back( CRyzomTime::evening );
_FilterPhase |= 0x100;
}
if ( t2 )
{
_TimeOfDayFilter.push_back( CRyzomTime::nightfall );
_TimeOfDayFilter.push_back( CRyzomTime::night );
_FilterPhase |= 0x200;
}
}
// Read auto-spawn properties
string assS;
if ( ! zone->getPropertyByName( "auto_spawn_sources", assS ) ) return malformed( "auto_spawn_sources", name );
if ( assS=="true" )
{
string aspS, asltS, asetS, amin; // not really 'average' anymore
if ( ! zone->getPropertyByName( "auto_spawn_average_period_s", aspS ) ) return malformed( "auto_spawn_average_period", name );
if ( ! zone->getPropertyByName( "auto_spawn_lifetime_s", asltS ) ) return malformed( "auto_spawn_lifetime", name );
if ( ! zone->getPropertyByName( "auto_spawn_extraction_time_s", asetS ) ) return malformed( "auto_spawn_sources", name );
if ( ! zone->getPropertyByName( "auto_spawn_min_source", amin ) ) return malformed( "auto_spawn_min_source", name );
_AutoSpawnSourcePt = new CAutoSpawnProperties;
NLMISC::fromString(aspS, _AutoSpawnSourcePt->SpawnPeriodGc);
_AutoSpawnSourcePt->SpawnPeriodGc *= 10;
NLMISC::fromString(asltS, _AutoSpawnSourcePt->LifeTimeGc);
_AutoSpawnSourcePt->LifeTimeGc *= 10;
NLMISC::fromString(asetS, _AutoSpawnSourcePt->ExtractionTimeGc);
_AutoSpawnSourcePt->ExtractionTimeGc *= 10;
NLMISC::fromString(amin, _AutoSpawnSourcePt->MinimumSpawnedSources);
// security!
_AutoSpawnSourcePt->MinimumSpawnedSources = min(uint32(100), _AutoSpawnSourcePt->MinimumSpawnedSources);
}
else
{
_AutoSpawnSourcePt = NULL;
}
// Read source FX index
string srcFXIndexS;
if ( ! zone->getPropertyByName( "source_fx", srcFXIndexS ) ) return malformed( "source_fx", name );
NLMISC::fromString(srcFXIndexS, _SourceFXIndex);
// Read other initial properties
string cpS, eS, ikaS, adpR;
if ( ! zone->getPropertyByName( "can_prospect", cpS ) ) return malformed( "can_prospect", name );
if ( ! zone->getPropertyByName( "enabled", eS ) ) return malformed( "enabled", name );
if ( ! zone->getPropertyByName( "initial_kami_anger", ikaS ) ) return malformed( "initial_kami_anger", name );
if ( ! zone->getPropertyByName( "can_have_depletion_risk", adpR ) ) return malformed( "can_have_depletion_risk", name );
_CanProspect = (cpS=="true");
_Enabled = (eS=="true");
_AllowDepletionRisk = (adpR=="true");
_KamiAnger = (float)atof( ikaS.c_str() );
if ( (_KamiAnger != -1.0f) && (_KamiAnger < 0) )
nlwarning( "Invalid initial_kami_anger %.1f in %s", _KamiAnger, name.c_str() );
// Apply filters
uint32 minEnergy, maxEnergy;
NLMISC::fromString(minEnergyS, minEnergy);
NLMISC::fromString(maxEnergyS, maxEnergy);
if ( exactRMCodesS->empty() && rmFamilyFilterS->empty() && itemPartsFilterS->empty() )
{
nlwarning( "FG: Deposit %s: No RM, exactRms or item parts specified!", name.c_str() );
return false;
}
else
selectRMsByFilters( *exactRMCodesS, *rmFamilyFilterS, *itemPartsFilterS, *craftCivS, minEnergy, maxEnergy );
nldebug( "FG: Built deposit %s %s %s %s %s", name.c_str(), _AutoSpawnSourcePt?"AUTO":"-", _CanProspect?"PRO":"-", _AllowDepletionRisk?"ADPR":"-", _Enabled?"ON":"OFF" );
return true;
}
/*
* Select the raw materials, using the specified filters and _Ecotype
*/
void CDeposit::selectRMsByFilters( std::vector& exactRMCodesS, const std::vector& rmFamilyFilterS, const std::vector& itemPartsFilterS, const std::vector& craftCivS, uint minEnergy, uint maxEnergy )
{
vector rmFamilyFilter;
convertRMFamiliesNames( _Name, rmFamilyFilterS, rmFamilyFilter );
vector itemPartsFilter;
convertItemPartsNames( _Name, itemPartsFilterS, itemPartsFilter );
vector craftCivFilter;
convertCraftCivNames( craftCivS, craftCivFilter );
for_each( exactRMCodesS.begin(), exactRMCodesS.end(), makeFullSItemCode );
if ( VerboseDepositFiltering )
nldebug( "%s", _Name.c_str() );
// Sort the items by sheetId so that the deposits are not dependant on the hash map ordering
const CAllStaticItems& allItems = CSheets::getItemMapForm();
vector< const CStaticItem* > sortedItems( allItems.size() );
uint i = 0;
for ( CAllStaticItems::const_iterator it=allItems.begin(); it!=allItems.end(); ++it, ++i )
{
sortedItems[i] = &((*it).second);
}
std::sort( sortedItems.begin(), sortedItems.end(), TCompareStaticItemPtrBySheetId() );
for ( vector< const CStaticItem* >::const_iterator it=sortedItems.begin(); it!=sortedItems.end(); ++it )
{
// Eliminate non raw materials
const CStaticItem& staticItem = *(*it);
if ( ! staticItem.Mp )
continue;
const string& sheetName = staticItem.SheetId.toString();
if ( (sheetName.size() < (CREATURE_OR_DEPOSIT_MP_CHAR+1)) || (sheetName[0] != 'm' ) )
continue;
// Keep raw material directly if matching one of specified exact raw material codes (including creature's raw materials)
if ( find( exactRMCodesS.begin(), exactRMCodesS.end(), sheetName ) == exactRMCodesS.end() )
{
// Eliminate non 'forage' raw materials
if ( VerboseDepositFiltering )
nldebug( "FG: Submitting %s", sheetName.c_str() );
if ( (sheetName[CREATURE_OR_DEPOSIT_MP_CHAR] != 'd') )
continue;
// Eliminate raw materials of incompatible ecosystem
if ( (_Ecotype != ECOSYSTEM::common_ecosystem) && (staticItem.Mp->Ecosystem != ECOSYSTEM::common_ecosystem) &&
(staticItem.Mp->Ecosystem != _Ecotype) )
{
if ( VerboseDepositFiltering ) nldebug( "-Ecotype %s", ECOSYSTEM::toString( staticItem.Mp->Ecosystem ).c_str() );
continue;
}
// Match energy filter (TEMP: 25 is the minimum maxEnergy threshold) (include boundaries)
if ( (staticItem.Mp->StatEnergy < minEnergy) || (staticItem.Mp->StatEnergy > maxEnergy /*max(maxEnergy,(uint)25)*/) )
{
if ( VerboseDepositFiltering ) nldebug( "-StatEnergy %hu", staticItem.Mp->StatEnergy );
continue;
}
// Match rmFamilyFilter or itemPartsFilter
if ( find( rmFamilyFilter.begin(), rmFamilyFilter.end(), staticItem.Mp->Family ) == rmFamilyFilter.end() )
{
// If not found in rmFamilyFilter, try if one of itemPartsFilter is matched along with civ
bool found = false;
for ( vector::iterator ipf=itemPartsFilter.begin(); ipf!=itemPartsFilter.end(); ++ipf )
{
uint itemPartIndex = (*ipf);
// Find if the current RM has the item part matching one of the filter
const CMP::TMpFaberParameters *mpFaberParam = staticItem.Mp->getMpFaberParameters( itemPartIndex );
if ( mpFaberParam && (mpFaberParam->Durability != 0) )
{
if ( (craftCivFilter.empty()) || // no civ constraint
(mpFaberParam->CraftCivSpec == ITEM_ORIGIN::COMMON) || // RM matches all civs
(find( craftCivFilter.begin(), craftCivFilter.end(), mpFaberParam->CraftCivSpec ) != craftCivFilter.end()) ) // RM matches civ constraint
{
if ( VerboseDepositFiltering ) nldebug( "+StatEnergy %hu +ItemPart %c +CivSpec %s", staticItem.Mp->StatEnergy, 'A' + (char)itemPartIndex, ITEM_ORIGIN::enumToString( mpFaberParam->CraftCivSpec ).c_str() );
found = true;
break; // RM is selected, no need to test other item parts
}
else
if ( VerboseDepositFiltering ) nldebug( "-CivSpec %s", ITEM_ORIGIN::enumToString( mpFaberParam->CraftCivSpec ).c_str() );
}
else
{
if ( VerboseDepositFiltering ) nldebug( "-ItemPart %c", 'A' + (char)itemPartIndex );
}
}
if ( ! found )
continue;
}
/*else
{
if ( VerboseDepositFiltering ) nldebug( "+rmFamilyFilter %u", staticItem.Mp->Family );
}*/
}
/*else
{
if ( VerboseDepositFiltering ) nldebug( "+exact_mp_item %s", sheetName.c_str() );
}*/
// Select if matching (do not check duplicate, keep them if item part matches explicit family)
CStaticDepositRawMaterial rm;
rm.MaterialSheet = staticItem.SheetId;
_RawMaterials.push_back( rm );
if ( VerboseDepositFiltering )
nldebug( "SELECTED %s", sheetName.c_str() );
// Export deposit contents report if requested
if ( ExportDepositContents )
{
static FILE *depositReportFile;
static bool depositReportCreated = false;
if ( ! depositReportCreated )
{
depositReportCreated = true;
depositReportFile = fopen( "deposit_contents.csv", "wt" ); // fclose() auto?
if ( depositReportFile )
{
fprintf( depositReportFile, "Deposit;RM;When in year;When in day;Weather;\n" );
}
}
if ( depositReportFile )
fprintf( depositReportFile, "%s;%s;%s;%s;%s;\n", _Name.c_str(), rm.MaterialSheet.toString().c_str(), getSeasonStr().c_str(), getTimeOfDayStr().c_str(), getWeatherStr().c_str() );
}
}
if ( _RawMaterials.empty() )
nlwarning( "FG: Selected 0 items in deposit %s", _Name.c_str() );
else
nldebug( "FG: Selected %u RM (on %u items) in deposit %s", _RawMaterials.size(), allItems.size(), _Name.c_str() );
}
/**
* Helper for CDeposit::hasFamily()
*/
struct CIsOfFamilyPred : public std::binary_function< CStaticDepositRawMaterial, RM_FAMILY::TRMFamily, bool >
{
/// Predicate
bool operator() ( const CStaticDepositRawMaterial& rm, RM_FAMILY::TRMFamily family ) const
{
const CAllStaticItems& allItems = CSheets::getItemMapForm();
CAllStaticItems::const_iterator it = allItems.find( rm.MaterialSheet );
if ( it != allItems.end() )
{
const CStaticItem& staticItem = (*it).second;
return staticItem.Mp && (staticItem.Mp->Family == family);
}
else
return false;
}
};
/**
* Helper for CDeposit::hasFamily()
*/
struct CIsOfGroupPred : public std::binary_function< CStaticDepositRawMaterial, RM_GROUP::TRMGroup, bool >
{
/// Predicate
bool operator() ( const CStaticDepositRawMaterial& rm, RM_GROUP::TRMGroup group ) const
{
const CAllStaticItems& allItems = CSheets::getItemMapForm();
CAllStaticItems::const_iterator it = allItems.find( rm.MaterialSheet );
if ( it != allItems.end() )
{
const CStaticItem& staticItem = (*it).second;
return staticItem.Mp && (staticItem.Mp->getGroup() == group);
}
else
return false;
}
};
/**
* Helper for CDeposit::hasRMForItemPart()
*/
struct CCanCraftItemPartPred : public std::binary_function< CStaticDepositRawMaterial, uint, bool >
{
/// Predicate
bool operator() ( const CStaticDepositRawMaterial& rm, uint itemPartIndex ) const
{
const CAllStaticItems& allItems = CSheets::getItemMapForm();
CAllStaticItems::const_iterator it = allItems.find( rm.MaterialSheet );
if ( it != allItems.end() )
{
const CStaticItem& staticItem = (*it).second;
if ( staticItem.Mp )
{
const CMP::TMpFaberParameters *mpFaberParam = staticItem.Mp->getMpFaberParameters( itemPartIndex );
return (mpFaberParam && (mpFaberParam->Durability != 0));
}
else
return false;
}
else
return false;
}
};
/**
* Helper for CDeposit::hasFamily()
*/
struct CMatchStatEnergyPred : public std::binary_function< CStaticDepositRawMaterial, uint8, bool >
{
/// Predicate
bool operator() ( const CStaticDepositRawMaterial& rm, uint8 maxStatEnergy ) const
{
const CAllStaticItems& allItems = CSheets::getItemMapForm();
CAllStaticItems::const_iterator it = allItems.find( rm.MaterialSheet );
if ( it != allItems.end() )
{
const CStaticItem& staticItem = (*it).second;
return staticItem.Mp && (staticItem.Mp->StatEnergy <= maxStatEnergy); // include boundary
}
else
{
nlwarning( "%s not found", rm.MaterialSheet.toString().c_str() );
return true;
}
}
};
/**
* Helper for CDeposit::hasFamily()
*/
struct CMatchExactStatEnergyPred : public std::binary_function< CStaticDepositRawMaterial, uint8, bool >
{
/// Predicate
bool operator() ( const CStaticDepositRawMaterial& rm, uint8 maxStatEnergy ) const
{
const CAllStaticItems& allItems = CSheets::getItemMapForm();
CAllStaticItems::const_iterator it = allItems.find( rm.MaterialSheet );
if ( it != allItems.end() )
{
const CStaticItem& staticItem = (*it).second;
return staticItem.Mp && (staticItem.Mp->StatEnergy == maxStatEnergy);
}
else
return false;
}
};
/*
* Return true if the deposit contains at least one RM of the specified family
*/
bool CDeposit::hasFamily( RM_FAMILY::TRMFamily family ) const
{
return find_if( _RawMaterials.begin(), _RawMaterials.end(), bind2nd( CIsOfFamilyPred(), family ) ) != _RawMaterials.end();
}
/*
* Return true if the deposit contains at least one RM of the specified group
*/
bool CDeposit::hasGroup( RM_GROUP::TRMGroup group ) const
{
return find_if( _RawMaterials.begin(), _RawMaterials.end(), bind2nd( CIsOfGroupPred(), group ) ) != _RawMaterials.end();
}
/*
* Return true if the deposit contains at least one RM than can craft the specified item part
*/
bool CDeposit::hasRMForItemPart( uint itemPartIndex ) const
{
return find_if( _RawMaterials.begin(), _RawMaterials.end(), bind2nd( CCanCraftItemPartPred(), itemPartIndex ) ) != _RawMaterials.end();
}
/*
* Return true if the deposit contains at least one RM with energy lower_eq than the specified value
*/
bool CDeposit::hasLowerStatEnergy( uint8 maxStatEnergy ) const
{
return find_if( _RawMaterials.begin(), _RawMaterials.end(), bind2nd( CMatchStatEnergyPred(), maxStatEnergy ) ) != _RawMaterials.end();
}
/*
* Return true if the deposit contains at least one RM with energy equalling the specifing value
*/
bool CDeposit::hasExactStatEnergy( uint8 statEnergy ) const
{
return find_if( _RawMaterials.begin(), _RawMaterials.end(), bind2nd( CMatchExactStatEnergyPred(), statEnergy ) ) != _RawMaterials.end();
}
/*
* Called by lowFreqUpdate and autoSpawnUpdate
*/
void CDeposit::autoSpawnSource(const CVector &cornerMin, const CVector &cornerMax)
{
/*
Yoyo: This method may fail because it selects a random position in a box and then check if it's in the zone.
We can do better by making a list of triangles of the concave polygon, then selecting randomly and
carefully in this list of triangle.
*/
// Randomize a position in the deposit
CVector pos;
pos.z = 0;
const uint NB_ATTEMPTS = 5;
uint iAttempt;
for ( iAttempt=0; iAttempt!=NB_ATTEMPTS; ++iAttempt )
{
pos.x = cornerMin.x + RandomGenerator.frand( cornerMax.x - cornerMin.x );
pos.y = cornerMin.y + RandomGenerator.frand( cornerMax.y - cornerMin.y );
if ( contains( pos ) )
break;
}
// If a valid position could not be found, abort spawning of this source
if ( iAttempt == NB_ATTEMPTS )
return;
// Spawn a source
CFgProspectionPhrase::autoSpawnSource( pos, this );
}
/*
* Update deposit (especially recent forage sites)
*/
void CDeposit::lowFreqUpdate()
{
if ( !HarvestSystemEnabled )
return;
// Update recent forage sites
for ( CRecentForageSites::iterator it=_RecentForageSites.begin(); it!=_RecentForageSites.end(); )
{
CRecentForageSite& forageSite = (*it);
if ( forageSite.lowFreqUpdate() )
{
++it;
}
else
{
--TotalNbRecentForageSites;
it = _RecentForageSites.erase( it );
}
}
// Decrease kami anger level
decKamiAnger( ForageKamiAngerDecreasePerHour.get() / 36000.0f * ((float)DepositUpdateFrequency.get()) );
// Auto-spawn a source from time to time (if the deposit has this flag)
if ( _AutoSpawnSourcePt && _Enabled )
{
// For Speed Test
//TTime testYoyoLT0= CTime::getLocalTime();
//uint countBefore= _CurrentNbAutoSpawnedSources;
uint spawnPeriodLFUpdates = (AutoSpawnForageSourcePeriodOverride.get() != 0) ? AutoSpawnForageSourcePeriodOverride.get() / DepositUpdateFrequency.get() : _AutoSpawnSourcePt->SpawnPeriodGc / DepositUpdateFrequency.get();
// get deposit bbox
CVector cornerMin, cornerMax;
getAABox( cornerMin, cornerMax );
// Previously, the rythm of spawning was randomized. Now it's constant.
//TGameCycle spawnAvgPeriod = (AutoSpawnForageSourceAveragePeriodOverride.get() != 0) ? AutoSpawnForageSourceAveragePeriodOverride.get() : _AutoSpawnSourcePt->SpawnAveragePeriodGc;
//sint32 r = RandomGenerator.rand( (uint16)(spawnAvgPeriod / DepositUpdateFrequency.get()) );
//if ( r == 0 )
bool spawnSourceBecauseOfFrequency= (CTickEventHandler::getGameCycle() / DepositUpdateFrequency.get()) % spawnPeriodLFUpdates == 0;
// Avoid infinite loop if the spawn of source is too buggy:
// count first the number of sources to force spawn then run this count, and forget the ones that fail.
uint numSourceToForceSpawn= 0;
if(_CurrentNbAutoSpawnedSources < _AutoSpawnSourcePt->MinimumSpawnedSources)
numSourceToForceSpawn= _AutoSpawnSourcePt->MinimumSpawnedSources - _CurrentNbAutoSpawnedSources;
uint numSourceSpawned= 0;
while ( spawnSourceBecauseOfFrequency || numSourceSpawned%d): %d ms", CTickEventHandler::getGameCycle()%1000, countBefore, _CurrentNbAutoSpawnedSources, testYoyoLT1 - testYoyoLT0);
}
// Beware: can return before
}
/*
* Update deposit that need to auto spawn because number of harvest sources is too low
*/
void CDeposit::autoSpawnUpdate()
{
if ( !HarvestSystemEnabled )
return;
// Auto-spawn a source from time to time (if the deposit has this flag)
if ( _AutoSpawnSourcePt && _Enabled && _CurrentNbAutoSpawnedSources < _AutoSpawnSourcePt->MinimumSpawnedSources)
{
// For Speed Test
//TTime testYoyoLT0= CTime::getLocalTime();
//uint countBefore= _CurrentNbAutoSpawnedSources;
// get deposit bbox
CVector cornerMin, cornerMax;
getAABox( cornerMin, cornerMax );
// Avoid infinite loop if the spawn of source is too buggy:
// count first the number of sources to force spawn then run this count, and forget the ones that fail.
uint numSourceToForceSpawn= _AutoSpawnSourcePt->MinimumSpawnedSources - _CurrentNbAutoSpawnedSources;
// for all sources to spawn
for( uint i= 0; i%d): %d ms", CTickEventHandler::getGameCycle()%1000, countBefore, _CurrentNbAutoSpawnedSources, testYoyoLT1 - testYoyoLT0);
}
// Beware: can return before
}
/*
* Auto Spawn Deposit mgt
*/
void CDeposit::decreaseAutoSpawnedSources()
{
BOMB_IF(_CurrentNbAutoSpawnedSources==0, "nb auto spawned sources should be > 0!", return);
_CurrentNbAutoSpawnedSources--;
// If the deposit has not enough ressources, must refill at next tick update
// NB: don't do it at next lowFreqUpdate(), to avoid big jump in CPU each 30 sec
// as soon an autospawned harvest source is unspawned, the deposit will refill at next tick.
if(_AutoSpawnSourcePt && _CurrentNbAutoSpawnedSources < _AutoSpawnSourcePt->MinimumSpawnedSources)
{
CZoneManager::getInstance().registerDepositToAutoSpawnUpdate(this);
}
}
void CDeposit::increaseAutoSpawnedSources()
{
_CurrentNbAutoSpawnedSources++;
}
//-----------------------------------------------------------------------------
// character take MP, kami eat this fool character ?
//-----------------------------------------------------------------------------
#if 0
//void CDeposit::characterTakeRM( const CEntityId& charId, uint32 depositIndexContent, uint16 quantity )
//{
// // check if depositIndexContent is in valid range
// if( depositIndexContent < _DepositContent.size() )
// {
// // process kami guardian reaction and fame impact
// CDepositRawMaterialSeasonParameters * seasonParameters;
// switch( CTimeDateSeasonManager::getRyzomTimeReference().getRyzomSeason() )
// {
// case EGSPD::CSeason::Spring:
// seasonParameters = &_DepositContent[ depositIndexContent ]->SpringParams;
// break;
// case EGSPD::CSeason::Summer:
// seasonParameters = &_DepositContent[ depositIndexContent ]->SummerParams;
// break;
// case EGSPD::CSeason::Autumn:
// seasonParameters = &_DepositContent[ depositIndexContent ]->AutumnParams;
// break;
// case EGSPD::CSeason::Winter:
// seasonParameters = &_DepositContent[ depositIndexContent ]->WinterParams;
// break;
// default:
// nlstop; //=> debug CTimeDateSeasonManager....
// }
//
// if( seasonParameters->AngryLevel >= _DepositContent[ depositIndexContent ]->CurrentQuantity )
// {
// sint32 fameImpact;
// uint32 kamiImpact;
//
// CMessage msgString("STATIC_STRING");
// msgString.serial( const_cast< CEntityId& > ( charId ) );
// // serial exclude set of empty exclude set
// set< CEntityId > empty;
// msgString.serialCont( empty );
//
// if( seasonParameters->BlackKamiLevel >= _DepositContent[ depositIndexContent ]->CurrentQuantity )
// {
// // kami impact for AIS when Kami expulse a black kami
// kamiImpact = 300;
//
// // set fame impact
// fameImpact = -15;
//
// string str("WOS_KAMI_BLACK_KAMI");
// msgString.serial(str);
// }
// else if( seasonParameters->FuryLevel >= _DepositContent[ depositIndexContent ]->CurrentQuantity )
// {
// // kami impact for AIS when Kami is fury against harvester
// kamiImpact = 200;
//
// // set fame impact
// fameImpact = -10;
//
// string str("WOS_KAMI_FURY");
// msgString.serial( str );
// }
// else if( seasonParameters->AngryLevel >= _DepositContent[ depositIndexContent ]->CurrentQuantity )
// {
// // kami impact for AIS when Kami is angry against harvester
// kamiImpact = 100;
//
// // set fame impact
// fameImpact = -5;
//
// string str("WOS_KAMI_ANGRY");
// msgString.serial( str );
// }
// sendMessageViaMirror( "IOS", msgString );
//
// // Send message to AIS for kami impact
// CMessage msgout("KAMI_IMPACT");
// // TODO : if this code reactivated, add the instance number and send to the corect AIS
// msgout.serial( const_cast (charId) );
// msgout.serial( _Alias );
// msgout.serial( kamiImpact );
// sendMessageViaMirror( "AIS", msgout );
//
//
// // send message to EGS for fame impact
// // TODO boris : update fame with new fame system
///* CCharacter * c = PlayerManager.getChar( charId );
// uint8 fameType = FAMES::kami;
// if( c )
// {
// c->fameChange( fameType, fameImpact );
// }
//*/ }
//
// // update deposit quantity
// if( _DepositContent[ depositIndexContent ]->CurrentQuantity < quantity )
// {
// _DepositContent[ depositIndexContent ]->CurrentQuantity = 0;
// }
// else
// {
// _DepositContent[ depositIndexContent ]->CurrentQuantity -= quantity;
// }
// }
// else
// {
// nlwarning("FG: Received invalid depositIndexContent %d (deposit content size id %d)", depositIndexContent, _DepositContent.size() );
// }
//}
#endif
string CDeposit::getSeasonStr() const
{
string s;
if ( _SeasonFilter.empty() )
return "All year";
for ( uint i=0; i!=_SeasonFilter.size(); ++i )
{
if ( ! s.empty() )
s += " ";
s += EGSPD::CSeason::toString( _SeasonFilter[i] );
}
return s;
}
string CDeposit::getTimeOfDayStr() const
{
string s;
if ( _TimeOfDayFilter.empty() )
return "Night & day";
for ( uint i=0; i!=_TimeOfDayFilter.size(); ++i )
{
if ( ! s.empty() )
s += " ";
s += toString( "%u", (uint)_TimeOfDayFilter[i] );
}
return s;
}
string CDeposit::getWeatherStr() const
{
string s;
if ( _WeatherFilter.empty() )
return "All weathers";
for ( uint i=0; i!=_WeatherFilter.size(); ++i )
{
if ( ! s.empty() )
s += " ";
s += toString( "%u", (uint)_WeatherFilter[i] );
}
return s;
}
//-----------------------------------------------------------------------------
// display deposit content
//-----------------------------------------------------------------------------
void CDeposit::displayContent( NLMISC::CLog * log, bool extendedInfo, NLMISC::CWordsDictionary *itemDictionary )
{
const CAllStaticItems& allItems = CSheets::getItemMapForm();
uint32 materialNumber = 0;
log->displayRawNL( "---- DEPOSIT %s ----", _Name.c_str() );
log->displayRawNL( "\t Centre: %s", getBarycentre().toString().c_str() );
if ( getContents().empty() )
log->displayRawNL( "\tNo RM was selected in this deposit" );
log->displayRawNL( "\t%u RMs:", getContents().size() );
for( std::vector< CStaticDepositRawMaterial >::const_iterator it = getContents().begin(); it != getContents().end(); ++it )
{
if ( extendedInfo )
log->displayRawNL( "\tRM #%d", materialNumber );
++materialNumber;
CSString sheetCode = (*it).MaterialSheet.toString();
string fullNameInfo;
if ( itemDictionary )
{
CVectorSString result;
sheetCode = sheetCode.splitTo( '.' );
itemDictionary->lookup( sheetCode, result );
if ( ! result.empty() )
fullNameInfo = string(" ") + result[0];
}
if ( fullNameInfo.empty() )
log->displayRawNL( "\t\tRM Id: %s", sheetCode.c_str() );
else
log->displayRawNL( "\t\tRM Id: %s", fullNameInfo.c_str() );
//log->displayNL( "\t\tMax Amount: %d", (*it).MaxAmount );
if ( extendedInfo )
{
CAllStaticItems::const_iterator itItem = allItems.find( (*it).MaterialSheet );
if ( itItem != allItems.end() && (*itItem).second.Mp )
{
const CStaticItem& staticItem = (*itItem).second;
log->displayRawNL( "\t\tFamily: %u", staticItem.Mp->Family );
log->displayRawNL( "\t\tGroup: %u", staticItem.Mp->getGroup() );
log->displayRawNL( "\t\tEcosystem: %s", ECOSYSTEM::toString( staticItem.Mp->Ecosystem ).c_str() );
//log->displayRawNL( "\t\tRarity: %hu", staticItem.Mp->Rarity );
log->displayRawNL( "\t\tStatEnergy: %hu", staticItem.Mp->StatEnergy );
}
}
}
log->displayRawNL( "\tEcotype: %s", ECOSYSTEM::toString( _Ecotype ).c_str() );
string ws;
if ( _WeatherFilter.empty() )
ws = "all";
else
{
for( vector::const_iterator iwf=_WeatherFilter.begin(); iwf!=_WeatherFilter.end(); ++iwf )
ws += toString( "%u:", *iwf );
}
log->displayRawNL( "\tWeathers: %s", ws.c_str() );
string ts;
if ( _TimeOfDayFilter.empty() )
ts = "all";
else
{
for( vector::const_iterator itf=_TimeOfDayFilter.begin(); itf!=_TimeOfDayFilter.end(); ++itf )
ts += toString( "%u:", *itf );
}
log->displayRawNL( "\tTimes of day: %s", ts.c_str() );
string ss;
if ( _SeasonFilter.empty() )
ss = "all";
else
{
for( vector::const_iterator isf=_SeasonFilter.begin(); isf!=_SeasonFilter.end(); ++isf )
ws += toString( "%u:", *isf );
}
log->displayRawNL( "\tSeasons: %s", ss.c_str() );
log->displayRaw( "\tKami anger: %.1f", _KamiAnger );
if ( _KamiAnger == -1.0f )
log->displayRawNL( " disabled" );
else
log->displayRawNL( " on %.1f, %.1f", ForageKamiAngerThreshold1.get(), ForageKamiAngerThreshold2.get() );
if ( _AutoSpawnSourcePt )
log->displayRawNL( "\tAutoSpawnSource: %s", _AutoSpawnSourcePt?"ON":"OFF" );
if ( ! _CanProspect )
log->displayRawNL( "\tCanProspect: %d", _CanProspect );
if ( ! _Enabled )
log->displayRawNL( "\tEnabled: %d", _Enabled );
if ( ! _AllowDepletionRisk )
log->displayRawNL( "\tAllowDepletionRisk: %d", _AllowDepletionRisk );
}
float GenRand = 1.0f;
float GenFreq = 0.05f;
NLMISC_COMMAND( setDepositRand, "Set CNoiseValue params for deposits", " " )
{
if ( args.size() < 2 )
return false;
GenRand = (float)atof( args[0].c_str() );
GenFreq = (float)atof( args[1].c_str() );
return true;
}
/*
*
*/
class CIndexNoiseValue : public CNoiseValue
{
public:
///
CIndexNoiseValue( uint arraySize, uint phase, float freq ) : CNoiseValue(0.0f, 1.0f, freq), _MaxIndex((float)(arraySize)), _Phase(1210.191f*phase, 523.8883f*phase, 403.57614f*phase) {}
/// Evaluate the value corresponding to the pos
uint eval( const CVector& pos ) const
{
uint indexOfMax = 0;
float maxNoise = 0;
// 1 dephased noise per possible value, and take the max result (otherwise with '(CNoiseValue::eval( pos ) * _MaxIndex)', the result would be biased)
for( uint i=0; i!=_MaxIndex; ++i )
{
float f = CNoiseValue::eval(
pos + _Phase + CVector(5245.346f*i, 785.67985f*i, 7842.783367f*i) );
if ( f > maxNoise )
{
maxNoise = f;
indexOfMax = i;
}
}
return indexOfMax;
}
private:
float _MaxIndex;
CVector _Phase;
};
/*
* Compute freq according to deposit size (small deposits under 64x64 have a proportionally higher frequency)
*/
inline float getFreqFromDepositArea( float depositBBoxArea )
{
if ( depositBBoxArea < SmallDepositAreaThreshold )
{
const float gradient = (0.05f-0.3f)*2.0f/(SmallDepositAreaThreshold-SmallDepositAreaReference);
return gradient*depositBBoxArea + (GenFreq - gradient*(SmallDepositAreaThreshold));
}
else
{
return GenFreq;
}
}
/*
* Get a random RM from the neighbourhood of the specified position.
* OptFastFloorBegin()/OptFastFloorEnd() must enclose one or more calls to this method.
*/
const CStaticDepositRawMaterial *CDeposit::getRandomRMAtPos( const NLMISC::CVector& pos, bool testIfSiteDepleted, bool& isDepleted )
{
const float cellWidth = 1.0f;
const uint16 nbNeighbours = 5; // center + 4 neighbour cells
//nldebug( "Selecting RM in deposit" );
if ( getContents().empty() )
return NULL;
// Test is local zone is depleted
if ( testIfSiteDepleted )
{
for ( CRecentForageSites::iterator ihs=_RecentForageSites.begin(); ihs!=_RecentForageSites.end(); ++ihs )
{
if ( (*ihs).isDepleted() && (*ihs).contains( pos ) )
{
//nldebug( "Harvest site depleted" );
isDepleted = true;
return NULL;
}
}
}
// Compute freq according to deposit size
float freq = getFreqFromDepositArea( getAreaOfAABox() );
// Get raw materials in neighbourhood
uint neighbourhoud [nbNeighbours];
CVector cellPos = pos;
CIndexNoiseValue noiseIndex( getContentSize(), _FilterPhase, freq ); // the phrase prevents to have the same map for two deposits with identical size but different seasons, etc.
neighbourhoud[0] = noiseIndex.eval( cellPos );
cellPos.x -= cellWidth;
cellPos.y -= cellWidth;
neighbourhoud[1] = noiseIndex.eval( cellPos );
cellPos.x = pos.x + cellWidth;
neighbourhoud[2] = noiseIndex.eval( cellPos );
cellPos.y = pos.y + cellWidth;
neighbourhoud[3] = noiseIndex.eval( cellPos );
cellPos.x = pos.x - cellWidth;
neighbourhoud[4] = noiseIndex.eval( cellPos );
//nldebug( "Indices: %u %u %u %u", neighbourhoud[0], neighbourhoud[1], neighbourhoud[2], neighbourhoud[3] );
// Select a random RM among neighbourhood
uint rmIndex = neighbourhoud[RandomGenerator.rand( nbNeighbours-1 )];
return &(getContents()[rmIndex]);
/*
if( seasonParameters->AngryLevel >= _DepositContent[ infos.DepositIndexContent ]->CurrentQuantity )
{
CMessage msgout("STATIC_STRING");
msgout.serial( const_cast< CEntityId& > ( charId ) );
// serial exclude set of empty exclude set
set< CEntityId > empty;
msgout.serialCont( empty );
string str("EGS_KAMI_ALERTE");
msgout.serial( str );
sendMessageViaMirror( "IOS", msgout );
}
*/
}
/*
* Get the RM at the specified position.
* OptFastFloorBegin()/OptFastFloorEnd() must enclose one or more calls to this method.
* \param testIfSiteDepleted Set it to false for map generation.
*/
const CStaticDepositRawMaterial *CDeposit::getRMAtPos( const NLMISC::CVector& pos, bool testIfSiteDepleted, bool& isDepleted )
{
//const float cellWidth = 5.0f;
//nldebug( "Selecting RM in deposit" );
if ( getContents().empty() )
return NULL;
// Test is local zone is depleted
if ( testIfSiteDepleted )
{
for ( CRecentForageSites::iterator ihs=_RecentForageSites.begin(); ihs!=_RecentForageSites.end(); ++ihs )
{
if ( (*ihs).isDepleted() && (*ihs).contains( pos ) )
{
//nldebug( "Harvest site depleted" );
isDepleted = true;
return NULL;
}
}
}
// Compute freq according to deposit size
float freq = getFreqFromDepositArea( getAreaOfAABox() );
// Get raw material at position
CIndexNoiseValue noiseIndex( getContentSize(), _FilterPhase, freq );
uint rmIndex = noiseIndex.eval( pos );
//nldebug( "Index: %u", rmIndex );
if ( rmIndex >= getContentSize() )
{
nlwarning( "FG: CIndexNoiseValue returned a value out of range" );
rmIndex = getContentSize()-1;
}
return &(getContents()[rmIndex]);
}
/*
* Return the current quantity. If 0 and the respawn time is elapsed, unlock.
*/
float CQuantityConstraints::getCurrentQuantity()
{
if ( CurrentQuantity == 0 )
{
// Time to unlock?
const CRyzomTime& ryzomTime = CTimeDateSeasonManager::getRyzomTimeReference();
if ( ryzomTime.getRyzomDay() < NextRespawnDay )
{
// Still locked
return 0;
}
else
{
// Unlock and retry
CurrentQuantity = InitialQuantity;
}
}
return CurrentQuantity;
}
/*
* Consume
*
* Last possible consumption => lock deposit for RespawnTimeRyzomDays.
* Does not unlock the deposit (see getCurrentQuantity() instead)
* Argument by value.
*/
float CQuantityConstraints::consumeQuantity( float consumed )
{
if ( CurrentQuantity > 0 )
{
float newQuantity = CurrentQuantity - consumed;
if ( newQuantity <= 0 )
{
// Lock if the deposit is now empty (and wasn't before)
const CRyzomTime& ryzomTime = CTimeDateSeasonManager::getRyzomTimeReference();
NextRespawnDay = ryzomTime.getRyzomDay() + (uint32)RespawnTimeRyzomDays;
// Give only the remaining quantity
consumed = CurrentQuantity;
newQuantity = 0;
}
// Consume
CurrentQuantity = newQuantity;
return consumed;
}
else
return 0;
}
/*
* Display forage sites info
*/
void CDeposit::displayRecentForageSites( NLMISC::CLog& log ) const
{
if ( ! _RecentForageSites.empty() )
{
log.displayNL( "Deposit %s:", _Name.c_str() );
for ( CRecentForageSites::const_iterator it=_RecentForageSites.begin(); it!=_RecentForageSites.end(); ++it )
{
const CRecentForageSite& forageSite = (*it);
forageSite.display( log );
}
}
}
/*
* Empty all forage sites
*/
void CDeposit::depleteAllRecentForageSites()
{
for ( CRecentForageSites::iterator it=_RecentForageSites.begin(); it!=_RecentForageSites.end(); ++it )
{
CRecentForageSite& forageSite = (*it);
forageSite.depleteAll();
}
}
/*
* Increment the kami anger level (delta < 768). React if a threshold is reached. Return the kami anger level.
*/
float CDeposit::incKamiAnger( float delta, const std::vector& foragers )
{
if ( _KamiAnger == -1.0f )
return -1.0f;
if ( ForageKamiAngerOverride.get() != 0 )
_KamiAnger = ForageKamiAngerOverride.get();
else
_KamiAnger += delta;
if ( _KamiAnger > ForageKamiAngerThreshold1.get() )
{
if ( _KamiAnger < ForageKamiAngerThreshold2.get() )
{
// Only a warning (and small punishment ?)
for ( vector::const_iterator itf=foragers.begin(); itf!=foragers.end(); ++itf )
{
// Send a warning message
const TDataSetRow& rowId = (*itf);
TDataSetRow noRow;
npcTellToPlayer( noRow, rowId, "FORAGE_KAMI_ANGER_WARNING", false );
}
}
else
{
// Full punishment
if( HarvestAreaEffectOn )
{
for ( vector::const_iterator itf=foragers.begin(); itf!=foragers.end(); ++itf )
{
const TDataSetRow& rowId = (*itf);
CEntityBase *entity = CEntityBaseManager::getEntityBasePtr( rowId );
if ( entity )
{
// Send a warning message
TDataSetRow noRow;
npcTellToPlayer( noRow, rowId, "FORAGE_KAMI_ANGER_PUNISH", false );
// Set target list for FX (must be done before behaviour)
CMirrorPropValueList targets( TheDataset, rowId, DSPropertyTARGET_LIST );
targets.clear();
targets.push_front( 0 ); // null distance (self)
uint32 urowId = *((uint32*)&rowId);
targets.push_front( urowId );
// Set behaviour for FX (must be done after target list)
MBEHAV::CBehaviour behav( MBEHAV::CAST_OFF_SUCCESS );
behav.Spell.SpellMode = MAGICFX::Bomb;
behav.Spell.SpellId = MAGICFX::Piercing;// + ForageKamiAngerPunishFX.get();
behav.Spell.SpellIntensity = 5;
behav.Spell.Resist = 0;
behav.Spell2.SelfSpell = 1; // 'self offensive cast' does not plays the cast anim, only impact & FX
PHRASE_UTILITIES::sendUpdateBehaviour( rowId, behav );
// tmp nico : stats about projectiles
projStatsIncrement();
// Damage
CHarvestSource::hitEntity( RYZOMID::creature, entity, ForageKamiAngerPunishDamage.get(), ForageKamiAngerPunishDamage.get(), false );
//Bsi.append( StatPath, NLMISC::toString("[FKWP] %s '%s' %.1f", entity->getId().toString().c_str(), name().c_str(), _KamiAnger) );
//EgsStat.displayNL("[FKWP] %s '%s' %.1f", entity->getId().toString().c_str(), name().c_str(), _KamiAnger);
// EGSPD::forageKamiWrathPunishment(entity->getId(), name(), _KamiAnger);
}
}
}
// Don't reset the kami anger level to the min value, but nearly to the threshold 2 value!
_KamiAnger = ForageKamiAngerThreshold2.get() - 1.0f; // +2 will reach the threshold 2 again
}
}
return _KamiAnger;
}
//NLMISC_VARIABLE( bool, VerboseDeposits, "Verbose info for deposits" );
NLMISC_COMMAND( forageDisplayRecentForageSitesNb, "Display the number of recent forage sites", "" )
{
log.displayNL( "%u forage sites", TotalNbRecentForageSites );
return true;
}
NLMISC_COMMAND( forageDisplayRecentForageSitesInfo, "Display the info on all recent forage sites", "" )
{
const std::vector& deposits = CZoneManager::getInstance().getDeposits();
std::vector::const_iterator it;
for ( it=deposits.begin(); it!=deposits.end(); ++it )
{
CDeposit *deposit = (*it);
deposit->displayRecentForageSites( log );
}
return true;
}
NLMISC_COMMAND( forageEmptyAllRecentForageSites, "Empty all forage sites", "" )
{
const std::vector& deposits = CZoneManager::getInstance().getDeposits();
std::vector::const_iterator it;
for ( it=deposits.begin(); it!=deposits.end(); ++it )
{
CDeposit *deposit = (*it);
deposit->depleteAllRecentForageSites();
}
return true;
}
NLMISC_COMMAND( forageDisplayDeposit, "Display info about one or all the deposits", " " )
{
string depName = "ALL";
bool extendedInfo = false;
if ( args.size() > 0 )
{
depName = args[0];
if ( args.size() > 1 )
extendedInfo = (args[1] == "1");
}
CZoneManager::getInstance().dumpDeposits( log, depName, extendedInfo );
return true;
}
struct TCompareDepositsByHighestKamiAnger : public std::binary_function
{
bool operator() ( const CDeposit* d1, const CDeposit* d2 )
{
return d1->kamiAnger() > d2->kamiAnger();
}
};
NLMISC_COMMAND( forageDisplayKamiAngerLevels, "Display the N deposits with the highest anger levels", "" )
{
uint nb = 10;
if ( args.size() > 0 )
NLMISC::fromString(args[0], nb);
std::vector newDepositList = CZoneManager::getInstance().getDeposits();
std::sort( newDepositList.begin(), newDepositList.end(), TCompareDepositsByHighestKamiAnger() );
log.displayNL( "%u first deposits by kami anger:", nb );
for ( uint i=0; i!=nb && ikamiAnger(), newDepositList[i]->name().c_str() );
}
return true;
}
NLMISC_VARIABLE( bool, VerboseDepositFiltering, "Verbose the items parsed when filtering deposits" );
NLMISC_VARIABLE( bool, ExportDepositContents, "When loading the deposit primitives, export contents report" );