PRC8/nwn/nwnprc/trunk/smp/xxx_s_forcemissi.nss
Jaysyn904 6ec137a24e Updated AMS marker feats
Updated AMS marker feats.  Removed arcane & divine marker feats.  Updated Dread Necromancer for epic progression. Updated weapon baseitem models.  Updated new weapons for crafting & npc equip.
 Updated prefix.  Updated release archive.
2024-02-11 14:01:05 -05:00

293 lines
11 KiB
Plaintext

/*:://////////////////////////////////////////////
//:: Spell Name Force Missiles
//:: Spell FileName XXX_S_ForceMissi
//:://////////////////////////////////////////////
//:: In Game Spell desctiption
//:://////////////////////////////////////////////
Evocation [Force]
Also known as: Mordenkainen's Force Missiles
Level: Sor/Wiz 4
Components: V, S
Casting Time: 1 standard action
Range: Medium (20M)
Targets: Up to four enemy creatures in a 1.67M-radius sphere (5ft)
Duration: Instantaneous
Saving Throw: None or Reflex half (see text)
Spell Resistance: Yes
Source: Various (WotC Spellbook)
You create a powerful missile of magical force, which darts from your
fingertips and unerringly strikes its target, dealing 2d6 points of damage.
The missile then bursts in a 1.67-meter blast of force that inflicts half
this amount of damage to any creatures in the area (other than the primary
target). The primary target is not entitled to a saving throw against the
burst, but creatures affected by the burst may attempt a Reflex save for
half damage.
If the missiles' burst areas overlap, secondary targets make only one saving
throw attempt (and only one SR check, if applicable). A character can be
struck by one missile (or more) and also be caught in the burst of another
missile. In such a case, the character may attempt a Reflex save to halve
the burst damage, and SR might apply.
The missile strikes unerringly, even if the target is in melee or has
anything less than total cover or concealment.
For every five caster levels, the caster gains one missile. A caster has two
missiles at 9th level or lower, three missiles from 10th to 14th level, and
four missiles at 15th level or higher. If you can fire more then one
missile, and target one person, all missiles fly towards that target. If you
target the ground, the nearest 2 or more enemies will get hit in a 3M
radius sphere.
//:://////////////////////////////////////////////
//:: Spell Effects Applied / Notes
//:://////////////////////////////////////////////
Using the magic missile effect, it'll do burst damage against those in
the area *at the time of casting*, thus it might look a little odd, oh well.
The way the "Only one save and SR check" is done is that a local is set
and checked for each target each blast.
//:://////////////////////////////////////////////
//:: Created By: Jasperre
//::////////////////////////////////////////////*/
#include "SMP_INC_SPELLS"
// Do the AOE blast using oOriginalTarget's location
// * Damages oOriginalTarget fully.
// * Uses the visuals eMissile and eVis.
void DoMissileAndBlast(object oOriginalTarget, object oCaster, int nSpellSaveDC, string sSRLocal, string sSaveLocal, int nMissiles, int nMetaMagic, effect eMissile, effect eVis);
// Reflex adjust nDamage, using stored stuff.
int ReturnReflexDamage(int nDamage, string sSaveLocal, object oTarget, int nDC, object oSaveVersus, float fDelay);
// Check SR, using stored stuff.
// * TRUE is resisted, FALSE isn't.
int ReturnSRCheck(object oTarget, object oCaster, string sSRLocal, float fDelay = 0.0);
void main()
{
// Spell Hook Check.
if(!SMP_SpellHookCheck(SMP_SPELL_FORCE_MISSILES)) return;
//Declare major variables
object oCaster = OBJECT_SELF;
object oTarget = GetSpellTargetObject();
location lTarget = GetSpellTargetLocation();
int nMetaMagic = SMP_GetMetaMagicFeat();
int nCasterLevel = SMP_GetCasterLevel();
int nSpellSaveDC = SMP_GetSpellSaveDC();
int nPeopleDone;
string sSRLocal = IntToString(SMP_SPELL_FORCE_MISSILES) + ObjectToString(oCaster) + "SR";
string sSaveLocal = IntToString(SMP_SPELL_FORCE_MISSILES) + ObjectToString(oCaster) + "SAVE";
// For every five caster levels, the caster gains one missile.
int nMissiles = SMP_LimitInteger(nCasterLevel/5, 4, 1);
// Declare Effects
effect eMissile = EffectVisualEffect(VFX_IMP_MIRV);
effect eVis = EffectVisualEffect(VFX_IMP_MAGBLUE);
// Is it one target?
if(GetIsObjectValid(oTarget) && GetObjectType(oTarget) == OBJECT_TYPE_CREATURE)
{
// Check PvP
if(!GetIsReactionTypeFriendly(oTarget))
{
// Use function
DoMissileAndBlast(oTarget, oCaster, nSpellSaveDC, sSRLocal, sSaveLocal, nMissiles, nMetaMagic, eMissile, eVis);
}
}
else
{
// We loop targets in the LOS of 2M sphere
oTarget = GetFirstObjectInShape(SHAPE_SPHERE, 2.0, lTarget, TRUE, OBJECT_TYPE_CREATURE);
while(GetIsObjectValid(oTarget) && nPeopleDone < nMissiles)
{
// Check if an enemy and is seen
if(GetIsReactionTypeHostile(oTarget) && GetObjectSeen(oTarget))
{
// Use function - 1 missile blast
DoMissileAndBlast(oTarget, oCaster, nSpellSaveDC, sSRLocal, sSaveLocal, 1, nMetaMagic, eMissile, eVis);
// Add one to done list
nPeopleDone++;
}
oTarget = GetNextObjectInShape(SHAPE_SPHERE, 2.0, lTarget, TRUE, OBJECT_TYPE_CREATURE);
}
}
}
// Do the AOE blast using oOriginalTarget's location
// * Damages oOriginalTarget fully.
// * Uses the visuals eMissile and eVis.
void DoMissileAndBlast(object oOriginalTarget, object oCaster, int nSpellSaveDC, string sSRLocal, string sSaveLocal, int nMissiles, int nMetaMagic, effect eMissile, effect eVis)
{
// Get delay for hitting.
float fDist = GetDistanceBetween(oCaster, oOriginalTarget);
float fDelay = fDist/(3.0 * log(fDist) + 2.0);
float fDelay2;
float fAOEDelay;
object oAOETarget;
int nCnt, nDam, nSR, nSave;
location lTarget;
// Apply visual effect (Magic Missiles)
for(nCnt = 1; nCnt <= nMissiles; nCnt++)
{
fDelay2 += 0.1;
DelayCommand(fDelay2, SMP_ApplyVFX(oOriginalTarget, eMissile));
// Signal Spell cast at, once
if(GetLocalInt(oOriginalTarget, sSRLocal) != FALSE)
{
SMP_SignalSpellCastAt(oOriginalTarget, SMP_SPELL_FORCE_MISSILES);
}
// Do damage to oOriginalTarget, if they fail thier SR. No save!
// Check SR
if(!ReturnSRCheck(oOriginalTarget, oCaster, sSRLocal, fDelay))
{
// Do damage
nDam = SMP_MaximizeOrEmpower(6, 2, nMetaMagic);
DelayCommand(fDelay, SMP_ApplyDamageVFXToObject(oOriginalTarget, eVis, nDam));
}
// For each blast, we loop targets in the AOE
// We loop targets in the LOS of 1.67M sphere
lTarget = GetLocation(oOriginalTarget);
oAOETarget = GetFirstObjectInShape(SHAPE_SPHERE, 1.67, lTarget, TRUE, OBJECT_TYPE_CREATURE);
while(GetIsObjectValid(oAOETarget))
{
// PvP Check and is NOT original target
if(!GetIsReactionTypeFriendly(oAOETarget) &&
oAOETarget != oOriginalTarget)
{
// Signal Spell cast at, once
if(GetLocalInt(oAOETarget, sSRLocal) != FALSE)
{
// Signal Spell cast at
SMP_SignalSpellCastAt(oAOETarget, SMP_SPELL_FORCE_MISSILES);
}
// Make sure we check SR as the original target
fAOEDelay = fDelay + GetDistanceBetweenLocations(lTarget, GetLocation(oAOETarget))/20;
// Do damage to oAOETarget, if they fail thier SR. No save!
if(!ReturnSRCheck(oAOETarget, oCaster, sSRLocal, fAOEDelay))
{
// Get damage to do
nDam = SMP_MaximizeOrEmpower(6, 1, nMetaMagic);
// Adjust it due to saves, done already or not.
nDam = ReturnReflexDamage(nDam, sSaveLocal, oAOETarget, nSpellSaveDC, oCaster, fAOEDelay);
// Do the damage, if any
if(nDam > 0)
{
DelayCommand(fDelay, SMP_ApplyDamageVFXToObject(oOriginalTarget, eVis, nDam));
}
}
}
oAOETarget = GetNextObjectInShape(SHAPE_SPHERE, 1.67, lTarget, TRUE, OBJECT_TYPE_CREATURE);
}
}
}
// Reflex adjust nDamage, using stored stuff.
int ReturnReflexDamage(int nDamage, string sSaveLocal, object oTarget, int nDC, object oSaveVersus, float fDelay)
{
// Check if saved already
int nSave = GetLocalInt(oTarget, sSaveLocal);
// 0 = not attempted
// 1 = saved
// 2 = Not saved, failed in the attempt.
if(nSave > 0)
{
if(nSave == 1)
{
// Saved
return SMP_ReflexAdjustDamage(TRUE, nDamage, oTarget);
}
else//if(nSave == 2)
{
// Failed
return SMP_ReflexAdjustDamage(FALSE, nDamage, oTarget);
}
}
else
{
// New save
nSave = SMP_SavingThrow(SAVING_THROW_REFLEX, oTarget, nDC, SAVING_THROW_TYPE_NONE, oSaveVersus, fDelay);
// Saved?
if(nSave == FALSE)
{
// Didn't save. Set and return.
SetLocalInt(oTarget, sSaveLocal, 2);
DelayCommand(0.1, DeleteLocalInt(oTarget, sSaveLocal));
// Failed
return SMP_ReflexAdjustDamage(FALSE, nDamage, oTarget);
}
else// is over 1
{
// Passed
SetLocalInt(oTarget, sSaveLocal, 1);
DelayCommand(0.1, DeleteLocalInt(oTarget, sSaveLocal));
// Passed
return SMP_ReflexAdjustDamage(TRUE, nDamage, oTarget);
}
}
return nDamage;
}
// Check SR, using stored stuff.
// * TRUE is resisted, FALSE isn't.
int ReturnSRCheck(object oTarget, object oCaster, string sSRLocal, float fDelay = 0.0)
{
// Get old result
int nResult = GetLocalInt(oTarget, sSRLocal);
if(nResult != TRUE)// Resisted already
{
// TRUE means we pass, and no damage, else we check if we need to do it.
// 1 Means we have passed, and do not need to do one.
// 2 means we failed the check, no more checks.
// 0 Means an absense of a check
if(nResult == 2)
{
// Failed already.
return FALSE;
}
else //if(nSR == 0)
{
// Make a new SR check - our first one!
nResult = SMP_SpellResistanceCheck(oCaster, oTarget, fDelay);
if(nResult == FALSE)
{
// Failed
// Set we always failed - 2
SetLocalInt(oTarget, sSRLocal, 2);
DelayCommand(0.1, DeleteLocalInt(oTarget, sSRLocal));
return FALSE;
}
else //must have passed somehow
{
// Passed/resisted always. Set always pass - 1
SetLocalInt(oTarget, sSRLocal, 1);
DelayCommand(0.1, DeleteLocalInt(oTarget, sSRLocal));
// Return TRUE, passed
return TRUE;
}
}
}
else
{
// Resisted, it was a 1
return TRUE;
}
// Return DIDN'T resist, as a defualt.
return FALSE;
}