2021-05-15 18:23:11 +02:00
|
|
|
/*
|
2021-05-15 21:04:26 +02:00
|
|
|
* RecruitHeroBehavior.cpp, part of VCMI engine
|
2021-05-15 18:23:11 +02:00
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
#include "StdInc.h"
|
|
|
|
#include "RecruitHeroBehavior.h"
|
2021-05-16 14:39:38 +02:00
|
|
|
#include "../AIGateway.h"
|
2021-05-15 18:23:11 +02:00
|
|
|
#include "../AIUtility.h"
|
|
|
|
#include "../Goals/RecruitHero.h"
|
2021-05-15 21:02:52 +02:00
|
|
|
#include "../Goals/ExecuteHeroChain.h"
|
2024-07-24 21:17:04 +02:00
|
|
|
#include "../lib/CHeroHandler.h"
|
2021-05-15 18:23:11 +02:00
|
|
|
|
2022-09-26 20:01:07 +02:00
|
|
|
namespace NKAI
|
|
|
|
{
|
|
|
|
|
2021-05-15 18:23:11 +02:00
|
|
|
using namespace Goals;
|
|
|
|
|
|
|
|
std::string RecruitHeroBehavior::toString() const
|
|
|
|
{
|
|
|
|
return "Recruit hero";
|
|
|
|
}
|
|
|
|
|
2024-03-31 17:39:00 +02:00
|
|
|
Goals::TGoalVec RecruitHeroBehavior::decompose(const Nullkiller * ai) const
|
2021-05-15 18:23:11 +02:00
|
|
|
{
|
|
|
|
Goals::TGoalVec tasks;
|
2024-03-31 17:39:00 +02:00
|
|
|
auto towns = ai->cb->getTownsInfo();
|
2021-05-15 18:23:11 +02:00
|
|
|
|
2024-03-31 17:39:00 +02:00
|
|
|
auto ourHeroes = ai->heroManager->getHeroRoles();
|
2022-10-14 10:24:43 +02:00
|
|
|
auto minScoreToHireMain = std::numeric_limits<float>::max();
|
|
|
|
|
|
|
|
for(auto hero : ourHeroes)
|
|
|
|
{
|
|
|
|
if(hero.second != HeroRole::MAIN)
|
|
|
|
continue;
|
|
|
|
|
2024-03-31 17:39:00 +02:00
|
|
|
auto newScore = ai->heroManager->evaluateHero(hero.first.get());
|
2022-10-14 10:24:43 +02:00
|
|
|
|
|
|
|
if(minScoreToHireMain > newScore)
|
|
|
|
{
|
|
|
|
// weakest main hero score
|
|
|
|
minScoreToHireMain = newScore;
|
|
|
|
}
|
|
|
|
}
|
2024-07-12 15:39:50 +02:00
|
|
|
// If we don't have any heros we might want to lower our expectations.
|
|
|
|
if (ourHeroes.empty())
|
|
|
|
minScoreToHireMain = 0;
|
2022-10-14 10:24:43 +02:00
|
|
|
|
2024-07-24 21:17:04 +02:00
|
|
|
const CGHeroInstance* bestHeroToHire = nullptr;
|
|
|
|
const CGTownInstance* bestTownToHireFrom = nullptr;
|
|
|
|
float bestScore = 0;
|
|
|
|
bool haveCapitol = false;
|
2024-08-18 09:50:32 +02:00
|
|
|
|
|
|
|
ai->dangerHitMap->updateHitMap();
|
2024-07-24 21:17:04 +02:00
|
|
|
|
2021-05-16 13:09:49 +02:00
|
|
|
for(auto town : towns)
|
2021-05-15 18:23:11 +02:00
|
|
|
{
|
2024-08-18 09:50:32 +02:00
|
|
|
uint8_t closestThreat = UINT8_MAX;
|
|
|
|
for (auto threat : ai->dangerHitMap->getTownThreats(town))
|
|
|
|
{
|
|
|
|
closestThreat = std::min(closestThreat, threat.turn);
|
|
|
|
}
|
2024-08-31 23:00:27 +02:00
|
|
|
//Don't hire a hero where there already is one present
|
|
|
|
if (town->visitingHero || town->garrisonHero)
|
2024-08-18 09:50:32 +02:00
|
|
|
continue;
|
2024-08-31 23:00:27 +02:00
|
|
|
float visitability = 0;
|
|
|
|
for (auto checkHero : ourHeroes)
|
|
|
|
{
|
|
|
|
if (ai->dangerHitMap->getClosestTown(checkHero.first.get()->pos) == town)
|
|
|
|
visitability++;
|
|
|
|
}
|
2024-03-31 17:39:00 +02:00
|
|
|
if(ai->heroManager->canRecruitHero(town))
|
2021-05-15 18:23:11 +02:00
|
|
|
{
|
2024-03-31 17:39:00 +02:00
|
|
|
auto availableHeroes = ai->cb->getAvailableHeroes(town);
|
2022-10-14 10:24:43 +02:00
|
|
|
|
|
|
|
for(auto hero : availableHeroes)
|
|
|
|
{
|
2024-03-31 17:39:00 +02:00
|
|
|
auto score = ai->heroManager->evaluateHero(hero);
|
2024-07-24 21:17:04 +02:00
|
|
|
if(score > minScoreToHireMain)
|
2022-10-14 10:24:43 +02:00
|
|
|
{
|
2024-07-24 21:17:04 +02:00
|
|
|
score *= score / minScoreToHireMain;
|
|
|
|
}
|
|
|
|
score *= hero->getArmyCost();
|
|
|
|
if (hero->type->heroClass->faction == town->getFaction())
|
|
|
|
score *= 1.5;
|
2024-09-05 16:36:07 +02:00
|
|
|
if (vstd::isAlmostZero(visitability))
|
2024-08-31 23:00:27 +02:00
|
|
|
score *= 30 * town->getTownLevel();
|
|
|
|
else
|
|
|
|
score *= town->getTownLevel() / visitability;
|
2024-07-24 21:17:04 +02:00
|
|
|
if (score > bestScore)
|
|
|
|
{
|
|
|
|
bestScore = score;
|
|
|
|
bestHeroToHire = hero;
|
|
|
|
bestTownToHireFrom = town;
|
2022-10-14 10:24:43 +02:00
|
|
|
}
|
|
|
|
}
|
2024-07-24 21:17:04 +02:00
|
|
|
}
|
|
|
|
if (town->hasCapitol())
|
|
|
|
haveCapitol = true;
|
|
|
|
}
|
|
|
|
if (bestHeroToHire && bestTownToHireFrom)
|
|
|
|
{
|
|
|
|
if (ai->cb->getHeroesInfo().size() < ai->cb->getTownsInfo().size() + 1
|
|
|
|
|| (ai->getFreeResources()[EGameResID::GOLD] > 10000 && !ai->buildAnalyzer->isGoldPressureHigh() && haveCapitol))
|
|
|
|
{
|
|
|
|
tasks.push_back(Goals::sptr(Goals::RecruitHero(bestTownToHireFrom, bestHeroToHire).setpriority((float)3 / ourHeroes.size())));
|
2021-05-15 18:23:11 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-15 21:02:52 +02:00
|
|
|
return tasks;
|
|
|
|
}
|
2022-09-26 20:01:07 +02:00
|
|
|
|
|
|
|
}
|