2018-12-01 10:30:37 +02:00
|
|
|
/*
|
|
|
|
* CollectRes.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"
|
2023-05-24 02:05:59 +03:00
|
|
|
#include "../../../lib/mapObjects/CGMarket.h"
|
2023-08-20 00:22:31 +03:00
|
|
|
#include "../../../lib/constants/StringConstants.h"
|
2018-12-01 10:30:37 +02:00
|
|
|
|
|
|
|
using namespace Goals;
|
|
|
|
|
|
|
|
bool CollectRes::operator==(const CollectRes & other) const
|
|
|
|
{
|
|
|
|
return resID == other.resID;
|
|
|
|
}
|
|
|
|
|
|
|
|
TGoalVec CollectRes::getAllPossibleSubgoals()
|
|
|
|
{
|
|
|
|
TGoalVec ret;
|
|
|
|
|
|
|
|
auto givesResource = [this](const CGObjectInstance * obj) -> bool
|
|
|
|
{
|
|
|
|
//TODO: move this logic to object side
|
|
|
|
//TODO: remember mithril exists
|
|
|
|
//TODO: water objects
|
|
|
|
//TODO: Creature banks
|
|
|
|
|
|
|
|
//return false first from once-visitable, before checking if they were even visited
|
|
|
|
switch (obj->ID.num)
|
|
|
|
{
|
|
|
|
case Obj::TREASURE_CHEST:
|
2023-04-05 03:26:29 +03:00
|
|
|
return resID == GameResID(EGameResID::GOLD);
|
2018-12-01 10:30:37 +02:00
|
|
|
break;
|
|
|
|
case Obj::RESOURCE:
|
2023-10-28 12:27:10 +03:00
|
|
|
return dynamic_cast<const CGResource*>(obj)->resourceID() == GameResID(resID);
|
2018-12-01 10:30:37 +02:00
|
|
|
break;
|
|
|
|
case Obj::MINE:
|
2023-10-28 12:27:10 +03:00
|
|
|
return (dynamic_cast<const CGMine*>(obj)->producedResource == GameResID(resID) &&
|
2018-12-01 10:30:37 +02:00
|
|
|
(cb->getPlayerRelations(obj->tempOwner, ai->playerID) == PlayerRelations::ENEMIES)); //don't capture our mines
|
|
|
|
break;
|
|
|
|
case Obj::CAMPFIRE:
|
|
|
|
return true; //contains all resources
|
|
|
|
break;
|
|
|
|
case Obj::WINDMILL:
|
2023-04-05 03:26:29 +03:00
|
|
|
switch (GameResID(resID).toEnum())
|
2018-12-01 10:30:37 +02:00
|
|
|
{
|
2023-04-05 03:26:29 +03:00
|
|
|
case EGameResID::GOLD:
|
|
|
|
case EGameResID::WOOD:
|
2018-12-01 10:30:37 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case Obj::MYSTICAL_GARDEN:
|
2023-04-05 03:26:29 +03:00
|
|
|
if ((resID != GameResID(EGameResID::GOLD)) && (resID != GameResID(EGameResID::GEMS)))
|
2018-12-01 10:30:37 +02:00
|
|
|
return false;
|
|
|
|
break;
|
2023-09-26 16:11:40 +03:00
|
|
|
case Obj::WATER_WHEEL:
|
2018-12-01 10:30:37 +02:00
|
|
|
case Obj::LEAN_TO:
|
|
|
|
case Obj::WAGON:
|
2023-04-05 03:26:29 +03:00
|
|
|
if (resID != GameResID(EGameResID::GOLD))
|
2018-12-01 10:30:37 +02:00
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return !vstd::contains(ai->alreadyVisited, obj); //for weekly / once visitable
|
|
|
|
};
|
|
|
|
|
|
|
|
std::vector<const CGObjectInstance *> objs;
|
|
|
|
for (auto obj : ai->visitableObjs)
|
|
|
|
{
|
|
|
|
if (givesResource(obj))
|
|
|
|
objs.push_back(obj);
|
|
|
|
}
|
|
|
|
for (auto h : cb->getHeroesInfo())
|
|
|
|
{
|
|
|
|
std::vector<const CGObjectInstance *> ourObjs(objs); //copy common objects
|
|
|
|
|
|
|
|
for (auto obj : ai->reservedHeroesMap[h]) //add objects reserved by this hero
|
|
|
|
{
|
|
|
|
if (givesResource(obj))
|
|
|
|
ourObjs.push_back(obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto obj : ourObjs)
|
|
|
|
{
|
|
|
|
auto waysToGo = ai->ah->howToVisitObj(h, ObjectIdRef(obj));
|
|
|
|
|
|
|
|
vstd::concatenate(ret, waysToGo);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
TSubgoal CollectRes::whatToDoToAchieve()
|
|
|
|
{
|
|
|
|
auto goals = getAllPossibleSubgoals();
|
|
|
|
auto trade = whatToDoToTrade();
|
|
|
|
if (!trade->invalid())
|
|
|
|
goals.push_back(trade);
|
|
|
|
|
|
|
|
if (goals.empty())
|
|
|
|
return sptr(Explore()); //we can always do that
|
|
|
|
else
|
|
|
|
return fh->chooseSolution(goals); //TODO: evaluate trading
|
|
|
|
}
|
|
|
|
|
|
|
|
TSubgoal CollectRes::whatToDoToTrade()
|
|
|
|
{
|
|
|
|
std::vector<const IMarket *> markets;
|
|
|
|
|
|
|
|
std::vector<const CGObjectInstance *> visObjs;
|
|
|
|
ai->retrieveVisitableObjs(visObjs, true);
|
2023-05-01 15:29:56 +04:00
|
|
|
for(const CGObjectInstance * obj : visObjs)
|
2018-12-01 10:30:37 +02:00
|
|
|
{
|
2024-02-14 12:56:37 +02:00
|
|
|
const auto * m = dynamic_cast<const IMarket*>(obj);
|
|
|
|
|
|
|
|
if(m && m->allowsTrade(EMarketMode::RESOURCE_RESOURCE))
|
2018-12-01 10:30:37 +02:00
|
|
|
{
|
2023-05-01 15:29:56 +04:00
|
|
|
if(obj->ID == Obj::TOWN)
|
|
|
|
{
|
|
|
|
if(obj->tempOwner == ai->playerID)
|
|
|
|
markets.push_back(m);
|
|
|
|
}
|
|
|
|
else
|
2018-12-01 10:30:37 +02:00
|
|
|
markets.push_back(m);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
boost::sort(markets, [](const IMarket * m1, const IMarket * m2) -> bool
|
|
|
|
{
|
|
|
|
return m1->getMarketEfficiency() < m2->getMarketEfficiency();
|
|
|
|
});
|
|
|
|
|
|
|
|
markets.erase(boost::remove_if(markets, [](const IMarket * market) -> bool
|
|
|
|
{
|
2023-04-27 00:18:43 +04:00
|
|
|
auto * o = dynamic_cast<const CGObjectInstance *>(market);
|
2024-04-22 15:46:51 +03:00
|
|
|
// FIXME: disabled broken visitation of external markets
|
|
|
|
//if(o && !(o->ID == Obj::TOWN && o->tempOwner == ai->playerID))
|
|
|
|
|
|
|
|
if(o && o->ID == Obj::TOWN)
|
2018-12-01 10:30:37 +02:00
|
|
|
{
|
2023-04-27 00:18:43 +04:00
|
|
|
if(!ai->isAccessible(o->visitablePos()))
|
2018-12-01 10:30:37 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}), markets.end());
|
|
|
|
|
|
|
|
if (!markets.size())
|
|
|
|
{
|
|
|
|
for (const CGTownInstance * t : cb->getTownsInfo())
|
|
|
|
{
|
|
|
|
if (cb->canBuildStructure(t, BuildingID::MARKETPLACE) == EBuildingState::ALLOWED)
|
|
|
|
return sptr(BuildThis(BuildingID::MARKETPLACE, t).setpriority(2));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const IMarket * m = markets.back();
|
|
|
|
//attempt trade at back (best prices)
|
|
|
|
int howManyCanWeBuy = 0;
|
2023-04-28 00:29:16 +03:00
|
|
|
for (GameResID i = EGameResID::WOOD; i <= EGameResID::GOLD; ++i)
|
2018-12-01 10:30:37 +02:00
|
|
|
{
|
2023-08-18 13:38:19 +03:00
|
|
|
if (i.getNum() == resID)
|
2018-12-01 10:30:37 +02:00
|
|
|
continue;
|
2024-01-09 22:38:54 +00:00
|
|
|
int toGive = -1;
|
|
|
|
int toReceive = -1;
|
2023-08-18 13:38:19 +03:00
|
|
|
m->getOffer(i, resID, toGive, toReceive, EMarketMode::RESOURCE_RESOURCE);
|
2018-12-01 10:30:37 +02:00
|
|
|
assert(toGive > 0 && toReceive > 0);
|
|
|
|
howManyCanWeBuy += toReceive * (ai->ah->freeResources()[i] / toGive);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (howManyCanWeBuy >= value)
|
|
|
|
{
|
2023-04-27 00:18:43 +04:00
|
|
|
auto * o = dynamic_cast<const CGObjectInstance *>(m);
|
|
|
|
auto backObj = cb->getTopObj(o->visitablePos()); //it'll be a hero if we have one there; otherwise marketplace
|
2018-12-01 10:30:37 +02:00
|
|
|
assert(backObj);
|
2023-04-27 00:18:43 +04:00
|
|
|
auto objid = o->id.getNum();
|
2018-12-01 10:30:37 +02:00
|
|
|
if (backObj->tempOwner != ai->playerID) //top object not owned
|
|
|
|
{
|
|
|
|
return sptr(VisitObj(objid)); //just go there
|
|
|
|
}
|
|
|
|
else //either it's our town, or we have hero there
|
|
|
|
{
|
2023-04-05 03:26:29 +03:00
|
|
|
return sptr(Trade(static_cast<EGameResID>(resID), value, objid).setisElementar(true)); //we can do this immediately
|
2018-12-01 10:30:37 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return sptr(Invalid()); //cannot trade
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CollectRes::fulfillsMe(TSubgoal goal)
|
|
|
|
{
|
|
|
|
if (goal->resID == resID)
|
|
|
|
if (goal->value >= value)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|