2021-05-15 21:04:26 +02:00
|
|
|
/*
|
|
|
|
* ExchangeSwapTownHeroes.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
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
#include "StdInc.h"
|
|
|
|
#include "ExchangeSwapTownHeroes.h"
|
|
|
|
#include "ExecuteHeroChain.h"
|
2021-05-16 14:39:38 +02:00
|
|
|
#include "../AIGateway.h"
|
2021-05-15 21:04:26 +02:00
|
|
|
#include "../Engine/Nullkiller.h"
|
|
|
|
|
2022-09-26 20:01:07 +02:00
|
|
|
namespace NKAI
|
|
|
|
{
|
|
|
|
|
2021-05-15 21:04:26 +02:00
|
|
|
using namespace Goals;
|
|
|
|
|
2021-05-16 13:13:56 +02:00
|
|
|
ExchangeSwapTownHeroes::ExchangeSwapTownHeroes(
|
|
|
|
const CGTownInstance * town,
|
|
|
|
const CGHeroInstance * garrisonHero,
|
|
|
|
HeroLockedReason lockingReason)
|
2021-05-16 13:38:26 +02:00
|
|
|
:ElementarGoal(Goals::EXCHANGE_SWAP_TOWN_HEROES), town(town), garrisonHero(garrisonHero), lockingReason(lockingReason)
|
2021-05-15 21:04:26 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2024-04-14 14:23:44 +02:00
|
|
|
std::vector<ObjectInstanceID> ExchangeSwapTownHeroes::getAffectedObjects() const
|
|
|
|
{
|
|
|
|
std::vector<ObjectInstanceID> affectedObjects = { town->id };
|
|
|
|
|
|
|
|
if(town->garrisonHero)
|
|
|
|
affectedObjects.push_back(town->garrisonHero->id);
|
|
|
|
|
|
|
|
if(town->visitingHero)
|
|
|
|
affectedObjects.push_back(town->visitingHero->id);
|
|
|
|
|
|
|
|
return affectedObjects;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ExchangeSwapTownHeroes::isObjectAffected(ObjectInstanceID id) const
|
|
|
|
{
|
|
|
|
return town->id == id
|
|
|
|
|| (town->visitingHero && town->visitingHero->id == id)
|
|
|
|
|| (town->garrisonHero && town->garrisonHero->id == id);
|
|
|
|
}
|
|
|
|
|
2021-05-16 13:38:26 +02:00
|
|
|
std::string ExchangeSwapTownHeroes::toString() const
|
2021-05-15 21:04:26 +02:00
|
|
|
{
|
2023-01-04 15:17:50 +02:00
|
|
|
return "Exchange and swap heroes of " + town->getNameTranslated();
|
2021-05-15 21:04:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ExchangeSwapTownHeroes::operator==(const ExchangeSwapTownHeroes & other) const
|
|
|
|
{
|
|
|
|
return town == other.town;
|
|
|
|
}
|
|
|
|
|
2021-05-16 14:39:38 +02:00
|
|
|
void ExchangeSwapTownHeroes::accept(AIGateway * ai)
|
2021-05-15 21:04:26 +02:00
|
|
|
{
|
|
|
|
if(!garrisonHero)
|
|
|
|
{
|
2022-03-05 15:51:03 +02:00
|
|
|
auto currentGarrisonHero = town->garrisonHero;
|
|
|
|
|
|
|
|
if(!currentGarrisonHero)
|
2021-05-15 21:04:26 +02:00
|
|
|
throw cannotFulfillGoalException("Invalid configuration. There is no hero in town garrison.");
|
|
|
|
|
|
|
|
cb->swapGarrisonHero(town);
|
2022-03-05 15:51:03 +02:00
|
|
|
|
|
|
|
if(currentGarrisonHero.get() != town->visitingHero.get())
|
|
|
|
{
|
2023-01-02 13:27:03 +02:00
|
|
|
logAi->error("VisitingHero is empty, expected %s", currentGarrisonHero->getNameTranslated());
|
2022-03-05 15:51:03 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-05-15 21:04:26 +02:00
|
|
|
ai->buildArmyIn(town);
|
2022-03-05 15:51:03 +02:00
|
|
|
ai->nullkiller->unlockHero(currentGarrisonHero.get());
|
2023-01-04 15:17:50 +02:00
|
|
|
logAi->debug("Extracted hero %s from garrison of %s", currentGarrisonHero->getNameTranslated(), town->getNameTranslated());
|
2021-05-15 21:04:26 +02:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(town->visitingHero && town->visitingHero.get() != garrisonHero)
|
|
|
|
cb->swapGarrisonHero(town);
|
|
|
|
|
2021-05-16 13:19:00 +02:00
|
|
|
ai->makePossibleUpgrades(town);
|
2021-05-15 21:04:26 +02:00
|
|
|
ai->moveHeroToTile(town->visitablePos(), garrisonHero);
|
|
|
|
|
2021-05-16 13:09:49 +02:00
|
|
|
auto upperArmy = town->getUpperArmy();
|
|
|
|
|
2021-05-16 14:00:24 +02:00
|
|
|
if(!town->garrisonHero)
|
2021-05-16 13:09:49 +02:00
|
|
|
{
|
2024-10-20 15:49:23 +02:00
|
|
|
if (!garrisonHero->canBeMergedWith(*town))
|
2021-05-16 13:09:49 +02:00
|
|
|
{
|
2024-10-20 15:49:23 +02:00
|
|
|
while (upperArmy->stacksCount() != 0)
|
|
|
|
{
|
|
|
|
cb->dismissCreature(upperArmy, upperArmy->Slots().begin()->first);
|
|
|
|
}
|
2021-05-16 13:09:49 +02:00
|
|
|
}
|
|
|
|
}
|
2021-05-16 14:00:24 +02:00
|
|
|
|
|
|
|
cb->swapGarrisonHero(town);
|
2021-05-16 13:09:49 +02:00
|
|
|
|
2023-02-28 09:07:59 +02:00
|
|
|
if(lockingReason != HeroLockedReason::NOT_LOCKED)
|
|
|
|
{
|
|
|
|
ai->nullkiller->lockHero(garrisonHero, lockingReason);
|
|
|
|
}
|
2021-05-15 21:04:26 +02:00
|
|
|
|
2021-05-16 13:09:49 +02:00
|
|
|
if(town->visitingHero && town->visitingHero != garrisonHero)
|
2021-05-15 21:04:26 +02:00
|
|
|
{
|
|
|
|
ai->nullkiller->unlockHero(town->visitingHero.get());
|
2021-05-16 13:19:00 +02:00
|
|
|
ai->makePossibleUpgrades(town->visitingHero);
|
2021-05-15 21:04:26 +02:00
|
|
|
}
|
|
|
|
|
2023-01-04 15:17:50 +02:00
|
|
|
logAi->debug("Put hero %s to garrison of %s", garrisonHero->getNameTranslated(), town->getNameTranslated());
|
2022-09-26 20:01:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|