| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565 | /* * TargetCondition.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 "TargetCondition.h"#include "../GameConstants.h"#include "../CBonusTypeHandler.h"#include "../battle/CBattleInfoCallback.h"#include "../battle/Unit.h"#include "../bonuses/BonusParams.h"#include "../bonuses/BonusList.h"#include "../json/JsonBonus.h"#include "../modding/IdentifierStorage.h"#include "../modding/ModUtility.h"#include "../serializer/JsonSerializeFormat.h"#include "../VCMI_Lib.h"#include <vcmi/spells/Spell.h>VCMI_LIB_NAMESPACE_BEGINnamespace spells{class TargetConditionItemBase : public TargetConditionItem{public:	bool inverted = false;	bool exclusive = false;	void setInverted(bool value) override	{		inverted = value;	}	void setExclusive(bool value) override	{		exclusive = value;	}	bool isExclusive() const override	{		return exclusive;	}	bool isReceptive(const Mechanics * m, const battle::Unit * target) const override	{		bool result = check(m, target);		return inverted != result;	}protected:	virtual bool check(const Mechanics * m, const battle::Unit * target) const = 0;};class SelectorCondition : public TargetConditionItemBase{public:	SelectorCondition(const CSelector & csel):		sel(csel)	{	}	SelectorCondition(const CSelector & csel, si32 minVal, si32 maxVal):		sel(csel),		minVal(minVal),		maxVal(maxVal)	{	}protected:	bool check(const Mechanics * m, const battle::Unit * target) const override	{		if(target->hasBonus(sel)) {			auto b = target->valOfBonuses(sel,"");			return b >= minVal && b <= maxVal;		}		return false;	}private:	CSelector sel;	si32 minVal = std::numeric_limits<si32>::min();	si32 maxVal = std::numeric_limits<si32>::max();};class ResistanceCondition : public TargetConditionItemBase{protected:	bool check(const Mechanics * m, const battle::Unit * target) const override	{		if(m->isPositiveSpell()) //Always pass on positive			return true;		return target->magicResistance() < 100;	}};class CreatureCondition : public TargetConditionItemBase{public:	CreatureCondition(const CreatureID & type_): type(type_) {}protected:	bool check(const Mechanics * m, const battle::Unit * target) const override	{		return target->creatureId() == type;	}private:	CreatureID type;};class AbsoluteLevelCondition : public TargetConditionItemBase{public:	AbsoluteLevelCondition()	{		inverted = false;		exclusive = true;	}protected:	bool check(const Mechanics * m, const battle::Unit * target) const override	{		if(!m->isMagicalEffect()) //Always pass on non-magical			return true;		std::stringstream cachingStr;		cachingStr << "type_" << vstd::to_underlying(BonusType::LEVEL_SPELL_IMMUNITY) << "addInfo_1";		TConstBonusListPtr levelImmunities = target->getBonuses(Selector::type()(BonusType::LEVEL_SPELL_IMMUNITY).And(Selector::info()(1)), cachingStr.str());		return (levelImmunities->size() == 0 || levelImmunities->totalValue() < m->getSpellLevel() || m->getSpellLevel() <= 0);	}};class AbsoluteSpellCondition : public TargetConditionItemBase{public:	AbsoluteSpellCondition()	{		inverted = false;		exclusive = true;	}protected:	bool check(const Mechanics * m, const battle::Unit * target) const override	{		std::stringstream cachingStr;		cachingStr << "type_" << vstd::to_underlying(BonusType::SPELL_IMMUNITY) << "subtype_" << m->getSpellIndex() << "addInfo_1";		return !target->hasBonus(Selector::typeSubtypeInfo(BonusType::SPELL_IMMUNITY, BonusSubtypeID(m->getSpellId()), 1), cachingStr.str());	}};class ElementalCondition : public TargetConditionItemBase{public:	ElementalCondition()	{		inverted = true;		exclusive = true;	}protected:	bool check(const Mechanics * m, const battle::Unit * target) const override	{		bool elementalImmune = false;		auto bearer = target->getBonusBearer();		m->getSpell()->forEachSchool([&](const SpellSchool & cnf, bool & stop) 		{			if (bearer->hasBonusOfType(BonusType::SPELL_SCHOOL_IMMUNITY, BonusSubtypeID(cnf)))			{				elementalImmune = true;				stop = true; //only bonus from one school is used			}			else if(!m->isPositiveSpell()) //negative or indifferent			{				if (bearer->hasBonusOfType(BonusType::NEGATIVE_EFFECTS_IMMUNITY, BonusSubtypeID(cnf)))				{					elementalImmune = true;					stop = true; //only bonus from one school is used				}			}		});		return elementalImmune;	}};class NormalLevelCondition : public TargetConditionItemBase{public:	NormalLevelCondition()	{		inverted = false;		exclusive = true;	}protected:	bool check(const Mechanics * m, const battle::Unit * target) const override	{		if(!m->isMagicalEffect()) //Always pass on non-magical			return true;		TConstBonusListPtr levelImmunities = target->getBonuses(Selector::type()(BonusType::LEVEL_SPELL_IMMUNITY));		return levelImmunities->size() == 0 ||		levelImmunities->totalValue() < m->getSpellLevel() ||		m->getSpellLevel() <= 0;	}};class NormalSpellCondition : public TargetConditionItemBase{public:	NormalSpellCondition()	{		inverted = false;		exclusive = true;	}protected:	bool check(const Mechanics * m, const battle::Unit * target) const override	{		return !target->hasBonusOfType(BonusType::SPELL_IMMUNITY, BonusSubtypeID(m->getSpellId()));	}};//for Hypnotizeclass HealthValueCondition : public TargetConditionItemBase{protected:	bool check(const Mechanics * m, const battle::Unit * target) const override	{		//todo: maybe do not resist on passive cast		//TODO: what with other creatures casting hypnotize, Faerie Dragons style?		int64_t subjectHealth = target->getAvailableHealth();		//apply 'damage' bonus for hypnotize, including hero specialty		auto maxHealth = m->applySpellBonus(m->getEffectValue(), target);		return subjectHealth <= maxHealth;	}};class SpellEffectCondition : public TargetConditionItemBase{public:	SpellEffectCondition(const SpellID & spellID_): spellID(spellID_)	{		std::stringstream builder;		builder << "source_" << vstd::to_underlying(BonusSource::SPELL_EFFECT) << "id_" << spellID.num;		cachingString = builder.str();		selector = Selector::source(BonusSource::SPELL_EFFECT, BonusSourceID(spellID));	}protected:	bool check(const Mechanics * m, const battle::Unit * target) const override	{		return target->hasBonus(selector, cachingString);	}private:	CSelector selector;	std::string cachingString;	SpellID spellID;};class ReceptiveFeatureCondition : public TargetConditionItemBase{protected:	bool check(const Mechanics * m, const battle::Unit * target) const override	{		return m->isPositiveSpell() && target->hasBonus(selector, cachingString);	}private:	CSelector selector = Selector::type()(BonusType::RECEPTIVE);	std::string cachingString = "type_RECEPTIVE";};class ImmunityNegationCondition : public TargetConditionItemBase{protected:	bool check(const Mechanics * m, const battle::Unit * target) const override	{		const bool battleWideNegation = target->hasBonusOfType(BonusType::NEGATE_ALL_NATURAL_IMMUNITIES, BonusCustomSubtype::immunityBattleWide);		const bool heroNegation = target->hasBonusOfType(BonusType::NEGATE_ALL_NATURAL_IMMUNITIES, BonusCustomSubtype::immunityEnemyHero);		//Non-magical effects is not affected by orb of vulnerability		if(!m->isMagicalEffect())			return false;		//anyone can cast on artifact holder`s stacks		if(heroNegation)		{			return true;		}		//this stack is from other player		else if(battleWideNegation)		{			if(m->ownerMatches(target, false))				return true;		}		return false;	}};class DefaultTargetConditionItemFactory : public TargetConditionItemFactory{public:	Object createAbsoluteLevel() const override	{		static auto antimagicCondition = std::make_shared<AbsoluteLevelCondition>();        return antimagicCondition;	}	Object createAbsoluteSpell() const override	{		static auto alCondition = std::make_shared<AbsoluteSpellCondition>();		return alCondition;	}	Object createElemental() const override	{		static auto elementalCondition = std::make_shared<ElementalCondition>();		return elementalCondition;	}	Object createResistance() const override	{		static auto elementalCondition = std::make_shared<ResistanceCondition>();		return elementalCondition;	}	Object createNormalLevel() const override	{		static auto nlCondition = std::make_shared<NormalLevelCondition>();		return nlCondition;	}	Object createNormalSpell() const override	{		static auto nsCondition = std::make_shared<NormalSpellCondition>();		return nsCondition;	}	Object createConfigurable(std::string scope, std::string type, std::string identifier) const override	{		if(type == "bonus")		{			//TODO: support custom bonus types			auto it = bonusNameMap.find(identifier);			if(it != bonusNameMap.end())				return std::make_shared<SelectorCondition>(Selector::type()(it->second));			auto params = BonusParams(identifier, "", -1);			if(params.isConverted)			{				if(params.val)					return std::make_shared<SelectorCondition>(params.toSelector(), *params.val, *params.val);				return std::make_shared<SelectorCondition>(params.toSelector());			}				logMod->error("Invalid bonus type %s in spell target condition.", identifier);		}		else if(type == "creature")		{			auto rawId = VLC->identifiers()->getIdentifier(scope, type, identifier, true);			if(rawId)				return std::make_shared<CreatureCondition>(CreatureID(rawId.value()));			else				logMod->error("Invalid creature %s type in spell target condition.", identifier);		}		else if(type == "spell")		{			auto rawId = VLC->identifiers()->getIdentifier(scope, type, identifier, true);			if(rawId)				return std::make_shared<SpellEffectCondition>(SpellID(rawId.value()));			else				logMod->error("Invalid spell %s in spell target condition.", identifier);		}		else if(type == "healthValueSpecial")		{			return  std::make_shared<HealthValueCondition>();		}		else		{			logMod->error("Invalid type %s in spell target condition.", type);		}		return Object();	}	Object createFromJsonStruct(const JsonNode & jsonStruct) const override	{		auto type = jsonStruct["type"].String();		auto parameters = jsonStruct["parameters"];		if(type == "selector")		{			auto minVal = std::numeric_limits<si32>::min();			auto maxVal = std::numeric_limits<si32>::max();			if(parameters["minVal"].isNumber())				minVal = parameters["minVal"].Integer();			if(parameters["maxVal"].isNumber())				maxVal = parameters["maxVal"].Integer();			auto sel = JsonUtils::parseSelector(parameters);			return std::make_shared<SelectorCondition>(sel, minVal, maxVal);		}		logMod->error("Invalid type %s in spell target condition.", type);		return Object();	}	Object createReceptiveFeature() const override	{		static auto condition = std::make_shared<ReceptiveFeatureCondition>();		return condition;	}	Object createImmunityNegation() const override	{		static auto condition = std::make_shared<ImmunityNegationCondition>();		return condition;	}};const TargetConditionItemFactory * TargetConditionItemFactory::getDefault(){	static std::unique_ptr<TargetConditionItemFactory> singleton;	if(!singleton)		singleton = std::make_unique<DefaultTargetConditionItemFactory>();	return singleton.get();}bool TargetCondition::isReceptive(const Mechanics * m, const battle::Unit * target) const{	if(!check(absolute, m, target))		return false;	for(const auto & item : negation)	{		if(item->isReceptive(m, target))			return true;	}	return check(normal, m, target);}void TargetCondition::serializeJson(JsonSerializeFormat & handler, const ItemFactory * itemFactory){	if(handler.saving)	{		logGlobal->error("Spell target condition saving is not supported");		return;	}	absolute.clear();	normal.clear();	negation.clear();	absolute.push_back(itemFactory->createAbsoluteSpell());	absolute.push_back(itemFactory->createAbsoluteLevel());	normal.push_back(itemFactory->createElemental());	normal.push_back(itemFactory->createResistance());	normal.push_back(itemFactory->createNormalLevel());	normal.push_back(itemFactory->createNormalSpell());	negation.push_back(itemFactory->createReceptiveFeature());	negation.push_back(itemFactory->createImmunityNegation());	{		auto anyOf = handler.enterStruct("anyOf");		loadConditions(anyOf->getCurrent(), false, false, itemFactory);	}	{		auto allOf = handler.enterStruct("allOf");		loadConditions(allOf->getCurrent(), true, false, itemFactory);	}	{		auto noneOf = handler.enterStruct("noneOf");		loadConditions(noneOf->getCurrent(), true, true, itemFactory);	}}bool TargetCondition::check(const ItemVector & condition, const Mechanics * m, const battle::Unit * target) const{	bool nonExclusiveCheck = false;	bool nonExclusiveExits = false;	for(const auto & item : condition)	{		if(item->isExclusive())		{			if(!item->isReceptive(m, target))				return false;		}		else		{			if(item->isReceptive(m, target))				nonExclusiveCheck = true;			nonExclusiveExits = true;		}	}	return !nonExclusiveExits || nonExclusiveCheck;}void TargetCondition::loadConditions(const JsonNode & source, bool exclusive, bool inverted, const ItemFactory * itemFactory){	for(const auto & keyValue : source.Struct())	{		bool isAbsolute;		const JsonNode & value = keyValue.second;		if(value.String() == "absolute")			isAbsolute = true;		else if(value.String() == "normal")			isAbsolute = false;		else if(value.isStruct()) //assume conditions have a new struct format			isAbsolute = value["absolute"].Bool();		else			continue;		std::shared_ptr<TargetConditionItem> item;		if(value.isStruct())			item = itemFactory->createFromJsonStruct(value);		else		{			std::string scope;			std::string type;			std::string identifier;			ModUtility::parseIdentifier(keyValue.first, scope, type, identifier);			item = itemFactory->createConfigurable(keyValue.second.getModScope(), type, identifier);		}		if(item)		{			item->setExclusive(exclusive);			item->setInverted(inverted);			if(isAbsolute)				absolute.push_back(item);			else				normal.push_back(item);		}	}}}VCMI_LIB_NAMESPACE_END
 |