| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269 | /* * Timed.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 "Timed.h"#include "Registry.h"#include "../ISpellMechanics.h"#include "../../MetaString.h"#include "../../battle/IBattleState.h"#include "../../battle/CBattleInfoCallback.h"#include "../../battle/Unit.h"#include "../../mapObjects/CGHeroInstance.h"#include "../../networkPacks/PacksForClientBattle.h"#include "../../networkPacks/SetStackEffect.h"#include "../../serializer/JsonSerializeFormat.h"VCMI_LIB_NAMESPACE_BEGINnamespace spells{namespace effects{static void describeEffect(std::vector<MetaString> & log, const Mechanics * m, const std::vector<Bonus> & bonuses, const battle::Unit * target) {	auto addLogLine = [&](const int32_t baseTextID, const boost::logic::tribool & plural)	{		MetaString line;		target->addText(line, EMetaText::GENERAL_TXT, baseTextID, plural);		target->addNameReplacement(line, plural);		log.push_back(std::move(line));	};	if(m->getSpellIndex() == SpellID::DISEASE)	{		addLogLine(553, boost::logic::indeterminate);		return;	}	for(const auto & bonus : bonuses)	{		switch(bonus.type)		{		case BonusType::NOT_ACTIVE:			{				switch(bonus.subtype.as<SpellID>().toEnum())				{				case SpellID::STONE_GAZE:					addLogLine(558, boost::logic::indeterminate);					return;				case SpellID::PARALYZE:					addLogLine(563, boost::logic::indeterminate);					return;				default:					break;				}			}			break;		case BonusType::POISON:			addLogLine(561, boost::logic::indeterminate);			return;		case BonusType::BIND_EFFECT:			addLogLine(-560, true);			return;		case BonusType::STACK_HEALTH:			{				if(bonus.val < 0)				{					BonusList unitHealth = *target->getBonuses(Selector::type()(BonusType::STACK_HEALTH));					auto oldHealth = unitHealth.totalValue();					unitHealth.push_back(std::make_shared<Bonus>(bonus));					auto newHealth = unitHealth.totalValue();					//"The %s shrivel with age, and lose %d hit points."					MetaString line;					target->addText(line, EMetaText::GENERAL_TXT, 551);					target->addNameReplacement(line);					line.replaceNumber(oldHealth - newHealth);					log.push_back(std::move(line));					return;				}			}			break;		default:			break;		}	}}void Timed::apply(ServerCallback * server, const Mechanics * m, const EffectTarget & target) const{	const bool describe = server->describeChanges();	int32_t duration = m->getEffectDuration();	std::vector<Bonus> converted;	convertBonus(m, duration, converted);	std::shared_ptr<const Bonus> peculiarBonus = nullptr;	std::shared_ptr<const Bonus> addedValueBonus = nullptr;	std::shared_ptr<const Bonus> fixedValueBonus = nullptr; 	const auto *casterHero = dynamic_cast<const CGHeroInstance *>(m->caster);	if(casterHero)	{ 		peculiarBonus = casterHero->getBonusLocalFirst(Selector::typeSubtype(BonusType::SPECIAL_PECULIAR_ENCHANT, BonusSubtypeID(m->getSpellId())));		addedValueBonus = casterHero->getBonusLocalFirst(Selector::typeSubtype(BonusType::SPECIAL_ADD_VALUE_ENCHANT, BonusSubtypeID(m->getSpellId())));		fixedValueBonus = casterHero->getBonusLocalFirst(Selector::typeSubtype(BonusType::SPECIAL_FIXED_VALUE_ENCHANT, BonusSubtypeID(m->getSpellId())));	}		//TODO: does hero specialty should affects his stack casting spells?	SetStackEffect sse;	BattleLogMessage blm;	blm.battleID = m->battle()->getBattle()->getBattleID();	sse.battleID = m->battle()->getBattle()->getBattleID();	for(const auto & t : target)	{		std::vector<Bonus> buffer;		std::copy(converted.begin(), converted.end(), std::back_inserter(buffer));		const battle::Unit * affected = t.unitValue;		if(!affected)		{			logGlobal->error("[Internal error] Invalid target for timed effect");			continue;		}		if(!affected->alive())			continue;		if(describe)			describeEffect(blm.lines, m, converted, affected);		//Apply hero specials - peculiar enchants		const auto tier = std::max(affected->creatureLevel(), 1); //don't divide by 0 for certain creatures (commanders, war machines)		if(peculiarBonus)		{			si32 power = 0;			switch (peculiarBonus->additionalInfo[0])			{			case 0: //normal				switch (tier)				{				case 1:				case 2:					power = 3;					break;				case 3:				case 4:					power = 2;					break;				case 5:				case 6:					power = 1;					break;				}				break;			case 1: 				//Coronius style specialty bonus.				//Please note that actual Coronius isnt here, because Slayer is a spell that doesnt affect monster stats and is used only in calculateDmgRange				break;			}			if(m->isNegativeSpell())			{				//negative spells like weakness are defined in json with negative numbers, so we need do same here				power = -1 * power;			}			for(Bonus& b : buffer)			{				b.val += power;			}		}		if(addedValueBonus)		{			for(Bonus & b : buffer)			{				b.val += addedValueBonus->additionalInfo[0];			}		}		if(fixedValueBonus)		{			for(Bonus & b : buffer)			{				b.val = fixedValueBonus->additionalInfo[0];			}		}		if(cumulative)			sse.toAdd.emplace_back(affected->unitId(), buffer);		else			sse.toUpdate.emplace_back(affected->unitId(), buffer);	}	if(!(sse.toAdd.empty() && sse.toUpdate.empty()))		server->apply(&sse);	if(describe && !blm.lines.empty())		server->apply(&blm);}void Timed::convertBonus(const Mechanics * m, int32_t & duration, std::vector<Bonus> & converted) const{	int32_t maxDuration = 0;	for(const auto & b : bonus)	{		Bonus nb(*b);		//use configured duration if present		if(nb.turnsRemain == 0)			nb.turnsRemain = duration;		vstd::amax(maxDuration, nb.turnsRemain);		nb.sid = BonusSourceID(m->getSpellId()); //for all		nb.source = BonusSource::SPELL_EFFECT;//for all		//fix to original config: shield should display damage reduction		if((nb.sid.as<SpellID>() == SpellID::SHIELD || nb.sid.as<SpellID>() == SpellID::AIR_SHIELD) && (nb.type == BonusType::GENERAL_DAMAGE_REDUCTION))			nb.val = 100 - nb.val;		//we need to know who cast Bind		else if(nb.sid.as<SpellID>() == SpellID::BIND && nb.type == BonusType::BIND_EFFECT && m->caster->getHeroCaster() == nullptr)			nb.additionalInfo = m->caster->getCasterUnitId();		converted.push_back(nb);	}	//if all spell effects have special duration, use it later for special bonuses	duration = maxDuration;}void Timed::serializeJsonUnitEffect(JsonSerializeFormat & handler){	assert(!handler.saving);	handler.serializeBool("cumulative", cumulative, false);	{		auto guard = handler.enterStruct("bonus");		const JsonNode & data = handler.getCurrent();		for(const auto & p : data.Struct())		{			//TODO: support JsonSerializeFormat in Bonus			auto guard = handler.enterStruct(p.first);			const JsonNode & bonusNode = handler.getCurrent();			auto b = JsonUtils::parseBonus(bonusNode);			if (b)				bonus.push_back(b);			else				logMod->error("Failed to parse bonus '%s'!", p.first);		}	}}}}VCMI_LIB_NAMESPACE_END
 |