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.
		
			
				
	
	
		
			722 lines
		
	
	
		
			32 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			722 lines
		
	
	
		
			32 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| //::///////////////////////////////////////////////
 | |
| //:: Shadowcasting include: Shadowcasting
 | |
| //:: shd_inc_myst
 | |
| //::///////////////////////////////////////////////
 | |
| /** @file
 | |
|     Defines structures and functions for handling
 | |
|     shadowcasting a mystery
 | |
| 
 | |
|     @author Stratovarius
 | |
|     @date   Created - 2019.2.7
 | |
|     @thanks to Ornedan for his work on Psionics upon which this is based.
 | |
| */
 | |
| //:://////////////////////////////////////////////
 | |
| //:://////////////////////////////////////////////
 | |
| 
 | |
| 
 | |
| //////////////////////////////////////////////////
 | |
| /*                 Constants                    */
 | |
| //////////////////////////////////////////////////
 | |
| 
 | |
| const string PRC_SHADOWCASTING_CLASS       = "PRC_CurrentMystery_ShadowcastingClass";
 | |
| const string PRC_MYSTERY_LEVEL             = "PRC_CurrentMystery_Level";
 | |
| const string MYST_DEBUG_IGNORE_CONSTRAINTS = "MYST_DEBUG_IGNORE_CONSTRAINTS";
 | |
| 
 | |
| /**
 | |
|  * The variable in which the mystery token is stored. If no token exists,
 | |
|  * the variable is set to point at the shadowcaster itself. That way OBJECT_INVALID
 | |
|  * means the variable is unitialised.
 | |
|  */
 | |
| //const string PRC_MYSTERY_TOKEN_VAR  = "PRC_MysteryToken";
 | |
| //const string PRC_MYSTERY_TOKEN_NAME = "PRC_MYSTTOKEN";
 | |
| //const float  PRC_MYSTERY_HB_DELAY   = 0.5f;
 | |
| 
 | |
| 
 | |
| //////////////////////////////////////////////////
 | |
| /*                 Structures                   */
 | |
| //////////////////////////////////////////////////
 | |
| 
 | |
| // struct mystery moved to shd_inc_metashd
 | |
| 
 | |
| //////////////////////////////////////////////////
 | |
| /*             Function prototypes              */
 | |
| //////////////////////////////////////////////////
 | |
| 
 | |
| /**
 | |
|  * Determines if the mystery that is currently being attempted to be TrueSpoken
 | |
|  * can in fact be truespoken. Determines metashadows used.
 | |
|  *
 | |
|  * @param oShadow       A creature attempting to shadowcast a mystery at this moment.
 | |
|  * @param oTarget       The target of the mystery, if any. For pure Area of Effect
 | |
|  *                      mysteries, this should be OBJECT_INVALID. Otherwise the main
 | |
|  *                      target of the mystery as returned by PRCGetSpellTargetObject().
 | |
|  * @param nMetaShadFlags The metashadows that may be used to modify this mystery. Any number
 | |
|  *                      of METASHADOW_* constants ORd together using the | operator.
 | |
|  *                      For example (METASHADOW_EMPOWER | METASHADOW_EXTEND)
 | |
|  *
 | |
|  * @return              A mystery structure that contains the data about whether
 | |
|  *                      the mystery was successfully shadowcast, what metashadows
 | |
|  *                      were used and some other commonly used data, like the 
 | |
|  *                      creator's shadowcaster level for this mystery.
 | |
|  */
 | |
| struct mystery EvaluateMystery(object oShadow, object oTarget, int nMetaShadFlags);
 | |
| 
 | |
| /**
 | |
|  * Causes OBJECT_SELF to use the given mystery.
 | |
|  *
 | |
|  * @param nMyst          The index of the mystery to use in spells.2da or a MYST_*
 | |
|  * @param nClass         The index of the class to use the mystery as in classes.2da or a CLASS_TYPE_*
 | |
|  * @param nLevelOverride An optional override to normal shadowcaster level. 
 | |
|  *                       Default: 0, which means the parameter is ignored.
 | |
|  */
 | |
| void UseMystery(int nMyst, int nClass, int nLevelOverride = 0);
 | |
| 
 | |
| /**
 | |
|  * A debugging function. Takes a mystery structure and
 | |
|  * makes a string describing the contents.
 | |
|  *
 | |
|  * @param myst A set of mystery data
 | |
|  * @return      A string describing the contents of myst
 | |
|  */
 | |
| string DebugMystery2Str(struct mystery myst);
 | |
| 
 | |
| /**
 | |
|  * Sets the evaluation functions to ignore constraints on shadowcasting.
 | |
|  * Call this just prior to EvaluateMystery() in a mystery script.
 | |
|  * That evaluation will then ignore lacking mystery ability score,
 | |
|  * and other restrictions
 | |
|  *
 | |
|  * @param oShadow A creature attempting to shadowcast a mystery at this moment.
 | |
|  */
 | |
| void ShadowcastDebugIgnoreConstraints(object oShadow);
 | |
| 
 | |
| /**
 | |
|  * Returns the uses per day already used
 | |
|  *
 | |
|  * @param oShadow    Caster of the Mystery
 | |
|  * @param nMystId    SpellId of the Mystery
 | |
|  * @return           Number of uses per day already used
 | |
|  */
 | |
| int GetMysteryUses(object oShadow, int nMystId);
 | |
| 
 | |
| /**
 | |
|  * Returns the bonus uses per day already used
 | |
|  *
 | |
|  * @param oShadow    Caster of the Mystery
 | |
|  * @param nMystLevel Level of the Mystery
 | |
|  * @return           Number of uses per day already used
 | |
|  */
 | |
| int GetBonusUses(object oShadow, int nMystLevel);
 | |
| 
 | |
| /**
 | |
|  * Adds a use per day
 | |
|  *
 | |
|  * @param oShadow    Caster of the Mystery
 | |
|  * @param nMystId    SpellId of the Mystery
 | |
|  */
 | |
| void SetMysteryUses(object oShadow, int nMystId);
 | |
| 
 | |
| /**
 | |
|  * Adds a bonus use per day
 | |
|  *
 | |
|  * @param oShadow    Caster of the Mystery
 | |
|  * @param nMystLevel Level of the Mystery
 | |
|  */
 | |
| void SetBonusUses(object oShadow, int nMystLevel);
 | |
| 
 | |
| /**
 | |
|  * Deletes all of the Local Ints stored by uses per day.
 | |
|  * Called OnRest and OnEnter
 | |
|  *
 | |
|  * @param oShadow    Caster of the Mystery
 | |
|  */
 | |
| void ClearMystLocalVars(object oShadow);
 | |
| 
 | |
| /**
 | |
|  * Returns total uses per day for the shadowcaster for a given mystery
 | |
|  *
 | |
|  * @param oShadow    Caster of the Mystery
 | |
|  * @param nMystId    SpellId of the Mystery
 | |
|  * @param nClass     Class to check against
 | |
|  */
 | |
| int MysteriesPerDay(object oShadow, int nMystId, int nClass);
 | |
| 
 | |
| /**
 | |
|  * Calculates bonus mysteries from a high intelligence
 | |
|  *
 | |
|  * @param oShadow    Caster of the Mystery
 | |
|  * @param nMystLevel Mystery level to check
 | |
|  */
 | |
| int BonusMysteriesPerDay(object oShadow, int nMystLevel);
 | |
| 
 | |
| /**
 | |
|  * Returns the name of the mystery
 | |
|  *
 | |
|  * @param nMystId        SpellId of the mystery
 | |
|  */
 | |
| string GetMysteryName(int nMystId);
 | |
| 
 | |
| /**
 | |
|  * Checks whether the mystery is supernatural or not
 | |
|  *
 | |
|  * @param nMystId The Mystery to Check
 | |
|  * @return        TRUE if Mystery is (Su), else FALSE
 | |
|  */
 | |
| int GetIsMysterySupernatural(object oShadow, int nMystId, int nClass);
 | |
| 
 | |
| /**
 | |
|  * Checks whether the mystery is a SLA or not
 | |
|  *
 | |
|  * @param nMystId The Mystery to Check
 | |
|  * @return        TRUE if Mystery is (Su), else FALSE
 | |
|  */
 | |
| int GetIsMysterySLA(object oShadow, int nMystId, int nClass);
 | |
| 
 | |
| /**
 | |
|  * Checks whether the mystery is a Fundamental or not
 | |
|  *
 | |
|  * @param nMystId The Mystery to Check
 | |
|  * @return        TRUE if Mystery is a Fundamental, else FALSE
 | |
|  */
 | |
| int GetIsFundamental(int nMystId);
 | |
| 
 | |
| /**
 | |
|  * Checks whether caster has Favored Mystery in the cast mystery
 | |
|  *
 | |
|  * @param oShadow The Shadowcaster
 | |
|  * @param nMyst   The Mystery to Check
 | |
|  * @return        TRUE if he has the feat, else FALSE
 | |
|  */
 | |
| int GetHasFavoredMystery(object oShadow, int nMyst);
 | |
| 
 | |
| /**
 | |
|  * Checks whether caster has Shadow Cast feat
 | |
|  *
 | |
|  * @param oShadow The Shadowcaster
 | |
|  * @return        TRUE if he has the feat, else FALSE
 | |
|  */
 | |
| int GetShadowCast(object oShadow);
 | |
| 
 | |
| //////////////////////////////////////////////////
 | |
| /*                  Includes                    */
 | |
| //////////////////////////////////////////////////
 | |
| 
 | |
| #include "shd_inc_metashd"
 | |
| #include "shd_inc_shdfunc"
 | |
| #include "prc_inc_combat"
 | |
| #include "inc_newspellbook"
 | |
| #include "inc_lookups"
 | |
| 
 | |
| //////////////////////////////////////////////////
 | |
| /*             Internal functions               */
 | |
| //////////////////////////////////////////////////
 | |
| 
 | |
| /** Internal function.
 | |
|  * Handles Spellfire absorption when a mystery is used on a friendly spellfire
 | |
|  * user.
 | |
|  */
 | |
| struct mystery _DoShadowcastSpellfireFriendlyAbsorption(struct mystery myst, object oTarget)
 | |
| {
 | |
|     if(GetLocalInt(oTarget, "SpellfireAbsorbFriendly") &&
 | |
|        GetIsFriend(oTarget, myst.oShadow)
 | |
|        )
 | |
|     {
 | |
|         if(CheckSpellfire(myst.oShadow, oTarget, TRUE))
 | |
|         {
 | |
|             PRCShowSpellResist(myst.oShadow, oTarget, SPELL_RESIST_MANTLE);
 | |
|             myst.bCanMyst = FALSE;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return myst;
 | |
| }
 | |
| 
 | |
| /** Internal function.
 | |
|  * Deletes mystery-related local variables.
 | |
|  *
 | |
|  * @param oShadow The creature currently shadowcasting a mystery
 | |
|  */
 | |
| void _CleanMysteryVariables(object oShadow)
 | |
| {
 | |
|     DeleteLocalInt(oShadow, PRC_SHADOWCASTING_CLASS);
 | |
|     DeleteLocalInt(oShadow, PRC_MYSTERY_LEVEL);
 | |
| }
 | |
| 
 | |
| /** Internal function.
 | |
|  * Sets mystery-related local variables.
 | |
|  *
 | |
|  * @param oShadow      The creature currently shadowcasting a mystery
 | |
|  * @param nClass       Mystery casting class constant
 | |
|  * @param nLevel       Mystery level
 | |
|  * @param bQuicken     If the mystery was quickened 1, else 0
 | |
|  */
 | |
| void _SetMysteryVariables(object oShadow, int nClass, int nLevel, int bQuicken)
 | |
| {
 | |
|     if (DEBUG) FloatingTextStringOnCreature(GetName(oShadow)+" is a "+IntToString(nClass)+" at "+IntToString(nLevel)+" level", oShadow);
 | |
|     SetLocalInt(oShadow, PRC_SHADOWCASTING_CLASS, nClass + 1);
 | |
|     SetLocalInt(oShadow, PRC_MYSTERY_LEVEL, nLevel);
 | |
|     SetLocalInt(oShadow, PRC_MYSTERY_IS_QUICKENED, bQuicken);
 | |
| }
 | |
| 
 | |
| // Makes sure radial spells are stored on the correct row number
 | |
| string _GetMysterySpellId(int nMystId)
 | |
| {
 | |
|     string nReturn = Get2DACache("spells", "Master", nMystId);
 | |
|     if (1 > StringToInt(nReturn)) nReturn = IntToString(nMystId); // SpellId invalid for the Master column
 | |
|     
 | |
|     return nReturn;
 | |
| }    
 | |
| 
 | |
| //////////////////////////////////////////////////
 | |
| /*             Function definitions             */
 | |
| //////////////////////////////////////////////////
 | |
| 
 | |
| struct mystery EvaluateMystery(object oShadow, object oTarget, int nMetaShadFlags)
 | |
| {
 | |
|     /* Get some data */
 | |
|     int bIgnoreConstraints = (DEBUG) ? GetLocalInt(oShadow, MYST_DEBUG_IGNORE_CONSTRAINTS) : FALSE;
 | |
|     // shadowcaster-related stuff
 | |
|     int nShadowcasterLevel = GetShadowcasterLevel(oShadow);
 | |
|     int nMystLevel         = GetMysteryLevel(oShadow);
 | |
|     int nClass             = GetShadowcastingClass(oShadow);
 | |
| 
 | |
|     if (DEBUG) FloatingTextStringOnCreature(GetName(oShadow)+" is a "+IntToString(nClass)+" casting a "+IntToString(nMystLevel)+" level mystery at "+IntToString(nShadowcasterLevel)+" shadowcaster level", oShadow);
 | |
| 
 | |
|     /* Initialise the mystery structure */
 | |
|     struct mystery myst;
 | |
|     myst.oShadow            = oShadow;
 | |
|     myst.bCanMyst           = TRUE;                                   // Assume successful mystery by default
 | |
|     myst.nShadowcasterLevel = nShadowcasterLevel;
 | |
|     myst.nMystId            = PRCGetSpellId();
 | |
|     myst.bIgnoreSR          = GetIsMysterySupernatural(oShadow, myst.nMystId, nClass);
 | |
| 
 | |
|     // Account for metashadows. This includes adding the appropriate DC boosts.
 | |
|     myst = EvaluateMetashadows(myst, nMetaShadFlags);
 | |
|     
 | |
|     // Skip paying anything if something has prevented a successful cast already by this point
 | |
|     // Fundamentals track their uses per day differently. 
 | |
|     if(myst.bCanMyst && !GetIsFundamental(myst.nMystId))
 | |
|     {
 | |
|         if (GetMysteryUses(oShadow, myst.nMystId) >= MysteriesPerDay(oShadow, myst.nMystId, nClass)) // Used up all regular uses
 | |
|         {
 | |
|             SetBonusUses(myst.oShadow, nMystLevel);
 | |
|             FloatingTextStringOnCreature("You have "+IntToString(BonusMysteriesPerDay(oShadow, nMystLevel)-GetBonusUses(oShadow, nMystLevel))+" bonus uses of level "+IntToString(nMystLevel)+" remaining", oShadow, FALSE);
 | |
|         }    
 | |
|         else
 | |
|         {
 | |
|             SetMysteryUses(myst.oShadow, myst.nMystId);
 | |
|             FloatingTextStringOnCreature("You have "+IntToString(MysteriesPerDay(myst.oShadow, myst.nMystId, nClass)-GetMysteryUses(myst.oShadow, myst.nMystId))+" uses of " + GetMysteryName(myst.nMystId) + " remaining", oShadow, FALSE);
 | |
|             FloatingTextStringOnCreature("You have "+IntToString(BonusMysteriesPerDay(oShadow, nMystLevel)-GetBonusUses(oShadow, nMystLevel))+" bonus uses of level "+IntToString(nMystLevel)+" remaining", oShadow, FALSE);
 | |
|         }    
 | |
|     }//end if
 | |
| 
 | |
|     if(DEBUG) DoDebug("EvaluateMystery(): Final result:\n" + DebugMystery2Str(myst));
 | |
| 
 | |
|     // Initiate mystery-related variable CleanUp
 | |
|     //DelayCommand(0.5f, _CleanMysteryVariables(oShadow));
 | |
| 
 | |
|     return myst;
 | |
| }
 | |
| 
 | |
| void UseMystery(int nMyst, int nClass, int nLevelOverride = 0)
 | |
| {
 | |
|     object oShadow = OBJECT_SELF;
 | |
|     int bQuickened = FALSE;
 | |
|     object oSkin   = GetPCSkin(oShadow);    
 | |
|     int nMystDur   = StringToInt(Get2DACache("spells", "ConjTime", nMyst)) + StringToInt(Get2DACache("spells", "CastTime", nMyst));
 | |
|     int nMystLevel = GetMysteryLevel(oShadow, nMyst);
 | |
|     
 | |
|     if(GetAbilityScore(oShadow, ABILITY_INTELLIGENCE, TRUE) < nMystLevel + 10)    
 | |
|     {
 | |
|         FloatingTextStringOnCreature("Your Intelligence score is too low to shadowcast this mystery", oShadow, FALSE);
 | |
|         return;
 | |
|     }      
 | |
|     
 | |
|     // Check uses per day. This is done here so that players don't waste an action on it
 | |
|     if ((GetMysteryUses(oShadow, nMyst) >= MysteriesPerDay(oShadow, nMyst, nClass)) && (GetBonusUses(oShadow, nMystLevel) >= BonusMysteriesPerDay(oShadow, nMystLevel)))
 | |
|     {
 | |
|         FloatingTextStringOnCreature("You have used " + GetMysteryName(nMyst) + " the maximum amount of times today.", oShadow, FALSE);
 | |
|         return;
 | |
|     }      
 | |
| 
 | |
|     // Quicken mystery check
 | |
|     if(nMystDur <= 6000                                 && // If the mystery could be quickened by having shadowcasting time of 1 round or less
 | |
|        GetLocalInt(oShadow, METASHADOW_QUICKEN_VAR) &&     // And the shadowcaster has Quicken active
 | |
|        TakeSwiftAction(oShadow))                           // And the shadowcaster can take a swift action
 | |
|     {
 | |
|         //Adding Auto-Quicken III for one round - deleted after casting is finished.
 | |
|         itemproperty ipAutoQuicken = ItemPropertyBonusFeat(IP_CONST_NSB_AUTO_QUICKEN);
 | |
|         ActionDoCommand(AddItemProperty(DURATION_TYPE_TEMPORARY, ipAutoQuicken, oSkin, nMystDur/1000.0f));
 | |
|         bQuickened = TRUE;
 | |
|         // Then clear the variable
 | |
|         DeleteLocalInt(oShadow, METASHADOW_QUICKEN_VAR);          
 | |
|     }
 | |
| 
 | |
|     if (nMyst == MYST_SHADOW_SKIN || nMyst == MYST_SHADOW_EVOCATION_CONV || nMyst == MYST_GREATER_SHADOW_EVO_CONV ||
 | |
|         nMyst == MYST_GREATER_SHADOW_EVO || nMyst == MYST_SHADOW_EVOCATION || (nMyst == MYST_ECHO_SPELL && GetLocalInt(oShadow, "EchoedSpell"))) // These are immediate actions
 | |
|     {
 | |
|         //Adding Auto-Quicken III for one round - deleted after casting is finished.
 | |
|         itemproperty ipAutoQuicken = ItemPropertyBonusFeat(IP_CONST_NSB_AUTO_QUICKEN);
 | |
|         ActionDoCommand(AddItemProperty(DURATION_TYPE_TEMPORARY, ipAutoQuicken, oSkin, nMystDur/1000.0f));
 | |
|         bQuickened = TRUE;
 | |
|     }
 | |
|     
 | |
|     // SLAs and Supernaturals both ignore the Somatic component
 | |
|     if(GetIsMysterySupernatural(oShadow, nMyst, nClass) || GetIsMysterySLA(oShadow, nMyst, nClass))
 | |
|     {
 | |
|         //Adding Auto-Still for one round - deleted after casting is finished. 7-9th level mysteries don't become SLA, hence lack of Still III
 | |
|         itemproperty ipAutoStill = ItemPropertyBonusFeat(IP_CONST_FEAT_EPIC_AUTO_STILL_II);
 | |
|         ActionDoCommand(AddItemProperty(DURATION_TYPE_TEMPORARY, ipAutoStill, oSkin, nMystDur/1000.0f));
 | |
|         ipAutoStill = ItemPropertyBonusFeat(IP_CONST_FEAT_EPIC_AUTO_STILL_I);
 | |
|         ActionDoCommand(AddItemProperty(DURATION_TYPE_TEMPORARY, ipAutoStill, oSkin, nMystDur/1000.0f));        
 | |
|     }    
 | |
|     // Supernaturals don't generate AoO, nor do casters with the Shadowcast feat in the right situation
 | |
|     if(GetIsMysterySupernatural(oShadow, nMyst, nClass) || GetShadowCast(oShadow))
 | |
|     {
 | |
|         //Adding Improved Combat Casting for one round - deleted after casting is finished.
 | |
|         itemproperty ipICC = ItemPropertyBonusFeat(IP_CONST_IMP_CC);
 | |
|         ActionDoCommand(AddItemProperty(DURATION_TYPE_TEMPORARY, ipICC, oSkin, nMystDur/1000.0f));        
 | |
|     } 
 | |
|     // Still mystery check. Both SLA and Su are automatically stilled.
 | |
|     if(!GetIsMysterySupernatural(oShadow, nMyst, nClass) && !GetIsMysterySLA(oShadow, nMyst, nClass) && GetLocalInt(oShadow, METASHADOW_STILL_VAR))
 | |
|     {
 | |
|         //Adding Auto-Still for one round - deleted after casting is finished. 
 | |
|         itemproperty ipAutoStill = ItemPropertyBonusFeat(IP_CONST_FEAT_EPIC_AUTO_STILL_II);
 | |
|         ActionDoCommand(AddItemProperty(DURATION_TYPE_TEMPORARY, ipAutoStill, oSkin, nMystDur/1000.0f)); 
 | |
|         ipAutoStill = ItemPropertyBonusFeat(IP_CONST_FEAT_EPIC_AUTO_STILL_I);
 | |
|         ActionDoCommand(AddItemProperty(DURATION_TYPE_TEMPORARY, ipAutoStill, oSkin, nMystDur/1000.0f));    
 | |
|         ipAutoStill = ItemPropertyBonusFeat(IP_CONST_FEAT_EPIC_AUTO_STILL_III);
 | |
|         ActionDoCommand(AddItemProperty(DURATION_TYPE_TEMPORARY, ipAutoStill, oSkin, nMystDur/1000.0f));    
 | |
|         // Then clear the variable
 | |
|         DeleteLocalInt(oShadow, METASHADOW_STILL_VAR);          
 | |
|     }    
 | |
| 
 | |
|     // Setup mystery-related variables
 | |
|     ActionDoCommand(_SetMysteryVariables(oShadow, nClass, StringToInt(lookup_spell_innate(nMyst)), bQuickened));
 | |
| 
 | |
|     // cast the actual mystery
 | |
|     ActionCastSpell(nMyst, nLevelOverride, 0, 0, METAMAGIC_NONE, CLASS_TYPE_INVALID, 0, 0, OBJECT_INVALID, FALSE);
 | |
| 
 | |
|     // Initiate mystery-related variable CleanUp
 | |
|     ActionDoCommand(_CleanMysteryVariables(oShadow));
 | |
| }
 | |
| 
 | |
| string DebugMystery2Str(struct mystery myst)
 | |
| {
 | |
|     string sRet;
 | |
| 
 | |
|     sRet += "oShadow = " + DebugObject2Str(myst.oShadow) + "\n";
 | |
|     sRet += "bCanMyst = " + DebugBool2String(myst.bCanMyst) + "\n";
 | |
|     sRet += "nShadowcasterLevel = "  + IntToString(myst.nShadowcasterLevel) + "\n";
 | |
| 
 | |
|     sRet += "bEmpower  = " + DebugBool2String(myst.bEmpower)  + "\n";
 | |
|     sRet += "bExtend   = " + DebugBool2String(myst.bExtend)   + "\n";
 | |
|     sRet += "bQuicken  = " + DebugBool2String(myst.bQuicken);//    + "\n";
 | |
| 
 | |
|     return sRet;
 | |
| }
 | |
| 
 | |
| void ShadowcastDebugIgnoreConstraints(object oShadow)
 | |
| {
 | |
|     SetLocalInt(oShadow, MYST_DEBUG_IGNORE_CONSTRAINTS, TRUE);
 | |
|     DelayCommand(0.0f, DeleteLocalInt(oShadow, MYST_DEBUG_IGNORE_CONSTRAINTS));
 | |
| }
 | |
| 
 | |
| int GetMysteryUses(object oShadow, int nMystId)
 | |
| {
 | |
|     // This makes sure everything is stored using the proper number
 | |
|     return GetLocalInt(oShadow, MYSTERY_USES + _GetMysterySpellId(nMystId));
 | |
| }
 | |
| 
 | |
| int GetBonusUses(object oShadow, int nMystLevel)
 | |
| {
 | |
|     // This makes sure everything is stored using the proper number
 | |
|     return GetLocalInt(oShadow, MYSTERY_BONUS_USES + IntToString(nMystLevel));
 | |
| }
 | |
| 
 | |
| void SetMysteryUses(object oShadow, int nMystId)
 | |
| {
 | |
|     // Apprentice mysteries only for Warp Spell
 | |
|     if(4 > GetMysteryLevel(oShadow, nMystId) && GetLocalInt(oShadow, "WarpSpellSuccess"))
 | |
|     {
 | |
|         DeleteLocalInt(oShadow, "WarpSpellSuccess");
 | |
|         FloatingTextStringOnCreature("Warped Spell used", oShadow, FALSE);
 | |
|         return;
 | |
|     }
 | |
|     if (nMystId == MYST_ECHO_SPELL && GetLocalInt(oShadow, "EchoedSpell")) 
 | |
|         return; //This is a free use and doesn't count. 
 | |
|         
 | |
|     if (GetLocalInt(oShadow, "MysteryFreeUse")) 
 | |
|         return; 
 | |
|       
 | |
|     if (GetLocalInt(oShadow, "InnateCounterSuccess") == GetMysteryLevel(oShadow, nMystId))
 | |
|     {
 | |
|         DeleteLocalInt(oShadow, "InnateCounterSuccess");
 | |
|         FloatingTextStringOnCreature("Innate Counterspell used", oShadow, FALSE);
 | |
|         return;
 | |
|     }
 | |
|         
 | |
|     // This makes sure everything is stored using the proper number
 | |
|     string sSpellId = _GetMysterySpellId(nMystId);
 | |
|     // Uses are stored for each Mystery by SpellId
 | |
|     int nNum = GetLocalInt(oShadow, MYSTERY_USES + sSpellId);
 | |
|     // Store the number of times per day its been cast succesfully
 | |
|     SetLocalInt(oShadow, MYSTERY_USES + sSpellId, (nNum + 1));
 | |
| }
 | |
| 
 | |
| void SetBonusUses(object oShadow, int nMystLevel)
 | |
| {
 | |
|     // Uses are stored by level
 | |
|     int nNum = GetLocalInt(oShadow, MYSTERY_BONUS_USES + IntToString(nMystLevel));
 | |
|     // Store the number of times per day its been cast succesfully
 | |
|     SetLocalInt(oShadow, MYSTERY_BONUS_USES + IntToString(nMystLevel), (nNum + 1));
 | |
| }
 | |
| 
 | |
| void ClearMystLocalVars(object oShadow)
 | |
| {
 | |
|     // Uses are stored for each Mystery by SpellId
 | |
|     // So we loop em all and blow em away
 | |
|     // Because there are only 60, this should not TMI
 | |
|     // i is the SpellId
 | |
|     int i;
 | |
|     for(i = 18352; i < 18429; i++)
 | |
|     {
 | |
|         DeleteLocalInt(oShadow, MYSTERY_USES + IntToString(i));
 | |
|     }
 | |
|     // Web Enhancement Mysteries
 | |
|     for(i = 18579; i < 18590; i++)
 | |
|     {
 | |
|         DeleteLocalInt(oShadow, MYSTERY_USES + IntToString(i));
 | |
|     }    
 | |
|     for(i = 0; i < 10; i++)
 | |
|     {
 | |
|         DeleteLocalInt(oShadow, MYSTERY_BONUS_USES + IntToString(i));
 | |
|     }        
 | |
| }
 | |
| 
 | |
| int MysteriesPerDay(object oShadow, int nMystId, int nClass)
 | |
| {
 | |
|     if (nClass == CLASS_TYPE_SHADOWSMITH) return 1; // They never get more than this.
 | |
| 
 | |
|     int nUses = 1; //always get at least 1
 | |
|     int nLevel = GetMysteryLevel(oShadow, nMystId);
 | |
|     // Done this way so it doesn't count for feats or other misc boosts
 | |
|     int nShadow = GetLevelByClass(nClass, oShadow) + GetShadowMagicPRCLevels(oShadow);
 | |
|     
 | |
|     //if(DEBUG) DoDebug("MysteriesPerDay(): GetMysteryLevel "+IntToString(nLevel));
 | |
|     //if(DEBUG) DoDebug("MysteriesPerDay(): GetShadowcasterLevel "+IntToString(nShadow));
 | |
|     
 | |
|     if (nMystId == MYST_ECHO_SPELL && GetLocalInt(oShadow, "EchoedSpell")) 
 | |
|         return 99; //This is a free use and doesn't count.
 | |
|         
 | |
|     if (GetLocalInt(oShadow, "MysteryFreeUse")) 
 | |
|         return 99;            
 | |
|     
 | |
|     if (nShadow >= 13 && 4 > nLevel)
 | |
|         nUses = 3;
 | |
|     else if (nShadow >= 13 && 7 > nLevel && nLevel > 3)
 | |
|         nUses = 2;    
 | |
|     else if (nShadow >= 7 && 4 > nLevel)
 | |
|         nUses = 2;     
 | |
|         
 | |
|     if (nShadow >= 13 && 4 > nLevel && GetHasFavoredMystery(oShadow, nMystId)) // Favored mystery can grant +1 use per day to Apprentice 
 | |
|         nUses += 1;
 | |
|     
 | |
|     return nUses;
 | |
| }
 | |
| 
 | |
| int BonusMysteriesPerDay(object oShadow, int nMystLevel)
 | |
| {
 | |
|     if(GetAbilityScore(oShadow, ABILITY_CHARISMA, TRUE) < nMystLevel + 10)
 | |
|         return 0;
 | |
|     int nSlots;
 | |
| 
 | |
|     // Both Mystery classes use Int for this
 | |
|     int nAbilityMod = GetAbilityModifier(ABILITY_CHARISMA, oShadow);
 | |
|     if (nAbilityMod >= nMystLevel) // Need an ability modifier at least equal to the spell level to gain bonus slots
 | |
|         nSlots += ((nAbilityMod - nMystLevel) / 4) + 1;
 | |
|     if(DEBUG) DoDebug("BonusMysteriesPerDay(): nSlots "+IntToString(nSlots));
 | |
|     return nSlots;
 | |
| }
 | |
| 
 | |
| string GetMysteryName(int nMystId)
 | |
| {
 | |
|     return GetStringByStrRef(StringToInt(Get2DACache("spells", "Name", nMystId)));
 | |
| }
 | |
| 
 | |
| int GetIsMysterySupernatural(object oShadow, int nMystId, int nClass)
 | |
| {
 | |
|     if (nClass == CLASS_TYPE_SHADOWSMITH) return FALSE;
 | |
|     int nLevel = GetMysteryLevel(oShadow, nMystId);
 | |
|     int nShadow = GetShadowcasterLevel(oShadow);
 | |
|     
 | |
|     if (nShadow >= 13 && 4 > nLevel)
 | |
|         return TRUE;
 | |
|     if (nShadow >= 13 && 7 > nLevel && nLevel > 3 && GetHasFavoredMystery(oShadow, nMystId)) // Bump up Initiate
 | |
|         return TRUE;        
 | |
| 
 | |
|     // If nothing returns TRUE, fail
 | |
|     return FALSE;
 | |
| }
 | |
| 
 | |
| int GetIsMysterySLA(object oShadow, int nMystId, int nClass)
 | |
| {
 | |
|     if (nClass == CLASS_TYPE_SHADOWSMITH) return FALSE;
 | |
|     int nLevel = GetMysteryLevel(oShadow, nMystId);
 | |
|     int nShadow = GetShadowcasterLevel(oShadow);
 | |
|     
 | |
|     if (nShadow >= 13 && 7 > nLevel && nLevel > 3)
 | |
|         return TRUE;
 | |
|     else if (nShadow >= 7 && 4 > nLevel)
 | |
|         return TRUE;   
 | |
|     else if (nShadow < 7 && 4 > nLevel && GetHasFavoredMystery(oShadow, nMystId)) // Bump up Apprentice
 | |
|         return TRUE;
 | |
|     if (nShadow < 13 && nShadow >= 7 && 7 > nLevel && nLevel > 3 && GetHasFavoredMystery(oShadow, nMystId)) // Bump up Initiate
 | |
|         return TRUE; 
 | |
|     if (nShadow >= 13 && nLevel >= 7 && GetHasFavoredMystery(oShadow, nMystId)) // Bump up Master
 | |
|         return TRUE;        
 | |
| 
 | |
|     // If nothing returns TRUE, fail
 | |
|     return FALSE;
 | |
| }
 | |
| 
 | |
| int GetIsFundamental(int nMystId)
 | |
| {
 | |
|     if(nMystId == FUND_ARROW_DUSK ||
 | |
|        nMystId == FUND_BLACK_CANDLE_LIGHT ||
 | |
|        nMystId == FUND_BLACK_CANDLE_DARK  ||
 | |
|        nMystId == FUND_CAUL_SHADOW        ||
 | |
|        nMystId == FUND_MYSTIC_REFLECTIONS ||
 | |
|        nMystId == FUND_SHADOW_HOOD        ||
 | |
|        nMystId == FUND_SIGHT_OBSCURED     ||
 | |
|        nMystId == FUND_UMBRAL_HAND        ||
 | |
|        nMystId == FUND_WIDENED_EYES)
 | |
|         return TRUE;
 | |
| 
 | |
|     return FALSE;
 | |
| }
 | |
| 
 | |
| int GetHasFavoredMystery(object oShadow, int nMyst)
 | |
| {
 | |
|     if (DEBUG) DoDebug("GetHasFavoredMystery(): Mystery "+IntToString(nMyst));
 | |
|     int nFavored, nReturn;
 | |
|     switch(nMyst)
 | |
|     {
 | |
|         case MYST_BEND_PERSPECTIVE     : nFavored = FEAT_FAV_MYST_BENDPERSPECTIVE        ; break;
 | |
|         case MYST_CARPET_SHADOW        : nFavored = FEAT_FAV_MYST_CARPETSHADOW           ; break;
 | |
|         case MYST_DUSK_AND_DAWN_DUSK   : nFavored = FEAT_FAV_MYST_DUSKANDDAWN            ; break;
 | |
|         case MYST_DUSK_AND_DAWN_DAWN   : nFavored = FEAT_FAV_MYST_DUSKANDDAWN            ; break;
 | |
|         case MYST_LIFE_FADES           : nFavored = FEAT_FAV_MYST_LIFEFADES              ; break;
 | |
|         case MYST_MESMERIZING_SHADE    : nFavored = FEAT_FAV_MYST_MESMERIZINGSHADE       ; break;
 | |
|         case MYST_STEEL_SHADOWS        : nFavored = FEAT_FAV_MYST_STEELSHADOWS           ; break;
 | |
|         case MYST_VOICE_SHADOW_APPROACH: nFavored = FEAT_FAV_MYST_VOICEOFSHADOW          ; break;
 | |
|         case MYST_VOICE_SHADOW_DROP    : nFavored = FEAT_FAV_MYST_VOICEOFSHADOW          ; break;
 | |
|         case MYST_VOICE_SHADOW_FALL    : nFavored = FEAT_FAV_MYST_VOICEOFSHADOW          ; break;
 | |
|         case MYST_VOICE_SHADOW_FLEE    : nFavored = FEAT_FAV_MYST_VOICEOFSHADOW          ; break;
 | |
|         case MYST_VOICE_SHADOW_HALT    : nFavored = FEAT_FAV_MYST_VOICEOFSHADOW          ; break;
 | |
|         case MYST_BLACK_FIRE           : nFavored = FEAT_FAV_MYST_BLACKFIRE              ; break;
 | |
|         case MYST_CONGRESS_SHADOWS     : nFavored = FEAT_FAV_MYST_CONGRESSSHADOWS        ; break;
 | |
|         case MYST_FLESH_FAILS_STR      : nFavored = FEAT_FAV_MYST_FLESHFAILS             ; break;
 | |
|         case MYST_FLESH_FAILS_DEX      : nFavored = FEAT_FAV_MYST_FLESHFAILS             ; break;
 | |
|         case MYST_FLESH_FAILS_CON      : nFavored = FEAT_FAV_MYST_FLESHFAILS             ; break;
 | |
|         case MYST_PIERCING_SIGHT       : nFavored = FEAT_FAV_MYST_PIERCINGSIGHT          ; break;
 | |
|         case MYST_SHADOW_SKIN          : nFavored = FEAT_FAV_MYST_SHADOWSKIN             ; break;
 | |
|         case MYST_SIGHT_ECLIPSED       : nFavored = FEAT_FAV_MYST_SIGHTECLIPSED          ; break;   
 | |
|         case MYST_THOUGHTS_SHADOW_INT  : nFavored = FEAT_FAV_MYST_THOUGHTSSHADOW         ; break;
 | |
|         case MYST_THOUGHTS_SHADOW_WIS  : nFavored = FEAT_FAV_MYST_THOUGHTSSHADOW         ; break;
 | |
|         case MYST_THOUGHTS_SHADOW_CHA  : nFavored = FEAT_FAV_MYST_THOUGHTSSHADOW         ; break;
 | |
|         case MYST_AFRAID_DARK          : nFavored = FEAT_FAV_MYST_AFRAIDOFTHEDARK        ; break;
 | |
|         case MYST_CLINGING_DARKNESS    : nFavored = FEAT_FAV_MYST_CLINGINGDARKNESS       ; break;
 | |
|         case MYST_DANCING_SHADOWS      : nFavored = FEAT_FAV_MYST_DANCINGSHADOWS         ; break;
 | |
|         case MYST_FLICKER              : nFavored = FEAT_FAV_MYST_FLICKER                ; break;
 | |
|         case MYST_KILLING_SHADOWS      : nFavored = FEAT_FAV_MYST_KILLINGSHADOWS         ; break;
 | |
|         case MYST_SHARP_SHADOWS        : nFavored = FEAT_FAV_MYST_SHARPSHADOWS           ; break;
 | |
|         case MYST_UMBRAL_TOUCH         : nFavored = FEAT_FAV_MYST_UMBRALTOUCH            ; break;
 | |
|         case MYST_AURA_OF_SHADE        : nFavored = FEAT_FAV_MYST_AURAOFSHADE            ; break;
 | |
|         case MYST_BOLSTER              : nFavored = FEAT_FAV_MYST_BOLSTER                ; break;
 | |
|         case MYST_SHADOW_EVOCATION     : nFavored = FEAT_FAV_MYST_SHADOWEVOCATION        ; break;
 | |
|         case MYST_SHADOW_VISION        : nFavored = FEAT_FAV_MYST_SHADOWVISION           ; break;
 | |
|         case MYST_SHADOWS_FADE         : nFavored = FEAT_FAV_MYST_SHADOWSFADE            ; break;
 | |
|         case MYST_STEP_SHADOW_SELF     : nFavored = FEAT_FAV_MYST_STEPINTOSHADOW         ; break;
 | |
|         case MYST_STEP_SHADOW_PARTY    : nFavored = FEAT_FAV_MYST_STEPINTOSHADOW         ; break;
 | |
|         case MYST_WARP_SPELL           : nFavored = FEAT_FAV_MYST_WARPSPELL              ; break;
 | |
|         case MYST_CURTAIN_SHADOWS      : nFavored = FEAT_FAV_MYST_CURTAINSHADOWS         ; break;  
 | |
|         case MYST_DARK_AIR             : nFavored = FEAT_FAV_MYST_DARKAIR                ; break;
 | |
|         case MYST_ECHO_SPELL           : nFavored = FEAT_FAV_MYST_FEIGNLIFE              ; break;
 | |
|         case MYST_FEIGN_LIFE           : nFavored = FEAT_FAV_MYST_DARKSOUL               ; break;
 | |
|         case MYST_LANGUOR_SLOW         : nFavored = FEAT_FAV_MYST_LANGUOR                ; break;
 | |
|         case MYST_LANGUOR_HOLD         : nFavored = FEAT_FAV_MYST_LANGUOR                ; break;
 | |
|         case MYST_PASS_SHADOW_SELF     : nFavored = FEAT_FAV_MYST_PASSINTOSHADOW         ; break;
 | |
|         case MYST_PASS_SHADOW_PARTY    : nFavored = FEAT_FAV_MYST_PASSINTOSHADOW         ; break;
 | |
|         case MYST_UNRAVEL_DWEOMER      : nFavored = FEAT_FAV_MYST_UNRAVELDWEOMER         ; break;
 | |
|         case MYST_FLOOD_SHADOW         : nFavored = FEAT_FAV_MYST_FLOODSHADOWS           ; break;
 | |
|         case MYST_GREATER_SHADOW_EVO   : nFavored = FEAT_FAV_MYST_GREATERSHADOWEVOCATION ; break;
 | |
|         case MYST_SHADOW_INVESTITURE   : nFavored = FEAT_FAV_MYST_SHADOWINVESTITURE      ; break;
 | |
|         case MYST_SHADOW_STORM         : nFavored = FEAT_FAV_MYST_SHADOWSTORM            ; break;
 | |
|         case MYST_SHADOWS_FADE_GREATER : nFavored = FEAT_FAV_MYST_SHADOWSFADE_GREATER    ; break;
 | |
|         case MYST_UNVEIL               : nFavored = FEAT_FAV_MYST_UNVEIL                 ; break;
 | |
|         case MYST_VOYAGE_SHADOW_SELF   : nFavored = FEAT_FAV_MYST_VOYAGESHADOW           ; break;
 | |
|         case MYST_VOYAGE_SHADOW_PARTY  : nFavored = FEAT_FAV_MYST_VOYAGESHADOW           ; break;
 | |
|         case MYST_DARK_SOUL            : nFavored = FEAT_FAV_MYST_DARKSOUL               ; break;
 | |
|         case MYST_EPHEMERAL_IMAGE      : nFavored = FEAT_FAV_MYST_EPHEMERALIMAGE         ; break;
 | |
|         case MYST_LIFE_FADES_GREATER   : nFavored = FEAT_FAV_MYST_LIFEFADESGREATER       ; break;   
 | |
|         case MYST_PRISON_NIGHT         : nFavored = FEAT_FAV_MYST_PRISONNIGHT            ; break;
 | |
|         case MYST_UMBRAL_SERVANT       : nFavored = FEAT_FAV_MYST_UMBRALSERVANT          ; break;
 | |
|         case MYST_TRUTH_REVEALED       : nFavored = FEAT_FAV_MYST_TRUTHREVEALED          ; break;
 | |
|         case MYST_FAR_SIGHT            : nFavored = FEAT_FAV_MYST_FARSIGHT               ; break;  
 | |
|         case MYST_GR_FLESH_FAILS_STR   : nFavored = FEAT_FAV_MYST_GRFLESHFAILS           ; break;
 | |
|         case MYST_GR_FLESH_FAILS_DEX   : nFavored = FEAT_FAV_MYST_GRFLESHFAILS           ; break;
 | |
|         case MYST_GR_FLESH_FAILS_CON   : nFavored = FEAT_FAV_MYST_GRFLESHFAILS           ; break;
 | |
|         case MYST_SHADOW_PLAGUE        : nFavored = FEAT_FAV_MYST_SHADOWPLAGUE           ; break;
 | |
|         case MYST_SOUL_PUPPET          : nFavored = FEAT_FAV_MYST_SOULPUPPET             ; break;
 | |
|         case MYST_TOMB_NIGHT           : nFavored = FEAT_FAV_MYST_TOMBNIGHT              ; break;
 | |
|         case MYST_UMBRAL_BODY          : nFavored = FEAT_FAV_MYST_UMBRALBODY             ; break;
 | |
|         case MYST_ARMY_SHADOW          : nFavored = FEAT_FAV_MYST_ARMYSHADOW             ; break;
 | |
|         case MYST_CONSUME_ESSENCE      : nFavored = FEAT_FAV_MYST_CONSUMEESSENCE         ; break;
 | |
|         case MYST_EPHEMERAL_STORM      : nFavored = FEAT_FAV_MYST_EPHEMERALSTORM         ; break;
 | |
|         case MYST_REFLECTIONS          : nFavored = FEAT_FAV_MYST_REFLECTIONS            ; break;
 | |
|         case MYST_SHADOW_SURGE         : nFavored = FEAT_FAV_MYST_SHADOWSURGE            ; break;
 | |
|         case MYST_SHADOW_TIME          : nFavored = FEAT_FAV_MYST_SHADOWTIME             ; break;    
 | |
|         case MYST_QUICKER_THAN_THE_EYE : nFavored = FEAT_FAV_MYST_QUICKERTHANTHEEYE      ; break;
 | |
|         case MYST_TRAIL_OF_HAZE        : nFavored = FEAT_FAV_MYST_TRAILHAZE              ; break;
 | |
|         case MYST_UMBRAL_FIST          : nFavored = FEAT_FAV_MYST_UMBRALFIST             ; break;
 | |
|         case MYST_FEARFUL_GLOOM        : nFavored = FEAT_FAV_MYST_FEARFULGLOOM           ; break;
 | |
|         case MYST_SICKENING_SHADOW     : nFavored = FEAT_FAV_MYST_SICKENINGSHADOW        ; break;
 | |
|         case MYST_DEADLY_SHADE_DR      : nFavored = FEAT_FAV_MYST_DEADLYSHADE            ; break;
 | |
|         case MYST_DEADLY_SHADE_NEG     : nFavored = FEAT_FAV_MYST_DEADLYSHADE            ; break;
 | |
|         case MYST_GRASPING_SHADOWS     : nFavored = FEAT_FAV_MYST_GRASPINGSHADOWS        ; break;
 | |
|         case MYST_MENAGERIE_OF_DARKNESS: nFavored = FEAT_FAV_MYST_MENAGERIEDARKNESS      ; break;
 | |
|         case MYST_BLACK_LABYRINTH      : nFavored = FEAT_FAV_MYST_BLACKLABYRINTH         ; break;           
 | |
|     }
 | |
|     if(GetHasFeat(nFavored, oShadow))
 | |
|         nReturn = 1;
 | |
| 
 | |
|     // If none of those trigger.
 | |
|     return nReturn;
 | |
| }
 | |
| 
 | |
| int GetShadowCast(object oShadow)
 | |
| {
 | |
|     if (DEBUG) DoDebug("GetShadowCast() enter");
 | |
|     if (!GetHasFeat(FEAT_SHADOW_CAST, oShadow)) return FALSE; // Need the feat
 | |
|     
 | |
|     location lTarget = GetLocation(oShadow);
 | |
|     int nCount, nReturn;
 | |
| 
 | |
|     // Use the function to get the closest creature as a target
 | |
|     object oCount = MyFirstObjectInShape(SHAPE_SPHERE, RADIUS_SIZE_SMALL, lTarget, TRUE, OBJECT_TYPE_CREATURE);
 | |
|     while(GetIsObjectValid(oCount))
 | |
|     {
 | |
|         if(GetIsEnemy(oCount, oShadow) && GetIsInMeleeRange(oShadow, oCount)) // Must be an enemy in melee range
 | |
|         {
 | |
|             nCount++;
 | |
|             if (DEBUG) DoDebug("GetShadowCast() nCount "+IntToString(nCount));
 | |
|         }
 | |
|     //Select the next target within the spell shape.
 | |
|     oCount = MyNextObjectInShape(SHAPE_SPHERE, RADIUS_SIZE_SMALL, lTarget, TRUE, OBJECT_TYPE_CREATURE);
 | |
|     } 
 | |
|     
 | |
|     if (nCount > 1) return FALSE;
 | |
| 
 | |
|     return TRUE;
 | |
| } |