1
0
mirror of https://github.com/vcmi/vcmi.git synced 2025-01-02 00:10:22 +02:00

Merged accurate shot bonus into death stare bonus

This commit is contained in:
Ivan Savenko 2024-01-13 15:55:07 +02:00
parent f6e0f46040
commit bb670cfb82
8 changed files with 50 additions and 43 deletions

View File

@ -727,14 +727,20 @@ Affected unit will deal additional damage after attack
### DEATH_STARE ### DEATH_STARE
Affected unit will kill additional units after attack Affected unit will kill additional units after attack. Used for Death stare (Mighty Gorgon) ability and for Accurate Shot (Pirates, HotA)
- subtype: - subtype:
- deathStareGorgon: random amount - deathStareGorgon: only melee attack, random amount of killed units
- deathStareCommander: fixed amount - deathStareNoRangePenalty: only ranged attacks without obstacle (walls) or range penalty
- deathStareRangePenalty: only ranged attacks with range penalty
- deathStareObstaclePenalty: only ranged attacks with obstacle (walls) penalty
- deathStareRangeObstaclePenalty: only ranged attacks with both range and obstacle penalty
- deathStareCommander: fixed amount, both melee and ranged attacks
- val: - val:
- for deathStareGorgon: chance to kill, counted separately for each unit in attacking stack, percentage. At most (stack size \* chance) units can be killed at once. TODO: recheck formula
- for deathStareCommander: number of creatures to kill, total amount of killed creatures is (attacker level / defender level) \* val - for deathStareCommander: number of creatures to kill, total amount of killed creatures is (attacker level / defender level) \* val
- for all other subtypes: chance to kill, counted separately for each unit in attacking stack, percentage. At most (stack size \* chance) units can be killed at once, rounded up
- addInfo:
- SpellID to be used as hit effect. If not set - 'deathStare' spell will be used
### SPECIAL_CRYSTAL_GENERATION ### SPECIAL_CRYSTAL_GENERATION
@ -744,15 +750,6 @@ If player has affected unit under his control in any army, he will receive addit
Affected unit will not use spellcast as default attack option Affected unit will not use spellcast as default attack option
### ACCURATE_SHOT
Affected unit will kill additional units after attack, similar to death stare - works only for ranged attack
- subtype:
spell identifier for spell that receives value that should be killed on input, spell.deathStare is used by default, use 'accurateShot' as part of spell name to allow detection for proper battle log description
- val:
chance to kill, counted separately for each unit in attacking stack, percentage. Chance gets lessened by 2/3 with range penalty and effect won't trigger with wall penalty. At most (stack size \* chance / 100 **[rounded up]**) units can be killed at once. TODO: recheck formula
## Creature spellcasting and activated abilities ## Creature spellcasting and activated abilities
### SPELLCASTER ### SPELLCASTER

View File

@ -515,7 +515,6 @@ static void loadBonusSubtype(BonusSubtypeID & subtype, BonusType type, const Jso
case BonusType::SPECIFIC_SPELL_POWER: case BonusType::SPECIFIC_SPELL_POWER:
case BonusType::ENCHANTED: case BonusType::ENCHANTED:
case BonusType::MORE_DAMAGE_FROM_SPELL: case BonusType::MORE_DAMAGE_FROM_SPELL:
case BonusType::ACCURATE_SHOT:
case BonusType::NOT_ACTIVE: case BonusType::NOT_ACTIVE:
{ {
VLC->identifiers()->requestIdentifier( "spell", node, [&subtype](int32_t identifier) VLC->identifiers()->requestIdentifier( "spell", node, [&subtype](int32_t identifier)

View File

@ -23,6 +23,10 @@ const BonusCustomSubtype BonusCustomSubtype::heroMovementLand(1);
const BonusCustomSubtype BonusCustomSubtype::heroMovementSea(0); const BonusCustomSubtype BonusCustomSubtype::heroMovementSea(0);
const BonusCustomSubtype BonusCustomSubtype::deathStareGorgon(0); const BonusCustomSubtype BonusCustomSubtype::deathStareGorgon(0);
const BonusCustomSubtype BonusCustomSubtype::deathStareCommander(1); const BonusCustomSubtype BonusCustomSubtype::deathStareCommander(1);
const BonusCustomSubtype BonusCustomSubtype::deathStareNoRangePenalty(2);
const BonusCustomSubtype BonusCustomSubtype::deathStareRangePenalty(3);
const BonusCustomSubtype BonusCustomSubtype::deathStareObstaclePenalty(4);
const BonusCustomSubtype BonusCustomSubtype::deathStareRangeObstaclePenalty(5);
const BonusCustomSubtype BonusCustomSubtype::rebirthRegular(0); const BonusCustomSubtype BonusCustomSubtype::rebirthRegular(0);
const BonusCustomSubtype BonusCustomSubtype::rebirthSpecial(1); const BonusCustomSubtype BonusCustomSubtype::rebirthSpecial(1);
const BonusCustomSubtype BonusCustomSubtype::visionsMonsters(0); const BonusCustomSubtype BonusCustomSubtype::visionsMonsters(0);

View File

@ -45,6 +45,10 @@ public:
static const BonusCustomSubtype deathStareGorgon; // 0 static const BonusCustomSubtype deathStareGorgon; // 0
static const BonusCustomSubtype deathStareCommander; static const BonusCustomSubtype deathStareCommander;
static const BonusCustomSubtype deathStareNoRangePenalty;
static const BonusCustomSubtype deathStareRangePenalty;
static const BonusCustomSubtype deathStareObstaclePenalty;
static const BonusCustomSubtype deathStareRangeObstaclePenalty;
static const BonusCustomSubtype rebirthRegular; // 0 static const BonusCustomSubtype rebirthRegular; // 0
static const BonusCustomSubtype rebirthSpecial; // 1 static const BonusCustomSubtype rebirthSpecial; // 1

View File

@ -174,7 +174,6 @@ class JsonNode;
BONUS_NAME(MAX_MORALE) /*cheat bonus*/ \ BONUS_NAME(MAX_MORALE) /*cheat bonus*/ \
BONUS_NAME(MAX_LUCK) /*cheat bonus*/ \ BONUS_NAME(MAX_LUCK) /*cheat bonus*/ \
BONUS_NAME(FEROCITY) /*extra attacks, only if at least some creatures killed while attacking target unit, val = amount of additional attacks, additional info = amount of creatures killed to trigger (default 1)*/ \ BONUS_NAME(FEROCITY) /*extra attacks, only if at least some creatures killed while attacking target unit, val = amount of additional attacks, additional info = amount of creatures killed to trigger (default 1)*/ \
BONUS_NAME(ACCURATE_SHOT) /*HotA Sea Dog-like ability - ranged only, val = full arrow trigger percent, subtype = spell identifier that killed value goes through (death stare by default) - use 'accurateShot' as part of spell name for proper battle log description*/ \
BONUS_NAME(ENEMY_ATTACK_REDUCTION) /*in % (value) eg. Nix (HotA)*/ \ BONUS_NAME(ENEMY_ATTACK_REDUCTION) /*in % (value) eg. Nix (HotA)*/ \
BONUS_NAME(REVENGE) /*additional damage based on how many units in stack died - formula: sqrt((number of creatures at battle start + 1) * creature health) / (total health now + 1 creature health) - 1) * 100% */ \ BONUS_NAME(REVENGE) /*additional damage based on how many units in stack died - formula: sqrt((number of creatures at battle start + 1) * creature health) / (total health now + 1 creature health) - 1) * 100% */ \
/* end of list */ /* end of list */

View File

@ -53,6 +53,10 @@ CIdentifierStorage::CIdentifierStorage()
registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "heroMovementSea", 0); registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "heroMovementSea", 0);
registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "deathStareGorgon", 0); registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "deathStareGorgon", 0);
registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "deathStareCommander", 1); registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "deathStareCommander", 1);
registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "deathStareNoRangePenalty", 2);
registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "deathStareRangePenalty", 3);
registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "deathStareObstaclePenalty", 4);
registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "deathStareRangeObstaclePenalty", 5);
registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "rebirthRegular", 0); registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "rebirthRegular", 0);
registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "rebirthSpecial", 1); registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "rebirthSpecial", 1);
registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "visionsMonsters", 0); registerObject(ModScope::scopeBuiltin(), "bonusSubtype", "visionsMonsters", 0);

View File

@ -1242,7 +1242,7 @@ void BattleActionProcessor::handleAttackBeforeCasting(const CBattleInfoCallback
attackCasting(battle, ranged, BonusType::SPELL_BEFORE_ATTACK, attacker, defender); //no death stare / acid breath needed? attackCasting(battle, ranged, BonusType::SPELL_BEFORE_ATTACK, attacker, defender); //no death stare / acid breath needed?
} }
void BattleActionProcessor::HandleDeathStareAndPirateShot(const CBattleInfoCallback & battle, bool ranged, const CStack * attacker, const CStack * defender) void BattleActionProcessor::handleDeathStare(const CBattleInfoCallback & battle, bool ranged, const CStack * attacker, const CStack * defender)
{ {
// mechanics of Death Stare as in H3: // mechanics of Death Stare as in H3:
// each gorgon have 10% chance to kill (counted separately in H3) -> binomial distribution // each gorgon have 10% chance to kill (counted separately in H3) -> binomial distribution
@ -1254,28 +1254,30 @@ void BattleActionProcessor::HandleDeathStareAndPirateShot(const CBattleInfoCallb
* X = 3 multiplier for shooting without penalty and X = 2 if shooting with penalty. Ability doesn't work if shooting at creatures behind walls. * X = 3 multiplier for shooting without penalty and X = 2 if shooting with penalty. Ability doesn't work if shooting at creatures behind walls.
*/ */
auto bonus = attacker->getBonus(Selector::type()(BonusType::DEATH_STARE)); auto subtype = BonusCustomSubtype::deathStareGorgon;
if(bonus == nullptr)
bonus = attacker->getBonus(Selector::type()(BonusType::ACCURATE_SHOT));
if(bonus->type == BonusType::ACCURATE_SHOT) //should not work from behind walls, except when being defender or under effect of golden bow etc. if (ranged)
{ {
if(!ranged) bool rangePenalty = battle.battleHasDistancePenalty(attacker, attacker->getPosition(), defender->getPosition());
return; bool obstaclePenalty = battle.battleHasWallPenalty(attacker, attacker->getPosition(), defender->getPosition());
if(battle.battleHasWallPenalty(attacker, attacker->getPosition(), defender->getPosition()))
return; if(rangePenalty)
{
if(obstaclePenalty)
subtype = BonusCustomSubtype::deathStareRangeObstaclePenalty;
else
subtype = BonusCustomSubtype::deathStareRangePenalty;
}
else
{
if(obstaclePenalty)
subtype = BonusCustomSubtype::deathStareObstaclePenalty;
else
subtype = BonusCustomSubtype::deathStareNoRangePenalty;
}
} }
int singleCreatureKillChancePercent; int singleCreatureKillChancePercent = attacker->valOfBonuses(BonusType::DEATH_STARE, subtype);
if(bonus->type == BonusType::ACCURATE_SHOT)
{
singleCreatureKillChancePercent = attacker->valOfBonuses(BonusType::ACCURATE_SHOT);
if(battle.battleHasDistancePenalty(attacker, attacker->getPosition(), defender->getPosition()))
singleCreatureKillChancePercent = (singleCreatureKillChancePercent * 2) / 3;
}
else //DEATH_STARE
singleCreatureKillChancePercent = attacker->valOfBonuses(BonusType::DEATH_STARE, BonusCustomSubtype::deathStareGorgon);
double chanceToKill = singleCreatureKillChancePercent / 100.0; double chanceToKill = singleCreatureKillChancePercent / 100.0;
vstd::amin(chanceToKill, 1); //cap at 100% vstd::amin(chanceToKill, 1); //cap at 100%
std::binomial_distribution<> distribution(attacker->getCount(), chanceToKill); std::binomial_distribution<> distribution(attacker->getCount(), chanceToKill);
@ -1284,16 +1286,16 @@ void BattleActionProcessor::HandleDeathStareAndPirateShot(const CBattleInfoCallb
int maxToKill = (attacker->getCount() * singleCreatureKillChancePercent + 99) / 100; int maxToKill = (attacker->getCount() * singleCreatureKillChancePercent + 99) / 100;
vstd::amin(killedCreatures, maxToKill); vstd::amin(killedCreatures, maxToKill);
if(bonus->type == BonusType::DEATH_STARE) killedCreatures += (attacker->level() * attacker->valOfBonuses(BonusType::DEATH_STARE, BonusCustomSubtype::deathStareCommander)) / defender->level();
killedCreatures += (attacker->level() * attacker->valOfBonuses(BonusType::DEATH_STARE, BonusCustomSubtype::deathStareCommander)) / defender->level();
if(killedCreatures) if(killedCreatures)
{ {
//TODO: death stare or accurate shot was not originally available for multiple-hex attacks, but... //TODO: death stare or accurate shot was not originally available for multiple-hex attacks, but...
SpellID spellID = SpellID(SpellID::DEATH_STARE); //also used as fallback spell for ACCURATE_SHOT SpellID spellID = SpellID(SpellID::DEATH_STARE); //also used as fallback spell for ACCURATE_SHOT
if(bonus->type == BonusType::ACCURATE_SHOT && bonus->subtype.as<SpellID>() != SpellID::NONE) auto bonus = attacker->getBonus(Selector::typeSubtype(BonusType::DEATH_STARE, subtype));
spellID = bonus->subtype.as<SpellID>(); if(bonus && bonus->additionalInfo[0] != SpellID::NONE)
spellID = SpellID(bonus->additionalInfo[0]);
const CSpell * spell = spellID.toSpell(); const CSpell * spell = spellID.toSpell();
spells::AbilityCaster caster(attacker, 0); spells::AbilityCaster caster(attacker, 0);
@ -1319,10 +1321,8 @@ void BattleActionProcessor::handleAfterAttackCasting(const CBattleInfoCallback &
return; return;
} }
if(attacker->hasBonusOfType(BonusType::DEATH_STARE) || attacker->hasBonusOfType(BonusType::ACCURATE_SHOT)) if(attacker->hasBonusOfType(BonusType::DEATH_STARE))
{ handleDeathStare(battle, ranged, attacker, defender);
HandleDeathStareAndPirateShot(battle, ranged, attacker, defender);
}
if(!defender->alive()) if(!defender->alive())
return; return;

View File

@ -45,7 +45,7 @@ class BattleActionProcessor : boost::noncopyable
void handleAttackBeforeCasting(const CBattleInfoCallback & battle, bool ranged, const CStack * attacker, const CStack * defender); void handleAttackBeforeCasting(const CBattleInfoCallback & battle, bool ranged, const CStack * attacker, const CStack * defender);
void HandleDeathStareAndPirateShot(const CBattleInfoCallback &battle, bool ranged, const CStack *attacker, const CStack *defender); void handleDeathStare(const CBattleInfoCallback &battle, bool ranged, const CStack *attacker, const CStack *defender);
void handleAfterAttackCasting(const CBattleInfoCallback & battle, bool ranged, const CStack * attacker, const CStack * defender); void handleAfterAttackCasting(const CBattleInfoCallback & battle, bool ranged, const CStack * attacker, const CStack * defender);
void attackCasting(const CBattleInfoCallback & battle, bool ranged, BonusType attackMode, const battle::Unit * attacker, const CStack * defender); void attackCasting(const CBattleInfoCallback & battle, bool ranged, BonusType attackMode, const battle::Unit * attacker, const CStack * defender);