Added Force Missiles spell. Fixed Shadowcaster Marker feat check. Updated Witchwwod step to grant immunity to Vine Mine & Spike Growth. Updated release archive.
		
			
				
	
	
		
			194 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			194 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| //::///////////////////////////////////////////////
 | |
| //:: Force Missiles
 | |
| //:: sp_forcemissiles
 | |
| //:: Copyright (c) 2022 PRC
 | |
| //:://////////////////////////////////////////////
 | |
| /*/
 | |
| Force Missiles
 | |
| (Spell Compendium, p. 98)
 | |
| 
 | |
| Evocation [Force]
 | |
| Level: Sorcerer 4, Wizard 4,
 | |
| Components: V, S,
 | |
| Casting Time: 1 Standard Action
 | |
| Range: Medium (100 ft. + 10 ft./level)
 | |
| Target: Up to four creatures, no two of which are more than 30 ft. apart
 | |
| Duration: Instantaneous
 | |
| Saving Throw: None
 | |
| Spell Resistance: Yes
 | |
| 
 | |
| Sparking bolts of blue magic, like giant magic missiles, streak from your
 | |
| outstretched hand to strike your foes and explode in sparkling bursts.
 | |
| 
 | |
| You create powerful missiles of magical force, each of which darts from your
 | |
| fingertips and unerringly strikes its target, dealing 2d6 points of damage.
 | |
| The missile then explodes in a burst of force that deals half this amount of
 | |
| damage to any creatures adjacent to the primary target.
 | |
| 
 | |
| The missile strikes unerringly, even if the target is in melee or has anything
 | |
| less than total cover or concealment. A caster cannot single out specific parts
 | |
| of a creature.
 | |
| 
 | |
| You gain one missile for every four caster levels. You can make more than one
 | |
| missile strike a single target, if desired. However,you must designate targets
 | |
| before rolling for spell resistance or damage.
 | |
| 
 | |
| /*/
 | |
| //:://////////////////////////////////////////////
 | |
| //:: Created By: Tsurani Nevericy
 | |
| //:: Created On: 05/15/2024
 | |
| //:://////////////////////////////////////////////
 | |
| //:: Last Updated By: Tsurani Nevericy
 | |
| //:: Last Updated On: 05/15/2024
 | |
| //:://////////////////////////////////////////////
 | |
| 
 | |
| #include "prc_sp_func"
 | |
| #include "prc_inc_spells"
 | |
| #include "x2_inc_spellhook"
 | |
| 
 | |
| void SendMissileBomb(object oCaster, object oTarget, float fDelay=0.0, float fTime=0.0)
 | |
| {
 | |
|     int nMetaMagic = PRCGetMetaMagicFeat();
 | |
|     ApplyEffectToObject(DURATION_TYPE_INSTANT, EffectVisualEffect(VFX_IMP_MIRV), oTarget);
 | |
|     location lLoc = GetLocation(oTarget);
 | |
|     object oLoop = GetFirstObjectInShape(SHAPE_SPHERE, 5.0, lLoc, TRUE);
 | |
|     while (GetIsObjectValid(oLoop))
 | |
|     {
 | |
|         SignalEvent(oLoop, EventSpellCastAt(oCaster, PRCGetSpellId()));
 | |
|         if (oLoop == oTarget)
 | |
|         {
 | |
|             int nDam = d6(2);
 | |
|             if (nMetaMagic == METAMAGIC_MAXIMIZE)
 | |
|                   nDam = 12;
 | |
|             if (nMetaMagic == METAMAGIC_EMPOWER)
 | |
|                   nDam += nDam/2;
 | |
|             DelayCommand(fTime, ApplyEffectToObject(DURATION_TYPE_INSTANT, EffectDamage(nDam, DAMAGE_TYPE_MAGICAL), oLoop));
 | |
|             DelayCommand(fTime, ApplyEffectToObject(DURATION_TYPE_TEMPORARY, EffectVisualEffect(VFX_IMP_MAGBLUE, FALSE, 4.0f), oLoop));
 | |
|         }
 | |
|         else if (!PRCDoResistSpell(oCaster, oLoop, FloatToInt(fDelay)))
 | |
|         {
 | |
|             int nDam = d6(1);
 | |
|             if (nMetaMagic == METAMAGIC_MAXIMIZE)
 | |
|                   nDam = 6;
 | |
|             if (nMetaMagic == METAMAGIC_EMPOWER)
 | |
|                   nDam += nDam/2;
 | |
|             DelayCommand(fTime, ApplyEffectToObject(DURATION_TYPE_INSTANT, EffectDamage(nDam, DAMAGE_TYPE_MAGICAL), oLoop));
 | |
|             DelayCommand(fTime, ApplyEffectToObject(DURATION_TYPE_TEMPORARY, EffectVisualEffect(VFX_IMP_MAGBLUE), oLoop));
 | |
|         }
 | |
|         oLoop = GetNextObjectInShape(SHAPE_SPHERE, 5.0, lLoc, TRUE);
 | |
|     }
 | |
| }
 | |
| 
 | |
| //Implements the spell impact, put code here
 | |
| //  if called in many places, return TRUE if
 | |
| //  stored charges should be decreased
 | |
| //  eg. touch attack hits
 | |
| //
 | |
| //  Variables passed may be changed if necessary
 | |
| int DoSpell(object oCaster, object oTarget, int nCasterLevel, int nEvent)
 | |
| {
 | |
|     int nMetaMagic = PRCGetMetaMagicFeat();
 | |
|     int nSaveDC = PRCGetSaveDC(oTarget, oCaster);
 | |
|     int nPenetr = nCasterLevel + SPGetPenetr();
 | |
| 	int i;
 | |
|     int nTargets;
 | |
|     int nCnt = 1;
 | |
| 	float fDist, fDelay, fDelay2, fTime;
 | |
| 
 | |
| 	if (nCasterLevel > 40) nCasterLevel = 40;
 | |
|     int nMissiles = nCasterLevel/4;
 | |
|     if (nMissiles < 1) nMissiles = 1;	
 | |
| 	
 | |
| 	location lTarget = PRCGetSpellTargetLocation();
 | |
| 
 | |
|     oTarget = GetFirstObjectInShape(SHAPE_SPHERE, 9.144, lTarget, TRUE, OBJECT_TYPE_CREATURE);
 | |
|     while (GetIsObjectValid(oTarget))
 | |
|     {
 | |
|         if (spellsIsTarget(oTarget, SPELL_TARGET_SELECTIVEHOSTILE, oCaster) && oTarget != oCaster)
 | |
|         {
 | |
|             nTargets++;
 | |
|         }
 | |
|         oTarget = GetNextObjectInShape(SHAPE_SPHERE, 9.144, lTarget, TRUE, OBJECT_TYPE_CREATURE);
 | |
|     }
 | |
| 	if (!nTargets)
 | |
| 		return FALSE;
 | |
| 
 | |
|     int nExtraMissiles = nMissiles / nTargets;
 | |
| 
 | |
|     if (nExtraMissiles <= 0)
 | |
|         nExtraMissiles = 1;
 | |
| 
 | |
|     int nRemainder = 0;
 | |
| 
 | |
|     if (nTargets > nMissiles) nTargets = nMissiles;
 | |
| 
 | |
|     nRemainder = nMissiles % nTargets;
 | |
| 
 | |
|     oTarget = GetFirstObjectInShape(SHAPE_SPHERE, 9.144, lTarget, TRUE, OBJECT_TYPE_CREATURE);
 | |
|     while (GetIsObjectValid(oTarget) && nCnt <= nTargets)
 | |
|     {
 | |
|         if (spellsIsTarget(oTarget, SPELL_TARGET_SELECTIVEHOSTILE, oCaster) && oTarget != oCaster)
 | |
|         {
 | |
|             if (!PRCDoResistSpell(oCaster, oTarget, FloatToInt(fDelay)))
 | |
|             {
 | |
|                 int i;
 | |
|                 for (i=1; i <= nExtraMissiles + nRemainder; i++)
 | |
|                 {
 | |
|                     fDist = GetDistanceBetween(oCaster, oTarget);
 | |
|                     fDelay = fDist/(3.0 * log(fDist) + 2.0);
 | |
|                     fTime = fDelay;
 | |
|                     fDelay2 += 0.1;
 | |
|                     fTime += fDelay2;
 | |
|                     DelayCommand(fDelay2, SendMissileBomb(oCaster, oTarget, fDelay, fTime));
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 ApplyEffectToObject(DURATION_TYPE_INSTANT, EffectVisualEffect(VFX_IMP_MIRV), oTarget);
 | |
|             }
 | |
|             nCnt++;
 | |
|             nRemainder = 0;
 | |
|         }
 | |
|         oTarget = GetNextObjectInShape(SHAPE_SPHERE, 9.144, lTarget, TRUE, OBJECT_TYPE_CREATURE);
 | |
|     }
 | |
| 	
 | |
| 	return TRUE;
 | |
| }
 | |
| 
 | |
| void main()
 | |
| {
 | |
|     object oCaster = OBJECT_SELF;
 | |
| 	object oTarget = PRCGetSpellTargetObject();
 | |
| 	
 | |
|     int nCasterLevel = PRCGetCasterLevel(oCaster);
 | |
| 	int i;
 | |
|     int nTargets;
 | |
|     int nCnt = 1;
 | |
| 	float fDist, fDelay, fDelay2, fTime;
 | |
| 	
 | |
|     PRCSetSchool(GetSpellSchool(PRCGetSpellId()));
 | |
| 	
 | |
|     if (!X2PreSpellCastCode()) return;
 | |
| 	
 | |
|     int nEvent = GetLocalInt(oCaster, PRC_SPELL_EVENT); //use bitwise & to extract flags
 | |
|     if(!nEvent) //normal cast
 | |
|     {
 | |
|         if(GetLocalInt(oCaster, PRC_SPELL_HOLD) && oCaster == oTarget)
 | |
|         {   //holding the charge, casting spell on self
 | |
|             SetLocalSpellVariables(oCaster, 1);   //change 1 to number of charges
 | |
|             return;
 | |
|         }
 | |
|         DoSpell(oCaster, oTarget, nCasterLevel, nEvent);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         if(nEvent & PRC_SPELL_EVENT_ATTACK)
 | |
|         {
 | |
|             if(DoSpell(oCaster, oTarget, nCasterLevel, nEvent))
 | |
|                 DecrementSpellCharges(oCaster);
 | |
|         }
 | |
|     }
 | |
|     PRCSetSchool();
 | |
| }
 | |
| 
 | |
| //:://///////////////////////////////////////////////////////////////// |