/// <summary> /// Calculates the damage, using a skill. /// </summary> /// <param name="attacker">The object which is attacking.</param> /// <param name="defender">The object which is defending.</param> /// <param name="skill">The skill which is used.</param> /// <returns>The hit information.</returns> public static HitInfo CalculateDamage(this IAttacker attacker, IAttackable defender, SkillEntry?skill) { if (!attacker.IsAttackSuccessfulTo(defender)) { return(new HitInfo(0, 0, DamageAttributes.Undefined)); } DamageAttributes attributes = DamageAttributes.Undefined; bool isCriticalHit = Rand.NextRandomBool(attacker.Attributes[Stats.CriticalDamageChance]); bool isExcellentHit = Rand.NextRandomBool(attacker.Attributes[Stats.ExcellentDamageChance]); bool isIgnoringDefense = Rand.NextRandomBool(attacker.Attributes[Stats.DefenseIgnoreChance]); attacker.GetBaseDmg(skill, out int baseMinDamage, out int baseMaxDamage); int dmg; if (isExcellentHit) { dmg = (int)(baseMaxDamage * 1.2); attributes |= DamageAttributes.Excellent; } else if (isCriticalHit) { dmg = baseMaxDamage; attributes |= DamageAttributes.Critical; } else if (baseMaxDamage <= baseMinDamage) { dmg = baseMinDamage; } else { dmg = Rand.NextInt(baseMinDamage, baseMaxDamage); } if (!isIgnoringDefense) { var defenseAttribute = defender.GetDefenseAttribute(attacker); var defense = (int)defender.Attributes[defenseAttribute]; if (defender.Attributes[Stats.IsShieldEquipped] > 0) { defense += (int)(defense * defender.Attributes[Stats.DefenseIncreaseWithEquippedShield]); } dmg -= defense; } else { attributes |= DamageAttributes.IgnoreDefense; } dmg = Math.Max(dmg, 0); dmg = (int)(dmg * defender.Attributes[Stats.DamageReceiveDecrement]); dmg = (int)(dmg * attacker.Attributes[Stats.AttackDamageIncrease]); if (skill != null) { dmg += (int)attacker.Attributes[Stats.SkillDamageBonus]; dmg = (int)(dmg * attacker.Attributes[Stats.SkillMultiplier]); } if (attacker.Attributes[Stats.IsTwoHandedWeaponEquipped] > 0) { dmg = (int)(dmg * attacker.Attributes[Stats.TwoHandedWeaponDamageIncrease]); } if (attacker is Player && defender is Player) { dmg += (int)attacker.Attributes[Stats.FinalDamageIncreasePvp]; } bool isDoubleDamage = Rand.NextRandomBool(attacker.Attributes[Stats.DoubleDamageChance]); if (isDoubleDamage) { dmg *= 2; attributes |= DamageAttributes.Double; } var minimumDamage = attacker.Attributes[Stats.Level] / 10; return(defender.GetHitInfo(Math.Max((uint)dmg, (uint)minimumDamage), attributes, attacker)); }
/// <summary> /// Calculates the damage, using a skill. /// </summary> /// <param name="attacker">The object which is attacking.</param> /// <param name="defender">The object which is defending.</param> /// <param name="skill">The skill which is used.</param> /// <returns>The hit information.</returns> public static HitInfo CalculateDamage(this IAttackable attacker, IAttackable defender, SkillEntry skill) { if (!attacker.IsAttackSuccessfulTo(defender)) { return(new HitInfo(0, 0, DamageAttributes.Undefined)); } DamageAttributes attributes = DamageAttributes.Undefined; bool isCriticalHit = Rand.NextRandomBool(attacker.Attributes[Stats.CriticalDamageChance]); bool isExcellentHit = Rand.NextRandomBool(attacker.Attributes[Stats.ExcellentDamageChance]); bool isIgnoringDefense = Rand.NextRandomBool(attacker.Attributes[Stats.DefenseIgnoreChance]); attacker.GetBaseDmg(skill, out int baseMinDamage, out int baseMaxDamage); int dmg; if (isExcellentHit) { dmg = (int)(baseMaxDamage * 1.2); attributes |= DamageAttributes.Excellent; } else if (isCriticalHit) { dmg = baseMaxDamage; attributes |= DamageAttributes.Critical; } else if (baseMaxDamage <= baseMinDamage) { dmg = baseMinDamage; } else { dmg = Rand.NextInt(baseMinDamage, baseMaxDamage); } if (!isIgnoringDefense) { var defenseAttribute = defender.GetDefenseAttribute(attacker); dmg -= (int)defender.Attributes[defenseAttribute]; } else { attributes |= DamageAttributes.IgnoreDefense; } dmg = Math.Max(dmg, 0); dmg = (int)(dmg * defender.Attributes[Stats.DamageReceiveDecrement]); dmg = (int)(dmg * attacker.Attributes[Stats.AttackDamageIncrease]); if (skill != null) { dmg = (int)(dmg * attacker.Attributes[Stats.SkillMultiplier]); } bool isDoubleDamage = Rand.NextRandomBool(attacker.Attributes[Stats.DoubleDamageChance]); if (isDoubleDamage) { dmg *= 2; attributes |= DamageAttributes.Double; } // now we have the final damage calculated. We have to calculate which part of the damage damages the shield and which the health. HitInfo hi; var shieldBypass = Rand.NextRandomBool(attacker.Attributes[Stats.ShieldBypassChance]); if (shieldBypass || defender.Attributes[Stats.CurrentShield] < 1) { hi = new HitInfo((uint)dmg, 0, 0); } else { var shieldRatio = 0.90; shieldRatio -= attacker.Attributes[Stats.ShieldDecreaseRateIncrease]; shieldRatio += defender.Attributes[Stats.ShieldRateIncrease]; shieldRatio = Math.Max(0, shieldRatio); shieldRatio = Math.Min(1, shieldRatio); hi = new HitInfo((uint)(dmg * (1 - shieldRatio)), (uint)(dmg * shieldRatio), attributes); } return(hi); }
int IRandomizer.NextInt(uint min, uint max) { return(Rand.NextInt(min, max)); }