khanat-opennel-code/code/ryzom/tools/leveldesign/mission_compiler_lib/mission_compiler.cpp

2283 lines
59 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/>.
#include "mission_compiler.h"
#include "step.h"
#include "nel/misc/i18n.h"
#include "nel/misc/common.h"
#include "nel/ligo/primitive_utils.h"
using namespace std;
using namespace NLMISC;
using namespace NLLIGO;
// hack to get access to string manager item enumeration to string without including
// almost all of the Ryzom server side project
namespace STRING_MANAGER
{
NL_BEGIN_STRING_CONVERSION_TABLE (TParamType)
NL_STRING_CONVERSION_TABLE_ENTRY( item )
NL_STRING_CONVERSION_TABLE_ENTRY( place )
NL_STRING_CONVERSION_TABLE_ENTRY( creature )
NL_STRING_CONVERSION_TABLE_ENTRY( skill )
NL_STRING_CONVERSION_TABLE_ENTRY( role )
NL_STRING_CONVERSION_TABLE_ENTRY( ecosystem )
NL_STRING_CONVERSION_TABLE_ENTRY( race )
NL_STRING_CONVERSION_TABLE_ENTRY( sbrick )
NL_STRING_CONVERSION_TABLE_ENTRY( faction )
NL_STRING_CONVERSION_TABLE_ENTRY( guild )
NL_STRING_CONVERSION_TABLE_ENTRY( player )
NL_STRING_CONVERSION_TABLE_ENTRY( bot )
{ "int", integer},
// NL_STRING_CONVERSION_TABLE_ENTRY( integer )
NL_STRING_CONVERSION_TABLE_ENTRY( time )
NL_STRING_CONVERSION_TABLE_ENTRY( money )
NL_STRING_CONVERSION_TABLE_ENTRY( compass )
NL_STRING_CONVERSION_TABLE_ENTRY( string_id )
NL_STRING_CONVERSION_TABLE_ENTRY( dyn_string_id )
NL_STRING_CONVERSION_TABLE_ENTRY( self )
NL_STRING_CONVERSION_TABLE_ENTRY( creature_model )
NL_STRING_CONVERSION_TABLE_ENTRY( entity )
NL_STRING_CONVERSION_TABLE_ENTRY( body_part )
NL_STRING_CONVERSION_TABLE_ENTRY( score )
NL_STRING_CONVERSION_TABLE_ENTRY( sphrase )
NL_STRING_CONVERSION_TABLE_ENTRY( characteristic )
NL_STRING_CONVERSION_TABLE_ENTRY( damage_type )
NL_STRING_CONVERSION_TABLE_ENTRY( bot_name)
NL_STRING_CONVERSION_TABLE_ENTRY( power_type )
NL_STRING_CONVERSION_TABLE_ENTRY( literal )
NL_END_STRING_CONVERSION_TABLE(TParamType, ParamTypeConversion, NB_PARAM_TYPES)
//-----------------------------------------------
// stringToParamType
//-----------------------------------------------
TParamType stringToParamType( const std::string & str )
{
return ParamTypeConversion.fromString( str );
}
//-----------------------------------------------
// stringToParamType
//-----------------------------------------------
const std::string & paramTypeToString( TParamType type )
{
return ParamTypeConversion.toString( type );
}
}
// utility to 'tabulate' the lines in a string
void tabulateLine(std::string &text, uint nbTabs)
{
if (text.empty())
return;
string::size_type pos = 0;
string tabs;
for (uint i=0; i<nbTabs; ++i)
tabs += "\t";
// add a tab at start
text = tabs + text;
// add a tab at each new line
while ((pos = text.find('\n', pos)) != string::npos)
{
if (pos < text.size()-1 && text[pos+1] == '\r')
{
// add after the '\r' char
++pos;
}
if (pos < text.size()-1)
text = text.substr(0, pos+1) + tabs + text.substr(pos+1);
}
}
class GenderExtractor
{
public:
GenderExtractor(const std::string & literal, const std::string& identifier, unsigned int level=0);
std::string operator()(unsigned int i) const;
unsigned int size() const;
~GenderExtractor();
private:
bool extractMarkup(const std::string& literal, const std::string & markup, std::string &before, std::string &inside, std::string & after);
bool parseMarkup(const std::string& literal, const std::string & markup, std::string& newPhrase, bool include = false );
std::string getPhrase(unsigned int i) const;
std::string getExtension(unsigned int i) const;
std::string getCondition(unsigned int i) const;
std::string getEntity(unsigned int i) const;
std::string getIdentifier(unsigned int i) const;
private:
bool _Entity;
std::string _EntityName;
std::string _Text;
std::string _Identifier;
GenderExtractor* _Female;
GenderExtractor* _Male;
};
GenderExtractor::~GenderExtractor()
{
delete _Female;
delete _Male;
}
std::string GenderExtractor::getIdentifier(unsigned int i) const
{
return _Identifier + getExtension(i);
}
std::string GenderExtractor::operator()(unsigned int i) const
{
std::string ret("\t");
std::string condition = getCondition(i);
ret += condition.empty() ? "":std::string("( ") + condition + " )" + NL + "\t\t";
ret += getIdentifier(i) + "\t[" + getPhrase(i) + "]" + NL;
return ret;
}
GenderExtractor::GenderExtractor(const std::string & literal, const std::string& identifier, unsigned int level)
{
static const char * es[] ={"e", "e1", "e2", "e3"};
static const char * fs[] ={"f", "f1", "f2", "f3"};
static const char * hs[] ={"h", "h1", "h2", "h3"};
const char * e = es[level];
const char * f = fs[level];
const char * h = hs[level];
_Identifier = toLower(identifier);
std::string newPhrase;
std::string before;
std::string after;
std::string femaleText;
std::string maleText;
_Entity = extractMarkup(literal, e, before, _EntityName, after);
if (_EntityName.size() > 2)
{
if (_EntityName[0] == '$' && _EntityName[_EntityName.size() - 1] == '$')
{
_EntityName = _EntityName.substr(1, _EntityName.size() - 2);
}
}
std::string newLiteral = before + after;
bool isFemale = parseMarkup(newLiteral,f,newPhrase, true);
if ( isFemale)
{
parseMarkup(newPhrase,h,newPhrase, false);
femaleText = newPhrase;
}
bool isMale = parseMarkup(newLiteral, h, newPhrase, true);
if (isMale)
{
parseMarkup(newPhrase, f, newPhrase, false);
maleText = newPhrase;
}
if (isMale != isFemale)
{
std::string goodMarkup = isMale ? std::string("") +"<" + h + "></" + h + ">" : std::string("")+"<"+f+"></"+f+">";
std::string badMarkup = isFemale ? std::string("") +"<" + h + "></" + h + ">" : std::string("")+"<"+f+"></"+f+">";
std::string exceptionText = std::string("Expression ") + identifier + " that contains a tag " + goodMarkup + " needs also tags " + badMarkup + " even empty.";
throw EParseException(0, exceptionText.c_str());
}
if (!isMale && !isFemale)
{
_Text = literal;
_Female = 0;
_Male = 0;
}
else
{
if (!_Entity) { _EntityName = "self"; }
_Female = new GenderExtractor(femaleText, identifier, level+1);
_Male = new GenderExtractor(maleText, identifier, level+1);
}
}
bool GenderExtractor::extractMarkup(const std::string& literal, const std::string & markup, std::string &before, std::string &inside, std::string & after)
{
std::string::size_type posBegin;
std::string::size_type posEnd;
std::string::size_type posInside;
std::string beginMarkup = std::string("<") + markup + std::string(">");
std::string endMarkup = std::string("</") + markup + std::string(">");
posBegin = literal.find(beginMarkup);
if ( posBegin != std::string::npos )
{
posEnd = literal.find(endMarkup, posBegin + beginMarkup.size());
if (posEnd != std::string::npos)
{
before = literal.substr(0, posBegin);
posInside = posBegin + beginMarkup.size();
inside = literal.substr(posInside, posEnd - posInside);
after = literal.substr(posEnd+endMarkup.size());
return true;
}
}
after = literal;
return false;
}
bool GenderExtractor::parseMarkup(const std::string& literal, const std::string & markup, std::string& newPhrase, bool include )
{
bool markupExist;
bool changed = false;
std::string oldPhrase = literal;
newPhrase.clear();
do
{
std::string before;
std::string inside;
std::string after;
markupExist = extractMarkup(oldPhrase, markup, before, inside, after);
newPhrase += before;
if (include){ newPhrase += inside; }
oldPhrase = after;
if (markupExist){ changed = true; }
} while(markupExist);
newPhrase += oldPhrase;
return changed;
}
std::string GenderExtractor::getPhrase(unsigned int i) const
{
if ( i%2 == 0) { return _Male ? _Male->getPhrase(i/2) : _Text; }
if ( i%2 == 1) { nlassert(_Female); return _Female->getPhrase(i/2);}
nlassert(0);
return "";
}
std::string GenderExtractor::getExtension(unsigned int i) const
{
if ( i%2 == 0) { return _Male ? std::string("_m") + _Male->getExtension(i/2) : ""; }
if ( i%2 == 1) { nlassert(_Female); return std::string("_f") + _Female->getExtension(i/2);}
nlassert(0);
return "";
}
std::string GenderExtractor::getCondition(unsigned int i) const
{
//if ( i%2 == 0) { return _Male ? std::string("\t(") + _Male->getExtension(i/2) : "\t"; }
//if ( i%2 == 1) { nlassert(_Female); return std::string("_f") + _Female->getExtension(i/2);}
if ( i%2 == 0)
{
if (_Male)
{
std::string next = _Male->getCondition(i/2);
std::string current = _EntityName + ".gender = male";
return next.size() ? current + " & " + next : current;
}
else
{
return "";
}
}
if ( i%2 == 1)
{
std::string next = _Female->getCondition(i/2);
std::string current = _EntityName + ".gender = female";
return next.size() ? current + " & " + next : current;
}
nlassert(0);
return "";
}
unsigned int GenderExtractor::size() const
{
return _Male ? _Male->size() + _Female->size(): 1;
}
string CPhrase::genPhrase()
{
string ret;
if (!_PhraseLiterals.empty())
{
for (uint p=0; p<_PhraseLiterals.size(); ++p)
{
string identifier = _PhraseId;
if (_NumEntry != 0)
identifier += toString("_%u", p+1);
GenderExtractor gender(_PhraseLiterals[p], identifier, 0);
ret += identifier + " (";
// generate default param list
if (_DefaultParams.size() > p)
{
for (uint i=0; i<_DefaultParams[p].size(); ++i)
{
ret += STRING_MANAGER::paramTypeToString(_DefaultParams[p][i].ParamType) + " "+_DefaultParams[p][i].ParamName;
if (i != _DefaultParams[p].size()-1 || !_AdditionalParams.empty())
ret += ", ";
}
}
// generate additional param list
for (uint i=0; i<_AdditionalParams.size(); ++i)
{
ret += STRING_MANAGER::paramTypeToString(_AdditionalParams[i].ParamType) + " "+_AdditionalParams[i].ParamName;
if (i != _AdditionalParams.size()-1)
ret += ", ";
}
ret += ")" + NL;
ret += "{" + NL;
for (unsigned int i = 0; i < gender.size(); ++i)
{
ret += gender(i);
}
ret += "}" + NL + NL;
}
}
nlinfo("genphrase: %s", ret.c_str());
return ret;
}
bool CMissionCompiler::generateDotScript(NLLIGO::IPrimitive *missionPrim, std::string &dotScript, std::string &log)
{
//assume that the mission is compiled in the last compiled mission slot
try
{
if (compileMission(missionPrim, string()))
{
dotScript = _CompiledMission.back()->generateDotScript();
return true;
}
else
{
return false;
}
}
catch(const EParseException & e)
{
log = e.Why;
return false;
}
}
/*
bool CMissionCompiler::parseGlobalMissionData(IPrimitive *mission, CMissionData &md)
{
// Mission name
string *s;
if (!mission->getPropertyByName("name", s) || s->empty())
throw EParseException(mission, "missing mission name !");
md.setMissionName(*s);
// giver primitive file
if (!mission->getPropertyByName("giver_primitive", s) || s->empty())
throw EParseException(mission, "missing giver primitive !");
md.setGiverPrimitive(*s);
// giver name
if (!mission->getPropertyByName("giver_primitive", s) || s->empty())
throw EParseException(mission, "missing giver primitive !");
md.setGiverName(*s);
// If the mission is under a npc_bot node, then the giver is directly taken
// from the npc name
if (mission->getParent())
{
if (mission->getParent()->getPropertyByName("class", s) && *s == "npc_bot")
{
if (mission->getParent()->getPropertyByName("name", s))
md.setGiverName(*s);
}
}
// TODO : read all other params...
return true;
}
*/
void CMissionData::initHeaderPhrase(IPrimitive *prim)
{
CPhrase::TPredefParams params;
params.resize(1);
params[0].push_back(CPhrase::TParamInfo("giver", STRING_MANAGER::bot));
_MissionTitle.initPhrase(*this, prim, _MissionTitleRaw, 0, params);
_MissionDescription.initPhrase(*this, prim, _MissionDescriptionRaw, 0, params);
_MissionAutoMenu.initPhrase(*this, prim, _MissionAutoMenuRaw);
}
bool CMissionCompiler::compileMission(NLLIGO::IPrimitive *rootPrim, const std::string &primFileName)
{
TPrimitiveClassPredicate pred("mission_tree");
if (!pred(rootPrim))
return false;
IPrimitive *mission = rootPrim;
CMissionData *pmd = new CMissionData;
CMissionData &md = *pmd;
// Read the mission name
string missionName = md.getProperty(mission, "name", false, false);
if( missionName.find(' ') != string::npos)
{
throw EParseException(mission, toString("Mission name '%s' must not contains space", missionName.c_str()).c_str());
}
md.setMissionName(missionName);
// Create a temporary primitive node to create default variable
{
// giver default var
IPrimitive *temp = new CPrimNode();
temp->addPropertyByName("class", new CPropertyString("var_npc"));
temp->addPropertyByName("name", new CPropertyString("giver = giver"));
temp->addPropertyByName("npc_name", new CPropertyString("giver"));
temp->addPropertyByName("var_name", new CPropertyString("giver"));
IVar *var = IVar::createVar(md, temp);
md.addVariable(NULL, var);
delete temp;
}
{
// player default var
IPrimitive *temp = new CPrimNode();
temp->addPropertyByName("class", new CPropertyString("var_npc"));
temp->addPropertyByName("name", new CPropertyString("player = player"));
temp->addPropertyByName("npc_name", new CPropertyString("player"));
temp->addPropertyByName("var_name", new CPropertyString("player"));
IVar *var = IVar::createVar(md, temp);
md.addVariable(NULL, var);
delete temp;
}
{
// guild_name default var
IPrimitive *temp = new CPrimNode();
temp->addPropertyByName("class", new CPropertyString("var_text"));
temp->addPropertyByName("name", new CPropertyString("guild_name = guild_name"));
temp->addPropertyByName("npc_name", new CPropertyString("guild_name"));
temp->addPropertyByName("var_name", new CPropertyString("guild_name"));
IVar *var = IVar::createVar(md, temp);
md.addVariable(NULL, var);
delete temp;
}
// first, start by reading mission variables
IPrimitive *variables;
{
TPrimitiveClassPredicate predTmp("variables");
variables= NLLIGO::getPrimitiveChild(mission, predTmp);
}
if (!variables)
{
nlwarning("Can't find variables !");
return false;
}
parseVariables(md, variables);
// read global mission data
md.parseMissionHeader(rootPrim);
// now, we can init the mission header phrase (they need variable knwoled)
md.initHeaderPhrase(rootPrim);
IPrimitive *preReq;
{
TPrimitiveClassPredicate predTmp("pre_requisite");
preReq = getPrimitiveChild(mission, predTmp);
}
if (!preReq)
{
nlwarning("Can't find pre requisite !");
return false;
}
parsePreRequisite(md, preReq);
/* IPrimitive *steps = getPrimitiveChild(mission, TPrimitivePropertyPredicate("step_tag", "true"));
if (!steps)
{
nlwarning("Can't find steps !");
return false;
}
*/ parseSteps(md, mission);
// Store the compiled mission
_CompiledMission.push_back(pmd);
string script = md.generateMissionScript(primFileName);
nlinfo("The script :");
nlinfo("%s", script.c_str());
string phrases = md.generatePhraseFile();
nlinfo("The phrase file is :");
{
vector<string> lines;
explode(phrases, string("\n"), lines, false);
for (uint i=0; i<lines.size(); ++i)
{
if(lines[i][0] == '\r') lines[i] = lines[i].substr(1);
nlinfo("%s", lines[i].c_str());
}
}
string dot = md.generateDotScript();
nlinfo("The dot script is :");
{
vector<string> lines;
explode(dot, string("\n"), lines, false);
for (uint i=0; i<lines.size(); ++i)
{
if(lines[i][0] == '\r') lines[i] = lines[i].substr(1);
nlinfo("%s", lines[i].c_str());
}
}
return true;
}
bool CMissionCompiler::compileMissions(IPrimitive *rootPrim, const std::string &primFileName)
{
bool ret = true;
// 1st, build a set of mission_scrip nodes
NLLIGO::TPrimitiveSet missionTrees;
CPrimitiveSet<TPrimitiveClassPredicate> scriptsSet;
TPrimitiveClassPredicate pred("mission_tree");
scriptsSet.buildSet(rootPrim, pred, missionTrees);
nlinfo("Found %u mission tree in the primitive file", missionTrees.size());
for (uint i=0; i<missionTrees.size(); ++i)
{
// try
// {
compileMission(missionTrees[i], primFileName);
// }
// catch (const EParseException &e)
// {
// nlwarning("Error while parsing a mission: '%s'", e.Why.c_str());
// ret = false;
// }
}
return ret;
}
bool CMissionCompiler::installCompiledMission(NLLIGO::CLigoConfig &ligoConfig, const std::string &primFileName)
{
// generate the mission script into the npcs...
{
map<string, TLoadedPrimitive > loadedPrimitives;
// store the previous alias value
map<string, uint32> missionAlias;
// First loop to remove any mission that belong to the compiled primitive file
for (uint i=0; i<_CompiledMission.size(); ++i)
{
CMissionData &mission = *(_CompiledMission[i]);
// first, look for the primitive file to load
string fileName = mission.getGiverPrimitive();
if (fileName.empty())
{
// use mission primitive instead
fileName = primFileName;
}
if (loadedPrimitives.find(toLower(fileName)) == loadedPrimitives.end())
{
string fullFileName = CPath::lookup(fileName, false);
if (fullFileName.empty())
{
throw EParseException(NULL, toString("Can't find primitive file '%s' in path", fileName.c_str()).c_str());
}
// we need to load this primitive file.
CPrimitives *primDoc = new CPrimitives;
CPrimitiveContext::instance().CurrentPrimitive = primDoc;
if (loadXmlPrimitiveFile(*primDoc, fullFileName, ligoConfig))
{
// the primitive file is loaded correctly
loadedPrimitives.insert(make_pair(toLower(fileName), TLoadedPrimitive(primDoc, fullFileName)));
CPrimitiveContext::instance().CurrentPrimitive = NULL;
}
else
{
CPrimitiveContext::instance().CurrentPrimitive = NULL;
throw EParseException(NULL, toString("Can't read primitive file '%s'", fullFileName.c_str()).c_str());
}
}
TLoadedPrimitive &loadedPrim = loadedPrimitives[toLower(fileName)];
CPrimitives *primDoc = loadedPrim.PrimDoc;
TPrimitiveSet scripts;
CPrimitiveSet<TPrimitiveClassPredicate> filter;
TPrimitiveClassPredicate pred("mission");
filter.buildSet(primDoc->RootNode, pred, scripts);
// for each script, check if it was generated, and if so, check the name
// of the source primitive file.
for (uint i=0; i<scripts.size(); ++i)
{
vector<string> *script;
if (scripts[i]->getPropertyByName("script", script) && !script->empty())
{
string missionName;
scripts[i]->getPropertyByName("name", missionName);
// Format should be : #compiled from <source_primitive_name>
if (script->front().find("generated from") != string::npos)
{
// we have a compiled mission
if (script->front().find(CFile::getFilename(primFileName)) != string::npos)
{
// ok, this mission is compiled from the same primitive
// store it's alias
TPrimitiveClassPredicate pred("alias");
IPrimitive *p = getPrimitiveChild(scripts[i], pred);
if (p)
{
CPrimAlias *pa = dynamic_cast<CPrimAlias*>(p);
if (pa)
{
uint32 alias = pa->getAlias();
missionAlias.insert(make_pair(missionName, alias));
}
}
else
{
nlwarning("Can't find alias prim in primitive '%s'", buildPrimPath(scripts[i]).c_str());
}
// and remove it
scripts[i]->getParent()->removeChild(scripts[i]);
}
}
}
}
}
// second loop to assign compiled mission to giver npc
for (uint i=0; i<_CompiledMission.size(); ++i)
{
CMissionData &mission = *(_CompiledMission[i]);
string fileName = mission.getGiverPrimitive();
if (fileName.empty())
{
// no giver primitive file specified in the mission, use the mission primitive instead
fileName = primFileName;
}
TLoadedPrimitive &loadedPrim = loadedPrimitives[toLower(fileName)];
CPrimitives *primDoc = loadedPrim.PrimDoc;
CPrimitiveContext::instance().CurrentPrimitive = primDoc;
TPrimitiveSet bots;
CPrimitiveSet<TPrimitiveClassAndNamePredicate> filter;
TPrimitiveClassAndNamePredicate pred("npc_bot", mission.getGiverName());
filter.buildSet(primDoc->RootNode, pred, bots);
if (bots.empty())
{
string err = toString("Can't find bot '%s' in primitive '%s' !",
mission.getGiverName().c_str(),
fileName.c_str());
throw EParseException(NULL, err.c_str());
}
else if (bots.size() > 1)
{
string err = toString("Found more than one bot named '%s' in primitive '%s' !",
mission.getGiverName().c_str(),
fileName.c_str());
throw EParseException(NULL, err.c_str());
}
// ok, all is good, we can add the mission node to the giver
IPrimitive *giver = bots.front();
// create a new node for the mission
IPrimitive *script = new CPrimNode;
// set the class
script->addPropertyByName("class", new CPropertyString("mission"));
// set the name
script->addPropertyByName("name", new CPropertyString(mission.getMissionName()));
// string alias(toString("%u", makeHash32(mission.getMissionName())));
// script->addPropertyByName("alias", new CPropertyString(mission.getAlias()));
string scriptLines = mission.generateMissionScript(primFileName);
vector<string> lines;
explode(scriptLines, NL, lines, false);
script->addPropertyByName("script", new CPropertyStringArray(lines));
// insert the script into the giver
giver->insertChild(script);
// add the alias
{
CPrimAlias *pa = new CPrimAlias;
pa->addPropertyByName("class", new CPropertyString ("alias"));
pa->addPropertyByName("name", new CPropertyString ("alias"));
if (missionAlias.find(mission.getMissionName()) != missionAlias.end())
{
// restore the previous alias
primDoc->forceAlias(pa, missionAlias.find(mission.getMissionName())->second);
}
// insert in first place
script->insertChild(pa, 0);
}
CPrimitiveContext::instance().CurrentPrimitive = NULL;
}
// Save the modified primitive files
while (!loadedPrimitives.empty())
{
TLoadedPrimitive &loadedPrim = loadedPrimitives.begin()->second;
if (!saveXmlPrimitiveFile(*(loadedPrim.PrimDoc), loadedPrim.FullFileName))
return false;
_FilesToPublish.push_back(loadedPrim.FullFileName);
// Free the memory
delete loadedPrim.PrimDoc;
loadedPrimitives.erase(loadedPrimitives.begin());
}
}
// generate the phrase file (if any)
{
string phraseFileName = CFile::getFilenameWithoutExtension(primFileName) + "_wk.txt";
CSString content;
for (uint i=0; i<_CompiledMission.size(); ++i)
{
content += _CompiledMission[i]->generatePhraseFile();
}
// transform NL (\n\r) into single \n
content = content.replace(NL.c_str(), "\n");
ucstring ucs;
ucs.fromUtf8(content);
CI18N::writeTextFile(phraseFileName, ucs, true);
_FilesToPublish.push_back(phraseFileName);
}
return true;
}
bool CMissionCompiler::publishFiles(const std::string &serverPathPrim, const std::string &serverPathText, const std::string &localPathText)
{
for (uint i=0 ; i<_FilesToPublish.size() ; i++)
{
string dst, src = _FilesToPublish[i];
string::size_type n = src.find("primitives");
if (n == string::npos)
{
// text files : copy it and check include in phrase_rites_wk.txt
// server
string textFile = CPath::standardizePath(serverPathText) + "phrase_rites_wk.txt";
includeText(textFile, string("#include \"") + src + string("\"\n"));
dst = CPath::standardizePath(serverPathText) + src;
NLMISC::CFile::copyFile(dst, src);
// local
textFile = CPath::standardizePath(localPathText) + "phrase_rites_wk.txt";
includeText(textFile, string("#include \"") + src + string("\"\n"));
dst = CPath::standardizePath(localPathText) + src;
NLMISC::CFile::copyFile(dst, src);
}
else
{
// primitive file : copy to server
dst = CPath::standardizePath(serverPathPrim) + string(src, n, src.size());
NLMISC::CFile::copyFile(dst, src);
}
}
return true;
}
bool CMissionCompiler::includeText(const std::string filename, const std::string text)
{
FILE *f = nlfopen(filename, "r+");
if (f == NULL)
return false;
bool isIn = false;
char buffer[1024];
// Check for UTF8 format
fread(buffer, 1, 3, f);
if (buffer[0] != -17 || buffer[1] != -69 || buffer[2] != -65)
fseek(f, 0, SEEK_SET);
// Compare each line
while(fgets(buffer, 1024, f))
{
if (!strcmp(text.c_str(), buffer))
{
isIn = true;
break;
}
}
if (!isIn)
fputs(text.c_str(), f);
fclose(f);
return true;
}
bool CMissionCompiler::parsePreRequisite(CMissionData &md, IPrimitive *preReq)
{
md.parsePrerequisites(preReq);
return true;
}
bool CMissionCompiler::parseOneStep(CMissionData &md, IPrimitive *stepToParse, IStep *parent, bool bEndOfBranch)
{
IStep *step = IStep::createStep(md, stepToParse);
if (step != NULL)
{
if (!step->isAJump() && !step->getStepName().empty())
{
if (md.getStepByName(step->getStepName()) != NULL)
{
string err = toString("Step '%s' already defined !", step->getStepName().c_str());
throw EParseException(step->getPrimitive(), err.c_str());
}
if (step->getStepName().find(' ') != string::npos)
{
throw EParseException(step->getPrimitive(), toString("Step name '%s' must not contains space", step->getStepName().c_str()).c_str());
}
md.addStepName(step->getStepName(), step);
}
TPrimitiveSet subBranchs = step->getSubBranchs();
// Add the step (if no parent add to the mission data)
if (parent == NULL)
{
if (!md.addStep(step))
{
throw EParseException(stepToParse, "Error parsing mission step");
}
}
else
{
parent->addSubStep(step);
}
CStepIf *pSI = dynamic_cast<CStepIf *>(step);
// If this is a IF step : parse with 'step' as a parent
IStep *pParentStep = NULL;
if ((dynamic_cast<CStepIf*>(step) != NULL) ||
(dynamic_cast<CStepPlayerReconnect*>(step) != NULL))
pParentStep = step;
if (!subBranchs.empty())
{
// need to parse subbranch before continuing
for (uint i=0; i<subBranchs.size(); ++i)
{
if (!parseOneStep(md, subBranchs[i], pParentStep, i==(subBranchs.size()-1)))
return false;
}
}
// if this is the last step, flag it as such
step->EndOfBranch = bEndOfBranch;
}
return true;
}
bool CMissionCompiler::parseSteps(CMissionData &md, IPrimitive *steps, IStep *parent)
{
TPrimitiveSet childs;
TPrimitivePropertyPredicate pred("step_tag", "true");
filterPrimitiveChilds(steps, pred, childs);
if (childs.empty())
{
CPrimNode node;
node.addPropertyByName("class", new CPropertyString("end"));
node.addPropertyByName("name", new CPropertyString(""));
IStep *step = IStep::createStep(md, &node);
delete step;
// md.addStep(step);
}
if (!childs.empty())
{
for (uint i=0; i<childs.size(); ++i)
{
IPrimitive *child = childs[i];
parseOneStep(md, childs[i], NULL, i == (childs.size()-1));
}
}
return true;
}
string CMissionCompiler::getProp(IPrimitive *prim, const string &propName)
{
string s;
bool ret = prim->getPropertyByName(propName.c_str(), s);
if (!ret)
throw EParseException(prim, toString("Property %s does't exist", propName.c_str()).c_str());
return s;
}
string CMissionCompiler::getClass(IPrimitive *prim)
{
string className;
bool ret = prim->getPropertyByName("class", className);
nlassert(ret);
return className;
}
bool CMissionCompiler::parseVariables(CMissionData &md, IPrimitive *variables)
{
for (uint i=0; i<variables->getNumChildren(); ++i)
{
IPrimitive *child;
if (variables->getChild(child, i))
{
IVar *var = IVar::createVar(md, child);
if (var)
{
nldebug("Adding variable '%s' as type %u", var->getVarName().c_str(), var->getVarType());
md.addVariable(child, var);
}
}
}
return true;
}
template <class VectorType>
bool strtokquote(const string &src, VectorType &tokens)
{
enum TMode
{
read_blank,
read_token,
read_quoted
};
string temp;
TMode mode = read_blank;
for (uint i=0; i<src.size(); ++i)
{
switch (mode)
{
case read_blank:
if (src[i] != ' ' && src[i] != '\t' && src[i] != '\n' && src[i] != '\r')
{
// end of blank !
if (src[i] == '\"')
{
// begin of a quoted string
temp = "\"";
mode = read_quoted;
}
else
{
// begin of a token
temp.clear();
temp += src[i];
mode = read_token;
}
}
break;
case read_token:
if (src[i] == ' ' || src[i] == '\t' || src[i] == '\n' || src[i] == '\r' || src[i] == '\"')
{
// end of token
tokens.push_back(temp);
temp.clear();
--i;
mode = read_blank;
}
else
{
temp += src[i];
}
break;
case read_quoted:
if (src[i] == '\\')
{
// special treatment for escape command
if (i < src.size()-1)
{
temp += src[i];
temp += src[i+1];
// skip escaped char
i++;
}
else
{
nlwarning("Error parsing escape char in quoted string");
return false;
}
}
else if (src[i] != '\"')
{
// just add this char
temp += src[i];
}
else
{
// end of quoted string
temp += src[i];
tokens.push_back(temp);
temp.clear();
mode = read_blank;
}
break;
}
}
if (!temp.empty())
{
if (mode == read_quoted)
{
nlwarning("Missing closing quote at end of string while reading text in '%s'", src.c_str());
return false;
}
tokens.push_back(temp);
}
return true;
}
template <class VectorType>
bool strtokquote(const vector<string> &src, VectorType &tokens)
{
for (uint i=0; i<src.size(); ++i)
{
if (!strtokquote(src[i], tokens))
return false;
}
return true;
}
struct TFindParamPred : std::unary_function<CPhrase::TParamInfo, bool>
{
string Name;
TFindParamPred(const std::string &name)
: Name (name)
{}
bool operator() (const CPhrase::TParamInfo &paramInfo) const
{
return paramInfo.ParamName == Name;
}
};
bool CPhrase::isEmpty()
{
return _PhraseId.empty();
}
bool CPhrase::asAdditionnalParams()
{
return !_AdditionalParams.empty();
}
void CPhrase::initPhrase (CMissionData &md,
IPrimitive *prim,
const vector<string> &texts,
uint32 numEntry,
const TPredefParams &predefParams )
{
// nlassert(numEntry == predefParams.size());
// store the predefined/default parameters
_DefaultParams = predefParams;
// store the number of entry to generate (for literal with variant)
_NumEntry = numEntry;
numEntry = max(uint32(1), numEntry);
_PhraseLiterals.clear();
// _PhraseLiterals.resize(numEntry);
// first, concatenate the text vector
string text;
for (uint i=0; i<texts.size(); ++i)
{
text = text + texts[i];
if (i != texts.size() -1)
text += "\n";
}
nldebug("phrase text: %s", text.c_str());
CVectorSString tokens;
if (!strtokquote(text, tokens))
throw EParseException(prim, toString("failed to tokenize the string '%s'", text.c_str()).c_str());
if (tokens.empty())
// nothing to parse
return;
// storage for additional parameters
vector<string> params;
retry:
// ok, the string is parsed, now we can analyze it
// look at the first letter of the first token to determine the type of data we have
if (tokens[0][0] == '\"')
{
// we have a literal, so we must found numEntry literal, then a suffix tag for the phrase name
if (tokens.size() != numEntry +1)
throw EParseException(prim, toString("bad number of tokens in phrase : need %u (%u entries + 1 suffix), found %u\n(in : '%s')",
numEntry+1,
numEntry,
tokens.size(),
text.c_str()
).c_str());
_PhraseLiterals.resize(numEntry);
for (uint i=0; i<numEntry; ++i)
{
CSString text = tokens[i];
// remove quotation marks
text = text.leftCrop(1);
text = text.rightCrop(1);
// store the literal phrase value
_PhraseLiterals[i] = text;
// escape any ']' in the string
_PhraseLiterals[i] = CSString(_PhraseLiterals[i]).replace("]", "\\]");
// now, we can analyse the string content, looking for parameters replacement
while (text.contains('$'))
{
// 'advance' to replacement point
text = text.splitFrom('$');
if (!text.empty())
{
if (text[0] != '$')
{
if (!text.contains('$'))
throw EParseException(prim, "missing parameter closing tag '$'");
string::size_type paramStart = _PhraseLiterals[i].size() - text.size();
// ok, we found a parameter
CSString p = text.splitTo('$', true);
// remove any subpart access
p = p.splitTo('.');
if (i >= predefParams.size() || find_if(predefParams[i].begin(), predefParams[i].end(), TFindParamPred(static_cast<string&>(p))) == predefParams[i].end())
{
// this param is not in the predefined params list, add it to the optional params
params.push_back(p);
}
// remove any compiler param from the phrase literal
if (p.find("@") != string::npos)
{
string::size_type pos = _PhraseLiterals[i].find(p, paramStart);
if (pos != string::npos)
{
string::size_type pos2 = _PhraseLiterals[i].find("@", pos);
if (pos2 != string::npos)
{
while (pos2 < _PhraseLiterals[i].size()
&& _PhraseLiterals[i][pos2] != '.'
&& _PhraseLiterals[i][pos2] != '$')
{
_PhraseLiterals[i].erase(pos2, 1);
}
}
}
}
}
else
{
// this is an escaped $, skip it
text.leftCrop(1);
}
}
}
}
// last, read the suffix
_Suffixe = tokens.back();
// generate identifier
_PhraseId = toUpper(md.getMissionName()+"_"+_Suffixe);
set<string> ps;
// select only unique params
ps.insert(params.begin(), params.end());
vector<string> temp(ps.begin(), ps.end());
params.swap(temp);
}
else if (tokens[0][0] == '$')
{
// we have a variable substitution. Retrieve the var and recall init
// do the var replacement
CVectorSString tokens2;
tokens[0] = md.replaceVar(prim, tokens[0]);
if (!strtokquote(tokens[0], tokens2))
throw EParseException(prim, toString("failed to tokenize the string ('%s')", tokens[0].c_str()).c_str());
tokens2.insert(tokens2.end(), tokens.begin()+1, tokens.end());
tokens.swap(tokens2);
// and retry the decoding
goto retry;
}
else
{
// this should be a simple identifier, followed by any number of additional parameters
// do the var replacement
// tokens = md.replaceVar(prim, tokens);
// untagVar(tokens[0]);
// ok, now extract the phrase label and the additional parameters
_PhraseId = tokens[0];
for (uint i=1; i<tokens.size(); ++i)
{
untagVar(tokens[i]);
if (predefParams.empty() || find_if(predefParams[0].begin(), predefParams[0].end(), TFindParamPred(static_cast<string&>(tokens[i]))) == predefParams[0].end())
{
// this param is not in the predefined params list, add it to the optional params
params.push_back(tokens[i]);
}
}
}
// now, build the parameter list
vector<string>::iterator first(params.begin()), last(params.end());
for (; first != last; ++first)
{
string name, param;
vector<string> parts;
NLMISC::explode(*first, string("@"), parts, false);
if (parts.size() > 0)
name = parts[0];
if (parts.size() > 1)
param = parts[1];
const string &varName = name;
if (varName != "self")
{
IVar *var = md.getVariable(varName);
if (var == NULL)
{
string err = toString("Can't find variable '%s' referenced from a phrase",
name.c_str());
throw EParseException(prim, err.c_str());
}
TParamInfo pi;
pi.ParamName = name;
pi.CompilerParam = param;
pi.ParamType = var->getStringManagerType();
_AdditionalParams.push_back(pi);
}
}
}
std::string CPhrase::genScript(CMissionData &md)
{
std::string ret;
ret = _PhraseId;
for (uint i=0; i<_AdditionalParams.size(); ++i)
{
IVar *var = md.getVariable(_AdditionalParams[i].ParamName);
if (var == NULL)
{
string err = toString("Can't find variable named '%s' to generate phrase param", _AdditionalParams[i].ParamName.c_str());
throw EParseException(NULL, err.c_str());
}
ret += "; " + var->evalVar(_AdditionalParams[i].CompilerParam);
}
return ret;
}
CMissionData::CMissionData()
{
// init all datas
_MonoInstance = false;
_MissionAuto = false;
_RunOnce = false;
_Replayable = false;
_Solo = false;
_Guild = false;
_NotInJournal = false;
_AutoRemoveFromJournal = false;
_PlayerReplayTimer = 0;
_GlobalReplayTimer = 0;
_NotProposed = false;
_NonAbandonnable = false;
_NeedValidation = false;
_FailIfInventoryIsFull = false;
}
CMissionData::~CMissionData()
{
while (!_Variables.empty())
{
delete _Variables.begin()->second;
_Variables.erase(_Variables.begin());
}
while (!_Steps.empty())
{
delete _Steps.back();
_Steps.pop_back();
}
}
void CMissionData::setMissionName(const string &missionName)
{
_MissionName = missionName;
}
const string &CMissionData::getMissionName() { return _MissionName;}
bool CMissionData::addVariable(NLLIGO::IPrimitive *prim, IVar *var)
{
if (_Variables.find(var->getVarName()) != _Variables.end())
throw EParseException(prim, toString("Variable '%s' already defined !", var->getVarName().c_str()).c_str());
_Variables.insert(make_pair(var->getVarName(), var));
_VariablesOrder.push_back(var);
return true;
}
IVar *CMissionData::getVariable(const string &varName)
{
map<string, IVar*>::iterator it(_Variables.find(varName));
if (it != _Variables.end())
return it->second;
return NULL;
}
IStep *CMissionData::getNextStep(IStep *current)
{
for (uint i=0; i<_Steps.size(); ++i)
{
if (_Steps[i] == current && i < _Steps.size()-1)
return _Steps[i+1];
}
return NULL;
}
IStep *CMissionData::getStepByName(const std::string &stepName)
{
if (_StepsByNames.find(stepName) != _StepsByNames.end())
{
return _StepsByNames[stepName];
}
return NULL;
}
bool CMissionData::addStep(IStep *step)
{
_Steps.push_back(step);
return true;
}
string CMissionData::genPreRequisites()
{
string ret;
if (!_ReqSkills.empty())
{
ret += "req_skill : ";
for (uint i=0; i<_ReqSkills.size(); ++i)
{
ret += _ReqSkills[i].Skill+" "+_ReqSkills[i].MinLevel+" "+_ReqSkills[i].MaxLevel;
if (i < _ReqSkills.size()-1)
ret +="; ";
else
ret += NL;
}
}
if (!_ReqMissionDone.empty())
{
for (uint i=0; i<_ReqMissionDone.size(); ++i)
{
ret += "req_mission : "+ _ReqMissionDone[i]+NL;
}
}
if (!_ReqMissionNotDone.empty())
{
for (uint i=0; i<_ReqMissionNotDone.size(); ++i)
{
ret += "req_mission_neg : "+_ReqMissionNotDone[i]+NL;
}
}
if (!_ReqMissionRunning.empty())
{
for (uint i=0; i<_ReqMissionRunning.size(); ++i)
{
ret += "req_mission_running : "+_ReqMissionRunning[i]+NL;
}
}
if (!_ReqMissionNotRunning.empty())
{
for (uint i=0; i<_ReqMissionNotRunning.size(); ++i)
{
ret += "req_mission_running_neg : "+_ReqMissionNotRunning[i]+NL;
}
}
if (!_ReqWearItem.empty())
{
ret += "req_wear : ";
for (uint i=0; i<_ReqWearItem.size(); ++i)
{
ret += _ReqWearItem[i];
if(i < _ReqWearItem.size()-1)
ret +="; ";
ret += NL;
}
}
if (!_ReqOwnItem.empty())
{
ret += "req_item : ";
for (uint i=0; i<_ReqOwnItem.size(); ++i)
{
ret += _ReqOwnItem[i];
if(i < _ReqOwnItem.size()-1)
ret +="; ";
ret += NL;
}
}
if (!_ReqTitle.empty())
{
ret += "req_title : "+_ReqTitle+NL;
}
if (!_ReqFames.empty())
{
for (uint i=0; i<_ReqFames.size(); ++i)
{
ret += "req_fame : "+_ReqFames[i].Faction+" "+_ReqFames[i].Fame;
ret += NL;
}
}
if(_ReqGuild)
{
ret += "req_guild"+NL;
}
if (!_ReqGrade.empty())
{
ret += "req_grade : "+_ReqGrade+NL;
}
if (!_ReqTeamSize.empty())
{
ret += "req_team_size : "+_ReqTeamSize+NL;
}
if (!_ReqBrick.empty())
{
ret += "req_brick : ";
for (uint i=0; i<_ReqBrick.size(); ++i)
{
ret += _ReqBrick[i];
if(i < _ReqBrick.size()-1)
ret +="; ";
ret += NL;
}
}
if (!_ReqCharacterAge.empty())
{
ret += "req_character_age : "+_ReqCharacterAge+NL;
}
if (!_ReqMaxPlayerID.empty())
{
ret += "req_max_player_id : "+_ReqMaxPlayerID+NL;
}
if (!_ReqSeason.empty())
{
ret += "req_season : "+_ReqSeason+NL;
}
if (!_ReqEncyclo.empty())
{
ret += "req_encyclo_thema : " + _ReqEncyclo + NL;
}
if (!_ReqEncycloNeg.empty())
{
ret += "req_encyclo_thema_neg : " + _ReqEncycloNeg + NL;
}
if (!_ReqEventFaction.empty())
{
ret += "req_event_faction : " + _ReqEventFaction + NL;
}
return ret;
}
string CMissionData::generateMissionScript(const std::string &primFileName)
{
_JumpPoints.clear();
// first, gather jump point list
for (uint i=0; i<_Steps.size(); ++i)
{
set<TJumpInfo> temp;
_Steps[i]->fillStepJump(*this, temp);
// remove any jump to the next step (normal flow)
if (i < _Steps.size()-1)
{
set<TJumpInfo>::iterator first(temp.begin()), last(temp.end());
for (; first != last; )
{
const TJumpInfo &ji = *first;
if (ji.StepName == _Steps[i+1]->getStepName() && ji.Discardable)
{
temp.erase(first);
first = temp.begin();
}
else
++first;
}
}
_JumpPoints.insert(temp.begin(), temp.end());
}
// generate the script
string script;
// generate mission header
script += "# script generated from '"+CFile::getFilename(primFileName)+"'"+NL+NL;
script += "#mission tags and pre-requisites"+NL;
if (_MonoInstance)
script += "mono"+NL;
if (_RunOnce)
script += "once"+NL;
if (_Replayable)
script += "replayable"+NL;
if (_Solo)
script += "solo"+NL;
if (_Guild)
script += "guild"+NL;
if (_NotInJournal)
script += "no_list"+NL;
if (_AutoRemoveFromJournal)
script += "auto_remove"+NL;
if (!_MissionCategory.empty())
script += "mission_category : "+_MissionCategory+NL;
if (_PlayerReplayTimer != 0)
script += "player_replay_timer : "+toString("%u", _PlayerReplayTimer)+NL;
if (_GlobalReplayTimer != 0)
script += "global_replay_timer : "+toString("%u", _GlobalReplayTimer)+NL;
if (_NotProposed)
script += "not_proposed"+NL;
if (_MissionAuto)
script += string("auto : ")+_MissionAutoMenu.genScript(*this)+NL;
if (_NonAbandonnable)
script += "non_abandonnable"+NL;
if (!_MissionIcon.empty())
script += "mission_icon : "+_MissionIcon+NL;
if (_NeedValidation)
script += "need_validation"+NL;
if (_FailIfInventoryIsFull)
script += "fail_if_inventory_is_full"+NL;
if (!_ParentMissions.empty())
{
set<string>::iterator first(_ParentMissions.begin()), last(_ParentMissions.end());
for (; first != last; ++first)
{
script += "parent : "+ *first+NL;
}
}
script += NL+"#Variables declaration"+NL;
// declare all the variables
{
std::vector<IVar*>::iterator first(_VariablesOrder.begin()), last(_VariablesOrder.end());
for (; first != last; ++first)
{
script += (*first)->genDecl(*this);
}
}
script += NL+"#pre-requisites"+NL;
script += genPreRequisites();
script += NL+"#script"+NL;
// generate mission title and desc
script += "mission_title : "+_MissionTitle.genScript(*this)+NL;
script += "mission_desc : "+_MissionDescription.genScript(*this)+NL;
// generate steps scripts
for (uint i=0; i<_Steps.size(); ++i)
{
script += "# "+_Steps[i]->getStepName()+NL;
if (_JumpPoints.find(_Steps[i]->getStepName()) != _JumpPoints.end()
&& !_Steps[i]->isAJump())
{
// insert a jump point
script += "jump_point : " + _Steps[i]->getStepName() + NL;
}
script += _Steps[i]->genCode(*this);
//if (_Steps[i]->EndOfBranch && !_Steps[i]->isAJump())
// script += "end"+NL;
}
return script;
}
string CMissionData::generatePhraseFile()
{
string ret;
// generate header phrase
ret = _MissionTitle.genPhrase();
ret += _MissionDescription.genPhrase();
ret += _MissionAutoMenu.genPhrase();
// generate var phrase
for (uint i=0; i<_VariablesOrder.size(); ++i)
{
ret += _VariablesOrder[i]->genPhrase();
}
// generate step phrase
for (uint i=0; i<_Steps.size(); ++i)
{
ret += _Steps[i]->genPhrase();
}
return ret;
}
string CMissionData::generateDotScript()
{
string ret = "digraph " + _MissionName + NL;
ret += "{" + NL;
// set default shape to 'record'
ret += "node [shape=record]"+NL;
ret += "\t__start__ [shape=\"point\", peripheries=2, label=\"\"]"+NL;
// 1st pass, generate node for each step
for (uint i=0; i<_Steps.size(); ++i)
{
if (!_Steps[i]->isEnd() && !_Steps[i]->isAJump())
{
ret += "\t"+_Steps[i]->getStepName();
ret += " [URL=\""+buildPrimPath(_Steps[i]->getPrimitive())+"\"]"+NL;
}
}
ret += "\t__end__ [shape=\"point\"]"+NL;
// activate red color for shapes that are created after this points
ret += "node [color=red]"+NL;
// 2nd pass, generate link between steps
for (uint i=0; i<_Steps.size(); ++i)
{
if (_Steps[i]->isAJump())
continue;
if (i == 0)
{
ret += "\t__start__ -> " + _Steps[i]->getStepName() + NL;
}
set<TJumpInfo> jumps;
_Steps[i]->fillStepJump(*this, jumps);
// there is a link there
while (!jumps.empty())
{
const TJumpInfo &ji = *(jumps.begin());
if (_StepsByNames.find(ji.StepName) != _StepsByNames.end()
&& _StepsByNames[ji.StepName]->isAJump())
{
// this step is a jump, skip to link to the jump destination
IStep *jumpStep = _StepsByNames[ji.StepName];
set<TJumpInfo> jumpJump;
jumpStep->fillStepJump(*this, jumpJump);
if (jumpJump.size() != 1)
{
string str = toString("Step jump contains %u jumps destination instead of 1", jumpJump.size());
throw EParseException(jumpStep->getPrimitive(), str.c_str());
}
ret += "\t"+_Steps[i]->getStepName() + " -> " + jumpJump.begin()->StepName+" [label=\""+ji.JumpName+"\"]" + NL;
}
else
{
ret += "\t"+_Steps[i]->getStepName() + " -> " + ji.StepName+" [label=\""+jumps.begin()->JumpName+"\"]" + NL;
}
jumps.erase(jumps.begin());
}
}
ret += "}" + NL;
return ret;
}
void CMissionData::parseMissionHeader(NLLIGO::IPrimitive *prim)
{
// _MissionName = getProperty(prim, "name", false, false);
// if( _MissionName.find(' ') != string::npos)
// {
// throw EParseException(prim, toString("Mission name '%s' must not contains space", _MissionName.c_str()).c_str());
// }
_GiverPrimitive = getProperty(prim,"giver_primitive", true, false);
_MissionGiver = getProperty(prim, "mission_giver", true, false);
// _Alias = getProperty(prim, "alias", false, false);
// If the mission is under a npc_bot node, then the giver is directly taken
// from the npc name
if (prim->getParent())
{
if (getProperty(prim->getParent(), "class", false, false) == "npc_bot")
{
_MissionGiver = getProperty(prim->getParent(), "name", false, false);
}
}
vector<string> vs;
_MissionTitleRaw = getPropertyArray(prim, "mission_title", false, false);
// _MissionTitle.init(*this, prim, vs);
_MissionDescriptionRaw = getPropertyArray(prim, "mission_description", false, false);
// _MissionDescription.init(*this, prim, vs);
_MonoInstance = toLower(getProperty(prim, "mono_instance", true, false)) == "true";
_RunOnce = toLower(getProperty(prim, "run_only_once", true, false)) == "true";
_Replayable = toLower(getProperty(prim, "replayable", true, false)) == "true";
_NeedValidation = toLower(getProperty(prim, "need_validation", true, false)) == "true";
_MissionAutoMenuRaw = getPropertyArray(prim, "phrase_auto_menu", false, false);
// audience setting
string s = getProperty(prim, "audience", false, false);
if (s == "solo")
_Solo = true;
else if (s == "guild")
_Guild = true;
_NotInJournal = NLMISC::toLower(getProperty(prim, "not_in_journal", false, false)) == "true";
_AutoRemoveFromJournal = NLMISC::toLower(getProperty(prim, "auto_remove_from_journal", false, false)) == "true";
_MissionCategory = getProperty(prim, "mission_category", false, false);
NLMISC::fromString(getProperty(prim, "player_replay_timer", true, false), _PlayerReplayTimer);
NLMISC::fromString(getProperty(prim, "global_replay_timer", true, false), _GlobalReplayTimer);
_NotProposed = NLMISC::toLower(getProperty(prim, "not_proposed", false, false)) == "true";
_MissionAuto = NLMISC::toLower(getProperty(prim, "automatic", false, false)) == "true";
_NonAbandonnable = NLMISC::toLower(getProperty(prim, "non_abandonnable", false, false)) == "true";
_FailIfInventoryIsFull = NLMISC::toLower(getProperty(prim, "fail_if_inventory_is_full", false, false)) == "true";
_MissionIcon = getProperty(prim, "mission_icon", false, false);
if (_MissionAuto)
{
if (_MissionAutoMenuRaw.empty())
{
string error = toString("Mission is flagged automatic, but no phrase_auto_menu defined !");
throw EParseException(prim, error.c_str());
}
}
vs = getPropertyArray(prim, "parent_missions", true, false);
_ParentMissions.insert(vs.begin(), vs.end());
}
void CMissionData::parsePrerequisites(NLLIGO::IPrimitive *prim)
{
// skills
vector<string> vs;
vs = getPropertyArray(prim, "require_skill/min_level/max_level", true, false);
for (uint i=0; i<vs.size(); ++i)
{
if (!vs[i].empty())
{
vector<string> parts;
strtokquote(vs[i], parts);
if (parts.size() != 3)
{
throw EParseException(prim, toString("Invalide argument count in line %u of require_skill array. Need 3, found %u", i, parts.size()).c_str());
}
TReqSkill rs;
rs.Skill = parts[0];
rs.MinLevel = parts[1];
rs.MaxLevel = parts[2];
_ReqSkills.push_back(rs);
}
}
// Mission done
vs = getPropertyArray(prim, "require_mission_done", true, false);
for (uint i=0; i<vs.size(); ++i)
{
if (!vs[i].empty())
{
vector<string> parts;
strtokquote(vs[i], parts);
if (parts.size() != 1)
{
throw EParseException(prim, toString("Invalide argument count in line %u of require_mission_done array. Need 1, found %u", i, parts.size()).c_str());
}
_ReqMissionDone.push_back(parts[0]);
}
}
// Mission not done
vs = getPropertyArray(prim, "require_mission_not_done", true, false);
for (uint i=0; i<vs.size(); ++i)
{
if (!vs[i].empty())
{
vector<string> parts;
strtokquote(vs[i], parts);
if (parts.size() != 1)
{
throw EParseException(prim, toString("Invalide argument count in line %u of require_mission_not_done array. Need 1, found %u", i, parts.size()).c_str());
}
_ReqMissionNotDone.push_back(parts[0]);
}
}
// Mission running
vs = getPropertyArray(prim, "require_mission_running", true, false);
for (uint i=0; i<vs.size(); ++i)
{
if (!vs[i].empty())
{
vector<string> parts;
strtokquote(vs[i], parts);
if (parts.size() != 1)
{
throw EParseException(prim, toString("Invalide argument count in line %u of require_mission_running array. Need 1, found %u", i, parts.size()).c_str());
}
_ReqMissionRunning.push_back(parts[0]);
}
}
// Mission not running
vs = getPropertyArray(prim, "require_mission_not_running", true, false);
for (uint i=0; i<vs.size(); ++i)
{
if (!vs[i].empty())
{
vector<string> parts;
strtokquote(vs[i], parts);
if (parts.size() != 1)
{
throw EParseException(prim, toString("Invalide argument count in line %u of require_mission_not_running array. Need 1, found %u", i, parts.size()).c_str());
}
_ReqMissionNotRunning.push_back(parts[0]);
}
}
// wearing item
vs = getPropertyArray(prim, "require_wearing_item", true, false);
for (uint i=0; i<vs.size(); ++i)
{
if (!vs[i].empty())
{
vector<string> parts;
strtokquote(vs[i], parts);
if (parts.size() != 1)
{
throw EParseException(prim, toString("Invalide argument count in line %u of require_wearing_item array. Need 1, found %u", i, parts.size()).c_str());
}
_ReqWearItem.push_back(parts[0]);
}
}
// own item
vs = getPropertyArray(prim, "require_own_item", true, false);
for (uint i=0; i<vs.size(); ++i)
{
if (!vs[i].empty())
{
vector<string> parts;
strtokquote(vs[i], parts);
if (parts.size() != 1)
{
throw EParseException(prim, toString("Invalide argument count in line %u of require_own_item array. Need 1, found %u", i, parts.size()).c_str());
}
_ReqOwnItem.push_back(parts[0]);
}
}
// title
_ReqTitle = getProperty(prim, "require_title", true, false);
// fame
vs = getPropertyArray(prim, "require_faction/fame", true, false);
for (uint i=0; i<vs.size(); ++i)
{
if (!vs[i].empty())
{
vector<string> parts;
strtokquote(vs[i], parts);
if (parts.size() != 2)
{
throw EParseException(prim, toString("Invalide argument count in line %u of require_faction/fame array. Need 2, found %u", i, parts.size()).c_str());
}
TReqFame rf;
rf.Faction = parts[0];
rf.Fame = parts[1];
_ReqFames.push_back(rf);
}
}
// guild
if (getProperty(prim, "require_guild_membership", true, false) == "true")
_ReqGuild = true;
else
_ReqGuild = false;
// grade
_ReqGrade = getProperty(prim, "require_guild_grade", true, false);
// team size
_ReqTeamSize = getProperty(prim, "require_team_size", true, false);
// character minimum age
_ReqCharacterAge = getProperty(prim, "require_character_age", true, false);
// maximum player ID
_ReqMaxPlayerID = getProperty(prim, "require_max_player_id", true, false);
// brick
vs = getPropertyArray(prim, "require_brick_knowledge", true, false);
for (uint i=0; i<vs.size(); ++i)
{
if (!vs[i].empty())
{
vector<string> parts;
strtokquote(vs[i], parts);
if (parts.size() != 1)
{
throw EParseException(prim, toString("Invalide argument count in line %u of require_brick_knowledge array. Need 1, found %u", i, parts.size()).c_str());
}
_ReqBrick.push_back(parts[0]);
}
}
// season
_ReqSeason = getProperty(prim, "require_season", true, false);
// encyclopedia
_ReqEncyclo = getProperty(prim, "require_encyclo_thema", true, false);
_ReqEncycloNeg = getProperty(prim, "require_encyclo_thema_neg", true, false);
if ((!_ReqEncyclo.empty() && !_ReqEncycloNeg.empty())
|| (!_ReqEncycloNeg.empty() && !_ReqEncyclo.empty()))
{
string err = toString("You can't mix positive and negative encyclopedy requirement");
throw EParseException(prim, err.c_str());
}
// event faction
_ReqEventFaction = getProperty(prim, "require_event_faction", true, false);
}
std::string CMissionData::replaceVar(NLLIGO::IPrimitive *prim, const std::string &str)
{
string::size_type pos = 0;
string::size_type pos2 = 0;
string ret;
while (pos < str.size())
{
if (str[pos] != '$')
{
ret += str[pos++];
}
else if (pos+1 < str.size() && str[pos+1] == '$')
{
// check that this $ is not escaped
ret += '$';
pos+=2;
}
else
{
// ok, this is not an escaped $
CSString varName;
// skip the initial '$'
pos++;
// while (str[pos] != ' ' && str[pos] != '\t' && str[pos] != '\n' && str[pos] != '\r')
while (pos < str.size() && str[pos] != '$')
varName += str[pos++];
if (pos >= str.size())
{
string err = toString("Error while parsing variable in '%s', missing closing '$'", str.c_str());
throw EParseException (NULL, err.c_str());
}
// skip the final '$'
pos++;
// split the var name and subpart
vector<string> varParts;
explode(string(varName), string("@"), varParts, true);
if (varParts.empty() || varParts.size() > 2)
{
throw EParseException(prim, toString("Error parsing varName '%s' in string '%s'", varName.c_str(), str.c_str()).c_str());
}
if (_Variables.find(varParts.front()) == _Variables.end())
{
string err = toString("Unknown variable '%s' in string '%s'", varParts.front().c_str(), str.c_str());
throw EParseException (prim, err.c_str());
}
IVar *var = _Variables[varParts[0]];
if (varParts.size() == 1)
ret += var->evalVar("");
else
ret += var->evalVar(varParts[1]);
}
}
return ret;
}
std::vector<std::string> CMissionData::replaceVar(NLLIGO::IPrimitive *prim, const std::vector<std::string> &strs)
{
vector<string> ret;
for (uint i=0; i<strs.size(); ++i)
{
ret.push_back(replaceVar(prim, strs[i]));
}
return ret;
}
std::string CMissionData::getProperty(NLLIGO::IPrimitive *prim, const std::string &propertyName, bool replaceVar, bool canFail)
{
string ret;
string *s;
if (!prim->getPropertyByName(propertyName.c_str(), s))
{
if (!canFail)
{
string err = toString("Can't find property '%s'", propertyName.c_str());
throw EParseException (prim, err.c_str());
}
}
else
{
ret = *s;
}
if (replaceVar)
{
ret = this->replaceVar(prim, ret);
}
return ret;
}
std::vector<std::string> CMissionData::getPropertyArray(NLLIGO::IPrimitive *prim, const std::string &propertyName, bool replaceVar, bool canFail)
{
vector<string> ret;
vector<string> *vs;
if (!prim->getPropertyByName(propertyName.c_str(), vs))
{
if (!canFail)
{
string err = toString("Can't find property '%s'", propertyName.c_str());
throw EParseException (prim, err.c_str());
}
}
else
{
ret = *vs;
}
if (replaceVar)
{
ret = this->replaceVar(prim, ret);
}
return ret;
}
bool CMissionData::isThereAJumpTo(const std::string &stepName)
{
if (_JumpPoints.find(stepName) != _JumpPoints.end())
return true;
else
return false;
}
void TCompilerVarName::init(const std::string &defaultName, STRING_MANAGER::TParamType type, CMissionData &md, NLLIGO::IPrimitive *prim, const std::string propName)
{
_DefaultName = defaultName;
_ParamType = type;
_VarName = md.getProperty(prim, propName, false, false);
// remove the variable tag if any
untagVar(_VarName);
_VarValue = md.getProperty(prim, propName, true, false);
}
void TCompilerVarName::initWithText(const std::string &defaultName, STRING_MANAGER::TParamType type, CMissionData &md, NLLIGO::IPrimitive *prim, const std::string &text)
{
_DefaultName = defaultName;
_ParamType = type;
_VarName = text;
// remove the variable tag if any
untagVar(_VarName);
_VarValue = md.replaceVar(prim, text);
}
CPhrase::TParamInfo TCompilerVarName::getParamInfo() const
{
if (_VarName.empty())
return CPhrase::TParamInfo(_DefaultName, _ParamType);
else
return CPhrase::TParamInfo(_VarName, _ParamType);
}
bool TCompilerVarName::empty() const
{
return _VarValue.empty();
}
TCompilerVarName::operator const std::string () const
{
return _VarValue;
}
TCompilerVarName::operator CPhrase::TParamInfo() const
{
return getParamInfo();
}
std::string operator+(const TCompilerVarName& left, const std::string & right) { return left._VarValue + right;}
std::string operator+(const std::string & left, const TCompilerVarName& right) { return left + right._VarValue;}
std::vector<TCompilerVarName> TCompilerVarName::getPropertyArrayWithText(const std::string &defaultName, STRING_MANAGER::TParamType type, CMissionData &md, NLLIGO::IPrimitive *prim, const std::string & arrayProperyName)
{
std::vector<TCompilerVarName> compilerParams;
std::vector<std::string> values = md.getPropertyArray(prim, arrayProperyName,false, false);
uint first = 0;
uint last = (uint)values.size();
compilerParams.resize(last);
for ( ; first != last; ++first)
{
compilerParams[first].initWithText( toString("%s%d", defaultName.c_str(), first+1) , type, md, prim, values[first]);
}
return compilerParams;
}
std::vector<TCompilerVarName> TCompilerVarName::getPropertyArrayWithTextStaticDefaultName(const std::string &defaultName, STRING_MANAGER::TParamType type, CMissionData &md, NLLIGO::IPrimitive *prim, const std::string & arrayProperyName)
{
std::vector<TCompilerVarName> compilerParams;
std::vector<std::string> values = md.getPropertyArray(prim, arrayProperyName,false, false);
uint first = 0;
uint last = (uint)values.size();
compilerParams.resize(last);
for ( ; first != last; ++first)
{
compilerParams[first].initWithText( defaultName, type, md, prim, values[first]);
}
return compilerParams;
}