123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163 |
- /*
- * GatherTroops.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 "Goals.h"
- #include "../VCAI.h"
- #include "../AIUtility.h"
- #include "../AIhelper.h"
- #include "../FuzzyHelper.h"
- #include "../ResourceManager.h"
- #include "../BuildingManager.h"
- #include "../../../lib/mapObjects/CGTownInstance.h"
- #include "../../../lib/constants/StringConstants.h"
- using namespace Goals;
- bool GatherTroops::operator==(const GatherTroops & other) const
- {
- return objid == other.objid;
- }
- int GatherTroops::getCreaturesCount(const CArmedInstance * army)
- {
- int count = 0;
- for(auto stack : army->Slots())
- {
- if(objid == stack.second->getCreatureID().num)
- {
- count += stack.second->count;
- }
- }
- return count;
- }
- TSubgoal GatherTroops::whatToDoToAchieve()
- {
- logAi->trace("Entering GatherTroops::whatToDoToAchieve");
- auto heroes = cb->getHeroesInfo(true);
- for(auto hero : heroes)
- {
- if(getCreaturesCount(hero) >= this->value)
- {
- logAi->trace("Completing GATHER_TROOPS by hero %s", hero->getNameTranslated());
- throw goalFulfilledException(sptr(*this));
- }
- }
- TGoalVec solutions = getAllPossibleSubgoals();
- if(solutions.empty())
- return sptr(Explore());
- return fh->chooseSolution(solutions);
- }
- TGoalVec GatherTroops::getAllPossibleSubgoals()
- {
- TGoalVec solutions;
- for(const CGTownInstance * t : cb->getTownsInfo())
- {
- int count = getCreaturesCount(t->getUpperArmy());
- if(count >= this->value)
- {
- if(t->visitingHero)
- {
- solutions.push_back(sptr(VisitObj(t->id.getNum()).sethero(t->visitingHero.get())));
- }
- else
- {
- vstd::concatenate(solutions, ai->ah->howToVisitObj(t));
- }
- continue;
- }
- auto creature = LIBRARY->creatures()->getByIndex(objid);
- if(t->getFactionID() == creature->getFactionID()) //TODO: how to force AI to build unupgraded creatures? :O
- {
- auto tryFindCreature = [&]() -> std::optional<std::vector<CreatureID>>
- {
- if(vstd::isValidIndex(t->getTown()->creatures, creature->getLevel() - 1))
- {
- auto itr = t->getTown()->creatures.begin();
- std::advance(itr, creature->getLevel() - 1);
- return make_optional(*itr);
- }
- return std::nullopt;
- };
- auto creatures = tryFindCreature();
- if(!creatures)
- continue;
- int upgradeNumber = vstd::find_pos(*creatures, creature->getId());
- if(upgradeNumber < 0)
- continue;
- BuildingID bid(BuildingID::getDwellingFromLevel(creature->getLevel(), upgradeNumber));
- if(t->hasBuilt(bid) && ai->ah->freeResources().canAfford(creature->getFullRecruitCost())) //this assumes only creatures with dwellings are assigned to faction
- {
- solutions.push_back(sptr(BuyArmy(t, creature->getAIValue() * this->value).setobjid(objid)));
- }
- /*else //disable random building requests for now - this code needs to know a lot of town/resource context to do more good than harm
- {
- return sptr(BuildThis(bid, t).setpriority(priority));
- }*/
- }
- }
- for(auto obj : ai->visitableObjs)
- {
- auto d = dynamic_cast<const CGDwelling *>(obj);
- if(!d || obj->ID == Obj::TOWN)
- continue;
- for(auto creature : d->creatures)
- {
- if(creature.first) //there are more than 0 creatures avaliabe
- {
- for(auto type : creature.second)
- {
- if(type.getNum() == objid && ai->ah->freeResources().canAfford(LIBRARY->creatures()->getById(type)->getFullRecruitCost()))
- vstd::concatenate(solutions, ai->ah->howToVisitObj(obj));
- }
- }
- }
- }
- CreatureID creID = CreatureID(objid);
- vstd::erase_if(solutions, [&](TSubgoal goal)->bool
- {
- return goal->hero && !goal->hero->getSlotFor(creID).validSlot() && !goal->hero->getFreeSlot().validSlot();
- });
- return solutions;
- //TODO: exchange troops between heroes
- }
- bool GatherTroops::fulfillsMe(TSubgoal goal)
- {
- if (!hero || hero == goal->hero) //we got army for desired hero or any hero
- if (goal->objid == objid) //same creature type //TODO: consider upgrades?
- if (goal->value >= value) //notify every time we get resources?
- return true;
- return false;
- }
|