| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056 |
- #define VCMI_DLL
- #include "../stdafx.h"
- #include "CObjectHandler.h"
- #include "CDefObjInfoHandler.h"
- #include "CLodHandler.h"
- #include "CGeneralTextHandler.h"
- #include "CDefObjInfoHandler.h"
- #include "CHeroHandler.h"
- #include "CSpellHandler.h"
- #include <boost/bind.hpp>
- #include <boost/algorithm/string/replace.hpp>
- #include <boost/random/linear_congruential.hpp>
- #include "CTownHandler.h"
- #include "CArtHandler.h"
- #include "../lib/VCMI_Lib.h"
- #include "../lib/IGameCallback.h"
- #include "../CGameState.h"
- #include "../lib/NetPacks.h"
- IGameCallback * IObjectInterface::cb = NULL;
- DLL_EXPORT void loadToIt(std::string &dest, std::string &src, int &iter, int mode);
- extern CLodHandler * bitmaph;
- extern boost::rand48 ran;
- void IObjectInterface::onHeroVisit(const CGHeroInstance * h)
- {};
- void IObjectInterface::onHeroLeave(const CGHeroInstance * h)
- {};
- void IObjectInterface::newTurn ()
- {};
- IObjectInterface::~IObjectInterface()
- {}
- IObjectInterface::IObjectInterface()
- {}
- void IObjectInterface::initObj()
- {}
- void CObjectHandler::loadObjects()
- {
- VLC->objh = this;
- // int ID=0; //TODO use me
- tlog5 << "\t\tReading OBJNAMES \n";
- std::string buf = bitmaph->getTextFile("OBJNAMES.TXT");
- int it=0; //hope that -1 will not break this
- while (it<buf.length()-1)
- {
- std::string nobj;
- loadToIt(nobj, buf, it, 3);
- if(nobj.size() && (nobj[nobj.size()-1]==(char)10 || nobj[nobj.size()-1]==(char)13 || nobj[nobj.size()-1]==(char)9))
- {
- nobj = nobj.substr(0, nobj.size()-1);
- }
- names.push_back(nobj);
- }
- tlog5 << "\t\tReading ADVEVENT \n";
- buf = bitmaph->getTextFile("ADVEVENT.TXT");
- it=0;
- std::string temp;
- while (it<buf.length()-1)
- {
- loadToIt(temp,buf,it,3);
- if (temp[0]=='\"')
- {
- temp = temp.substr(1,temp.length()-2);
- }
- boost::algorithm::replace_all(temp,"\"\"","\"");
- advobtxt.push_back(temp);
- }
- tlog5 << "\t\tReading XTRAINFO \n";
- buf = bitmaph->getTextFile("XTRAINFO.TXT");
- it=0;
- while (it<buf.length()-1)
- {
- loadToIt(temp,buf,it,3);
- xtrainfo.push_back(temp);
- }
- tlog5 << "\t\tReading MINENAME \n";
- buf = bitmaph->getTextFile("MINENAME.TXT");
- it=0;
- while (it<buf.length()-1)
- {
- loadToIt(temp,buf,it,3);
- mines.push_back(std::pair<std::string,std::string>(temp,""));
- }
- tlog5 << "\t\tReading MINEEVNT \n";
- buf = bitmaph->getTextFile("MINEEVNT.TXT");
- it=0;
- int i=0;
- while (it<buf.length()-1)
- {
- loadToIt(temp,buf,it,3);
- temp = temp.substr(1,temp.length()-2);
- mines[i++].second = temp;
- }
- tlog5 << "\t\tReading RESTYPES \n";
- buf = bitmaph->getTextFile("RESTYPES.TXT");
- it=0;
- while (it<buf.length()-1)
- {
- loadToIt(temp,buf,it,3);
- restypes.push_back(temp);
- }
- tlog5 << "\t\tReading cregens \n";
- cregens.resize(110); //TODO: hardcoded value - change
- for(size_t i=0; i < cregens.size(); ++i)
- {
- cregens[i]=-1;
- }
- std::ifstream ifs("config/cregens.txt");
- while(!ifs.eof())
- {
- int dw, cr;
- ifs >> dw >> cr;
- cregens[dw]=cr;
- }
- ifs.close();
- ifs.clear();
- tlog5 << "\t\tReading ZCRGN1 \n";
- buf = bitmaph->getTextFile("ZCRGN1.TXT");
- it=0;
- while (it<buf.length()-1)
- {
- loadToIt(temp,buf,it,3);
- creGens.push_back(temp);
- }
- tlog5 << "\t\tDone loading objects!\n";
- }
- int CGObjectInstance::getOwner() const
- {
- //if (state)
- // return state->owner;
- //else
- return tempOwner; //won't have owner
- }
- CGObjectInstance::CGObjectInstance(): animPhaseShift(rand()%0xff)
- {
- pos = int3(-1,-1,-1);
- //std::cout << "Tworze obiekt "<<this<<std::endl;
- //state = new CLuaObjectScript();
- ID = subID = id = -1;
- defInfo = NULL;
- state = NULL;
- info = NULL;
- tempOwner = 254;
- blockVisit = false;
- }
- CGObjectInstance::~CGObjectInstance()
- {
- //std::cout << "Usuwam obiekt "<<this<<std::endl;
- //if (state)
- // delete state;
- //state=NULL;
- }
- CGObjectInstance::CGObjectInstance(const CGObjectInstance & right)
- {
- pos = right.pos;
- ID = right.ID;
- subID = right.subID;
- id = right.id;
- defInfo = right.defInfo;
- info = right.info;
- blockVisit = right.blockVisit;
- //state = new CLuaObjectScript(right.state->);
- //*state = *right.state;
- //state = right.state;
- tempOwner = right.tempOwner;
- }
- CGObjectInstance& CGObjectInstance::operator=(const CGObjectInstance & right)
- {
- pos = right.pos;
- ID = right.ID;
- subID = right.subID;
- id = right.id;
- defInfo = right.defInfo;
- info = right.info;
- blockVisit = right.blockVisit;
- //state = new CLuaObjectScript();
- //*state = *right.state;
- tempOwner = right.tempOwner;
- return *this;
- }
- const std::string & CGObjectInstance::getHoverText() const
- {
- return hoverName;
- }
- void CGObjectInstance::setOwner(int ow)
- {
- //if (state)
- // state->owner = ow;
- //else
- tempOwner = ow;
- }
- int CGObjectInstance::getWidth() const//returns width of object graphic in tiles
- {
- return defInfo->width;
- }
- int CGObjectInstance::getHeight() const //returns height of object graphic in tiles
- {
- return defInfo->width;
- }
- bool CGObjectInstance::visitableAt(int x, int y) const //returns true if object is visitable at location (x, y) form left top tile of image (x, y in tiles)
- {
- if(x<0 || y<0 || x>=getWidth() || y>=getHeight() || defInfo==NULL)
- return false;
- if((defInfo->visitMap[y+6-getHeight()] >> (7-(8-getWidth()+x) )) & 1)
- return true;
- return false;
- }
- bool CGObjectInstance::blockingAt(int x, int y) const
- {
- if(x<0 || y<0 || x>=getWidth() || y>=getHeight() || defInfo==NULL)
- return false;
- if((defInfo->blockMap[y+6-getHeight()] >> (7-(8-getWidth()+x) )) & 1)
- return true;
- return false;
- }
- bool CGObjectInstance::operator<(const CGObjectInstance & cmp) const //screen printing priority comparing
- {
- if(defInfo->printPriority==1 && cmp.defInfo->printPriority==0)
- return true;
- if(cmp.defInfo->printPriority==1 && defInfo->printPriority==0)
- return false;
- if(this->pos.y<cmp.pos.y)
- return true;
- if(this->pos.y>cmp.pos.y)
- return false;
- if(cmp.ID==34 && ID!=34)
- return true;
- if(cmp.ID!=34 && ID==34)
- return false;
- if(!defInfo->isVisitable() && cmp.defInfo->isVisitable())
- return true;
- if(!cmp.defInfo->isVisitable() && defInfo->isVisitable())
- return false;
- if(this->pos.x<cmp.pos.x)
- return true;
- return false;
- }
- void CGObjectInstance::initObj()
- {
- }
- int lowestSpeed(const CGHeroInstance * chi)
- {
- if(!chi->army.slots.size())
- {
- tlog1 << "Error! Hero " << chi->id << " ("<<chi->name<<") has no army!\n";
- return 20;
- }
- std::map<si32,std::pair<ui32,si32> >::const_iterator i = chi->army.slots.begin();
- ui32 ret = VLC->creh->creatures[(*i++).second.first].speed;
- for (;i!=chi->army.slots.end();i++)
- {
- ret = std::min(ret,VLC->creh->creatures[(*i).second.first].speed);
- }
- return ret;
- }
- unsigned int CGHeroInstance::getTileCost(const EterrainType & ttype, const Eroad & rdtype, const Eriver & rvtype) const
- {
- unsigned int ret = type->heroClass->terrCosts[ttype];
- //applying pathfinding skill
- switch(getSecSkillLevel(0))
- {
- case 1: //basic
- switch(ttype)
- {
- case rough:
- ret = 100;
- break;
- case sand: case snow:
- if(ret>125)
- ret = 125;
- break;
- case swamp:
- if(ret>150)
- ret = 150;
- break;
- default:
- //TODO do something nasty here throw maybe? or some def value asing
- break;
- }
- break;
- case 2: //advanced
- switch(ttype)
- {
- case rough:
- case sand:
- case snow:
- ret = 100;
- break;
- case swamp:
- if(ret>125)
- ret = 125;
- break;
- default:
- //TODO look up
- break;
- }
- break;
- case 3: //expert
- ret = 100;
- break;
- default:
- //TODO look up
- break;
- }
- //calculating road influence
- switch(rdtype)
- {
- case dirtRoad:
- ret*=0.75;
- break;
- case grazvelRoad:
- ret*=0.667;
- break;
- case cobblestoneRoad:
- ret*=0.5;
- break;
- default:
- //TODO killllll me
- break;
- }
- return ret;
- }
- unsigned int CGHeroInstance::getLowestCreatureSpeed() const
- {
- unsigned int sl = 100;
- for(size_t h=0; h < army.slots.size(); ++h)
- {
- if(VLC->creh->creatures[army.slots.find(h)->first].speed<sl)
- sl = VLC->creh->creatures[army.slots.find(h)->first].speed;
- }
- return sl;
- }
- int3 CGHeroInstance::convertPosition(int3 src, bool toh3m) //toh3m=true: manifest->h3m; toh3m=false: h3m->manifest
- {
- if (toh3m)
- {
- src.x+=1;
- return src;
- }
- else
- {
- src.x-=1;
- return src;
- }
- }
- int3 CGHeroInstance::getPosition(bool h3m) const //h3m=true - returns position of hero object; h3m=false - returns position of hero 'manifestation'
- {
- if (h3m)
- {
- return pos;
- }
- else
- {
- return convertPosition(pos,false);
- }
- }
- int CGHeroInstance::getSightDistance() const //returns sight distance of this hero
- {
- return 6 + getSecSkillLevel(3); //default + scouting
- }
- int CGHeroInstance::manaLimit() const
- {
- double modifier = 1.0;
- switch(getSecSkillLevel(24)) //intelligence level
- {
- case 1: modifier+=0.25; break;
- case 2: modifier+=0.5; break;
- case 3: modifier+=1.0; break;
- }
- return 10*getPrimSkillLevel(3)*modifier;
- }
- //void CGHeroInstance::setPosition(int3 Pos, bool h3m) //as above, but sets position
- //{
- // if (h3m)
- // pos = Pos;
- // else
- // pos = convertPosition(Pos,true);
- //}
- bool CGHeroInstance::canWalkOnSea() const
- {
- //TODO: write it - it should check if hero is flying, or something similiar
- return false;
- }
- int CGHeroInstance::getCurrentLuck() const
- {
- //TODO: write it
- return 0;
- }
- int CGHeroInstance::getCurrentMorale() const
- {
- //TODO: write it
- return 0;
- }
- int CGHeroInstance::getPrimSkillLevel(int id) const
- {
- return primSkills[id];
- }
- int CGHeroInstance::getSecSkillLevel(const int & ID) const
- {
- for(size_t i=0; i < secSkills.size(); ++i)
- if(secSkills[i].first==ID)
- return secSkills[i].second;
- return 0;
- }
- int CGHeroInstance::maxMovePoints(bool onLand) const
- {
- int ret = 1270+70*lowestSpeed(this);
- if (ret>2000)
- ret=2000;
- if(onLand)
- {
- //logistics:
- switch(getSecSkillLevel(2))
- {
- case 1:
- ret *= 1.1f;
- break;
- case 2:
- ret *= 1.2f;
- break;
- case 3:
- ret *= 1.3f;
- break;
- }
- }
- else
- {
- //navigation:
- switch(getSecSkillLevel(2))
- {
- case 1:
- ret *= 1.5f;
- break;
- case 2:
- ret *= 2.0f;
- break;
- case 3:
- ret *= 2.5f;
- break;
- }
- }
- return ret;
- }
- ui32 CGHeroInstance::getArtAtPos(ui16 pos) const
- {
- if(pos<19)
- if(vstd::contains(artifWorn,pos))
- return artifWorn.find(pos)->second;
- else
- return -1;
- else
- if(pos-19 < artifacts.size())
- return artifacts[pos-19];
- else
- return -1;
- }
- void CGHeroInstance::setArtAtPos(ui16 pos, int art)
- {
- if(art<0)
- {
- if(pos<19)
- artifWorn.erase(pos);
- else
- artifacts -= artifacts[pos-19];
- }
- else
- {
- if(pos<19)
- artifWorn[pos] = art;
- else
- if(pos-19 < artifacts.size())
- artifacts[pos-19] = art;
- else
- artifacts.push_back(art);
- }
- }
- const CArtifact * CGHeroInstance::getArt(int pos) const
- {
- int id = getArtAtPos(pos);
- if(id>=0)
- return &VLC->arth->artifacts[id];
- else
- return NULL;
- }
- int CGHeroInstance::getSpellSecLevel(int spell) const
- {
- int bestslvl = 0;
- if(VLC->spellh->spells[spell].air)
- if(getSecSkillLevel(15) >= bestslvl)
- {
- bestslvl = getSecSkillLevel(15);
- }
- if(VLC->spellh->spells[spell].fire)
- if(getSecSkillLevel(14) >= bestslvl)
- {
- bestslvl = getSecSkillLevel(14);
- }
- if(VLC->spellh->spells[spell].water)
- if(getSecSkillLevel(16) >= bestslvl)
- {
- bestslvl = getSecSkillLevel(16);
- }
- if(VLC->spellh->spells[spell].earth)
- if(getSecSkillLevel(17) >= bestslvl)
- {
- bestslvl = getSecSkillLevel(17);
- }
- return bestslvl;
- }
- CGHeroInstance::CGHeroInstance()
- {
- ID = 34;
- tacticFormationEnabled = inTownGarrison = false;
- mana = movement = portrait = level = -1;
- isStanding = true;
- moveDir = 4;
- exp = 0xffffffff;
- visitedTown = NULL;
- type = NULL;
- secSkills.push_back(std::make_pair(-1, -1));
- }
- void CGHeroInstance::initHero(int SUBID)
- {
- subID = SUBID;
- initHero();
- }
- void CGHeroInstance::initHero()
- {
- if(!defInfo)
- {
- defInfo = new CGDefInfo();
- defInfo->id = 34;
- defInfo->subid = subID;
- defInfo->printPriority = 0;
- defInfo->visitDir = 0xff;
- }
- if(!type)
- type = VLC->heroh->heroes[subID];
- for(int i=0;i<6;i++)
- {
- defInfo->blockMap[i]=255;
- defInfo->visitMap[i]=0;
- }
- defInfo->handler=NULL;
- defInfo->blockMap[5] = 253;
- defInfo->visitMap[5] = 2;
- artifWorn[16] = 3;
- if(type->heroType % 2 == 1) //it's a magical hero
- {
- artifWorn[17] = 0; //give him spellbook
- }
- if(portrait < 0 || portrait == 255)
- portrait = subID;
- if((!primSkills.size()) || (getPrimSkillLevel(0)<0))
- {
- primSkills.resize(4);
- primSkills[0] = type->heroClass->initialAttack;
- primSkills[1] = type->heroClass->initialDefence;
- primSkills[2] = type->heroClass->initialPower;
- primSkills[3] = type->heroClass->initialKnowledge;
- }
- if(secSkills.size() == 1 && secSkills[0] == std::pair<ui8,ui8>(-1, -1)) //set secondary skills to default
- secSkills = type->secSkillsInit;
- if(mana < 0)
- mana = manaLimit();
- if (!name.length())
- name = type->name;
- if (exp == 0xffffffff)
- {
- exp=40+ (ran()) % 50;
- level = 1;
- }
- else
- {
- level = VLC->heroh->level(exp);
- }
- if (!army.slots.size()) //standard army//initial army
- {
- int pom, pom2=0;
- for(int x=0;x<3;x++)
- {
- pom = (VLC->creh->nameToID[type->refTypeStack[x]]);
- if(pom>=145 && pom<=149) //war machine
- {
- pom2++;
- switch (pom)
- {
- case 145: //catapult
- artifWorn[16] = 3;
- break;
- default:
- artifWorn[9+CArtHandler::convertMachineID(pom,true)] = CArtHandler::convertMachineID(pom,true);
- break;
- }
- continue;
- }
- army.slots[x-pom2].first = pom;
- if((pom = (type->highStack[x]-type->lowStack[x])) > 0)
- army.slots[x-pom2].second = (ran()%pom)+type->lowStack[x];
- else
- army.slots[x-pom2].second = +type->lowStack[x];
- army.formation = false;
- }
- }
- hoverName = VLC->generaltexth->allTexts[15];
- boost::algorithm::replace_first(hoverName,"%s",name);
- boost::algorithm::replace_first(hoverName,"%s", type->heroClass->name);
- }
- CGHeroInstance::~CGHeroInstance()
- {
- }
- bool CGHeroInstance::needsLastStack() const
- {
- return true;
- }
- void CGHeroInstance::onHeroVisit(const CGHeroInstance * h)
- {
- //TODO: check for allies
- if(tempOwner == h->tempOwner) //our hero
- {
- //exchange
- }
- else
- {
- cb->startBattleI(
- &h->army,
- &army,
- h->pos,
- h,
- this,
- 0);
- }
- }
- const std::string & CGHeroInstance::getBiography() const
- {
- if (biography.length())
- return biography;
- else
- return VLC->generaltexth->hTxts[subID].biography;
- }
- void CGHeroInstance::initObj()
- {
- cb->setBlockVis(id,true);
- }
- int CGTownInstance::getSightDistance() const //returns sight distance
- {
- return 10;
- }
- int CGTownInstance::fortLevel() const //0 - none, 1 - fort, 2 - citadel, 3 - castle
- {
- if((builtBuildings.find(9))!=builtBuildings.end())
- return 3;
- if((builtBuildings.find(8))!=builtBuildings.end())
- return 2;
- if((builtBuildings.find(7))!=builtBuildings.end())
- return 1;
- return 0;
- }
- int CGTownInstance::hallLevel() const // -1 - none, 0 - village, 1 - town, 2 - city, 3 - capitol
- {
- if ((builtBuildings.find(13))!=builtBuildings.end())
- return 3;
- if ((builtBuildings.find(12))!=builtBuildings.end())
- return 2;
- if ((builtBuildings.find(11))!=builtBuildings.end())
- return 1;
- if ((builtBuildings.find(10))!=builtBuildings.end())
- return 0;
- return -1;
- }
- int CGTownInstance::mageGuildLevel() const
- {
- if ((builtBuildings.find(4))!=builtBuildings.end())
- return 5;
- if ((builtBuildings.find(3))!=builtBuildings.end())
- return 4;
- if ((builtBuildings.find(2))!=builtBuildings.end())
- return 3;
- if ((builtBuildings.find(1))!=builtBuildings.end())
- return 2;
- if ((builtBuildings.find(0))!=builtBuildings.end())
- return 1;
- return 0;
- }
- bool CGTownInstance::creatureDwelling(const int & level, bool upgraded) const
- {
- return builtBuildings.find(30+level+upgraded*7)!=builtBuildings.end();
- }
- int CGTownInstance::getHordeLevel(const int & HID) const//HID - 0 or 1; returns creature level or -1 if that horde structure is not present
- {
- return town->hordeLvl[HID];
- }
- int CGTownInstance::creatureGrowth(const int & level) const
- {
- int ret = VLC->creh->creatures[town->basicCreatures[level]].growth;
- switch(fortLevel())
- {
- case 3:
- ret*=2;break;
- case 2:
- ret*=(1.5); break;
- }
- if(builtBuildings.find(26)!=builtBuildings.end()) //grail
- ret+=VLC->creh->creatures[town->basicCreatures[level]].growth;
- if(getHordeLevel(0)==level)
- if((builtBuildings.find(18)!=builtBuildings.end()) || (builtBuildings.find(19)!=builtBuildings.end()))
- ret+=VLC->creh->creatures[town->basicCreatures[level]].hordeGrowth;
- if(getHordeLevel(1)==level)
- if((builtBuildings.find(24)!=builtBuildings.end()) || (builtBuildings.find(25)!=builtBuildings.end()))
- ret+=VLC->creh->creatures[town->basicCreatures[level]].hordeGrowth;
- return ret;
- }
- int CGTownInstance::dailyIncome() const
- {
- int ret = 0;
- if ((builtBuildings.find(26))!=builtBuildings.end())
- ret+=5000;
- if ((builtBuildings.find(13))!=builtBuildings.end())
- ret+=4000;
- else if ((builtBuildings.find(12))!=builtBuildings.end())
- ret+=2000;
- else if ((builtBuildings.find(11))!=builtBuildings.end())
- ret+=1000;
- else if ((builtBuildings.find(10))!=builtBuildings.end())
- ret+=500;
- return ret;
- }
- bool CGTownInstance::hasFort() const
- {
- return (builtBuildings.find(7))!=builtBuildings.end();
- }
- bool CGTownInstance::hasCapitol() const
- {
- return (builtBuildings.find(13))!=builtBuildings.end();
- }
- CGTownInstance::CGTownInstance()
- {
- builded=-1;
- destroyed=-1;
- garrisonHero=NULL;
- town=NULL;
- visitingHero = NULL;
- }
- CGTownInstance::~CGTownInstance()
- {}
- int CGTownInstance::spellsAtLevel(int level, bool checkGuild) const
- {
- if(checkGuild && mageGuildLevel() < level)
- return 0;
- int ret = 6 - level; //how many spells are available at this level
- if(subID == 2 && vstd::contains(builtBuildings,22)) //magic library in Tower
- ret++;
- return ret;
- }
- bool CGTownInstance::needsLastStack() const
- {
- if(garrisonHero)
- return true;
- else return false;
- }
- void CGTownInstance::onHeroVisit(const CGHeroInstance * h)
- {
- if(getOwner() != h->getOwner())
- {
- return;
- }
- cb->heroVisitCastle(id,h->id);
- }
- void CGTownInstance::onHeroLeave(const CGHeroInstance * h)
- {
- cb->stopHeroVisitCastle(id,h->id);
- }
- void CGTownInstance::initObj()
- {
- MetaString ms;
- ms << name << ", " << town->Name();
- cb->setHoverName(id,&ms);
- }
- //std::vector<int> CVisitableOPH::yourObjects()
- //{
- // std::vector<int> ret;
- // ret.push_back(51);//camp
- // ret.push_back(23);//tower
- // ret.push_back(61);//axis
- // ret.push_back(32);//garden
- // ret.push_back(100);//stone
- // ret.push_back(102);//tree
- // return ret;
- //}
- void CGVisitableOPH::onHeroVisit( const CGHeroInstance * h )
- {
- if(visitors.find(h->id)==visitors.end())
- {
- onNAHeroVisit(h->id, false);
- if(ID != 102) //not tree
- visitors.insert(h->id);
- }
- else
- {
- onNAHeroVisit(h->id, true);
- }
- }
- void CGVisitableOPH::initObj()
- {
- if(ID==102)
- ttype = ran()%3;
- else
- ttype = -1;
- }
- void CGVisitableOPH::treeSelected( int heroID, int resType, int resVal, int expVal, ui32 result )
- {
- if(result==0) //player agreed to give res for exp
- {
- cb->giveResource(cb->getOwner(heroID),resType,-resVal); //take resource
- cb->changePrimSkill(heroID,4,expVal); //give exp
- visitors.insert(heroID); //set state to visited
- }
- }
- void CGVisitableOPH::onNAHeroVisit(int heroID, bool alreadyVisited)
- {
- int id=0, subid=0, ot=0, val=1;
- switch(ID)
- {
- case 51:
- subid=0;
- ot=80;
- break;
- case 23:
- subid=1;
- ot=39;
- break;
- case 61:
- subid=2;
- ot=100;
- break;
- case 32:
- subid=3;
- ot=59;
- break;
- case 100:
- id=5;
- ot=143;
- val=1000;
- break;
- case 102:
- id = 5;
- subid = 1;
- ot = 146;
- val = 1;
- break;
- }
- if (!alreadyVisited)
- {
- switch (ID)
- {
- case 51:
- case 23:
- case 61:
- case 32:
- {
- cb->changePrimSkill(heroID,subid,val);
- InfoWindow iw;
- iw.components.push_back(Component(0,subid,val,0));
- iw.text << std::pair<ui8,ui32>(11,ot);
- iw.player = cb->getOwner(heroID);
- cb->showInfoDialog(&iw);
- break;
- }
- case 100: //give exp
- {
- InfoWindow iw;
- iw.components.push_back(Component(id,subid,val,0));
- iw.player = cb->getOwner(heroID);
- iw.text << std::pair<ui8,ui32>(11,ot);
- cb->showInfoDialog(&iw);
- cb->changePrimSkill(heroID,4,val);
- break;
- }
- case 102:
- {
- const CGHeroInstance *h = cb->getHero(heroID);
- val = VLC->heroh->reqExp(h->level+val) - VLC->heroh->reqExp(h->level);
- if(!ttype)
- {
- visitors.insert(heroID);
- InfoWindow iw;
- iw.components.push_back(Component(id,subid,1,0));
- iw.player = cb->getOwner(heroID);
- iw.text << std::pair<ui8,ui32>(11,148);
- cb->showInfoDialog(&iw);
- cb->changePrimSkill(heroID,4,val);
- break;
- }
- else
- {
- int res, resval;
- if(ttype==1)
- {
- res = 6;
- resval = 2000;
- ot = 149;
- }
- else
- {
- res = 5;
- resval = 10;
- ot = 151;
- }
- if(cb->getResource(h->tempOwner,res) < resval) //not enough resources
- {
- ot++;
- InfoWindow iw;
- iw.player = h->tempOwner;
- iw.text << std::pair<ui8,ui32>(11,ot);
- cb->showInfoDialog(&iw);
- return;
- }
- YesNoDialog sd;
- sd.player = cb->getOwner(heroID);
- sd.text << std::pair<ui8,ui32>(11,ot);
- sd.components.push_back(Component(id,subid,val,0));
- cb->showYesNoDialog(&sd,boost::bind(&CGVisitableOPH::treeSelected,this,heroID,res,resval,val,_1));
- }
- break;
- }
- }
- }
- else
- {
- ot++;
- InfoWindow iw;
- iw.player = cb->getOwner(heroID);
- iw.text << std::pair<ui8,ui32>(11,ot);
- cb->showInfoDialog(&iw);
- }
- }
- const std::string & CGVisitableOPH::getHoverText() const
- {
- int pom;
- switch(ID)
- {
- case 51:
- pom = 8;
- break;
- case 23:
- pom = 7;
- break;
- case 61:
- pom = 11;
- break;
- case 32:
- pom = 4;
- break;
- case 100:
- pom = 5;
- break;
- case 102:
- pom = 18;
- break;
- default:
- throw "Wrong CGVisitableOPH object ID!\n";
- }
- hoverName = VLC->objh->names[ID] + " " + VLC->objh->xtrainfo[pom];
- const CGHeroInstance *h = cb->getSelectedHero(cb->getCurrentPlayer());
- if(h)
- {
- hoverName += (vstd::contains(visitors,h->id))
- ? (VLC->generaltexth->allTexts[353]) //not visited
- : ( VLC->generaltexth->allTexts[352]); //visited
- }
- return hoverName;
- }
- bool CArmedInstance::needsLastStack() const
- {
- return false;
- }
- void CGCreature::onHeroVisit( const CGHeroInstance * h )
- {
- army.slots[0].first = subID;
- cb->startBattleI(h->id,army,pos,boost::bind(&CGCreature::endBattle,this,_1));
- }
- void CGCreature::endBattle( BattleResult *result )
- {
- if(result->winner==0)
- {
- cb->removeObject(id);
- }
- else
- {
- int killedAmount=0;
- for(std::set<std::pair<ui32,si32> >::iterator i=result->casualties[1].begin(); i!=result->casualties[1].end(); i++)
- if(i->first == subID)
- killedAmount += i->second;
- cb->setAmount(id, army.slots[0].second - killedAmount);
- }
- }
- void CGCreature::initObj()
- {
- si32 &amount = army.slots[0].second;
- CCreature &c = VLC->creh->creatures[subID];
- if(!amount)
- if(c.ammMax == c.ammMin)
- amount = c.ammMax;
- else
- amount = c.ammMin + (ran() % (c.ammMax - c.ammMin));
- }
|