PRC8/nwn/nwnprc/trunk/scripts/prc_onhitcast.nss
Jaysyn904 e641b42f84 Exalted update
Updated Vow of Poverty. Added Sanctify Ki Strike, Holy Strike, Fist of Heavens, Vow of Abstinence, Vow of Chastity & Gift of Faith.  (@fenac).  Turned off the Taunt & Parry skills.  Re-disabled AC & save bonuses from Tumble & Spellcraft.   Updated min() & max() to PRCmin() & PRCmax() to not conflict with similarly named NUI adjacent functions.  Set Point Blank Shot to 30' per PnP.  Added icon for Chosen of Evil.  Started work on Hidden Talent.  Created Psionics function cheatsheet.  Updated release archive.
2025-01-29 22:46:38 -05:00

734 lines
32 KiB
Plaintext

//::///////////////////////////////////////////////
//:: User Defined OnHitCastSpell code
//:: prc_onhitcast
//:://////////////////////////////////////////////
/*
This file holds all the OnHitCastSpell events
used by PRC.
It was created to replace x2_s3_onhitcast so that
it wouldn't override module-specific onhitcast events.
Add your own entries after the previous ones. Try to
keep variable scope as little as possible. ie, no top-
level variables if you just can avoid it.
If your entry is long (over 20 lines), consider placing
the guts of it outside the main to improve readability
for the rest of us :D
Please remember to comment your entry.
At least mention what class ability / spell / whatever
it is part of.
*/
//:://////////////////////////////////////////////
//:: Created By: Various people
//:://////////////////////////////////////////////
#include "psi_inc_onhit"
#include "inc_rend"
#include "psi_inc_ac_const"
#include "prc_inc_onhit"
#include "tob_inc_tobfunc"
#include "prc_inc_sp_tch"
void SetRancorVar(object oPC);
void SetImprovedRicochetVar(object oPC);
void DoImprovedRicochet(object oPC, object oTarget);
object GetProperTarget(object oPC, object oTarget)
{
location lTarget = GetLocation(oPC);
// Use the function to get the closest creature as a target
object oAreaTarget = MyFirstObjectInShape(SHAPE_SPHERE, RADIUS_SIZE_MEDIUM, lTarget, TRUE, OBJECT_TYPE_CREATURE);
while(GetIsObjectValid(oAreaTarget))
{
if(oAreaTarget != oPC && // Not you.
GetIsEnemy(oPC, oAreaTarget) && // Enemies only, please
GetIsInMeleeRange(oPC, oAreaTarget)) // They must be in melee range
{
return oAreaTarget;
}
//Select the next target within the spell shape.
oAreaTarget = MyNextObjectInShape(SHAPE_SPHERE, RADIUS_SIZE_MEDIUM, lTarget, TRUE, OBJECT_TYPE_CREATURE);
}
return oTarget;
}
void main()
{
object oSpellOrigin = OBJECT_SELF; // On a weapon: The one wielding the weapon. On an armor: The one wearing an armor
// Call the normal OnHitCastSpell: Unique script
if (DEBUG) DoDebug("prc_onhitcast: entered, executing normal onhitcastspell unique power script x2_s3_onhitcast for "+GetName(oSpellOrigin));
ExecuteScript("x2_s3_onhitcast", oSpellOrigin);
// motu99: setting a local int, so that onhitcast impact spell scripts can find out, whether they were called from prc_onhitcast
// or from somewhere else (presumably the aurora engine). This local int will be deleted just before we exit prc_onhitcast
// Note that any scripts that are called by a DelayCommand (or AssignCommand) from prc_onhitcast will not find this local int
SetLocalInt(oSpellOrigin, "prc_ohc", TRUE);
object oSpellTarget = PRCGetSpellTargetObject(oSpellOrigin); // On a weapon: The one being hit. On an armor: The one hitting the armor
// Make sure you don't hit yourself. Some idiot didn't check that.
if (oSpellOrigin == oSpellTarget)
oSpellTarget = GetProperTarget(oSpellOrigin, oSpellTarget);
// motu99: replacing call to Bioware's GetSpellCastItem with new PRC wrapper function
// will ensure that we retrieve a valid item when we are called from scripted combat (prc_inc_combat) or
object oItem = PRCGetSpellCastItem(oSpellOrigin); // The item casting triggering this spellscript
int iItemBaseType = GetBaseItemType(oItem);
// DEBUG code, remove
if (DEBUG)
{
if(!GetIsOnHitCastSpell(oSpellTarget, oItem, oSpellOrigin))
DoDebug("prc_onhitcast: Warning, the currently running instance of prc_onhitcast was not recognized as an onhitcastspell");
}
// DEBUG code, remove
if (DEBUG) DoDebug("prc_onhitcast: now executing prc specific routines with item = "+ GetName(oItem)+", target = "+GetName(oSpellTarget)+", caller = "+GetName(oSpellOrigin)+", item type = "+IntToString(iItemBaseType));
// int nBArcher; // Blood Archer level
int nFoeHunter = GetLevelByClass(CLASS_TYPE_FOE_HUNTER, oSpellOrigin); // Foe Hunter Level
//int bItemIsWeapon;
//nBArcher = GetLevelByClass(CLASS_TYPE_BLARCHER, oSpellOrigin);
//// Swashbuckler Weakening and Wounding Criticals
if(GetHasFeat(INSIGHTFUL_STRIKE, oSpellOrigin))
ExecuteScript("prc_swashweak", oSpellOrigin);
//// Champion of Corellon damage healing for sneak/critical immune creatures
if(GetHasFeat(FEAT_COC_ELEGANT_STRIKE, oSpellOrigin))
ExecuteScript("prc_coc_heal", oSpellOrigin);
//// Stormlord Shocking & Thundering Spear
if(GetHasFeat(FEAT_THUNDER_WEAPON, oSpellOrigin))
ExecuteScript("ft_shockweap", oSpellOrigin);
if(GetHasSpellEffect(TEMPUS_ENCHANT_WEAPON, oItem))
{
int nRace = MyPRCGetRacialType(oSpellTarget);
if((GetLocalInt(oSpellOrigin, "WeapEchant1") == TEMPUS_ABILITY_VICIOUS &&
nRace == GetLocalInt(oSpellOrigin,"WeapEchantRace1")) ||
(GetLocalInt(oSpellOrigin,"WeapEchant2")==TEMPUS_ABILITY_VICIOUS &&
nRace == GetLocalInt(oSpellOrigin,"WeapEchantRace2")) ||
(GetLocalInt(oSpellOrigin,"WeapEchant3")==TEMPUS_ABILITY_VICIOUS &&
nRace == GetLocalInt(oSpellOrigin,"WeapEchantRace3"))
)
ApplyEffectToObject(DURATION_TYPE_INSTANT,EffectDamage(d6()),oSpellOrigin);
}
if (GetIsPC(oSpellOrigin))
SetLocalInt(oSpellOrigin,"DmgDealt",GetTotalDamageDealt());
/// Vassal of Bahamut Dragonwrack
if(GetLevelByClass(CLASS_TYPE_VASSAL, oSpellOrigin) > 3)
{
if (iItemBaseType == BASE_ITEM_ARMOR)
ExecuteScript("prc_vb_dw_armor", oSpellOrigin);
else
ExecuteScript("prc_vb_dw_weap", oSpellOrigin);
}
/* /// Blood Archer Acidic Blood
if (nBArcher >= 2)
{
if(iItemBaseType == BASE_ITEM_ARMOR)
ExecuteScript("prc_bldarch_ab", oSpellOrigin);
}
// Blood Archer Poison Blood
if (nBArcher > 0 && iItemBaseType != BASE_ITEM_ARMOR)
{
// poison number based on archer level
// gives proper DC for poison
int iPoison = 104 + nBArcher;
effect ePoison = EffectPoison(iPoison);
ApplyEffectToObject(DURATION_TYPE_PERMANENT, ePoison, oSpellTarget);
}*/
// Frenzied Berserker Auto Frenzy
if(iItemBaseType == BASE_ITEM_ARMOR && GetHasFeat(FEAT_FRENZY, oSpellOrigin))
{
if(!GetHasFeatEffect(FEAT_FRENZY, oSpellOrigin))
{
DelayCommand(0.01, ExecuteScript("prc_fb_auto_fre", oSpellOrigin) );
}
else if(GetCurrentHitPoints(oSpellOrigin) == 1 && GetHasFeat(FEAT_DEATHLESS_FRENZY, oSpellOrigin))
{
DelayCommand(0.01, ExecuteScript("prc_fb_deathless", oSpellOrigin) );
}
}
// Warsling Sniper Improved Ricochet
if (iItemBaseType == BASE_ITEM_BULLET
&& GetLevelByClass(CLASS_TYPE_HALFLING_WARSLINGER, oSpellOrigin) > 5
&& GetLocalInt(oSpellOrigin, "CanRicochet") != 2)
{
// Deactivates Ability
SetLocalInt(oSpellOrigin, "CanRicochet", 2);
DoImprovedRicochet(oSpellOrigin, oSpellTarget);
// Prevents the heartbeat script from running multiple times
if(GetLocalInt(oSpellOrigin, "ImpRicochetVarRunning") != 1)
{
DelayCommand(6.0, SetImprovedRicochetVar(oSpellOrigin) );
SetLocalInt(oSpellOrigin, "ImpRicochetVarRunning", 1);
}
}
// Warchief Devoted Bodyguards
if(iItemBaseType == BASE_ITEM_ARMOR && GetLevelByClass(CLASS_TYPE_WARCHIEF, oSpellOrigin) > 7)
{
// Warchief must make a reflex save
if(!GetLocalInt(oSpellOrigin, "WarChiefDelay"))
{
// Done this way for formatting reasons in game
if(ReflexSave(oSpellOrigin, 15))
DelayCommand(0.01, ExecuteScript("prc_wchf_bodygrd", oSpellOrigin));
// He can only do this once a round, so put a limit on it
SetLocalInt(oSpellOrigin, "WarChiefDelay", TRUE);
DelayCommand(6.0, DeleteLocalInt(oSpellOrigin, "WarChiefDelay"));
}
}
// Foe Hunter Damage Resistance
if(iItemBaseType == BASE_ITEM_ARMOR && nFoeHunter > 1)
{
DelayCommand(0.01, ExecuteScript("prc_fh_dr",oSpellOrigin) );
}
// Foe Hunter Rancor Attack
if(iItemBaseType != BASE_ITEM_ARMOR && nFoeHunter > 0)
{
if( GetLocalInt(oSpellOrigin, "PRC_CanUseRancor") != 2
&& GetLocalInt(oSpellOrigin, "HatedFoe") == MyPRCGetRacialType(oSpellTarget) )
{
int iFHLevel = GetLevelByClass(CLASS_TYPE_FOE_HUNTER, oSpellOrigin);
int iRancorDice = FloatToInt( (( iFHLevel + 1.0 ) /2) );
int iDamage = d6(iRancorDice);
int iDamType = GetWeaponDamageType(oItem);
int iDamPower = GetDamagePowerConstant(oItem, oSpellTarget, oSpellOrigin);
effect eDam = EffectDamage(iDamage, iDamType, iDamPower);
string sMess = "*Rancor Attack*";
FloatingTextStringOnCreature(sMess, oSpellOrigin, FALSE);
ApplyEffectToObject(DURATION_TYPE_INSTANT, eDam, oSpellTarget);
// Deactivates Ability
SetLocalInt(oSpellOrigin, "PRC_CanUseRancor", 2);
// Prevents the heartbeat script from running multiple times
if(GetLocalInt(oSpellOrigin, "PRC_RancorVarRunning") != 1)
{
DelayCommand(6.0, SetRancorVar(oSpellOrigin) );
SetLocalInt(oSpellOrigin, "PRC_RancorVarRunning", 1);
}
}
}
//Creatures with a necrotic cyst take +1d6 damage from natural attacks of undead
if(GetHasNecroticCyst(oSpellOrigin) && iItemBaseType == BASE_ITEM_ARMOR)
{
//if enemy is undead
if(MyPRCGetRacialType(oSpellTarget) == RACIAL_TYPE_UNDEAD)
{
//and unarmed
if (!GetIsObjectValid(GetItemInSlot(INVENTORY_SLOT_LEFTHAND, oSpellTarget)) &&
!GetIsObjectValid(GetItemInSlot(INVENTORY_SLOT_RIGHTHAND, oSpellTarget)))
{
effect eDam = EffectDamage(d6(1), DAMAGE_TYPE_MAGICAL);
SPApplyEffectToObject(DURATION_TYPE_INSTANT, eDam, oSpellOrigin);
}
}
}
/*//////////////////////////////////////////////////
//////////////// PSIONICS //////////////////////////
//////////////////////////////////////////////////*/
// SweepingStrike OnHit
if(iItemBaseType != BASE_ITEM_ARMOR && GetLocalInt(oItem, "SweepingStrike"))
{
SweepingStrike(oSpellOrigin, oSpellTarget);
}
// Mind Stab OnHit
if(iItemBaseType != BASE_ITEM_ARMOR && GetLocalInt(oSpellOrigin, "ShadowMindStab") && !GetLocalInt(oSpellOrigin, "MindStabDelay"))
{
//FloatingTextStringOnCreature("MindStab Stage 1", oSpellOrigin, FALSE);
// Only works when able to sneak attack a target
if (GetCanSneakAttack(oSpellTarget, oSpellOrigin))
{
//FloatingTextStringOnCreature("MindStab Stage 2", oSpellOrigin, FALSE);
MindStab(oSpellOrigin, oSpellTarget);
// Only once per round
SetLocalInt(oSpellOrigin, "MindStabDelay", TRUE);
DelayCommand(6.0, DeleteLocalInt(oSpellOrigin, "MindStabDelay"));
}
}
// Astral Construct's Poison Touch special ability
if(GetLocalInt(oSpellOrigin, ASTRAL_CONSTRUCT_POISON_TOUCH))
{
ExecuteScript("psi_ast_con_ptch", oSpellOrigin);
}
// Pyrokineticist bonus damage
int nPyroLevel = GetLevelByClass(CLASS_TYPE_PYROKINETICIST, oSpellOrigin);
if(nPyroLevel)
{
int iElement = GetPersistantLocalInt(oSpellOrigin, "PyroDamageType");
int nDam1 = 0;
int nDam2 = 0;
int nDam = 0;
int nDice = 0;
if(iItemBaseType == BASE_ITEM_ARMOR)
{ //Nimbus Onhit
if(nPyroLevel >= 5 && GetHasSpellEffect(SPELL_NIMBUS, oSpellOrigin))
{
nDice = 2;
nDam = (iElement == DAMAGE_TYPE_SONIC) ? d4(nDice) : d6(nDice); //reduced damage dice
if((iElement == DAMAGE_TYPE_COLD) || (iElement == DAMAGE_TYPE_ELECTRICAL) || (iElement == DAMAGE_TYPE_ACID))
nDam = PRCMax(nDice, nDam - nDice); //minimum of 1 per die
}
}
else
{ //Assume unarmed/weapon damage
if(nPyroLevel >= 2)
{
nDice = (GetLevelByClass(CLASS_TYPE_PYROKINETICIST, oSpellOrigin) >= 8) ? 4 : 2;
nDam1 = (iElement == DAMAGE_TYPE_SONIC) ? d4(nDice) : d6(nDice); //reduced damage dice
if((iElement == DAMAGE_TYPE_COLD) || (iElement == DAMAGE_TYPE_ELECTRICAL) || (iElement == DAMAGE_TYPE_ACID))
nDam1 = PRCMax(nDice, nDam1 - nDice); //minimum of 1 per die
}
if(GetTag(oItem) == "PRC_PYRO_LASH_WHIP")
{ //Extra damage from whip
nDam2 += (iElement == DAMAGE_TYPE_SONIC) ? d6(1) : d8(1); //reduced damage dice
if((iElement == DAMAGE_TYPE_COLD) || (iElement == DAMAGE_TYPE_ELECTRICAL) || (iElement == DAMAGE_TYPE_ACID))
nDam2 = PRCMax(1, nDam2 - 1); //minimum of 1
}
nDam = nDam1 + nDam2;
}
if(nDam > 0)
{
SPApplyEffectToObject(DURATION_TYPE_INSTANT, EffectDamage(nDam, iElement), oSpellTarget);
SPApplyEffectToObject(DURATION_TYPE_INSTANT, EffectVisualEffect(GetPersistantLocalInt(oSpellOrigin, "PyroImpactVFX")), oSpellTarget);
}
}
/*//////////////////////////////////////////////////
//////////////// END PSIONICS //////////////////////
//////////////////////////////////////////////////*/
/*//////////////////////////////////////////////////
//////////////// Blade Magic ///////////////////////
//////////////////////////////////////////////////*/
// Martial Spirit
if(GetHasSpellEffect(MOVE_DS_MARTIAL_SPIRIT, oSpellOrigin) && GetBaseItemType(oItem) != BASE_ITEM_ARMOR && GetObjectType(oSpellTarget) == OBJECT_TYPE_CREATURE && !GetIsDead(oSpellTarget))
{
object oHealTarget = GetCrusaderHealTarget(oSpellOrigin, 30.0);
ApplyEffectToObject(DURATION_TYPE_INSTANT, EffectHeal(2), oHealTarget);
ApplyEffectToObject(DURATION_TYPE_INSTANT, EffectVisualEffect(VFX_IMP_HEALING_L_LAW), oHealTarget);
}
// Blood in the Water
if(GetHasSpellEffect(MOVE_TC_BLOOD_WATER, oSpellOrigin) && GetBaseItemType(oItem) != BASE_ITEM_ARMOR && !GetIsImmune(oSpellTarget, IMMUNITY_TYPE_CRITICAL_HIT))
{
// Fake critical hit check
if(d20() >= GetWeaponCriticalRange(oSpellOrigin, oItem))
{
effect eBuff = EffectLinkEffects(EffectAttackIncrease(1), EffectDamageIncrease(DAMAGE_BONUS_1, DAMAGE_TYPE_SLASHING));
DelayCommand(0.0, ApplyEffectToObject(DURATION_TYPE_TEMPORARY, eBuff, oSpellOrigin, TurnsToSeconds(1)));
ApplyEffectToObject(DURATION_TYPE_INSTANT, EffectVisualEffect(VFX_COM_BLOOD_CRT_YELLOW_HEAD), oSpellOrigin);
}
}
// Fire Riposte
if(GetHasSpellEffect(MOVE_DW_FIRE_RIPOSTE, oSpellOrigin) && GetBaseItemType(oItem) == BASE_ITEM_ARMOR)
{
int nTouchAttack = PRCDoMeleeTouchAttack(oSpellTarget);
if(nTouchAttack > 0)
{
// Apply the damage and VFX
ApplyTouchAttackDamage(oSpellOrigin, oSpellTarget, nTouchAttack, d6(4), DAMAGE_TYPE_FIRE);
effect eVis = EffectVisualEffect(VFX_COM_HIT_FIRE);
SPApplyEffectToObject(DURATION_TYPE_INSTANT, eVis, oSpellTarget);
FloatingTextStringOnCreature("Fire Riposte Hit", oSpellOrigin, FALSE);
// Clean up
PRCRemoveSpellEffects(MOVE_DW_FIRE_RIPOSTE, oSpellOrigin, oSpellOrigin);
}
}
// Holocaust Cloak
if(GetHasSpellEffect(MOVE_DW_HOLOCAUST_CLOAK, oSpellOrigin) && GetBaseItemType(oItem) == BASE_ITEM_ARMOR)
{
if(GetIsInMeleeRange(oSpellOrigin, oSpellTarget))
{
// Apply the damage and VFX
effect eVis = EffectVisualEffect(VFX_COM_HIT_FIRE);
effect eDam = EffectDamage(5, DAMAGE_TYPE_FIRE);
ApplyEffectToObject(DURATION_TYPE_INSTANT, eDam, oSpellTarget);
SPApplyEffectToObject(DURATION_TYPE_INSTANT, eVis, oSpellTarget);
FloatingTextStringOnCreature("Holocaust Cloak Hit", oSpellOrigin, FALSE);
}
}
// Defensive Rebuke
if(GetHasSpellEffect(MOVE_DS_DEFENSIVE_REBUKE, oSpellOrigin) && GetBaseItemType(oItem) != BASE_ITEM_ARMOR)
{
SetLocalObject(oSpellTarget, "DefensiveRebuke", oSpellOrigin);
DelayCommand(3.0, ExecuteScript("tob_dvsp_defrbka", oSpellTarget));
}
// Defensive Rebuke
if(GetHasSpellEffect(MOVE_DM_PEARL_BLACK_DOUBT, oSpellOrigin) && GetBaseItemType(oItem) == BASE_ITEM_ARMOR)
{
// Will reset to 0.
DeleteLocalInt(oSpellOrigin, "PearlOfBlackDoubtBonus");
}
// Tactics of the Wolf
if(GetHasSpellEffect(MOVE_WR_TACTICS_WOLF, oSpellOrigin) && GetBaseItemType(oItem) != BASE_ITEM_ARMOR)
{
if (GetIsFlanked(oSpellTarget, oSpellOrigin))
{
int nWolfDam = GetLocalInt(oSpellOrigin, "TacticsWolf");
ApplyEffectToObject(DURATION_TYPE_INSTANT, EffectDamage(nWolfDam), oSpellTarget);
}
}
// Aura of Triumph
if(GetHasSpellEffect(MOVE_DS_AURA_TRIUMPH, oSpellOrigin) && GetBaseItemType(oItem) != BASE_ITEM_ARMOR)
{
// Heal both
object oHealTarget = GetLocalObject(oSpellOrigin, "DSTriumph");
// Must be within 10 feet
if (10.0 >= FeetToMeters(GetDistanceBetween(oSpellOrigin, oSpellOrigin)))
{
SPApplyEffectToObject(DURATION_TYPE_INSTANT, EffectHeal(4), oHealTarget);
SPApplyEffectToObject(DURATION_TYPE_INSTANT, EffectVisualEffect(VFX_IMP_HEALING_L_LAW), oHealTarget);
}
SPApplyEffectToObject(DURATION_TYPE_INSTANT, EffectHeal(4), oSpellOrigin);
SPApplyEffectToObject(DURATION_TYPE_INSTANT, EffectVisualEffect(VFX_IMP_HEALING_L_LAW), oSpellOrigin);
}
// Immortal Fortitude
if(GetHasSpellEffect(MOVE_DS_IMMORTAL_FORTITUDE, oSpellOrigin) && GetBaseItemType(oItem) == BASE_ITEM_ARMOR)
{
// He's immortal, so now we run a script to see if he should/would have died (Saving, etc)
if(GetCurrentHitPoints(oSpellOrigin) == 1)
{
DelayCommand(0.01, ExecuteScript("tob_dvsp_imfrtoh", oSpellOrigin));
}
}
// Immortal Fortitude
if(GetHasSpellEffect(MOVE_WR_CLARION_CALL, oSpellOrigin) && GetBaseItemType(oItem) != BASE_ITEM_ARMOR)
{
// Check to see if the target is dead, triggers each time
if(GetIsDead(oSpellTarget))
{
location lTarget = GetLocation(oSpellOrigin);
object oAreaTarget = MyFirstObjectInShape(SHAPE_SPHERE, FeetToMeters(60.0), lTarget, TRUE, OBJECT_TYPE_CREATURE);
while(GetIsObjectValid(oAreaTarget))
{
if(GetIsFriend(oAreaTarget, oSpellOrigin))
{
// Apply extra attack for one round
ApplyEffectToObject(DURATION_TYPE_TEMPORARY, ExtraordinaryEffect(EffectModifyAttacks(1)), oAreaTarget, 6.0);
}
oAreaTarget = MyNextObjectInShape(SHAPE_SPHERE, FeetToMeters(60.0), lTarget, TRUE, OBJECT_TYPE_CREATURE);
}
}
}
/*//////////////////////////////////////////////////
//////////////// End Blade Magic ///////////////////
//////////////////////////////////////////////////*/
if(iItemBaseType != BASE_ITEM_ARMOR && GetLocalInt(oSpellOrigin,"doarcstrike"))
{
int nDice = GetLocalInt(oSpellOrigin,"curentspell");
int nDamage = d4(nDice);
effect eDam = EffectDamage(nDamage);
ApplyEffectToObject(DURATION_TYPE_INSTANT,eDam,oSpellTarget);
}
//spellsword & arcane archer
//will also fire for other OnHit:UniquePower items that have a SpellSequencer property (such as Duskblade)
if(GetLocalInt(oItem, "X2_L_NUMCHANNELTRIGGERS"))
{
if (DEBUG) DoDebug("prc_onhitcast: Triggering Sequencer Discharge");
SetLocalObject(oSpellOrigin, "ChannelSpellTarget", oSpellTarget);
ExecuteScript("x2_s3_sequencer", oSpellOrigin);
}
// Handle Rend. Creature weapon damage + 1.5x STR bonus.
// Only happens when attacking with a creature weapon
if(GetIsCreatureWeaponType(iItemBaseType) && (GetHasFeat(FEAT_REND, oSpellOrigin) || GetLevelByClass(CLASS_TYPE_BLACK_BLOOD_CULTIST, oSpellOrigin) >= 6))
{
DoRend(oSpellTarget, oSpellOrigin, oItem);
}
// Handle Frostrager Rend. 2d8 damage + 1.5x STR bonus.
// Only happens when attacking with a creature weapon
if(GetIsCreatureWeaponType(iItemBaseType) && (GetLevelByClass(CLASS_TYPE_FROSTRAGER, oSpellOrigin) > 4))
{
DoFrostRend(oSpellTarget, oSpellOrigin, oItem);
}
// Handle Spine Rend. 2d6 damage + 1.5x STR bonus.
// Only happens when attacking with a skarn spine
if(GetTag(oItem) == "skarn_spine" && (GetLevelByClass(CLASS_TYPE_SPINEMELD_WARRIOR, oSpellOrigin) >= 5))
{
DoSpineRend(oSpellTarget, oSpellOrigin, oItem);
}
// now cycle through all onhitcast spells on the item
// we must exclude unique power (which is associated with prc_onhitcast), because otherwise we would get infinite recursions
// it is of utmost importance to devise a *safe* way to cycle through all onhitcast spells on the item. The safe way is provided
// by the function ApplyAllOnHitCastSpellsOnItemExcludingSubType defined in prc_inc_spells
// There are two ways to call this function: Either with all necessary parameters passed explicitly to the function
// or with no parameters passed to the function (in this case default values are used, which also works, at least in prc_onhitcast)
// VERSION 1:
// generally it is more efficient to call ApplyAllOnHitCastSpellsOnItemExcludingSubType by explicitly passing the parameters to the function
// this will set up the overrides in the PRC-wrappers for the spell information functions, which generally is much faster
// ApplyAllOnHitCastSpellsOnItemExcludingSubType(IP_CONST_ONHIT_CASTSPELL_ONHIT_UNIQUEPOWER, oSpellTarget, oItem, oSpellOrigin);
// VERSION 2:
// motu99: It might be safer to call this only with defaults in order disallow overrides being set.
// (they could have been set beforehand, though - in fact they *are* if we were called from prc_inc_combat)
// VERSION 2 has also been tested to work; however, if Bioware changes its implementation the code below is more likely to break
ApplyAllOnHitCastSpellsOnItemExcludingSubType(IP_CONST_ONHIT_CASTSPELL_ONHIT_UNIQUEPOWER);
// ONLY FOR TESTING
// if(GetPRCSwitch(PRC_TIMESTOP_LOCAL)) CastSpellAtLocation(SPELL_FIREBALL, GetLocation(oSpellTarget) /*, METAMAGIC_ANY, GetLevelByTypeArcane(oSpellOrigin), CLASS_TYPE_WIZARD */);
// if(GetPRCSwitch(PRC_PNP_TRUESEEING)) CastSpellAtObject(SPELL_FIREBALL, oSpellTarget /*, METAMAGIC_ANY, GetLevelByTypeArcane(oSpellOrigin), CLASS_TYPE_WIZARD */);
/*
// motu99: This is the old (unsafe) way to cycle through the onhitcast spells.
// This method fails, whenever one of the called impact spell script cycles through the item properties of the SpellCastItem on its own
// (or calls a function that cycles through the item properties - such as PRCGetMetaMagicFeat)
// The safe way to do things is to use the functions ApplyAllOnHitCastSpells* to be found in prc_inc_spells
// Left the piece of code here as an example and a warning, how perfectly reasonable code can break without the fault of the scripter
// Such an "error" can easily happen to any of us. It is quite difficult to spot an error, caused by nested loops
// over item properties in *different* scripts. Runtime behavior is erratic. If you are lucky, you get an
// infinite recursion (then you will notice that something is wrong). If you are not lucky, the loop will just skip over
// some item properties. And this simply because you put a completely harmless looking function like PRCGetMetaMagicFeat
// into your Spell script. How could you possibly know that you just broke your script, because of an unsafe implementation
// in a *different* script (here: prc_onhitcast)? You might not even know, that this different scripts exists.
//handle other OnHit:CastSpell properties
DoDebug("prc_onhitcast: now doing other OnHitCastSpell properties on item = "+GetName(oItem));
itemproperty ipTest = GetFirstItemProperty(oItem);
while(GetIsItemPropertyValid(ipTest))
{
if(GetItemPropertyType(ipTest) == ITEM_PROPERTY_ONHITCASTSPELL)
{
int nIPSpell = GetItemPropertySubType(ipTest);
if(nIPSpell == 125)
{
ipTest = GetNextItemProperty(oItem);
continue; //abort if its OnHit:CastSpell:UniquePower otherwise it would TMI.
}
int nSpell = StringToInt(Get2DACache("iprp_onhitspell", "SpellIndex", nIPSpell));
// int nLevel = GetItemPropertyCostTableValue(ipTest);
string sScript = Get2DACache("spells", "ImpactScript", nSpell);
DoDebug("prc_onhitcast: Now executing Impact spell script "+sScript);
// motu99: Never execute complicated scripts within an GetFirst* / GetNext* loop !!!
// The code will break, whenever the script does a loop over item properties (or effects) on its own
// rather store all found scripts in a local array, and execute the scripts in a separate loop
ExecuteScript(sScript,oSpellOrigin);
}
ipTest = GetNextItemProperty(oItem);
}
*/
/*//////////////////////////////////////////////////
/////////////////// SPELLFIRE ////////////////////
//////////////////////////////////////////////////*/
int nSpellfire = GetLevelByClass(CLASS_TYPE_SPELLFIRE, oSpellOrigin);
if(nSpellfire && (iItemBaseType == BASE_ITEM_ARMOR))
{
int nStored = GetPersistantLocalInt(oSpellOrigin, "SpellfireLevelStored");
int nCON = GetAbilityScore(oSpellOrigin, ABILITY_CONSTITUTION);
int nFlare = 0;
int bFlare = FALSE;
if(nStored > 4 * nCON)
{
nFlare = d6(2);
bFlare = TRUE;
}
else if(nStored > 3 * nCON)
{
nFlare = d6();
bFlare = TRUE;
}
else if(nStored > 2 * nCON)
nFlare = d4();
else if(nStored > nCON)
nFlare = 1;
if(nFlare)
{
nStored -= nFlare;
if(nStored < 0) nStored = 0;
SetPersistantLocalInt(oSpellOrigin, "SpellfireLevelStored", nStored);
}
if(bFlare)
{
int nDC = 10 + nFlare;
location lTarget = GetLocation(oSpellOrigin);
object oFlareTarget = MyFirstObjectInShape(SHAPE_SPHERE, RADIUS_SIZE_SMALL, lTarget, TRUE, OBJECT_TYPE_CREATURE);
while(GetIsObjectValid(oFlareTarget))
{
if(spellsIsTarget(oFlareTarget, SPELL_TARGET_STANDARDHOSTILE, oSpellOrigin))
{
if(!PRCDoResistSpell(oSpellOrigin, oFlareTarget, nSpellfire))
{
if (PRCMySavingThrow(SAVING_THROW_FORT, oFlareTarget, nDC))
{
if (GetHasMettle(oFlareTarget, SAVING_THROW_FORT))
// This script does nothing if it has Mettle, bail
return;
SPApplyEffectToObject(DURATION_TYPE_INSTANT, EffectVisualEffect(VFX_IMP_FLAME_S), oFlareTarget);
ApplyEffectToObject(DURATION_TYPE_TEMPORARY, EffectDazzle(), oFlareTarget, 60.0);
}
}
}
oFlareTarget = MyNextObjectInShape(SHAPE_SPHERE, RADIUS_SIZE_SMALL, lTarget, TRUE, OBJECT_TYPE_CREATURE);
}
}
if(GetLocalInt(oSpellOrigin, "SpellfireCrown")) //melts non-magical melee weapons
{ //can't really get which weapon hit you, so...
object oWeapon = GetItemInSlot(INVENTORY_SLOT_RIGHTHAND, oSpellTarget);
if(GetIsObjectValid(oWeapon))
{
if(IPGetIsMeleeWeapon(oWeapon) && !GetIsMagicItem(oWeapon))
{
DestroyObject(oWeapon);
FloatingTextStringOnCreature("*Your weapon has melted!*", oSpellTarget);
}
}
else
{
oWeapon = GetItemInSlot(INVENTORY_SLOT_LEFTHAND, oSpellTarget);
if(GetIsObjectValid(oWeapon))
{
if(IPGetIsMeleeWeapon(oWeapon) && !GetIsMagicItem(oWeapon))
{
DestroyObject(oWeapon);
FloatingTextStringOnCreature("*Your weapon has melted!*", oSpellTarget);
}
}
else //You're putting your arms and legs through something that melts weapons?
{ //Silly monk/brawler/fool with molten weapons!
ApplyEffectToObject(DURATION_TYPE_INSTANT, EffectDamage(d20()), oSpellTarget);
ApplyEffectToObject(DURATION_TYPE_INSTANT, EffectDamage(d20(), DAMAGE_TYPE_FIRE), oSpellTarget);
}
}
}
}
/*//////////////////////////////////////////////////
////////////////// END SPELLFIRE ///////////////////
//////////////////////////////////////////////////*/
// Handle poisoned weapons
/*
if(GetLocalInt(oItem, "pois_wpn_uses"))
{
ExecuteScript("poison_wpn_onhit", OBJECT_SELF);
}
*/
// Execute scripts hooked to this event for the player triggering it
if (DEBUG) DoDebug("prc_onhitcast: executing all scripts hooked to onhit events of attacker and item");
ExecuteAllScriptsHookedToEvent(oSpellOrigin, EVENT_ONHIT);
ExecuteAllScriptsHookedToEvent(oItem, EVENT_ITEM_ONHIT);
DeleteLocalInt(oSpellOrigin, "prc_ohc");
}
void SetRancorVar(object oPC)
{
// Turn Rancor on
SetLocalInt(oPC, "PRC_CanUseRancor", 1);
//FloatingTextStringOnCreature("Rancor Attack Possible", oPC, FALSE);
int iMain = GetMainHandAttacks(oPC);
float fDelay = 6.0 / IntToFloat(iMain);
// Turn Rancor off after one attack is made
DelayCommand(fDelay, SetLocalInt(oPC, "PRC_CanUseRancor", 2));
//DelayCommand((fDelay + 0.01), FloatingTextStringOnCreature("Rancor Attack Not Possible", oPC, FALSE));
// Call again if the character is still in combat.
// this allows the ability to keep running even if the
// player does not score a rancor hit during the allotted time
if( PRCGetIsFighting() )
{
DelayCommand(6.0, SetRancorVar(oPC) );
}
else
{
DelayCommand(2.0, SetLocalInt(oPC, "PRC_CanUseRancor", 1));
DelayCommand(2.1, SetLocalInt(oPC, "PRC_RancorVarRunning", 2));
//DelayCommand(2.2, FloatingTextStringOnCreature("Rancor Enabled After Combat", oPC, FALSE));
}
}
void DoImprovedRicochet(object oPC, object oTarget)
{
int nTargetsLeft = 1;
effect eVis = EffectVisualEffect(VFX_IMP_DUST_EXPLOSION);
location lTarget = GetLocation(oTarget);
//Declare the spell shape, size and the location. Capture the first target object in the shape.
object oAreaTarget = MyFirstObjectInShape(SHAPE_SPHERE, RADIUS_SIZE_LARGE, lTarget, TRUE, OBJECT_TYPE_CREATURE);
//Cycle through the targets within the spell shape until you run out of targets.
while (GetIsObjectValid(oAreaTarget) && nTargetsLeft > 0)
{
if (spellsIsTarget(oAreaTarget, SPELL_TARGET_SELECTIVEHOSTILE, OBJECT_SELF) && oAreaTarget != OBJECT_SELF && oAreaTarget != oTarget)
{
PerformAttack(oAreaTarget, oPC, eVis, 0.0, -2, 0, 0, "*Improved Ricochet Hit*", "*Improved Ricochet Missed*");
// Use up a target slot only if we actually did something to it
nTargetsLeft -= 1;
}
//Select the next target within the spell shape.
oAreaTarget = MyNextObjectInShape(SHAPE_SPHERE, RADIUS_SIZE_LARGE, lTarget, TRUE, OBJECT_TYPE_CREATURE);
}
}
void SetImprovedRicochetVar(object oPC)
{
// Turn Retort on
SetLocalInt(oPC, "CanRicochet", 1);
// Turn Retort off after one attack is made
DelayCommand(0.01, SetLocalInt(oPC, "CanRicochet", 0));
// Call again if the character is still in combat.
// this allows the ability to keep running even if the
// player does not score a retort hit during the allotted time
if( PRCGetIsFighting() )
{
DelayCommand(6.0, SetImprovedRicochetVar(oPC));
}
else
{
DelayCommand(2.0, SetLocalInt(oPC, "CanRicochet", 1));
DelayCommand(2.1, SetLocalInt(oPC, "ImpRicochetVarRunning", 2));
}
}