mirror of
https://github.com/vcmi/vcmi.git
synced 2025-01-28 03:57:02 +02:00
ea03273c8e
-It is possible to toggle adventure AI on / off typing "ai" in the console (default is true) -General cleaning and formatting of code -Some minor optimizations -Teal console log for AI AI tends to hang in the maze of mutexes and current player interface, further investigation is needed. Fixed #580 minor tweaks
130 lines
4.6 KiB
C++
130 lines
4.6 KiB
C++
#include "GeneralAI.h"
|
|
#include "../../CCallback.h"
|
|
#include "ExpertSystem.h"
|
|
|
|
/*
|
|
* ExpertSystem.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
|
|
*
|
|
*/
|
|
|
|
template <typename ruleType, typename facts> template <typename cond> void ExpertSystemShell<ruleType, facts>::DataDrivenReasoning(runType type)
|
|
{
|
|
std::set<ruleType>::iterator ir;
|
|
std::list<fact>::iterator iF;
|
|
std::set<std::pair<cond, input*>>::iterator ic;
|
|
bool factWasAdded = false; //carry it over inner loop
|
|
switch (type)
|
|
{
|
|
case ANY_GOAL: //first produced decision ends reasoning
|
|
{
|
|
int goalCounter = 0;
|
|
while(!goalCounter) //we reach goal or can't modify knowledge anymore
|
|
{
|
|
for (ir = knowledge.begin(); ir != knowledge.end(); ir++)
|
|
{
|
|
for (iF = factList.begin(); iF != factList.end(); iF++)
|
|
{
|
|
for (ic = ir->conditions.begin(); ic != ir->conditions.end(), ic++)
|
|
{
|
|
if (ic->first.object.matchesFact(iF->object)) //condition matches held object
|
|
{
|
|
(ic->second = *iF);
|
|
++(ir->conditionCounter);
|
|
}
|
|
}
|
|
if (ir->conditions.size() >= ir->conditionCounter());
|
|
{
|
|
ir->fireRule(); //all conditions matched
|
|
ir->conditionCounter = 0; //do it here or in rule itself?
|
|
if (dynamic_cast<&Goal>(*ir))
|
|
++goalCounter;
|
|
}
|
|
}
|
|
//modify set until something happens (hopefully!)
|
|
for (iF = factsToAdd.begin(); iF != factsToAdd.end(); iF++)
|
|
factList.insert(*iF);
|
|
if (factsToAdd.size())
|
|
{
|
|
factsToAdd.clear();
|
|
factWasAdded = true;
|
|
}
|
|
}
|
|
for (iF = factsToErase.begin(); iF != factsToErase.end(); iF++) //remove facts discarded in this run
|
|
factList.erase(knowledge.find(*iF));
|
|
factsToErase.clear(); //erase only after all rules had a chance to trigger
|
|
for (ir = rulesToErase.begin(); ir != rulesToErase.end(); ir++)
|
|
knowledge.erase(knowledge.find(*ir));
|
|
rulesToErase.clear();
|
|
for (ir = rulesToAdd.begin(); ir != rulesToAdd.end(); ir++)
|
|
knowledge.insert(*ir);
|
|
if (!(factWasAdded || rulesToAdd.size())) //we can't do anything more
|
|
break;
|
|
rulesToAdd.clear();
|
|
};
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
void BonusRule::fireRule()
|
|
{
|
|
for (std::set<std::pair<BonusCondition, BonusHolder*>>::iterator it = cons.begin(); it != cons.end(); it++)
|
|
{
|
|
switch (it->first.parameter)
|
|
{ //compare fact with condition
|
|
case BonusCondition::type:
|
|
if (!it->first.functor(it->second->object->type, it->first.value)) return;
|
|
break;
|
|
case BonusCondition::subtype: //probably suprfluous, Selector already handles that
|
|
if (!it->first.functor(it->second->object->subtype, it->first.value)) return;
|
|
break;
|
|
case BonusCondition::val:
|
|
if (!it->first.functor(it->second->object->val, it->first.value)) return;
|
|
break;
|
|
case BonusCondition::duration:
|
|
if (!it->first.functor(it->second->object->duration, it->first.value)) return;
|
|
break;
|
|
case BonusCondition::source: //likely to handle by selector
|
|
if (!it->first.functor(it->second->object->source, it->first.value)) return;
|
|
break;
|
|
case BonusCondition::id:
|
|
if (!it->first.functor(it->second->object->id, it->first.value)) return;
|
|
break;
|
|
case BonusCondition::valType: //ever needed?
|
|
if (!it->first.functor(it->second->object->valType, it->first.value)) return;
|
|
break;
|
|
case BonusCondition::additionalInfo:
|
|
if (!it->first.functor(it->second->object->additionalInfo, it->first.value)) return;
|
|
break;
|
|
case BonusCondition::effectRange:
|
|
if (!it->first.functor(it->second->object->effectRange, it->first.value)) return;
|
|
break;
|
|
default: //ignore or accept?
|
|
break;
|
|
};
|
|
}
|
|
//TODO: add new fact or modify existing one
|
|
}
|
|
//TODO: find out why it does not compile
|
|
//template <typename input, typename conType> void Rule<input, conType>::refreshRule(std::set<conType> &conditionSet)
|
|
//{
|
|
// cons.clear();
|
|
// for (std::set<conType>::iterator it = conditionSet.begin(); it != conditionSet.end(); it++)
|
|
// cons.insert(std::make_pair<conType,input*>(*it, NULL)); //pointer to condition and null fact
|
|
//}
|
|
//template <typename input, typename conType> void Rule<input, conType>::refreshRule()
|
|
//{
|
|
// for (std::set<std::pair<conType, input*>>::iterator it = cons.begin(); it != cons.end(); it++)
|
|
// *it->second = NULL;
|
|
//}
|
|
bool BonusCondition::matchesFact(Bonus &fact)
|
|
{
|
|
if (object(fact)) //Bonus(fact) matches local Selector(object)
|
|
return true;
|
|
return false;
|
|
}
|