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.
190 lines
7.0 KiB
Plaintext
190 lines
7.0 KiB
Plaintext
//::///////////////////////////////////////////////
|
|
//:: Name Rapture of Rupture
|
|
//:: FileName sp_rapt_rupt.nss
|
|
//:://////////////////////////////////////////////
|
|
/** @file
|
|
Rapture of Rupture
|
|
Transmutation [Evil]
|
|
Level: Corrupt 7
|
|
Components: V, S, Corrupt
|
|
Casting Time: 1 action
|
|
Range: Touch
|
|
Target: One living creature touched per level
|
|
Duration: Instantaneous
|
|
Saving Throw: Fortitude half
|
|
Spell Resistance: Yes
|
|
|
|
With this spell, the caster's touch deals grievous
|
|
wounds to multiple targets. After rapture of rupture
|
|
is cast, the caster can touch one target per round
|
|
until she has touched a number of targets equal to
|
|
her caster level. The same creature cannot be
|
|
affected twice by the same rapture of rupture. A
|
|
creature with no discernible anatomy is unaffected by
|
|
this spell.
|
|
|
|
When the caster touches a subject, his flesh bursts
|
|
open suddenly in multiple places. Each subject takes
|
|
6d6 points of damage and is stunned for 1 round; a
|
|
successful Fortitude save reduces damage by half and
|
|
negates the stun effect. Subjects who fail their
|
|
Fortitude save continue to take 1d6 points of damage
|
|
per round until they receive magical healing, succeed
|
|
at a Heal check (DC 20), or die. If a subject takes 6
|
|
points of damage from rapture of rupture in a single
|
|
round, he is stunned in the following round.
|
|
|
|
Corruption Cost: 1 point of Strength damage per target
|
|
touched.
|
|
|
|
*/
|
|
// Author: Tenjac
|
|
// Created: 5/31/2006
|
|
//:://////////////////////////////////////////////
|
|
//:://////////////////////////////////////////////
|
|
|
|
#include "prc_inc_sp_tch"
|
|
#include "prc_sp_func"
|
|
#include "prc_add_spell_dc"
|
|
|
|
int DoSpell(object oCaster, object oTarget, int nCasterLevel, int nEvent, string sScript);
|
|
void WoundLoop(object oTarget, object oCaster, int nDamage = 0);
|
|
|
|
|
|
void main()
|
|
{
|
|
if(!X2PreSpellCastCode()) return;
|
|
|
|
PRCSetSchool(SPELL_SCHOOL_TRANSMUTATION);
|
|
|
|
object oCaster = OBJECT_SELF;
|
|
int nCasterLevel = PRCGetCasterLevel(oCaster);
|
|
|
|
/*if(GetRunningEvent() == EVENT_VIRTUAL_ONDAMAGED)
|
|
{
|
|
ConcentrationLossCode(oCaster, GetLocalObject(oCaster, PRC_SPELL_CONC_TARGET), nCasterLevel);
|
|
return;
|
|
}*/
|
|
|
|
object oTarget = PRCGetSpellTargetObject();
|
|
int nSpellID = GetSpellId();
|
|
string sScript = Get2DACache("spells", "ImpactScript", nSpellID);
|
|
|
|
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, PRCGetCasterLevel(oCaster));
|
|
|
|
// Setup target tracking set. If one remains from a previous casting, delete it first
|
|
if(set_exists(oCaster, "PRC_Spell_RoRTargets"))
|
|
set_delete(oCaster, "PRC_Spell_RoRTargets");
|
|
set_create(oCaster, "PRC_Spell_RoRTargets");
|
|
|
|
return;
|
|
}
|
|
DoSpell(oCaster, oTarget, nCasterLevel, nEvent, sScript);
|
|
}
|
|
else
|
|
{
|
|
if(nEvent & PRC_SPELL_EVENT_ATTACK)
|
|
{
|
|
if(DoSpell(oCaster, oTarget, nCasterLevel, nEvent, sScript))
|
|
DecrementSpellCharges(oCaster);
|
|
}
|
|
}
|
|
|
|
//SPEvilShift(oCaster);
|
|
|
|
//Corrupt spells get mandatory 10 pt evil adjustment, regardless of switch
|
|
AdjustAlignment(oCaster, ALIGNMENT_EVIL, 10, FALSE);
|
|
|
|
PRCSetSchool();
|
|
}
|
|
|
|
//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, string sScript)
|
|
{
|
|
int nMetaMagic = PRCGetMetaMagicFeat();
|
|
int nSaveDC = PRCGetSaveDC(oTarget, oCaster);
|
|
int nPenetr = nCasterLevel + SPGetPenetr();
|
|
string sCaster = GetLocalString(oCaster, "PRCRuptureID");
|
|
string sTest = GetLocalString(oTarget, "PRCRuptureTargetID");
|
|
|
|
// Roll to hit
|
|
int iAttackRoll = PRCDoMeleeTouchAttack(oTarget);
|
|
if (iAttackRoll > 0)
|
|
{
|
|
// Target validity check - should not have been targeted before
|
|
// and should have discernible anatomy (excludes constructs, elementals, oozes, plants, undead)
|
|
int nRace = MyPRCGetRacialType(oTarget);
|
|
if(!set_contains_object(oCaster, "PRC_Spell_RoRTargets", oTarget) &&
|
|
nRace != RACIAL_TYPE_CONSTRUCT &&
|
|
nRace != RACIAL_TYPE_ELEMENTAL &&
|
|
nRace != RACIAL_TYPE_OOZE &&
|
|
//nRace != RACIAL_TYPE_PLANT &&
|
|
nRace != RACIAL_TYPE_UNDEAD
|
|
)
|
|
{
|
|
if(!PRCDoResistSpell(oCaster, oTarget, nPenetr))
|
|
{
|
|
// Roll damage and apply metamagic
|
|
int nDam = d6(6);
|
|
if(nMetaMagic & METAMAGIC_MAXIMIZE)
|
|
nDam = 36;
|
|
if(nMetaMagic & METAMAGIC_EMPOWER)
|
|
nDam += nDam / 2;
|
|
nDam += SpellDamagePerDice(oCaster, 6);
|
|
// Save for half damage
|
|
if(PRCMySavingThrow(SAVING_THROW_FORT, oTarget, nSaveDC, SAVING_THROW_TYPE_EVIL))
|
|
{
|
|
nDam /= 2;
|
|
|
|
// If the target has Mettle, do no damage. However, the target will still count as having been affected by the spell
|
|
if (GetHasMettle(oTarget, SAVING_THROW_FORT))
|
|
nDam = 0;
|
|
}
|
|
// On failed save, stun for a round and start bleeding
|
|
else
|
|
{
|
|
SPApplyEffectToObject(DURATION_TYPE_TEMPORARY, EffectStunned(), oTarget, 6.0f);
|
|
WoundLoop(oTarget, oCaster); // Init the wound loop. This will deal d6 damage, which in turn determines whether the target will be stunned again next round
|
|
}
|
|
|
|
// Apply Damage
|
|
SPApplyEffectToObject(DURATION_TYPE_INSTANT, PRCEffectDamage(oTarget, nDam, DAMAGE_TYPE_MAGICAL), oTarget);
|
|
|
|
// If this spell was a held charge, it can have multiple targets. So we need to keep track of who has been already affected
|
|
if(nEvent)
|
|
set_add_object(oCaster, "PRC_Spell_RoRTargets", oTarget);
|
|
}
|
|
}
|
|
|
|
// Corruption cost is paid whenever something is hit
|
|
DoCorruptionCost(oCaster, ABILITY_STRENGTH, 1, 0);
|
|
}
|
|
|
|
// Return the result of the touch attack, will be used to determine whether to reduce number of touch attacks remaining
|
|
return iAttackRoll;
|
|
}
|
|
|
|
void WoundLoop(object oTarget, object oCaster, int nDamage = 0)
|
|
{
|
|
// If previous round's damage roll was 6, stun this round
|
|
if(nDamage == 6)
|
|
SPApplyEffectToObject(DURATION_TYPE_TEMPORARY, EffectStunned(), oTarget, 6.0f);
|
|
|
|
// Roll and apply new damage
|
|
nDamage = d6();
|
|
nDamage += SpellDamagePerDice(oCaster, 1);
|
|
SPApplyEffectToObject(DURATION_TYPE_INSTANT, PRCEffectDamage(oTarget, nDamage, DAMAGE_TYPE_MAGICAL), oTarget);
|
|
|
|
DelayCommand(6.0f, WoundLoop(oTarget, oCaster, nDamage));
|
|
}
|