ES_PRC8/_module/nss/re_rndenc.nss
Jaysyn904 08e84b4e71 Initial upload
Initial upload.
2023-11-14 12:09:02 -05:00

591 lines
25 KiB
Plaintext

//::///////////////////////////////////////////////
//:: Name re_rndenc
//:: FileName re_rndenc.nss
//:: Copyright (c) 2002 Raymond Miller
//:://////////////////////////////////////////////
/*
This script creates functions called RandomEncounter(),
CleanHouse(), and SetRndEncProperties() for use in the NWN
scripting language. This script is meant to be used as an #include
and is part of the BESIE Random Encounter package by Ray Miller
*/
//:://////////////////////////////////////////////
//:: Created By: Ray Miller
//:: Created On: 7/6/2002
//:://////////////////////////////////////////////
// Encounter Type Constants
int ENCOUNTER_TYPE_AREA = 0;
int ENCOUNTER_TYPE_PARTY = 1;
int ENCOUNTER_TYPE_IND = 2;
// PRIVATE FUNCTION DECLARATIONS
// Sets properties for random encounters that are likely to seldom change
// - iDifficulty 1 to 10
// - bConsiderCR: If true takes CR of creature into consideration when
// choosing an encounter.
// - sCreatureTable: "re_***" - where *** is a string of letter and/or numbers to indicate to the function what type
// of creatures to spawn. They are as follows:
// a - animals
// c - construct
// d - dragon
// e - elemental
// g - giant
// h - humanoid
// i - insect
// m - miscellaneous
// p - planar
// u - undead
// b - bandit
// 0 through 9 - These are for custom encounter tables (see Custom Encounter Tables, later in this section.)
// - iLifeTime: Time in seconds before unengaged encounters decay.
// - Mph: Should equal the Minutes Per Hour setting of the Module.
// - iEncountertype:
// ENCOUNTER_TYPE_PARTY (default) - Takes into consideration the average level of the entire party of the PC who is to
// receive the encounter when choosing an encounter of appropriate difficulty level.
// ENCOUNTER_TYPE_AREA - Takes into consideration the levels off all PCs and henchmen within a 20m radius of the PC
// who is to receive the encounter.
// ENCOUNTER_TYPE_IND - Takes into consideration only the levels of the individual PC who is to receive the encounter.
// - bLOSCheck: Dependant upon a broken scripting function. (future use!)
// Note: This function is best called by the OnModuleLoad or OnAreaLoad handler.
void SetRndEncProperties(int iDifficulty = 4, int bConsiderCR = TRUE, string sCreatureTable = "re_ceghimpub", int iLifeTime = 180, int iMph = 2, int iEncounterType = 1, int bLOSCheck = FALSE);
// Generates the likelihood of a random encounter.
// - fChanceOfEncounter: Odds of encounter spawning when funciton is called. Accurate to two
// decimal places. .01 to 100.00 percent chance.
// - oEncounterObject: The object about which the encounter will spawn, whose levels, party
// or area is considered when determining an appropriate creature.
// - sTemplate: When used as the sCreatureTable parameter in the SetRndEncProperties()
// function this parameter has higher priority. It can also be set to the tag of a
// specific creature, or to "random" to use the default table set by SetRndEncProperties()
// - iMinNumberOfCreatures: If > 1, a random number of creatures between this and iMaxNumberOfCreatures
// will spawn. If set to 0, then exactly the number of creatures set by iMaxNumberOfCreatures will
// spawn.
// - iMaxNumberOfCreatures: If this and iMinNumberOfCreatures is set to 0 then the number of Creatures
// spawned will be determined by the CR of the creature spawned compared to the levels of the player(s).
// - iMinEncounterDistance: If set to 0, encounter distance will always be at the number set by iMaxEncounterDistance.
// - iMaxEncounterDistance: Farthest distance the encounter can be from oEncounterObject.
// - iOrientation: 0 to 360. Counterclockwise representing the angle from facing where the encounter will spawn.
// a value of 0 will spawn the encounter directly in front of oEncounterObject. 360 will generate a random angle.
// - iTolerance: The number of degrees by which the angle can randomly be off from iOrientation.
// - iLevelOverride: Use this to force the function to base the encounter on a character level other than that
// determined by oEncounterObject.
// - iDifficulty: Overrides the difficulty setting determined by the SetRndEncProperties() function.
object RandomEncounter(float fChanceOfEncounter = 100.0, object oEncounterObject = OBJECT_SELF, string sTemplate = "random", int iMinNumberOfCreatures = 0, int iMaxNumberOfCreatures = 0, int iMinEncounterDistance = 1, int iMaxEncounterDistance = 15, int iOrientation = 360, int iTolerance = 0, int iLevelOverride = 0, int iDifficulty = 0);
// Used to "clean up" an area that has become littered by random encounters.
// - bDestroyPlotItems - Tells the function whether or not to destroy items with their plot flags set. If set to TRUE,
// plot items will be destroyed just like any other item.
// - oArea - The area to clean up.
// - iSpawnOverride - Overrides the default (set by the SetRndEncProperties() function) time to destroy random encounter
// creatures who are not engaged by PCs.
// - iItemOverride - Overrides the default time of 30 minutes after which to destroy items dropped by PCs
// Note: Only works if the "re_moditemdrop" script included with the BESIE Random Encounter package
// is placed in the module OnItemUnacquire handler.
// - iBodyBagOverride - Overrides the default time of 5 minutes after which to destroy loot that was dropped by creatures
// who were killed.
// NOTE: If there is bDestroyPlotItems is FALSE and there is a plot item or items inside a container or body bag, the container
// and all non-plot items will decay but the plot item(s) will be left.
// NOTE: A value of zero assigned to the override parameters will cause the function to use the default value for that parameter.
void CleanHouse(int bDestroyPlotItems = FALSE, object oArea = OBJECT_SELF, int iSpawnOverride = 0, int iItemOverride = 0, int iBodyBagOverride = 0);
void SetRndEncProperties(int iDifficulty = 4, int bConsiderCR = TRUE, string sCreatureTable = "re_ceghimpub", int iLifeTime = 180, int iMph = 2, int iEncounterType = 1, int bLOSCheck = FALSE)
{
// INITIALIZE SOME GLOBAL VARIABLES
object oMod = GetModule();
object oPC;
if(GetIsObjectValid(oMod))
{
SetLocalInt(oMod, "bInitialized", TRUE);
SetLocalInt(oMod, "bConsiderCR", bConsiderCR);
SetLocalInt(oMod, "iMph", iMph);
SetLocalInt(oMod, "iLifeTime", iLifeTime);
SetLocalInt(oMod, "bLOSCheck", bLOSCheck);
SetLocalInt(oMod, "iDifficulty", iDifficulty);
SetLocalInt(oMod, "iEncounterType", iEncounterType);
SetLocalString(oMod, "sCreatureTable", sCreatureTable);
SetLocalInt(oMod, "iMinNumberOfCreatures", 0);
SetLocalInt(oMod, "iMaxNumberOfCreatures", 0);
}
}
#include "re_table"
object RandomEncounter(float fChanceOfEncounter = 100.0, object oEncounterObject = OBJECT_SELF, string sTemplate = "random", int iMinNumberOfCreatures = 0, int iMaxNumberOfCreatures = 0, int iMinEncounterDistance = 1, int iMaxEncounterDistance = 15, int iOrientation = 360, int iTolerance = 0, int iLevelOverride = 0, int iDifficulty = 0)
{
// IF PROPERTIES ARE NOT SET THEN SET THEM WITH DEFAULTS
if(!GetLocalInt(GetModule(), "bInitialized"))
{
SetRndEncProperties();
}
// DETERMINE IF ENCOUNTER HAPPENS
// The following two lines allow for a chance of encounter with a precision of up to
// two decimal places. ie. 100.00. An encounter can have as little as a 0.01 chance
// of occuring.
int iHappens = Random(10000)+1;
int iChanceOfEncounter = FloatToInt(fChanceOfEncounter * 100);
if(iChanceOfEncounter < iHappens)
{
return OBJECT_INVALID;
}
// DECLARE AND INITIALIZE VARIABLES
object oMod = GetModule();
int iLifeTime = GetLocalInt(oMod, "iLifeTime");
int bLOSCheck = GetLocalInt(oMod, "bLOSCheck");
if(!iDifficulty) iDifficulty = GetLocalInt(oMod, "iDifficulty") * 2;
int iEncounterType = GetLocalInt(oMod, "iEncounterType");
int iCounter1 = 1; // Used to count the creatures when spawning them.
int iCounter2 = 1; // Used in loop to set difficulty level.
int iCounter3 = 1; // Used in loop to check line of sight float fEncounterDistance (future use!);
int iCounter4;
int iNumberOfCreatures;
int iEncounterDistance;
int iFacingSameWay;
int iLevels;
int bNumberByLevel = FALSE;
int bNoEncounter = FALSE;
int bComplete1 = FALSE;
int bComplete2 = FALSE;
int iMph;
float fMinCR;
float fMaxCR;
float fEncounterDistance;
float fNewEncounterDistance;
float fCreatureFacing;
float fEncounterAngle;
float fEncounterVector;
float fAngleOffset;
float fLevels;
float fDifficulty = 0.167;
vector vEncounterVector;
vector vVectorOffset;
vector vCreatureVector;
object oAmIAPC;
object oCreature;
object oArea;
if(oEncounterObject == GetModule())
{
oAmIAPC = GetFirstPC();
while(GetIsObjectValid(oAmIAPC))
{
SetLocalObject(oMod, "oEncounterObject" + IntToString(iCounter4), oAmIAPC);
iCounter4++;
oAmIAPC = GetNextPC();
}
oEncounterObject = GetLocalObject(oMod, "oEncounterObject" + IntToString(Random(iCounter4)));
}
else if(GetObjectType(oEncounterObject) == 0 && oEncounterObject != GetModule())
{
oArea = oEncounterObject;
oAmIAPC = GetFirstObjectInArea(oArea);
while(GetIsObjectValid(oAmIAPC))
{
if(GetIsPC(oAmIAPC))
{
SetLocalObject(oArea, "oEncounterObject" + IntToString(iCounter4), oAmIAPC);
iCounter4++;
}
oAmIAPC = GetNextObjectInArea(oArea);
}
oEncounterObject = GetLocalObject(oArea, "oEncounterObject" + IntToString(Random(iCounter4)));
}
else
{
oArea = GetArea(oEncounterObject);
}
if(!GetIsPC(oEncounterObject))
iEncounterType = ENCOUNTER_TYPE_AREA;
location lCreatureLocation;
vector vEncounterObjectVector = GetPosition(oEncounterObject);
if(!GetLocalInt(GetModule(), "iMph"))
{
iMph = 2;
}
else
{
iMph = GetLocalInt(GetModule(), "iMph");
}
int iMin = 60;
int iHr = iMin * iMph;
int iDay = iHr * 24;
int iMth = iDay * 28;
int iYr = iMth * 12;
if(iDifficulty > 10)
{
iDifficulty = 10;
}
if(iDifficulty == 0)
{
iDifficulty = GetGameDifficulty() * 2;
}
while(iCounter2 <= iDifficulty)
{
fDifficulty = fDifficulty * 1.5;
iCounter2++;
}
// FURTHER CHECKS TO DETERMINE IF ENCOUNTER HAPPENS
oAmIAPC = GetFirstObjectInShape(SHAPE_SPHERE, 35.0, GetLocation(oEncounterObject), FALSE, OBJECT_TYPE_CREATURE);
if(GetIsObjectValid(oAmIAPC))
{
while(GetIsObjectValid(oAmIAPC))
{
if(GetIsPC(oAmIAPC))
{
if(IsInConversation(oAmIAPC))
{
return OBJECT_INVALID;
}
}
oAmIAPC = GetNextObjectInShape(SHAPE_SPHERE, 25.0, GetLocation(oEncounterObject), FALSE, OBJECT_TYPE_CREATURE);
}
}
// ERROR CORRECTION
if(iMaxNumberOfCreatures < iMinNumberOfCreatures)
{
iMaxNumberOfCreatures = iMinNumberOfCreatures;
}
if(iMaxEncounterDistance < iMinEncounterDistance)
{
iMaxEncounterDistance = iMinEncounterDistance;
}
if(!GetIsPC(oEncounterObject))
{
iEncounterType = ENCOUNTER_TYPE_AREA;
}
// DETERMINE THE ANGLE OFFSET OF THE SPAWN
if(iOrientation == 360)
{
fEncounterAngle = IntToFloat(Random(360));
}
else
{
fEncounterAngle = GetFacingFromLocation(GetLocation(oEncounterObject)) + IntToFloat(iOrientation);
fEncounterAngle = (fEncounterAngle + (IntToFloat(iTolerance) * 0.5)) - (IntToFloat(Random(iTolerance)));
}
// DETERMINE THE DISTANCE FROM THE SPAWNING OBJECT
if(iMinEncounterDistance == 0)
{
iMinEncounterDistance = iMaxEncounterDistance;
fEncounterDistance = IntToFloat(iMaxEncounterDistance);
}
else
{
fEncounterDistance = IntToFloat(iMinEncounterDistance + Random((iMaxEncounterDistance - iMinEncounterDistance) + 1));
}
iEncounterDistance = FloatToInt(fEncounterDistance);
// DETERMINE THE FACING OF THE SPAWN
fCreatureFacing = IntToFloat(Random(360));
iFacingSameWay = Random(2); // Note: If there is more than one creature there is a 50% chance they will all be facing the same direction
// DETERMINE TOTAL CHARACTER LEVELS TO CONSIDER WHEN CHOOSING A CREATURE
// AND/OR DETERMINING THE NUMBER OF CREATURES TO SPAWN.
// If the variable iEncounterType is AREA, this routine
// determines the total character levels
// based upon the character levels of all PCs
// in a 20 meter radius around the object that spawned
// the encounter.
// Later on the total character levels will be compared to
// the challenge rating of the creature spawned, and a number
// of creatures will be determined from that comparison.
if(iEncounterType == ENCOUNTER_TYPE_AREA)
{
oAmIAPC = GetFirstObjectInShape(SHAPE_SPHERE, 20.0, GetLocation(oEncounterObject), FALSE, OBJECT_TYPE_CREATURE);
while(GetIsObjectValid(oAmIAPC))
{
if(GetIsPC(oAmIAPC))
{
iLevels = iLevels + GetLevelByPosition(1, oAmIAPC) + GetLevelByPosition(2, oAmIAPC) + GetLevelByPosition(3, oAmIAPC);
if(GetIsObjectValid(GetHenchman(oAmIAPC)))
{
iLevels = iLevels + GetLevelByPosition(1, GetHenchman(oAmIAPC)) + GetLevelByPosition(2, GetHenchman(oAmIAPC)) + GetLevelByPosition(3, GetHenchman(oAmIAPC));
}
}
oAmIAPC = GetNextObjectInShape(SHAPE_SPHERE, 20.0, GetLocation(oEncounterObject), FALSE, OBJECT_TYPE_CREATURE);
}
}
else if(iEncounterType == ENCOUNTER_TYPE_PARTY)
{
iLevels = GetFactionAverageLevel(oEncounterObject);
}
else
{
// If the variable iEncounterType is set to IND, this
// routine determines the total character levels based upon the
// character level of the object that spawned the encounter.
// if the object that spawned the encounter is NOT a PC then
// the number of creatures spawned will be one. This shouldn't
// happen since the the encounter type sets itself to AREA if
// the encounter object is a placeable.
if(GetIsPC(oEncounterObject))
{
iLevels = GetLevelByPosition(1, oEncounterObject) + GetLevelByPosition(2, oEncounterObject) + GetLevelByPosition(3, oEncounterObject);
}
}
// Modify the float representing the total levels by the difficulty level.
if(iLevelOverride)
{
iLevels = iLevelOverride;
}
fLevels = IntToFloat(iLevels) * fDifficulty;
// CHOOSE A CREATURE TO SPAWN
if(GetStringLowerCase(sTemplate) == "random" || GetStringLowerCase(GetStringLeft(sTemplate, 3)) == "re_")
{
if(GetStringLowerCase(GetStringLeft(sTemplate, 3)) == "re_")
{
sTemplate = GetStringRight(sTemplate, GetStringLength(sTemplate) - 3);
}
if(fLevels < 0.25)
{
fMaxCR = 0.25;
}
else
{
fMaxCR = fLevels;
}
fMinCR = IntToFloat(FloatToInt(fMaxCR * 0.2));
//If there is a definative number of creatures to spawn passed to
//the RandomEncounter function when it is called, then do not
//allow as much play in the low end, and a little more in the
// high end challange ratings.
if(iMinNumberOfCreatures == 0 && iMaxNumberOfCreatures > 1)
{
fMinCR = IntToFloat(FloatToInt(fMaxCR * 0.4));
fMaxCR = fMaxCR * 1.2;
}
if(iMinNumberOfCreatures == 0 && iMaxNumberOfCreatures == 1)
{
fMinCR = IntToFloat(FloatToInt(fMaxCR * 0.6));
fMaxCR = fMinCR * 1.2;
}
if(GetLocalInt(oMod, "bConsiderCR") == FALSE)
{
fMaxCR = 9999.0;
fMinCR = 0.0;
}
sTemplate = GetRndEncCreature(fMinCR, fMaxCR, sTemplate);
}
// DETERMINE LOCATION AND SPAWN ONE CREATURE
// NOTE: Line Of Sight checks have a bug. Bioware says they are looking
// into the bug. I have spent an ungodly amount of hours trying to come
// up with an acceptable work-around to the Line Of Sight functionality
// of Get**ObjectInShape(). Unless somebody else can come up with a working
// LOS check, I have no choice but to disregard LOS checks until they are
// fixed.
//
// if(LOSCheck = TRUE)
// {
// <LOS code goes here>
// }
//
// note: one creature is spawned in now so its challange rating can be
// used to determine if more are needed. (if that option is set)
vEncounterVector = AngleToVector(fEncounterAngle);
vVectorOffset = vEncounterVector * fEncounterDistance;
vCreatureVector = vEncounterObjectVector + vVectorOffset;
lCreatureLocation = Location(oArea, vCreatureVector, fCreatureFacing);
oCreature = CreateObject(OBJECT_TYPE_CREATURE, sTemplate, lCreatureLocation, TRUE);
// DETERMINE THE NUMBER OF ADDITIONAL CREATURES TO SPAWN.
// If the min and max number of creatures in the function call are zero
// then get the min and max number from the local variables in the module
// object.
if(iMinNumberOfCreatures == 0 && iMaxNumberOfCreatures == 0)
{
iMinNumberOfCreatures = GetLocalInt(oMod, "iMinNumberOfCreatures");
iMaxNumberOfCreatures = GetLocalInt(oMod, "iMaxNumberOfCreatures");
}
// Now that we are done with these local integers, we need to clean reset
// them to their defaults so we don't accidentally use old numbers later.
SetLocalInt(oMod, "iMinNumberOfCreatures", 0);
SetLocalInt(oMod, "iMaxNumberOfCreatures", 0);
if(iMinNumberOfCreatures == 0 && iMaxNumberOfCreatures != 0)
{
iNumberOfCreatures = iMaxNumberOfCreatures;
}
if(iMinNumberOfCreatures != 0 && iMaxNumberOfCreatures != 0)
{
iNumberOfCreatures = iMinNumberOfCreatures + Random((iMaxNumberOfCreatures - iMinNumberOfCreatures) + 1);
}
if(iMinNumberOfCreatures == 0 && iMaxNumberOfCreatures == 0)
{
// This is the routine that sets the number of creatures to spawn
// based on their challenge rating and the total character levels.
// It chooses a random number between one half (truncated) and 125
// percent (1 for every 4) of the number of creatures ideal for the
// difficulty level set.
iMaxNumberOfCreatures = FloatToInt(fLevels / GetChallengeRating(oCreature));
iMinNumberOfCreatures = FloatToInt(IntToFloat(iMaxNumberOfCreatures) * 0.75);
iMaxNumberOfCreatures = FloatToInt(IntToFloat(iMaxNumberOfCreatures) * 1.25);
iNumberOfCreatures = iMinNumberOfCreatures + Random((iMaxNumberOfCreatures - iMinNumberOfCreatures) + 1);
if((iNumberOfCreatures < 1) && (iLevels > 0))
{
iNumberOfCreatures = 1;
}
}
// SPAWN THOSE SUCKERS!
while(iCounter1 <= iNumberOfCreatures)
{
// Stick some labels on the creature for record keeping and reference (future use!)
SetLocalInt(oCreature, "bRandomEncounter", TRUE);
SetLocalObject(oCreature, "oRandomEncounterSpawner", oEncounterObject);
SetLocalInt(oCreature, "iRandomEncounterCounter", 1);
SetLocalInt(oCreature, "iRandomEncounterSpawnTime", (GetCalendarYear() * iYr) + (GetCalendarMonth() * iMth) + (GetCalendarDay()* iDay) + (GetTimeHour()* iHr) + (GetTimeMinute() * iMin) + GetTimeSecond());
SetLocalInt(oCreature, "iRandomEncounterLifeTime", iLifeTime);
if(iCounter1 < iNumberOfCreatures)
{
oCreature = CreateObject(OBJECT_TYPE_CREATURE, sTemplate, lCreatureLocation, TRUE);
}
iCounter1++;
// Determine the facing of the next creature
if(iFacingSameWay == FALSE)
{
fCreatureFacing = IntToFloat(Random(360));
lCreatureLocation = Location(oArea, vCreatureVector, fCreatureFacing);
}
}
// Stick a lable on the spawning object for record keeping and reference (future use?)
SetLocalString(oEncounterObject, "sLastRandomEncounterSpawned", GetTag(oCreature));
return oCreature;
}
void CleanHouse(int bDestroyPlotItems = FALSE, object oArea = OBJECT_SELF, int iSpawnOverride = 0, int iItemOverride = 0, int iBodyBagOverride = 0)
{
// IF NOT INITIALIZED THEN DO SO
if(!GetLocalInt(GetModule(), "bInitialized"))
{
SetRndEncProperties();
}
int iMph;
if(!GetLocalInt(GetModule(), "iMph"))
{
iMph = 2;
}
else
{
iMph = GetLocalInt(GetModule(), "iMph");
}
// DECLARE AND INTIALIZE VARIABLES
int iMin = 60;
int iHr = iMin * iMph;
int iDay = iHr * 24;
int iMth = iDay * 28;
int iYr = iMth * 12;
int bShouldIKillHim = TRUE;
int iLifeTime;
int iItemLifeTime;
int iBodyBagLifeTime;
int iPresentTime = (GetCalendarYear() * iYr) + (GetCalendarMonth() * iMth) + (GetCalendarDay() * iDay) + (GetTimeHour() * iHr) + (GetTimeMinute() * iMin) + GetTimeSecond();
object oObject;
// GET EACH OBJECT IN THE AREA AND TEST FOR VALIDITY
object oAmIASpawn = GetFirstObjectInArea();
while(GetIsObjectValid(oAmIASpawn))
{
// IS IT A BODY BAG?
if(GetTag(oAmIASpawn) == "Body Bag" && !GetLocalInt(oAmIASpawn, "bDroppedItem"))
{
SetLocalInt(oAmIASpawn, "bDroppedItem", TRUE);
SetLocalInt(oAmIASpawn, "iDropTime", iPresentTime);
}
// IS IT A DROPPED ITEM?
if(GetLocalInt(oAmIASpawn, "bDroppedItem"))
{
// HAS IT BEEN AROUND TOO LONG?
if(iItemOverride)
{
iItemLifeTime = iItemOverride;
}
else
{
iItemLifeTime = 1800;
}
if(iBodyBagOverride)
{
iBodyBagLifeTime = iBodyBagOverride;
}
else
{
iBodyBagLifeTime = 300;
}
if((iPresentTime - GetLocalInt(oAmIASpawn, "iDropTime") > iItemLifeTime && GetTag(oAmIASpawn) != "Body Bag") || (iPresentTime - GetLocalInt(oAmIASpawn, "iDropTime") > iBodyBagLifeTime && GetTag(oAmIASpawn) == "Body Bag"))// && !GetPlotFlag(oAmIASpawn))
{
if(GetHasInventory(oAmIASpawn))
{
oObject = GetFirstItemInInventory(oAmIASpawn);
while(GetIsObjectValid(oObject))
{
if(!GetPlotFlag(oObject) || bDestroyPlotItems)
{
DestroyObject(oObject, 0.0);
}
oObject = GetNextItemInInventory(oAmIASpawn);
}
}
if(!GetPlotFlag(oAmIASpawn) || bDestroyPlotItems)
{
DestroyObject(oAmIASpawn, 0.0);
}
}
}
// IS HE IS A RANDOM ENCOUNTER?
if(GetLocalInt(oAmIASpawn, "bRandomEncounter"))
{
// HAS HE BEEN AROUND TOO LONG?
if(iSpawnOverride)
{
iLifeTime = iSpawnOverride;
}
else
{
iLifeTime = GetLocalInt(oAmIASpawn, "iRandomEncounterLifeTime");
}
if(iPresentTime - GetLocalInt(oAmIASpawn, "iRandomEncounterSpawnTime") > iLifeTime)
{
// IS HE IN COMBAT?
if(!GetIsInCombat(oAmIASpawn))
{
// GET EACH PC AND TEST IF THE CREATURE SEES HIM
// Note: this is because the creature might be charmed
// or influenced not to attack the PCs by other means.
object oPC = GetFirstPC();
if(GetIsObjectValid(oPC))
{
while(GetIsObjectValid(oPC))
{
if(GetObjectSeen(oPC, oAmIASpawn))
{
bShouldIKillHim = FALSE;
}
oPC = GetNextPC();
}
}
// IF THE CREATURE HAS PASSED ALL OF THESE CHECKS, DESTROY HIM.
if(bShouldIKillHim)
{
DestroyObject(oAmIASpawn, 0.0);
}
}
}
}
oAmIASpawn = GetNextObjectInArea();
}
}