2023-10-19 16:19:09 +02:00
|
|
|
/*
|
|
|
|
* ReachabilityInfo.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 "ReachabilityInfo.h"
|
|
|
|
#include "Unit.h"
|
|
|
|
|
|
|
|
VCMI_LIB_NAMESPACE_BEGIN
|
|
|
|
|
|
|
|
ReachabilityInfo::Parameters::Parameters(const battle::Unit * Stack, BattleHex StartPosition):
|
2024-08-11 20:22:35 +00:00
|
|
|
perspective(static_cast<BattleSide>(Stack->unitSide())),
|
2023-10-19 16:19:09 +02:00
|
|
|
startPosition(StartPosition),
|
|
|
|
doubleWide(Stack->doubleWide()),
|
|
|
|
side(Stack->unitSide()),
|
|
|
|
flying(Stack->hasBonusOfType(BonusType::FLYING))
|
|
|
|
{
|
|
|
|
knownAccessible = battle::Unit::getHexes(startPosition, doubleWide, side);
|
|
|
|
}
|
|
|
|
|
|
|
|
ReachabilityInfo::ReachabilityInfo()
|
|
|
|
{
|
|
|
|
distances.fill(INFINITE_DIST);
|
|
|
|
predecessors.fill(BattleHex::INVALID);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ReachabilityInfo::isReachable(BattleHex hex) const
|
|
|
|
{
|
|
|
|
return distances[hex] < INFINITE_DIST;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t ReachabilityInfo::distToNearestNeighbour(
|
|
|
|
const std::vector<BattleHex> & targetHexes,
|
|
|
|
BattleHex * chosenHex) const
|
|
|
|
{
|
|
|
|
uint32_t ret = 1000000;
|
|
|
|
|
|
|
|
for(auto targetHex : targetHexes)
|
|
|
|
{
|
|
|
|
for(auto & n : targetHex.neighbouringTiles())
|
|
|
|
{
|
|
|
|
if(distances[n] < ret)
|
|
|
|
{
|
|
|
|
ret = distances[n];
|
|
|
|
if(chosenHex)
|
|
|
|
*chosenHex = n;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t ReachabilityInfo::distToNearestNeighbour(
|
|
|
|
const battle::Unit * attacker,
|
|
|
|
const battle::Unit * defender,
|
|
|
|
BattleHex * chosenHex) const
|
|
|
|
{
|
|
|
|
auto attackableHexes = defender->getHexes();
|
|
|
|
|
|
|
|
if(attacker->doubleWide())
|
|
|
|
{
|
2024-07-14 11:39:43 +03:00
|
|
|
if(defender->doubleWide())
|
|
|
|
{
|
|
|
|
// It can be back to back attack o==o or head to head =oo=.
|
|
|
|
// In case of back-to-back the distance between heads (unit positions) may be up to 3 tiles
|
|
|
|
vstd::concatenate(attackableHexes, battle::Unit::getHexes(defender->occupiedHex(), true, defender->unitSide()));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
vstd::concatenate(attackableHexes, battle::Unit::getHexes(defender->getPosition(), true, defender->unitSide()));
|
|
|
|
}
|
2023-10-19 16:19:09 +02:00
|
|
|
}
|
|
|
|
|
2024-07-14 11:39:43 +03:00
|
|
|
vstd::removeDuplicates(attackableHexes);
|
|
|
|
|
|
|
|
vstd::erase_if(attackableHexes, [defender](BattleHex h) -> bool
|
|
|
|
{
|
|
|
|
return h.getY() != defender->getPosition().getY() || !h.isAvailable();
|
|
|
|
});
|
|
|
|
|
2023-10-19 16:19:09 +02:00
|
|
|
return distToNearestNeighbour(attackableHexes, chosenHex);
|
|
|
|
}
|
|
|
|
|
|
|
|
VCMI_LIB_NAMESPACE_END
|