mirror of
https://github.com/vcmi/vcmi.git
synced 2025-01-14 02:33:51 +02:00
94fa59cf45
* New settings.txt file (needed for kingdom overview to work) * Last remainings of h3m format: hero placeholders parsing * Minor improvements
386 lines
13 KiB
C++
386 lines
13 KiB
C++
//#define BOOST_SPIRIT_DEBUG
|
|
#include "CConfigHandler.h"
|
|
#include <boost/bind.hpp>
|
|
#include <boost/function.hpp>
|
|
#include <boost/spirit.hpp>
|
|
#include <fstream>
|
|
using namespace config;
|
|
using namespace boost::spirit;
|
|
using namespace phoenix;
|
|
|
|
/*
|
|
* CConfigHandler.cpp, part of VCMI engine
|
|
*
|
|
* Authors: listed in file AUTHORS in main folder
|
|
*
|
|
* License: GNU General Public License v2.0 or later
|
|
* Full text of license available in license.txt file, in main folder
|
|
*
|
|
*/
|
|
|
|
CConfigHandler conf;
|
|
GUIOptions *current = NULL;
|
|
std::pair<int,int> curRes;
|
|
ButtonInfo *currentButton;
|
|
int gnb=-1;
|
|
|
|
struct lerror
|
|
{
|
|
std::string txt;
|
|
lerror(const std::string & TXT):txt(TXT){};
|
|
void operator()() const
|
|
{
|
|
tlog1 << txt << std::endl;
|
|
}
|
|
template<typename IteratorT>
|
|
void operator()(IteratorT t1, IteratorT t2) const
|
|
{
|
|
tlog1 << txt << std::endl;
|
|
}
|
|
};
|
|
struct SetCurButton
|
|
{
|
|
template<typename IteratorT>
|
|
void operator()(IteratorT t1, IteratorT t2) const
|
|
{
|
|
std::string str(t1,t2);
|
|
if(str=="KingdomOv")
|
|
currentButton = ¤t->ac.kingOverview;
|
|
else if(str=="Underground")
|
|
currentButton = ¤t->ac.underground;
|
|
else if(str=="QuestLog")
|
|
currentButton = ¤t->ac.questlog;
|
|
else if(str=="SleepWake")
|
|
currentButton = ¤t->ac.sleepWake;
|
|
else if(str=="MoveHero")
|
|
currentButton = ¤t->ac.moveHero;
|
|
else if(str=="Spellbook")
|
|
currentButton = ¤t->ac.spellbook;
|
|
else if(str=="AdvOptions")
|
|
currentButton = ¤t->ac.advOptions;
|
|
else if(str=="SysOptions")
|
|
currentButton = ¤t->ac.sysOptions;
|
|
else if(str=="NextHero")
|
|
currentButton = ¤t->ac.nextHero;
|
|
else if(str=="EndTurn")
|
|
currentButton = ¤t->ac.endTurn;
|
|
}
|
|
};
|
|
struct lerror2
|
|
{
|
|
std::string txt;
|
|
lerror2(const std::string & TXT):txt(TXT){};
|
|
template<typename IteratorT>
|
|
void operator()(IteratorT t1, IteratorT t2) const
|
|
{
|
|
std::string txt2(t1,t2);
|
|
tlog1 << txt << txt2 << std::endl;
|
|
}
|
|
};
|
|
|
|
struct dummy
|
|
{
|
|
boost::function<void()> func;
|
|
dummy(const boost::function<void()> & F)
|
|
:func(F){}
|
|
template<typename IteratorT>
|
|
void operator()(IteratorT t1, IteratorT t2) const
|
|
{
|
|
func();
|
|
}
|
|
};
|
|
|
|
template<typename T>struct SetButtonProp
|
|
{
|
|
T point;
|
|
SetButtonProp(T p)
|
|
:point(p){}
|
|
template <typename Z>
|
|
void operator()(const Z & val) const
|
|
{
|
|
currentButton->*point = val;
|
|
}
|
|
};
|
|
template<typename T> SetButtonProp<T> SetButtonProp_a(T p)
|
|
{
|
|
return SetButtonProp<T>(p);
|
|
}
|
|
struct SetButtonStr
|
|
{
|
|
std::string ButtonInfo::*point;
|
|
SetButtonStr(std::string ButtonInfo::* p)
|
|
:point(p){}
|
|
template <typename Z>
|
|
void operator()(const Z first, const Z last) const
|
|
{
|
|
std::string str(first,last);
|
|
currentButton->*point = str;
|
|
}
|
|
};
|
|
template<typename T>struct SetAdventureProp
|
|
{
|
|
T point;
|
|
SetAdventureProp(T p)
|
|
:point(p){}
|
|
template <typename Z>
|
|
void operator()(const Z & val) const
|
|
{
|
|
current->ac.*point = val;
|
|
}
|
|
};
|
|
template<typename T> SetAdventureProp<T> SetAdventureProp_a(T p)
|
|
{
|
|
return SetAdventureProp<T>(p);
|
|
}
|
|
struct SetAdventureStr
|
|
{
|
|
std::string AdventureMapConfig::*point;
|
|
SetAdventureStr(std::string AdventureMapConfig::* p)
|
|
:point(p){}
|
|
template <typename Z>
|
|
void operator()(const Z first, const Z last) const
|
|
{
|
|
std::string str(first,last);
|
|
current->ac.*point = str;
|
|
}
|
|
};
|
|
struct AddDefForButton
|
|
{
|
|
template <typename Z>
|
|
void operator()(const Z first, const Z last) const
|
|
{
|
|
std::string str(first,last);
|
|
currentButton->additionalDefs.push_back(str);
|
|
}
|
|
};
|
|
struct ClearAdditionalDefs
|
|
{
|
|
template <typename Z>
|
|
void operator()(const Z first, const Z last) const
|
|
{
|
|
currentButton->additionalDefs.clear();
|
|
}
|
|
};
|
|
static void addGRes()
|
|
{
|
|
if(current)
|
|
conf.guiOptions[curRes] = *current; //we'll use by default settings from previous resolution
|
|
current = &conf.guiOptions[curRes];
|
|
}
|
|
static void setGem(int x, int val)
|
|
{
|
|
if(x)
|
|
current->ac.gemX[gnb] = val;
|
|
else
|
|
current->ac.gemY[gnb] = val;
|
|
}
|
|
struct AddGemName
|
|
{
|
|
template <typename Z>
|
|
void operator()(const Z first, const Z last) const
|
|
{
|
|
current->ac.gemG.push_back(std::string(first,last));
|
|
}
|
|
};
|
|
struct SettingsGrammar : public grammar<SettingsGrammar>
|
|
{
|
|
template <typename ScannerT>
|
|
struct definition
|
|
{
|
|
rule<ScannerT> r, clientOption, clientOptionsSequence, ClientSettings;
|
|
rule<ScannerT> GUISettings, GUIOption, GUIOptionsSequence, AdvMapOptionsSequence, AdvMapOption;
|
|
rule<ScannerT> GUIResolution, fname;
|
|
definition(SettingsGrammar const& self)
|
|
{
|
|
fname = lexeme_d[+(alnum_p | '.')];
|
|
clientOption
|
|
= str_p("resolution=") >> (uint_p[assign_a(conf.cc.resx)] >> 'x' >> uint_p[assign_a(conf.cc.resy)] | eps_p[lerror("Wrong resolution!")])
|
|
| str_p("port=") >> (uint_p[assign_a(conf.cc.port)] | eps_p[lerror("Wrong port!")])
|
|
| str_p("bpp=") >> (uint_p[assign_a(conf.cc.bpp)] | eps_p[lerror("Wrong bpp!")])
|
|
| str_p("localInformation=") >> (uint_p[assign_a(conf.cc.localInformation)] | eps_p[lerror("Wrong localInformation!")])
|
|
| str_p("fullscreen=") >> (uint_p[assign_a(conf.cc.fullscreen)] | eps_p[lerror("Wrong fullscreen!")])
|
|
| str_p("server=") >> ( ( +digit_p >> *('.' >> +digit_p) )[assign_a(conf.cc.server)] | eps_p[lerror("Wrong server!")])
|
|
| str_p("defaultAI=") >> ((+(anychar_p - ';'))[assign_a(conf.cc.defaultAI)] | eps_p[lerror("Wrong defaultAI!")])
|
|
| (+(anychar_p - '}'))[lerror2("Unrecognized client option: ")]
|
|
;
|
|
clientOptionsSequence = *(clientOption >> (';' | eps_p[lerror("Semicolon lacking after client option!")]));
|
|
ClientSettings = '{' >> clientOptionsSequence >> '}';
|
|
|
|
AdvMapOption
|
|
= str_p("Buttons") >> ((ch_p('{') >> '}') | eps_p[lerror("Wrong Buttons!")])
|
|
| str_p("Minimap: ") >>
|
|
*(
|
|
"width=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::minimapW)]//[assign_a(current->ac.minimapW)]
|
|
| "height=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::minimapH)]
|
|
| "x=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::minimapX)]
|
|
| "y=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::minimapY)]
|
|
)
|
|
| str_p("Statusbar:") >>
|
|
*(
|
|
( "x=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::statusbarX)]
|
|
| "y=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::statusbarY)]
|
|
| "graphic=" >> fname[SetAdventureStr(&AdventureMapConfig::statusbarG)]
|
|
)
|
|
)
|
|
| str_p("ResDataBar:") >>
|
|
*(
|
|
( "x=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::resdatabarX)]
|
|
| "y=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::resdatabarY)]
|
|
| "offsetX=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::resOffsetX)]
|
|
| "offsetY=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::resOffsetY)]
|
|
| "resSpace=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::resDist)]
|
|
| "resDateSpace=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::resDateDist)]
|
|
| "graphic=" >> fname[SetAdventureStr(&AdventureMapConfig::resdatabarG)]
|
|
)
|
|
)
|
|
| str_p("InfoBox:") >>
|
|
*(
|
|
( "x=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::infoboxX)]
|
|
| "y=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::infoboxY)]
|
|
)
|
|
)
|
|
| str_p("AdvMap:") >>
|
|
*(
|
|
( "x=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::advmapX)]
|
|
| "y=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::advmapY)]
|
|
| "width=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::advmapW)]
|
|
| "height=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::advmapH)]
|
|
| "smoothMove=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::smoothMove)]
|
|
| "puzzleSepia=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::puzzleSepia)]
|
|
)
|
|
)
|
|
| str_p("background=") >> fname[SetAdventureStr(&AdventureMapConfig::mainGraphic)]
|
|
| str_p("Button") >> (+(anychar_p-':'))[SetCurButton()] >> ':' >>
|
|
*(
|
|
( "x=" >> uint_p[SetButtonProp_a(&ButtonInfo::x)]
|
|
| "y=" >> uint_p[SetButtonProp_a(&ButtonInfo::y)]
|
|
| "playerColoured=" >> uint_p[SetButtonProp_a(&ButtonInfo::playerColoured)]
|
|
| "graphic=" >> fname[SetButtonStr(&ButtonInfo::defName)]
|
|
| str_p("additionalDefs=")[ClearAdditionalDefs()]
|
|
>> ch_p('(') >> fname[AddDefForButton()]
|
|
>> *(',' >> fname[AddDefForButton()]) >> ')'
|
|
)
|
|
)
|
|
| str_p("HeroList:") >>
|
|
*(
|
|
( "x=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::hlistX)]
|
|
| "y=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::hlistY)]
|
|
| "size=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::hlistSize)]
|
|
| "movePoints=" >> fname[SetAdventureStr(&AdventureMapConfig::hlistMB)]
|
|
| "manaPoints=" >> fname[SetAdventureStr(&AdventureMapConfig::hlistMN)]
|
|
| "arrowUp=" >> fname[SetAdventureStr(&AdventureMapConfig::hlistAU)]
|
|
| "arrowDown=" >> fname[SetAdventureStr(&AdventureMapConfig::hlistAD)]
|
|
)
|
|
)
|
|
| str_p("TownList:") >>
|
|
*(
|
|
( "x=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::tlistX)]
|
|
| "y=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::tlistY)]
|
|
| "size=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::tlistSize)]
|
|
| "arrowUp=" >> fname[SetAdventureStr(&AdventureMapConfig::tlistAU)]
|
|
| "arrowDown=" >> fname[SetAdventureStr(&AdventureMapConfig::tlistAD)]
|
|
)
|
|
)
|
|
| str_p("gem") >> uint_p[var(gnb) = arg1] >> ':' >>
|
|
*(
|
|
( "x=" >> uint_p[bind(&setGem,1,_1)]
|
|
| "y=" >> uint_p[bind(&setGem,0,_1)]
|
|
| "graphic=" >> fname[AddGemName()]
|
|
)
|
|
)
|
|
| str_p("InGameConsole:") >>
|
|
*(
|
|
( "maxInputPerLine=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::inputLineLength)]
|
|
| "maxOutputPerLine=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::outputLineLength)]
|
|
)
|
|
)
|
|
| str_p("Overview:") >>
|
|
*(
|
|
( "pics=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::overviewPics)]
|
|
| "size=" >> uint_p[SetAdventureProp_a(&AdventureMapConfig::overviewSize)]
|
|
| "graphic=" >> fname[SetAdventureStr(&AdventureMapConfig::overviewBg)]
|
|
)
|
|
)
|
|
;
|
|
AdvMapOptionsSequence = *(AdvMapOption >> (';' | eps_p[lerror("Semicolon lacking in advmapopt!")]));
|
|
GUIResolution = (uint_p[assign_a(curRes.first)] >> 'x' >> uint_p[assign_a(curRes.second)])
|
|
[dummy(&addGRes)];
|
|
GUIOption = str_p("AdventureMap") >> ('{' >> AdvMapOptionsSequence >> '}' | eps_p[lerror("Wrong AdventureMap!")]);
|
|
GUIOptionsSequence = *(GUIOption >> (';' | eps_p[lerror("Semicolon after GUIOption lacking!")]));
|
|
GUISettings = +(GUIResolution >> '{' >> GUIOptionsSequence >> '}');
|
|
|
|
|
|
r
|
|
= str_p("clientSettings") >> (ClientSettings | eps_p[lerror("Wrong clientSettings!")])
|
|
>> str_p("GUISettings") >> ('{' >> GUISettings >> '}' | eps_p[lerror("Wrong GUISettings!")]);
|
|
#ifdef BOOST_SPIRIT_DEBUG
|
|
BOOST_SPIRIT_DEBUG_RULE(clientOption);
|
|
BOOST_SPIRIT_DEBUG_RULE(clientOptionsSequence);
|
|
BOOST_SPIRIT_DEBUG_RULE(ClientSettings);
|
|
BOOST_SPIRIT_DEBUG_RULE(AdvMapOption);
|
|
BOOST_SPIRIT_DEBUG_RULE(AdvMapOptionsSequence);
|
|
BOOST_SPIRIT_DEBUG_RULE(GUIOption);
|
|
BOOST_SPIRIT_DEBUG_RULE(GUIOptionsSequence);
|
|
BOOST_SPIRIT_DEBUG_RULE(GUISettings);
|
|
BOOST_SPIRIT_DEBUG_RULE(GUIResolution);
|
|
BOOST_SPIRIT_DEBUG_RULE(r);
|
|
#endif
|
|
}
|
|
|
|
rule<ScannerT> const& start() const { return r; }
|
|
};
|
|
};
|
|
|
|
struct CommentsGrammar : public grammar<CommentsGrammar>
|
|
{
|
|
template <typename ScannerT>
|
|
struct definition
|
|
{
|
|
rule<ScannerT> comment;
|
|
definition(CommentsGrammar const& self)
|
|
{
|
|
comment = comment_p("//") | comment_p("/*","*/") | space_p;
|
|
BOOST_SPIRIT_DEBUG_RULE(comment);
|
|
}
|
|
rule<ScannerT> const& start() const { return comment; }
|
|
};
|
|
};
|
|
|
|
CConfigHandler::CConfigHandler(void)
|
|
{
|
|
}
|
|
|
|
CConfigHandler::~CConfigHandler(void)
|
|
{
|
|
}
|
|
|
|
void config::CConfigHandler::init()
|
|
{
|
|
std::vector<char> settings;
|
|
std::ifstream ifs(DATA_DIR "/config/settings.txt");
|
|
if(!ifs)
|
|
{
|
|
tlog1 << "Cannot open " DATA_DIR "/config/settings.txt !" << std::endl;
|
|
return;
|
|
}
|
|
ifs.unsetf(std::ios::skipws); // Turn of white space skipping on the stream
|
|
std::copy(std::istream_iterator<char>(ifs),std::istream_iterator<char>(),std::back_inserter(settings));
|
|
std::vector<char>::const_iterator first = settings.begin(), last = settings.end();
|
|
SettingsGrammar sg;
|
|
BOOST_SPIRIT_DEBUG_NODE(sg);
|
|
CommentsGrammar cg;
|
|
BOOST_SPIRIT_DEBUG_NODE(cg);
|
|
|
|
|
|
parse_info<std::vector<char>::const_iterator> info = parse(first,last,sg,cg);
|
|
if(!info.hit)
|
|
tlog1 << "Cannot parse config/settings.txt file!\n";
|
|
else if(!info.full)
|
|
tlog2 << "Not entire config/settings.txt parsed!\n";
|
|
}
|
|
|
|
GUIOptions * config::CConfigHandler::go()
|
|
{
|
|
return &guiOptions[std::pair<int,int>(cc.resx,cc.resy)];
|
|
}
|