/* * CaptureObjectsBehavior.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 "../VCAI.h" #include "../Engine/Nullkiller.h" #include "../AIhelper.h" #include "../Goals/ExecuteHeroChain.h" #include "CaptureObjectsBehavior.h" #include "../AIUtility.h" #include "lib/mapping/CMap.h" //for victory conditions #include "lib/CPathfinder.h" extern boost::thread_specific_ptr cb; extern boost::thread_specific_ptr ai; extern FuzzyHelper * fh; using namespace Goals; std::string CaptureObjectsBehavior::toString() const { return "Capture objects"; } std::shared_ptr getFirstBlockedAction(const AIPath & path) { for(auto node : path.nodes) { if(node.specialAction && !node.specialAction->canAct(node.targetHero)) return node.specialAction; } return std::shared_ptr(); } Goals::TGoalVec CaptureObjectsBehavior::getTasks() { Goals::TGoalVec tasks; auto captureObjects = [&](const std::vector & objs) -> void{ if(objs.empty()) { return; } logAi->trace("Scanning objects, count %d", objs.size()); for(auto objToVisit : objs) { #ifdef VCMI_TRACE_PATHFINDER logAi->trace("Checking object %s, %s", objToVisit->getObjectName(), objToVisit->visitablePos().toString()); #endif if(!shouldVisitObject(objToVisit)) continue; const int3 pos = objToVisit->visitablePos(); auto paths = ai->ah->getPathsToTile(pos); std::vector> waysToVisitObj; std::shared_ptr closestWay; #ifdef VCMI_TRACE_PATHFINDER logAi->trace("Found %d paths", paths.size()); #endif for(auto & path : paths) { #ifdef VCMI_TRACE_PATHFINDER logAi->trace("Path found %s", path.toString()); #endif if(getFirstBlockedAction(path)) { #ifdef VCMI_TRACE_PATHFINDER // TODO: decomposition? logAi->trace("Ignore path. Action is blocked."); #endif continue; } if(ai->nullkiller->dangerHitMap->enemyCanKillOurHeroesAlongThePath(path)) { #ifdef VCMI_TRACE_PATHFINDER logAi->trace("Ignore path. Target hero can be killed by enemy"); #endif continue; } if(!shouldVisit(path.targetHero, objToVisit)) continue; auto hero = path.targetHero; auto danger = path.getTotalDanger(); if(danger == 0 && path.exchangeCount > 1) continue; auto isSafe = isSafeToVisit(hero, path.heroArmy, danger); #ifdef VCMI_TRACE_PATHFINDER logAi->trace( "It is %s to visit %s by %s with army %lld, danger %lld and army loss %lld", isSafe ? "safe" : "not safe", objToVisit->instanceName, hero->name, path.getHeroStrength(), danger, path.armyLoss); #endif if(isSafe) { auto newWay = std::make_shared(path, objToVisit); waysToVisitObj.push_back(newWay); if(!closestWay || closestWay->evaluationContext.movementCost > newWay->evaluationContext.movementCost) closestWay = newWay; } } if(waysToVisitObj.empty()) continue; for(auto way : waysToVisitObj) { way->evaluationContext.closestWayRatio = way->evaluationContext.movementCost / closestWay->evaluationContext.movementCost; if(way->hero && ai->nullkiller->canMove(way->hero.h)) { tasks.push_back(sptr(*way)); } } } }; if(specificObjects) { captureObjects(objectsToCapture); } else { captureObjects(std::vector(ai->visitableObjs.begin(), ai->visitableObjs.end())); } return tasks; } bool CaptureObjectsBehavior::shouldVisitObject(ObjectIdRef obj) const { const CGObjectInstance* objInstance = obj; if(!objInstance) return false; if(objectTypes.size() && !vstd::contains(objectTypes, objInstance->ID.num)) { return false; } if(objectSubTypes.size() && !vstd::contains(objectSubTypes, objInstance->subID)) { return false; } const int3 pos = objInstance->visitablePos(); if(objInstance->ID != Obj::CREATURE_GENERATOR1 && vstd::contains(ai->alreadyVisited, objInstance) || obj->wasVisited(ai->playerID)) { return false; } auto playerRelations = cb->getPlayerRelations(ai->playerID, objInstance->tempOwner); if(playerRelations != PlayerRelations::ENEMIES && !isWeeklyRevisitable(objInstance)) { return false; } //it may be hero visiting this obj //we don't try visiting object on which allied or owned hero stands // -> it will just trigger exchange windows and AI will be confused that obj behind doesn't get visited const CGObjectInstance * topObj = cb->getTopObj(pos); if(!topObj) return false; // partly visible obj but its visitable pos is not visible. if(topObj->ID == Obj::HERO && cb->getPlayerRelations(ai->playerID, topObj->tempOwner) != PlayerRelations::ENEMIES) return false; else return true; //all of the following is met }