#define VCMI_DLL #include "../stdafx.h" #include "map.h" #include "../hch/CObjectHandler.h" #include "../hch/CDefObjInfoHandler.h" #include "VCMI_Lib.h" #include #include #include "../hch/CLodHandler.h" #include /* * map.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 * */ static std::set convertBuildings(const std::set h3m, int castleID) { std::map mapa; std::set ret; std::ifstream b5("config/buildings5.txt"); while(!b5.eof()) { int a, b; b5 >> a >> b; if(castleID==8 && b==17) //magic university ID 17 (h3m) => 21 (vcmi) b=21; mapa[a]=b; } for(std::set::const_iterator i=h3m.begin();i!=h3m.end();i++) { if(mapa[*i]>=0) ret.insert(mapa[*i]); else if(mapa[*i] >= (-CREATURES_PER_TOWN)) // horde buildings { int level = (-mapa[*i]); if(h3m.find(20+(level*3)) != h3m.end()) //upgraded creature horde building { if(((castleID==1 && level==5) || (castleID==3 && level==3))) ret.insert(25); else ret.insert(19); } //else //{ if(((castleID==1 && level==5) || (castleID==3 && level==3))) ret.insert(24); else ret.insert(18); //} } else { tlog3<<"Conversion warning: unknown building "<<*i<<" in castle "<>i) { ret |= (128>>i); } } return ret; } static EDefType getDefType(CGDefInfo * a) { switch(a->id) { case 5: case 65: case 66: case 67: case 68: case 69: return ARTIFACT_DEF; //handled case 6: return PANDORA_DEF; //hanled case 10: return EVENTOBJ_DEF; //??? case 26: return EVENTOBJ_DEF; //handled case 33: return GARRISON_DEF; //handled case 34: case 70: case 62: //70 - random hero //62 - prison return HERO_DEF; //handled case 36: return GRAIL_DEF; //hanled case 53: case 17: case 18: case 19: case 20: case 42: case 87: case 220://cases 17 - 20 and 42 - tests return PLAYERONLY_DEF; //handled case 54: case 71: case 72: case 73: case 74: case 75: case 162: case 163: case 164: return CREATURES_DEF; //handled case 59: return SIGN_DEF; //handled case 77: return TOWN_DEF; //can be problematic, but handled case 79: case 76: return RESOURCE_DEF; //handled case 81: return SCHOLAR_DEF; //handled case 83: return SEERHUT_DEF; //handled case 91: return SIGN_DEF; //handled case 88: case 89: case 90: return SHRINE_DEF; //handled case 93: return SPELLSCROLL_DEF; //handled case 98: return TOWN_DEF; //handled case 113: return WITCHHUT_DEF; //handled case 214: return HEROPLACEHOLDER_DEF; //partially handled case 215: case 9: //??? return BORDERGUARD_DEF; //handled by analogy to seer huts ;] case 216: return CREGEN2_DEF; //handled case 217: return CREGEN_DEF; //handled case 218: return CREGEN3_DEF; //handled case 219: return GARRISON_DEF; //handled default: return TERRAINOBJ_DEF; // nothing to be handled } } static int readNormalNr (unsigned char * bufor, int pos, int bytCon = 4, bool cyclic = false) { int ret=0; int amp=1; for (int ir=0; ir=amp/2) { ret = ret-amp; } return ret; } static char readChar(unsigned char * bufor, int &i) { return bufor[i++]; } static std::string readString(unsigned char * bufor, int &i) { int len = readNormalNr(bufor,i); i+=4; std::string ret; ret.reserve(len); for(int gg=0; ggRoE maps { if(version) { CCreatureSet ret; std::pair ins; for(int ir=0;ir32768) rettt = 65536-rettt+VLC->creh->creatures.size()-16; ins.first = rettt; ins.second = readNormalNr(bufor,i+ir*4+2, 2); std::pair > tt(ir,ins); ret.slots.insert(tt); } i+=number*4; return ret; } else { CCreatureSet ret; std::pair ins; for(int ir=0;ir220) rettt = 256-rettt+VLC->creh->creatures.size()-16; ins.first = rettt; ins.second = readNormalNr(bufor,i+ir*3+1, 2); std::pair > tt(ir,ins); ret.slots.insert(tt); } i+=number*3; return ret; } } CMapHeader::CMapHeader(unsigned char *map) { int i=0; initFromMemory(map,i); } CMapHeader::CMapHeader() { areAnyPLayers = difficulty = levelLimit = howManyTeams = 0; height = width = twoLevel = -1; } void CMapHeader::initFromMemory( unsigned char *bufor, int &i ) { version = (Eformat)(readNormalNr(bufor,i)); i+=4; //map version areAnyPLayers = readChar(bufor,i); //invalid on some maps height = width = (readNormalNr(bufor,i)); i+=4; // wymiary mapy twoLevel = readChar(bufor,i); //czy sa lochy int pom; name = readString(bufor,i); description= readString(bufor,i); difficulty = readChar(bufor,i); // reading map difficulty if(version != RoE) levelLimit = readChar(bufor,i); // hero level limit else levelLimit = 0; loadPlayerInfo(pom, bufor, i); loadViCLossConditions(bufor, i); howManyTeams=bufor[i++]; //read number of teams if(howManyTeams>0) //read team numbers { for(int rr=0; rr<8; ++rr) { players[rr].team = bufor[i++]; } } else//no alliances { for(int i=0;iRoE) //probably reserved for further heroes i+=4; } void CMapHeader::loadPlayerInfo( int &pom, unsigned char * bufor, int &i ) { players.resize(8); for (pom=0;pom<8;pom++) { players[pom].canHumanPlay = bufor[i++]; players[pom].canComputerPlay = bufor[i++]; if ((!(players[pom].canHumanPlay || players[pom].canComputerPlay))) { switch(version) { case SoD: case WoG: i+=13; break; case AB: i+=12; break; case RoE: i+=6; break; } continue; } players[pom].AITactic = bufor[i++]; if(version == SoD || version == WoG) players[pom].p7= bufor[i++]; else players[pom].p7= -1; players[pom].allowedFactions = 0; players[pom].allowedFactions += bufor[i++]; if(version != RoE) players[pom].allowedFactions += (bufor[i++])*256; players[pom].isFactionRandom = bufor[i++]; players[pom].hasMainTown = bufor[i++]; if (players[pom].hasMainTown) { if(version != RoE) { players[pom].generateHeroAtMainTown = bufor[i++]; players[pom].generateHero = bufor[i++]; } else { players[pom].generateHeroAtMainTown = false; players[pom].generateHero = false; } players[pom].posOfMainTown.x = bufor[i++]; players[pom].posOfMainTown.y = bufor[i++]; players[pom].posOfMainTown.z = bufor[i++]; } players[pom].p8= bufor[i++]; players[pom].p9= bufor[i++]; if(players[pom].p9!=0xff) { players[pom].mainHeroPortrait = bufor[i++]; players[pom].mainHeroName = readString(bufor,i); } if(version != RoE) { i++; ////unknown byte int heroCount = bufor[i++]; i+=3; for (int pp=0;pp2) victoryCondition.pos = int3(-1,-1,-1); else { victoryCondition.pos.x = bufor[i+2]; victoryCondition.pos.y = bufor[i+3]; victoryCondition.pos.z = bufor[i+4]; } nr = 3; break; } case beatHero: case captureCity: case beatMonster: { victoryCondition.pos.x = bufor[i+2]; victoryCondition.pos.y = bufor[i+3]; victoryCondition.pos.z = bufor[i+4]; nr = 3; break; } case takeDwellings: case takeMines: { nr = 0; break; } case transportItem: { victoryCondition.ID = bufor[i+2]; victoryCondition.pos.x = bufor[i+3]; victoryCondition.pos.y = bufor[i+4]; victoryCondition.pos.z = bufor[i+5]; nr = 4; break; } } victoryCondition.allowNormalVictory = bufor[i++]; victoryCondition.appliesToAI = bufor[i++]; i+=nr; } lossCondition.typeOfLossCon = (ElossCon)bufor[i++]; switch (lossCondition.typeOfLossCon) //read loss conditions { case lossCastle: { lossCondition.castlePos.x=bufor[i++]; lossCondition.castlePos.y=bufor[i++]; lossCondition.castlePos.z=bufor[i++]; break; } case lossHero: { lossCondition.heroPos.x=bufor[i++]; lossCondition.heroPos.y=bufor[i++]; lossCondition.heroPos.z=bufor[i++]; break; } case timeExpires: { lossCondition.timeLimit = readNormalNr(bufor,i++,2); i++; break; } } } CMapHeader::~CMapHeader() { } void Mapa::initFromBytes(unsigned char * bufor) { int i=0; initFromMemory(bufor,i); timeHandler th; th.getDif(); readHeader(bufor, i); tlog0<<"\tReading header: "<defInfo) continue; addBlockVisTiles(objects[f]); } tlog0<<"\tCalculating blocked/visitable tiles: "<pos.x + fx - 7; int yVal = obj->pos.y + fy - 5; int zVal = obj->pos.z; if(xVal>=0 && xVal=0 && yValdefInfo->visitMap[fy] >> (7 - fx)) & 1)) { curt.visitableObjects -= obj; curt.visitable = curt.visitableObjects.size(); } if(total || !((obj->defInfo->blockMap[fy] >> (7 - fx)) & 1)) { curt.blockingObjects -= obj; curt.blocked = curt.blockingObjects.size(); } } } } } void Mapa::addBlockVisTiles(CGObjectInstance * obj) { for(int fx=0; fx<8; ++fx) { for(int fy=0; fy<6; ++fy) { int xVal = obj->pos.x + fx - 7; int yVal = obj->pos.y + fy - 5; int zVal = obj->pos.z; if(xVal>=0 && xVal=0 && yValdefInfo->visitMap[fy] >> (7 - fx)) & 1)) { curt.visitableObjects.push_back(obj); curt.visitable = true; } if(!((obj->defInfo->blockMap[fy] >> (7 - fx)) & 1)) { curt.blockingObjects.push_back(obj); curt.blocked = true; } } } } } Mapa::Mapa(std::string filename) { const int bufsize = 65536; int mapsize = 0; tlog0<<"Opening map file: "< mapstr; // Read a map by chunks // We could try to read the map size directly (cf RFC 1952) and then read // directly the whole map, but that would create more problems. do { unsigned char *buf = new unsigned char[bufsize]; int ret = gzread(map, buf, bufsize); if (ret == 0 || ret == -1) { delete [] buf; break; } mapstr.push_back(buf); mapsize += ret; } while(1); gzclose(map); // Now that we know the uncompressed size, reassemble the chunks unsigned char *initTable = new unsigned char[mapsize]; std::vector::iterator it; int offset; int tocopy = mapsize; for (it = mapstr.begin(), offset = 0; it != mapstr.end(); it++, offset+=bufsize ) { memcpy(&initTable[offset], *it, tocopy > bufsize ? bufsize : tocopy); tocopy -= bufsize; delete [] *it; } tlog0<<"done."<serial < 0) //def not present in the main vector in defobjinfo // delete defy[i]; if(terrain) { for (int ii=0;ii::iterator i = events.begin(); i != events.end(); i++) delete *i; } CGHeroInstance * Mapa::getHero(int ID, int mode) { if (mode != 0) #ifndef __GNUC__ throw new std::exception("gs->getHero: This mode is not supported!"); #else throw new std::exception(); #endif for(unsigned int i=0; isubID == ID) return heroes[i]; return NULL; } int Mapa::loadSeerHut( unsigned char * bufor, int i, CGObjectInstance *& nobj ) { CGSeerHut *hut = new CGSeerHut(); nobj = hut; if(version>RoE) { loadQuest(hut,bufor,i); } else //RoE { int artID = bufor[i]; ++i; if(artID!=255) //not none quest { hut->m5arts.push_back(artID); hut->missionType = 5; } else { hut->missionType = 255; } } if(hut->missionType!=255) { unsigned char rewardType = bufor[i]; ++i; hut->rewardType = rewardType; switch(rewardType) { case 1: { hut->rVal = readNormalNr(bufor,i); i+=4; break; } case 2: { hut->rVal = readNormalNr(bufor,i); i+=4; break; } case 3: { hut->rVal = bufor[i]; ++i; break; } case 4: { hut->rVal = bufor[i]; ++i; break; } case 5: { hut->rID = bufor[i]; ++i; hut->rVal = readNormalNr(bufor,i, 3); i+=3; i+=1; break; } case 6: { hut->rID = bufor[i]; ++i; hut->rVal = bufor[i]; ++i; break; } case 7: { hut->rID = bufor[i]; ++i; hut->rVal = bufor[i]; ++i; break; } case 8: { hut->rID = readNormalNr(bufor,i, (version == RoE ? 1 : 2)); i+=(version == RoE ? 1 : 2); break; } case 9: { hut->rID = bufor[i]; ++i; break; } case 10: { if(version>RoE) { hut->rID = readNormalNr(bufor,i, 2); i+=2; hut->rVal = readNormalNr(bufor,i, 2); i+=2; } else { hut->rID = bufor[i]; ++i; hut->rVal = readNormalNr(bufor,i, 2); i+=2; } break; } }// end of internal switch i+=2; } else //missionType==255 { i+=3; } return i; } void Mapa::loadTown( CGObjectInstance * &nobj, unsigned char * bufor, int &i, int subid) { CGTownInstance * nt = new CGTownInstance(); //(*(static_cast(nt))) = *nobj; //delete nobj; nobj = nt; nt->identifier = 0; if(version>RoE) { readNormalNr(bufor,i); i+=4; } nt->tempOwner = bufor[i]; ++i; if(readChar(bufor,i)) //has name nt->name = readString(bufor,i); if(readChar(bufor,i))//true if garrison isn't empty nt->army = readCreatureSet(bufor,i,7,(version>RoE)); nt->army.formation = bufor[i]; ++i; if(readChar(bufor,i)) //custom buildings info { //built buildings for(int byte=0;byte<6;byte++) { for(int bit=0;bit<8;bit++) if(bufor[i] & (1<builtBuildings.insert(byte*8+bit); i++; } //forbidden buildings for(int byte=6;byte<12;byte++) { for(int bit=0;bit<8;bit++) if(bufor[i] & (1<forbiddenBuildings.insert((byte-6)*8+bit); i++; } nt->builtBuildings = convertBuildings(nt->builtBuildings,subid); nt->forbiddenBuildings = convertBuildings(nt->forbiddenBuildings,subid); } else //standard buildings { if(readChar(bufor,i)) //has fort nt->builtBuildings.insert(7); nt->builtBuildings.insert(-50); //means that set of standard building should be included } int ist = i; if(version>RoE) { for(i; iobligatorySpells.push_back((i-ist)*8+yy); } } } } ist = i; for(i; ipossibleSpells.push_back((i-ist)*8+yy); } } } /////// reading castle events ////////////////////////////////// int numberOfEvent = readNormalNr(bufor,i); i+=4; for(int gh = 0; gh AB) { nce.forHuman = bufor[i]; ++i; } else nce.forHuman = true; nce.forComputer = bufor[i]; ++i; nce.firstShow = readNormalNr(bufor,i, 2); i+=2; nce.forEvery = bufor[i]; ++i; i+=17; for(int kk=0; kk<6; ++kk) { nce.bytes[kk] = bufor[i]; ++i; } for(int vv=0; vv<7; ++vv) { nce.gen[vv] = readNormalNr(bufor,i, 2); i+=2; } i+=4; nt->events.insert(nce); }//castle events have been read if(version > AB) { nt->alignment = bufor[i]; ++i; } else nt->alignment = 0xff; i+=3; nt->builded = 0; nt->destroyed = 0; nt->garrisonHero = NULL; } void Mapa::loadHero( CGObjectInstance * &nobj, unsigned char * bufor, int &i ) { CGHeroInstance * nhi = new CGHeroInstance; nobj=nhi; int identifier = 0; if(version>RoE) { identifier = readNormalNr(bufor,i, 4); i+=4; } ui8 owner = bufor[i++]; nobj->subID = readNormalNr(bufor,i, 1); ++i; for(unsigned int j=0; jsubID == nobj->subID) { *nhi = *predefinedHeroes[j]; break; } } nobj->setOwner(owner); //(*(static_cast(nhi))) = *nobj; //delete nobj; nhi->portrait = nhi->subID; for(unsigned int j=0; jsubID) { nhi->name = disposedHeroes[j].name; nhi->portrait = disposedHeroes[j].portrait; break; } } nhi->identifier = identifier; if(readChar(bufor,i))//true if hero has nonstandard name nhi->name = readString(bufor,i); if(version>AB) { if(readChar(bufor,i))//true if hero's experience is greater than 0 { nhi->exp = readNormalNr(bufor,i); i+=4; } else nhi->exp = 0xffffffff; } else { nhi->exp = readNormalNr(bufor,i); i+=4; if(!nhi->exp) //0 means "not set" in <=AB maps nhi->exp = 0xffffffff; } bool portrait=bufor[i]; ++i; if (portrait) nhi->portrait = bufor[i++]; if(readChar(bufor,i))//true if hero has specified abilities { int howMany = readNormalNr(bufor,i); i+=4; nhi->secSkills.resize(howMany); for(int yy=0; yysecSkills[yy].first = readNormalNr(bufor,i, 1); ++i; nhi->secSkills[yy].second = readNormalNr(bufor,i, 1); ++i; } } if(readChar(bufor,i))//true if hero has nonstandard garrison nhi->army = readCreatureSet(bufor,i,7,(version>RoE)); nhi->army.formation =bufor[i]; ++i; //formation bool artSet = bufor[i]; ++i; //true if artifact set is not default (hero has some artifacts) int artmask = version == RoE ? 0xff : 0xffff; int artidlen = version == RoE ? 1 : 2; if(artSet) { for(int pom=0;pom<16;pom++) { int id = readNormalNr(bufor,i, artidlen); i+=artidlen; if(id!=artmask) nhi->artifWorn[pom] = id; } //misc5 art //17 if(version>=SoD) { int id = readNormalNr(bufor,i, artidlen); i+=artidlen; if(id!=artmask) nhi->artifWorn[16] = id; else nhi->artifWorn[16] = 3; //catapult by default } //spellbook int id = readNormalNr(bufor,i, artidlen); i+=artidlen; if(id!=artmask) nhi->artifWorn[17] = id; //19 //???what is that? gap in file or what? - it's probably fifth slot.. if(version>RoE) { id = readNormalNr(bufor,i, artidlen); i+=artidlen; if(id!=artmask) nhi->artifWorn[18] = id; } else i+=1; //bag artifacts //20 int amount = readNormalNr(bufor,i, 2); i+=2; //number of artifacts in hero's bag if(amount>0) { for(int ss=0; ssartifacts.push_back(id); } } } //artifacts nhi->patrol.patrolRadious = readNormalNr(bufor,i, 1); ++i; if(nhi->patrol.patrolRadious == 0xff) nhi->patrol.patrolling = false; else nhi->patrol.patrolling = true; if(version>RoE) { if(readChar(bufor,i))//true if hero has nonstandard (mapmaker defined) biography nhi->biography = readString(bufor,i); nhi->sex = !(bufor[i]); ++i; } //spells if(version>AB) { bool areSpells = bufor[i]; ++i; if(areSpells) //TODO: sprawdzić //seems to be ok - tow { nhi->spells.insert(0xffffffff); //placeholder "preset spells" int ist = i; for(i; ispells.insert((i-ist)*8+yy); } } } } } else if(version==AB) //we can read one spell { unsigned char buff = bufor[i]; ++i; if(buff != 254) { nhi->spells.insert(0xffffffff); //placeholder "preset spells" if(buff < 254) //255 means no spells nhi->spells.insert(buff); } } //spells loaded if(version>AB) { if(readChar(bufor,i))//customPrimSkills { nhi->primSkills.resize(4); for(int xx=0;xx<4;xx++) nhi->primSkills[xx] = bufor[i++]; } } i+=16; nhi->moveDir = 4; nhi->isStanding = true; nhi->level = -1; nhi->mana = -1; nhi->movement = -1; } void Mapa::readRumors( unsigned char * bufor, int &i) { int rumNr = readNormalNr(bufor,i,4);i+=4; for (int it=0;it=SoD) { disp = bufor[i++]; disposedHeroes.resize(disp); for(int g=0; g=SoD) { //reading allowed spells (9 bytes) ist=i; //starting i for loop for(i; iID = HEROI_TYPE; cgh->subID = z; if(readChar(bufor,i))//true if hore's experience is greater than 0 { cgh->exp = readNormalNr(bufor,i); i+=4; } else cgh->exp = 0; if(readChar(bufor,i))//true if hero has specified abilities { int howMany = readNormalNr(bufor,i); i+=4; cgh->secSkills.resize(howMany); for(int yy=0; yysecSkills[yy].first = readNormalNr(bufor,i, 1); ++i; cgh->secSkills[yy].second = readNormalNr(bufor,i, 1); ++i; } } bool artSet = bufor[i]; ++i; //true if artifact set is not default (hero has some artifacts) int artmask = version == RoE ? 0xff : 0xffff; int artidlen = version == RoE ? 1 : 2; if(artSet) { for(int pom=0;pom<16;pom++) { int id = readNormalNr(bufor,i, artidlen); i+=artidlen; if(id!=artmask) cgh->artifWorn[pom] = id; } //misc5 art //17 if(version>=SoD) { i+=2; //int id = readNormalNr(bufor,i, artidlen); i+=artidlen; //if(id!=artmask) // spec->artifWorn[16] = id; } //spellbook int id = readNormalNr(bufor,i, artidlen); i+=artidlen; if(id!=artmask) cgh->artifWorn[17] = id; //19 //???what is that? gap in file or what? - it's probably fifth slot.. if(version>RoE) { id = readNormalNr(bufor,i, artidlen); i+=artidlen; if(id!=artmask) cgh->artifWorn[18] = id; } else i+=1; //bag artifacts //20 int amount = readNormalNr(bufor,i, 2); i+=2; //number of artifacts in hero's bag if(amount>0) { for(int ss=0; ssartifacts.push_back(id); } } } //artifacts if(readChar(bufor,i))//customBio cgh->biography = readString(bufor,i); int sex = bufor[i++]; // 0xFF is default, 00 male, 01 female if(readChar(bufor,i))//are spells { int ist = i; for(i; ispells.insert((i-ist)*8+yy); } } } } if(readChar(bufor,i))//customPrimSkills { cgh->primSkills.resize(4); for(int xx=0;xx<4;xx++) cgh->primSkills[xx] = bufor[i++]; } predefinedHeroes.push_back(cgh); } break; } case RoE: i+=0; break; } } void Mapa::readTerrain( unsigned char * bufor, int &i) { terrain = new TerrainTile**[width]; // allocate memory for (int ii=0;ii(bufor[i++]); terrain[z][c][0].terview = bufor[i++]; terrain[z][c][0].nuine = static_cast(bufor[i++]); terrain[z][c][0].rivDir = bufor[i++]; terrain[z][c][0].malle = static_cast(bufor[i++]); terrain[z][c][0].roadDir = bufor[i++]; terrain[z][c][0].siodmyTajemniczyBajt = bufor[i++]; terrain[z][c][0].blocked = 0; terrain[z][c][0].visitable = 0; } } if (twoLevel) // read underground terrain { for (int c=0; c(bufor[i++]); terrain[z][c][1].terview = bufor[i++]; terrain[z][c][1].nuine = static_cast(bufor[i++]); terrain[z][c][1].rivDir = bufor[i++]; terrain[z][c][1].malle = static_cast(bufor[i++]); terrain[z][c][1].roadDir = bufor[i++]; terrain[z][c][1].siodmyTajemniczyBajt = bufor[i++]; terrain[z][c][1].blocked = 0; terrain[z][c][1].visitable = 0; } } } } void Mapa::readDefInfo( unsigned char * bufor, int &i) { int defAmount = readNormalNr(bufor,i); i+=4; defy.reserve(defAmount); for (int idd = 0 ; iddname.reserve(nameLength); for (int cd=0;cdname += bufor[i++]; } std::transform(vinya->name.begin(),vinya->name.end(),vinya->name.begin(),(int(*)(int))toupper); unsigned char bytes[12]; for (int v=0; v<12; v++) // read info { bytes[v] = bufor[i++]; } vinya->terrainAllowed = readNormalNr(bufor,i,2);i+=2; vinya->terrainMenu = readNormalNr(bufor,i,2);i+=2; vinya->id = readNormalNr(bufor,i,4);i+=4; vinya->subid = readNormalNr(bufor,i,4);i+=4; vinya->type = bufor[i++]; vinya->printPriority = bufor[i++]; for (int zi=0; zi<6; zi++) { vinya->blockMap[zi] = reverse(bytes[zi]); } for (int zi=0; zi<6; zi++) { vinya->visitMap[zi] = reverse(bytes[6+zi]); } i+=16; if(vinya->id!=HEROI_TYPE && vinya->id!=70) { CGDefInfo *h = VLC->dobjinfo->gobjs[vinya->id][vinya->subid]; if(!h) { //remove fake entry VLC->dobjinfo->gobjs[vinya->id].erase(vinya->subid); if(VLC->dobjinfo->gobjs[vinya->id].size()) VLC->dobjinfo->gobjs.erase(vinya->id); tlog2<<"\t\tWarning: no defobjinfo entry for object ID="<id<<" subID=" << vinya->subid<visitDir = VLC->dobjinfo->gobjs[vinya->id][vinya->subid]->visitDir; } } else { vinya->visitDir = 0xff; } if(vinya->id == 26) std::memset(vinya->blockMap,255,6); //calculating coverageMap std::string nameCopy = vinya->name; std::string msk = spriteh->getTextFile(nameCopy.replace( nameCopy.size()-4, 4, ".MSK" )); for(int i=0; i<6; ++i) { vinya->coverageMap[i] = msk[i+2]; } defy.push_back(vinya); // add this def to the vector } } void Mapa::readObjects( unsigned char * bufor, int &i) { int howManyObjs = readNormalNr(bufor,i, 4); i+=4; for(int ww=0; wwid) { case 26: //for event objects { CGEvent *evnt = new CGEvent(); nobj = evnt; bool guardMess = bufor[i]; ++i; if(guardMess) { int messLong = readNormalNr(bufor,i, 4); i+=4; if(messLong>0) { for(int yy=0; yymessage +=bufor[i+yy]; } i+=messLong; } if(bufor[i++]) { evnt->army = readCreatureSet(bufor,i,7,(version>RoE)); } i+=4; } evnt->gainedExp = readNormalNr(bufor,i, 4); i+=4; evnt->manaDiff = readNormalNr(bufor,i, 4); i+=4; evnt->moraleDiff = readNormalNr(bufor,i, 1, true); ++i; evnt->luckDiff = readNormalNr(bufor,i, 1, true); ++i; evnt->resources.resize(RESOURCE_QUANTITY); for(int x=0; x<7; x++) { evnt->resources[x] = readNormalNr(bufor,i); i+=4; } evnt->primskills.resize(PRIMARY_SKILLS); for(int x=0; x<4; x++) { evnt->primskills[x] = readNormalNr(bufor,i, 1); i++; } int gabn; //number of gained abilities gabn = readNormalNr(bufor,i, 1); ++i; for(int oo = 0; ooabilities.push_back(readNormalNr(bufor,i, 1)); ++i; evnt->abilityLevels.push_back(readNormalNr(bufor,i, 1)); ++i; } int gart = readNormalNr(bufor,i, 1); ++i; //number of gained artifacts for(int oo = 0; ooartifacts.push_back(readNormalNr(bufor,i, (version == RoE ? 1 : 2))); i+=(version == RoE ? 1 : 2); } int gspel = readNormalNr(bufor,i, 1); ++i; //number of gained spells for(int oo = 0; oospells.push_back(readNormalNr(bufor,i, 1)); ++i; } int gcre = readNormalNr(bufor,i, 1); ++i; //number of gained creatures evnt->creatures = readCreatureSet(bufor,i,gcre,(version>RoE)); i+=8; evnt->availableFor = readNormalNr(bufor,i, 1); ++i; evnt->computerActivate = readNormalNr(bufor,i, 1); ++i; evnt->removeAfterVisit = readNormalNr(bufor,i, 1); ++i; evnt->humanActivate = true; i+=4; break; } case 34: case 70: case 62: //34 - hero; 70 - random hero; 62 - prison { loadHero(nobj, bufor, i); break; } case 4: //Arena case 51: //Mercenary Camp case 23: //Marletto Tower case 61: // Star Axis case 32: // Garden of Revelation case 100: //Learning Stone case 102: //Tree of Knowledge case 41: //Library of Enlightenment case 47: //School of Magic case 107: //School of War { nobj = new CGVisitableOPH(); break; } case 55: //mystical garden case 112://windmill case 109://water wheel { nobj = new CGVisitableOPW(); break; } case 43: //teleport case 44: //teleport case 45: //teleport case 103://subterranean gate { nobj = new CGTeleport(); break; } case 12: //campfire case 29: //Flotsam case 82: //Sea Chest case 86: //Shipwreck Survivor case 101://treasure chest { nobj = new CGPickable(); break; } case 54: //Monster case 71: case 72: case 73: case 74: case 75: // Random Monster 1 - 4 case 162: case 163: case 164: // Random Monster 5 - 7 { CGCreature *cre = new CGCreature(); nobj = cre; if(version>RoE) { cre->identifier = readNormalNr(bufor,i); i+=4; } cre->army.slots[0].second = readNormalNr(bufor,i, 2); i+=2; cre->character = bufor[i]; ++i; bool isMesTre = bufor[i]; ++i; //true if there is message or treasury if(isMesTre) { cre->message = readString(bufor,i); cre->resources.resize(RESOURCE_QUANTITY); for(int j=0; j<7; j++) { cre->resources[j] = readNormalNr(bufor,i); i+=4; } int artID = readNormalNr(bufor,i, (version == RoE ? 1 : 2)); i+=(version == RoE ? 1 : 2); if(version==RoE) { if(artID!=0xff) cre->gainedArtifact = artID; else cre->gainedArtifact = -1; } else { if(artID!=0xffff) cre->gainedArtifact = artID; else cre->gainedArtifact = -1; } } cre->neverFlees = bufor[i]; ++i; cre->notGrowingTeam = bufor[i]; ++i; i+=2;; break; } case 59: case 91: //ocean bottle and sign { CGSignBottle *sb = new CGSignBottle(); nobj = sb; sb->message = readString(bufor,i); i+=4; break; } case 83: //seer's hut { i = loadSeerHut(bufor, i, nobj); break; } case 113: //witch hut { CGWitchHut *wh = new CGWitchHut(); nobj = wh; if(version>RoE) //in reo we cannot specify it - all are allowed (I hope) { int ist=i; //starting i for loop for(i; iallowedAbilities.push_back((i-ist)*8+yy); } } } } else //(RoE map) { for(int gg=0; ggallowedAbilities.push_back(gg); } } break; } case 81: //scholar { CGScholar *sch = new CGScholar(); nobj = sch; sch->bonusType = bufor[i++]; sch->bonusID = bufor[i++]; i+=6; break; } case 33: case 219: //garrison { CGGarrison *gar = new CGGarrison(); nobj = gar; nobj->setOwner(bufor[i++]); i+=3; gar->army = readCreatureSet(bufor,i,7,(version>RoE)); if(version > RoE) { gar->removableUnits = bufor[i]; ++i; } else gar->removableUnits = true; i+=8; break; } case 5: //artifact case 65: case 66: case 67: case 68: case 69: //random artifact case 93: //spell scroll { CGArtifact *art = new CGArtifact(); nobj = art; bool areSettings = bufor[i]; ++i; if(areSettings) { art->message = readString(bufor,i); bool areGuards = bufor[i]; ++i; if(areGuards) { art->army = readCreatureSet(bufor,i,7,(version>RoE)); } i+=4; } if(defInfo->id==93) { art->spell = readNormalNr(bufor,i); i+=4; } break; } case 76: case 79: //random resource; resource { CGResource *res = new CGResource(); nobj = res; bool isMessGuard = bufor[i]; ++i; if(isMessGuard) { res->message = readString(bufor,i); if(bufor[i++]) { res->army = readCreatureSet(bufor,i,7,(version>RoE)); } i+=4; } res->amount = readNormalNr(bufor,i); i+=4; if (defInfo->subid == 6) // Gold is multiplied by 100. res->amount *= 100; i+=4; break; } case 77: case 98: //random town; town { loadTown(nobj, bufor, i, defInfo->subid); break; } case 53: { nobj = new CGMine(); nobj->setOwner(bufor[i++]); i+=3; break; } case 17: case 18: case 19: case 20: //dwellings { nobj = new CGDwelling(); nobj->setOwner(bufor[i++]); i+=3; break; } case 42: //lighthouse case 220://mine (?) { nobj = new CGObjectInstance(); nobj->setOwner(bufor[i++]); i+=3; break; } case 88: case 89: case 90: //spell shrine { CGShrine * shr = new CGShrine(); nobj = shr; shr->spell = bufor[i]; i+=4; break; } case 6: { CGPandoraBox *box = new CGPandoraBox(); nobj = box; bool messg = bufor[i]; ++i; if(messg) { box->message = readString(bufor,i); if(bufor[i++]) { box->army = readCreatureSet(bufor,i,7,(version>RoE)); } i+=4; } box->gainedExp = readNormalNr(bufor,i, 4); i+=4; box->manaDiff = readNormalNr(bufor,i, 4); i+=4; box->moraleDiff = readNormalNr(bufor,i, 1, true); ++i; box->luckDiff = readNormalNr(bufor,i, 1, true); ++i; box->resources.resize(RESOURCE_QUANTITY); for(int x=0; x<7; x++) { box->resources[x] = readNormalNr(bufor,i); i+=4; } box->primskills.resize(PRIMARY_SKILLS); for(int x=0; x<4; x++) { box->primskills[x] = readNormalNr(bufor,i, 1); i++; } int gabn; //number of gained abilities gabn = readNormalNr(bufor,i, 1); ++i; for(int oo = 0; ooabilities.push_back(readNormalNr(bufor,i, 1)); ++i; box->abilityLevels.push_back(readNormalNr(bufor,i, 1)); ++i; } int gart = readNormalNr(bufor,i, 1); ++i; //number of gained artifacts for(int oo = 0; oo RoE) { box->artifacts.push_back(readNormalNr(bufor,i, 2)); i+=2; } else { box->artifacts.push_back(readNormalNr(bufor,i, 1)); i+=1; } } int gspel = readNormalNr(bufor,i, 1); ++i; //number of gained spells for(int oo = 0; oospells.push_back(readNormalNr(bufor,i, 1)); ++i; } int gcre = readNormalNr(bufor,i, 1); ++i; //number of gained creatures box->creatures = readCreatureSet(bufor,i,gcre,(version>RoE)); i+=8; break; } case 36: { grailPos = pos; grailRadious = readNormalNr(bufor,i); i+=4; continue; } case 217: { nobj = new CGDwelling(); CCreGenObjInfo * spec = new CCreGenObjInfo; spec->player = readNormalNr(bufor,i); i+=4; spec->identifier = readNormalNr(bufor,i); i+=4; if(!spec->identifier) { spec->asCastle = false; spec->castles[0] = bufor[i]; ++i; spec->castles[1] = bufor[i]; ++i; } else { spec->asCastle = true; } nobj->setOwner(spec->player); nobj->info = spec; break; } case 216: { nobj = new CGDwelling(); CCreGen2ObjInfo * spec = new CCreGen2ObjInfo; spec->player = readNormalNr(bufor,i); i+=4; spec->identifier = readNormalNr(bufor,i); i+=4; if(!spec->identifier) { spec->asCastle = false; spec->castles[0] = bufor[i]; ++i; spec->castles[1] = bufor[i]; ++i; } else { spec->asCastle = true; } spec->minLevel = bufor[i]; ++i; spec->maxLevel = bufor[i]; ++i; nobj->setOwner(spec->player); nobj->info = spec; break; } case 218: { nobj = new CGDwelling(); CCreGen3ObjInfo * spec = new CCreGen3ObjInfo; spec->player = bufor[i]; ++i; i+=3; spec->minLevel = bufor[i]; ++i; spec->maxLevel = bufor[i]; ++i; if(spec->maxLevel>7) spec->maxLevel = 7; if(spec->minLevel<1) spec->minLevel = 1; nobj->setOwner(spec->player); nobj->info = spec; break; } case 215: { CGQuestGuard *guard = new CGQuestGuard(); nobj = guard; loadQuest(guard, bufor, i); break; } case 28: //faerie ring case 14: //Swan pond case 38: //idol of fortune case 30: //Fountain of Fortune case 64: //Rally Flag case 56: //oasis case 96: //temple case 110://Watering Hole case 31: //Fountain of Youth case 11: //Buoy case 52: //Mermaid case 94: //Stables { nobj = new CGBonusingObject(); break; } case 49: //Magic Well { nobj = new CGMagicWell(); break; } case 58: //Redwood Observatory case 60: //Pillar of Fire { nobj = new CGObservatory(); break; } case 22: //Corpse case 39: //Lean To case 105://Wagon case 108://Warrior's Tomb { nobj = new CGOnceVisitable(); break; } case 8: //Boat { nobj = new CGBoat(); break; } case 92: //Sirens { nobj = new CGSirens(); break; } case 87: //Shipyard { nobj = new CGShipyard(); nobj->setOwner(readNormalNr(bufor,i)); i+=4; break; } case 214: //hero placeholder { i+=3; //TODO: handle it more properly } case 10: //Keymaster { nobj = new CGKeymasterTent(); break; } case 9: //Border Guard { nobj = new CGBorderGuard(); break; } case 212: //Border Gate { nobj = new CGBorderGate(); break; } case 27: case 37: //Eye and Hut of Magi { nobj = new CGMagi(); break; } case 16: case 24: case 25: case 84: case 85: //treasure bank { nobj = new CBank(); break; } case 63: //Pyramid { nobj = new CGPyramid(); break; } case 13: //Cartographer { nobj = new CCartographer(); break; } case 48: //Magic Spring { nobj = new CGMagicSpring(); break; } default: nobj = new CGObjectInstance(); } //end of main switch nobj->pos = pos; nobj->ID = defInfo->id; nobj->id = objects.size(); if(nobj->ID != HEROI_TYPE) nobj->subID = defInfo->subid; nobj->defInfo = defInfo; objects.push_back(nobj); if(nobj->ID==TOWNI_TYPE) towns.push_back(static_cast(nobj)); if(nobj->ID==HEROI_TYPE) heroes.push_back(static_cast(nobj)); } std::sort(heroes.begin(), heroes.end(), boost::bind(&CGHeroInstance::subID, _1) < boost::bind(&CGHeroInstance::subID, _2)); } void Mapa::readEvents( unsigned char * bufor, int &i ) { int numberOfEvents = readNormalNr(bufor,i); i+=4; for(int yyoo=0; yyooname = std::string(); ne->message = std::string(); int nameLen = readNormalNr(bufor,i); i+=4; for(int qq=0; qqname += bufor[i]; ++i; } int messLen = readNormalNr(bufor,i); i+=4; for(int qq=0; qqmessage +=bufor[i]; ++i; } ne->resources.resize(RESOURCE_QUANTITY); for(int k=0; k < 7; k++) { ne->resources[k] = readNormalNr(bufor,i); i+=4; } ne->players = bufor[i]; ++i; if(version>AB) { ne->humanAffected = bufor[i]; ++i; } else ne->humanAffected = true; ne->computerAffected = bufor[i]; ++i; ne->firstOccurence = readNormalNr(bufor,i, 2); i+=2; ne->nextOccurence = bufor[i]; ++i; char unknown[17]; memcpy(unknown, bufor+i, 17); i+=17; events.push_back(ne); } } bool Mapa::isInTheMap(const int3 &pos) const { if(pos.x<0 || pos.y<0 || pos.z<0 || pos.x >= width || pos.y >= height || pos.z > twoLevel) return false; else return true; } void Mapa::loadQuest(CQuest * guard, unsigned char * bufor, int & i) { guard->missionType = bufor[i]; ++i; int len1, len2, len3; switch(guard->missionType) { case 0: return; case 2: { guard->m2stats.resize(4); for(int x=0; x<4; x++) { guard->m2stats[x] = bufor[i++]; } } break; case 1: case 3: case 4: { guard->m13489val = readNormalNr(bufor,i); i+=4; break; } case 5: { int artNumber = bufor[i]; ++i; for(int yy=0; yym5arts.push_back(readNormalNr(bufor,i, 2)); i+=2; } break; } case 6: { int typeNumber = bufor[i]; ++i; for(int hh=0; hhm6creatures.push_back(std::make_pair(creType,creNumb)); } break; } case 7: { guard->m7resources.resize(7); for(int x=0; x<7; x++) { guard->m7resources[x] = readNormalNr(bufor,i); i+=4; } break; } case 8: case 9: { guard->m13489val = bufor[i]; ++i; break; } } int limit = readNormalNr(bufor,i); i+=4; if(limit == ((int)0xffffffff)) { guard->lastDay = -1; } else { guard->lastDay = limit; } guard->firstVisitText = readString(bufor,i); guard->nextVisitText = readString(bufor,i); guard->completedText = readString(bufor,i); } TerrainTile & Mapa::getTile( int3 tile ) { return terrain[tile.x][tile.y][tile.z]; } const TerrainTile & Mapa::getTile( int3 tile ) const { return terrain[tile.x][tile.y][tile.z]; } bool Mapa::isWaterTile(const int3 &pos) const { return isInTheMap(pos) && getTile(pos).tertype == TerrainTile::water; } void CMapInfo::countPlayers() { playerAmnt = humenPlayers = 0; for(int i=0;i