1
0
mirror of https://github.com/vcmi/vcmi.git synced 2025-01-26 03:52:01 +02:00

Fixed #1385 (and possible some related issues)

This commit is contained in:
Michał W. Urbańczyk 2013-09-03 13:18:09 +00:00
parent a50e9219c2
commit a2610de87d
2 changed files with 37 additions and 1 deletions

View File

@ -1532,6 +1532,8 @@ int CGHeroInstance::getSpellCost(const CSpell *sp) const
void CGHeroInstance::pushPrimSkill( PrimarySkill::PrimarySkill which, int val ) void CGHeroInstance::pushPrimSkill( PrimarySkill::PrimarySkill which, int val )
{ {
assert(!hasBonus(Selector::typeSubtype(Bonus::PRIMARY_SKILL, which)
.And(Selector::sourceType(Bonus::HERO_BASE_SKILL))));
addNewBonus(new Bonus(Bonus::PERMANENT, Bonus::PRIMARY_SKILL, Bonus::HERO_BASE_SKILL, val, id.getNum(), which)); addNewBonus(new Bonus(Bonus::PERMANENT, Bonus::PRIMARY_SKILL, Bonus::HERO_BASE_SKILL, val, id.getNum(), which));
} }
@ -4969,7 +4971,7 @@ const CGHeroInstance * CGSeerHut::getHeroToKill(bool allowNull) const
const CGObjectInstance *o = cb->getObjByQuestIdentifier(quest->m13489val); const CGObjectInstance *o = cb->getObjByQuestIdentifier(quest->m13489val);
if(allowNull && !o) if(allowNull && !o)
return nullptr; return nullptr;
assert(o && o->ID == Obj::HERO); assert(o && (o->ID == Obj::HERO || o->ID == Obj::PRISON));
return static_cast<const CGHeroInstance*>(o); return static_cast<const CGHeroInstance*>(o);
} }

View File

@ -589,6 +589,14 @@ void CMapLoaderH3M::loadArtifactsOfHero(CGHeroInstance * hero)
// True if artifact set is not default (hero has some artifacts) // True if artifact set is not default (hero has some artifacts)
if(artSet) if(artSet)
{ {
if(hero->artifactsWorn.size() || hero->artifactsInBackpack.size())
{
logGlobal->warnStream() << boost::format("Hero %s at %s has set artifacts twice (in map properties and on adventure map instance). Using the latter set...") % hero->name % hero->pos;
hero->artifactsInBackpack.clear();
while(hero->artifactsWorn.size())
hero->eraseArtSlot(hero->artifactsWorn.begin()->first);
}
for(int pom = 0; pom < 16; pom++) for(int pom = 0; pom < 16; pom++)
{ {
loadArtifactToSlot(hero, pom); loadArtifactToSlot(hero, pom);
@ -597,9 +605,11 @@ void CMapLoaderH3M::loadArtifactsOfHero(CGHeroInstance * hero)
// misc5 art //17 // misc5 art //17
if(map->version >= EMapFormat::SOD) if(map->version >= EMapFormat::SOD)
{ {
assert(!hero->getArt(ArtifactPosition::MACH4));
if(!loadArtifactToSlot(hero, ArtifactPosition::MACH4)) if(!loadArtifactToSlot(hero, ArtifactPosition::MACH4))
{ {
// catapult by default // catapult by default
assert(!hero->getArt(ArtifactPosition::MACH4));
hero->putArtifact(ArtifactPosition::MACH4, createArtifact(ArtifactID::CATAPULT)); hero->putArtifact(ArtifactPosition::MACH4, createArtifact(ArtifactID::CATAPULT));
} }
} }
@ -1537,6 +1547,8 @@ CGObjectInstance * CMapLoaderH3M::readHero(ObjectInstanceID idToBeGiven)
PlayerColor owner = PlayerColor(reader.readUInt8()); PlayerColor owner = PlayerColor(reader.readUInt8());
nhi->subID = reader.readUInt8(); nhi->subID = reader.readUInt8();
assert(!nhi->getArt(ArtifactPosition::MACH4));
//If hero of this type has been predefined, use that as a base. //If hero of this type has been predefined, use that as a base.
//Instance data will overwrite the predefined values where appropriate. //Instance data will overwrite the predefined values where appropriate.
for(auto & elem : map->predefinedHeroes) for(auto & elem : map->predefinedHeroes)
@ -1600,6 +1612,12 @@ CGObjectInstance * CMapLoaderH3M::readHero(ObjectInstanceID idToBeGiven)
bool hasSecSkills = reader.readBool(); bool hasSecSkills = reader.readBool();
if(hasSecSkills) if(hasSecSkills)
{ {
if(nhi->secSkills.size())
{
nhi->secSkills.clear();
//logGlobal->warnStream() << boost::format("Hero %s subID=%d has set secondary skills twice (in map properties and on adventure map instance). Using the latter set...") % nhi->name % nhi->subID;
}
int howMany = reader.readUInt32(); int howMany = reader.readUInt32();
nhi->secSkills.resize(howMany); nhi->secSkills.resize(howMany);
for(int yy = 0; yy < howMany; ++yy) for(int yy = 0; yy < howMany; ++yy)
@ -1651,6 +1669,12 @@ CGObjectInstance * CMapLoaderH3M::readHero(ObjectInstanceID idToBeGiven)
if(map->version > EMapFormat::AB) if(map->version > EMapFormat::AB)
{ {
bool hasCustomSpells = reader.readBool(); bool hasCustomSpells = reader.readBool();
if(nhi->spells.size())
{
nhi->clear();
logGlobal->warnStream() << boost::format("Hero %s subID=%d has spells set twice (in map properties and on adventure map instance). Using the latter set...") % nhi->name % nhi->subID;
}
if(hasCustomSpells) if(hasCustomSpells)
{ {
nhi->spells.insert(SpellID::PRESET); //placeholder "preset spells" nhi->spells.insert(SpellID::PRESET); //placeholder "preset spells"
@ -1677,6 +1701,16 @@ CGObjectInstance * CMapLoaderH3M::readHero(ObjectInstanceID idToBeGiven)
bool hasCustomPrimSkills = reader.readBool(); bool hasCustomPrimSkills = reader.readBool();
if(hasCustomPrimSkills) if(hasCustomPrimSkills)
{ {
auto ps = nhi->getAllBonuses(Selector::type(Bonus::PRIMARY_SKILL)
.And(Selector::sourceType(Bonus::HERO_BASE_SKILL)), nullptr);
if(ps->size())
{
logGlobal->warnStream() << boost::format("Hero %s subID=%d has set primary skills twice (in map properties and on adventure map instance). Using the latter set...") % nhi->name % nhi->subID;
for(auto b : *ps)
nhi->removeBonus(b);
}
for(int xx = 0; xx < GameConstants::PRIMARY_SKILLS; ++xx) for(int xx = 0; xx < GameConstants::PRIMARY_SKILLS; ++xx)
{ {
nhi->pushPrimSkill(static_cast<PrimarySkill::PrimarySkill>(xx), reader.readUInt8()); nhi->pushPrimSkill(static_cast<PrimarySkill::PrimarySkill>(xx), reader.readUInt8());