2021-05-15 21:00:02 +02:00
|
|
|
/*
|
|
|
|
* HeroManager.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
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2021-05-16 13:55:57 +02:00
|
|
|
#include "../StdInc.h"
|
2020-05-04 17:58:43 +02:00
|
|
|
#include "../Engine/Nullkiller.h"
|
2021-05-16 13:55:57 +02:00
|
|
|
#include "../../../lib/mapObjects/MapObjects.h"
|
|
|
|
#include "../../../lib/CHeroHandler.h"
|
2023-04-22 13:47:02 +02:00
|
|
|
#include "../../../lib/GameSettings.h"
|
2021-05-15 21:00:02 +02:00
|
|
|
|
2022-09-26 20:01:07 +02:00
|
|
|
namespace NKAI
|
|
|
|
{
|
|
|
|
|
2021-05-15 21:00:02 +02:00
|
|
|
SecondarySkillEvaluator HeroManager::wariorSkillsScores = SecondarySkillEvaluator(
|
|
|
|
{
|
|
|
|
std::make_shared<SecondarySkillScoreMap>(
|
|
|
|
std::map<SecondarySkill, float>
|
|
|
|
{
|
|
|
|
{SecondarySkill::DIPLOMACY, 2},
|
|
|
|
{SecondarySkill::LOGISTICS, 2},
|
|
|
|
{SecondarySkill::EARTH_MAGIC, 2},
|
|
|
|
{SecondarySkill::ARMORER, 2},
|
|
|
|
{SecondarySkill::OFFENCE, 2},
|
|
|
|
{SecondarySkill::AIR_MAGIC, 1},
|
|
|
|
{SecondarySkill::WISDOM, 1},
|
|
|
|
{SecondarySkill::LEADERSHIP, 1},
|
|
|
|
{SecondarySkill::INTELLIGENCE, 1},
|
|
|
|
{SecondarySkill::RESISTANCE, 1},
|
|
|
|
{SecondarySkill::MYSTICISM, -1},
|
|
|
|
{SecondarySkill::SORCERY, -1},
|
|
|
|
{SecondarySkill::ESTATES, -1},
|
|
|
|
{SecondarySkill::FIRST_AID, -1},
|
|
|
|
{SecondarySkill::LEARNING, -1},
|
|
|
|
{SecondarySkill::SCHOLAR, -1},
|
|
|
|
{SecondarySkill::EAGLE_EYE, -1},
|
|
|
|
{SecondarySkill::NAVIGATION, -1}
|
|
|
|
}),
|
|
|
|
std::make_shared<ExistingSkillRule>(),
|
|
|
|
std::make_shared<WisdomRule>(),
|
|
|
|
std::make_shared<AtLeastOneMagicRule>()
|
|
|
|
});
|
|
|
|
|
|
|
|
SecondarySkillEvaluator HeroManager::scountSkillsScores = SecondarySkillEvaluator(
|
|
|
|
{
|
|
|
|
std::make_shared<SecondarySkillScoreMap>(
|
|
|
|
std::map<SecondarySkill, float>
|
|
|
|
{
|
|
|
|
{SecondarySkill::LOGISTICS, 2},
|
|
|
|
{SecondarySkill::ESTATES, 2},
|
|
|
|
{SecondarySkill::PATHFINDING, 1},
|
|
|
|
{SecondarySkill::SCHOLAR, 1}
|
|
|
|
}),
|
|
|
|
std::make_shared<ExistingSkillRule>()
|
|
|
|
});
|
|
|
|
|
2021-05-15 21:02:27 +02:00
|
|
|
float HeroManager::evaluateSecSkill(SecondarySkill skill, const CGHeroInstance * hero) const
|
|
|
|
{
|
|
|
|
auto role = getHeroRole(hero);
|
|
|
|
|
|
|
|
if(role == HeroRole::MAIN)
|
|
|
|
return wariorSkillsScores.evaluateSecSkill(hero, skill);
|
|
|
|
|
|
|
|
return scountSkillsScores.evaluateSecSkill(hero, skill);
|
|
|
|
}
|
|
|
|
|
2021-05-15 21:00:02 +02:00
|
|
|
float HeroManager::evaluateSpeciality(const CGHeroInstance * hero) const
|
|
|
|
{
|
2023-10-21 13:50:42 +02:00
|
|
|
auto heroSpecial = Selector::source(BonusSource::HERO_SPECIAL, BonusSourceID(hero->type->getId()));
|
2023-05-01 00:20:01 +02:00
|
|
|
auto secondarySkillBonus = Selector::targetSourceType()(BonusSource::SECONDARY_SKILL);
|
2021-05-15 21:00:02 +02:00
|
|
|
auto specialSecondarySkillBonuses = hero->getBonuses(heroSpecial.And(secondarySkillBonus));
|
2023-05-01 00:20:01 +02:00
|
|
|
auto secondarySkillBonuses = hero->getBonuses(Selector::sourceTypeSel(BonusSource::SECONDARY_SKILL));
|
2021-05-15 21:00:02 +02:00
|
|
|
float specialityScore = 0.0f;
|
|
|
|
|
2023-03-05 02:27:04 +02:00
|
|
|
for(auto bonus : *secondarySkillBonuses)
|
2021-05-15 21:00:02 +02:00
|
|
|
{
|
2023-03-05 02:27:04 +02:00
|
|
|
auto hasBonus = !!specialSecondarySkillBonuses->getFirst(Selector::typeSubtype(bonus->type, bonus->subtype));
|
2021-05-15 21:00:02 +02:00
|
|
|
|
2023-03-05 02:27:04 +02:00
|
|
|
if(hasBonus)
|
|
|
|
{
|
2023-10-10 17:05:18 +02:00
|
|
|
SecondarySkill bonusSkill = bonus->sid.as<SecondarySkill>();
|
2023-03-05 02:27:04 +02:00
|
|
|
float bonusScore = wariorSkillsScores.evaluateSecSkill(hero, bonusSkill);
|
|
|
|
|
|
|
|
if(bonusScore > 0)
|
|
|
|
specialityScore += bonusScore * bonusScore * bonusScore;
|
|
|
|
}
|
2021-05-15 21:00:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return specialityScore;
|
|
|
|
}
|
|
|
|
|
|
|
|
float HeroManager::evaluateFightingStrength(const CGHeroInstance * hero) const
|
|
|
|
{
|
|
|
|
return evaluateSpeciality(hero) + wariorSkillsScores.evaluateSecSkills(hero) + hero->level * 1.5f;
|
|
|
|
}
|
|
|
|
|
2021-05-16 13:15:03 +02:00
|
|
|
void HeroManager::update()
|
2021-05-15 21:00:02 +02:00
|
|
|
{
|
2021-05-16 13:15:03 +02:00
|
|
|
logAi->trace("Start analysing our heroes");
|
|
|
|
|
2021-05-16 13:56:27 +02:00
|
|
|
std::map<const CGHeroInstance *, float> scores;
|
2020-05-04 17:58:43 +02:00
|
|
|
auto myHeroes = cb->getHeroesInfo();
|
2021-05-15 21:00:02 +02:00
|
|
|
|
|
|
|
for(auto & hero : myHeroes)
|
|
|
|
{
|
2020-05-04 17:58:43 +02:00
|
|
|
scores[hero] = evaluateFightingStrength(hero);
|
2021-05-15 21:00:02 +02:00
|
|
|
}
|
|
|
|
|
2021-05-16 13:56:27 +02:00
|
|
|
auto scoreSort = [&](const CGHeroInstance * h1, const CGHeroInstance * h2) -> bool
|
2021-05-15 21:00:02 +02:00
|
|
|
{
|
2021-05-15 21:02:27 +02:00
|
|
|
return scores.at(h1) > scores.at(h2);
|
2021-05-16 13:56:27 +02:00
|
|
|
};
|
2021-05-15 21:00:02 +02:00
|
|
|
|
2023-03-05 15:42:15 +02:00
|
|
|
int globalMainCount = std::min(((int)myHeroes.size() + 2) / 3, cb->getMapSize().x / 50 + 1);
|
2021-05-16 13:56:27 +02:00
|
|
|
|
2023-03-11 11:42:44 +02:00
|
|
|
//vstd::amin(globalMainCount, 1 + (cb->getTownsInfo().size() / 3));
|
|
|
|
if(cb->getTownsInfo().size() < 4 && globalMainCount > 2)
|
|
|
|
{
|
|
|
|
globalMainCount = 2;
|
|
|
|
}
|
|
|
|
|
2021-05-16 13:56:27 +02:00
|
|
|
std::sort(myHeroes.begin(), myHeroes.end(), scoreSort);
|
2023-07-24 21:19:09 +02:00
|
|
|
heroRoles.clear();
|
2021-05-16 13:56:27 +02:00
|
|
|
|
|
|
|
for(auto hero : myHeroes)
|
|
|
|
{
|
|
|
|
heroRoles[hero] = (globalMainCount--) > 0 ? HeroRole::MAIN : HeroRole::SCOUT;
|
|
|
|
}
|
|
|
|
|
2020-05-04 17:58:43 +02:00
|
|
|
for(auto hero : myHeroes)
|
2021-05-15 21:00:02 +02:00
|
|
|
{
|
2023-01-02 13:27:03 +02:00
|
|
|
logAi->trace("Hero %s has role %s", hero->getNameTranslated(), heroRoles[hero] == HeroRole::MAIN ? "main" : "scout");
|
2021-05-15 21:00:02 +02:00
|
|
|
}
|
2021-05-15 21:02:27 +02:00
|
|
|
}
|
2021-05-15 21:00:02 +02:00
|
|
|
|
2021-05-15 21:02:27 +02:00
|
|
|
HeroRole HeroManager::getHeroRole(const HeroPtr & hero) const
|
|
|
|
{
|
|
|
|
return heroRoles.at(hero);
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::map<HeroPtr, HeroRole> & HeroManager::getHeroRoles() const
|
|
|
|
{
|
|
|
|
return heroRoles;
|
2021-05-15 21:00:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int HeroManager::selectBestSkill(const HeroPtr & hero, const std::vector<SecondarySkill> & skills) const
|
|
|
|
{
|
2021-05-15 21:02:27 +02:00
|
|
|
auto role = getHeroRole(hero);
|
2021-05-15 21:00:02 +02:00
|
|
|
auto & evaluator = role == HeroRole::MAIN ? wariorSkillsScores : scountSkillsScores;
|
|
|
|
|
|
|
|
int result = 0;
|
|
|
|
float resultScore = -100;
|
|
|
|
|
|
|
|
for(int i = 0; i < skills.size(); i++)
|
|
|
|
{
|
|
|
|
auto score = evaluator.evaluateSecSkill(hero.get(), skills[i]);
|
|
|
|
|
|
|
|
if(score > resultScore)
|
|
|
|
{
|
|
|
|
resultScore = score;
|
|
|
|
result = i;
|
|
|
|
}
|
|
|
|
|
|
|
|
logAi->trace(
|
|
|
|
"Hero %s is proposed to learn %d with score %f",
|
|
|
|
hero.name,
|
|
|
|
skills[i].toEnum(),
|
|
|
|
score);
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-05-15 21:02:52 +02:00
|
|
|
float HeroManager::evaluateHero(const CGHeroInstance * hero) const
|
|
|
|
{
|
|
|
|
return evaluateFightingStrength(hero);
|
|
|
|
}
|
|
|
|
|
2023-06-04 15:02:02 +02:00
|
|
|
bool HeroManager::heroCapReached() const
|
|
|
|
{
|
|
|
|
const bool includeGarnisoned = true;
|
|
|
|
int heroCount = cb->getHeroCount(ai->playerID, includeGarnisoned);
|
|
|
|
|
|
|
|
return heroCount >= ALLOWED_ROAMING_HEROES
|
|
|
|
|| heroCount >= VLC->settings()->getInteger(EGameSettings::HEROES_PER_PLAYER_ON_MAP_CAP);
|
|
|
|
}
|
|
|
|
|
2023-09-24 12:07:42 +02:00
|
|
|
float HeroManager::getMagicStrength(const CGHeroInstance * hero) const
|
|
|
|
{
|
|
|
|
auto hasFly = hero->spellbookContainsSpell(SpellID::FLY);
|
|
|
|
auto hasTownPortal = hero->spellbookContainsSpell(SpellID::TOWN_PORTAL);
|
|
|
|
auto manaLimit = hero->manaLimit();
|
|
|
|
auto spellPower = hero->getPrimSkillLevel(PrimarySkill::SPELL_POWER);
|
|
|
|
auto hasEarth = hero->getSpellSchoolLevel(SpellID(SpellID::TOWN_PORTAL).toSpell()) > 0;
|
|
|
|
|
|
|
|
auto score = 0.0f;
|
|
|
|
|
|
|
|
for(auto spellId : hero->getSpellsInSpellbook())
|
|
|
|
{
|
|
|
|
auto spell = spellId.toSpell();
|
|
|
|
auto schoolLevel = hero->getSpellSchoolLevel(spell);
|
|
|
|
|
|
|
|
score += (spell->getLevel() + 1) * (schoolLevel + 1) * 0.05f;
|
|
|
|
}
|
|
|
|
|
|
|
|
vstd::amin(score, 1);
|
|
|
|
|
|
|
|
score *= std::min(1.0f, spellPower / 10.0f);
|
|
|
|
|
|
|
|
if(hasFly)
|
|
|
|
score += 0.3f;
|
|
|
|
|
|
|
|
if(hasTownPortal && hasEarth)
|
|
|
|
score += 0.6f;
|
|
|
|
|
|
|
|
vstd::amin(score, 1);
|
|
|
|
|
|
|
|
score *= std::min(1.0f, manaLimit / 100.0f);
|
|
|
|
|
|
|
|
return std::min(score, 1.0f);
|
|
|
|
}
|
|
|
|
|
2023-04-22 13:47:02 +02:00
|
|
|
bool HeroManager::canRecruitHero(const CGTownInstance * town) const
|
|
|
|
{
|
|
|
|
if(!town)
|
|
|
|
town = findTownWithTavern();
|
|
|
|
|
|
|
|
if(!town || !townHasFreeTavern(town))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if(cb->getResourceAmount(EGameResID::GOLD) < GameConstants::HERO_GOLD_COST)
|
|
|
|
return false;
|
|
|
|
|
2023-06-04 15:02:02 +02:00
|
|
|
if(heroCapReached())
|
2023-04-22 13:47:02 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
if(!cb->getAvailableHeroes(town).size())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
const CGTownInstance * HeroManager::findTownWithTavern() const
|
|
|
|
{
|
|
|
|
for(const CGTownInstance * t : cb->getTownsInfo())
|
|
|
|
if(townHasFreeTavern(t))
|
|
|
|
return t;
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
const CGHeroInstance * HeroManager::findHeroWithGrail() const
|
|
|
|
{
|
|
|
|
for(const CGHeroInstance * h : cb->getHeroesInfo())
|
|
|
|
{
|
|
|
|
if(h->hasArt(ArtifactID::GRAIL))
|
|
|
|
return h;
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2023-07-28 13:17:01 +02:00
|
|
|
const CGHeroInstance * HeroManager::findWeakHeroToDismiss(uint64_t armyLimit) const
|
|
|
|
{
|
|
|
|
const CGHeroInstance * weakestHero = nullptr;
|
|
|
|
auto myHeroes = ai->cb->getHeroesInfo();
|
|
|
|
|
|
|
|
for(auto existingHero : myHeroes)
|
|
|
|
{
|
2023-07-30 11:21:47 +02:00
|
|
|
if(ai->getHeroLockedReason(existingHero) == HeroLockedReason::DEFENCE
|
2023-07-28 13:17:01 +02:00
|
|
|
|| existingHero->getArmyStrength() >armyLimit
|
|
|
|
|| getHeroRole(existingHero) == HeroRole::MAIN
|
|
|
|
|| existingHero->movementPointsRemaining()
|
|
|
|
|| existingHero->artifactsWorn.size() > (existingHero->hasSpellbook() ? 2 : 1))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!weakestHero || weakestHero->getFightingStrength() > existingHero->getFightingStrength())
|
|
|
|
{
|
|
|
|
weakestHero = existingHero;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return weakestHero;
|
|
|
|
}
|
|
|
|
|
2021-05-15 21:00:02 +02:00
|
|
|
SecondarySkillScoreMap::SecondarySkillScoreMap(std::map<SecondarySkill, float> scoreMap)
|
|
|
|
:scoreMap(scoreMap)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void SecondarySkillScoreMap::evaluateScore(const CGHeroInstance * hero, SecondarySkill skill, float & score) const
|
|
|
|
{
|
|
|
|
auto it = scoreMap.find(skill);
|
|
|
|
|
|
|
|
if(it != scoreMap.end())
|
|
|
|
{
|
|
|
|
score = it->second;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ExistingSkillRule::evaluateScore(const CGHeroInstance * hero, SecondarySkill skill, float & score) const
|
|
|
|
{
|
|
|
|
int upgradesLeft = 0;
|
|
|
|
|
|
|
|
for(auto & heroSkill : hero->secSkills)
|
|
|
|
{
|
|
|
|
if(heroSkill.first == skill)
|
|
|
|
return;
|
|
|
|
|
2023-10-05 15:13:52 +02:00
|
|
|
upgradesLeft += MasteryLevel::EXPERT - heroSkill.second;
|
2021-05-15 21:00:02 +02:00
|
|
|
}
|
|
|
|
|
2021-05-15 21:04:31 +02:00
|
|
|
if(score >= 2 || (score >= 1 && upgradesLeft <= 1))
|
2021-05-15 21:00:02 +02:00
|
|
|
score += 1.5;
|
|
|
|
}
|
|
|
|
|
|
|
|
void WisdomRule::evaluateScore(const CGHeroInstance * hero, SecondarySkill skill, float & score) const
|
|
|
|
{
|
|
|
|
if(skill != SecondarySkill::WISDOM)
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto wisdomLevel = hero->getSecSkillLevel(SecondarySkill::WISDOM);
|
|
|
|
|
2023-10-05 15:13:52 +02:00
|
|
|
if(hero->level > 10 && wisdomLevel == MasteryLevel::NONE)
|
2021-05-15 21:00:02 +02:00
|
|
|
score += 1.5;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<SecondarySkill> AtLeastOneMagicRule::magicSchools = {
|
|
|
|
SecondarySkill::AIR_MAGIC,
|
|
|
|
SecondarySkill::EARTH_MAGIC,
|
|
|
|
SecondarySkill::FIRE_MAGIC,
|
|
|
|
SecondarySkill::WATER_MAGIC
|
|
|
|
};
|
|
|
|
|
|
|
|
void AtLeastOneMagicRule::evaluateScore(const CGHeroInstance * hero, SecondarySkill skill, float & score) const
|
|
|
|
{
|
|
|
|
if(!vstd::contains(magicSchools, skill))
|
|
|
|
return;
|
|
|
|
|
|
|
|
bool heroHasAnyMagic = vstd::contains_if(magicSchools, [&](SecondarySkill skill) -> bool
|
|
|
|
{
|
2023-10-05 15:13:52 +02:00
|
|
|
return hero->getSecSkillLevel(skill) > MasteryLevel::NONE;
|
2021-05-15 21:00:02 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
if(!heroHasAnyMagic)
|
|
|
|
score += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
SecondarySkillEvaluator::SecondarySkillEvaluator(std::vector<std::shared_ptr<ISecondarySkillRule>> evaluationRules)
|
|
|
|
: evaluationRules(evaluationRules)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
float SecondarySkillEvaluator::evaluateSecSkills(const CGHeroInstance * hero) const
|
|
|
|
{
|
|
|
|
float totalScore = 0;
|
|
|
|
|
|
|
|
for(auto skill : hero->secSkills)
|
|
|
|
{
|
|
|
|
totalScore += skill.second * evaluateSecSkill(hero, skill.first);
|
|
|
|
}
|
|
|
|
|
|
|
|
return totalScore;
|
|
|
|
}
|
|
|
|
|
|
|
|
float SecondarySkillEvaluator::evaluateSecSkill(const CGHeroInstance * hero, SecondarySkill skill) const
|
|
|
|
{
|
|
|
|
float score = 0;
|
|
|
|
|
|
|
|
for(auto rule : evaluationRules)
|
|
|
|
rule->evaluateScore(hero, skill, score);
|
|
|
|
|
|
|
|
return score;
|
|
|
|
}
|
2022-09-26 20:01:07 +02:00
|
|
|
|
|
|
|
}
|