PRC8/nwn/nwnprc/trunk/include/psi_inc_powknown.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

719 lines
30 KiB
Plaintext

//::///////////////////////////////////////////////
//:: Psionics include: Powers known
//:: psi_inc_powknown
//::///////////////////////////////////////////////
/** @file
Defines functions for adding & removing
powers known.
Data stored:
- For each power list
-- Total number of powers known
-- A modifier value to maximum powers known on this list to account for stuff like
Apopsi, Expanded Knowledge and Psychic Chirurgery that add or remove powers
-- An array related to powers the knowledge of which is not dependent on character level
--- Each array entry specifies the spells.2da row of the known power's class-specific entry
-- For each character level on which powers have been gained from this list
--- An array of powers gained on this level
---- Each array entry specifies the spells.2da row of the known power's class-specific entry
@author Ornedan
@date Created - 2006.01.06
@date Modified - 2006.04.21
Changed the stored value from cls_psipw_*.2da row to spells.2da row of the
class-specific entry because the adding policy to cls_psipw_*.2da means that the rows
after the added entry are moved. Which would break compatibility with old characters.
On the other hand, if the spells.2da entries are moved, the result would likely be a
backwards-compatibility loss anyway.
*/
//:://////////////////////////////////////////////
//:://////////////////////////////////////////////
//////////////////////////////////////////////////
/* Constants */
//////////////////////////////////////////////////
// Constants are provided via psi_inc_core
//////////////////////////////////////////////////
/* Function prototypes */
//////////////////////////////////////////////////
/**
* Gives the creature the control feats for the given power and marks the power
* in a powers known array.
* If the power's data is already stored in one of the powers known arrays for
* the list or adding the power's data to the array fails, the function aborts.
*
* @param oCreature The creature to gain the power
* @param nList The list the power comes from. One of POWER_LIST_*
* @param n2daRow The 2da row in the lists's 2da file that specifies the power.
* @param bLevelDependent If this is TRUE, the power is tied to a certain level and can
* be lost via level loss. If FALSE, the power is not dependent
* of a level and cannot be lost via level loss.
* @param nLevelToTieTo If bLevelDependent is TRUE, this specifies the level the power
* is gained on. Otherwise, it's ignored.
* The default value (-1) means that the current level of oCreature
* will be used.
*
* @return TRUE if the power was successfully stored and control feats added.
* FALSE otherwise.
*/
int AddPowerKnown(object oCreature, int nList, int n2daRow, int bLevelDependent = FALSE, int nLevelToTieTo = -1, int nExpandedKnowledge = 0);
/**
*
*/
//int RemoveSpecificPowerKnown(object oCreature, int nList, int n2daRow);
/**
* Removes all powers gained from each list on the given level.
*
* @param oCreature The creature whose powers to remove
* @param nLevel The level to clear
*/
void RemovePowersKnownOnLevel(object oCreature, int nLevel);
/**
* Gets the value of the powers known modifier, which is a value that is added
* to the 2da-specified maximum powers known to determine the actual maximum.
*
* @param oCreature The creature whose modifier to get
* @param nList The list the maximum powers known from which the modifier
* modifies. One of POWER_LIST_*
*/
int GetKnownPowersModifier(object oCreature, int nList);
/**
* Sets the value of the powers known modifier, which is a value that is added
* to the 2da-specified maximum powers known to determine the actual maximum.
*
* @param oCreature The creature whose modifier to set
* @param nList The list the maximum powers known from which the modifier
* modifies. One of POWER_LIST_*
*/
void SetKnownPowersModifier(object oCreature, int nList, int nNewValue);
/**
* Gets the number of powers a character character possesses from a
* specific list.
*
* @param oCreature The creature whose powers to check
* @param nList The list to check. One of POWER_LIST_*
* @return The number of powers known oCreature has from nList
*/
int GetPowerCount(object oCreature, int nList);
/**
* Gets the maximum number of powers a character may posses from a given list
* at this time. Calculated based on class levels, feats and a misceallenous
* modifier.
*
* @param oCreature Character to determine maximum powers for
* @param nList POWER_LIST_* of the list to determine maximum powers for
* @return Maximum number of powers that oCreature may know from the given list.
*/
int GetMaxPowerCount(object oCreature, int nList);
/**
* Determines whether a character has a given power, gained via some power list.
* Psi-like abilities do not count.
*
* @param nPower POWER_* of the power to test
* @param oCreature Character to test for the possession of the power
* @return TRUE if the character has the power, FALSE otherwise
*/
int GetHasPower(int nPower, object oCreature = OBJECT_SELF);
/**
* Converts a CLASS_TYPE_* constant to a corresponding POWER_LIST_* constant.
* Special: CLASS_TYPE_INVALID corresponds to POWER_LIST_EXP_KNOWLEDGE.
*
* @param nClassType CLASS_TYPE_* to determine POWER_LIST_* for
* @return POWER_LIST_* constant for nClassType. 0 if there is no
* power list for the given CLASS_TYPE_*
*/
//int ClassTypeToPowerList(int nClassType);
/**
* Converts a POWER_LIST_* constant to a corresponding CLASS_TYPE_* constant.
* Special: POWER_LIST_EXP_KNOWLEDGE corresponds to CLASS_TYPE_INVALID.
*
* @param nPowerList POWER_LIST_* to determine CLASS_TYPE_* for
* @return CLASS_TYPE_* constant for nPowerList. 0 if there is no
* class type for the given POWER_LIST_*
*/
//int PowerListToClassType(int nPowerList);
//////////////////////////////////////////////////
/* Includes */
//////////////////////////////////////////////////
#include "psi_inc_core"
//#include "prc_inc_spells" // Access to lookup and 2da
//////////////////////////////////////////////////
/* Internal functions */
//////////////////////////////////////////////////
void _RecurseRemoveArray(object oCreature, string sArrayName, string sPowerFile, int nArraySize, int nCurIndex)
{
if(DEBUG) DoDebug("_RecurseRemoveArray():\n"
+ "oCreature = " + DebugObject2Str(oCreature) + "\n"
+ "sArrayName = '" + sArrayName + "'\n"
+ "sPowerFile = '" + sPowerFile + "'\n"
+ "nArraySize = " + IntToString(nArraySize) + "\n"
+ "nCurIndex = " + IntToString(nCurIndex) + "\n"
);
// Determine whether we've already parsed the whole array or not
if(nCurIndex >= nArraySize)
{
if(DEBUG) DoDebug("_RecurseRemoveArray(): Running itemproperty removal loop.");
// Loop over itemproperties on the skin and remove each match
object oSkin = GetPCSkin(oCreature);
itemproperty ipTest = GetFirstItemProperty(oSkin);
while(GetIsItemPropertyValid(ipTest))
{
// Check if the itemproperty is a bonus feat that has been marked for removal
if(GetItemPropertyType(ipTest) == ITEM_PROPERTY_BONUS_FEAT &&
GetLocalInt(oCreature, "PRC_PowerFeatRemovalMarker_" + IntToString(GetItemPropertySubType(ipTest)))
)
{
if(DEBUG) DoDebug("_RecurseRemoveArray(): Removing bonus feat itemproperty:\n" + DebugIProp2Str(ipTest));
// If so, remove it
RemoveItemProperty(oSkin, ipTest);
}
ipTest = GetNextItemProperty(oSkin);
}
}
// Still parsing the array
else
{
// Set the marker
string sName = "PRC_PowerFeatRemovalMarker_" + Get2DACache(sPowerFile, "IPFeatID",
GetPowerfileIndexFromSpellID(persistant_array_get_int(oCreature, sArrayName, nCurIndex))
);
if(DEBUG) DoDebug("_RecurseRemoveArray(): Recursing through array, marker set:\n" + sName);
SetLocalInt(oCreature, sName, TRUE);
// Recurse to next array index
_RecurseRemoveArray(oCreature, sArrayName, sPowerFile, nArraySize, nCurIndex + 1);
// After returning, delete the local
DeleteLocalInt(oCreature, sName);
}
}
void _RemovePowerArray(object oCreature, int nList, int nLevel)
{
if(DEBUG) DoDebug("_RemovePowerArray():\n"
+ "oCreature = " + DebugObject2Str(oCreature) + "\n"
+ "nList = " + IntToString(nList) + "\n"
);
string sBase = _POWER_LIST_NAME_BASE + IntToString(nList);
string sArray = sBase + _POWER_LIST_LEVEL_ARRAY + IntToString(nLevel);
int nSize = persistant_array_get_size(oCreature, sArray);
// Reduce the total by the array size
SetPersistantLocalInt(oCreature, sBase + _POWER_LIST_TOTAL_KNOWN,
GetPersistantLocalInt(oCreature, sBase + _POWER_LIST_TOTAL_KNOWN) - nSize
);
// Remove each power in the array
_RecurseRemoveArray(oCreature, sArray, GetAMSDefinitionFileName(/*PowerListToClassType(*/nList/*)*/), nSize, 0);
// Remove the array itself
persistant_array_delete(oCreature, sArray);
}
//////////////////////////////////////////////////
/* Function definitions */
//////////////////////////////////////////////////
int AddPowerKnown(object oCreature, int nList, int n2daRow, int bLevelDependent = FALSE, int nLevelToTieTo = -1, int nExpandedKnowledge = 0)
{
string sBase = nExpandedKnowledge < 0 ? _POWER_LIST_NAME_BASE + IntToString(nExpandedKnowledge):
_POWER_LIST_NAME_BASE + IntToString(nList);
string sArray = sBase;
string sPowerFile = GetAMSDefinitionFileName(/*PowerListToClassType(*/nList/*)*/);
string sTestArray;
int i, j, nSize, bReturn;
// Get the spells.2da row corresponding to the cls_psipw_*.2da row
int nSpells2daRow = StringToInt(Get2DACache(sPowerFile, "SpellID", n2daRow));
// Determine the array name.
if(bLevelDependent)
{
// If no level is specified, default to the creature's current level
if(nLevelToTieTo == -1)
nLevelToTieTo = GetHitDice(oCreature);
sArray += _POWER_LIST_LEVEL_ARRAY + IntToString(nLevelToTieTo);
}
else
{
sArray += _POWER_LIST_GENERAL_ARRAY;
}
// Make sure the power isn't already in an array. If it is, abort and return FALSE
// Loop over each level array and check that it isn't there.
for(i = 1; i <= GetHitDice(oCreature); i++)
{
sTestArray = sBase + _POWER_LIST_LEVEL_ARRAY + IntToString(i);
if(persistant_array_exists(oCreature, sTestArray))
{
nSize = persistant_array_get_size(oCreature, sTestArray);
for(j = 0; j < nSize; j++)
if(persistant_array_get_int(oCreature, sArray, j) == nSpells2daRow)
return FALSE;
}
}
// Check the non-level-dependent array
sTestArray = sBase + _POWER_LIST_GENERAL_ARRAY;
if(persistant_array_exists(oCreature, sTestArray))
{
nSize = persistant_array_get_size(oCreature, sTestArray);
for(j = 0; j < nSize; j++)
if(persistant_array_get_int(oCreature, sArray, j) == nSpells2daRow)
return FALSE;
}
// All checks are made, now start adding the new power
// Create the array if it doesn't exist yet
if(!persistant_array_exists(oCreature, sArray))
persistant_array_create(oCreature, sArray);
// Store the power in the array
if(persistant_array_set_int(oCreature, sArray, persistant_array_get_size(oCreature, sArray), nSpells2daRow) != SDL_SUCCESS)
{
if(DEBUG) DoDebug("psi_inc_powknown: AddPowerKnown(): ERROR: Unable to add power to known array\n"
+ "oCreature = " + DebugObject2Str(oCreature) + "\n"
+ "nList = " + IntToString(nList) + "\n"
+ "n2daRow = " + IntToString(n2daRow) + "\n"
+ "bLevelDependent = " + DebugBool2String(bLevelDependent) + "\n"
+ "nLevelToTieTo = " + IntToString(nLevelToTieTo) + "\n"
+ "nSpells2daRow = " + IntToString(nSpells2daRow) + "\n"
+ "nExpandedKnowledge = " + IntToString(nExpandedKnowledge) + "\n"
);
return FALSE;
}
// Increment powers known total
SetPersistantLocalInt(oCreature, sBase + _POWER_LIST_TOTAL_KNOWN,
GetPersistantLocalInt(oCreature, sBase + _POWER_LIST_TOTAL_KNOWN) + 1
);
// Give the power's control feats
object oSkin = GetPCSkin(oCreature);
string sPowerFeatIP = Get2DACache(sPowerFile, "IPFeatID", n2daRow);
itemproperty ipFeat = PRCItemPropertyBonusFeat(StringToInt(sPowerFeatIP));
IPSafeAddItemProperty(oSkin, ipFeat, 0.0f, X2_IP_ADDPROP_POLICY_KEEP_EXISTING, FALSE, FALSE);
// Second power feat, if any
sPowerFeatIP = Get2DACache(sPowerFile, "IPFeatID2", n2daRow);
if(sPowerFeatIP != "")
{
ipFeat = PRCItemPropertyBonusFeat(StringToInt(sPowerFeatIP));
IPSafeAddItemProperty(oSkin, ipFeat, 0.0f, X2_IP_ADDPROP_POLICY_KEEP_EXISTING, FALSE, FALSE);
}
return TRUE;
}
/** @todo If ever required
int RemoveSpecificPowerKnown(object oCreature, int nList, int n2daRow)
{
}
*/
void RemovePowersKnownOnLevel(object oCreature, int nLevel)
{
if(DEBUG) DoDebug("psi_inc_powknown: RemovePowersKnownOnLevel():\n"
+ "oCreature = " + DebugObject2Str(oCreature) + "\n"
+ "nLevel = " + IntToString(nLevel) + "\n"
);
string sPostFix = _POWER_LIST_LEVEL_ARRAY + IntToString(nLevel);
// For each power list, determine if an array exists for this level.
if(persistant_array_exists(oCreature, _POWER_LIST_NAME_BASE + IntToString(POWER_LIST_PSION) + sPostFix))
// If one does exist, clear it
_RemovePowerArray(oCreature, POWER_LIST_PSION, nLevel);
if(persistant_array_exists(oCreature, _POWER_LIST_NAME_BASE + IntToString(POWER_LIST_WILDER) + sPostFix))
_RemovePowerArray(oCreature, POWER_LIST_WILDER, nLevel);
if(persistant_array_exists(oCreature, _POWER_LIST_NAME_BASE + IntToString(POWER_LIST_PSYWAR) + sPostFix))
_RemovePowerArray(oCreature, POWER_LIST_PSYWAR, nLevel);
if(persistant_array_exists(oCreature, _POWER_LIST_NAME_BASE + IntToString(POWER_LIST_PSYROG) + sPostFix))
_RemovePowerArray(oCreature, POWER_LIST_PSYROG, nLevel);
if(persistant_array_exists(oCreature, _POWER_LIST_NAME_BASE + IntToString(POWER_LIST_FIST_OF_ZUOKEN) + sPostFix))
_RemovePowerArray(oCreature, POWER_LIST_FIST_OF_ZUOKEN, nLevel);
if(persistant_array_exists(oCreature, _POWER_LIST_NAME_BASE + IntToString(POWER_LIST_WARMIND) + sPostFix))
_RemovePowerArray(oCreature, POWER_LIST_WARMIND, nLevel);
if(persistant_array_exists(oCreature, _POWER_LIST_NAME_BASE + IntToString(POWER_LIST_EXP_KNOWLEDGE) + sPostFix))
_RemovePowerArray(oCreature, POWER_LIST_EXP_KNOWLEDGE, nLevel);
if(persistant_array_exists(oCreature, _POWER_LIST_NAME_BASE + IntToString(POWER_LIST_EPIC_EXP_KNOWLEDGE) + sPostFix))
_RemovePowerArray(oCreature, POWER_LIST_EPIC_EXP_KNOWLEDGE, nLevel);
if(persistant_array_exists(oCreature, _POWER_LIST_NAME_BASE + IntToString(POWER_LIST_MISC) + sPostFix))
_RemovePowerArray(oCreature, POWER_LIST_MISC, nLevel);
}
int GetKnownPowersModifier(object oCreature, int nList)
{
return GetPersistantLocalInt(oCreature, _POWER_LIST_NAME_BASE + IntToString(nList) + _POWER_LIST_MODIFIER);
}
void SetKnownPowersModifier(object oCreature, int nList, int nNewValue)
{
SetPersistantLocalInt(oCreature, _POWER_LIST_NAME_BASE + IntToString(nList) + _POWER_LIST_MODIFIER, nNewValue);
}
int GetPowerCount(object oCreature, int nList)
{
return GetPersistantLocalInt(oCreature, _POWER_LIST_NAME_BASE + IntToString(nList) + _POWER_LIST_TOTAL_KNOWN);
}
int GetMaxPowerCount(object oCreature, int nList)
{
int nMaxPowers = 0;
switch(nList)
{
case POWER_LIST_PSION:{
// Determine base powers known
int nLevel = GetLevelByClass(CLASS_TYPE_PSION, oCreature);
nLevel += GetPsionicPRCLevels(oCreature, CLASS_TYPE_PSION);
//nLevel += GetPrimaryPsionicClass(oCreature) == CLASS_TYPE_PSION ? GetPsionicPRCLevels(oCreature) : 0;
if(nLevel == 0)
break;
nMaxPowers = StringToInt(Get2DACache(GetAMSKnownFileName(CLASS_TYPE_PSION), "PowersKnown", nLevel - 1));
// Calculate feats
// Apply the epic feat Power Knowledge - +2 powers known per
int nFeat = FEAT_POWER_KNOWLEDGE_PSION_1;
while(nFeat <= FEAT_POWER_KNOWLEDGE_PSION_10 &&
GetHasFeat(nFeat, oCreature))
{
nMaxPowers += 2;
nFeat++;
}
// Add in the custom modifier
nMaxPowers += GetKnownPowersModifier(oCreature, nList);
break;
}
case POWER_LIST_WILDER:{
// Determine base powers known
int nLevel = GetLevelByClass(CLASS_TYPE_WILDER, oCreature);
nLevel += GetPsionicPRCLevels(oCreature, CLASS_TYPE_WILDER);
//nLevel += GetPrimaryPsionicClass(oCreature) == CLASS_TYPE_WILDER ? GetPsionicPRCLevels(oCreature) : 0;
if(nLevel == 0)
break;
nMaxPowers = StringToInt(Get2DACache(GetAMSKnownFileName(CLASS_TYPE_WILDER), "PowersKnown", nLevel - 1));
// Calculate feats
// Apply the epic feat Power Knowledge - +2 powers known per
int nFeat = FEAT_POWER_KNOWLEDGE_WILDER_1;
while(nFeat <= FEAT_POWER_KNOWLEDGE_WILDER_10 &&
GetHasFeat(nFeat, oCreature))
{
nMaxPowers += 2;
nFeat++;
}
// Add in the custom modifier
nMaxPowers += GetKnownPowersModifier(oCreature, nList);
break;
}
case POWER_LIST_PSYWAR:{
// Determine base powers known
int nLevel = GetLevelByClass(CLASS_TYPE_PSYWAR, oCreature);
nLevel += GetPsionicPRCLevels(oCreature, CLASS_TYPE_PSYWAR);
//nLevel += GetPrimaryPsionicClass(oCreature) == CLASS_TYPE_PSYWAR ? GetPsionicPRCLevels(oCreature) : 0;
if(nLevel == 0)
break;
nMaxPowers = StringToInt(Get2DACache(GetAMSKnownFileName(CLASS_TYPE_PSYWAR), "PowersKnown", nLevel - 1));
// Calculate feats
// Apply the epic feat Power Knowledge - +2 powers known per
int nFeat = FEAT_POWER_KNOWLEDGE_PSYWAR_1;
while(nFeat <= FEAT_POWER_KNOWLEDGE_PSYWAR_10 &&
GetHasFeat(nFeat, oCreature))
{
nMaxPowers += 2;
nFeat++;
}
// Add in the custom modifier
nMaxPowers += GetKnownPowersModifier(oCreature, nList);
break;
}
case POWER_LIST_PSYROG:{
// Determine base powers known
int nLevel = GetLevelByClass(CLASS_TYPE_PSYCHIC_ROGUE, oCreature);
nLevel += GetPsionicPRCLevels(oCreature, CLASS_TYPE_PSYCHIC_ROGUE);
//nLevel += GetPrimaryPsionicClass(oCreature) == CLASS_TYPE_PSYCHIC_ROGUE ? GetPsionicPRCLevels(oCreature) : 0;
if(nLevel == 0)
break;
nMaxPowers = StringToInt(Get2DACache(GetAMSKnownFileName(CLASS_TYPE_PSYCHIC_ROGUE), "PowersKnown", nLevel - 1));
// Calculate feats
// Apply the epic feat Power Knowledge - +2 powers known per
int nFeat = FEAT_POWER_KNOWLEDGE_PSYROG_1;
while(nFeat <= FEAT_POWER_KNOWLEDGE_PSYROG_10 &&
GetHasFeat(nFeat, oCreature))
{
nMaxPowers += 2;
nFeat++;
}
// Add in the custom modifier
nMaxPowers += GetKnownPowersModifier(oCreature, nList);
break;
}
case POWER_LIST_FIST_OF_ZUOKEN:{
// Determine base powers known
int nLevel = GetLevelByClass(CLASS_TYPE_FIST_OF_ZUOKEN, oCreature);
nLevel += GetPsionicPRCLevels(oCreature, CLASS_TYPE_FIST_OF_ZUOKEN);
//nLevel += GetPrimaryPsionicClass(oCreature) == CLASS_TYPE_FIST_OF_ZUOKEN ? GetPsionicPRCLevels(oCreature) : 0;
if(nLevel == 0)
break;
nMaxPowers = StringToInt(Get2DACache(GetAMSKnownFileName(CLASS_TYPE_FIST_OF_ZUOKEN), "PowersKnown", nLevel - 1));
// Calculate feats
// Apply the epic feat Power Knowledge - +2 powers known per
int nFeat = FEAT_POWER_KNOWLEDGE_FIST_OF_ZUOKEN_1;
while(nFeat <= FEAT_POWER_KNOWLEDGE_FIST_OF_ZUOKEN_10 &&
GetHasFeat(nFeat, oCreature))
{
nMaxPowers += 2;
nFeat++;
}
// Add in the custom modifier
nMaxPowers += GetKnownPowersModifier(oCreature, nList);
break;
}
case POWER_LIST_WARMIND:{
// Determine base powers known
int nLevel = GetLevelByClass(CLASS_TYPE_WARMIND, oCreature);
nLevel += GetPsionicPRCLevels(oCreature, CLASS_TYPE_WARMIND);
//nLevel += GetPrimaryPsionicClass(oCreature) == CLASS_TYPE_WARMIND ? GetPsionicPRCLevels(oCreature) : 0;
if(nLevel == 0)
break;
nMaxPowers = StringToInt(Get2DACache(GetAMSKnownFileName(CLASS_TYPE_WARMIND), "PowersKnown", nLevel - 1));
// Calculate feats
// Apply the epic feat Power Knowledge - +2 powers known per
int nFeat = FEAT_POWER_KNOWLEDGE_WARMIND_1;
while(nFeat <= FEAT_POWER_KNOWLEDGE_WARMIND_10 &&
GetHasFeat(nFeat, oCreature))
{
nMaxPowers += 2;
nFeat++;
}
// Add in the custom modifier
nMaxPowers += GetKnownPowersModifier(oCreature, nList);
break;
}
case POWER_LIST_EXP_KNOWLEDGE:{
// Calculate feats
// Apply the feat Expanded Knowledge - +1 powers known per feat
int nFeat = FEAT_EXPANDED_KNOWLEDGE_1;
while(nFeat <= FEAT_EXPANDED_KNOWLEDGE_10 &&
GetHasFeat(nFeat, oCreature))
{
nMaxPowers += 1;
nFeat++;
}
// Add in the custom modifier
nMaxPowers += GetKnownPowersModifier(oCreature, nList);
break;
}
case POWER_LIST_EPIC_EXP_KNOWLEDGE:{
int nFeat = FEAT_EPIC_EXPANDED_KNOWLEDGE_1;
while(nFeat <= FEAT_EPIC_EXPANDED_KNOWLEDGE_10 &&
GetHasFeat(nFeat, oCreature))
{
nMaxPowers += 1;
nFeat++;
}
// Add in the custom modifier
nMaxPowers += GetKnownPowersModifier(oCreature, nList);
break;
}
case POWER_LIST_MISC:{
//no limits here, should not be checked
}
default:{
string sErr = "GetMaxPowerCount(): ERROR: Unknown power list value: " + IntToString(nList);
if(DEBUG) DoDebug(sErr);
else WriteTimestampedLogEntry(sErr);
}
}
return nMaxPowers;
}
int GetHasPower(int nPower, object oCreature = OBJECT_SELF)
{
if((GetLevelByClass(CLASS_TYPE_PSION, oCreature)
&& GetHasFeat(GetClassFeatFromPower(nPower, CLASS_TYPE_PSION), oCreature)
) ||
(GetLevelByClass(CLASS_TYPE_PSYWAR, oCreature)
&& GetHasFeat(GetClassFeatFromPower(nPower, CLASS_TYPE_PSYWAR), oCreature)
) ||
(GetLevelByClass(CLASS_TYPE_PSYCHIC_ROGUE, oCreature)
&& GetHasFeat(GetClassFeatFromPower(nPower, CLASS_TYPE_PSYCHIC_ROGUE), oCreature)
) ||
(GetLevelByClass(CLASS_TYPE_WILDER, oCreature)
&& GetHasFeat(GetClassFeatFromPower(nPower, CLASS_TYPE_WILDER), oCreature)
) ||
(GetLevelByClass(CLASS_TYPE_FIST_OF_ZUOKEN, oCreature)
&& GetHasFeat(GetClassFeatFromPower(nPower, CLASS_TYPE_FIST_OF_ZUOKEN), oCreature)
) ||
(GetLevelByClass(CLASS_TYPE_WARMIND, oCreature)
&& GetHasFeat(GetClassFeatFromPower(nPower, CLASS_TYPE_WARMIND), oCreature)
)
// add new psionic classes here
)
return TRUE;
return FALSE;
}
string DebugListKnownPowers(object oCreature)
{
string sReturn = "Powers known by " + DebugObject2Str(oCreature) + ":\n";
int i, j, k, numPowerLists = 8;
int nPowerList, nSize;
string sTemp, sArray, sArrayBase, sPowerFile;
// Loop over all power lists
for(i = 1; i <= numPowerLists; i++)
{
// Some padding
sReturn += " ";
// Get the power list for this loop
switch(i)
{
case 1: nPowerList = POWER_LIST_PSION; sReturn += "Psion"; break;
case 2: nPowerList = POWER_LIST_WILDER; sReturn += "Wilder"; break;
case 3: nPowerList = POWER_LIST_PSYWAR; sReturn += "Psychic Warrior"; break;
case 4: nPowerList = POWER_LIST_PSYROG; sReturn += "Psychic Rogue"; break;
case 5: nPowerList = POWER_LIST_FIST_OF_ZUOKEN; sReturn += "Fist of Zuoken"; break;
case 6: nPowerList = POWER_LIST_WARMIND; sReturn += "Warmind"; break;
// This should always be last
case 7: nPowerList = POWER_LIST_EXP_KNOWLEDGE; sReturn += "Expanded Knowledge"; break;
case 8: nPowerList = POWER_LIST_EPIC_EXP_KNOWLEDGE; sReturn += "Epic Expanded Knowledge"; break;
case 9: nPowerList = POWER_LIST_MISC; sReturn += "Misceallenous"; break;
}
sReturn += " powers known:\n";
// Determine if the character has any powers from this list
sPowerFile = GetAMSDefinitionFileName(nPowerList);
sArrayBase = _POWER_LIST_NAME_BASE + IntToString(nPowerList);
// Loop over levels
for(j = 1; j <= GetHitDice(oCreature); j++)
{
sArray = sArrayBase + _POWER_LIST_LEVEL_ARRAY + IntToString(j);
if(persistant_array_exists(oCreature, sArray))
{
sReturn += " Gained on level " + IntToString(j) + ":\n";
nSize = persistant_array_get_size(oCreature, sArray);
for(k = 0; k < nSize; k++)
sReturn += " " + GetStringByStrRef(StringToInt(Get2DACache("spells", "Name",
GetPowerFromSpellID(persistant_array_get_int(oCreature, sArray, k))
)
)
)
+ "\n";
}
}
// Non-leveldependent powers
sArray = sArrayBase + _POWER_LIST_GENERAL_ARRAY;
if(persistant_array_exists(oCreature, sArray))
{
sReturn += " Non-leveldependent:\n";
nSize = persistant_array_get_size(oCreature, sArray);
for(k = 0; k < nSize; k++)
sReturn += " " + GetStringByStrRef(StringToInt(Get2DACache("spells", "Name",
GetPowerFromSpellID(persistant_array_get_int(oCreature, sArray, k))
)
)
)
+ "\n";
}
}
return sReturn;
}
/* Probably unnecessary
int ClassTypeToPowerList(int nClassType)
{
int nReturn = 0;
switch(nClassType)
{
case CLASS_TYPE_PSION: nReturn = POWER_LIST_PSION; break;
case CLASS_TYPE_WILDER: nReturn = POWER_LIST_WILDER; break;
case CLASS_TYPE_PSYWAR: nReturn = POWER_LIST_PSYWAR; break;
case CLASS_TYPE_FIST_OF_ZUOKEN: nReturn = POWER_LIST_FIST_OF_ZUOKEN; break;
case CLASS_TYPE_INVALID: nReturn = POWER_LIST_EXP_KNOWLEDGE; break;
case -2: nReturn = POWER_LIST_EPIC_EXP_KNOWLEDGE; break;
case -3: nReturn = POWER_LIST_MISC; break;
default:{
string sErr = "ClassTypeToPowerList(): ERROR: Unknown class type value: " + IntToString(nClassType);
if(DEBUG) DoDebug(sErr);
else WriteTimestampedLogEntry(sErr);
}
}
return nReturn;
}
int PowerListToClassType(int nPowerList)
{
int nReturn = 0;
switch(nPowerList)
{
case POWER_LIST_PSION: nReturn = CLASS_TYPE_PSION; break;
case POWER_LIST_WILDER: nReturn = CLASS_TYPE_WILDER; break;
case POWER_LIST_PSYWAR: nReturn = CLASS_TYPE_PSYWAR; break;
case POWER_LIST_FIST_OF_ZUOKEN: nReturn = CLASS_TYPE_FIST_OF_ZUOKEN; break;
case POWER_LIST_EXP_KNOWLEDGE: nReturn = CLASS_TYPE_INVALID; break;
case POWER_LIST_EPIC_EXP_KNOWLEDGE: nReturn = -2; break;
case POWER_LIST_MISC: nReturn = -3; break;
default:{
string sErr = "PowerListToClassType(): ERROR: Unknown power list value: " + IntToString(nPowerList);
if(DEBUG) DoDebug(sErr);
else WriteTimestampedLogEntry(sErr);
}
}
return nReturn;
}
*/
// Test main
//void main(){}