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.
421 lines
15 KiB
Plaintext
421 lines
15 KiB
Plaintext
//::///////////////////////////////////////////////
|
||
//:: Tensor's Transformation
|
||
//:: NW_S0_TensTrans.nss
|
||
//:: Copyright (c) 2001 Bioware Corp.
|
||
//:://////////////////////////////////////////////
|
||
/*
|
||
Gives the caster the following bonuses:
|
||
+1 Attack per 2 levels
|
||
+4 Natural AC
|
||
20 STR and DEX and CON
|
||
1d6 Bonus HP per level
|
||
+5 on Fortitude Saves
|
||
-10 Intelligence
|
||
*/
|
||
//:://////////////////////////////////////////////
|
||
//:: Created By: Preston Watamaniuk
|
||
//:: Created On: Oct 26, 2001
|
||
//:://////////////////////////////////////////////
|
||
//: Sep2002: losing hit-points won't get rid of the rest of the bonuses
|
||
|
||
//:: modified by mr_bumpkin Dec 4, 2003
|
||
//:: modified by motu99 Apr 3, 2007
|
||
|
||
/*
|
||
PnP TT:
|
||
Tenser's Transformation
|
||
Transmutation
|
||
Level: Sor/Wiz 6
|
||
Components: V, S, M
|
||
Casting Time: 1 action
|
||
Range: Personal
|
||
Target: You
|
||
Duration: 1 round/level
|
||
You become a virtual fighting machine—
|
||
stronger, tougher, faster, and more skilled
|
||
in combat. Your mind-set changes so that
|
||
you relish combat and you can’t cast spells,
|
||
even from magic items.
|
||
You gain 1d6 temporary hit points per
|
||
caster level, a +4 natural armor bonus to
|
||
AC, a +2d4 Strength enhancement bonus,
|
||
a +2d4 Dexterity enhancement bonus, a +1
|
||
base attack bonus per two caster levels
|
||
(which may give you an extra attack), a +5
|
||
competence bonus on Fortitude saves, and
|
||
proficiency with all simple and martial
|
||
weapons. You attack opponents with
|
||
melee or ranged weapons if you can, even
|
||
resorting to unarmed attacks if that’s all
|
||
you can do.
|
||
Material Component: A potion of bull’s
|
||
strength, which you drink (and whose
|
||
effects are subsumed by the spell effects).
|
||
*/
|
||
#include "prc_inc_spells"
|
||
|
||
|
||
|
||
#include "prc_alterations"
|
||
#include "x2_inc_shifter"
|
||
|
||
#include "pnp_shft_poly"
|
||
#include "prc_inc_combat"
|
||
|
||
|
||
/*
|
||
int CalculateAttackBonus(object oPC)
|
||
{
|
||
int iBAB = GetBaseAttackBonus(oPC);
|
||
int iHD = GetHitDice(oPC);
|
||
int iBonus = (iHD > 20) ? ((20 + (iHD - 19) / 2) - iBAB) : (iHD - iBAB); // most confusing line ever. :)
|
||
|
||
return (iBonus > 0) ? iBonus : 0;
|
||
}
|
||
*/
|
||
|
||
// this is used for PnP Tenser's in order to ensure that we attack anything in our sight
|
||
// and to turn off that behavior (and restore # attacks), when spell has ended
|
||
void PnPTensTransPseudoHB()
|
||
{
|
||
// object oPC = OBJECT_SELF;
|
||
if (DEBUG) DoDebug("entered PnPTensTransPseudoHB");
|
||
|
||
// if we don't have the spell effect any more, do clean up
|
||
if(!GetHasSpellEffect(SPELL_TENSERS_TRANSFORMATION))
|
||
{
|
||
//remove IPs for simple + martial weapon prof
|
||
object oSkin = GetPCSkin(OBJECT_SELF);
|
||
int nSimple;
|
||
int nMartial;
|
||
itemproperty ipTest = GetFirstItemProperty(oSkin);
|
||
while(GetIsItemPropertyValid(ipTest))
|
||
{
|
||
if(!nSimple
|
||
&& GetItemPropertyType(ipTest) == ITEM_PROPERTY_BONUS_FEAT
|
||
&& GetItemPropertyDurationType(ipTest) == DURATION_TYPE_TEMPORARY
|
||
&& GetItemPropertyCostTableValue(ipTest) == IP_CONST_FEAT_WEAPON_PROF_SIMPLE)
|
||
{
|
||
RemoveItemProperty(oSkin, ipTest);
|
||
nSimple = TRUE;
|
||
}
|
||
if(!nMartial
|
||
&& GetItemPropertyType(ipTest) == ITEM_PROPERTY_BONUS_FEAT
|
||
&& GetItemPropertyDurationType(ipTest) == DURATION_TYPE_TEMPORARY
|
||
&& GetItemPropertyCostTableValue(ipTest) == IP_CONST_FEAT_WEAPON_PROF_MARTIAL)
|
||
{
|
||
RemoveItemProperty(oSkin, ipTest);
|
||
nMartial = TRUE;
|
||
}
|
||
ipTest = GetNextItemProperty(oSkin);
|
||
}
|
||
// motu99: added this, to facilitate logic in prc_bab_caller
|
||
// DeleteLocalInt(OBJECT_SELF, "AttackCount_TensersTrans");
|
||
DeleteLocalInt(OBJECT_SELF, "CasterLvl_TensersTrans");
|
||
|
||
// now execute prc_bab_caller to set the base attack count to normal again
|
||
ExecuteScript("prc_bab_caller", OBJECT_SELF);
|
||
|
||
//end the pseudoHB
|
||
return;
|
||
}
|
||
|
||
// the spell is still active: make sure that we attack everything in sight
|
||
if(GetCurrentAction() != ACTION_ATTACKOBJECT)
|
||
{
|
||
//DoDebug("PnPTensTransPseudoHB: not attacking, look for enemies to attack");
|
||
// look for any living enemies
|
||
object oTarget = GetNearestCreature(CREATURE_TYPE_REPUTATION,
|
||
REPUTATION_TYPE_ENEMY, OBJECT_SELF, 1,
|
||
CREATURE_TYPE_PERCEPTION, PERCEPTION_SEEN_AND_HEARD,
|
||
CREATURE_TYPE_IS_ALIVE, TRUE);
|
||
|
||
// if we find a living enemy, we will attack it, otherwise don't bother
|
||
if (GetIsObjectValid(oTarget) && !GetIsDead(oTarget))
|
||
{
|
||
//DoDebug("PnPTensTransPseudoHB: found living enemy - clear all actions and attack it");
|
||
// stop anything we have been doing
|
||
ClearAllActions();
|
||
|
||
// if we don't yet have a weapon equipped (unless we are unarmed with a creature weapon equipped or a monk)
|
||
// we equip the most damaging ranged or melee weapon, depending on the distance to our enemies
|
||
object oWeapon = GetItemInSlot(INVENTORY_SLOT_RIGHTHAND, OBJECT_SELF);
|
||
if( ( !GetIsObjectValid(oWeapon) && !GetIsUnarmedFighter(OBJECT_SELF) )
|
||
|| GetBaseItemType(oWeapon) == BASE_ITEM_TORCH)
|
||
{
|
||
//DoDebug("PnPTensTransPseudoHB: no weapon equipped, equip most damaging weapon");
|
||
if(GetDistanceToObject(oTarget) > 10.0)
|
||
{
|
||
ActionEquipMostDamagingRanged(oTarget);
|
||
}
|
||
else
|
||
{
|
||
int nTwoWeapon;
|
||
if(GetHasFeat(FEAT_AMBIDEXTERITY))
|
||
nTwoWeapon++;
|
||
if(GetHasFeat(FEAT_TWO_WEAPON_FIGHTING))
|
||
nTwoWeapon++;
|
||
if(GetHasFeat(FEAT_IMPROVED_TWO_WEAPON_FIGHTING))
|
||
nTwoWeapon++;
|
||
ActionEquipMostDamagingMelee(oTarget, nTwoWeapon);
|
||
}
|
||
}
|
||
// now attack the enemy
|
||
ActionAttack(oTarget);
|
||
}
|
||
}
|
||
// do the pseudo heart beat again in 6 seconds, continue as long as spell lasts
|
||
DelayCommand(6.0, PnPTensTransPseudoHB());
|
||
}
|
||
|
||
void main()
|
||
{
|
||
DeleteLocalInt(OBJECT_SELF, "X2_L_LAST_SPELLSCHOOL_VAR");
|
||
SetLocalInt(OBJECT_SELF, "X2_L_LAST_SPELLSCHOOL_VAR", SPELL_SCHOOL_TRANSMUTATION);
|
||
|
||
// check if we should the PnP version
|
||
if(GetPRCSwitch(PRC_PNP_TENSERS_TRANSFORMATION))
|
||
{
|
||
// motu99: shouldn't this already have been checked before?
|
||
// in any case it will prevent casting a second tenser's while the first is still running
|
||
// (because no spell casting at all is allowed for a creature who has PnP tenser's on it)
|
||
if (!X2PreSpellCastCode())
|
||
{
|
||
return;
|
||
}
|
||
|
||
//Declare major variables
|
||
object oTarget = PRCGetSpellTargetObject();
|
||
object oSkin = GetPCSkin(oTarget);
|
||
|
||
int nCasterLvl = PRCGetCasterLevel(OBJECT_SELF);
|
||
//DoDebug("nw_s0_tensTrans: "+GetName(OBJECT_SELF) +" casts level " + IntToString(nCasterLvl)+ " TensersTrans on " + GetName(oTarget));
|
||
float fDuration = RoundsToSeconds(nCasterLvl);
|
||
int nMeta = PRCGetMetaMagicFeat();
|
||
|
||
// Attack Bonus Increase
|
||
int nAB = nCasterLvl / 2;
|
||
|
||
int nHP;
|
||
int nStr;
|
||
int nDex;
|
||
int nCon;
|
||
|
||
// find out the number of attacks with the bonus (can be higher than 4)
|
||
// motu99: commented this out, because it is done in prc_bab_caller
|
||
// int nTotalAttacks = GetMainHandAttacks(oTarget, nAB);
|
||
|
||
//Determine bonus HP and Ability adjustments
|
||
//Metamagic
|
||
if(nMeta & METAMAGIC_MAXIMIZE)
|
||
{
|
||
nHP = nCasterLvl * 6;
|
||
nStr = 8;
|
||
nDex = 8;
|
||
nCon = 8;
|
||
}
|
||
else
|
||
{
|
||
nHP = d6(nCasterLvl);
|
||
nStr = d4(2);
|
||
nDex = d4(2);
|
||
nCon = d4(2);
|
||
}
|
||
|
||
if(nMeta & METAMAGIC_EMPOWER)
|
||
{
|
||
nHP += nHP/2;
|
||
nStr += nStr/2;
|
||
nDex += nDex/2;
|
||
nCon += nCon/2;
|
||
}
|
||
|
||
if(nMeta & METAMAGIC_EXTEND)
|
||
{
|
||
fDuration += fDuration;
|
||
}
|
||
|
||
effect eEffect = EffectAbilityIncrease(ABILITY_STRENGTH, nStr);
|
||
eEffect = EffectLinkEffects(eEffect, EffectAbilityIncrease(ABILITY_DEXTERITY, nDex));
|
||
eEffect = EffectLinkEffects(eEffect, EffectAbilityIncrease(ABILITY_CONSTITUTION, nCon));
|
||
eEffect = EffectLinkEffects(eEffect, EffectAttackIncrease(nAB));
|
||
eEffect = EffectLinkEffects(eEffect, EffectSavingThrowIncrease(SAVING_THROW_FORT, 5));
|
||
//accounted for in prc_bab_caller
|
||
// eEffect = EffectLinkEffects(eEffect, EffectModifyAttacks(nAttacks));
|
||
|
||
//apply separately (so that we don't loose any other boni when we loose the temporary HP)
|
||
effect eHP = EffectTemporaryHitpoints(nHP);
|
||
|
||
effect eVis = EffectVisualEffect(VFX_IMP_SUPER_HEROISM);
|
||
|
||
SPApplyEffectToObject(DURATION_TYPE_INSTANT, eVis, oTarget);
|
||
SPApplyEffectToObject(DURATION_TYPE_TEMPORARY, eHP, oTarget, fDuration, TRUE, -1, nCasterLvl);
|
||
SPApplyEffectToObject(DURATION_TYPE_TEMPORARY, eEffect, oTarget, fDuration, TRUE, -1, nCasterLvl);
|
||
|
||
itemproperty ipSimple = PRCItemPropertyBonusFeat(IP_CONST_FEAT_WEAPON_PROF_SIMPLE);
|
||
itemproperty ipMartial = PRCItemPropertyBonusFeat(IP_CONST_FEAT_WEAPON_PROF_MARTIAL);
|
||
//AddItemProperty(DURATION_TYPE_TEMPORARY, ipSimple, oSkin, fDuration);
|
||
//AddItemProperty(DURATION_TYPE_TEMPORARY, ipMartial, oSkin, fDuration);
|
||
IPSafeAddItemProperty(oSkin, ipSimple, fDuration);
|
||
IPSafeAddItemProperty(oSkin, ipMartial, fDuration);
|
||
|
||
// remember the caster level in order to properly calculate the number of attacks in prc_bab_caller
|
||
SetLocalInt(oTarget, "CasterLvl_TensersTrans", nCasterLvl);
|
||
// SetLocalInt(oTarget, "AttackCount_TensersTrans", nTotalAttacks);
|
||
// prc_bab_caller must be executed *after* the spell effects have been applied to the target (otherwise it won't detect Tenser's on the target)
|
||
ExecuteScript("prc_bab_caller", oTarget);
|
||
|
||
//motu99: why don't we signal the spell event, as in Bioware tenser's?
|
||
// SignalEvent(oTarget, EventSpellCastAt(oTarget, SPELL_TENSERS_TRANSFORMATION, FALSE));
|
||
|
||
// put the pseudo heart beat on the target of the spell
|
||
DelayCommand(6.0, AssignCommand(oTarget,PnPTensTransPseudoHB()));
|
||
|
||
DeleteLocalInt(OBJECT_SELF, "X2_L_LAST_SPELLSCHOOL_VAR");
|
||
// Getting rid of the integer used to hold the spells spell school
|
||
|
||
// finished with PnP version
|
||
return;
|
||
}
|
||
|
||
|
||
|
||
//----------------------------------------------------------------------------
|
||
// GZ, Nov 3, 2003
|
||
// There is a serious problems with creatures turning into unstoppable killer
|
||
// machines when affected by tensors transformation. NPC AI can't handle that
|
||
// spell anyway, so I added this code to disable the use of Tensors by any
|
||
// NPC.
|
||
//----------------------------------------------------------------------------
|
||
if (!GetIsPC(OBJECT_SELF))
|
||
{
|
||
WriteTimestampedLogEntry(GetName(OBJECT_SELF) + "[" + GetTag (OBJECT_SELF) +"] tried to cast Tensors Transformation. Bad! Remove that spell from the creature");
|
||
return;
|
||
}
|
||
|
||
/*
|
||
Spellcast Hook Code
|
||
Added 2003-06-23 by GeorgZ
|
||
If you want to make changes to all spells,
|
||
check x2_inc_spellhook.nss to find out more
|
||
*/
|
||
|
||
if (!X2PreSpellCastCode())
|
||
{
|
||
return;
|
||
}
|
||
|
||
// End of Spell Cast Hook
|
||
|
||
//Declare major variables
|
||
object oTarget = PRCGetSpellTargetObject();
|
||
int CasterLvl = PRCGetCasterLevel(OBJECT_SELF);
|
||
int nDuration = CasterLvl;
|
||
int nMeta = PRCGetMetaMagicFeat();
|
||
int nHP, nAB, nCnt;
|
||
|
||
/*
|
||
//Determine bonus HP
|
||
for(nCnt; nCnt <= CasterLvl; nCnt++)
|
||
{
|
||
nHP += d6();
|
||
}
|
||
*/
|
||
//Metamagic
|
||
//Determine bonus HP and Ability adjustments
|
||
//Metamagic
|
||
if(nMeta & METAMAGIC_MAXIMIZE)
|
||
nHP = CasterLvl * 6;
|
||
else
|
||
nHP = d6(CasterLvl);
|
||
|
||
if(nMeta & METAMAGIC_EMPOWER)
|
||
nHP += nHP/2;
|
||
|
||
if(nMeta & METAMAGIC_EXTEND)
|
||
nDuration += nDuration;
|
||
|
||
// attack bonus
|
||
nAB = CasterLvl / 2;
|
||
|
||
// Get The PC's Equipment
|
||
object oWeaponOld = GetItemInSlot(INVENTORY_SLOT_RIGHTHAND, oTarget);
|
||
object oArmorOld = GetItemInSlot(INVENTORY_SLOT_CHEST, oTarget);
|
||
object oHelmetOld = GetItemInSlot(INVENTORY_SLOT_HEAD, oTarget);
|
||
object oShieldOld = GetItemInSlot(INVENTORY_SLOT_LEFTHAND, oTarget);
|
||
|
||
if (GetBaseItemType(oShieldOld) != BASE_ITEM_SMALLSHIELD &&
|
||
GetBaseItemType(oShieldOld) != BASE_ITEM_LARGESHIELD &&
|
||
GetBaseItemType(oShieldOld) != BASE_ITEM_TOWERSHIELD)
|
||
{
|
||
oShieldOld = OBJECT_INVALID;
|
||
}
|
||
|
||
//Declare effects
|
||
effect eAttack = EffectAttackIncrease(nAB);
|
||
effect eSave = EffectSavingThrowIncrease(SAVING_THROW_FORT, 5);
|
||
effect eDur = EffectVisualEffect(VFX_DUR_CESSATE_POSITIVE);
|
||
|
||
// motu99: This could be calculated better (see below), but I left it at 2 extra attacks, because this is bioware's strange implementation
|
||
// effect eSwing = EffectModifyAttacks(GetMainHandAttacks(oTarget, nAB)-GetMainHandAttacks(oTarget));
|
||
effect eSwing = EffectModifyAttacks(2);
|
||
|
||
effect ePoly = EffectPolymorph(28);
|
||
effect eHP = EffectTemporaryHitpoints(nHP);
|
||
effect eVis = EffectVisualEffect(VFX_IMP_SUPER_HEROISM);
|
||
|
||
//Link effects
|
||
effect eLink = EffectLinkEffects(eAttack, eSave);
|
||
eLink = EffectLinkEffects(eLink, eDur);
|
||
eLink = EffectLinkEffects(eLink, eSwing);
|
||
eLink = EffectLinkEffects(eLink, ePoly);
|
||
|
||
//Signal Spell Event
|
||
SignalEvent(oTarget, EventSpellCastAt(OBJECT_SELF, SPELL_TENSERS_TRANSFORMATION, FALSE));
|
||
|
||
// abort if mounted
|
||
if (!GetLocalInt(GetModule(),"X3_NO_SHAPESHIFT_SPELL_CHECK"))
|
||
{ // check to see if abort due to being mounted
|
||
if (PRCHorseGetIsMounted(oTarget))
|
||
{ // abort
|
||
if (GetIsPC(oTarget)) FloatingTextStrRefOnCreature(111982,oTarget,FALSE);
|
||
return;
|
||
} // abort
|
||
} // check to see if abort due to being mounted
|
||
|
||
//this command will make shore that polymorph plays nice with the shifter
|
||
ShifterCheck(oTarget);
|
||
|
||
AssignCommand(oTarget, ClearAllActions()); // prevents an exploit
|
||
|
||
SPApplyEffectToObject(DURATION_TYPE_INSTANT, eVis, oTarget);
|
||
SPApplyEffectToObject(DURATION_TYPE_TEMPORARY, eHP, oTarget, RoundsToSeconds(nDuration), TRUE, -1, CasterLvl);
|
||
SPApplyEffectToObject(DURATION_TYPE_TEMPORARY, eLink, oTarget, RoundsToSeconds(nDuration), TRUE, -1, CasterLvl);
|
||
|
||
// Get the Polymorphed form's stuff
|
||
object oWeaponNew = GetItemInSlot(INVENTORY_SLOT_RIGHTHAND,oTarget);
|
||
object oArmorNew = GetItemInSlot(INVENTORY_SLOT_CARMOUR,oTarget);
|
||
|
||
// Tenser's Sword is overpowerful with equipment merging -- no longer do you
|
||
// need a +3 flaming longsword if you have reasonably good equipment.
|
||
itemproperty ip = GetFirstItemProperty(oWeaponNew);
|
||
while (GetIsItemPropertyValid(ip))
|
||
{
|
||
RemoveItemProperty(oWeaponNew, ip);
|
||
ip = GetNextItemProperty(oWeaponNew);
|
||
}
|
||
|
||
// I like that nice flaming effect though.
|
||
itemproperty ipFlaming = ItemPropertyVisualEffect(ITEM_VISUAL_FIRE);
|
||
IPSafeAddItemProperty(oWeaponNew, ipFlaming);
|
||
|
||
// Merges in your stuff so that you're not weakened by the morph.
|
||
IPWildShapeCopyItemProperties(oWeaponOld, oWeaponNew, TRUE);
|
||
IPWildShapeCopyItemProperties(oArmorOld, oArmorNew, TRUE);
|
||
IPWildShapeCopyItemProperties(oHelmetOld, oArmorNew, TRUE);
|
||
IPWildShapeCopyItemProperties(oShieldOld, oArmorNew, TRUE);
|
||
|
||
DeleteLocalInt(OBJECT_SELF, "X2_L_LAST_SPELLSCHOOL_VAR");
|
||
// Getting rid of the integer used to hold the spells spell school
|
||
}
|