private int OnGetDamageBonus(void *pStats, void *pCreature, int bOffHand) { int damageBonus = getDamageBonusHook.CallOriginal(pStats, pCreature, bOffHand); CNWSCreatureStats stats = CNWSCreatureStats.FromPointer(pStats); CNWSCreature creature = stats.m_pBaseCreature; bool offHand = bOffHand.ToBool(); CNWSItem?weapon = GetEquippedWeapon(creature, offHand); uint baseItem = weapon != null ? weapon.m_nBaseItem : (uint)BaseItem.Gloves; bool hasApplicableFeat = false; bool applicableFeatExists = greaterWeaponSpecializationMap.TryGetValue(baseItem, out HashSet <ushort>?types); if (applicableFeatExists) { foreach (ushort feat in types !) { hasApplicableFeat = stats.HasFeat(feat).ToBool(); if (hasApplicableFeat) { break; } } } if (applicableFeatExists && hasApplicableFeat) { damageBonus += GreaterWeaponSpecializationDamageBonus; if ((*NWNXLib.EnableCombatDebugging()).ToBool() && stats.m_bIsPC.ToBool()) { CNWSCombatAttackData currentAttack = creature.m_pcCombatRound.GetAttack(creature.m_pcCombatRound.m_nCurrentAttack); StringBuilder debugMessage = new StringBuilder(currentAttack.m_sDamageDebugText.ToString()); debugMessage.Append(" + "); if (currentAttack.m_nAttackResult == 3) { int criticalThreat = stats.GetCriticalHitMultiplier(bOffHand); debugMessage.Append(GreaterWeaponSpecializationDamageBonus * criticalThreat); debugMessage.Append(" (Greater Weapon Specialization Feat) (Critical x"); debugMessage.Append(criticalThreat); debugMessage.Append(")"); } else { debugMessage.Append(GreaterWeaponSpecializationDamageBonus); debugMessage.Append(" (Greater Weapon Specialization Feat) "); } currentAttack.m_sDamageDebugText = debugMessage.ToString().ToExoString(); } } return(damageBonus); }
private int OnGetEpicWeaponDevastatingCritical(void *pStats, void *pWeapon) { CNWSCreatureStats stats = CNWSCreatureStats.FromPointer(pStats); CNWSItem weapon = CNWSItem.FromPointer(pWeapon); uint weaponType = weapon == null ? (uint)BaseItem.Gloves : CNWSItem.FromPointer(pWeapon).m_nBaseItem; bool hasApplicableFeat = false; bool applicableFeatExists = epicWeaponDevastatingCriticalMap.TryGetValue(weaponType, out HashSet <ushort>?types); if (applicableFeatExists) { foreach (ushort feat in types !) { hasApplicableFeat = stats.HasFeat(feat).ToBool(); if (hasApplicableFeat) { break; } } } bool canUseFeat = applicableFeatExists && hasApplicableFeat || getEpicWeaponDevastatingCriticalHook.CallOriginal(pStats, pWeapon).ToBool(); if (weapon != null && canUseFeat && OnDevastatingCriticalHit != null) { CNWSCreature creature = stats.m_pBaseCreature; CNWSCombatRound combatRound = creature.m_pcCombatRound; CNWSCombatAttackData attackData = combatRound.GetAttack(combatRound.m_nCurrentAttack); DevastatingCriticalData devastatingCriticalData = new DevastatingCriticalData { Weapon = weapon.ToNwObject <NwItem>() !, Target = creature.m_oidAttackTarget.ToNwObject <NwGameObject>() !, Damage = attackData.GetTotalDamage(1), }; OnDevastatingCriticalHit(devastatingCriticalData); if (devastatingCriticalData.Bypass) { attackData.m_bKillingBlow = 0; return(0); } } return(canUseFeat.ToInt()); }
private static OnCreatureAttack GetEventData(NwCreature creature, NwGameObject target, CNWSCombatRound combatRound, int attackNumber) { CNWSCombatAttackData combatAttackData = combatRound.GetAttack(attackNumber); return(new OnCreatureAttack { CombatAttackData = combatAttackData, Attacker = creature, Target = target, AttackNumber = attackNumber + 1, // 1-based for backwards compatibility WeaponAttackType = (WeaponAttackType)combatAttackData.m_nWeaponAttackType, SneakAttack = (SneakAttack)(combatAttackData.m_bSneakAttack + (combatAttackData.m_bDeathAttack << 1)), KillingBlow = combatAttackData.m_bKillingBlow.ToBool(), AttackType = combatAttackData.m_nAttackType, AttackRoll = combatAttackData.m_nToHitRoll, AttackModifier = combatAttackData.m_nToHitMod, IsRangedAttack = combatAttackData.m_bRangedAttack.ToBool(), IsCoupDeGrace = combatAttackData.m_bCoupDeGrace.ToBool(), IsAttackDeflected = combatAttackData.m_bAttackDeflected.ToBool(), IsCriticalThreat = combatAttackData.m_bCriticalThreat.ToBool(), DamageData = new DamageData <short>(combatAttackData.m_nDamage), TotalDamage = combatAttackData.GetTotalDamage(), }); }
private int OnGetAttackModifierVersus(void *pStats, void *pCreature) { int attackMod = getAttackModifierVersusHook.CallOriginal(pStats, pCreature); CNWSCreatureStats stats = CNWSCreatureStats.FromPointer(pStats); CNWSCreature creature = stats.m_pBaseCreature; CNWSCombatRound combatRound = creature.m_pcCombatRound; if (combatRound == null) { return(attackMod); } CNWSItem weapon = combatRound.GetCurrentAttackWeapon(combatRound.GetWeaponAttackType()); if (weapon == null) { return(attackMod); } uint baseItem = weapon.m_nBaseItem; bool hasApplicableFeat = false; bool applicableFeatExists = greaterWeaponFocusMap.TryGetValue(baseItem, out HashSet <ushort>?types); if (applicableFeatExists) { foreach (ushort feat in types !) { hasApplicableFeat = stats.HasFeat(feat).ToBool(); if (hasApplicableFeat) { break; } } } if (applicableFeatExists && hasApplicableFeat) { attackMod += GreaterWeaponFocusAttackBonus; if ((*NWNXLib.EnableCombatDebugging()).ToBool() && stats.m_bIsPC.ToBool()) { CNWSCombatAttackData currentAttack = combatRound.GetAttack(combatRound.m_nCurrentAttack); StringBuilder debugMessage = new StringBuilder(currentAttack.m_sDamageDebugText.ToString()); debugMessage.Append(" + "); debugMessage.Append(GreaterWeaponFocusAttackBonus); debugMessage.Append(" (Greater Weapon Focus Feat)"); currentAttack.m_sDamageDebugText = debugMessage.ToString().ToExoString(); } } if (EnableSlingGoodAimFeat && baseItem == (uint)BaseItem.Sling && stats.m_nRace != (ushort)RacialType.Halfling && stats.HasFeat((ushort)Feat.GoodAim).ToBool()) { int goodAimModifier = NWNXLib.Rules().GetRulesetIntEntry("GOOD_AIM_MODIFIER".ToExoString(), 1); attackMod += goodAimModifier; if ((*NWNXLib.EnableCombatDebugging()).ToBool() && stats.m_bIsPC.ToBool()) { CNWSCombatAttackData currentAttack = combatRound.GetAttack(combatRound.m_nCurrentAttack); StringBuilder debugMessage = new StringBuilder(currentAttack.m_sDamageDebugText.ToString()); debugMessage.Append(" + "); debugMessage.Append(goodAimModifier); debugMessage.Append(" (Good Aim Feat)"); currentAttack.m_sDamageDebugText = debugMessage.ToString().ToExoString(); } } return(attackMod); }