WizardryEE/Module/nss/spawn_functions.nss
Jaysyn904 71a604d04a Added NESS. Updated Level 10
Added NESS.  Updated Level 10.  Full compile.
2023-04-20 23:24:24 -04:00

3333 lines
105 KiB
Plaintext

//
// NESS V8.1.3
// Spawn Functions
//
// Do NOT Modify this File
// See 'spawn__readme' for Instructions
//
#include "spawn_timefuncs"
#include "spawn_flags"
#include "spawn_cfg_cusflg"
#include "x0_i0_corpses"
void InitFlags(object oSpawn, string sSpawnName, string sSpawnTag);
int SetSpawns(location lBase);
string PadIntToString(int nInt, int nDigits);
void ResetSpawn(object oSpawn, int nTimeNow);
int IsRestoreBlocked(object oSpawn, location lChildLoc, int iExpireTime,
int nTimeNow);
void SetupSpawned(object oSpawn, object oSpawned, location lHome, int nTimeNow,
int nWalkToHome = FALSE);
void SetupCampSpawned(object oSpawn, object oSpawned, vector vCampPosition,
location lHome, string sFlags);
// Returns new nNextSpawnTime
int SetupSpawnDelay(int nSpawnDelay, int nDelayMinimum, int nDelayRandom,
int nTimeNow);
// Writes all the necessary info onto a spawn and its child after spawning
void RecordSpawned(object oSpawn, object oSpawned, location lHome,
location lEntranceExit, float fSpawnedFacing);
// Saves the state of one child onto the spawn (or a camp object) for respawning
void SaveStateOnDespawn(object oSpawned, object oSpawn, int nCamp=FALSE);
// Saves a camp object onto the spawn for respawning
void SaveCampStateOnDespawn(object oCamp, object oSpawn);
// Respawns all saved children/camps
void RestorePCDespawns(object oSpawn, int nTimeNow);
void ReturnHome(location lHome);
int FindNextEmptyChildSlot(object oSpawn);
//
// Pseudo-heartbeat support
//
const string SPAWN_INTERVAL = "Spawn_Interval";
const string SPAWN_PCS_IN_AREA = "Spawn_PCsInArea";
const string SPAWN_AREA_COUNT = "AreaSpawnCount";
const string SPAWN_HEARTBEAT_SCRIPT = "SpawnHeartbeatScript";
const string SPAWN_HEARTBEAT_SCHEDULED = "SpawnHeartbeatScheduled";
// This checks conditions to determine if a pseudo-heartbeat should be called
int NeedPseudoHeartbeat( object oArea );
// ... and if it should, this schedules it.
void ScheduleNextPseudoHeartbeat( object oArea );
// Pseudo-heartbeat area enter and exit functions
void Spawn_OnAreaEnter( string sHeartbeatScript = "spawn_sample_hb",
float fHeartbeatInterval = 6.0, float fFirstDelay = 0.0 );
void Spawn_OnAreaExit();
// Externals
void LootTable(object oSpawn, object oSpawned, int nLootTable);
effect ObjectEffect(object oSpawn);
int SPAWN_DELAY_DEBUG = FALSE;
int SPAWN_COUNT_DEBUG = FALSE;
int CONSOLE_DEBUG = TRUE;
void SpawnDelayDebug(object oSpawn, string str)
{
if (SPAWN_DELAY_DEBUG)
{
WriteTimestampedLogEntry("[sd " + GetName(GetArea(oSpawn)) + "] " +
GetLocalString(oSpawn, "f_Template") + " (" + ObjectToString(oSpawn) + "): " + str);
if (CONSOLE_DEBUG)
{
SendMessageToAllDMs("[sd " + GetName(GetArea(oSpawn)) + "] "
+ GetLocalString(oSpawn, "f_Template") + " (" + ObjectToString(oSpawn) + "): " + str);
}
/*
object oPC = GetFirstPC();
if (! GetIsDM(oPC))
SendMessageToPC(oPC, str);
*/
}
}
void SpawnCountDebug(object oSpawn, string str)
{
if (SPAWN_COUNT_DEBUG)
{
WriteTimestampedLogEntry("[sc " + GetName(GetArea(oSpawn)) + "] " +
GetLocalString(oSpawn, "f_Template") + " (" +ObjectToString(oSpawn) + "): " + str);
if (CONSOLE_DEBUG)
{
SendMessageToAllDMs("[sc " + GetName(GetArea(oSpawn)) + "] "
+ GetLocalString(oSpawn, "f_Template") + " (" +ObjectToString(oSpawn) + "): " + str);
}
/*
object oPC = GetFirstPC();
if (! GetIsDM(oPC))
SendMessageToPC(oPC, str);
*/
}
}
//
// Custom Functions
//
object GetChildByTag(object oSpawn, string sChildTag);
object GetChildByNumber(object oSpawn, int nChildNum);
object GetSpawnByID(int nSpawnID);
void DeactivateSpawn(object oSpawn);
void DeactivateSpawnsByTag(string sSpawnTag);
void DeactivateAllSpawns();
void DespawnChildren(object oSpawn);
void DespawnChildrenByTag(object oSpawn, string sSpawnTag);
void AddChild(object oSpawn, object oSpawned);
void ReportSpawns(int nAreaSpawns, int nModuleSpawns);
void TrackModuleSpawns(int nAreaSpawnCount, int nTrackModuleSpawns);
void DumpModuleSpawns();
void DumpModuleSpawns();
//
// PC and NPC Functions
//
int CountPCsInArea(object oArea = OBJECT_INVALID, int nDM = FALSE);
int CountPCsInRadius(location lCenter, float fRadius, int nDM = FALSE);
object GetRandomPCInArea(object oArea, object oSpawn);
// This struct is returned by TransferAllInventorySlots() to return back the
// object ids of the armor, right hand weapon, and left hand weapons (if any)
// both left on the (original) corpse and copied to the lootable corpse
struct NESS_CorpseInfo
{
object origLftWpn;
object origRgtWpn;
object origArmor;
object lootLftWpn;
object lootRgtWpn;
object lootArmor;
};
// This is equivalent to (and adapted from) the Bioware LootInventorySlots() function.
// However, it does not schedule deletions for a hardcoded
// corpse decay time later (the corpse decay script handles those).
struct NESS_CorpseInfo TransferAllInventorySlots(object oCreature, object oTarget,
int bDropWielded=FALSE);
int IsCreatureItem(object oItem);
void RandomWalk(object oSpawn, float fWalkingRadius, int nRun);
void FindSeat(object oSpawn, object oSpawned);
void NESS_CleanInventory(object oSpawned);
// for back compatibility (grr) will remove at some point
void CleanInventory(object oSpawned)
{
NESS_CleanInventory(oSpawned);
}
void NESS_CleanEquipped(object oSpawned);
void NESS_CleanCorpse(object oSpawned);
object NESS_CopyCorpseItem(object oSource, object oInventory);
void StripNonDroppables(object oSpawned);
void DestroyIfNonDrop(object oItem);
//
// Date and Time Functions
//
int IsBetweenDays(int nCheckDay, int nDayStart, int nDayEnd);
int IsBetweenHours(int nCheckHour, int nHourStart, int nHourEnd);
//
// Patrol Route Functions
//
void SetPatrolRoute(int nPatrolRoute, int nStartClosest=FALSE);
void DoPatrolRoute(int nPatrolRoute, int nRouteType);
void AddPatrolStop(int nPatrolRoute, int nStopNumber, int bJump=FALSE);
void CheckForStuckPatrol(object oCreature, int nPatrolRoute, int nRouteType);
//
// Camp Functions
//
int ProcessCamp(object oCamp);
void DestroyCamp(object oCamp, float fCampDecay, int nSaveState);
//
// Outside Functions
//
void SpawnFlags(object oSpawn, int nFlagTableNumber);
//
//
// Functions for External Use.
object NESS_GetSpawnByID(int nSpawnID, object oArea);
void NESS_ActivateSpawnByID(int nSpawnID, object oAreaD);
void NESS_DeactivateSpawnByID(int nSpawnID, object oArea);
void NESS_ActivateSpawn(object oSpawn);
void NESS_DeactivateSpawn(object oSpawn);
void NESS_ForceProcess(object oSpawn);
void NESS_TrackModuleSpawns(int flag=TRUE);
int NESS_IsModuleSpawnTracking();
void NESS_DumpModuleSpawns(int flag=TRUE);
int NESS_IsModuleSpawnDumping();
void NESS_ReturnHome(object oCreature, int bRun=FALSE);
void NESS_ProcessDeadCreature(object oCreature, object oSpawn=OBJECT_INVALID);
// This Function Initializes the Flags
void InitFlags(object oSpawn, string sSpawnName, string sSpawnTag)
{
// These are true when certain flags are present, false otherwise
// Retreive Defaults
object oModule = GetModule();
// These have values associated with them, although in some cases value
// of 0 is treated as a non-existent flag
int dfProcessFrequency = GetLocalInt(oModule, "df_processFrequency");
int dfProcessOffset = GetLocalInt(oModule, "df_processOffest");
int dfInitialState = GetLocalInt(oModule, "df_InitialState");
int dfInitialDelay = GetLocalInt(oModule, "df_InitialDelay");
int dfFlagTableNumber = GetLocalInt(oModule, "df_FlagTableNumber");
int dfSpawnDelay = GetLocalInt(oModule, "df_SpawnDelay");
int dfDelayMinimum = GetLocalInt(oModule, "df_DelayMinimum");
int dfSpawnNumber = GetLocalInt(oModule, "df_SpawnNumber");
int dfSpawnNumberMin = GetLocalInt(oModule, "df_SpawnNumberMin");
int dfSpawnNumberAtOnce = GetLocalInt(oModule, "df_SpawnNumberAtOnce");
int dfSpawnNumberAtOnceMin = GetLocalInt(oModule, "df_SpawnNumberAtOnceMin");
int dfSpawnDayStart = GetLocalInt(oModule, "df_SpawnDayStart");
int dfSpawnDayEnd = GetLocalInt(oModule, "df_SpawnDayEnd");
int dfSpawnHourStart = GetLocalInt(oModule, "df_SpawnHourStart");
int dfSpawnHourEnd = GetLocalInt(oModule, "df_SpawnHourEnd");
int dfWanderRange = GetLocalInt(oModule, "df_WanderRange");
int dfReturnHomeRange = GetLocalInt(oModule, "df_ReturnHomeRange");
int dfPCCheckDelay = GetLocalInt(oModule, "df_PCCheckDelay");
int dfRandomGold = GetLocalInt(oModule, "df_RandomGold");
int dfRandomGoldMin = GetLocalInt(oModule, "df_RandomGoldMin");
int dfGoldChance = GetLocalInt(oModule, "df_GoldChance");
int dfSpawnEffect = GetLocalInt(oModule, "df_SpawnEffect");
int dfDespawnEffect = GetLocalInt(oModule, "df_DespawnEffect");
int dfPatrolRoute = GetLocalInt(oModule, "df_PatrolRoute");
int dfRouteType = GetLocalInt(oModule, "df_RouteType");
int dfPlaceableType = GetLocalInt(oModule, "df_PlaceableType");
int dfTrapDisabled = GetLocalInt(oModule, "df_TrapDisabled");
int dfPlaceableRefreshPeriod = GetLocalInt(oModule, "df_PlaceableRefreshPeriod");
int dfLootTable = GetLocalInt(oModule, "df_LootTable");
int dfLootTable1ItemChance = GetLocalInt(oModule, "df_LootTable1ItemChance");
int dfLootTable2ItemChance = GetLocalInt(oModule, "df_LootTable2ItemChance");
int dfLootTable3ItemChance = GetLocalInt(oModule, "df_LootTable3ItemChance");
int dfDeactivateSpawn = GetLocalInt(oModule, "df_DeactivateSpawn");
int dfDeactivateScript = GetLocalInt(oModule, "df_DeactivateScript");
int dfDeactivationInfo = GetLocalInt(oModule, "df_DeactivationInfo");
int dfChildLifespanMax = GetLocalInt(oModule, "df_ChildLifespanMax");
int dfChildLifespanMin = GetLocalInt(oModule, "df_ChildLifespanMin");
int dfSpawnRadius = GetLocalInt(oModule, "df_SpawnRadius");
int dfSpawnRadiusMin = GetLocalInt(oModule, "df_SpawnRadiusMin");
int dfSpawnUnseen = GetLocalInt(oModule, "df_SpawnUnseen");
int dfUnseenRetryCount = GetLocalInt(oModule, "df_dfUnseenRetryCount");
int dfCorpseDecay = GetLocalInt(oModule, "df_CorpseDecay");
int dfCorpseDecayType = GetLocalInt(oModule, "df_CorpseDecayType");
int dfCorpseRemainsType = GetLocalInt(oModule, "df_CorpseRemainsType");
int dfCampDecay = GetLocalInt(oModule, "df_CampDecay");
int dfSpawnScript = GetLocalInt(oModule, "df_SpawnScript");
int dfDespawnScript = GetLocalInt(oModule, "df_DespawnScript");
int dfDeathScript = GetLocalInt(oModule, "df_DeathScript");
int dfSpawnCheckCustom = GetLocalInt(oModule, "df_SpawnCheckCustom");
int dfSpawnCheckPCs = GetLocalInt(oModule, "df_SpawnCheckPCs");
int dfCheckPCsRadius = GetLocalInt(oModule, "f_CheckPCsRadius");
int dfSpawnTrigger = GetLocalInt(oModule, "df_SpawnTrigger");
int dfDespawnTrigger = GetLocalInt(oModule, "df_DespawnTrigger");
int dfSpawnAreaEffect = GetLocalInt(oModule, "df_SpawnAreaEffect");
int dfAreaEffectDuration = GetLocalInt(oModule, "df_AreaEffectDuration");
int dfObjectEffect = GetLocalInt(oModule, "df_ObjectEffect");
int dfObjectEffectDuration = GetLocalInt(oModule, "df_ObjectEffectDuration");
int dfRandomSpawn = GetLocalInt(oModule, "df_RandomSpawn");
int dfSpawnFaction = GetLocalInt(oModule, "df_SpawnFaction");
int dfSpawnAlignment = GetLocalInt(oModule, "df_SpawnAlignment");
int dfAlignmentShift = GetLocalInt(oModule, "df_AlignmentShift");
int dfHeartbeatScript = GetLocalInt(oModule, "df_HeartbeatScript");
int dfSpawnLocation = GetLocalInt(oModule, "df_SpawnLocation");
int dfSpawnLocationMin = GetLocalInt(oModule, "df_SpawnLocationMin");
int dfSpawnFacing = GetLocalInt(oModule, "df_SpawnFacing");
int dfEntranceExit = GetLocalInt(oModule, "df_EntranceExit");
int dfEntranceExitMin = GetLocalInt(oModule, "df_EntranceExitMin");
int dfExit = GetLocalInt(oModule, "df_Exit");
int dfExitMin = GetLocalInt(oModule, "df_ExitMin");
int dfHealChildren = GetLocalInt(oModule, "df_HealChildren");
int dfGlobalSuppressDR = GetLocalInt(oModule, "df_GlobalSuppressDR");
int dfSuppressDR = GetLocalInt(oModule, "df_SuppressDR");
int dfEncounterLevel = GetLocalInt(oModule, "df_EncounterLevel");
//debug("init flags: " + sSpawnName);
SetLocalString(oSpawn, "f_Flags", sSpawnName);
SetLocalString(oSpawn, "f_Template", sSpawnTag);
// Initialize FlagTable
int nFlagTable = IsFlagPresent(sSpawnName, "_FT");
int nFlagTableNumber = GetFlagValue(sSpawnName, "_FT", dfFlagTableNumber);
if (nFlagTable == TRUE)
{
SpawnFlags(oSpawn, nFlagTableNumber);
if (GetStringLeft(GetLocalString(oSpawn, "f_Flags"), 2) == "SP")
{
sSpawnName = GetLocalString(oSpawn, "f_Flags");
}
else if (GetStringLeft(GetLocalString(oSpawn, "f_Flags"), 1) == "_")
{
sSpawnName = sSpawnName + GetLocalString(oSpawn, "f_Flags");
}
SetLocalString(oSpawn, "f_Flags", sSpawnName);
sSpawnTag = GetLocalString(oSpawn, "f_Template");
}
// Initialize CustomFlag
string sCustomFlag;
int nCustomFlag = IsFlagPresent(sSpawnName, "CF");
if (nCustomFlag == TRUE)
{
sCustomFlag = GetStringRight(sSpawnName, GetStringLength(sSpawnName) -
(FindSubString(sSpawnName, "CF") + 2));
sSpawnName = GetStringLeft(sSpawnName, GetStringLength(sSpawnName) -
(GetStringLength(sCustomFlag) + 3));
SetLocalString(oSpawn, "f_Flags", sSpawnName);
}
// Record CustomFlag
SetLocalString(oSpawn, "f_CustomFlag", sCustomFlag);
ParseCustomFlags(oSpawn, sCustomFlag);
// Initialize Process Frequency
int nProcessFrequency = GetFlagValue(sSpawnName, "SP", dfProcessFrequency);
int nProcessOffset = GetSubFlagValue(sSpawnName, "SP", "O", dfProcessOffset);
if (nProcessFrequency <= 0)
{
nProcessFrequency = 1;
}
// Record Process Frequency
SetLocalInt(oSpawn, "f_ProcessFrequency", nProcessFrequency);
SetLocalInt(oSpawn, "f_ProcessOffset", nProcessOffset);
// Initialize InitialState
int nInitialState = GetFlagValue(sSpawnName, "IS", dfInitialState);
int nInitialDelay = GetSubFlagValue(sSpawnName, "IS", "D", dfInitialDelay);
nInitialDelay *= 60; // convert to seconds
// Record InitialState
SetLocalInt(oSpawn, "f_InitialState", nInitialState);
SetLocalInt(oSpawn, "f_InitialDelay", nInitialDelay);
// Initialize SpawnID
int nSpawnID = GetFlagValue(sSpawnName, "ID", 0);
// Record SpawnID
if (nSpawnID > 0)
{
SetLocalInt(oSpawn, "SpawnID", nSpawnID);
}
// Initialize SpawnDelay
int nSpawnDelay = GetFlagValue(sSpawnName, "SD", dfSpawnDelay);
nSpawnDelay *= 60; // convert to seconds
int nDelayRandom = IsSubFlagPresent(sSpawnName, "SD", "M");
int nDelayMinimum = GetSubFlagValue(sSpawnName, "SD", "M", dfDelayMinimum);
nDelayMinimum *= 60; // convert to seconds
int nSpawnDelayPeriodic = IsSubFlagPresent(sSpawnName, "SD", "P");
if (nDelayMinimum > nSpawnDelay)
{
nDelayRandom = FALSE;
nDelayMinimum = 0;
}
// Record SpawnDelay
SetLocalInt(oSpawn, "f_SpawnDelay", nSpawnDelay);
SetLocalInt(oSpawn, "f_DelayRandom", nDelayRandom);
SetLocalInt(oSpawn, "f_DelayMinimum", nDelayMinimum);
SetLocalInt(oSpawn, "f_SpawnDelayPeriodic", nSpawnDelayPeriodic);
// Initialize SpawnNumber
int nSpawnNumber = GetFlagValue(sSpawnName, "SN", dfSpawnNumber);
int nSpawnNumberMax = nSpawnNumber;
int nSpawnNumberMin = GetSubFlagValue(sSpawnName, "SN", "M", dfSpawnNumberMin);
int nSpawnAllAtOnce = IsFlagPresent(sSpawnName, "SA");
int nSpawnNumberAtOnce = GetFlagValue(sSpawnName, "SA", dfSpawnNumberAtOnce);
int nSpawnNumberAtOnceMin = GetSubFlagValue(sSpawnName, "SA", "M",
dfSpawnNumberAtOnceMin);
if (nSpawnNumberMin > nSpawnNumber)
{
nSpawnNumberMin = -1;
}
if (nSpawnNumberMin > -1)
{
int nRndSpawnNumber = Random(nSpawnNumberMax + 1);
while (nRndSpawnNumber < nSpawnNumberMin)
{
nRndSpawnNumber = Random(nSpawnNumberMax + 1);
}
nSpawnNumber = nRndSpawnNumber;
}
if (nSpawnNumberAtOnce == 1)
{
nSpawnAllAtOnce = FALSE;
}
if (nSpawnNumberAtOnceMin > nSpawnNumberAtOnce)
{
nSpawnNumberAtOnceMin = 0;
}
// Record SpawnNumber
SetLocalInt(oSpawn, "f_SpawnNumber", nSpawnNumber);
SetLocalInt(oSpawn, "f_SpawnNumberMin", nSpawnNumberMin);
SetLocalInt(oSpawn, "f_SpawnNumberMax", nSpawnNumberMax);
SetLocalInt(oSpawn, "f_SpawnAllAtOnce", nSpawnAllAtOnce);
SetLocalInt(oSpawn, "f_SpawnNumberAtOnce", nSpawnNumberAtOnce);
SetLocalInt(oSpawn, "f_SpawnNumberAtOnceMin", nSpawnNumberAtOnceMin);
// Initialize Day/Night Only
int nDayOnly = IsFlagPresent(sSpawnName, "DO");
int nDayOnlyDespawn = IsSubFlagPresent(sSpawnName, "DO", "D");
int nNightOnly = IsFlagPresent(sSpawnName, "NO");
int nNightOnlyDespawn = IsSubFlagPresent(sSpawnName, "NO", "D");
// Record Day/Night Only
SetLocalInt(oSpawn, "f_DayOnly", nDayOnly);
SetLocalInt(oSpawn, "f_DayOnlyDespawn", nDayOnlyDespawn);
SetLocalInt(oSpawn, "f_NightOnly", nNightOnly);
SetLocalInt(oSpawn, "f_NightOnlyDespawn", nNightOnlyDespawn);
// Initialize Day/Hour Spawns
int nSpawnDayStart = GetFlagValue(sSpawnName, "DY", dfSpawnDayStart);
int nSpawnDayEnd = GetSubFlagValue(sSpawnName, "DY", "T", dfSpawnDayEnd);
if (nSpawnDayEnd > nSpawnDayStart)
{
nSpawnDayEnd = -1;
}
int nSpawnHourStart = GetFlagValue(sSpawnName, "HR", dfSpawnHourStart);
int nSpawnHourEnd = GetSubFlagValue(sSpawnName, "HR", "T", dfSpawnHourEnd);
if (nSpawnHourStart > nSpawnHourEnd)
{
nSpawnHourEnd = -1;
}
// Record Day/Hour Spawns
SetLocalInt(oSpawn, "f_SpawnDayStart", nSpawnDayStart);
SetLocalInt(oSpawn, "f_SpawnDayEnd", nSpawnDayEnd);
SetLocalInt(oSpawn, "f_SpawnHourStart", nSpawnHourStart);
SetLocalInt(oSpawn, "f_SpawnHourEnd", nSpawnHourEnd);
// Initialize RandomWalk
int nRandomWalk = IsFlagPresent(sSpawnName, "RW");
float fWanderRange = IntToFloat(GetSubFlagValue(sSpawnName, "RW", "R", dfWanderRange));
// Record RandomWalk
SetLocalInt(oSpawn, "f_RandomWalk", nRandomWalk);
SetLocalFloat(oSpawn, "f_WanderRange", fWanderRange);
// Initialize ReturnHome
int nReturnHome = IsFlagPresent(sSpawnName, "RH");
float fReturnHomeRange = IntToFloat(GetFlagValue(sSpawnName, "RH", dfReturnHomeRange));
// Record ReturnHome
SetLocalInt(oSpawn, "f_ReturnHome", nReturnHome);
SetLocalFloat(oSpawn, "f_ReturnHomeRange", fReturnHomeRange);
// Initialize PCCheck
int nPCCheck = IsFlagPresent(sSpawnName, "PC");
int nPCCheckDelay = GetFlagValue(sSpawnName, "PC", dfPCCheckDelay);
nPCCheckDelay *= 60; // convert to seconds
int nPCReset = IsSubFlagPresent(sSpawnName, "PC", "R");
// Record PCCheck
SetLocalInt(oSpawn, "f_PCCheck", nPCCheck);
SetLocalInt(oSpawn, "f_PCCheckDelay", nPCCheckDelay);
SetLocalInt(oSpawn, "f_PCReset", nPCReset);
// Initialize RandomGold
int nRandomGold = GetFlagValue(sSpawnName, "RG", dfRandomGold);
int nRandomGoldMin = GetSubFlagValue(sSpawnName, "RG", "M", dfRandomGoldMin);
int nGoldChance = GetSubFlagValue(sSpawnName, "RG", "C", dfGoldChance);
// Record RandomGold
SetLocalInt(oSpawn, "f_RandomGold", nRandomGold);
SetLocalInt(oSpawn, "f_RandomGoldMin", nRandomGoldMin);
SetLocalInt(oSpawn, "f_GoldChance", nGoldChance);
// Initialize SpawnEffects
int nSpawnEffect = GetFlagValue(sSpawnName, "FX", dfSpawnEffect);
int nDespawnEffect = GetSubFlagValue(sSpawnName, "FX", "D", dfDespawnEffect);
// Record SpawnEffects
SetLocalInt(oSpawn, "f_SpawnEffect", nSpawnEffect);
SetLocalInt(oSpawn, "f_DespawnEffect", nDespawnEffect);
// Initialize PatrolRoutes
int nPatrolRoute = GetFlagValue(sSpawnName, "PR", dfPatrolRoute);
int nRouteType = GetSubFlagValue(sSpawnName, "PR", "T", dfRouteType);
int nStartAtClosest = IsSubFlagPresent(sSpawnName, "PR", "C");
// Record PatrolRoutes
SetLocalInt(oSpawn, "f_PatrolRoute", nPatrolRoute);
SetLocalInt(oSpawn, "f_RouteType", nRouteType);
SetLocalInt(oSpawn, "f_PatrolStartAtClosest", nStartAtClosest);
// Initialize Placeables
int nPlaceable = IsFlagPresent(sSpawnName, "PL");
int nPlaceableType = GetFlagValue(sSpawnName, "PL", dfPlaceableType);
int nTrapDisabled = GetSubFlagValue(sSpawnName, "PL", "T", dfTrapDisabled);
int nPlaceableRefreshPeriod = GetSubFlagValue(sSpawnName, "PL", "P", dfPlaceableRefreshPeriod);
nPlaceableRefreshPeriod *= 60; // convert to seconds
// Record Placeables
SetLocalInt(oSpawn, "f_Placeable", nPlaceable);
SetLocalInt(oSpawn, "f_PlaceableType", nPlaceableType);
SetLocalInt(oSpawn, "f_TrapDisabled", nTrapDisabled);
SetLocalInt(oSpawn, "f_PlaceableRefreshPeriod", nPlaceableRefreshPeriod);
// Initialize SpawnGroups
int nSpawnGroup = IsFlagPresent(sSpawnName, "SG");
// Record SpawnGroups
SetLocalInt(oSpawn, "f_SpawnGroup", nSpawnGroup);
// Initialize LootTable
int nLootTable = GetFlagValue(sSpawnName, "LT", dfLootTable);
int nLootTable1ItemChance = GetSubFlagValue(sSpawnName, "LT", "A", dfLootTable1ItemChance);
int nLootTable2ItemChance = GetSubFlagValue(sSpawnName, "LT", "B", dfLootTable2ItemChance);
int nLootTable3ItemChance = GetSubFlagValue(sSpawnName, "LT", "C", dfLootTable3ItemChance);
if (nLootTable1ItemChance > 100) nLootTable1ItemChance = 100;
if (nLootTable2ItemChance > 100) nLootTable2ItemChance = 100;
if (nLootTable3ItemChance > 100) nLootTable3ItemChance = 100;
// Record LootTable
SetLocalInt(oSpawn, "f_LootTable", nLootTable);
SetLocalInt(oSpawn, "f_LootTable1ItemChance", nLootTable1ItemChance);
SetLocalInt(oSpawn, "f_LootTable2ItemChance", nLootTable2ItemChance);
SetLocalInt(oSpawn, "f_LootTable3ItemChance", nLootTable3ItemChance);
// Initialize SpawnDeactivation
int nDeactivateSpawn = GetFlagValue(sSpawnName, "DS", dfDeactivateSpawn);
int nDeactivateScript = GetSubFlagValue(sSpawnName, "DS", "S", dfDeactivateScript);
int nDeactivationInfo = GetFlagValue(sSpawnName, "DI", dfDeactivationInfo);
// Record SpawnDeactivations
SetLocalInt(oSpawn, "f_DeactivateSpawn", nDeactivateSpawn);
SetLocalInt(oSpawn, "f_DeactivateScript", nDeactivateScript);
if (nDeactivateSpawn == 4)
{
nDeactivationInfo *= 60; // convert minutes to seconds
}
else if (nDeactivateSpawn == 5)
{
nDeactivationInfo *= 6; // convert cycles to seconds
}
SetLocalInt(oSpawn, "f_DeactivationInfo", nDeactivationInfo);
// Initialize ChildLifespan
int nChildLifespanMax = GetFlagValue(sSpawnName, "CL", dfChildLifespanMax);
nChildLifespanMax *= 60; // convert to seconds
int nChildLifespanMin = GetSubFlagValue(sSpawnName, "CL", "M", dfChildLifespanMin);
nChildLifespanMin *= 60; // convert to seconds
if (nChildLifespanMin > nChildLifespanMax)
{
nChildLifespanMin = -1;
}
// Record ChildLifespan
SetLocalInt(oSpawn, "f_ChildLifespanMax", nChildLifespanMax);
SetLocalInt(oSpawn, "f_ChildLifespanMin", nChildLifespanMin);
// Initialize SpawnRadius
float fSpawnRadius = IntToFloat(GetFlagValue(sSpawnName, "SR", dfSpawnRadius));
float fSpawnRadiusMin = IntToFloat(GetSubFlagValue(sSpawnName, "SR", "M", dfSpawnRadiusMin));
int nSpawnNearPCs = IsSubFlagPresent(sSpawnName, "SR", "P");
if (fSpawnRadiusMin > fSpawnRadius)
{
fSpawnRadiusMin = 0.0;
}
// Record SpawnRadius
SetLocalFloat(oSpawn, "f_SpawnRadius", fSpawnRadius);
SetLocalFloat(oSpawn, "f_SpawnRadiusMin", fSpawnRadiusMin);
SetLocalInt(oSpawn, "f_SpawnNearPCs", nSpawnNearPCs);
// Initialize SpawnUnseen
float fSpawnUnseen = IntToFloat(GetFlagValue(sSpawnName, "SU", dfSpawnUnseen));
int nUnseenIndividual = IsSubFlagPresent(sSpawnName, "SU", "I");
int nUnseenRetryCount = GetSubFlagValue(sSpawnName, "SU", "I", dfUnseenRetryCount);
// Record SpawnUnseen
SetLocalFloat(oSpawn, "f_SpawnUnseen", fSpawnUnseen);
SetLocalInt(oSpawn, "f_UnseenIndividual", nUnseenIndividual);
SetLocalInt(oSpawn, "f_UnseenRetryCount", nUnseenRetryCount);
// Initialize CorpseDecay
float fCorpseDecay = IntToFloat(GetFlagValue(sSpawnName, "CD", dfCorpseDecay));
int nCorpseDecayType = GetSubFlagValue(sSpawnName, "CD", "T", dfCorpseDecayType);
int nCorpseRemainsType = GetSubFlagValue(sSpawnName, "CD", "R", dfCorpseRemainsType);
int bDropWielded = IsSubFlagPresent(sSpawnName, "CD", "D");
string sCorpseRemainsResRef;
int bDeleteLootOnDecay = FALSE;
switch (nCorpseRemainsType)
{
case 0: sCorpseRemainsResRef = "invis_corpse_obj"; break;
case 1: sCorpseRemainsResRef = "invis_corpse_bdy"; break;
case 2: sCorpseRemainsResRef = "invis_corpse_bon"; break;
case 3: sCorpseRemainsResRef = "invis_corpse_pot"; break;
case 4: sCorpseRemainsResRef = "invis_corpse_pch"; break;
case 5: sCorpseRemainsResRef = "invis_corpse_scr"; break;
case 6: sCorpseRemainsResRef = "invis_corpse_tre"; break;
case 7:
sCorpseRemainsResRef = "invis_corpse_obj";
bDeleteLootOnDecay = TRUE;
break;
}
// Record CorpseDecay
SetLocalFloat(oSpawn, "f_CorpseDecay", fCorpseDecay);
SetLocalInt(oSpawn, "f_CorpseDecayType", nCorpseDecayType);
SetLocalString(oSpawn, "f_CorpseRemainsResRef", sCorpseRemainsResRef);
SetLocalInt(oSpawn, "f_CorpseDropWielded", bDropWielded);
SetLocalInt(oSpawn, "f_CorpseDeleteLootOnDecay", bDeleteLootOnDecay);
// Initialize SpawnCamp
int nSpawnCamp = IsFlagPresent(sSpawnName, "CM");
float fCampDecay = IntToFloat(GetSubFlagValue(sSpawnName, "CM", "D", dfCampDecay));
// Record SpawnCamp
SetLocalInt(oSpawn, "f_SpawnCamp", nSpawnCamp);
SetLocalFloat(oSpawn, "f_CampDecay", fCampDecay);
// Initialize Spawn Scripts
int nSpawnScript = GetFlagValue(sSpawnName, "SS", dfSpawnScript);
int nDespawnScript = GetSubFlagValue(sSpawnName, "SS", "D", dfDespawnScript);
// Record Spawn Scripts
SetLocalInt(oSpawn, "f_SpawnScript", nSpawnScript);
SetLocalInt(oSpawn, "f_DespawnScript", nDespawnScript);
// Initialize Death Scripts
int nDeathScript = GetFlagValue(sSpawnName, "DT", dfDeathScript);
// Record Death Scripts
SetLocalInt(oSpawn, "f_DeathScript", nDeathScript);
// Initialize SpawnCheckCustom
int nSpawnCheckCustom = GetFlagValue(sSpawnName, "CC", dfSpawnCheckCustom);
// Record SpawnCheckCustom
SetLocalInt(oSpawn, "f_SpawnCheckCustom", nSpawnCheckCustom);
// Initialize SpawnCheckPCs
int nSpawnCheckPCs = GetFlagValue(sSpawnName, "CP", dfSpawnCheckPCs);
float fCheckPCsRadius = IntToFloat(GetSubFlagValue(sSpawnName, "CP", "R", dfCheckPCsRadius));
// Record SpawnCheckPCs
SetLocalInt(oSpawn, "f_SpawnCheckPCs", nSpawnCheckPCs);
SetLocalFloat(oSpawn, "f_CheckPCsRadius", fCheckPCsRadius);
// Intialize SpawnTrigger
float fSpawnTrigger = IntToFloat(GetFlagValue(sSpawnName, "TR", dfSpawnTrigger));
float fDespawnTrigger = IntToFloat(GetSubFlagValue(sSpawnName, "TR", "D", dfDespawnTrigger));
// Record SpawnTrigger
SetLocalFloat(oSpawn, "f_SpawnTrigger", fSpawnTrigger);
SetLocalFloat(oSpawn, "f_DespawnTrigger", fDespawnTrigger);
// Initialize AreaEffect
int nSpawnAreaEffect = GetFlagValue(sSpawnName, "AE", dfSpawnAreaEffect);
float fAreaEffectDuration = IntToFloat(GetSubFlagValue(sSpawnName, "AE", "D", dfAreaEffectDuration));
// Record AreaEffect
SetLocalInt(oSpawn, "f_SpawnAreaEffect", nSpawnAreaEffect);
SetLocalFloat(oSpawn, "f_AreaEffectDuration", fAreaEffectDuration);
// Initialize ObjectEffect
int nObjectEffect = GetFlagValue(sSpawnName, "OE", dfObjectEffect);
float fObjectEffectDuration = IntToFloat(GetSubFlagValue(sSpawnName, "OE", "D", dfObjectEffectDuration));
if (fObjectEffectDuration == 0.0)
{
fObjectEffectDuration = -1.0;
}
// Record ObjectEffect
SetLocalInt(oSpawn, "f_ObjectEffect", nObjectEffect);
SetLocalFloat(oSpawn, "f_ObjectEffectDuration", fObjectEffectDuration);
// Initialize RandomSpawn
int nRandomSpawn = GetFlagValue(sSpawnName, "RS", dfRandomSpawn);
// Record RandomSpawn
SetLocalInt(oSpawn, "f_RandomSpawn", nRandomSpawn);
// Initialize SpawnFaction
int nSpawnFaction = GetFlagValue(sSpawnName, "FC", dfSpawnFaction);
// Record SpawnFaction
SetLocalInt(oSpawn, "f_SpawnFaction", nSpawnFaction);
// Initialize SpawnAlignment
int nSpawnAlignment = GetFlagValue(sSpawnName, "AL", dfSpawnAlignment);
int nAlignmentShift = GetSubFlagValue(sSpawnName, "AL", "S", dfAlignmentShift);
// Record SpawnAlignment
SetLocalInt(oSpawn, "f_SpawnAlignment", nSpawnAlignment);
SetLocalInt(oSpawn, "f_AlignmentShift", nAlignmentShift);
// Initialize Heartbeat
int nHeartbeatScript = GetFlagValue(sSpawnName, "HB", dfHeartbeatScript);
// Record HeartBeat
SetLocalInt(oSpawn, "f_HeartbeatScript", nHeartbeatScript);
// Initialize SpawnLocation
int nSpawnLocation = GetFlagValue(sSpawnName, "SL", dfSpawnLocation);
int nSpawnLocationMin = GetSubFlagValue(sSpawnName, "SL", "R", dfSpawnLocationMin);
int nSpawnLocationInd = IsSubFlagPresent(sSpawnName, "SL", "I");
if (nSpawnLocationMin > nSpawnLocation)
{
nSpawnLocationMin = -1;
}
// Record SpawnLocation
SetLocalInt(oSpawn, "f_SpawnLocation", nSpawnLocation);
SetLocalInt(oSpawn, "f_SpawnLocationMin", nSpawnLocationMin);
SetLocalInt(oSpawn, "f_SpawnLocationInd", nSpawnLocationInd);
// Initialize SpawnFacing
float fSpawnFacing;
int nSpawnFacing = IsFlagPresent(sSpawnName, "SF");
if (nSpawnFacing == TRUE)
{
fSpawnFacing = GetFacingFromLocation(GetLocation(oSpawn));
}
else
{
// If f_Facing is false, fSpawnFacing is now calculated for
// each individual creature
fSpawnFacing = 0.0;
// fSpawnFacing = IntToFloat(Random(360));
}
// Record SpawnFacing
SetLocalInt(oSpawn, "f_Facing", nSpawnFacing);
SetLocalFloat(oSpawn, "f_SpawnFacing", fSpawnFacing);
// Initialize EntranceExit
int nEntranceExit = GetFlagValue(sSpawnName, "EE", dfEntranceExit);
int nEntranceExitMin = GetSubFlagValue(sSpawnName, "EE", "R", dfEntranceExitMin);
if (nEntranceExitMin > nEntranceExit)
{
nEntranceExitMin = -1;
}
int nExit = GetFlagValue(sSpawnName, "EX", dfExit);
int nExitMin = GetSubFlagValue(sSpawnName, "EX", "R", dfExitMin);
if (nExitMin > nExit)
{
nExitMin = -1;
}
// Record EntranceExit
SetLocalInt(oSpawn, "f_EntranceExit", nEntranceExit);
SetLocalInt(oSpawn, "f_EntranceExitMin", nEntranceExitMin);
SetLocalInt(oSpawn, "f_Exit", nExit);
SetLocalInt(oSpawn, "f_ExitMin", nExitMin);
// Initialize HealChildren
int nHealChildren = GetFlagValue(sSpawnName, "HL", dfHealChildren);
int nHealEffects = IsSubFlagPresent(sSpawnName, "HL", "E");
if (nHealChildren == 1)
{
nHealChildren == 100;
}
// Record HealChildren
SetLocalInt(oSpawn, "f_HealChildren", nHealChildren);
SetLocalInt(oSpawn, "f_HealEffects", nHealEffects);
// Initialize SpawnItem
int nSpawnItem = IsFlagPresent(sSpawnName, "IT");
// Record SpawnItem
SetLocalInt(oSpawn, "f_SpawnItem", nSpawnItem);
// Initialize SpawnSit
int nSpawnSit = IsFlagPresent(sSpawnName, "ST");
// Record SpawnSit
SetLocalInt(oSpawn, "f_SpawnSit", nSpawnSit);
// Initialize SpawnPlot
int nSpawnPlot = IsFlagPresent(sSpawnName, "PT");
// Record SpawnPlot
SetLocalInt(oSpawn, "f_SpawnPlot", nSpawnPlot);
// Initialize SpawnMerchant
int nSpawnMerchant = IsFlagPresent(sSpawnName, "SM");
// Record SpawnMerchant
SetLocalInt(oSpawn, "f_SpawnMerchant", nSpawnMerchant);
// Initialize Dim Returns Suppression
int nSuppressDimReturns = IsFlagPresent(sSpawnName, "SX");
if (nSuppressDimReturns)
{
// If the flag is present, get suppression mode from its value
nSuppressDimReturns = GetFlagValue(sSpawnName, "SX", dfSuppressDR);
}
else
{
// Use the global setting
nSuppressDimReturns = dfGlobalSuppressDR;
}
// Record Dim Returns Suppression
SetLocalInt(oSpawn, "f_SuppressDimReturns", nSuppressDimReturns);
// Initialize Loot Suppression
int nSuppressLooting = IsFlagPresent(sSpawnName, "NL");
// Record Loot Suppression
SetLocalInt(oSpawn, "f_SuppressLooting", nSuppressLooting);
// Initialize Subdual Mode
int nSubdualMode = IsFlagPresent(sSpawnName, "SB");
// Record Subdual Mode
SetLocalInt(oSpawn, "f_SubdualMode", nSubdualMode);
int nEncounterLevel;
// Initialize Encounter Level
if (IsFlagPresent(sSpawnName, "EL"))
{
nEncounterLevel = GetFlagValue(sSpawnName, "EL", dfEncounterLevel);
SetLocalInt(oSpawn, "f_EncounterLevel", nEncounterLevel);
}
// Record Flags Initialized
SetLocalInt(oSpawn, "FlagsInitialized", TRUE);
}
//
// This Function Sets the Spawns
int SetSpawns(location lBase)
{
string sSpawnName, sSpawnNum, sSpawnTag;
int nNth = 1;
int nSpawnNum = 0;
// Enumerate Waypoints in the Area
object oSpawn = GetFirstObjectInArea(OBJECT_SELF);
while (oSpawn != OBJECT_INVALID)
{
// Check for a local string called "NESS" on the waypoint
// first. If it exists, use it instead of the name
sSpawnName = GetLocalString(oSpawn, "NESS");
if (GetStringLeft(sSpawnName, 2) != "SP")
{
// Retrieve Name
sSpawnName = GetName(oSpawn);
}
// Check if Waypoint is a Spawn Controller
if (GetStringLeft(sSpawnName, 2) == "SP")
{
// Set Spawn
nSpawnNum++;
sSpawnNum = "Spawn" + PadIntToString(nSpawnNum, 2);
SetLocalObject(OBJECT_SELF, sSpawnNum, oSpawn);
sSpawnTag = GetLocalString(oSpawn, "NESS_TAG");
if (sSpawnTag == "")
{
sSpawnTag = GetTag(oSpawn);
}
DelayCommand(0.0, InitFlags(oSpawn, sSpawnName, sSpawnTag));
}
nNth++;
oSpawn = GetNextObjectInArea(OBJECT_SELF);
}
SetLocalInt(OBJECT_SELF, "Spawns", nSpawnNum);
return nSpawnNum;
}
//
// This Function returns the Number of PCs in an Area
int CountPCsInArea(object oArea = OBJECT_INVALID, int nDM = FALSE)
{
int retVal = 0;
if (oArea == OBJECT_INVALID)
{
oArea = GetArea(OBJECT_SELF);
}
object oPC = GetFirstPC();
while (oPC != OBJECT_INVALID)
{
if (GetArea(oPC) == oArea)
{
if (GetIsDM(oPC) == TRUE)
{
if (nDM == TRUE)
{
retVal++;
}
}
else
{
retVal++;
}
}
oPC = GetNextPC();
}
return retVal;
}
//
// This Function Returns the Number of PCs in a Radius
int CountPCsInRadius(location lCenter, float fRadius, int nDM = FALSE)
{
int nPCs = 0;
object oPC = GetFirstObjectInShape(SHAPE_SPHERE, fRadius, lCenter, FALSE,
OBJECT_TYPE_ALL);
while (oPC != OBJECT_INVALID)
{
if (GetIsPC(oPC))
{
if (GetIsDM(oPC))
{
//debug(GetName(oPC) + " is a DM ");
if (nDM == TRUE)
{
nPCs++;
}
}
else
{
//debug("found a real PC");
nPCs++;
}
}
oPC = GetNextObjectInShape(SHAPE_SPHERE, fRadius, lCenter, FALSE,
OBJECT_TYPE_ALL);
}
return nPCs;
}
//
int IsCreatureItem(object oItem)
{
if (GetBaseItemType(oItem) == BASE_ITEM_CREATUREITEM ||
GetBaseItemType(oItem) == BASE_ITEM_CBLUDGWEAPON ||
GetBaseItemType(oItem) == BASE_ITEM_CPIERCWEAPON ||
GetBaseItemType(oItem) == BASE_ITEM_CSLASHWEAPON ||
GetBaseItemType(oItem) == BASE_ITEM_CSLSHPRCWEAP)
{
return TRUE;
}
return FALSE;
}
// This Function Returns a Random PC from Area
object GetRandomPCInArea(object oArea, object oSpawn)
{
int nPCsInArea = CountPCsInArea(oArea, TRUE);
int nNth = Random(nPCsInArea) + 1;
object oRandomPC = GetNearestCreature(CREATURE_TYPE_PLAYER_CHAR, PLAYER_CHAR_IS_PC, oSpawn, nNth);
return oRandomPC;
}
object NESS_CopyCorpseItem(object oSource, object oInventory)
{
int bWasPlot = GetPlotFlag(oSource);
object oNewItem = CopyItem(oSource, oInventory);
if (bWasPlot == TRUE)
{
SetPlotFlag(oNewItem,TRUE);
}
return oNewItem;
}
//
struct NESS_CorpseInfo TransferAllInventorySlots(object oVictim,
object oCorpse, int bDropWielded=FALSE)
{
int i=0;
object oItem = OBJECT_INVALID;
location locItem;
struct NESS_CorpseInfo stCorpseInfo;
// Initialize
stCorpseInfo.origRgtWpn = OBJECT_INVALID;
stCorpseInfo.origLftWpn = OBJECT_INVALID;
stCorpseInfo.origArmor = OBJECT_INVALID;
stCorpseInfo.lootRgtWpn = OBJECT_INVALID;
stCorpseInfo.lootLftWpn = OBJECT_INVALID;
stCorpseInfo.lootArmor = OBJECT_INVALID;
for (i=0; i < NUM_INVENTORY_SLOTS; i++)
{
oItem = GetItemInSlot(i, oVictim);
// See if we're going to allow looting of this item.
if (GetIsObjectValid(oItem) && GetDroppableFlag(oItem))
{
// Handle different items slightly differently.
if (i == INVENTORY_SLOT_CHEST && GetIsVictimDressed(oVictim))
{
// The victim is wearing the armor. We don't want to destroy
// it while the corpse is around, since that would leave the
// body naked.
stCorpseInfo.origArmor = oItem;
stCorpseInfo.lootArmor = NESS_CopyCorpseItem(oItem, oCorpse);
}
else if (i == INVENTORY_SLOT_RIGHTHAND)
{
if (bDropWielded)
{
// This is a wielded item. Drop it nearby.
locItem = GetStepRightLocation(oVictim);
CreateObject(OBJECT_TYPE_ITEM, GetResRef(oItem), locItem);
DestroyObject(oItem, 0.1);
}
else
{
stCorpseInfo.origRgtWpn = oItem;
stCorpseInfo.lootRgtWpn = NESS_CopyCorpseItem(oItem, oCorpse);
}
}
else if (i == INVENTORY_SLOT_LEFTHAND)
{
if(bDropWielded)
{
// This is a wielded item. Drop it nearby.
locItem = GetStepLeftLocation(oVictim);
CreateObject(OBJECT_TYPE_ITEM, GetResRef(oItem), locItem);
DestroyObject(oItem, 0.1);
}
else
{
stCorpseInfo.origLftWpn = oItem;
stCorpseInfo.lootLftWpn = NESS_CopyCorpseItem(oItem, oCorpse);
}
}
else
{
NESS_CopyCorpseItem(oItem, oCorpse);
DestroyObject(oItem, 0.1);
}
}
}
return stCorpseInfo;
}
//
// This Function Checks if the Party is within fDistance Meters of Each Other
int IsPartyTogether(object oPC, float fDistance)
{
int nTogether = TRUE;
object oMember = GetFirstFactionMember(oPC, TRUE);
while (oMember != OBJECT_INVALID)
{
if (GetIsDead(oMember) == FALSE)
{
if (GetDistanceBetween(oPC, oMember) > fDistance)
{
nTogether = FALSE;
oMember = OBJECT_INVALID;
}
}
oMember = GetNextFactionMember(oPC, TRUE);
}
return nTogether;
}
//
// This Function Returns the Number of PCs in a Party
int CountMembersInParty(object oPC, int bPCOnly = TRUE)
{
int nCount;
object oMember = GetFirstFactionMember(oPC, bPCOnly);
while (oMember != OBJECT_INVALID)
{
nCount++;
oMember = GetNextFactionMember(oPC, bPCOnly);
}
return nCount;
}
//
// This Function Checks if nCheckDay is Between Days
int IsBetweenDays(int nCheckDay, int nDayStart, int nDayEnd)
{
if (nDayEnd > -1)
{
if (nCheckDay >= nDayStart && nCheckDay <= nDayEnd)
{
return TRUE;
}
}
else
{
if (nCheckDay == nDayStart)
{
return TRUE;
}
}
return FALSE;
}
//
// This Function Checks if nCheckHour is Between Hours
int IsBetweenHours(int nCheckHour, int nHourStart, int nHourEnd)
{
if (nHourEnd > -1)
{
if (nCheckHour >= nHourStart && nCheckHour <= nHourEnd)
{
return TRUE;
}
}
else
{
if (nCheckHour == nHourStart)
{
return TRUE;
}
}
return FALSE;
}
//
// This Function Pads an IntToString with 0s
string PadIntToString(int nInt, int nDigits)
{
string sRetString;
string sTempInt = IntToString(nInt);
int iCount;
sRetString = "";
for (iCount = 1; iCount <= (nDigits - GetStringLength(sTempInt)); iCount++)
{
sRetString = sRetString + "0";
}
sRetString = sRetString + sTempInt;
return sRetString;
}
//
// This Function returns a Child Object by Tag
object GetChildByTag(object oSpawn, string sChildTag)
{
object oChild;
object oRetChild = OBJECT_INVALID;
string sChildSlot;
int nChildSlot;
string sSpawnName = GetLocalString(oSpawn, "f_Flags");
int nSpawnNumber = GetFlagValue(sSpawnName, "SN", 1);
// Cycle through Children
for (nChildSlot = 1; nChildSlot <= nSpawnNumber; nChildSlot++)
{
// Retrieve Child
sChildSlot = "ChildSlot" + PadIntToString(nChildSlot, 2);
oChild = GetLocalObject(oSpawn, sChildSlot);
if (GetTag(oChild) == sChildTag)
{
oRetChild = oChild;
}
}
return oRetChild;
}
//
// This Function returns a Child Object by Slot Number
object GetChildByNumber(object oSpawn, int nChildNum)
{
object oRetChild = OBJECT_INVALID;
string sChildSlot;
string sSpawnName = GetLocalString(oSpawn, "f_Flags");
int nSpawnNumber = GetFlagValue(sSpawnName, "SN", 1);
// Check if Valid Number
if (nChildNum > nSpawnNumber)
{
return oRetChild;
}
// Retrieve Child
sChildSlot = "ChildSlot" + PadIntToString(nChildNum, 2);
oRetChild = GetLocalObject(oSpawn, sChildSlot);
// Return Child
return oRetChild;
}
//
object NESS_GetSpawnByID(int nSpawnID, object oArea)
{
string sSpawnName;
object oRetSpawn;
// Enumerate Waypoints in the Area
object oSpawn = GetFirstObjectInArea(oArea);
while (oSpawn != OBJECT_INVALID)
{
// Retrieve Name
sSpawnName = GetLocalString(oSpawn, "f_Flags");
// Check if Waypoint is a Spawn Controller
if (GetStringLeft(sSpawnName, 2) == "SP")
{
if (GetLocalInt(oSpawn, "SpawnID") == nSpawnID)
{
oRetSpawn = oSpawn;
}
}
oSpawn = GetNextObjectInArea(oArea);
}
return oRetSpawn;
}
//
// This Function returns a Spawn Object by ID
object GetSpawnByID(int nSpawnID)
{
return NESS_GetSpawnByID(nSpawnID, OBJECT_SELF);
}
//
// This Function Sets Children to Despawn
void DespawnChildren(object oSpawn)
{
object oChild;
string sChildSlot;
int nChildSlot;
string sSpawnName = GetLocalString(oSpawn, "f_Flags");
int nSpawnNumber = GetFlagValue(sSpawnName, "SN", 1);
// Cycle through Children
for (nChildSlot = 1; nChildSlot <= nSpawnNumber; nChildSlot++)
{
// Retrieve Child
sChildSlot = "ChildSlot" + PadIntToString(nChildSlot, 2);
oChild = GetLocalObject(oSpawn, sChildSlot);
SetLocalInt(oChild, "ForceDespawn", TRUE);
}
}
//
// This Function Sets Children to Despawn by Tag
void DespawnChildrenByTag(object oSpawn, string sChildTag)
{
object oChild;
string sChildSlot;
int nChildSlot;
string sSpawnName = GetLocalString(oSpawn, "f_Flags");
int nSpawnNumber = GetFlagValue(sSpawnName, "SN", 1);
// Cycle through Children
for (nChildSlot = 1; nChildSlot <= nSpawnNumber; nChildSlot++)
{
// Retrieve Child
sChildSlot = "ChildSlot" + PadIntToString(nChildSlot, 2);
oChild = GetLocalObject(oSpawn, sChildSlot);
if (GetTag(oChild) == sChildTag)
{
SetLocalInt(oChild, "ForceDespawn", TRUE);
}
}
}
//
// This Function Adds a Child to a Spawn
void AddChild(object oSpawn, object oSpawned)
{
// Declare Variables
int nEmptyChildSlot, nChildSlot;
int nSpawnNumber, nSpawnCount, nChildrenSpawned;
string sChildSlot, sEmptyChildSlot;
object oChild;
// Retreive Values
nSpawnNumber = GetLocalInt(oSpawn, "f_SpawnNumber");
nChildrenSpawned = GetLocalInt(oSpawn, "ChildrenSpawned");
nSpawnCount = GetLocalInt(oSpawn, "SpawnCount");
// Find Empty Child Slot
nEmptyChildSlot = 0;
for (nChildSlot = 1; nChildSlot <= nSpawnNumber; nChildSlot++)
{
// Retrieve Child
sChildSlot = "ChildSlot" + PadIntToString(nChildSlot, 2);
oChild = GetLocalObject(oSpawn, sChildSlot);
// Check if this is Child Slot is Valid
if (GetIsObjectValid(oChild) == FALSE || GetIsDead(oChild))
{
// Empty Slot
if (nEmptyChildSlot == 0)
{
nEmptyChildSlot = nChildSlot;
sEmptyChildSlot = sChildSlot;
}
}
}
if (nEmptyChildSlot != 0)
{
// Assign Values to oSpawned
SetLocalObject(oSpawned, "ParentSpawn", oSpawn);
SetLocalFloat(oSpawned, "HomeX", GetPositionFromLocation(GetLocation(oSpawned)).x);
SetLocalFloat(oSpawned, "HomeY", GetPositionFromLocation(GetLocation(oSpawned)).y);
// Assign Child Slot
SetLocalObject(oSpawn, sEmptyChildSlot, oSpawned);
SetLocalString(oSpawned, "ParentChildSlot", sEmptyChildSlot);
// Assign Values to oSpawn
nChildrenSpawned++;
SetLocalInt(oSpawn, "ChildrenSpawned", nChildrenSpawned);
nSpawnCount++;
SetLocalInt(oSpawn, "SpawnCount", nSpawnCount);
}
}
//
void NESS_DeactivateSpawnByID(int nSpawnID, object oArea)
{
object oSpawn = NESS_GetSpawnByID(nSpawnID, oArea);
NESS_DeactivateSpawn(oSpawn);
}
// Identical to DeactivateSpawn, but included for interface consistency
void NESS_DeactivateSpawn(object oSpawn)
{
DeactivateSpawn(oSpawn);
}
// This Function Sets a Spawn to Deactivate
void DeactivateSpawn(object oSpawn)
{
SetLocalInt(oSpawn, "ForceDeactivateSpawn", TRUE);
NESS_ForceProcess(oSpawn);
}
void NESS_ActivateSpawnByID(int nSpawnID, object oArea)
{
object oSpawn = NESS_GetSpawnByID(nSpawnID, oArea);
NESS_ActivateSpawn(oSpawn);
}
// This Function Sets a Spawn to Activate
void NESS_ActivateSpawn(object oSpawn)
{
SetLocalInt(oSpawn, "SpawnDeactivated", FALSE);
NESS_ForceProcess(oSpawn);
}
void NESS_ForceProcess(object oSpawn)
{
SetLocalInt(oSpawn, "SpawnForceProcess", TRUE);
}
void NESS_TrackModuleSpawns(int flag=TRUE)
{
SetLocalInt(GetModule(), "TrackModuleSpawns", flag);
}
int NESS_IsModuleSpawnTracking()
{
return GetLocalInt(GetModule(), "TrackModuleSpawns");
}
//
void NESS_DumpModuleSpawns(int flag=TRUE)
{
SetLocalInt(GetModule(), "DumpModuleSpawns", flag);
}
int NESS_IsModuleSpawnDumping()
{
return GetLocalInt(GetModule(), "DumpModuleSpawns");
}
//
// This Function Sets all Spawns by Tag to Deactivate
void DeactivateSpawnsByTag(string sSpawnTag)
{
int nNth;
object oSpawn;
string sSpawnNum;
int nSpawns = GetLocalInt(GetArea(OBJECT_SELF), "Spawns");
for (nNth = 1; nNth <= nSpawns; nNth++)
{
// Retrieve Spawn
sSpawnNum = "Spawn" + PadIntToString(nNth, 2);
oSpawn = GetLocalObject(OBJECT_SELF, sSpawnNum);
if (GetTag(oSpawn) == sSpawnTag)
{
SetLocalInt(oSpawn, "ForceDeactivateSpawn", TRUE);
}
}
}
//
// This Function Sets all Spawns to Deactivate
void DeactivateAllSpawns()
{
int nNth;
object oSpawn;
string sSpawnNum;
int nSpawns = GetLocalInt(GetArea(OBJECT_SELF), "Spawns");
for (nNth = 1; nNth <= nSpawns; nNth++)
{
// Retrieve Spawn
sSpawnNum = "Spawn" + PadIntToString(nNth, 2);
oSpawn = GetLocalObject(OBJECT_SELF, sSpawnNum);
SetLocalInt(oSpawn, "ForceDeactivateSpawn", TRUE);
}
}
//
location GetRandomLocationInRadius(object oSpawn, float fWalkingRadius)
{
vector vCurrentLocation, vTargetLocation;
float fRadiusX, fRadiusY;
location lTargetLocation;
// Create our Random Location
fRadiusX = IntToFloat(Random(FloatToInt(fWalkingRadius)));
fRadiusY = IntToFloat(Random(FloatToInt(fWalkingRadius)));
if (d2() == 2)
{
fRadiusX = -fRadiusX;
}
if (d2() == 2)
{
fRadiusY = -fRadiusY;
}
vTargetLocation = Vector(fRadiusX, fRadiusY);
vCurrentLocation = GetPositionFromLocation(GetLocation(oSpawn));
lTargetLocation = Location(OBJECT_SELF, vCurrentLocation + vTargetLocation,
0.0);
return lTargetLocation;
}
// Random Walking with Range
void RandomWalk(object oSpawn, float fWalkingRadius, int nRun)
{
// Walk to the New Location
float fRadiusX, fRadiusY;
// Create our Random Location
fRadiusX = IntToFloat(Random(FloatToInt(fWalkingRadius)));
fRadiusY = IntToFloat(Random(FloatToInt(fWalkingRadius)));
if (d2() == 2)
{
fRadiusX = -fRadiusX;
}
if (d2() == 2)
{
fRadiusY = -fRadiusY;
}
location lSpawnLocation = GetLocation(oSpawn);
vector vNewPosition = GetPositionFromLocation(lSpawnLocation);
vNewPosition.x += fRadiusX;
vNewPosition.y += fRadiusY;
location lRandomWalkLocation = Location(GetArea(OBJECT_SELF), vNewPosition,
0.0);
ActionMoveToLocation(lRandomWalkLocation, nRun);
}
//
// This Function Finds aaSeatnd Sits in It
void FindSeat(object oSpawn, object oSpawned)
{
object oSeat;
string sSeatTag = GetLocalString(GetModule(), "df_SeatTag");
location lSeatLocation;
int nNth = 1;
object oSittable = GetNearestObjectByTag(sSeatTag, oSpawned, nNth);
while (oSittable != OBJECT_INVALID && oSeat == OBJECT_INVALID)
{
if (GetSittingCreature(oSittable) == OBJECT_INVALID)
{
oSeat = oSittable;
}
nNth++;
oSittable = GetNearestObjectByTag(sSeatTag, oSpawned, nNth);
}
if (GetIsObjectValid(oSeat))
{
lSeatLocation = GetLocation(oSeat);
AssignCommand(oSpawned, ClearAllActions());
AssignCommand(oSpawned, ActionMoveToLocation(lSeatLocation));
AssignCommand(oSpawned, ActionSit(oSeat));
}
}
//
// This Function Cleans an Object's Inventory
void NESS_CleanCorpse(object oSpawned)
{
NESS_CleanEquipped(oSpawned);
NESS_CleanInventory(oSpawned);
}
void NESS_CleanEquipped(object oSpawned)
{
int i = 0;
object oItem = OBJECT_INVALID;
for (i=0; i < NUM_INVENTORY_SLOTS; i++)
{
oItem = GetItemInSlot(i, oSpawned);
if (GetIsObjectValid(oItem))
{
// Why the delay? This is used to remove equipped items from corpses. If the
// corpse is still around, it do it's unequip animation...
DestroyObject(oItem, 1.0);
oItem = GetNextItemInInventory(oSpawned);
}
}
}
void NESS_CleanInventory(object oSpawned)
{
// Clean out oSpawned's Inventory
//debug("in clean inventory");
object oItem = GetFirstItemInInventory(oSpawned);
while (oItem != OBJECT_INVALID)
{
//debug("destroying " + GetName(oItem));
DestroyObject(oItem);
oItem = GetNextItemInInventory(oSpawned);
}
}
//
// This Functions Sets up the Patrol Route
void SetPatrolRoute(int nPatrolRoute, int nStartClosest=FALSE)
{
object oStop;
int nRouteNumber, nStopNumber, iCount, nNumStops;
string sStop;
// These 3 vars only used if nStartClosest is TRUE;
float fLeastDistance = 9999999.0;// any distance returned should be smaller
float fCurrentDistance;
int nClosestStopNum;
// Cycle through Available Patrol Route Stops
iCount = 0;
nNumStops = 0;
oStop = GetNearestObject(OBJECT_TYPE_WAYPOINT, OBJECT_SELF, iCount);
while (oStop != OBJECT_INVALID)
{
sStop = GetTag(oStop);
// Check Route Number
nRouteNumber = GetFlagValue(sStop, "PR", -1);
if (nRouteNumber == nPatrolRoute)
{
// Identical Route Number, Add this Stop to oSpawned
nNumStops++;
nStopNumber = GetFlagValue(sStop, "SN", 0);
SetLocalObject(OBJECT_SELF, "PR_SN" + PadIntToString(nStopNumber, 2), oStop);
if (nStartClosest)
{
fCurrentDistance = GetDistanceToObject(oStop);
if (fCurrentDistance < fLeastDistance)
{
nClosestStopNum = nStopNumber;
fLeastDistance = fCurrentDistance;
}
}
}
iCount++;
oStop = GetNearestObject(OBJECT_TYPE_WAYPOINT, OBJECT_SELF, iCount);
}
SetLocalInt(OBJECT_SELF, "PR_STOPS", nNumStops);
if (nStartClosest && (nClosestStopNum > 0))
{
SetLocalInt(OBJECT_SELF, "PR_LASTSTOP", nClosestStopNum - 1);
// Force it to go to a new waypoint
SetLocalInt(OBJECT_SELF, "PR_NEXTSTOP", -1);
}
}
//
// This Function Performs the Patrol Route
void DoPatrolRoute(int nPatrolRoute, int nRouteType)
{
int nNumStops, nDespawn;
int nNextPatrolStop = -1;
int nLastPatrolStop = -1;
int nReturnRoute = FALSE;
if (GetCurrentAction() != ACTION_INVALID)
return;
ClearAllActions();
// Retreive Stop Information
nNumStops = GetLocalInt(OBJECT_SELF, "PR_STOPS");
nNextPatrolStop = GetLocalInt(OBJECT_SELF, "PR_NEXTSTOP");
nLastPatrolStop = GetLocalInt(OBJECT_SELF, "PR_LASTSTOP");
nReturnRoute = GetLocalInt(OBJECT_SELF, "PR_RETURNROUTE");
// Add New Stop to Route
if (nNextPatrolStop == -1)
{
// Sequential Route
if (nRouteType == 0 || nRouteType == 3)
{
if (nReturnRoute == TRUE)
{
if (nLastPatrolStop == 0)
{
nReturnRoute = FALSE;
SetLocalInt(OBJECT_SELF, "PR_RETURNROUTE", nReturnRoute);
nNextPatrolStop = nLastPatrolStop + 1;
}
else
{
nNextPatrolStop = nLastPatrolStop - 1;
}
}
else
{
if (nLastPatrolStop == nNumStops)
{
if (nRouteType == 3)
{
// End of Line, Despawn
nDespawn = TRUE;
}
else
{
nReturnRoute = TRUE;
SetLocalInt(OBJECT_SELF, "PR_RETURNROUTE", nReturnRoute);
nNextPatrolStop = nLastPatrolStop - 2;
}
}
else
{
nNextPatrolStop = nLastPatrolStop + 1;
}
}
}
// Circular Route
if (nRouteType == 1)
{
if (nLastPatrolStop == nNumStops)
{
nNextPatrolStop = 0;
}
else
{
nNextPatrolStop = nLastPatrolStop + 1;
}
}
// Random Route
if (nRouteType == 2)
{
nNextPatrolStop = Random(nNumStops);
while (nNextPatrolStop == nLastPatrolStop)
{
nNextPatrolStop = Random(nNumStops);
}
}
}
if (nDespawn == TRUE)
{
ClearAllActions();
SetLocalInt(OBJECT_SELF, "ForceDespawn", TRUE);
}
else
{
// Set Next Stop
SetLocalInt(OBJECT_SELF, "PR_NEXTSTOP", nNextPatrolStop);
// Add Stop to Patrol
AddPatrolStop(nPatrolRoute, nNextPatrolStop);
// Repeat the Process
ActionDoCommand(DoPatrolRoute(nPatrolRoute, nRouteType));
}
}
//
// This Function adds a Stop to the Patrol Route
void AddPatrolStop(int nPatrolRoute, int nStopNumber, int bJump=FALSE)
{
object oStop;
int nRun, nScript, nFacing;
int nDayOnly, nNightOnly;
float fPause;
// Danmar: Added below for random pause setup
int nRandomPause;
int nRandomRoute;
// End Danmar changes
string sStop;
int nValid = TRUE;
// Gather Stop Information
oStop = GetLocalObject(OBJECT_SELF, "PR_SN" +
PadIntToString(nStopNumber, 2));
if (GetIsObjectValid( oStop ) )
{
sStop = GetTag(oStop);
nRun = IsFlagPresent(sStop, "RN");
fPause = IntToFloat(GetFlagValue(sStop, "PS", 1));
// Danmar: Added RP###/RR### flag to patrol points to allow randomization
// of the pause time and stops.
nRandomPause = GetFlagValue(sStop, "RP", 0);
nRandomRoute = GetFlagValue(sStop, "RR", 0);
// End Danmar changes.
nScript = GetFlagValue(sStop, "SC", -1);
nFacing = IsFlagPresent(sStop, "SF");
nDayOnly = IsFlagPresent(sStop, "DO");
nNightOnly = IsFlagPresent(sStop, "NO");
// Day Only
if (nDayOnly == TRUE && (GetIsDay() == FALSE && GetIsDawn() == FALSE))
{
nValid = FALSE;
}
// Night Only
if (nNightOnly == TRUE && (GetIsNight() == FALSE && GetIsDusk() == FALSE))
{
nValid = FALSE;
}
// Check if Valid
if (nValid == TRUE)
{
// Move to Stop
// Modified by Danmar
// ActionMoveToObject(oStop, nRun); // Original NESS line.
// if d% is less than nRandomRoute (RRxxx) then we move to the next stop.
// If its not then we skip that one and move to the next.
if ((nRandomRoute == 0) || ((Random(100) + 1) < nRandomRoute))
{
if (bJump)
{
ActionJumpToLocation(GetLocation(oStop));
}
else
{
ActionMoveToObject(oStop, nRun);
}
}
// End Danmar Changes
if (nFacing == TRUE)
{
ActionDoCommand(SetFacing(GetFacingFromLocation(GetLocation(oStop))));
}
// Execute Script
if (nScript > -1)
{
SetLocalInt(OBJECT_SELF, "PatrolScript", nScript);
ActionDoCommand(SetLocalInt(OBJECT_SELF, "PatrolScriptRunning", TRUE));
ExecuteScript("spawn_sc_patrol", OBJECT_SELF);
ActionDoCommand(SetLocalInt(OBJECT_SELF, "PatrolScriptRunning", FALSE));
}
// Pause
/* Danmar: If fRandomPause!=0 then let's pick a random pause length and
add it to the existing fpause. This way you can use PS to set the
minimum pause and RP to set the maximum pause.
Example: PR01_SN01_PS010_RR011 would cause the creature to pause between
10 to 20 seconds. */
if (nRandomPause != 0)
{
fPause = fPause + IntToFloat(Random(nRandomPause));
}
// End Danmar Changes
ActionWait(fPause);
}
}
// Record this Stop and Clear Next Stop
ActionDoCommand(SetLocalInt(OBJECT_SELF, "PR_LASTSTOP", nStopNumber));
ActionDoCommand(SetLocalInt(OBJECT_SELF, "PR_NEXTSTOP", -1));
}
void CheckForStuckPatrol(object oCreature, int nPatrolRoute, int nRouteType)
{
// are we at the same location as last time?
location lLast = GetLocalLocation(oCreature, "NESSLastLoc");
location lCurrent = GetLocation(oCreature);
if (lLast != lCurrent)
{
SetLocalLocation(oCreature, "NESSLastLoc", lCurrent);
SetLocalInt(oCreature, "NESSStuckCount", 0);
return;
}
int nStuckCount = GetLocalInt(oCreature, "NESSStuckCount");
nStuckCount += 1;
if (nStuckCount < 3)
{
SetLocalInt(oCreature, "NESSStuckCount", nStuckCount);
return;
}
AssignCommand(oCreature, ClearAllActions());
// unstuck 'im
int nLastStop = GetLocalInt(OBJECT_SELF, "PR_LASTSTOP");
// force a move
AssignCommand(oCreature, AddPatrolStop(nPatrolRoute, nLastStop, TRUE));
// reset
SetLocalInt(oCreature, "NESSStuckCount", 0);
}
//
// This Function Checks Camp State
int ProcessCamp(object oCamp)
{
int iCount;
int nIsAlive = FALSE;
int nCampNumC, nCampNumP, nPlaceableType, nDeathScript;
float fCorpseDecay;
object oSpawned, oCampTrigger, oItem;
string sObject, sFlags, sCampTrigger;
// Check Creatures
nCampNumC = GetLocalInt(oCamp, "CampNumC");
// Suppress despawning on creatureless camps
if ( nCampNumC == 0 )
{
nIsAlive = TRUE;
}
for (iCount = 1; iCount <= nCampNumC; iCount++)
{
sObject = "CampC" + IntToString(iCount - 1);
sFlags = GetLocalString(oCamp, sObject + "_Flags");
fCorpseDecay = IntToFloat(GetFlagValue(sFlags, "CD", 0));
nDeathScript = GetFlagValue(sFlags, "DT", -1);
oSpawned = GetLocalObject(oCamp, sObject);
if (oSpawned != OBJECT_INVALID)
{
if (GetIsDead(oSpawned) == FALSE)
{
nIsAlive = TRUE;
int nIsBusy = FALSE;
if (GetIsInCombat(oSpawned) || IsInConversation(oSpawned) ||
(GetCurrentAction(oSpawned) != ACTION_INVALID))
{
nIsBusy = TRUE;
}
if (! nIsBusy)
{
// Do return home processing
int nReturnHome = GetLocalInt(oSpawned, "f_ReturnHome");
if (nReturnHome)
{
// retrieve lHome and fReturnRange
float fReturnHomeRange = GetLocalFloat(oSpawned,
"f_ReturnHomeRange");
float fHomeX = GetLocalFloat(oSpawned, "HomeX");
float fHomeY = GetLocalFloat(oSpawned, "HomeY");
vector vHome = Vector(fHomeX, fHomeY, 0.0);
location lHome = Location(OBJECT_SELF, vHome, 0.0);
if (GetDistanceBetweenLocations(lHome, GetLocation(oSpawned))
> fReturnHomeRange)
{
AssignCommand(oSpawned, ReturnHome(lHome));
nIsBusy = TRUE;
}
}
}
// Do random walk
if (! nIsBusy)
{
int nRandomWalk = GetLocalInt(oSpawned, "f_RandomWalk");
if (nRandomWalk &&
GetCurrentAction(oSpawned) != ACTION_WAIT &&
GetCurrentAction(oSpawned) != ACTION_CASTSPELL &&
(d2(1) == 2))
{
AssignCommand(oSpawned, ClearAllActions());
AssignCommand(oSpawned, ActionRandomWalk());
}
}
}
else
{
NESS_ProcessDeadCreature(oSpawned);
DeleteLocalObject(oCamp, sObject);
}
}
}
// Check Camp Trigger
if (nIsAlive)
{
sCampTrigger = GetLocalString(oCamp, "CampTrigger");
if (sCampTrigger != "")
{
oCampTrigger = GetLocalObject(oCamp, "Camp" + sCampTrigger);
if (oCampTrigger == OBJECT_INVALID || GetIsDead(oCampTrigger) == TRUE)
{
// Run Trigger Script
ExecuteScript("spawn_sc_cmptrig", oCamp);
}
}
}
// Check Placeable
nCampNumP = GetLocalInt(oCamp, "CampNumP");
for (iCount = 1; iCount <= nCampNumP; iCount++)
{
sObject = "CampP" + IntToString(iCount - 1);
sFlags = GetLocalString(oCamp, sObject + "_Flags");
nPlaceableType = GetFlagValue(sFlags, "PL", 0);
if (nPlaceableType == 1)
{
// Despawn if Empty
oSpawned = GetLocalObject(oCamp, sObject);
if (GetFirstItemInInventory(oSpawned) == OBJECT_INVALID)
{
DestroyObject(oSpawned);
}
}
}
return nIsAlive;
}
//
// This Function will Despawn a Camp
void DestroyCamp(object oCamp, float fCampDecay, int nSaveState)
{
int iCount;
object oSpawned;
string sObject;
int nCampNumP = GetLocalInt(oCamp, "CampNumP");
int nCampNumC = GetLocalInt(oCamp, "CampNumC");
// Destroy Placeables
for (iCount = 1; iCount <= nCampNumP; iCount++)
{
sObject = "CampP" + IntToString(iCount - 1);
oSpawned = GetLocalObject(oCamp, sObject);
if (nSaveState)
{
//debug("Saving " + sObject);
SaveStateOnDespawn(oSpawned, oCamp, TRUE);
}
SpawnCountDebug(oCamp, "despawning camp object " + ObjectToString(oSpawned));
DestroyObject(oSpawned, fCampDecay);
}
// Destroy Creatures
for (iCount = 1; iCount <= nCampNumC; iCount++)
{
sObject = "CampC" + IntToString(iCount - 1);
oSpawned = GetLocalObject(oCamp, sObject);
if (nSaveState)
{
//debug("Saving " + sObject);
SaveStateOnDespawn(oSpawned, oCamp, TRUE);
}
AssignCommand(oSpawned, ClearAllActions());
if (oSpawned != OBJECT_INVALID)
{
NESS_CleanInventory(oSpawned);
AssignCommand(oSpawned, SetIsDestroyable(TRUE));
SpawnCountDebug(oCamp, "despawning camp creature " + ObjectToString(oSpawned));
DestroyObject(oSpawned);
}
// remove from camp object
DeleteLocalObject(oCamp, sObject);
}
}
//
object GetSpawnLocationObject(object oSpawn, int nSpawnLocationMin,
int nSpawnLocation, int nSpawnLocationInd)
{
string sSpawnLocation;
object oSpawnLocation;
if (nSpawnLocationInd)
{
int nNextEmptySlot;
nNextEmptySlot = FindNextEmptyChildSlot(oSpawn);
if (nNextEmptySlot > 0)
{
sSpawnLocation = "SL" + PadIntToString(nSpawnLocation + nNextEmptySlot - 1,
2);
}
else
{
// no empty slots? Just use base, I guess
sSpawnLocation = "SL" + PadIntToString(nSpawnLocation, 2);
}
oSpawnLocation = GetNearestObjectByTag(sSpawnLocation, oSpawn);
}
else
{
int nRndSpawnLocation;
if (nSpawnLocationMin > -1)
{
nRndSpawnLocation = Random(nSpawnLocation + 1);
while (nRndSpawnLocation < nSpawnLocationMin)
{
nRndSpawnLocation = Random(nSpawnLocation + 1);
}
nSpawnLocation = nRndSpawnLocation;
}
sSpawnLocation = "SL" + PadIntToString(nSpawnLocation, 2);
oSpawnLocation = GetNearestObjectByTag(sSpawnLocation, oSpawn);
}
return oSpawnLocation;
}
vector GetSpawnRadiusPosition(vector vSpawnPos, float fSpawnRadius,
float fSpawnRadiusMin, int nRadialDistribution=FALSE)
{
float fX, fY;
vector vRadius;
if (nRadialDistribution)
{
float fSpawnAngle, fRadius;
fSpawnAngle = IntToFloat(Random(361));
if (fSpawnRadiusMin == fSpawnRadius)
{
fX = fSpawnRadius * cos(fSpawnAngle);
fY = fSpawnRadius * sin(fSpawnAngle);
}
else
{
fRadius = IntToFloat(Random(FloatToInt(fSpawnRadius) + 1));
while (fRadius < fSpawnRadiusMin)
{
fRadius = IntToFloat(Random(FloatToInt(fSpawnRadius) + 1));
}
fX = fRadius * cos(fSpawnAngle);
fY = fRadius * sin(fSpawnAngle);
}
}
else
{
float fTestDistSquared, fMaxRadiusSquared, fMinRadiusSquared;
int nSpawnRadius = FloatToInt(fSpawnRadius);
// Set up comparators
fMaxRadiusSquared = fSpawnRadius * fSpawnRadius;
fMinRadiusSquared = fSpawnRadiusMin * fSpawnRadiusMin;
// Calculate first attempt
fX = IntToFloat(Random((2 * nSpawnRadius) + 1) -
nSpawnRadius + 1);
fY = IntToFloat(Random((2 * nSpawnRadius) + 1) -
nSpawnRadius + 1);
fTestDistSquared = (fX * fX) + (fY * fY);
while (fTestDistSquared < fMinRadiusSquared ||
fTestDistSquared > fMaxRadiusSquared)
{
fX = IntToFloat(Random((2 * nSpawnRadius) + 1) -
nSpawnRadius + 1);
fY = IntToFloat(Random((2 * nSpawnRadius) + 1) -
nSpawnRadius + 1);
fTestDistSquared = (fX * fX) + (fY * fY);
}
}
vRadius = Vector(vSpawnPos.x + fX, vSpawnPos.y + fY);
return vRadius;
}
int CheckPositionUnseen(vector vSpawnPos, float fUnseenRadius)
{
location lSpawn = Location(OBJECT_SELF, vSpawnPos, 0.0);
object oCreature = GetFirstObjectInShape(SHAPE_SPHERE, fUnseenRadius,
lSpawn, FALSE, OBJECT_TYPE_CREATURE);
while (oCreature != OBJECT_INVALID)
{
if (GetIsPC(oCreature) == TRUE)
{
return FALSE;
}
oCreature = GetNextObjectInShape(SHAPE_SPHERE, fUnseenRadius,
lSpawn, FALSE, OBJECT_TYPE_CREATURE);
}
return TRUE;
}
void ReportSpawns(int nAreaSpawns, int nModuleSpawns)
{
SendMessageToAllDMs("New creature count in " +
GetName(OBJECT_SELF) + ": " + IntToString(nAreaSpawns) +
" Module spawn count: " + IntToString(nModuleSpawns) +
(SPAWN_DELAY_DEBUG ? " (sd+)" : " (sd-)") +
(SPAWN_COUNT_DEBUG ? " (sc+)" : " (sc-)"));
}
void TrackModuleSpawns(int nAreaSpawnCount, int nTrackModuleSpawns)
{
int nNewAreaSpawnCount = GetLocalInt(OBJECT_SELF, SPAWN_AREA_COUNT );
int nSpawnDifference = nNewAreaSpawnCount - nAreaSpawnCount;
if (nSpawnDifference)
{
int nModuleSpawns = GetLocalInt(GetModule(), "ModuleSpawnCount");
nModuleSpawns += nSpawnDifference;
SetLocalInt(GetModule(), "ModuleSpawnCount", nModuleSpawns);
if (nTrackModuleSpawns)
{
ReportSpawns(nNewAreaSpawnCount, nModuleSpawns);
}
}
}
void DumpModuleSpawns()
{
int nAreaSpawnCount = GetLocalInt(OBJECT_SELF, SPAWN_AREA_COUNT );
if (nAreaSpawnCount > 0)
{
SendMessageToAllDMs("Area " + GetName(OBJECT_SELF) + ": " +
IntToString(nAreaSpawnCount) +
(SPAWN_DELAY_DEBUG ? " (sd+)" : " (sd-)") +
(SPAWN_COUNT_DEBUG ? " (sc+)" : " (sc-)"));
}
}
void NESS_ReturnHome(object oCreature, int bRun=FALSE)
{
float fHomeX = GetLocalFloat(oCreature, "HomeX");
float fHomeY = GetLocalFloat(oCreature, "HomeY");
vector vHome = Vector(fHomeX, fHomeY, 0.0);
location lHome = Location(GetArea(oCreature), vHome, 0.0);
ClearAllActions();
ActionMoveToLocation(lHome, bRun);
}
void ReturnHome(location lHome)
{
if (GetCurrentAction() != ACTION_INVALID)
return;
ClearAllActions();
ActionMoveToLocation(lHome);
}
void DestroyIfNonDrop(object oItem)
{
if(GetIsObjectValid(oItem))
{
if (! GetDroppableFlag(oItem))
{
//debug("Destroying non-drop item named " + GetName(oItem));
DestroyObject(oItem);
}
}
}
void StripNonDroppables(object oSpawned)
{
DestroyIfNonDrop(GetItemInSlot(INVENTORY_SLOT_ARMS, oSpawned));
DestroyIfNonDrop(GetItemInSlot(INVENTORY_SLOT_ARROWS, oSpawned));
DestroyIfNonDrop(GetItemInSlot(INVENTORY_SLOT_BELT, oSpawned));
DestroyIfNonDrop(GetItemInSlot(INVENTORY_SLOT_BOLTS, oSpawned));
DestroyIfNonDrop(GetItemInSlot(INVENTORY_SLOT_BOOTS, oSpawned));
DestroyIfNonDrop(GetItemInSlot(INVENTORY_SLOT_BULLETS, oSpawned));
DestroyIfNonDrop(GetItemInSlot(INVENTORY_SLOT_CHEST, oSpawned));
DestroyIfNonDrop(GetItemInSlot(INVENTORY_SLOT_CLOAK, oSpawned));
DestroyIfNonDrop(GetItemInSlot(INVENTORY_SLOT_HEAD, oSpawned));
DestroyIfNonDrop(GetItemInSlot(INVENTORY_SLOT_LEFTHAND, oSpawned));
DestroyIfNonDrop(GetItemInSlot(INVENTORY_SLOT_LEFTRING, oSpawned));
DestroyIfNonDrop(GetItemInSlot(INVENTORY_SLOT_NECK, oSpawned));
DestroyIfNonDrop(GetItemInSlot(INVENTORY_SLOT_RIGHTHAND, oSpawned));
DestroyIfNonDrop(GetItemInSlot(INVENTORY_SLOT_RIGHTRING, oSpawned));
object oItem = GetFirstItemInInventory(oSpawned);
while (oItem != OBJECT_INVALID)
{
DestroyIfNonDrop(oItem);
oItem = GetNextItemInInventory(oSpawned);
}
}
void NESS_ProcessDeadCreature(object oCreature, object oSpawn=OBJECT_INVALID)
{
int nCampCreature;
// Only do once per creature. Since this may be called from onDeath
// or within a NESS HB, check/mark for having been processed
int nProcessedDeath = GetLocalInt(oCreature, "ProcessedDeath");
if (nProcessedDeath)
{
return;
}
SetLocalInt(oCreature, "ProcessedDeath", TRUE);
// Find parent if not passed in
if (oSpawn == OBJECT_INVALID)
{
oSpawn = GetLocalObject(oCreature, "ParentSpawn");
}
object oArea = GetArea(oCreature);
SPAWN_DELAY_DEBUG = GetLocalInt(oArea, "SpawnDelayDebug");
SPAWN_COUNT_DEBUG = GetLocalInt(oArea, "SpawnCountDebug");
SpawnDelayDebug(oSpawn, "creature " + ObjectToString(oCreature) + " died");
// Remove non-drop items
// StripNonDroppables(oCreature);
// Initialize DeathScripts
int nDeathScript;
float fCorpseDecay;
if (oSpawn == OBJECT_INVALID)
{
//debug("camp creature died");
string sCampFlags;
nCampCreature = TRUE;
sCampFlags = GetLocalString(oCreature, "CreatureFlags");
//debug("in process dead with flags " + sCampFlags);
fCorpseDecay = IntToFloat(GetFlagValue(sCampFlags, "CD", 0));
nDeathScript = GetFlagValue(sCampFlags, "DT", -1);
//debug("camp creature: cd = " + FloatToString(fCorpseDecay));
}
else
{
//debug("non-camp creature died");
nCampCreature = FALSE;
nDeathScript = GetLocalInt(oSpawn, "f_DeathScript");
fCorpseDecay = GetLocalFloat(oSpawn, "f_CorpseDecay");
//debug("not a camp creature: cd = " + FloatToString(fCorpseDecay));
}
// Run Death Script
if (nDeathScript > -1 && GetLocalInt(oCreature, "DeathScriptRan") == FALSE)
{
SetLocalInt(oCreature, "DeathScript", nDeathScript);
ExecuteScript("spawn_sc_death", oCreature);
}
// Spawn Corpse if Dead and No Corpse
if (fCorpseDecay > 0.0)
{
if (GetLocalObject(oCreature, "Corpse") == OBJECT_INVALID)
{
//debug("calling spawn_corpse_dth");
ExecuteScript("spawn_corpse_dth", oCreature);
}
}
else
{
if (GetLocalInt(GetModule(), "AlwaysDestroyCorpses"))
{
AssignCommand(oCreature, SetIsDestroyable(TRUE, FALSE, FALSE));
}
}
}
void ResetSpawn(object oSpawn, int nTimeNow)
{
// Reset the Spawn
SetLocalInt(oSpawn, "NextSpawnTime", nTimeNow);
//debug("set next spawn time on reset: " + IntToString(nTimeNow));
SetLocalInt(oSpawn, "SpawnDeactivated", FALSE);
SetLocalInt(oSpawn, "ChildrenSpawned", 0);
SetLocalInt(oSpawn, "CurrentProcessTick", 0);
SetLocalInt(oSpawn, "SpawnAgeTime", 0);
//debug("PC reset");
}
//
void SaveStateOnDespawn(object oSpawned, object oSpawn, int nCamp=FALSE)
{
string sSaveVarName;
// These are the values needed to recreate the spawned object where
// it was when it despawned
int nObjectType = 0;
string sTemplate;
location lLastLocation;
float fLastFacing;
float fHomeX;
float fHomeY;
int nReturnHome;
if (nCamp && !GetIsObjectValid(oSpawned))
{
SpawnCountDebug(oSpawn, "creating slot for dead camp object" +
ObjectToString(oSpawned));
sTemplate = "";
}
else
{
nObjectType = GetObjectType(oSpawned);
sTemplate = GetResRef(oSpawned);
lLastLocation = GetLocation(oSpawned);
fLastFacing = GetFacing(oSpawned);
fHomeX = GetLocalFloat(oSpawned, "HomeX");
fHomeY = GetLocalFloat(oSpawned, "HomeY");
// if the return home flag is on, respawn at the home point
if (nCamp)
{
nReturnHome = GetLocalInt(oSpawned, "f_ReturnHome");
}
else
{
nReturnHome = GetLocalInt(oSpawn, "f_ReturnHome");
}
if (nReturnHome)
{
vector vHome = Vector(fHomeX, fHomeY, 0.0);
location lHome = Location(GetArea(oSpawned), vHome, fLastFacing);
lLastLocation = lHome;
}
}
int nNumberSaveStates = GetLocalInt(oSpawn, "SpawnNumSavedStates");
SetLocalInt(oSpawn, "SpawnNumSavedStates", ++nNumberSaveStates);
string sSaveVarNameBase = "SpawnedSaveState" + IntToString(nNumberSaveStates);
SpawnCountDebug(oSpawn, "PC despawn save count: " + IntToString(nNumberSaveStates));
// Save the stuff needed to respawn
sSaveVarName = sSaveVarNameBase + "ObjectType";
SetLocalInt(oSpawn, sSaveVarName, nObjectType);
sSaveVarName = sSaveVarNameBase + "Template";
SetLocalString(oSpawn, sSaveVarName, sTemplate);
sSaveVarName = sSaveVarNameBase + "LastLocation";
SetLocalLocation(oSpawn, sSaveVarName, lLastLocation);
sSaveVarName = sSaveVarNameBase + "HomeX";
SetLocalFloat(oSpawn, sSaveVarName, fHomeX);
sSaveVarName = sSaveVarNameBase + "HomeY";
SetLocalFloat(oSpawn, sSaveVarName, fHomeY);
// These need to be preserved so they can be rewritten to the respawned
// object; they may be used in ProcessSpawn
if (! nCamp)
{
float fSpawnFacing = GetLocalFloat(oSpawned, "SpawnFacing");
float fEntranceExitX = GetLocalFloat(oSpawned, "EntranceExitX");
float fEntranceExitY = GetLocalFloat(oSpawned, "EntranceExitY");
int iExpireTime = GetLocalInt(oSpawned, "LifespanExpireTime");
// Save the stuff we need to rewrite onto the spawn
sSaveVarName = sSaveVarNameBase + "SpawnFacing";
SetLocalFloat(oSpawn, sSaveVarName, fSpawnFacing);
sSaveVarName = sSaveVarNameBase + "EntranceExitX";
SetLocalFloat(oSpawn, sSaveVarName, fEntranceExitX);
sSaveVarName = sSaveVarNameBase + "EntranceExitY";
SetLocalFloat(oSpawn, sSaveVarName, fEntranceExitY);
sSaveVarName = sSaveVarNameBase + "LifeSpanExpireTime";
SetLocalInt(oSpawn, sSaveVarName, iExpireTime);
}
}
//
void SaveCampStateOnDespawn(object oCamp, object oSpawn)
{
// Note that most of what needs to be saved is on the camp object itself.
// This saves the camp object onto oSpawn
int nNumSavedCampStates = GetLocalInt(oSpawn,"SpawnNumSavedCampStates");
SetLocalInt(oSpawn, "SpawnNumSavedCampStates",++nNumSavedCampStates);
SpawnCountDebug(oSpawn, "PC despawn camp save count: " + IntToString(nNumSavedCampStates));
string sSaveCampVarName = "SpawnedSaveCampState" +
IntToString(nNumSavedCampStates);
SetLocalObject(oSpawn, sSaveCampVarName, oCamp);
}
//
void RestorePCDespawns(object oSpawn, int nTimeNow)
{
int nSpawnNumSavedStates = GetLocalInt(oSpawn, "SpawnNumSavedStates");
string sSaveVarNameBase;
string sSaveVarName;
int nChildNum;
object oSpawned;
int nObjectType;
string sTemplate;
location lLastLocation;
location lEntranceExit;
float fHomeX;
float fHomeY;
location lHome;
int iExpireTime;
int i;
SpawnCountDebug(oSpawn, "restoring " + IntToString(nSpawnNumSavedStates) + " objects");
for (i = 0; i < nSpawnNumSavedStates; i++)
{
nChildNum = i + 1;
sSaveVarNameBase = "SpawnedSaveState" + IntToString(nChildNum);
sSaveVarName = sSaveVarNameBase + "ObjectType";
nObjectType = GetLocalInt(oSpawn, sSaveVarName);
DeleteLocalInt(oSpawn, sSaveVarName);
sSaveVarName = sSaveVarNameBase + "Template";
sTemplate = GetLocalString(oSpawn, sSaveVarName);
DeleteLocalString (oSpawn, sSaveVarName);
sSaveVarName = sSaveVarNameBase + "LastLocation";
lLastLocation = GetLocalLocation(oSpawn, sSaveVarName);
DeleteLocalLocation(oSpawn, sSaveVarName);
sSaveVarName = sSaveVarNameBase + "SpawnFacing";
float fSpawnFacing = GetLocalFloat(oSpawn, sSaveVarName);
DeleteLocalFloat(oSpawn, sSaveVarName);
sSaveVarName = sSaveVarNameBase + "HomeX";
fHomeX = GetLocalFloat(oSpawn, sSaveVarName);
DeleteLocalFloat(oSpawn, sSaveVarName);
sSaveVarName = sSaveVarNameBase + "HomeY";
fHomeY = GetLocalFloat(oSpawn, sSaveVarName);
DeleteLocalFloat(oSpawn, sSaveVarName);
sSaveVarName = sSaveVarNameBase + "EntranceExitX";
float fEntranceExitX = GetLocalFloat(oSpawn, sSaveVarName);
DeleteLocalFloat(oSpawn, sSaveVarName);
sSaveVarName = sSaveVarNameBase + "EntranceExitY";
float fEntranceExitY = GetLocalFloat(oSpawn, sSaveVarName);
DeleteLocalFloat(oSpawn, sSaveVarName);
sSaveVarName = sSaveVarNameBase + "LifeSpanExpireTime";
iExpireTime = GetLocalInt(oSpawn, sSaveVarName);
DeleteLocalInt(oSpawn, sSaveVarName);
if (! IsRestoreBlocked(oSpawn, lLastLocation, iExpireTime, nTimeNow))
{
object oSpawned = CreateObject(nObjectType, sTemplate, lLastLocation);
SpawnCountDebug(oSpawn, "restored " + sSaveVarNameBase + "object id: " +
ObjectToString(oSpawned));
lHome = Location(OBJECT_SELF, Vector(fHomeX, fHomeY, 0.), fSpawnFacing);
lEntranceExit = Location(OBJECT_SELF, Vector(fEntranceExitX,
fEntranceExitY, 0.), fSpawnFacing);
RecordSpawned(oSpawn, oSpawned, lHome, lEntranceExit, fSpawnFacing);
SetupSpawned(oSpawn, oSpawned, lHome, nTimeNow, FALSE);
// Lifespan expire time needs to be rewritten the the spawned object,
// since SetupSpawned generated a new one based on the current time...
SetLocalInt(oSpawned, "LifespanExpireTime", iExpireTime);
}
else
{
SpawnCountDebug(oSpawn, "restore blocked: " + sSaveVarNameBase);
}
}
int nNumSavedCampStates = GetLocalInt(oSpawn, "SpawnNumSavedCampStates");
// restore camps
for (i = 0; i < nNumSavedCampStates; i++)
{
int j;
int nCampNum = i + 1;
string sSaveCampVarName = "SpawnedSaveCampState" +
IntToString(nCampNum);
object oCamp = GetLocalObject(oSpawn, sSaveCampVarName);
DeleteLocalObject(oSpawn, sSaveCampVarName);
// respawn camp objects
//debug("restoring " + sSaveCampVarName);
nSpawnNumSavedStates = GetLocalInt(oCamp, "SpawnNumSavedStates");
int nPlaceableCount = 0;
int nCreatureCount = 0;
for (j = 0; j < nSpawnNumSavedStates; j++)
{
nChildNum = j + 1;
sSaveVarNameBase = "SpawnedSaveState" + IntToString(nChildNum);
//debug("restoring " + sSaveVarNameBase);
sSaveVarName = sSaveVarNameBase + "ObjectType";
//debug("Getting objtype with var name " + sSaveVarName);
nObjectType = GetLocalInt(oCamp, sSaveVarName);
DeleteLocalInt(oCamp, sSaveVarName);
sSaveVarName = sSaveVarNameBase + "Template";
sTemplate = GetLocalString(oCamp, sSaveVarName);
//debug("template: " + sTemplate);
DeleteLocalString (oCamp, sSaveVarName);
sSaveVarName = sSaveVarNameBase + "LastLocation";
lLastLocation = GetLocalLocation(oCamp, sSaveVarName);
DeleteLocalLocation(oCamp, sSaveVarName);
sSaveVarName = sSaveVarNameBase + "HomeX";
fHomeX = GetLocalFloat(oCamp, sSaveVarName);
DeleteLocalFloat(oCamp, sSaveVarName);
sSaveVarName = sSaveVarNameBase + "HomeY";
fHomeY = GetLocalFloat(oCamp, sSaveVarName);
DeleteLocalFloat(oCamp, sSaveVarName);
lHome = Location(OBJECT_SELF, Vector(fHomeX, fHomeY, 0.), 0.);
sSaveVarName = sSaveVarNameBase + "LifeSpanExpireTime";
iExpireTime = GetLocalInt(oSpawn, sSaveVarName);
DeleteLocalInt(oSpawn, sSaveVarName);
string sObject;
if (nObjectType == OBJECT_TYPE_CREATURE)
{
sObject = "CampC" + IntToString(nCreatureCount++);
}
else if (nObjectType == OBJECT_TYPE_PLACEABLE)
{
sObject = "CampP"+ IntToString(nPlaceableCount++);
}
if (sTemplate != "")
{
if (! IsRestoreBlocked(oCamp, lLastLocation, iExpireTime, nTimeNow))
{
oSpawned = CreateObject(nObjectType, sTemplate, lLastLocation);
SetLocalObject(oCamp, sObject, oSpawned);
vector vCamp = GetPositionFromLocation(GetLocation(oSpawn));
string sFlags = GetLocalString(oCamp, sObject + "_Flags");
SetupCampSpawned(oSpawn, oSpawned, vCamp, lHome, sFlags);
// Lifespan expire time needs to be rewritten the the spawned object,
// since SetupSpawned generated a new one based on the current time...
SetLocalInt(oSpawned, "LifespanExpireTime", iExpireTime);
}
else
{
SetLocalObject(oCamp, sObject, OBJECT_INVALID);
}
}
else
{
SetLocalObject(oCamp, sObject, OBJECT_INVALID);
}
}
location lCampLocation = GetLocation(oSpawn);
float fCampFacing = GetFacing(oSpawn);
RecordSpawned(oSpawn, oCamp, lCampLocation, lCampLocation, fCampFacing);
}
SetLocalInt(oSpawn, "SpawnNumSavedStates", 0);
SetLocalInt(oSpawn, "SpawnNumSavedCampStates", 0);
}
//
void RecordSpawned(object oSpawn, object oSpawned, location lHome,
location lEntranceExit, float fSpawnedFacing)
{
int nChildrenSpawned;
int nSpawnCount;
int nChildSlot;
int nEmptyChildSlot;
object oChild;
string sChildSlot, sEmptyChildSlot;
string sCustomFlag = GetLocalString(oSpawn, "f_CustomFlag");
int nSpawnNumber = GetLocalInt(oSpawn, "f_SpawnNumber");
// Assign Values to oSpawned
SetLocalObject(oSpawned, "ParentSpawn", oSpawn);
SetLocalFloat(oSpawned, "SpawnFacing", fSpawnedFacing);
SetLocalFloat(oSpawned, "HomeX", GetPositionFromLocation(lHome).x);
SetLocalFloat(oSpawned, "HomeY", GetPositionFromLocation(lHome).y);
SetLocalFloat(oSpawned, "EntranceExitX", GetPositionFromLocation(lEntranceExit).x);
SetLocalFloat(oSpawned, "EntranceExitY", GetPositionFromLocation(lEntranceExit).y);
SetLocalString(oSpawned, "CustomFlag", sCustomFlag);
// Assign Values to oSpawn
nChildrenSpawned = GetLocalInt(oSpawn, "ChildrenSpawned");
nChildrenSpawned++;
SetLocalInt(oSpawn, "ChildrenSpawned", nChildrenSpawned);
nSpawnCount = GetLocalInt(oSpawn, "SpawnCount");
nSpawnCount++;
SetLocalInt(oSpawn, "SpawnCount", nSpawnCount);
// Find Empty Child Slot
nChildSlot = 1;
nEmptyChildSlot = 0;
for (nChildSlot = 1; nChildSlot <= nSpawnNumber; nChildSlot++)
{
// Retrieve Child
sChildSlot = "ChildSlot" + PadIntToString(nChildSlot, 2);
oChild = GetLocalObject(oSpawn, sChildSlot);
// Check if this is Child Slot is Valid
if (GetIsObjectValid(oChild) == FALSE || GetIsDead(oChild))
{
// Empty Slot
if (nEmptyChildSlot == 0)
{
nEmptyChildSlot = nChildSlot;
sEmptyChildSlot = sChildSlot;
}
}
}
// Assign Child Slot
SpawnCountDebug(oSpawn, "recorded spawn " + GetTag(oSpawned) + " (" +
ObjectToString(oSpawned) + ") as " + sEmptyChildSlot);
SetLocalObject(oSpawn, sEmptyChildSlot, oSpawned);
SetLocalString(oSpawned, "ParentChildSlot", sEmptyChildSlot);
object oIdiot = GetLocalObject(oSpawn, sEmptyChildSlot);
string sValid = GetIsObjectValid(oIdiot) ? "valid" : "invalid";
}
//
// Returns 0 if no empty slots
int FindNextEmptyChildSlot(object oSpawn)
{
int nChildSlot;
int nEmptyChildSlot;
object oChild;
string sChildSlot;
int nSpawnNumber = GetLocalInt(oSpawn, "f_SpawnNumber");
nChildSlot = 1;
nEmptyChildSlot = 0;
for (nChildSlot = 1; nChildSlot <= nSpawnNumber; nChildSlot++)
{
// Retrieve Child
sChildSlot = "ChildSlot" + PadIntToString(nChildSlot, 2);
oChild = GetLocalObject(oSpawn, sChildSlot);
// Check if this is Child Slot is Valid
if (GetIsObjectValid(oChild) == FALSE || GetIsDead(oChild))
{
// Empty Slot
nEmptyChildSlot = nChildSlot;
break;
}
}
return nEmptyChildSlot;
}
void SetupSpawned(object oSpawn, object oSpawned, location lHome,
int nTimeNow, int nWalkToHome = FALSE)
{
object oFaction;
int nChildLifespanExpireTime;
int nGoldAmount;
effect eObject;
int nSpawnFaction = GetLocalInt(oSpawn, "f_SpawnFaction");
int nSpawnSit = GetLocalInt(oSpawn, "f_SpawnSit");
int nSpawnPlot = GetLocalInt(oSpawn, "f_SpawnPlot");
int nSpawnAlignment = GetLocalInt(oSpawn, "f_SpawnAlignment");
int nAlignmentShift = GetLocalInt(oSpawn, "f_AlignmentShift");
int nChildLifespanMax = GetLocalInt(oSpawn, "f_ChildLifespanMax");
int nChildLifespanMin = GetLocalInt(oSpawn, "f_ChildLifespanMin");
int nRandomGold = GetLocalInt(oSpawn, "f_RandomGold");
int nRandomGoldMin = GetLocalInt(oSpawn, "f_RandomGoldMin");
int nGoldChance = GetLocalInt(oSpawn, "f_GoldChance");
int nLootTable = GetLocalInt(oSpawn, "f_LootTable");
int nTrapDisabled = GetLocalInt(oSpawn, "f_TrapDisabled");
float fCorpseDecay = GetLocalFloat(oSpawn, "f_CorpseDecay");
int nCorpseDecayType = GetLocalInt(oSpawn, "f_CorpseDecayType");
int bDropWielded = GetLocalInt(oSpawn, "f_CorpseDropWielded");
int bDeleteLootOnDecay = GetLocalInt(oSpawn, "f_CorpseDeleteLootOnDecay");
string sCorpseRemainsResRef = GetLocalString(oSpawn, "f_CorpseRemainsResRef");
int nDeathScript = GetLocalInt(oSpawn, "f_DeathScript");
int nSpawnScript = GetLocalInt(oSpawn, "f_SpawnScript");
int nSpawnAreaEffect = GetLocalInt(oSpawn, "f_SpawnAreaEffect");
float fAreaEffectDuration = GetLocalFloat(oSpawn, "f_AreaEffectDuration");
int nObjectEffect = GetLocalInt(oSpawn, "f_ObjectEffect");
float fObjectEffectDuration = GetLocalFloat(oSpawn, "f_ObjectEffectDuration");
string sSpawnTag = GetLocalString(oSpawn, "f_Template");
int nPatrolRoute = GetLocalInt(oSpawn, "f_PatrolRoute");
int nPatrolStartAtClosest = GetLocalInt(oSpawn, "f_PatrolStartAtClosest");
int nRouteType = GetLocalInt(oSpawn, "f_RouteType");
int nRandomWalk = GetLocalInt(oSpawn, "f_RandomWalk");
float fWanderRange = GetLocalFloat(oSpawn, "f_WanderRange");
int nSuppressLooting = GetLocalInt(oSpawn, "f_SuppressLooting");
int nSubdualMode = GetLocalInt(oSpawn, "f_SubdualMode");
int nEncounterLevel = GetLocalInt(oSpawn, "f_EncounterLevel");
if (nWalkToHome)
{
AssignCommand(oSpawned, ActionMoveToLocation(lHome));
}
// Spawn it in with the right facing, and you don't need this; and
// then it works for placeables as well!
// AssignCommand(oSpawned, ActionDoCommand(SetFacing(fSpawnedFacing)));
// Set up SpawnPlot
if (nSpawnPlot == TRUE)
{
SetPlotFlag(oSpawned, TRUE);
}
// Set up Faction
if (nSpawnFaction > -1)
{
switch (nSpawnFaction)
{
case 0:
ChangeToStandardFaction(oSpawned, STANDARD_FACTION_COMMONER);
break;
case 1:
ChangeToStandardFaction(oSpawned, STANDARD_FACTION_DEFENDER);
break;
case 2:
ChangeToStandardFaction(oSpawned, STANDARD_FACTION_MERCHANT);
break;
case 3:
ChangeToStandardFaction(oSpawned, STANDARD_FACTION_HOSTILE);
break;
case 4:
oFaction = GetNearestObjectByTag("SpawnFaction", oSpawned);
if (oFaction != OBJECT_INVALID)
{
ChangeFaction(oSpawned, oFaction);
}
break;
}
}
// Set up Alignment
if (nSpawnAlignment > -1)
{
switch (nSpawnAlignment)
{
case 0:
AdjustAlignment(oSpawned, ALIGNMENT_NEUTRAL, nAlignmentShift);
break;
case 1:
AdjustAlignment(oSpawned, ALIGNMENT_LAWFUL, nAlignmentShift);
break;
case 2:
AdjustAlignment(oSpawned, ALIGNMENT_CHAOTIC, nAlignmentShift);
break;
case 3:
AdjustAlignment(oSpawned, ALIGNMENT_GOOD, nAlignmentShift);
break;
case 4:
AdjustAlignment(oSpawned, ALIGNMENT_EVIL, nAlignmentShift);
break;
case 5:
AdjustAlignment(oSpawned, ALIGNMENT_ALL, nAlignmentShift);
break;
}
}
// Set up Lifespan
if (nChildLifespanMax > -1)
{
if (nChildLifespanMin > -1)
{
nChildLifespanExpireTime = -1;
while (nChildLifespanExpireTime < nChildLifespanMin)
{
nChildLifespanExpireTime = nTimeNow + Random(nChildLifespanMax) + 1;
}
}
else
{
nChildLifespanExpireTime = nTimeNow + nChildLifespanMax;
}
SetLocalInt(oSpawned, "LifespanExpireTime", nChildLifespanExpireTime);
}
// Give Creature Loot
if (nLootTable > -1)
{
DelayCommand(1.0, LootTable(oSpawn, oSpawned, nLootTable));
}
// Give RandomGold
if (nRandomGold > 0)
{
// One in Four Creatures give Gold
if (d100(1) <= nGoldChance)
{
// Calculate Gold to Drop
nGoldAmount = Random(nRandomGold + 1);
while (nGoldAmount < nRandomGoldMin)
{
nGoldAmount = Random(nRandomGold + 1);
}
// Give Gold
CreateItemOnObject("nw_it_gold001", oSpawned, nGoldAmount);
}
}
// Set up Trap on Placeable
if (GetIsTrapped(oSpawned))
{
if (d100(1) <= nTrapDisabled)
{
SetTrapDisabled(oSpawned);
}
}
// Set up Corpse Decay
if (fCorpseDecay > 0.0)
{
SetLocalFloat(oSpawned, "CorpseDecay", fCorpseDecay);
SetLocalInt(oSpawned, "CorpseDecayType", nCorpseDecayType);
SetLocalString(oSpawned, "CorpseRemainsResRef", sCorpseRemainsResRef);
SetLocalInt(oSpawned, "CorpseDropWielded", bDropWielded);
SetLocalInt(oSpawned, "CorpseDeleteLootOnDecay", bDeleteLootOnDecay);
AssignCommand(oSpawned, SetIsDestroyable(FALSE, FALSE, FALSE));
}
// Set up Death Script
if (nDeathScript > -1)
{
AssignCommand(oSpawned, SetIsDestroyable(FALSE, FALSE, FALSE));
}
// Set up Object Effects
if (nObjectEffect > 0)
{
eObject = ObjectEffect(oSpawn);
if (fObjectEffectDuration == -1.0)
{
ApplyEffectToObject(DURATION_TYPE_PERMANENT, eObject, oSpawned, 0.0);
}
else
{
ApplyEffectToObject(DURATION_TYPE_TEMPORARY, eObject, oSpawned, fObjectEffectDuration);
}
}
// Set up Area Effect
if (nSpawnAreaEffect > 0 && sSpawnTag == "AE" && fAreaEffectDuration > 0.0)
{
DestroyObject(oSpawned, fAreaEffectDuration);
}
// Run the Spawn Script
if (nSpawnScript > -1)
{
SetLocalInt(oSpawned, "SpawnScript", nSpawnScript);
ExecuteScript("spawn_sc_spawn", oSpawned);
}
// Set up Random Walking
if (nRandomWalk == TRUE)
{
if (fWanderRange > 0.0)
{
AssignCommand(oSpawned, RandomWalk(oSpawn, fWanderRange, FALSE));
}
else
{
AssignCommand(oSpawned, ActionRandomWalk());
}
}
// Set up the Patrol Route
if (nPatrolRoute > -1)
{
AssignCommand(oSpawned, SetPatrolRoute(nPatrolRoute,
nPatrolStartAtClosest));
AssignCommand(oSpawned, DoPatrolRoute(nPatrolRoute, nRouteType));
}
// Set up Spawn Sit
if (nSpawnSit == TRUE)
{
FindSeat(oSpawn, oSpawned);
}
// Set up loot suppression
if (nSuppressLooting)
{
SetLocalInt(oSpawned, "DoNotLoot", TRUE);
}
// Set up subdual mode
if (nSubdualMode)
{
SetLocalInt(oSpawned, "SubdualMode", TRUE);
}
// Set up encounter level
if (nEncounterLevel > 0)
{
SetLocalInt(oSpawned, "AlfaEncounterLevel", nEncounterLevel);
}
SetupCustomFlags(oSpawn, oSpawned);
}
//
void SetupCampSpawned(object oSpawn, object oSpawned, vector vCampPosition,
location lHome, string sFlags)
{
//debug("in setupCampSpawned");
// This is the closest we get to an "InitFlags" call for camp creatures
// write the flags onto the spawned creature
SetLocalString(oSpawned, "CreatureFlags", sFlags);
int nSpawnFacing = IsFlagPresent(sFlags, "SF");
int nLootTable = GetFlagValue(sFlags, "LT", -1);
int nTrapDisabled = GetSubFlagValue(sFlags, "PL", "T", 100);
int nRandomWalk = IsFlagPresent(sFlags, "RW");
SetLocalInt(oSpawned, "f_RandomWalk", nRandomWalk);
float fCorpseDecay = IntToFloat(GetFlagValue(sFlags, "CD", 0));
int nCorpseDecayType = GetSubFlagValue(sFlags, "CD", "T", 0);
int nCorpseRemainsType = GetSubFlagValue(sFlags, "CD", "R", 0);
int bDropWielded = IsSubFlagPresent(sFlags, "CD", "D");
int nDeathScript = GetFlagValue(sFlags, "DT", -1);
int nReturnHome = IsFlagPresent(sFlags, "RH");
SetLocalInt(oSpawned, "f_ReturnHome", nReturnHome);
if (nReturnHome)
{
int dfReturnHomeRange = GetLocalInt(GetModule(), "df_ReturnHomeRange");
float fReturnHomeRange = IntToFloat(GetFlagValue(sFlags, "RH",
dfReturnHomeRange));
SetLocalFloat(oSpawned, "f_ReturnHomeRange", fReturnHomeRange);
SetLocalFloat(oSpawned, "HomeX", GetPositionFromLocation(lHome).x);
SetLocalFloat(oSpawned, "HomeY", GetPositionFromLocation(lHome).y);
}
// Spawn Facing
if (nSpawnFacing == TRUE)
{
AssignCommand(oSpawned, SetFacingPoint(vCampPosition));
}
else
{
AssignCommand(oSpawned, SetFacing(IntToFloat(Random(360))));
}
// Loot Table
if (nLootTable > -1)
{
LootTable(oSpawn, oSpawned, nLootTable);
}
// Trap Disabled
if (GetIsTrapped(oSpawned))
{
if (d100(1) <= nTrapDisabled)
{
SetTrapDisabled(oSpawned);
}
}
// RandomWalk
if (nRandomWalk == TRUE)
{
AssignCommand(oSpawned, ActionRandomWalk());
}
// Corpse Decay
if (fCorpseDecay > 0.0)
{
string sCorpseRemainsResRef;
int bDeleteLootOnDecay = FALSE;
switch (nCorpseRemainsType)
{
case 0: sCorpseRemainsResRef = "invis_corpse_obj"; break;
case 1: sCorpseRemainsResRef = "invis_corpse_bdy"; break;
case 2: sCorpseRemainsResRef = "invis_corpse_bon"; break;
case 3: sCorpseRemainsResRef = "invis_corpse_pot"; break;
case 4: sCorpseRemainsResRef = "invis_corpse_pch"; break;
case 5: sCorpseRemainsResRef = "invis_corpse_scr"; break;
case 6: sCorpseRemainsResRef = "invis_corpse_tre"; break;
case 7:
sCorpseRemainsResRef = "invis_corpse_obj";
bDeleteLootOnDecay = TRUE;
break;
}
// Record CorpseDecay
SetLocalString(oSpawned, "CorpseRemainsResRef", sCorpseRemainsResRef);
SetLocalInt(oSpawned, "CorpseDropWielded", bDropWielded);
SetLocalInt(oSpawned, "CorpseDeleteLootOnDecay", bDeleteLootOnDecay);
SetLocalFloat(oSpawned, "CorpseDecay", fCorpseDecay);
SetLocalInt(oSpawned, "CorpseDecayType", nCorpseDecayType);
AssignCommand(oSpawned, SetIsDestroyable(FALSE, FALSE, FALSE));
}
// Death Script
if (nDeathScript > -1)
{
AssignCommand(oSpawned, SetIsDestroyable(FALSE, FALSE, FALSE));
}
}
int SetupSpawnDelay(int nSpawnDelay, int nDelayRandom, int nDelayMinimum,
int nTimeNow)
{
int nNextSpawnTime;
if (nDelayRandom == TRUE)
{
// Setup Next Spawn Randomly
nNextSpawnTime = Random(nSpawnDelay) + 1;
while (nNextSpawnTime < nDelayMinimum)
{
nNextSpawnTime = Random(nSpawnDelay) + 1;
}
nNextSpawnTime += nTimeNow;
}
else
{
// Setup Next Spawn
nNextSpawnTime = nTimeNow + nSpawnDelay;
}
return nNextSpawnTime;
}
int IsRestoreBlocked(object oSpawn, location lChildLoc, int iExpireTime,
int nTimeNow)
{
int nSpawnBlock = FALSE;
// Initialize Day/Night Only
int nDayOnly = GetLocalInt(oSpawn, "f_DayOnly");
int nDayOnlyDespawn = GetLocalInt(oSpawn, "f_DayOnlyDespawn");
int nNightOnly = GetLocalInt(oSpawn, "f_NightOnly");
int nNightOnlyDespawn = GetLocalInt(oSpawn, "f_NightOnlyDespawn");
// Initialize Day/Hour Spawns
int nDay, nHour;
int nSpawnDayStart = GetLocalInt(oSpawn, "f_SpawnDayStart");
int nSpawnDayEnd = GetLocalInt(oSpawn, "f_SpawnDayEnd");
int nSpawnHourStart = GetLocalInt(oSpawn, "f_SpawnHourStart");
int nSpawnHourEnd = GetLocalInt(oSpawn, "f_SpawnHourEnd");
// Initialize Child Lifespan
int nChildLifespanMax = GetLocalInt(oSpawn, "f_ChildLifespanMax");
// Initialize SpawnUnseen
float fSpawnUnseen = GetLocalFloat(oSpawn, "f_SpawnUnseen");
int nUnseenIndividual = GetLocalInt(oSpawn, "f_UnseenIndividual");
// Check Against Spawn Unseen (_SUnn|I_)
if (fSpawnUnseen > 0.0)
{
if (nUnseenIndividual)
{
vector vChildPos = GetPositionFromLocation(lChildLoc);
if (CheckPositionUnseen(vChildPos, fSpawnUnseen) == FALSE)
{
nSpawnBlock = TRUE;
}
}
else
{
vector vSpawnPos = GetPositionFromLocation(GetLocation(oSpawn));
if (CheckPositionUnseen(vSpawnPos, fSpawnUnseen) == FALSE)
{
nSpawnBlock = TRUE;
}
}
}
// Check Against Night Only. Since this is a restore of something already
// spawned, it should only be blocked if despawn has been specified for the
// creature (_NOD_)
if (nNightOnly && nNightOnlyDespawn)
{
if (GetIsDay() || GetIsDawn())
{
nSpawnBlock = TRUE;
}
}
// Check Against Day Only (_DOD_)
if (nDayOnly && nDayOnlyDespawn)
{
if (GetIsDay() == FALSE && GetIsDawn() == FALSE)
{
nSpawnBlock = TRUE;
}
}
// Check Against Specific Day(s) (_DYnn_)
if (nSpawnDayStart > -1)
{
nDay = GetCalendarDay();
if (IsBetweenDays(nDay, nSpawnDayStart, nSpawnDayEnd) == FALSE)
{
nSpawnBlock = TRUE;
}
}
// Check Against Specific Hour(s) (_HRnn_)
if (nSpawnHourStart > -1)
{
nHour = GetTimeHour();
if (IsBetweenHours(nHour, nSpawnHourStart, nSpawnHourEnd) == FALSE)
{
nSpawnBlock = TRUE;
}
}
// Check Lifespan (_CLnn_)
if (nChildLifespanMax > -1)
{
if (nTimeNow >= iExpireTime)
{
//debug("restore: lifespawn exceeded");
nSpawnBlock = TRUE;
}
}
return nSpawnBlock;
}
int NeedPseudoHeartbeat( object oArea )
{
int bPCsInArea = GetLocalInt( oArea, SPAWN_PCS_IN_AREA );
int nAreaSpawnCount = GetLocalInt( oArea, SPAWN_AREA_COUNT );
int bHeartbeatScheduled = GetLocalInt( oArea, SPAWN_HEARTBEAT_SCHEDULED );
int bLeftoversForceProcessing = GetLocalInt( GetModule(),
"LeftoversForceProcessing");
// Do a heartbeat if there are PCs in the area or any spawns up, and we
// don't already have a heartbeat scheduled
if (bLeftoversForceProcessing)
{
return ( (bPCsInArea || nAreaSpawnCount) && ! bHeartbeatScheduled );
}
return ( bPCsInArea && ! bHeartbeatScheduled );
}
void Spawn_OnAreaEnter( string sHeartbeatScript = "spawn_sample_hb",
float fHeartbeatInterval = 6.0, float fFirstDelay = 0.0 )
{
object oPC = GetEnteringObject();
object oArea = OBJECT_SELF;
SetLocalString( oArea, SPAWN_HEARTBEAT_SCRIPT, sHeartbeatScript );
SetLocalFloat( oArea, SPAWN_INTERVAL, fHeartbeatInterval );
if ( GetIsPC( oPC ) )
{
SetLocalInt(oPC, "NESS_Player", TRUE);
int nPCsInArea = GetLocalInt( oArea, SPAWN_PCS_IN_AREA );
int nAreaSpawnCount = GetLocalInt( oArea, SPAWN_AREA_COUNT );
nPCsInArea++;
SetLocalInt( oArea, SPAWN_PCS_IN_AREA, nPCsInArea );
if ( NeedPseudoHeartbeat( oArea ) )
{
if ( fFirstDelay > 0.0 )
{
DelayCommand( fFirstDelay, ExecuteScript( "spawn_pseudohb", oArea ) );
}
else
{
ExecuteScript( "spawn_pseudohb", oArea );
}
}
}
}
void Spawn_OnAreaExit()
{
object oPC = GetExitingObject();
object oArea = OBJECT_SELF;
int bIsPC = GetLocalInt(oPC, "NESS_Player");
if ( bIsPC )
{
int nPCsInArea = GetLocalInt( oArea, SPAWN_PCS_IN_AREA );
nPCsInArea--;
SetLocalInt( oArea, SPAWN_PCS_IN_AREA, nPCsInArea );
}
}
void ScheduleNextPseudoHeartbeat( object oArea )
{
float fInterval = GetLocalFloat( oArea, SPAWN_INTERVAL );
if ( fInterval == 0.0 )
{
fInterval = 6.0;
}
DelayCommand( fInterval, ExecuteScript( "spawn_pseudohb", oArea ) );
SetLocalInt( oArea, SPAWN_HEARTBEAT_SCHEDULED, TRUE );
}