2021-05-15 18:22:44 +02:00
|
|
|
/*
|
|
|
|
* AILayerTransitionRule.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 "AILayerTransitionRule.h"
|
2020-05-04 17:58:43 +02:00
|
|
|
#include "../../Engine/Nullkiller.h"
|
2023-09-16 11:33:02 +02:00
|
|
|
#include "../../../../lib/pathfinder/CPathfinder.h"
|
|
|
|
#include "../../../../lib/pathfinder/TurnInfo.h"
|
2021-05-15 18:22:44 +02:00
|
|
|
|
2022-09-26 20:01:07 +02:00
|
|
|
namespace NKAI
|
|
|
|
{
|
2021-05-15 18:22:44 +02:00
|
|
|
namespace AIPathfinding
|
|
|
|
{
|
2020-05-04 17:58:43 +02:00
|
|
|
AILayerTransitionRule::AILayerTransitionRule(CPlayerSpecificInfoCallback * cb, Nullkiller * ai, std::shared_ptr<AINodeStorage> nodeStorage)
|
2021-05-15 18:22:44 +02:00
|
|
|
:cb(cb), ai(ai), nodeStorage(nodeStorage)
|
|
|
|
{
|
|
|
|
setup();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AILayerTransitionRule::process(
|
|
|
|
const PathNodeInfo & source,
|
|
|
|
CDestinationNodeInfo & destination,
|
|
|
|
const PathfinderConfig * pathfinderConfig,
|
|
|
|
CPathfinderHelper * pathfinderHelper) const
|
|
|
|
{
|
|
|
|
LayerTransitionRule::process(source, destination, pathfinderConfig, pathfinderHelper);
|
|
|
|
|
|
|
|
if(!destination.blocked)
|
|
|
|
{
|
2023-09-16 11:33:02 +02:00
|
|
|
if(source.node->layer == EPathfindingLayer::LAND
|
|
|
|
&& (destination.node->layer == EPathfindingLayer::AIR || destination.node->layer == EPathfindingLayer::WATER))
|
|
|
|
{
|
|
|
|
if(pathfinderHelper->getTurnInfo()->isLayerAvailable(destination.node->layer))
|
|
|
|
return;
|
|
|
|
else
|
|
|
|
destination.blocked = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2021-05-15 18:22:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if(source.node->layer == EPathfindingLayer::LAND && destination.node->layer == EPathfindingLayer::SAIL)
|
|
|
|
{
|
|
|
|
std::shared_ptr<const VirtualBoatAction> virtualBoat = findVirtualBoat(destination, source);
|
|
|
|
|
2023-09-16 11:33:02 +02:00
|
|
|
if(virtualBoat && tryUseSpecialAction(destination, source, virtualBoat, EPathNodeAction::EMBARK))
|
2021-05-15 18:22:44 +02:00
|
|
|
{
|
2022-09-26 20:01:07 +02:00
|
|
|
#if NKAI_PATHFINDER_TRACE_LEVEL >= 1
|
2021-05-15 18:22:44 +02:00
|
|
|
logAi->trace("Embarking to virtual boat while moving %s -> %s!", source.coord.toString(), destination.coord.toString());
|
2023-09-16 11:33:02 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(source.node->layer == EPathfindingLayer::LAND && destination.node->layer == EPathfindingLayer::WATER)
|
|
|
|
{
|
2024-01-20 23:31:57 +02:00
|
|
|
if(nodeStorage->getAINode(source.node)->dayFlags & DayFlags::WATER_WALK_CAST)
|
2023-12-23 15:53:30 +02:00
|
|
|
{
|
|
|
|
destination.blocked = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-09-16 11:33:02 +02:00
|
|
|
auto action = waterWalkingActions.find(nodeStorage->getHero(source.node));
|
|
|
|
|
|
|
|
if(action != waterWalkingActions.end() && tryUseSpecialAction(destination, source, action->second, EPathNodeAction::NORMAL))
|
|
|
|
{
|
|
|
|
#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
|
|
|
|
logAi->trace("Casting water walk while moving %s -> %s!", source.coord.toString(), destination.coord.toString());
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(source.node->layer == EPathfindingLayer::LAND && destination.node->layer == EPathfindingLayer::AIR)
|
|
|
|
{
|
2024-01-20 23:31:57 +02:00
|
|
|
if(nodeStorage->getAINode(source.node)->dayFlags & DayFlags::FLY_CAST)
|
2023-12-23 15:53:30 +02:00
|
|
|
{
|
|
|
|
destination.blocked = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-09-16 11:33:02 +02:00
|
|
|
auto action = airWalkingActions.find(nodeStorage->getHero(source.node));
|
|
|
|
|
|
|
|
if(action != airWalkingActions.end() && tryUseSpecialAction(destination, source, action->second, EPathNodeAction::NORMAL))
|
|
|
|
{
|
|
|
|
#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
|
|
|
|
logAi->trace("Casting fly while moving %s -> %s!", source.coord.toString(), destination.coord.toString());
|
2021-05-15 18:22:44 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AILayerTransitionRule::setup()
|
2023-09-16 11:33:02 +02:00
|
|
|
{
|
|
|
|
SpellID waterWalk = SpellID::WATER_WALK;
|
|
|
|
SpellID airWalk = SpellID::FLY;
|
|
|
|
|
|
|
|
for(const CGHeroInstance * hero : nodeStorage->getAllHeroes())
|
|
|
|
{
|
2023-12-02 14:03:20 +02:00
|
|
|
if(hero->canCastThisSpell(waterWalk.toSpell()) && hero->mana >= hero->getSpellCost(waterWalk.toSpell()))
|
2023-09-16 11:33:02 +02:00
|
|
|
{
|
|
|
|
waterWalkingActions[hero] = std::make_shared<WaterWalkingAction>(hero);
|
|
|
|
}
|
|
|
|
|
2023-12-02 14:03:20 +02:00
|
|
|
if(hero->canCastThisSpell(airWalk.toSpell()) && hero->mana >= hero->getSpellCost(airWalk.toSpell()))
|
2023-09-16 11:33:02 +02:00
|
|
|
{
|
|
|
|
airWalkingActions[hero] = std::make_shared<AirWalkingAction>(hero);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
collectVirtualBoats();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AILayerTransitionRule::collectVirtualBoats()
|
2021-05-15 18:22:44 +02:00
|
|
|
{
|
|
|
|
std::vector<const IShipyard *> shipyards;
|
|
|
|
|
|
|
|
for(const CGTownInstance * t : cb->getTownsInfo())
|
|
|
|
{
|
2023-04-22 17:37:27 +02:00
|
|
|
if(t->hasBuilt(BuildingID::SHIPYARD))
|
2021-05-15 18:22:44 +02:00
|
|
|
shipyards.push_back(t);
|
|
|
|
}
|
|
|
|
|
2020-05-04 17:58:43 +02:00
|
|
|
for(const CGObjectInstance * obj : ai->memory->visitableObjs)
|
2021-05-15 18:22:44 +02:00
|
|
|
{
|
2023-04-22 17:37:27 +02:00
|
|
|
if(obj->ID != Obj::TOWN) //towns were handled in the previous loop
|
2021-05-15 18:22:44 +02:00
|
|
|
{
|
|
|
|
if(const IShipyard * shipyard = IShipyard::castFrom(obj))
|
|
|
|
shipyards.push_back(shipyard);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for(const IShipyard * shipyard : shipyards)
|
|
|
|
{
|
|
|
|
if(shipyard->shipyardStatus() == IShipyard::GOOD)
|
|
|
|
{
|
|
|
|
int3 boatLocation = shipyard->bestLocation();
|
2021-05-16 14:08:56 +02:00
|
|
|
virtualBoats[boatLocation] = std::make_shared<BuildBoatAction>(cb, shipyard);
|
2021-05-15 18:22:44 +02:00
|
|
|
logAi->debug("Virtual boat added at %s", boatLocation.toString());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-15 18:22:49 +02:00
|
|
|
for(const CGHeroInstance * hero : nodeStorage->getAllHeroes())
|
2021-05-15 18:22:44 +02:00
|
|
|
{
|
2021-05-15 18:22:49 +02:00
|
|
|
auto summonBoatSpell = SpellID(SpellID::SUMMON_BOAT).toSpell();
|
|
|
|
|
|
|
|
if(hero->canCastThisSpell(summonBoatSpell)
|
2023-10-05 15:13:52 +02:00
|
|
|
&& hero->getSpellSchoolLevel(summonBoatSpell) >= MasteryLevel::ADVANCED)
|
2021-05-15 18:22:49 +02:00
|
|
|
{
|
|
|
|
// TODO: For lower school level we might need to check the existance of some boat
|
|
|
|
summonableVirtualBoats[hero] = std::make_shared<SummonBoatAction>();
|
|
|
|
}
|
2021-05-15 18:22:44 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::shared_ptr<const VirtualBoatAction> AILayerTransitionRule::findVirtualBoat(
|
|
|
|
CDestinationNodeInfo & destination,
|
|
|
|
const PathNodeInfo & source) const
|
|
|
|
{
|
|
|
|
std::shared_ptr<const VirtualBoatAction> virtualBoat;
|
|
|
|
|
|
|
|
if(vstd::contains(virtualBoats, destination.coord))
|
|
|
|
{
|
|
|
|
virtualBoat = virtualBoats.at(destination.coord);
|
|
|
|
}
|
2021-05-15 18:22:49 +02:00
|
|
|
else
|
2021-05-15 18:22:44 +02:00
|
|
|
{
|
2021-05-15 18:22:49 +02:00
|
|
|
const CGHeroInstance * hero = nodeStorage->getHero(source.node);
|
|
|
|
|
|
|
|
if(vstd::contains(summonableVirtualBoats, hero)
|
2021-05-16 13:38:53 +02:00
|
|
|
&& summonableVirtualBoats.at(hero)->canAct(nodeStorage->getAINode(source.node)))
|
2021-05-15 18:22:49 +02:00
|
|
|
{
|
|
|
|
virtualBoat = summonableVirtualBoats.at(hero);
|
|
|
|
}
|
2021-05-15 18:22:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return virtualBoat;
|
|
|
|
}
|
|
|
|
|
2023-09-16 11:33:02 +02:00
|
|
|
bool AILayerTransitionRule::tryUseSpecialAction(
|
2021-05-15 18:22:44 +02:00
|
|
|
CDestinationNodeInfo & destination,
|
|
|
|
const PathNodeInfo & source,
|
2023-09-16 11:33:02 +02:00
|
|
|
std::shared_ptr<const SpecialAction> specialAction,
|
|
|
|
EPathNodeAction targetAction) const
|
2021-05-15 18:22:44 +02:00
|
|
|
{
|
|
|
|
bool result = false;
|
|
|
|
|
2023-09-16 11:33:02 +02:00
|
|
|
nodeStorage->updateAINode(destination.node, [&](AIPathNode * node)
|
2021-05-15 18:22:44 +02:00
|
|
|
{
|
2023-09-16 11:33:02 +02:00
|
|
|
auto castNodeOptional = nodeStorage->getOrCreateNode(
|
|
|
|
node->coord,
|
|
|
|
node->layer,
|
|
|
|
specialAction->getActor(node->actor));
|
2021-05-15 18:22:44 +02:00
|
|
|
|
2023-09-16 11:33:02 +02:00
|
|
|
if(castNodeOptional)
|
2021-05-15 18:22:44 +02:00
|
|
|
{
|
2023-09-16 11:33:02 +02:00
|
|
|
AIPathNode * castNode = castNodeOptional.value();
|
|
|
|
|
|
|
|
if(castNode->action == EPathNodeAction::UNKNOWN)
|
|
|
|
{
|
|
|
|
castNode->addSpecialAction(specialAction);
|
|
|
|
destination.blocked = false;
|
|
|
|
destination.action = targetAction;
|
|
|
|
destination.node = castNode;
|
|
|
|
result = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
#if NKAI_PATHFINDER_TRACE_LEVEL >= 1
|
|
|
|
logAi->trace(
|
|
|
|
"Special transition node already allocated. Blocked moving %s -> %s",
|
|
|
|
source.coord.toString(),
|
|
|
|
destination.coord.toString());
|
|
|
|
#endif
|
|
|
|
}
|
2021-05-15 18:22:44 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-09-16 11:33:02 +02:00
|
|
|
logAi->debug(
|
|
|
|
"Can not allocate special transition node while moving %s -> %s",
|
2021-05-15 18:22:44 +02:00
|
|
|
source.coord.toString(),
|
|
|
|
destination.coord.toString());
|
|
|
|
}
|
2023-09-16 11:33:02 +02:00
|
|
|
});
|
2021-05-15 18:22:44 +02:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
2022-09-26 20:01:07 +02:00
|
|
|
|
|
|
|
}
|