mirror of
https://port.numenaute.org/aleajactaest/khanat-opennel-code.git
synced 2024-12-24 18:08:44 +00:00
55de2f0bd2
--HG-- branch : develop
2965 lines
76 KiB
C++
2965 lines
76 KiB
C++
// 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 NL_SRG_UTILITIES_H
|
|
#define NL_SRG_UTILITIES_H
|
|
|
|
|
|
// Misc
|
|
#include <nel/misc/types_nl.h>
|
|
#include <nel/misc/sstring.h>
|
|
#include "nel/misc/path.h"
|
|
#include "nel/misc/file.h"
|
|
#include "nel/misc/smart_ptr.h"
|
|
#include "nel/misc/command.h"
|
|
#include "nel/misc/common.h"
|
|
#include "nel/misc/path.h"
|
|
#include <nel/misc/diff_tool.h>
|
|
#include <nel/misc/random.h>
|
|
// Georges
|
|
#include "nel/georges/u_form.h"
|
|
#include "nel/georges/u_form_elm.h"
|
|
#include "nel/georges/u_form_dfn.h"
|
|
#include "nel/georges/u_form_loader.h"
|
|
#include "nel/georges/u_type.h"
|
|
// Georges, bypassing interface
|
|
#include "georges/stdgeorges.h"
|
|
#include "georges/form.h"
|
|
// C
|
|
#include <time.h>
|
|
#include <conio.h>
|
|
// stl
|
|
#include <set>
|
|
#include <map>
|
|
#include <hash_map>
|
|
|
|
using namespace NLMISC;
|
|
using namespace NLGEORGES;
|
|
using namespace std;
|
|
|
|
|
|
typedef sint TFaberInterestLevel;
|
|
const TFaberInterestLevel NAInterestLevel = -1;
|
|
const uint32 NbNomenclaturedFaberLevel = 5; // excluding N/A
|
|
const char *sNomenclaturedInterestLevels [NbNomenclaturedFaberLevel+1] = { "N/A", "Worst", "Bad", "Average", "Good", "Best" }; // from -1 to 5
|
|
CRandom RandomGenerator;
|
|
|
|
typedef CVectorSString vs;
|
|
|
|
typedef map <uint32, vector<uint32> > CRulesFilter;
|
|
typedef map <CSString, vector<uint32>, CUnsensitiveSStringLessPred > CRulesStrFilter;
|
|
typedef map <CSString, vs, CUnsensitiveSStringLessPred > CRulesStr2Filter;
|
|
typedef map< CSString, CSString, CUnsensitiveSStringLessPred > mss;
|
|
typedef vector<uint32> vu;
|
|
typedef hash_map< string, string, hash<string> > CTitles; // code -> utf8 title
|
|
|
|
// Write sheet files, or display to screen only
|
|
bool WriteSheetsToDisk = true;
|
|
|
|
// Overriden by command-line argument after -n
|
|
string ExtractNamesCsv;
|
|
|
|
|
|
const uint32 NbFaberElements = 26;
|
|
|
|
const uint32 RM_INDEX_FAMILY_CODE = 1;
|
|
const uint32 NB_FAMILY_CODE_CHARS = 4;
|
|
|
|
const uint32 RM_INDEX_CREATURE_CODE = 5;
|
|
const uint32 NB_CREATURE_CODE_CHARS = 5;
|
|
|
|
const uint32 RM_INDEX_ECOSYSTEM_CODE = 8;
|
|
const uint32 NB_ECOSYSTEM_CODE_CHARS = 1;
|
|
|
|
const uint32 RM_INDEX_LEVELZONE_CODE = 9;
|
|
|
|
const uint32 SIZE_RAW_MATERIAL_SHEET_FILENAME = 18;
|
|
|
|
/*const uint32 RM_INDEX_INTEREST_LEVEL = 6;
|
|
const uint32 NB_INTEREST_LEVEL_CHARS = 2;
|
|
|
|
const uint32 RM_INDEX_FABERELEMS_CODE = 8;
|
|
const uint32 NB_FABERELEMS_CODE_CHARS = 6;*/
|
|
|
|
const uint32 CR_INDEX_ECOSYSTEM_CODE = 3; // in creature code
|
|
const uint32 CR_INDEX_LEVELZONE_CODE = 4;
|
|
const uint32 CR_INDEX_LOCAL_LEVEL_CODE = 5;
|
|
|
|
//const uint32 NB_UNIQUE_LEVELS_PER_ZONE = 4; // not counting the bosses, the fifth equals the first one of next
|
|
const uint32 NB_UNIQUE_LEVELZONES_PER_CONTINENT = 5;
|
|
const uint32 MAX_NB_LOCAL_LEVELS = 8;
|
|
//const uint32 NB_ZONE_LEVELS = 5; // TEMP for forest ecosystem
|
|
//const uint32 NbFaberInterestLevels = (NB_ZONE_LEVELS * NB_UNIQUE_LEVELS_PER_ZONE) + 1; // excluding N/A
|
|
|
|
string inputSheetPath;
|
|
bool inputSheetPathLoaded = false;
|
|
map<string, string> inputSheetPathContent; // short filename without ext, full filename with path
|
|
string TranslationPath;
|
|
string SystemMPPath;
|
|
|
|
|
|
// These vectors have the same indices : by family
|
|
vs families;
|
|
vs familyCodes, ecosystemCodes, propertyCodes, creatureCodes;
|
|
|
|
sint MaxFamilyNum = -1;
|
|
|
|
enum TColor { Red, Beige, Green, Turquoise, Blue, Violet, White, Black, NbColors, InvalidColor=NbColors };
|
|
|
|
// By ecosystem, color, property, rm group, creature, season
|
|
vs ecosystems, colors, properties, groups, creatures, seasons;
|
|
mss adjectives;
|
|
|
|
// DtName must be the 1st one
|
|
enum TDataCol { DtName, DtTitle, DtRMFamily, DtGroup, DtEcosystem, DtLevelZone, DtStatQuality, DtProp, DtCreature, DtCreaTitle, DtCraftSlotName, DtCraftCivSpec, DtColor, DtAverageEnergy, DtMaxLevel, DtJewelProtectionType, DtCustomizedProperties, DtNbCols };
|
|
const char *DataColStr [DtNbCols] = { "Code", "Name", "Family", "Group", "Ecosystem", "LevelZone", "Stat Quality", "Properties", "Creature sheets", "Creatures", "Item parts", "Craft civ spec", "Color", "Average energy", "Max level", "Jewel protection type", "Customized" };
|
|
|
|
const uint32 NbPropertyDepths = 5;
|
|
const char *PropertyDepths [NbPropertyDepths] = { "Unknown", "Slightly", "Moderately", "Quite", "Extremely" };
|
|
|
|
typedef uint32 TGroup; // index in groups, and value in groups .typ
|
|
|
|
//enum TLocation { Under, Over, Flora, Creatures, NB_LOCATIONS, NB_DEPOSITS=Flora+1 };
|
|
enum TLocation { InDeposits, InCreatures, NB_LOCATIONS };
|
|
vu DepositFamilyIndices;
|
|
|
|
enum TCiv { Fyros, Matis, Tryker, Zorai, AllCiv, NbCiv };
|
|
const char *CivNames [NbCiv] = { "Fyros", "Matis", "Tryker", "Zorai", "All" };
|
|
const char *CivEcosystemCodes [NbCiv] = { "D", "F", "L", "J", "X" };
|
|
|
|
enum TEcosystem { CommonEcosystem, Desert, Forest, Lacustre, Jungle, PrimeRoots, NbEcosystems, Goo=NbEcosystems, Invasion, Raid, Event, N, S, T, U, V, W, X, Y, Z, NbEcosystemsPlusExtensions };
|
|
|
|
TCiv EcosystemToCiv[NbEcosystems] = { AllCiv, Fyros, Matis, Tryker, Zorai, AllCiv };
|
|
|
|
enum TStatQuality { Basic, Fine, Choice, Excellent, Supreme, NbStatQualities, InvalidStatQuality=NbStatQualities };
|
|
char *StatQualityStr [NbStatQualities+1] = { "Basic", "Fine", "Choice", "Excellent", "Supreme", "N/A" };
|
|
|
|
TStatQuality CreatureLocalLevelToStatQuality [MAX_NB_LOCAL_LEVELS] =
|
|
{
|
|
Basic, // 1 (index 0)
|
|
Fine, // 2
|
|
Basic, // 3
|
|
Fine, // 4
|
|
Excellent, // 5 (named creatures)
|
|
InvalidStatQuality, // 6 (mission creatures, their RMs have no craft stats)
|
|
Supreme, // 7 (bosses)
|
|
Choice // 8 (mini-bosses)
|
|
};
|
|
|
|
|
|
enum TIndexOfRemarkableStatIndex { RBestA, RBest=RBestA, RWorstA1, RWorst1=RWorstA1, RWorstA2, RWorst2=RWorstA2, RBestB, RWorstB1, RWorstB2, NB_REMARKABLE_STAT_INDICES };
|
|
|
|
struct TSkeletonInfo
|
|
{
|
|
TSkeletonInfo() : IsUsed(false) {}
|
|
|
|
//vs CreaturesOfSke;
|
|
CSString Name;
|
|
CSString AbbrevName;
|
|
//CSString SkGroup;
|
|
bool IsUsed; // true if has RM
|
|
};
|
|
|
|
typedef map<CSString, TSkeletonInfo, CUnsensitiveSStringLessPred> CSkeletonMap;
|
|
|
|
// By skeleton group, by skeleton, bu creature model
|
|
CRulesStr2Filter SkgroupToModels;
|
|
CSkeletonMap CreatureModels;
|
|
mss CreatureToModel;
|
|
//vs skeletonGroupColumns;
|
|
set<CSString, CUnsensitiveSStringLessPred> CreatureMainModelsWithoutRM;
|
|
|
|
// By creature model
|
|
CRulesStrFilter RMFamilyIndicesByCreatureModel;
|
|
vu GooCreatureFamilyIndices;
|
|
|
|
typedef map<char, vu> CCreatureTypeToFamilyIndices; // the key is the 2nd char of the creature code (h for herbivore, k for kitin...)
|
|
CCreatureTypeToFamilyIndices InvasionRaidCreatureFamilyIndices;
|
|
|
|
struct TBool
|
|
{
|
|
TBool() : Done(false) {}
|
|
bool Done;
|
|
};
|
|
|
|
typedef map< uint, TBool > CDoneMap; // indexed by levelzone
|
|
map< string, TBool > IsRMSheetGenerated; // indexed by sheetname
|
|
|
|
string rawMaterialPath, creaturePath, creatureAssignmentPath, depositPath;
|
|
string dirbase;
|
|
|
|
|
|
const string oldRmSheetType = "item";
|
|
const string rmSheetType = "sitem";
|
|
const string crSheetType = "creature";
|
|
const string dpSheetType = "deposit";
|
|
|
|
|
|
//const uint32 NbPropertySlots = 10; // obsolete
|
|
uint32 UndefinedProperty = ~0;
|
|
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
struct CIconInfo
|
|
{
|
|
CSString IconBackground, Icon, IconOver, IconOver2;
|
|
};
|
|
|
|
map< CSString, CIconInfo, CUnsensitiveSStringLessPred > Icons;
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
class CMainStat
|
|
{
|
|
public:
|
|
|
|
///
|
|
CMainStat() : SumNbFaberElemsFilled(0), MaxNbFaberElemsFilled(0), MinNbFaberElemsFilled(~0),
|
|
NbRMByFaberElem( NbFaberElements, 0 ) {}
|
|
|
|
/// Call it when families etc. are ready
|
|
void init()
|
|
{
|
|
NbRMByFaberElemByFamilyAndCiv.resize( families.size() );
|
|
for ( uint32 i=0; i!=families.size(); ++i )
|
|
{
|
|
NbRMByFaberElemByFamilyAndCiv[i].resize( NbCiv );
|
|
for ( uint32 c=0; c!=NbCiv; ++c )
|
|
NbRMByFaberElemByFamilyAndCiv[i][c].resize( NbFaberElements );
|
|
}
|
|
|
|
for ( uint32 iEcosystem=0; iEcosystem!=ecosystems.size(); ++iEcosystem )
|
|
{
|
|
for ( uint32 iLoc=0; iLoc!=NB_LOCATIONS; ++iLoc )
|
|
{
|
|
NbRMHavingProperty[iLoc][iEcosystem].resize( properties.size() );
|
|
NbRMByFaberElemByEcosystem[iEcosystem].resize( NbFaberElements );
|
|
}
|
|
}
|
|
}
|
|
|
|
///
|
|
bool updateCraftStatistics( uint32 rFaberElem, uint32 iEcosystem, uint32 iFam, TCiv civ );
|
|
|
|
///
|
|
void updateMainStats( uint32 nbFaberElemsFilled )
|
|
{
|
|
SumNbFaberElemsFilled += nbFaberElemsFilled;
|
|
if ( nbFaberElemsFilled < MinNbFaberElemsFilled )
|
|
MinNbFaberElemsFilled = nbFaberElemsFilled;
|
|
if ( nbFaberElemsFilled> MaxNbFaberElemsFilled )
|
|
MaxNbFaberElemsFilled = nbFaberElemsFilled;
|
|
}
|
|
|
|
// By property and by ecosystem and location
|
|
vu NbRMHavingProperty [NB_LOCATIONS][NbEcosystems];
|
|
|
|
uint32 SumNbFaberElemsFilled;
|
|
uint32 MaxNbFaberElemsFilled;
|
|
uint32 MinNbFaberElemsFilled;
|
|
|
|
vector<uint32> NbRMByFaberElem;;
|
|
vector<uint32> NbRMByFaberElemByEcosystem [NbEcosystems];
|
|
vector< vector < vector< uint32 > > > NbRMByFaberElemByFamilyAndCiv;
|
|
|
|
};
|
|
|
|
|
|
/*
|
|
* From georges2csv
|
|
*/
|
|
void loadSheetPath()
|
|
{
|
|
if (inputSheetPathLoaded)
|
|
return;
|
|
|
|
NLMISC::createDebug();
|
|
NLMISC::WarningLog->addNegativeFilter( "CPath::insertFileInMap" );
|
|
|
|
CPath::addSearchPath(inputSheetPath, true, false); // for Georges to work properly
|
|
|
|
vector<string> files;
|
|
CPath::getPathContent (inputSheetPath, true, false, true, files);
|
|
|
|
uint32 i;
|
|
for (i=0; i<files.size(); ++i)
|
|
{
|
|
string filename = files[i];
|
|
string filebase = CFile::getFilenameWithoutExtension(filename);
|
|
inputSheetPathContent[filebase] = filename;
|
|
}
|
|
|
|
inputSheetPathLoaded = true;
|
|
}
|
|
|
|
|
|
/*
|
|
* Get a random value in range [0..nbPossibleValues[
|
|
* Precondition: nbPossibleValues < 65536
|
|
*/
|
|
inline uint32 getRandomValue( uint32 nbPossibleValues )
|
|
{
|
|
/*double r = (double) rand();
|
|
r/= (double) (RAND_MAX+1); // exclude the ceiling value
|
|
return (uint32)(r * nbPossibleValues);*/
|
|
return RandomGenerator.rand( (uint16)(nbPossibleValues-1) );
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
struct CDfnFieldInfo
|
|
{
|
|
CDfnFieldInfo() {}
|
|
CDfnFieldInfo( const vector<string>& values, const vector<string>& labels ) : TypePredefinedValues(values), TypePredefinedLabels(labels) {}
|
|
|
|
vector<string> TypePredefinedValues;
|
|
vector<string> TypePredefinedLabels;
|
|
};
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
struct TIconMapping
|
|
{
|
|
const char *FamilyName;
|
|
const char *IconFilename;
|
|
};
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
sint getNomenclaturedInterestLevel( TFaberInterestLevel level, TFaberInterestLevel nbInterestLevels )
|
|
{
|
|
return (level == NAInterestLevel) ? 0 : (level * NbNomenclaturedFaberLevel / nbInterestLevels) + 1;
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
//struct CFaberCode
|
|
//{
|
|
// char Ch[NB_FABERELEMS_CODE_CHARS];
|
|
//};
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
//struct CFaberCombination
|
|
//{
|
|
// CFaberCombination( TFaberInterestLevel firstLevel, const string& code ) : FirstLevel(firstLevel)
|
|
// {
|
|
// memcpy( Code.Ch, &code[0], NB_FABERELEMS_CODE_CHARS );
|
|
// }
|
|
//
|
|
// TFaberInterestLevel FirstLevel;
|
|
// CFaberCode Code;
|
|
//};
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
//class CSheetNameRepository
|
|
//{
|
|
//public:
|
|
//
|
|
// ///
|
|
// void resize( uint32 nbEcosystems, uint32 nbFamilies )
|
|
// {
|
|
// _Container.resize( nbEcosystems );
|
|
// for ( uint32 i=0; i!=nbEcosystems; ++i )
|
|
// {
|
|
// _Container[i].resize( nbFamilies );
|
|
// }
|
|
// }
|
|
//
|
|
// ///
|
|
// void insert( uint32 iEcosystem, uint32 iFamily, uint32 iCreature, TFaberInterestLevel level, const string& faberCombinationCode )
|
|
// {
|
|
// // nlassert( faberCombinationCode.size() == NB_FABERELEMS_CODE_CHARS );
|
|
// _Container[iEcosystem][iFamily][iCreature].push_back( CFaberCombination( level, faberCombinationCode ) );
|
|
// }
|
|
//
|
|
// ///
|
|
// void getFaberCombinationCodes( uint32 iEcosystem, uint32 iFamily, uint32 iCreature, vector<CFaberCombination> **codes )
|
|
// {
|
|
// map < uint32, vector< CFaberCombination > >::iterator im;
|
|
// /*nldebug( "%u %u -> %u cr (searching for %u)", iEcosystem, iFamily, _Container[iEcosystem][iFamily].size(), iCreature );
|
|
// for ( im=_Container[iEcosystem][iFamily].begin(); im!=_Container[iEcosystem][iFamily].end(); ++im )
|
|
// nldebug( "cr %u -- %u combinations", (*im).first, (*im).second.size() );*/
|
|
// im = _Container[iEcosystem][iFamily].find( iCreature );
|
|
// if ( im == _Container[iEcosystem][iFamily].end() )
|
|
// *codes = NULL;
|
|
// else
|
|
// *codes = &((im)->second);
|
|
// }
|
|
//
|
|
//private:
|
|
//
|
|
// /// Indexs: iEcosystem, iFamily, rCreatureSpecialization, rFaberCombination
|
|
// vector< vector < map < uint32, vector< CFaberCombination > > > > _Container;
|
|
//};
|
|
|
|
|
|
//void CSheetNameRepository::getFaberCombinationCodes( uint32 iEcosystem, uint32 iFamily, uint32 iCreature, vector<CFaberCombination> **codes )
|
|
|
|
|
|
//CSheetNameRepository RawMaterialRepository;
|
|
|
|
/**
|
|
* Characteristics.
|
|
* When adding a new characteristic, ADD IT INTO "v3_source_tables.xls!Item Parts v3"/"rm_item_parts.csv" and mark compatible item parts
|
|
*/
|
|
enum TFaberCharacteristic {
|
|
Durability, Weight, SapLoad, DMG, Speed, Range,
|
|
DodgeModifier, ParryModifier, AdversaryDodgeModifier, AdversaryParryModifier,
|
|
ProtectionFactor, MaxSlashProtect, MaxBluntProtect, MaxPierceProtect,
|
|
ECTF, EPF,
|
|
OACTF, OAPF,
|
|
HCTP, HPF,
|
|
DACTF, DAPF,
|
|
JewelProtection,
|
|
CraftCivSpec,
|
|
NbCharacs };
|
|
|
|
const char *sCharacs [NbCharacs] = {
|
|
"Durability", "Weight", "SapLoad", "DMG", "Speed", "Range",
|
|
"DodgeModifier", "ParryModifier", "AdversaryDodgeModifier", "AdversaryParryModifier",
|
|
"ProtectionFactor", "MaxSlashingProtection", "MaxBluntProtection", "MaxPiercingProtection",
|
|
"ElementalCastingTimeFactor", "ElementalPowerFactor",
|
|
"OffensiveAfflictionCastingTimeFactor", "OffensiveAfflictionPowerFactor",
|
|
"HealCastingTimeFactor", "HealPowerFactor",
|
|
"DefensiveAfflictionCastingTimeFactor", "DefensiveAfflictionPowerFactor",
|
|
"JewelProtectionType",
|
|
"CraftCivSpec" };
|
|
|
|
//const bool PositiveCharacs [NbCharacs] = { true, false, true, false, true, true };
|
|
//const float MinCharacValues [NbCharacs] = { 100, 0.1f, 10, 0.2f, 100, 0 };
|
|
//const float MaxCharacValues [NbCharacs] = { 500, 1.5f, 400, 2.0f, 500, 60 };
|
|
//const float PeakCharacValues [NbCharacs] = { 2000, 2.5f, 800, 5.0f, 2000, 80 };
|
|
bool PositiveCharacs [NbCharacs];
|
|
float MinCharacValues [NbCharacs];
|
|
float MaxCharacValues [NbCharacs];
|
|
float PeakCharacValues [NbCharacs];
|
|
vector<bool> CharacSlotFilter [NbCharacs]; // it's a positive filter
|
|
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
struct TFamInfo
|
|
{
|
|
vs Properties; // ex: propForA, propForB, propForC
|
|
CSString CompatibleCraftParts; // ex: ABC
|
|
vu CraftPartsByProp; // ex: 0, 1, 2 (indices in ompatibleCraftParts)
|
|
vector<TCiv> Civs; // ex: Fyros, All, All
|
|
vu Freqs; // ex: 1, 2, 2
|
|
TGroup Group;
|
|
bool IsActive; // False if not in rm_fam_prop.csv
|
|
bool IsInDeposits;
|
|
bool IsInCreatures;
|
|
CSString SpecialCreatureTag;
|
|
bool GenerateOnly;
|
|
bool IsForMission;
|
|
sint8 RemarkableStatIndex [NB_REMARKABLE_STAT_INDICES];
|
|
sint8 ColorIndex;
|
|
sint8 JewelProtIndex;
|
|
|
|
static uint UseGenerateOnly; // if 0, generate all; otherwise, generate only families that have GenerateOnly set to true
|
|
|
|
///
|
|
TFamInfo() : Group(~0), IsInDeposits(false), IsActive(false), IsInCreatures(false), SpecialCreatureTag(), GenerateOnly(false), IsForMission(false), ColorIndex(-1), JewelProtIndex(-1) {}
|
|
|
|
///
|
|
uint getCompatibleCraftPart( uint iCompatibleCP ) const
|
|
{
|
|
return (uint)(CompatibleCraftParts[iCompatibleCP] - 'A');
|
|
}
|
|
|
|
/// whichProp: index in Properties
|
|
uint getBeginCraftPartForProp( uint whichProp ) const
|
|
{
|
|
return CraftPartsByProp[whichProp];
|
|
}
|
|
|
|
/// whichProp: index in Properties
|
|
uint getEndCraftPartForProp( uint whichProp ) const
|
|
{
|
|
if ( whichProp == Properties.size()-1 )
|
|
return CompatibleCraftParts.size();
|
|
else
|
|
return CraftPartsByProp[whichProp+1];
|
|
}
|
|
|
|
/// whichProp: index in Properties
|
|
CSString getCraftPartForProp( uint whichProp ) const
|
|
{
|
|
uint start = getBeginCraftPartForProp( whichProp );
|
|
return CompatibleCraftParts.substr( start, getEndCraftPartForProp( whichProp ) - start );
|
|
}
|
|
|
|
/** With the returned index, you can get elt in Property, CraftPartsByProp, Civs and Freqs.
|
|
* Returns ~0 if not found.
|
|
*/
|
|
uint getPropIndexByCraftPart( char itemPart ) const
|
|
{
|
|
for ( uint i=0; i!=CraftPartsByProp.size(); ++i )
|
|
{
|
|
char itemPartCode [2];
|
|
itemPartCode[0] = itemPart;
|
|
itemPartCode[1] = '\0';
|
|
if ( getCraftPartForProp( i ).find( itemPartCode ) != string::npos )
|
|
return i;
|
|
}
|
|
return ~0;
|
|
}
|
|
|
|
///
|
|
bool existsInEcosystem( TEcosystem iEcosystem, TStatQuality statQuality ) const;
|
|
|
|
///
|
|
static bool mustGenerateFamily( uint iFamily );
|
|
};
|
|
|
|
|
|
const uint ITEM_PART_JEWEL_GEM = 17; // R
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
struct CFaberCharacteristics
|
|
{
|
|
/// Default constructor (for reading)
|
|
CFaberCharacteristics()
|
|
: FaberElement(~0), ActualEnergy(0.0f), ActualOriginality(0.0f)
|
|
{
|
|
for ( uint32 i=0; i!=NbCharacs; ++i )
|
|
Values[i] = 0.0f;
|
|
}
|
|
|
|
///
|
|
void serial( NLMISC::IStream& s )
|
|
{
|
|
s.serial( FaberElement );
|
|
s.serial( ActualEnergy );
|
|
s.serial( ActualOriginality );
|
|
for ( uint32 c=0; c!=NbCharacs; ++c )
|
|
s.serial( Values[c] );
|
|
}
|
|
|
|
///
|
|
void initFaberElement( uint32 rFaberElement ) { FaberElement = rFaberElement; }
|
|
|
|
/// Returns false if the RM must NOT be generated.
|
|
bool randomizeValues( TFaberInterestLevel interestLevel, TFaberInterestLevel nbInterestLevels, uint iVariant, float widthRatio, float peakOccurRatio, float baseBoost, TEcosystem iEcosystem, uint iFreq, uint iFamily );
|
|
|
|
/// Returns -1 if the RM must NOT be generated, otherwise return the stat average
|
|
sint32 computeValues( TStatQuality statQuality, const TFamInfo& famInfo, sint remarkableIndicesSetBaseIndex,
|
|
TEcosystem iEcosystem, uint iFreq, uint iFamily );
|
|
|
|
///
|
|
void calcQualitativeValues();
|
|
|
|
/// Index of faber element in "MpParam"
|
|
uint32 FaberElement;
|
|
|
|
/// Values
|
|
float Values [NbCharacs];
|
|
|
|
/// Average of the actual interest of the random values between 0 and 1
|
|
float ActualEnergy;
|
|
|
|
///
|
|
float ActualOriginality;
|
|
|
|
protected:
|
|
|
|
// Returns false if the RM must NOT be generated.
|
|
//bool randomizeJewelProtection( TStatQuality statQuality, TEcosystem iEcosystem, uint iFreq, uint iFamily );
|
|
|
|
void computeJewelProtection( TStatQuality statQuality, TEcosystem iEcosystem, uint iFamily );
|
|
};
|
|
|
|
|
|
//
|
|
void CFaberCharacteristics::calcQualitativeValues()
|
|
{
|
|
float actualInterests [NbCharacs];
|
|
|
|
// Calculate the average of ratio between [0,1]
|
|
float sumActualInterest = 0.0f;
|
|
uint nbCharacsUsed = 0;
|
|
for ( uint r=0; r!=NbCharacs; ++r )
|
|
{
|
|
if ( MaxCharacValues[r] == 0 )
|
|
continue;
|
|
if ( ! CharacSlotFilter[r][FaberElement] )
|
|
continue;
|
|
|
|
float interestRatio = (Values[r] / PeakCharacValues[r]); // new: Peak is taken as the max => the Energy is in [0..100]
|
|
if ( ! PositiveCharacs[r] )
|
|
interestRatio = 1.0f - interestRatio; // thus, can be negative
|
|
|
|
actualInterests[r] = interestRatio;
|
|
sumActualInterest += interestRatio;
|
|
++nbCharacsUsed;
|
|
}
|
|
|
|
if ( nbCharacsUsed == 0 )
|
|
return;
|
|
|
|
ActualEnergy = (sumActualInterest / (float)nbCharacsUsed);
|
|
if ( ActualEnergy > 1.0f )
|
|
ActualEnergy = 1.0f;
|
|
|
|
// Calculate the standard deviation (SQRT(SUM((Ai-Aavg)^2)/N))
|
|
float varianceSum = 0.0f;
|
|
for ( uint r=0; r!=NbCharacs; ++r )
|
|
{
|
|
if ( MaxCharacValues[r] == 0 )
|
|
continue;
|
|
if ( ! CharacSlotFilter[r][FaberElement] )
|
|
continue;
|
|
|
|
varianceSum += sqr( actualInterests[r] - ActualEnergy );
|
|
}
|
|
|
|
// Don't normalize standard deviation, otherwise low energy materials will be considered more
|
|
// original (by average) than high energy materials. They wouldn't be comparable.
|
|
|
|
//if ( ActualEnergy != 0.0f )
|
|
ActualOriginality = (float)sqrt( (double)(varianceSum / (float)nbCharacsUsed) ); // / ActualEnergy;
|
|
//else
|
|
// nlinfo( "Null energy for craft slot %u", FaberElement );
|
|
}
|
|
|
|
|
|
/*
|
|
* Fill childrenToGet if rootNameForGetChildren is not null
|
|
*/
|
|
void fillFromDFN( UFormLoader *formLoader, map<string, CDfnFieldInfo>& dfnFields, UFormDfn *formDfn, const string& rootName, const string& dfnFilename,
|
|
const char *rootNameForGetChildren=NULL, vector<string>& childrenToGet=vector<string>() )
|
|
{
|
|
uint32 i;
|
|
for ( i=0; i!=formDfn->getNumEntry(); ++i )
|
|
{
|
|
string entryName, rootBase;
|
|
formDfn->getEntryName( i, entryName );
|
|
rootBase = rootName.empty() ? "" : (rootName+".");
|
|
|
|
UFormDfn::TEntryType entryType;
|
|
bool array;
|
|
formDfn->getEntryType( i, entryType, array );
|
|
switch ( entryType )
|
|
{
|
|
case UFormDfn::EntryVirtualDfn:
|
|
{
|
|
CSmartPtr<UFormDfn> subFormDfn = formLoader->loadFormDfn( (entryName + ".dfn").c_str() );
|
|
if ( ! subFormDfn )
|
|
nlwarning( "Can't load virtual DFN %s", entryName.c_str() );
|
|
else
|
|
fillFromDFN( formLoader, dfnFields, subFormDfn, rootBase + entryName, entryName + ".dfn", rootNameForGetChildren, childrenToGet ); // recurse
|
|
break;
|
|
}
|
|
case UFormDfn::EntryDfn: // .dfn
|
|
{
|
|
UFormDfn *subFormDfn;
|
|
if ( formDfn->getEntryDfn( i, &subFormDfn) )
|
|
{
|
|
string filename;
|
|
formDfn->getEntryFilename( i, filename );
|
|
fillFromDFN( formLoader, dfnFields, subFormDfn, rootBase + entryName, filename, rootNameForGetChildren, childrenToGet ); // recurse
|
|
}
|
|
if ( rootNameForGetChildren && (rootName == rootNameForGetChildren) )
|
|
{
|
|
childrenToGet.push_back( rootBase + entryName );
|
|
}
|
|
break;
|
|
}
|
|
case UFormDfn::EntryType: // .typ
|
|
{
|
|
vector<string> values, labels;
|
|
UType *subType;
|
|
if ( formDfn->getEntryType( i, &subType ) )
|
|
{
|
|
uint32 listSize = subType->getNumDefinition();
|
|
if ( listSize > 0 )
|
|
{
|
|
string label, value;
|
|
for ( uint32 j=0; j!=listSize; ++j )
|
|
{
|
|
subType->getDefinition( j, label, value );
|
|
if ( (subType->getIncrement() == "1") && (subType->getType() == UType::UnsignedInt || subType->getType() == UType::SignedInt) )
|
|
{
|
|
// Fill blank entry for skipped identifier values (to allow indexing by identifier value)
|
|
sint num = atoi( value.c_str() );
|
|
while ( num - (sint)values.size() > 0 )
|
|
{
|
|
values.push_back( "" );
|
|
labels.push_back( "" );
|
|
}
|
|
}
|
|
values.push_back( value );
|
|
labels.push_back( label );
|
|
}
|
|
}
|
|
}
|
|
dfnFields.insert( make_pair( rootBase + entryName, CDfnFieldInfo(values, labels) ) );
|
|
//nlinfo( "DFN entry: %s (in %s)", (rootBase + entryName).c_str(), dfnFilename.c_str() );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
CForm *loadTemplateForm( UFormLoader *formLoader, const string& sheetType )
|
|
{
|
|
CForm *form = (CForm*)formLoader->loadForm( (string("_empty.")+sheetType).c_str() );
|
|
if ( ! form )
|
|
nlerror( "Can't load sheet _empty.%s", sheetType.c_str() );
|
|
return form;
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void eraseCarriageReturns( string& s )
|
|
{
|
|
const char CR = '\n';
|
|
string::size_type p = s.find( CR );
|
|
while ( (p=s.find( CR )) != string::npos )
|
|
s.erase( p, 1 );
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
string getNomenclatureCode( const string& longName, set<string>& usedCodes, uint32 nbLetters )
|
|
{
|
|
if ( nbLetters > longName.size() )
|
|
nlerror( "Wrong nbLetters for %s", longName.c_str() );
|
|
|
|
// Start with beginning of name
|
|
string code = strlwr(longName.substr( 0, nbLetters ));
|
|
uint32 i = nbLetters-1;
|
|
while ( usedCodes.find( code ) != usedCodes.end() )
|
|
{
|
|
++i;
|
|
if ( i < longName.size() )
|
|
{
|
|
// Substitute last code char by a char from the name (except ' ')
|
|
if ( longName[i] != ' ' )
|
|
code[nbLetters-1] = tolower(longName[i]);
|
|
else
|
|
continue;
|
|
}
|
|
else
|
|
{
|
|
// If no char from the name is suitable, increment the last char of the code until suitable
|
|
char c=1;
|
|
while ( usedCodes.find( code ) != usedCodes.end() )
|
|
{
|
|
code[nbLetters-1] = tolower(longName[nbLetters-1]) + c;
|
|
++c;
|
|
if ( code[1] > 'z' )
|
|
nlerror( "Impossible to make code for %s", longName.c_str() );
|
|
}
|
|
}
|
|
}
|
|
strlwr( code );
|
|
usedCodes.insert( code );
|
|
return code;
|
|
}
|
|
|
|
|
|
/*
|
|
* Displays mapping if title not null.
|
|
*/
|
|
void buildNomenclatureCodes( const char *title, const vector<string>& longNames, vector<string>& codes, uint32 nbLetters )
|
|
{
|
|
set<string> usedCodeSet;
|
|
uint32 i;
|
|
for ( i=0; i!=longNames.size(); ++i )
|
|
{
|
|
codes[i] = getNomenclatureCode( longNames[i], usedCodeSet, nbLetters );
|
|
if ( title )
|
|
nlinfo( "%s %s -> %s", title, longNames[i].c_str(), codes[i].c_str() );
|
|
//DebugLog->displayRawNL( "%s", longNames[i].c_str() );
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Set the size of a family code to NB_FAMILY_CODE_CHARS
|
|
*/
|
|
void normalizeFamilyCode( std::string& s )
|
|
{
|
|
if ( s.size() > NB_FAMILY_CODE_CHARS )
|
|
{
|
|
nlerror( "Family codes limited to %u chars (%s)", NB_FAMILY_CODE_CHARS, s.c_str() );
|
|
}
|
|
else
|
|
{
|
|
uint p = s.size();
|
|
while ( p < NB_FAMILY_CODE_CHARS )
|
|
{
|
|
s = "0" + s;
|
|
++p;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void loadNomenclatureCodes( const char *title, const vector<string>& longNames, vector<string>& codes, const char *filename )
|
|
{
|
|
if ( longNames.empty() )
|
|
{
|
|
nlwarning( "No nomenclature codes to load. %s", title ? title : "" );
|
|
return;
|
|
}
|
|
codes.resize( longNames.size() );
|
|
|
|
char lineBuffer[2048];
|
|
FILE *rulesFile;
|
|
const char *SEPARATOR = ";";
|
|
vector<string> args;
|
|
vector<string>::iterator iarg;
|
|
vector<string>::const_iterator ivs;
|
|
|
|
if ( (rulesFile = NLMISC::nlfopen( filename, "r" )) == NULL )
|
|
{
|
|
nlwarning( "Can't find file %s", filename );
|
|
}
|
|
else
|
|
{
|
|
while ( ! feof(rulesFile) )
|
|
{
|
|
// Get from file
|
|
fgets( lineBuffer, 2048, rulesFile );
|
|
explode( lineBuffer, SEPARATOR, args );
|
|
|
|
// Get rid of carriage returns!
|
|
for ( iarg=args.begin(); iarg!=args.end(); ++iarg )
|
|
{
|
|
eraseCarriageReturns( *iarg );
|
|
}
|
|
|
|
// Read
|
|
const uint32 MIN_COLS = 6;
|
|
const uint32 NAME_COL = 4;
|
|
const uint32 C_COL = 2;
|
|
const uint32 R_COL = 3;
|
|
const uint32 NB_CODE_CHARS = 2;
|
|
if ( (args.size()>=MIN_COLS) && (! args[0].empty()) && (args[0].find( "name" )==string::npos) ) // skip blank lines, and lines with blank header or "name" in the first column
|
|
{
|
|
if ( args[NAME_COL].empty() )
|
|
continue;
|
|
|
|
ivs = find( longNames.begin(), longNames.end(), args[NAME_COL] );
|
|
if ( ivs == longNames.end() )
|
|
nlwarning( "Name %s is not in the names array", args[NAME_COL].c_str() );
|
|
else
|
|
{
|
|
string code = args[C_COL] + args[R_COL];
|
|
if ( code.size() < NB_CODE_CHARS )
|
|
{
|
|
nlwarning( "Invalid partial code for %s: %s", (*ivs).c_str(), code.c_str() );
|
|
continue;
|
|
}
|
|
else if ( code.size() > NB_CODE_CHARS )
|
|
{
|
|
nlinfo( "Compacting code '%s' for %s", code.c_str(), (*ivs).c_str() );
|
|
string::size_type p;
|
|
while ( (p = code.find( ' ' )) != string::npos )
|
|
{
|
|
code.erase( p, 1 );
|
|
}
|
|
}
|
|
|
|
if ( codes[ivs-longNames.begin()].empty() )
|
|
{
|
|
if ( title )
|
|
nlinfo( "%s %s -> %s", title, (*ivs).c_str(), code.c_str() );
|
|
codes[ivs-longNames.begin()] = code;
|
|
}
|
|
else
|
|
{
|
|
if ( code != codes[ivs-longNames.begin()] )
|
|
nlwarning( "Invalid nomenclature: (%s and %s for %s: ", codes[ivs-longNames.begin()].c_str(), code.c_str(), (*ivs).c_str() );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for ( ivs=codes.begin(); ivs!=codes.end(); ++ivs )
|
|
{
|
|
if ( (*ivs).empty() )
|
|
nlwarning( "No code found for %s", (*(longNames.begin() + (ivs - codes.begin()))).c_str() );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
inline sint getLastUsedPropertySlot( uint32 *iProperties, sint lastPropertySlot, uint32 undefinedProperty )
|
|
{
|
|
for ( sint r=lastPropertySlot; r>=0; --r )
|
|
{
|
|
if ( iProperties[r] != undefinedProperty )
|
|
return r;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
inline bool passNegativeFilter( const vector<uint32>& incompatibilityList, uint32 iValue )
|
|
{
|
|
vector<uint32>::const_iterator ip = find( incompatibilityList.begin(), incompatibilityList.end(), iValue );
|
|
return (ip == incompatibilityList.end());
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
inline bool passPositiveFilter( const vector<uint32>& compatibilityList, uint32 iValue )
|
|
{
|
|
vector<uint32>::const_iterator ip = find( compatibilityList.begin(), compatibilityList.end(), iValue );
|
|
return (ip != compatibilityList.end());
|
|
}
|
|
|
|
|
|
/*
|
|
* Reject a prop if it is in the incompatibility list of a family
|
|
*/
|
|
bool passPropFamilyFilter( const vector<uint32>& iFamilyRelatedProperties, uint32 iProp )
|
|
{
|
|
return passNegativeFilter( iFamilyRelatedProperties, iProp );
|
|
}
|
|
|
|
|
|
/*
|
|
* Reject a creature if NOT in the creature list of a family
|
|
*/
|
|
/*bool passCreatureFilter( const vector<uint32>& iFamilyRelatedCreatures, uint32 iCreature )
|
|
{
|
|
//nldebug( "%u related creatures, %s", iFamilyRelatedCreatures.size(), passPositiveFilter( iFamilyRelatedCreatures, iCreature ) ? "TRUE": "FALSE" );
|
|
return passPositiveFilter( iFamilyRelatedCreatures, iCreature );
|
|
}*/
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
class CStrIComparator : public binary_function<string, string, bool>
|
|
{
|
|
public:
|
|
bool operator() ( const string& s1, const string& s2 ) const
|
|
{
|
|
return (nlstricmp( s1, s2 ) == 0);
|
|
}
|
|
};
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void displayList( const vector<string>& v, CLog *log=DebugLog )
|
|
{
|
|
vector<string>::const_iterator ist;
|
|
for ( ist=v.begin(); ist!=v.end(); ++ist )
|
|
log->displayRaw( "%s ", (*ist).c_str() );
|
|
log->displayRawNL( "" );
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
uint32 getIndexFromString( const string& s, const vector<string>& v, bool displayWarning=true )
|
|
{
|
|
if ( v.empty() )
|
|
{
|
|
if ( displayWarning )
|
|
nlwarning( "Can't find '%s' in empty array", s.c_str() );
|
|
return ~0;
|
|
}
|
|
else
|
|
{
|
|
vector<string>::const_iterator ist = find_if( v.begin(), v.end(), bind2nd(CStrIComparator(), s) );
|
|
if ( ist == v.end() )
|
|
{
|
|
if ( displayWarning )
|
|
{
|
|
nlwarning( "Can't find '%s' in:", s.c_str() );
|
|
displayList( v, WarningLog );
|
|
}
|
|
return ~0;
|
|
}
|
|
else
|
|
return ist - v.begin();
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
uint32 getIndexFromString( const string& s, const char **array, uint arraySize, bool displayWarning=true )
|
|
{
|
|
if ( arraySize == 0 )
|
|
{
|
|
if ( displayWarning )
|
|
nlwarning( "Can't find '%s' in empty array", s.c_str() );
|
|
return ~0;
|
|
}
|
|
else
|
|
{
|
|
for ( uint i=0; i!=arraySize; ++i )
|
|
{
|
|
if ( strlwr(string(array[i])) == strlwr(s) )
|
|
return i;
|
|
}
|
|
|
|
if ( displayWarning )
|
|
{
|
|
nlwarning( "Can't find '%s' in:", s.c_str() );
|
|
//displayList( v, WarningLog );
|
|
}
|
|
return ~0;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Returns the index of the erased element, ~0 if not found
|
|
*/
|
|
uint32 removeEntryFromList( vector<string>& v, const string& entry )
|
|
{
|
|
vector<string>::iterator ivs;
|
|
ivs = find( v.begin(), v.end(), entry );
|
|
uint32 index;
|
|
if ( ivs != v.end() )
|
|
{
|
|
index = ivs - v.begin();
|
|
v.erase( ivs );
|
|
return index;
|
|
}
|
|
else
|
|
return ~0;
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
bool removeEntryFromListByIndex( vector<string>& v, uint32 index )
|
|
{
|
|
if ( index < v.size() )
|
|
{
|
|
v.erase( v.begin() + index );
|
|
return true;
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
|
|
typedef void (*TMapDeliveryCallback) ( mss& );
|
|
typedef void (*TVectorDeliveryCallback) ( vs& );
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void loadCSVFile( const char *filename, TMapDeliveryCallback deliveryCallback, bool firstColWithoutName=false )
|
|
{
|
|
char lineBuffer[2048];
|
|
FILE *file;
|
|
const char *SEPARATOR = ";";
|
|
vector<string> args;
|
|
vector<string>::iterator iarg;
|
|
|
|
if ( (file = NLMISC::nlfopen( filename, "r" )) == NULL )
|
|
{
|
|
nlwarning( "Can't find file %s", filename );
|
|
}
|
|
else
|
|
{
|
|
// Read first line as header with column names
|
|
lineBuffer[0] = '\0';
|
|
fgets( lineBuffer, 2048, file );
|
|
explode( lineBuffer, SEPARATOR, args );
|
|
|
|
// Store column names (and get rid of carriage returns!)
|
|
vector < string > columnNames;
|
|
mss valuesByName;
|
|
for ( iarg=args.begin(); iarg!=args.end(); ++iarg )
|
|
{
|
|
if ( firstColWithoutName && (iarg == args.begin()) )
|
|
{
|
|
*iarg = "<>"; // override column name for the 1st column
|
|
}
|
|
eraseCarriageReturns( *iarg );
|
|
columnNames.push_back( *iarg );
|
|
valuesByName.insert( make_pair( *iarg, string("") ) );
|
|
}
|
|
|
|
while ( ! feof(file) )
|
|
{
|
|
// Get from file
|
|
lineBuffer[0] = '\0';
|
|
fgets( lineBuffer, 2048, file );
|
|
explode( lineBuffer, SEPARATOR, args );
|
|
|
|
// Set values (and get rid of carriage returns!)
|
|
for ( iarg=args.begin(); iarg!=args.end(); ++iarg )
|
|
{
|
|
eraseCarriageReturns( *iarg );
|
|
valuesByName[columnNames[iarg-args.begin()]] = *iarg;
|
|
}
|
|
|
|
// Deliver the wanted fields
|
|
deliveryCallback( valuesByName );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void loadCSVFile( const char *filename, TVectorDeliveryCallback deliveryCallback )
|
|
{
|
|
char lineBuffer[2048];
|
|
FILE *file;
|
|
const char *SEPARATOR = ";";
|
|
vs args;
|
|
vs::iterator iarg;
|
|
|
|
if ( (file = NLMISC::nlfopen( filename, "r" )) == NULL )
|
|
{
|
|
nlwarning( "Can't find file %s", filename );
|
|
}
|
|
else
|
|
{
|
|
while ( ! feof(file) )
|
|
{
|
|
// Get from file
|
|
lineBuffer[0] = '\0';
|
|
fgets( lineBuffer, 2048, file );
|
|
explode( lineBuffer, SEPARATOR, args );
|
|
|
|
// Get rid of carriage returns!
|
|
for ( iarg=args.begin(); iarg!=args.end(); ++iarg )
|
|
{
|
|
eraseCarriageReturns( *iarg );
|
|
}
|
|
|
|
// Deliver the wanted fields
|
|
deliveryCallback( args );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void loadValueFile( const char *filename, const vector<string>& keyStrings,
|
|
vector<sint>& contents, sint defaultValue )
|
|
{
|
|
nlassert( keyStrings.size() == contents.size() );
|
|
char lineBuffer[2048];
|
|
FILE *rulesFile;
|
|
const char *SEPARATOR = ";";
|
|
vector<string> args;
|
|
vector<string>::iterator iarg;
|
|
|
|
if ( (rulesFile = NLMISC::nlfopen( filename, "r" )) == NULL )
|
|
{
|
|
nlwarning( "Can't find file %s", filename );
|
|
}
|
|
else
|
|
{
|
|
while ( ! feof(rulesFile) )
|
|
{
|
|
// Get from file
|
|
lineBuffer[0] = '\0';
|
|
fgets( lineBuffer, 2048, rulesFile );
|
|
explode( lineBuffer, SEPARATOR, args );
|
|
|
|
// Get rid of carriage returns!
|
|
for ( iarg=args.begin(); iarg!=args.end(); ++iarg )
|
|
{
|
|
eraseCarriageReturns( *iarg );
|
|
}
|
|
|
|
// Read
|
|
if ( (! args.empty()) && (! args[0].empty()) ) // skip blank lines, and lines with blank header
|
|
{
|
|
sint value = defaultValue;
|
|
for ( uint32 a=0; a!=args.size()-1; ++a )
|
|
{
|
|
if ( ! args[a+1].empty() ) // skip blank entries
|
|
value = atoi( args[a+1].c_str() );
|
|
}
|
|
uint32 index = getIndexFromString( args[0], keyStrings );
|
|
if ( index != ~0 )
|
|
{
|
|
contents[index] = value;
|
|
}
|
|
}
|
|
}
|
|
fclose( rulesFile );
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void loadRulesFile( const char *filename, const vector<string>& keyStrings,
|
|
const vector<string>& contentStrings, CRulesFilter& filter,
|
|
const string& matchExtKeyAtFirstColumn=string() )
|
|
{
|
|
char lineBuffer[2048];
|
|
FILE *rulesFile;
|
|
const char *SEPARATOR = ";";
|
|
uint32 firstColumn = matchExtKeyAtFirstColumn.empty() ? 0 : 1;
|
|
vector<string> args;
|
|
vector<string>::iterator iarg;
|
|
|
|
if ( (rulesFile = NLMISC::nlfopen( filename, "r" )) == NULL )
|
|
{
|
|
nlwarning( "Can't find file %s", filename );
|
|
}
|
|
else
|
|
{
|
|
while ( ! feof(rulesFile) )
|
|
{
|
|
// Get from file
|
|
lineBuffer[0] = '\0';
|
|
fgets( lineBuffer, 2048, rulesFile );
|
|
explode( lineBuffer, SEPARATOR, args );
|
|
|
|
// Get rid of carriage returns!
|
|
for ( iarg=args.begin(); iarg!=args.end(); ++iarg )
|
|
{
|
|
eraseCarriageReturns( *iarg );
|
|
}
|
|
|
|
// Match with ext key string if set
|
|
if ( (! matchExtKeyAtFirstColumn.empty()) && (args[0]!=matchExtKeyAtFirstColumn) )
|
|
continue;
|
|
|
|
// Read
|
|
if ( (! args.empty()) && (! args[firstColumn].empty()) ) // skip blank lines, and lines with blank header
|
|
{
|
|
vector<uint32> contents;
|
|
for ( uint32 a=firstColumn; a!=args.size()-1; ++a )
|
|
{
|
|
if ( ! args[a+1].empty() ) // skip blank entries
|
|
contents.push_back( getIndexFromString( args[a+1], contentStrings ) );
|
|
}
|
|
filter.insert( make_pair( getIndexFromString( args[firstColumn], keyStrings ), contents ) );
|
|
}
|
|
}
|
|
fclose( rulesFile );
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* 1st column: extKeyStrings (corresponding to the filters 'vector'); 2nd: keyStrings
|
|
*/
|
|
void loadRulesFileMulti( const char *filename, const vector<string>& extKeyStrings, const vector<string>& keyStrings, const vector<string>& contentStrings, vector<CRulesFilter>& filters )
|
|
{
|
|
filters.resize( extKeyStrings.size() );
|
|
for ( uint32 i=0; i!=filters.size(); ++i )
|
|
{
|
|
loadRulesFile( filename, keyStrings, contentStrings, filters[i], extKeyStrings[i] );
|
|
/*CRulesFilter::const_iterator irf;
|
|
nldebug( "%s", extKeyStrings[i].c_str() );
|
|
for ( irf=filters[i].begin(); irf!=filters[i].end(); ++irf )
|
|
{
|
|
nldebug( "%s", keyStrings[(*irf).first].c_str() );
|
|
vector<uint32>::const_iterator ivi;
|
|
for ( ivi=(*irf).second.begin(); ivi!=(*irf).second.end(); ++ivi )
|
|
{
|
|
nldebug( "%u", *ivi );
|
|
}
|
|
}*/
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Clear the form to reuse it (and all contents below node)
|
|
*/
|
|
void clearSheet( CForm *form, UFormElm* node )
|
|
{
|
|
((CFormElm*)node)->clean();
|
|
form->clean();
|
|
}
|
|
|
|
|
|
/*
|
|
* Saves to disk if bool WriteSheetsToDisk is true
|
|
*/
|
|
void flushSheetToDisk( const string& fullFilename, UForm *form )
|
|
{
|
|
if ( WriteSheetsToDisk )
|
|
{
|
|
COFile output( fullFilename );
|
|
form->write( output, false );
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
string::size_type findCapital( const string& s, string::size_type startPos )
|
|
{
|
|
string::size_type p;
|
|
for ( p=startPos; p!=s.size(); ++p )
|
|
{
|
|
if ( (s[p] >= 'A') && (s[p] <= 'Z') )
|
|
return p;
|
|
}
|
|
return string::npos;
|
|
}
|
|
|
|
|
|
/*
|
|
* Transform "MyString " into "My string"
|
|
*/
|
|
void detachValue( string& s )
|
|
{
|
|
if ( s.size() < 2 )
|
|
return;
|
|
|
|
string::size_type p;
|
|
while ( (p = findCapital( s, 1 )) != string::npos )
|
|
{
|
|
s.insert( p, " " );
|
|
s[p+1] = tolower( s[p+1] );
|
|
}
|
|
|
|
// Rip off any blank at the end
|
|
if ( s[s.size()-1] == ' ' )
|
|
{
|
|
s.resize( s.size()-1 );
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void getTransposedMap( CRulesFilter& dest, const CRulesFilter& src )
|
|
{
|
|
CRulesFilter::const_iterator im;
|
|
for ( im=src.begin(); im!=src.end(); ++im )
|
|
{
|
|
vector<uint32>::const_iterator iv;
|
|
for ( iv=(*im).second.begin(); iv!=(*im).second.end(); ++iv )
|
|
{
|
|
dest[*iv].push_back( (*im).first );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
string makeFaberElementCode( uint32 iFaberElement, TFaberInterestLevel level, TFaberInterestLevel nbInterestLevels )
|
|
{
|
|
return toString( "%c%d", 'a' + iFaberElement, getNomenclaturedInterestLevel( level, nbInterestLevels ) );
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
inline bool hasMatchingFaberLevel( TFaberInterestLevel storedLevel, TFaberInterestLevel submittedLevel )
|
|
{
|
|
return storedLevel <= submittedLevel;
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
//void keepOnlyHighestLevel( vector<CFaberCombination*>& codes )
|
|
//{
|
|
// nlassert( ! codes.empty() );
|
|
// sint maxLevel = -1;
|
|
// uint32 i;
|
|
// for ( i=0; i!=codes.size(); ++i )
|
|
// {
|
|
// if ( codes[i]->FirstLevel > maxLevel )
|
|
// {
|
|
// maxLevel = codes[i]->FirstLevel;
|
|
// }
|
|
// }
|
|
// vector<CFaberCombination*> remainingCodes;
|
|
// for ( i=0; i!=codes.size(); ++i )
|
|
// {
|
|
// if ( codes[i]->FirstLevel == maxLevel )
|
|
// remainingCodes.push_back( codes[i] );
|
|
// }
|
|
//
|
|
// //nldebug( "%u codes, highest level = %u with %u occurences", codes.size(), maxLevel, remainingCodes.size() );
|
|
// //nlassert( remainingCodes.size() <= codes.size() );
|
|
// codes = remainingCodes;
|
|
// nlassert( ! codes.empty() );
|
|
//}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
bool allIncludedIn( const vu& subset, const vu& bigset )
|
|
{
|
|
vu::const_iterator iv;
|
|
for ( iv=subset.begin(); iv!=subset.end(); ++iv )
|
|
{
|
|
if ( find( bigset.begin(), bigset.end(), *iv ) == bigset.end() )
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void loadConfigFlag( CConfigFile& configFile, const char *varTitle, bool &flag )
|
|
{
|
|
CConfigFile::CVar *var = configFile.getVarPtr( varTitle );
|
|
if ( var )
|
|
flag = (var->asInt() == 1);
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
string::size_type getCapitalFromPos( const string& s, string::size_type startPos )
|
|
{
|
|
//nldebug( "%s %u", s.c_str(), startPos );
|
|
string::size_type p;
|
|
for ( p=startPos; p<s.size(); ++p )
|
|
{
|
|
if ( (s[p] >= 'A') && (s[p] <= 'Z') )
|
|
return p;
|
|
}
|
|
return string::npos;
|
|
}
|
|
|
|
|
|
/*
|
|
* Also used to make system_mp filenames.
|
|
* Converts "My Identifier" or "MyIdentifier" to "my_identifier" ("My identifier" to "Myidentifier")
|
|
*/
|
|
string conventionalDirectory( const string& dirname )
|
|
{
|
|
if ( dirname.empty() )
|
|
return "";
|
|
|
|
string result = dirname;
|
|
|
|
// Remove blanks
|
|
string::size_type p = 0;
|
|
while ( (p = result.find( ' ' )) != string::npos )
|
|
{
|
|
result.erase( p, 1 );
|
|
}
|
|
|
|
// Convert capitals to underscores
|
|
result[0] = tolower( result[0] );
|
|
p = 1;
|
|
while ( (p = getCapitalFromPos( result, p )) != string::npos )
|
|
{
|
|
result.insert( p, "_" );
|
|
++p;
|
|
result[p] = tolower( result[p] );
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
mss UniqueRMNamesAndSheetCodeHead;
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void readRMNames( mss& values )
|
|
{
|
|
string& name = values["basics.name"];
|
|
if ( ! name.empty() )
|
|
{
|
|
string radix = values["FILE"].substr( 0, 5 );
|
|
UniqueRMNamesAndSheetCodeHead.insert( make_pair( name, radix ) );
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void loadTitles( const string& sourceWords, const string& sourceBase, const string& languageCode, CTitles& dest )
|
|
{
|
|
STRING_MANAGER::TWorksheet worksheet;
|
|
STRING_MANAGER::loadExcelSheet( TranslationPath + sourceBase + "/" + sourceWords + "_words_" + languageCode + ".txt", worksheet );
|
|
uint cp, cn, nbTitles = 0;
|
|
if ( worksheet.findCol( ucstring(sourceWords + " ID"), cp ) && worksheet.findCol( ucstring("name"), cn ) )
|
|
{
|
|
for ( std::vector<STRING_MANAGER::TWorksheet::TRow>::iterator ip = worksheet.begin(); ip!=worksheet.end(); ++ip )
|
|
{
|
|
if ( ip == worksheet.begin() ) // skip first row
|
|
continue;
|
|
STRING_MANAGER::TWorksheet::TRow& row = *ip;
|
|
dest.insert( make_pair( row[cp].toString(), row[cn].toUtf8() ) );
|
|
++nbTitles;
|
|
}
|
|
}
|
|
else
|
|
nlwarning( "%s ID or name not found", sourceWords.c_str() );
|
|
|
|
nlinfo( "Loaded %u %s titles", nbTitles, sourceWords.c_str() );
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void extractRawMaterialNames()
|
|
{
|
|
loadCSVFile( ExtractNamesCsv.c_str(), readRMNames );
|
|
FILE *output = NLMISC::nlfopen( CFile::getFilenameWithoutExtension( ExtractNamesCsv ) + "_output.csv", "wt" );
|
|
fprintf( output, "Code;Name\n" );
|
|
for ( mss::const_iterator iun=UniqueRMNamesAndSheetCodeHead.begin(); iun!=UniqueRMNamesAndSheetCodeHead.end(); ++iun )
|
|
{
|
|
const string& codeRadix = (*iun).second;
|
|
const string& name = (*iun).first;
|
|
fprintf( output, "%s;%s\n", codeRadix.c_str(), name.c_str() );
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void cleanExteriorWhitespace( vs& line )
|
|
{
|
|
for ( vs::iterator it=line.begin(); it!=line.end(); ++it )
|
|
{
|
|
CSString& s = (*it);
|
|
string::size_type p;
|
|
for ( p=0; p!=s.size(); ++p )
|
|
{
|
|
if ( s[p] != ' ' )
|
|
break;
|
|
}
|
|
if ( (p != 0) && (p != s.size()) )
|
|
s = s.substr( p );
|
|
|
|
for ( p=0; p!=s.size(); ++p )
|
|
{
|
|
if ( s[s.size()-1-p] != ' ' )
|
|
break;
|
|
}
|
|
if ( (p != 0) && (p != s.size()) )
|
|
s = s.rightCrop( p );
|
|
}
|
|
}
|
|
|
|
|
|
uint TFamInfo::UseGenerateOnly = 0;
|
|
|
|
// Only used for deposits; for creature, works with the creature sheets found
|
|
/*bool TFamInfo::existsInEcosystem( TEcosystem iEcosystem, TStatQuality statQuality ) const
|
|
{
|
|
switch ( iEcosystem )
|
|
{
|
|
case CommonEcosystem: // The Common rm exists if the rm family has a freq=2 (or 0 but only in Supreme)
|
|
return ( (find( Freqs.begin(), Freqs.end(), 2 ) != Freqs.end())
|
|
|| ((find( Freqs.begin(), Freqs.end(), 0 ) != Freqs.end()) && (statQuality == Supreme)) );
|
|
// was: find_if ... bind2nd( equals<uint>(), 1 )
|
|
break;
|
|
case PrimeRoots: // Only freq 1 families exist if the PrimeRoots
|
|
return find( Freqs.begin(), Freqs.end(), 1 ) != Freqs.end();
|
|
break;
|
|
default: // A rm family exists in the ecosystem matching a civ if the corresponding freq is 1 or 3
|
|
{
|
|
uint iCiv = getIndexFromString( ecosystemCodes[iEcosystem], CivEcosystemCodes, NbCiv, false );
|
|
vector<TCiv>::const_iterator it = find( Civs.begin(), Civs.end(), (TCiv)iCiv );
|
|
if ( it != Civs.end() )
|
|
return (Freqs[it-Civs.begin()] == 1) || (Freqs[it-Civs.begin()] == 3);
|
|
else
|
|
return false;
|
|
}
|
|
}
|
|
}*/
|
|
|
|
// Only used for deposits;
|
|
bool TFamInfo::existsInEcosystem( TEcosystem iEcosystem, TStatQuality statQuality ) const
|
|
{
|
|
if ( find( Freqs.begin(), Freqs.end(), 0 ) != Freqs.end() )
|
|
{
|
|
// Freq 0 => only Common/Supreme
|
|
return (statQuality == Supreme) && (iEcosystem == CommonEcosystem);
|
|
}
|
|
else if ( statQuality <= Fine )
|
|
{
|
|
// Basic, Fine => Common
|
|
return (iEcosystem == CommonEcosystem);
|
|
}
|
|
else
|
|
{
|
|
// Choice to Supreme => One per ecosystem
|
|
return (iEcosystem != CommonEcosystem) && (iEcosystem < NbEcosystems);
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
struct TCraftPartInfo
|
|
{
|
|
CSString Name;
|
|
CSString Path;
|
|
uint8 PartIndex;
|
|
bool Enabled;
|
|
};
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
class CCraftParts
|
|
{
|
|
public:
|
|
|
|
///
|
|
CCraftParts() : CraftParts( NbFaberElements )
|
|
{
|
|
for ( uint i=0; i!=NbFaberElements; ++i )
|
|
{
|
|
CraftParts[i].PartIndex = i;
|
|
CraftParts[i].Enabled = false;
|
|
}
|
|
}
|
|
|
|
///
|
|
void registerPartChars( const CSString& parts )
|
|
{
|
|
for ( string::size_type p=0; p!=parts.size(); ++p )
|
|
{
|
|
uint index = (uint)(parts[p] - 'A');
|
|
CraftParts[index].Enabled = true;
|
|
}
|
|
}
|
|
|
|
///
|
|
bool isEnabled( uint index ) const
|
|
{
|
|
return CraftParts[index].Enabled;
|
|
}
|
|
|
|
///
|
|
void getNamesAndPaths( const vector<string>& paths )
|
|
{
|
|
uint i = 0;
|
|
vector<string>::const_iterator ip;
|
|
for ( ip=paths.begin(); ip!=paths.end(); ++ip )
|
|
{
|
|
if ( i >= CraftParts.size() )
|
|
nlerror( "Mismatch between sitem DFN and constant (nb of craft parts)" );
|
|
|
|
CraftParts[i].Path = (*ip);
|
|
string::size_type p = (*ip).rfind( '.' ) + 1; // string::npos+1 gives 0
|
|
CraftParts[i].Name = (*ip).substr( p );
|
|
nldebug( "%u: %s", ip-paths.begin(), CraftParts[i].Name.c_str() );
|
|
++i;
|
|
}
|
|
}
|
|
|
|
///
|
|
TCraftPartInfo& operator[] ( uint index ) { return CraftParts[index]; }
|
|
|
|
vector< TCraftPartInfo > CraftParts;
|
|
|
|
};
|
|
|
|
typedef map<CSString, TFamInfo, CUnsensitiveSStringLessPred > CFamMap;
|
|
CFamMap FamSet;
|
|
CCraftParts CraftParts;
|
|
|
|
enum TFamAndPropLine {
|
|
LFam, LGroup, LCraftParts, LCiv, LFreq, LLoc,
|
|
LIconMain, LIconBk, LIconOv1, LIconOv2, LIconSpecial,
|
|
LCraftPlans, LGenerateOnly,
|
|
LBaseOfRemarkableStatIndices,
|
|
LColorIndex = LBaseOfRemarkableStatIndices + NB_REMARKABLE_STAT_INDICES,
|
|
LJewelProtIndex,
|
|
NbFamAndPropCols };
|
|
|
|
|
|
// static
|
|
bool TFamInfo::mustGenerateFamily( uint iFamily )
|
|
{
|
|
if ( families[iFamily].empty() )
|
|
return false;
|
|
else if ( ! TFamInfo::UseGenerateOnly )
|
|
return true;
|
|
else
|
|
{
|
|
TFamInfo& famInfo = FamSet[families[iFamily]];
|
|
return ( famInfo.GenerateOnly );
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
CSString getShortFaberElemString( uint rFaberElem )
|
|
{
|
|
string& longString = CraftParts[rFaberElem].Name;
|
|
return reinterpret_cast<CSString&>(longString.substr( longString.find( "(" ) + 1 )).rightCrop( 1 );
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
TCiv getCivFromStr( const CSString& civStr )
|
|
{
|
|
if ( civStr.empty() )
|
|
return AllCiv;
|
|
else
|
|
{
|
|
for ( uint i=0; i!=NbCiv; ++i )
|
|
{
|
|
if ( civStr == CSString(CivNames[i]) )
|
|
return (TCiv)i;
|
|
}
|
|
nlwarning( "Unknown civ '%s'", civStr.c_str() );
|
|
return AllCiv;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
uint getFreqFromStr( const CSString& freqStr )
|
|
{
|
|
uint f = atoi( freqStr.c_str() );
|
|
if ( (f < 1) && (f > 5) )
|
|
nlwarning( "Unknown freq '%s'", freqStr.c_str() );
|
|
return f;
|
|
}
|
|
|
|
|
|
/*
|
|
* Returns ~0 if s is empty
|
|
*/
|
|
TGroup getNewOrExistingGroupFromStr( const CSString& s )
|
|
{
|
|
uint i = getIndexFromString( s, groups, false );
|
|
if ( i == ~0 )
|
|
{
|
|
if ( s.empty() )
|
|
return ~0;
|
|
else
|
|
{
|
|
i = groups.size();
|
|
groups.push_back( s );
|
|
nlinfo( "New group: %s (%u)", s.c_str(), i );
|
|
}
|
|
}
|
|
return i;
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void deliverFamAndProp( vs& line )
|
|
{
|
|
if ( line.size() < NbFamAndPropCols )
|
|
line.resize( NbFamAndPropCols );
|
|
|
|
cleanExteriorWhitespace( line );
|
|
|
|
if ( line[LFam].empty() )
|
|
{
|
|
// Load special icons
|
|
if ( (line.size() >= LIconSpecial+1) && (! line[LIconSpecial].empty()) && (line[LIconMain].empty()) )
|
|
{
|
|
/*if ( line.size() >= LIconMain+1 )
|
|
Icons[line[LIconSpecial]].Icon = line[LIconMain];*/
|
|
if ( line.size() >= LIconBk+1 )
|
|
Icons[line[LIconSpecial]].IconBackground = line[LIconBk];
|
|
if ( line.size() >= LIconOv1+1 )
|
|
Icons[line[LIconSpecial]].IconOver = line[LIconOv1];
|
|
}
|
|
return;
|
|
}
|
|
|
|
// Load icons of families
|
|
if ( line.size() >= LIconMain+1 )
|
|
{
|
|
Icons[line[LFam]].Icon = line[LIconMain];
|
|
if ( ! line[LGroup].empty() )
|
|
{
|
|
// For group, set icon of first family of group found! (for forage source knowledge)
|
|
if ( Icons.find( line[LGroup] ) == Icons.end() )
|
|
{
|
|
Icons[line[LGroup]].Icon = line[LIconMain];
|
|
}
|
|
}
|
|
}
|
|
if ( line.size() >= LIconBk+1 )
|
|
Icons[line[LFam]].IconBackground = line[LIconBk];
|
|
if ( line.size() >= LIconOv1+1 )
|
|
Icons[line[LFam]].IconOver = line[LIconOv1];
|
|
|
|
TFamInfo& famInfo = FamSet[line[LFam]];
|
|
famInfo.IsActive = true;
|
|
/*if ( ! line[LCraftParts].empty() )
|
|
{
|
|
// Store by property (line[LProp])
|
|
famInfo.Properties.push_back( line[LProp] );
|
|
famInfo.CraftPartsByProp.push_back( famInfo.CompatibleCraftParts.size() ); // beginning of craft parts chars
|
|
famInfo.CompatibleCraftParts += line[LCraftParts];
|
|
CraftParts.registerPartChars( line[LCraftParts] );
|
|
famInfo.Civs.push_back( getCivFromStr( line[LCiv] ) );
|
|
famInfo.Freqs.push_back( getFreqFromStr( line[LFreq] ) );
|
|
famInfo.IsInDeposits = line[LLoc].contains( "D" );
|
|
famInfo.IsInCreatures = line[LLoc].contains( "C" );
|
|
if ( ! (famInfo.IsInDeposits || famInfo.IsInCreatures) )
|
|
nlwarning( "Unknown loc for %s", line[LFam].c_str() );
|
|
}*/
|
|
|
|
for ( string::size_type p=0; p!=line[LCraftParts].size(); ++p )
|
|
{
|
|
// Store by property = craft part (each char of line[LCraftParts])
|
|
CSString craftPart = string( 1, line[LCraftParts][p]);
|
|
famInfo.Properties.push_back( craftPart );
|
|
famInfo.CraftPartsByProp.push_back( famInfo.CompatibleCraftParts.size() );
|
|
famInfo.CompatibleCraftParts += craftPart;
|
|
CraftParts.registerPartChars( craftPart );
|
|
famInfo.Civs.push_back( getCivFromStr( line[LCiv] ) );
|
|
famInfo.Freqs.push_back( getFreqFromStr( line[LFreq] ) );
|
|
}
|
|
if ( line[LCraftParts].empty() )
|
|
{
|
|
famInfo.Freqs.push_back( getFreqFromStr( line[LFreq] ) ); // freq needed for Rarity computation
|
|
}
|
|
famInfo.Group = getNewOrExistingGroupFromStr( line[LGroup] );
|
|
famInfo.IsInDeposits = line[LLoc].contains( "D" );
|
|
famInfo.IsInCreatures = line[LLoc].contains( "C" );
|
|
if ( ! (famInfo.IsInDeposits || famInfo.IsInCreatures) )
|
|
{
|
|
famInfo.SpecialCreatureTag = line[LLoc];
|
|
if ( (famInfo.SpecialCreatureTag[0] != 'G') && (famInfo.SpecialCreatureTag[0] != 'I') )
|
|
nlwarning( "Unknown loc %s for %s", line[LLoc].c_str(), line[LFam].c_str() );
|
|
}
|
|
famInfo.IsForMission = line[LCraftParts].empty();
|
|
for ( uint i=0; i!=NB_REMARKABLE_STAT_INDICES; ++i )
|
|
{
|
|
if ( line[LBaseOfRemarkableStatIndices+i].empty() && (! line[LCraftParts].empty()) && (line[LFreq] != "0") )
|
|
nlerror( "%s has empty stat index %u", line[LFam].c_str(), i );
|
|
famInfo.RemarkableStatIndex[i] = atoi( line[LBaseOfRemarkableStatIndices+i].c_str() );
|
|
}
|
|
if ( ! line[LColorIndex].empty() )
|
|
famInfo.ColorIndex = atoi( line[LColorIndex].c_str() );
|
|
if ( ! line[LJewelProtIndex].empty() )
|
|
famInfo.JewelProtIndex = atoi( line[LJewelProtIndex].c_str() );
|
|
bool markedForGeneration = (line[LGenerateOnly] == "X");
|
|
if ( (!markedForGeneration) && famInfo.GenerateOnly )
|
|
{
|
|
nlwarning( "Found duplicate family line with different GenerateOnly setting" );
|
|
}
|
|
else
|
|
{
|
|
famInfo.GenerateOnly = markedForGeneration;
|
|
}
|
|
if ( famInfo.GenerateOnly )
|
|
++TFamInfo::UseGenerateOnly;
|
|
}
|
|
|
|
|
|
typedef map< TGroup, set<uint32> > CGroupMap;
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void loadFamAndProp( const string& filename, bool displayAll )
|
|
{
|
|
loadCSVFile( filename.c_str(), deliverFamAndProp );
|
|
|
|
if ( displayAll )
|
|
{
|
|
set<CSString, CUnsensitiveSStringLessPred> propSet;
|
|
CGroupMap groupMap;
|
|
|
|
/// Generate contents of item_mp_family.typ (and fill group map)
|
|
nlinfo( "item_mp_family.typ:" );
|
|
InfoLog->displayRawNL( "<DEFINITION Label=\"Undefined\" Value=\"0\"/>" );
|
|
uint i = 1;
|
|
for ( CFamMap::iterator iss=FamSet.begin(); iss!=FamSet.end(); ++iss )
|
|
{
|
|
const CSString& famStr = (*iss).first;
|
|
TFamInfo& famInfo = (*iss).second;
|
|
InfoLog->displayRawNL( "<DEFINITION Label=\"%s\" Value=\"%u\"/>", famStr.c_str(), i );
|
|
|
|
// Get info about props and group
|
|
for ( vs::iterator ip=famInfo.Properties.begin(); ip!=famInfo.Properties.end(); ++ip )
|
|
{
|
|
propSet.insert( *ip );
|
|
}
|
|
groupMap[ famInfo.Group ].insert( i ); // ~0 is for "no group" (creature's RMs only)
|
|
++i;
|
|
}
|
|
|
|
/*
|
|
/// Generate family-specialized forage search bricks (TODO)
|
|
nlinfo( "Family-specialized forage search bricks:");
|
|
i = 1;
|
|
for ( CGroupMap::iterator igm=groupMap.begin(); igm!=groupMap.end(); ++igm )
|
|
{
|
|
CSString skill = toString( "SHFM%u", i );
|
|
CSString rmgrpBrickCode = toString( "BHFPMB%02u", i );
|
|
CSString rmfamBrickFamCode = "BHFPMI" + string( 1, (char)'A' + ((char)(i-1)) );
|
|
uint j = 1;
|
|
for ( set<uint32>:::iterator ifs=(*igm).begin(); ifs!=(*igm).end(); ++ifs )
|
|
{
|
|
// TODO: modifier of modifier
|
|
CSString brickCode = rmfamBrickFamCode + toString( "%02u", j );
|
|
InfoLog->displayRawNL( "%s\t80\t%s\t%u\t\t%s\t\tFG_RMFAM_FILT: %u\t", brickCode.c_str(), rmgrpBrickCode.c_str(), j, skill.c_str(), (*ifs) );
|
|
++j;
|
|
}
|
|
}
|
|
|
|
/// Generate family-specialized forage search phrases (TODO)
|
|
nlinfo( "Family-specialized forage search phrases:");
|
|
i = 1;
|
|
for ( CFamMap::iterator iss=FamSet.begin(); iss!=FamSet.end(); ++iss )
|
|
{
|
|
const CSString& famStr = (*iss).first;
|
|
TFamInfo& famInfo = (*iss).second;
|
|
InfoLog->displayRawNL( "<DEFINITION Label=\"%s\" Value=\"%u\"/>", famStr.c_str(), i );
|
|
}
|
|
|
|
/// Generate family-specialized forage extraction bricks (TODO)
|
|
nlinfo( "Family-specialized forage extraction bricks:");
|
|
i = 1;
|
|
for ( CFamMap::iterator iss=FamSet.begin(); iss!=FamSet.end(); ++iss )
|
|
{
|
|
const CSString& famStr = (*iss).first;
|
|
TFamInfo& famInfo = (*iss).second;
|
|
InfoLog->displayRawNL( "<DEFINITION Label=\"%s\" Value=\"%u\"/>", famStr.c_str(), i );
|
|
++i;
|
|
}
|
|
|
|
/// Generate family-specialized forage extraction phrases (TODO)
|
|
nlinfo( "Family-specialized forage extraction phrases:");
|
|
i = 1;
|
|
for ( CFamMap::iterator iss=FamSet.begin(); iss!=FamSet.end(); ++iss )
|
|
{
|
|
const CSString& famStr = (*iss).first;
|
|
TFamInfo& famInfo = (*iss).second;
|
|
InfoLog->displayRawNL( "<DEFINITION Label=\"%s\" Value=\"%u\"/>", famStr.c_str(), i );
|
|
++i;
|
|
}*/
|
|
|
|
/// Generate item_mp_property.typ
|
|
nlinfo( "Item parts as props:" );
|
|
InfoLog->displayRawNL( "<DEFINITION Label=\"Undefined\" Value=\"0\"/>" );
|
|
i = 1;
|
|
for ( set<CSString, CUnsensitiveSStringLessPred>::iterator iss=propSet.begin(); iss!=propSet.end(); ++iss )
|
|
{
|
|
InfoLog->displayRawNL( "<DEFINITION Label=\"%s\" Value=\"%u\"/>", (*iss).c_str(), i );
|
|
++i;
|
|
}
|
|
|
|
/// Generate item_mp_group.typ
|
|
nlinfo( "Groups:" );
|
|
InfoLog->displayRawNL( "<DEFINITION Label=\"Undefined\" Value=\"0\"/>" );
|
|
i = 1;
|
|
for ( CGroupMap::iterator igm=groupMap.begin(); igm!=groupMap.end(); ++igm )
|
|
{
|
|
if ( (*igm).first != ~0 )
|
|
{
|
|
InfoLog->displayRawNL( "<DEFINITION Label=\"%s\" Value=\"%u\"/>", (groups[(*igm).first]).c_str(), i );
|
|
++i;
|
|
}
|
|
}
|
|
|
|
/*
|
|
/// Generate group-specialized forage search bricks (TODO)
|
|
nlinfo( "Group-specialized forage search bricks:");
|
|
i = 1;
|
|
for ( CGroupMap::iterator igm=groupMap.begin(); igm!=groupMap.end(); ++igm )
|
|
{
|
|
CSString skill = toString( "SHFM%u", i );
|
|
CSString rmgrpBrickCode = toString( "BHFPMB%02u", i );
|
|
++i;
|
|
}
|
|
|
|
/// Generate group-specialized forage search phrases
|
|
nlinfo( "Group-specialized forage search phrases:");
|
|
i = 1;
|
|
for ( set<CSString, CUnsensitiveSStringLessPred>::iterator iss=groupSet.begin(); iss!=groupSet.end(); ++iss )
|
|
{
|
|
if ( (*iss).empty() )
|
|
continue;
|
|
InfoLog->displayRawNL( "<DEFINITION Label=\"%s\" Value=\"%u\"/>", (*iss).c_str(), i );
|
|
++i;
|
|
}
|
|
|
|
/// Generate group-specialized forage extraction bricks
|
|
nlinfo( "Group-specialized forage extraction bricks:");
|
|
i = 1;
|
|
for ( set<CSString, CUnsensitiveSStringLessPred>::iterator iss=groupSet.begin(); iss!=groupSet.end(); ++iss )
|
|
{
|
|
if ( (*iss).empty() )
|
|
continue;
|
|
InfoLog->displayRawNL( "<DEFINITION Label=\"%s\" Value=\"%u\"/>", (*iss).c_str(), i );
|
|
++i;
|
|
}
|
|
|
|
/// Generate group-specialized forage extraction phrases
|
|
nlinfo( "Group-specialized forage extraction phrases:");
|
|
i = 1;
|
|
for ( set<CSString, CUnsensitiveSStringLessPred>::iterator iss=groupSet.begin(); iss!=groupSet.end(); ++iss )
|
|
{
|
|
if ( (*iss).empty() )
|
|
continue;
|
|
InfoLog->displayRawNL( "<DEFINITION Label=\"%s\" Value=\"%u\"/>", (*iss).c_str(), i );
|
|
++i;
|
|
}*/
|
|
|
|
nlinfo( "TODO: Keep old values when adding new entries" );
|
|
nlinfo( "Don't forget to regen craft plans and to map localized texts" );
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Multi-indexed array.
|
|
* NC is the number of columns.
|
|
*/
|
|
template <uint32 NC>
|
|
class CSortableData
|
|
{
|
|
public:
|
|
|
|
/// A row is made of fields, usually 1 per column but there may be more than one (each one is a key)
|
|
struct TSortableItem
|
|
{
|
|
std::vector<std::string> Fields [NC];
|
|
|
|
///
|
|
void push( uint32 column, const std::string& f, bool allowDuplicates=false )
|
|
{
|
|
if ( (allowDuplicates) || (find( Fields[column].begin(), Fields[column].end(), f ) == Fields[column].end()) )
|
|
Fields[column].push_back( f );
|
|
}
|
|
|
|
/**
|
|
* Display the item as a row of a HTML table.
|
|
* If (key!=previousKey) and (name==previousName), the row will not be displayed entirely to save space
|
|
*
|
|
* \param keyColumn If not ~0, column used for sorting => this column displays only the field matching the key
|
|
* \param key The key used for sorting (see keyColumn)
|
|
* \param previousKey Previous key
|
|
* \param nameColumn If not ~0, column used for the unique name (column must have exaclty one element)
|
|
* \param previousName Previous name
|
|
*/
|
|
std::string toHTMLRow( uint32 keyColumn=~0, const string& key=string(), const string& previousKey=string(),
|
|
uint32 nameColumn=~0, const string& previousName=string() ) const
|
|
{
|
|
std::string s = "<tr>";
|
|
bool lightMode = (nameColumn == ~0) ? false : ((key != previousKey) && (Fields[nameColumn][0] == previousName));
|
|
for ( uint32 c=0; c!=NC; ++c )
|
|
{
|
|
s += "<td>";
|
|
if ( c == keyColumn )
|
|
s += key; // key should be a substr of toString( c )
|
|
else
|
|
{
|
|
if ( lightMode )
|
|
s += "\"";
|
|
else
|
|
s += columnToString( c );
|
|
}
|
|
s += "</td>";
|
|
}
|
|
s += "</tr>\n";
|
|
return s;
|
|
}
|
|
|
|
|
|
///
|
|
std::string toCSVLine( char columnSeparator=',', string internalSeparator=" - ", uint32 keyColumn=~0, const string& key=string(), const string& previousKey=string(),
|
|
uint32 nameColumn=~0, const string& previousName=string() ) const
|
|
{
|
|
std::string s;
|
|
bool lightMode = (nameColumn == ~0) ? false : ((key != previousKey) && (Fields[nameColumn][0] == previousName));
|
|
for ( uint32 c=0; c!=NC; ++c )
|
|
{
|
|
if ( c == keyColumn )
|
|
s += key; // key should be a substr of columnToString( c )
|
|
else
|
|
{
|
|
if ( lightMode )
|
|
s += "\"";
|
|
else
|
|
s += columnToString( c, internalSeparator );
|
|
}
|
|
s += columnSeparator;
|
|
}
|
|
s += "\n";
|
|
return s;
|
|
}
|
|
|
|
///
|
|
std::string columnToString( uint32 column, const std::string& internalSeparator=", " ) const
|
|
{
|
|
std::string s;
|
|
std::vector<std::string>::const_iterator ivs;
|
|
for ( ivs=Fields[column].begin(); ivs!=Fields[column].end(); ++ivs )
|
|
{
|
|
if ( ivs!=Fields[column].begin() )
|
|
s += internalSeparator;
|
|
s += (*ivs);
|
|
}
|
|
return s;
|
|
}
|
|
};
|
|
|
|
typedef std::multimap< std::string, uint32 > CLookup; // key to index (not pt because reallocation invalidates pointers)
|
|
typedef std::vector< TSortableItem > CItems;
|
|
|
|
/// Init
|
|
void init( bool enabled )
|
|
{
|
|
_Enabled = enabled;
|
|
}
|
|
|
|
/// Add a row
|
|
void addItem( const TSortableItem& item )
|
|
{
|
|
if ( ! _Enabled )
|
|
return;
|
|
|
|
_Items.push_back( item );
|
|
for ( uint32 c=0; c!=NC; ++c )
|
|
{
|
|
for ( std::vector<std::string>::const_iterator ik=item.Fields[c].begin(); ik!=item.Fields[c].end(); ++ik )
|
|
{
|
|
_Indices[c].insert( make_pair( *ik, _Items.size()-1 ) );
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Update a row (found by the first column, which must have exactly one element).
|
|
* Returns true if it existed before, false if it's being created.
|
|
* If it existed before:
|
|
* - Does not remove elements that already exist and are not in the new item
|
|
* - Adds the new elements found in the new item at the specified columns, and updates lookup map
|
|
*/
|
|
bool updateItemAppend( const TSortableItem& item, uint32 column )
|
|
{
|
|
if ( ! _Enabled )
|
|
return true; // quiet
|
|
|
|
uint32 nameColumn = 0;
|
|
CLookup::iterator ilk = _Indices[nameColumn].find( item.Fields[nameColumn][0] );
|
|
if ( ilk != _Indices[nameColumn].end() )
|
|
{
|
|
uint32& index = (*ilk).second;
|
|
|
|
// Update map for the specified column
|
|
// and update item column
|
|
for ( std::vector<std::string>::const_iterator ivs=item.Fields[column].begin(); ivs!=item.Fields[column].end(); ++ivs )
|
|
{
|
|
ilk = _Indices[column].find( *ivs );
|
|
if ( (ilk == _Indices[column].end()) || ( getRow( (*ilk).second ).Fields[nameColumn][0] != item.Fields[nameColumn][0]) )
|
|
{
|
|
_Indices[column].insert( make_pair( *ivs, index ) );
|
|
_Items[index].Fields[column].push_back( *ivs );
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
addItem( item );
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Update a row (found by the first column, which must have exactly one element).
|
|
* Returns true if it existed before, false if it's being created.
|
|
* If it existed before:
|
|
* - Does not update lookup maps or item for columns that were already present.
|
|
* - Adds entries in lookup maps and updates item for new columns (fields that were empty).
|
|
*/
|
|
/*bool updateItemAppend( const TSortableItem& item )
|
|
{
|
|
if ( ! _Enabled )
|
|
return true; // quiet
|
|
|
|
CLookup::iterator ilk = _Indices[0].find( item.Fields[0][0] );
|
|
if ( ilk != _Indices[0].end() )
|
|
{
|
|
uint32& index = (*ilk).second;
|
|
|
|
for ( uint32 c=1; c!=NC; ++c )
|
|
{
|
|
// Update maps for previously empty columns
|
|
if ( _Items[index].Fields[c].empty() )
|
|
{
|
|
for ( std::vector<std::string>::iterator ivs=item.Fields[c].begin(); ivs!=item.Fields[c].end(); ++ivs )
|
|
_Indices[c].insert( make_pair( *ivs, index ) );
|
|
}
|
|
|
|
// Update item column
|
|
_Items[index].Fields[c] = item.Fields[c];
|
|
}
|
|
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
addItem( item );
|
|
return false;
|
|
}
|
|
}*/
|
|
|
|
/// Find or browse by key
|
|
CLookup& lookup( uint32 column )
|
|
{
|
|
return _Indices[column];
|
|
}
|
|
|
|
/// Browse by adding order
|
|
CItems& items()
|
|
{
|
|
return _Items;
|
|
}
|
|
|
|
/// Get a row by index
|
|
TSortableItem& getRow( uint32 index )
|
|
{
|
|
return _Items[index];
|
|
}
|
|
|
|
private:
|
|
|
|
CLookup _Indices [NC];
|
|
|
|
CItems _Items;
|
|
|
|
bool _Enabled;
|
|
};
|
|
|
|
|
|
typedef CSortableData<DtNbCols> CRMData;
|
|
typedef CRMData::TSortableItem TRMItem;
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
class CProducedDocHtml
|
|
{
|
|
public:
|
|
|
|
///
|
|
CProducedDocHtml() : _File(NULL), _Enabled(false) {}
|
|
|
|
///
|
|
void open( const std::string& filename, const std::string& title, bool enableFlag )
|
|
{
|
|
_Enabled = enableFlag;
|
|
if ( ! _Enabled )
|
|
return;
|
|
|
|
_File = NLMISC::nlfopen( filename, "wt" );
|
|
fprintf( _File, ("<html><head>\n<meta http-equiv=\"content-type\" content=\"text/html; charset=UTF-8\">\n<title>" + title + "</title>\n</head><body>\n").c_str() );
|
|
}
|
|
|
|
///
|
|
void write( const std::string& htmlCode )
|
|
{
|
|
if ( ! _Enabled )
|
|
return;
|
|
|
|
fprintf( _File, htmlCode.c_str() );
|
|
}
|
|
|
|
///
|
|
void writeln( const std::string& htmlCode )
|
|
{
|
|
write( htmlCode + "\n" );
|
|
}
|
|
|
|
///
|
|
void writebln( const std::string& htmlCode )
|
|
{
|
|
write( htmlCode + "<br>\n" );
|
|
}
|
|
|
|
///
|
|
void writepln( const std::string& htmlCode )
|
|
{
|
|
write( "<p>" + htmlCode + "</p>\n" );
|
|
}
|
|
|
|
///
|
|
void save()
|
|
{
|
|
if ( ! _Enabled )
|
|
return;
|
|
|
|
fprintf( _File, "</body></html>\n" );
|
|
fclose( _File );
|
|
}
|
|
|
|
private:
|
|
|
|
FILE *_File;
|
|
bool _Enabled;
|
|
};
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
class CProducedDocCSV
|
|
{
|
|
public:
|
|
|
|
///
|
|
CProducedDocCSV() : _File(NULL), _Enabled(false) {}
|
|
|
|
///
|
|
void open( const std::string& filename, bool enableFlag )
|
|
{
|
|
_Enabled = enableFlag;
|
|
if ( ! _Enabled )
|
|
return;
|
|
|
|
_File = NLMISC::nlfopen( filename, "wt" );
|
|
}
|
|
|
|
///
|
|
void write( const std::string& data )
|
|
{
|
|
if ( ! _Enabled )
|
|
return;
|
|
|
|
fprintf( _File, data.c_str() );
|
|
}
|
|
|
|
///
|
|
void writeln( const std::string& data )
|
|
{
|
|
write( data + "\n" );
|
|
}
|
|
|
|
///
|
|
void save()
|
|
{
|
|
if ( ! _Enabled )
|
|
return;
|
|
|
|
fclose( _File );
|
|
}
|
|
|
|
private:
|
|
|
|
FILE *_File;
|
|
bool _Enabled;
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
class CGenRawMaterial
|
|
{
|
|
public:
|
|
|
|
/// Constructor
|
|
CGenRawMaterial( const std::string& sheetName = std::string() ) : SheetName(sheetName), ILocation(~0), IFamily(~0), IEcosystem(NbEcosystems), StatQuality(InvalidStatQuality), Color(InvalidColor), StatEnergyAvg(0)
|
|
{}
|
|
|
|
/// Serial
|
|
void serial( NLMISC::IStream& s )
|
|
{
|
|
s.serial( SheetName );
|
|
s.serial( (uint32&)ILocation );
|
|
s.serial( (uint32&)IFamily );
|
|
s.serial( (uint32&)Group );
|
|
s.serial( (uint32&)IEcosystem );
|
|
s.serial( (uint32&)StatQuality );
|
|
s.serial( (sint32&)Color );
|
|
//s.serial( (uint32&)SapLoadLevel );
|
|
//s.serial( (uint32&)Rarity );
|
|
s.serial( (sint32&)StatEnergyAvg );
|
|
s.serial( (uint32&)MaxLevel );
|
|
s.serialCont( RMProperties );
|
|
s.serialCont( IPropertyDepths );
|
|
s.serialCont( RMCraftCharacs );
|
|
}
|
|
|
|
/// Computes randomly RMCraftCharacs, IPropertyDepths... Returns false if the RM must NOT be generated.
|
|
bool computeCraftCharacs( uint iVariant, const CSString& sheetName );
|
|
|
|
///
|
|
void writeSheet( CForm *form );
|
|
|
|
///
|
|
void loadSheet( CForm *form, const std::string& sheetName, bool full );
|
|
|
|
///
|
|
void collectStats( TRMItem& item, CMainStat& mainStats );
|
|
|
|
/// Return average of energies (including max quality as half of the balance)
|
|
/*float getEnergyAvg() const
|
|
{
|
|
if ( RMCraftCharacs.empty() )
|
|
return 0.0f;
|
|
else
|
|
{
|
|
float sum = 0.0f;
|
|
for ( list<CFaberCharacteristics>::const_iterator ics=RMCraftCharacs.begin(); ics!=RMCraftCharacs.end(); ++ics )
|
|
{
|
|
sum += (*ics).ActualEnergy;
|
|
}
|
|
//return (sum + (float)MaxQuality / 250.0f) / ((float)RMCraftCharacs.size() + 1);
|
|
return (sum / (float)RMCraftCharacs.size()); // now, MaxQuality is not part of the average
|
|
}
|
|
}*/
|
|
|
|
///
|
|
float getOriginalityAvg() const
|
|
{
|
|
float sum = 0.0f;
|
|
for ( list<CFaberCharacteristics>::const_iterator ics=RMCraftCharacs.begin(); ics!=RMCraftCharacs.end(); ++ics )
|
|
{
|
|
sum += (*ics).ActualOriginality;
|
|
}
|
|
return sum / (float)RMCraftCharacs.size();
|
|
}
|
|
|
|
///
|
|
float getOriginalityMax() const
|
|
{
|
|
float maxOriginality = 0.0f;
|
|
for ( list<CFaberCharacteristics>::const_iterator ics=RMCraftCharacs.begin(); ics!=RMCraftCharacs.end(); ++ics )
|
|
{
|
|
if ( (*ics).ActualOriginality > maxOriginality )
|
|
maxOriginality = (*ics).ActualOriginality;
|
|
}
|
|
return maxOriginality;
|
|
}
|
|
|
|
///
|
|
void fillPropertiesFromFamily()
|
|
{
|
|
vs& props = FamSet[familyStr()].Properties;
|
|
RMProperties.resize( props.size() );
|
|
for ( vs::iterator ip=props.begin(); ip!=props.end(); ++ip )
|
|
{
|
|
RMProperties[ip-props.begin()] = getIndexFromString( *ip, properties );
|
|
}
|
|
}
|
|
|
|
///
|
|
bool hasCraftPart( uint craftPartIndex )
|
|
{
|
|
return CraftParts.isEnabled( craftPartIndex ) && (FamSet[familyStr()].CompatibleCraftParts.find( string( 1, (char)'A' + craftPartIndex ).c_str() ) != string::npos);
|
|
}
|
|
|
|
///
|
|
/*TCiv getCivSpec( uint craftPartIndex, const TFamInfo& famInfo )
|
|
{
|
|
TCiv civ = NbCiv;
|
|
for ( vector<TCiv>::const_iterator ivc=famInfo.Civs.begin(); ivc!=famInfo.Civs.end(); ++ivc )
|
|
{
|
|
// Skip those not matching the current rFaberElem
|
|
if ( famInfo.getCraftPartForProp( ivc-famInfo.Civs.begin() ).find( string( 1, (char)('A' + craftPartIndex) ).c_str() ) != string::npos )
|
|
{
|
|
if ( (civ != NbCiv) && ((*ivc) != civ) )
|
|
{
|
|
nlwarning( "Different civ specializations for %s, %s (%s and %s)", familyStr().c_str(), getShortFaberElemString( craftPartIndex ).c_str(), CivNames[civ], CivNames[*ivc] );
|
|
return AllCiv;
|
|
}
|
|
civ = (*ivc);
|
|
}
|
|
}
|
|
if ( civ == NbCiv )
|
|
return AllCiv;
|
|
else if ( civ =
|
|
return civ;
|
|
}*/
|
|
|
|
///
|
|
TCiv getCivSpec( TEcosystem iEcosystem, TStatQuality statQuality )
|
|
{
|
|
if ( (statQuality <= Fine) || (iEcosystem >= NbEcosystems) )
|
|
return AllCiv;
|
|
else
|
|
return EcosystemToCiv[iEcosystem];
|
|
}
|
|
|
|
///
|
|
/*TCiv getMainCivSpec( const TFamInfo& famInfo )
|
|
{
|
|
TCiv civ = NbCiv;
|
|
for ( list<CFaberCharacteristics>::const_iterator ics=RMCraftCharacs.begin(); ics!=RMCraftCharacs.end(); ++ics )
|
|
{
|
|
TCiv civOfCraftPart = getCivSpec( (*ics).FaberElement, famInfo );
|
|
if ( (civ != NbCiv) && (civOfCraftPart != civ) )
|
|
{
|
|
return AllCiv;
|
|
}
|
|
civ = civOfCraftPart;
|
|
}
|
|
if ( civ == NbCiv )
|
|
return AllCiv;
|
|
else
|
|
return civ;
|
|
}*/
|
|
|
|
///
|
|
/*const char * getMainEcosystemSpec( const TFamInfo& famInfo )
|
|
{
|
|
return CivEcosystemCodes[getMainCivSpec( famInfo )];
|
|
}*/
|
|
|
|
/// Code
|
|
CSString SheetName;
|
|
|
|
/// Index in locations
|
|
uint32 ILocation;
|
|
|
|
/// Index in families
|
|
uint32 IFamily;
|
|
|
|
///
|
|
CSString familyStr() const { return families[IFamily]; }
|
|
|
|
/// Group number
|
|
TGroup Group;
|
|
|
|
///
|
|
CSString groupStr() const { return Group==~0 ? "" : groups[Group]; }
|
|
|
|
/// Index in ecosystems
|
|
TEcosystem IEcosystem;
|
|
|
|
///
|
|
CSString ecosystemStr() const { return ecosystems[IEcosystem]; }
|
|
|
|
/// From Basic) to Supreme)
|
|
TStatQuality StatQuality;
|
|
|
|
/// From 'b' (Basic) to 'f' (Supreme)
|
|
char levelZoneLChar() const { return 'b' + (char)StatQuality; }
|
|
|
|
/// Same
|
|
void setStatQuality( char levelZoneChar ) { StatQuality = (TStatQuality)(levelZoneChar - 'b'); }
|
|
|
|
/// For creatures
|
|
uint32 ILevelZone;
|
|
|
|
/// Index in colors
|
|
TColor Color;
|
|
|
|
///
|
|
CSString colorStr() const { return colors[Color]; }
|
|
|
|
/// Sap load level
|
|
//uint32 SapLoadLevel;
|
|
|
|
/// Rarity
|
|
//uint32 Rarity;
|
|
|
|
sint32 StatEnergyAvg;
|
|
|
|
/// Max quality
|
|
uint32 MaxLevel;
|
|
|
|
/// Indices in properties
|
|
vu RMProperties;
|
|
|
|
///
|
|
CSString propertyStr( uint32 p ) const { return properties[RMProperties[p]]; }
|
|
|
|
///
|
|
vu IPropertyDepths;
|
|
|
|
///
|
|
CSString propertyDepthStr( uint32 p ) const { return PropertyDepths[IPropertyDepths[p]]; }
|
|
|
|
///
|
|
CFaberCharacteristics *getCraftSlot( uint rFaberElem )
|
|
{
|
|
std::list< CFaberCharacteristics >::iterator icl;
|
|
for ( icl=RMCraftCharacs.begin(); icl!=RMCraftCharacs.end(); ++icl )
|
|
{
|
|
if ( (*icl).FaberElement == rFaberElem )
|
|
return &(*icl);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/// Randomly generated characs
|
|
std::list< CFaberCharacteristics > RMCraftCharacs;
|
|
};
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
class COriginalitySorter
|
|
{
|
|
public:
|
|
|
|
typedef std::set< CGenRawMaterial* > CRMSet;
|
|
typedef std::multimap< uint32, CGenRawMaterial*, std::greater<uint32> > CMultiMapByOriginality;
|
|
|
|
///
|
|
COriginalitySorter() : RMByOriginalityByCraftSlot( NbFaberElements ) {}
|
|
|
|
///
|
|
void pushRM( CGenRawMaterial *rawMaterial )
|
|
{
|
|
RawMaterials.insert( rawMaterial );
|
|
|
|
//InfoLog->displayRawNL( "Inserting RM" );
|
|
std::list< CFaberCharacteristics >::const_iterator ilc;
|
|
for ( ilc=rawMaterial->RMCraftCharacs.begin(); ilc!=rawMaterial->RMCraftCharacs.end(); ++ilc )
|
|
{
|
|
//InfoLog->displayRawNL( " %u: %s orig=%u", (*ilc).FaberElement, rawMaterial->SheetName.c_str(), (uint32)((*ilc).ActualOriginality*100.0f) );
|
|
RMByOriginalityByCraftSlot[(*ilc).FaberElement].insert( make_pair( (uint32)((*ilc).ActualOriginality*100.0f), rawMaterial ) );
|
|
}
|
|
}
|
|
|
|
///
|
|
void popAndDeleteRM( CGenRawMaterial *rawMaterial )
|
|
{
|
|
delete rawMaterial;
|
|
RawMaterials.erase( rawMaterial );
|
|
}
|
|
|
|
///
|
|
bool alreadyPopped( CGenRawMaterial *rawMaterial ) const
|
|
{
|
|
return RawMaterials.find( rawMaterial ) == RawMaterials.end();
|
|
}
|
|
|
|
/// fromPos and the returned iterator are the pos internal to the COriginalitySorter RM set
|
|
/*CRMSet::const_iterator getFirstRMNotInFamilyListFromPos( const set<uint>& familyList, TStatQuality statQuality, CRMSet::const_iterator fromPos ) const
|
|
{
|
|
CRMSet::const_iterator irm;
|
|
for ( irm=fromPos; irm!=RawMaterials.end(); ++irm )
|
|
{
|
|
if ( ((*irm)->StatQuality == statQuality) &&
|
|
(familyList.find( (*irm)->IFamily ) == familyList.end()) )
|
|
return irm;
|
|
}
|
|
return RawMaterials.end();
|
|
}*/
|
|
|
|
///
|
|
CRMSet::iterator getRMSetBegin() const { return RawMaterials.begin(); }
|
|
|
|
///
|
|
CRMSet::iterator getRMSetEnd() const { return RawMaterials.end(); }
|
|
|
|
///
|
|
void deleteAllRemainingRM()
|
|
{
|
|
CRMSet::iterator irm;
|
|
for ( irm=RawMaterials.begin(); irm!=RawMaterials.end(); ++irm )
|
|
{
|
|
delete (*irm);
|
|
}
|
|
RawMaterials.clear();
|
|
// Does not clear the maps by originality
|
|
}
|
|
|
|
std::vector< CMultiMapByOriginality > RMByOriginalityByCraftSlot;
|
|
|
|
private:
|
|
|
|
CRMSet RawMaterials;
|
|
};
|
|
|
|
|
|
#define checkColor( c ) nlassert( colors[c] == #c );
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void loadDFNs( UFormLoader *formLoader )
|
|
{
|
|
map<string, CDfnFieldInfo> dfnFields;
|
|
NLMISC::CSmartPtr<UFormDfn> formDfn;
|
|
formDfn = formLoader->loadFormDfn( (rmSheetType + ".dfn").c_str() );
|
|
if ( ! formDfn )
|
|
nlerror( "Can't find DFN for %s", rmSheetType.c_str() );
|
|
vector<string> craftPartsPaths;
|
|
fillFromDFN( formLoader, dfnFields, formDfn, "", rmSheetType, "mp.MpParam", craftPartsPaths );
|
|
|
|
// Get craft parts
|
|
CraftParts.getNamesAndPaths( craftPartsPaths );
|
|
|
|
formDfn = formLoader->loadFormDfn( (crSheetType + ".dfn").c_str() );
|
|
if ( ! formDfn )
|
|
nlerror( "Can't find DFN for %s", crSheetType.c_str() );
|
|
fillFromDFN( formLoader, dfnFields, formDfn, "", crSheetType );
|
|
|
|
// Get lists of predefined values from sitem and creature DFN
|
|
families = dfnFields["mp.Family"].TypePredefinedLabels;
|
|
familyCodes = dfnFields["mp.Family"].TypePredefinedValues;
|
|
groups = dfnFields["mp.Group"].TypePredefinedLabels;
|
|
|
|
//properties = dfnFields["mp.Material property 1"].TypePredefinedLabels;
|
|
//nlverify( removeEntryFromList( properties, "Undefined" ) != ~0 );
|
|
properties.resize( craftPartsPaths.size() );
|
|
for ( uint i=0; i!=properties.size(); ++i ) // now, use properties as item part list
|
|
properties[i] = string( 1, (char)('A' + i) );
|
|
|
|
ecosystems = dfnFields["mp.Ecosystem"].TypePredefinedLabels,
|
|
colors = dfnFields["mp.MpColor"].TypePredefinedLabels,
|
|
creatures = dfnFields["Basics.Race"].TypePredefinedLabels;
|
|
seasons.push_back( "Winter" );
|
|
seasons.push_back( "Spring" );
|
|
seasons.push_back( "Summer" );
|
|
seasons.push_back( "Autumn" );
|
|
//removeEntryFromList( families, "Undefined" );
|
|
//removeEntryFromList( familyCodes, "0" );
|
|
nlverify( removeEntryFromList( ecosystems, "unknown" ) != ~0 );
|
|
nlverify( removeEntryFromList( ecosystems, "Goo" ) != ~0 );
|
|
nlassert( ecosystems[0] == "Common" );
|
|
nlassert( ecosystems[1] == "Desert" ); // ensure we match with enum TEcosystem!
|
|
nlassert( ecosystems[2] == "Forest" );
|
|
nlassert( ecosystems[3] == "Lacustre" );
|
|
nlassert( ecosystems[4] == "Jungle" );
|
|
nlassert( ecosystems[5] == "PrimeRoots" );
|
|
//removeEntryFromList( ecosystems, "Common" ); // TODO
|
|
nlassert( NbEcosystems == ecosystems.size() );
|
|
nlverify( removeEntryFromList( colors, "None" ) != ~0 );
|
|
nlverify( removeEntryFromList( colors, "UserColor") != ~0 );
|
|
nlassert( colors.size() == NbColors );
|
|
checkColor( Red );
|
|
checkColor( Beige );
|
|
checkColor( Green );
|
|
checkColor( Turquoise );
|
|
checkColor( Blue );
|
|
checkColor( Violet );
|
|
checkColor( White );
|
|
checkColor( Black );
|
|
|
|
/*UndefinedProperty = getIndexFromString( "Undefined", properties );
|
|
nlassert( UndefinedProperty != ~0 );*/
|
|
}
|
|
|
|
|
|
/*
|
|
* Build RMFamilyIndicesByCreatureModel and DepositFamilyIndices
|
|
*/
|
|
void dispatchFamiliesToLocations()
|
|
{
|
|
for ( CFamMap::iterator iss=FamSet.begin(); iss!=FamSet.end(); ++iss )
|
|
{
|
|
const CSString& famStr = (*iss).first;
|
|
TFamInfo& famInfo = (*iss).second;
|
|
uint iFam = getIndexFromString( famStr, families );
|
|
if ( famInfo.IsInDeposits )
|
|
{
|
|
// Deposits
|
|
nlassert( iFam != ~0 );
|
|
DepositFamilyIndices.push_back( iFam );
|
|
}
|
|
if ( famInfo.IsInCreatures )
|
|
{
|
|
// Extract creature name from left of family name (ASSUMES there's no blank in creature name)
|
|
CSString creaNameForRMFamily = famStr.splitTo( ' ' );
|
|
|
|
// Dispatch
|
|
for ( CSkeletonMap::iterator icm=CreatureModels.begin(); icm!=CreatureModels.end(); ++icm )
|
|
{
|
|
const CSString& creaModel = (*icm).first;
|
|
TSkeletonInfo& modelInfo = (*icm).second;
|
|
|
|
if ( modelInfo.Name == creaNameForRMFamily )
|
|
{
|
|
RMFamilyIndicesByCreatureModel[creaModel].push_back( iFam );
|
|
//nlinfo( "+ %s for %s (now %u models registered)", famStr.c_str(), creaModel.c_str(), RMFamilyIndicesByCreatureModel.size() );
|
|
modelInfo.IsUsed = true; // Name and AbbrevName are set by deliverCreatureModels()
|
|
}
|
|
}
|
|
}
|
|
else switch ( famInfo.SpecialCreatureTag[0] )
|
|
{
|
|
// Goo & invasion/raid creatures
|
|
case 'G':
|
|
{
|
|
GooCreatureFamilyIndices.push_back( iFam );
|
|
nldebug( "Family %s selected for goo creatures", famStr.c_str() );
|
|
break;
|
|
}
|
|
case 'I':
|
|
{
|
|
if ( famInfo.SpecialCreatureTag.size() == 1 )
|
|
{
|
|
InvasionRaidCreatureFamilyIndices['*'].push_back( iFam );
|
|
nldebug( "Family %s selected for all invasion creatures", famStr.c_str() );
|
|
}
|
|
else
|
|
{
|
|
for ( uint c=1; c!=famInfo.SpecialCreatureTag.size(); ++c )
|
|
{
|
|
InvasionRaidCreatureFamilyIndices[famInfo.SpecialCreatureTag[c]].push_back( iFam );
|
|
nldebug( "Family %s selected for invasion creature of type %c", famStr.c_str(), famInfo.SpecialCreatureTag[c] );
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Returns the number of models used
|
|
*/
|
|
uint checkSkeletons()
|
|
{
|
|
uint32 nbSkeUsed = 0;
|
|
for ( CSkeletonMap::const_iterator isc=CreatureModels.begin(); isc!=CreatureModels.end(); ++isc )
|
|
{
|
|
const string& skeFilename = (*isc).first;
|
|
const TSkeletonInfo& ske = (*isc).second;
|
|
const bool& used = (*isc).second.IsUsed;
|
|
if ( used )
|
|
{
|
|
nldebug( "Model %s (%s) %s", skeFilename.c_str(), ske.AbbrevName.c_str(), used?"used":"NOT USED" );
|
|
++nbSkeUsed;
|
|
}
|
|
else
|
|
nlwarning( "Model %s %s", skeFilename.c_str(), used?"":"NOT USED" );
|
|
}
|
|
return nbSkeUsed;
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*/
|
|
void createDirectoryStructure()
|
|
{
|
|
// Create the directory structure
|
|
if ( WriteSheetsToDisk )
|
|
{
|
|
for ( uint32 i=0; i!=ecosystems.size(); ++i )
|
|
{
|
|
string dirname = conventionalDirectory( ecosystems[i] );
|
|
if ( ! CFile::isExists( rawMaterialPath + dirname ) )
|
|
{
|
|
CFile::createDirectory( rawMaterialPath + dirname );
|
|
}
|
|
else
|
|
{
|
|
if ( ! CFile::isDirectory( rawMaterialPath + dirname ) )
|
|
{
|
|
nlwarning( "%s already existing but not a directory!", (rawMaterialPath + dirname).c_str() );
|
|
}
|
|
}
|
|
}
|
|
string dirname = "_parent";
|
|
if ( ! CFile::isExists( rawMaterialPath + dirname ) )
|
|
{
|
|
CFile::createDirectory( rawMaterialPath + dirname );
|
|
}
|
|
else
|
|
{
|
|
if ( ! CFile::isDirectory( rawMaterialPath + dirname ) )
|
|
{
|
|
nlwarning( "%s already existing but not a directory!", (rawMaterialPath + dirname).c_str() );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
#endif // NL_SRG_UTILITIES_H
|
|
|
|
/* End of srg_utilities.h */
|