RoT2_PRC8/_module/nss/tab_spawnplacabl.nss
Jaysyn904 499aba4eb3 Initial upload
Initial upload
2023-09-25 18:13:22 -04:00

342 lines
11 KiB
Plaintext

/**
* This script an include used as part of the container respawning system in
* dmc_containergen.
*
* Feel free to use or modify this script freely, with the exception that this
* paragraph must remain unchanged. Please leave some mention of the original
* authors in the comments of the script, and if you have an area in your module
* where you give credit to the developers/contributors, please be sure to
* include us.
*
* @author <a href="mailto:david@carr.name">David Carr</a>
* @author <a href="http://nwvault.ign.com/Files/scripts/data/1046937481401.shtml">
* R.Lowe
* </a>
* @version 0.3
*/
#include "tab_copyfunction"
#include "tab_array"
/**
* PUBLIC: When a container receives user-defined event
* UE_SET_RESPAWN_VARIABLES, this local float should be set to the number of
* seconds between respawns
*/
const string RESPAWN_DURATION_TAG = "RESPAWN_DURATION";
/**
* PUBLIC: When a container receives user-defined event
* UE_SET_RESPAWN_VARIABLES, this local string should be set to the blueprint
* resref of the invisible object to use as a respawner
*/
const string RESPAWNER_RESREF_TAG = "RESPAWNER_RESREF";
/**
* PUBLIC: When a container receives user-defined event
* UE_SET_RESPAWN_VARIABLES, this local int may be set to TRUE if initial
* inventory respawning is desired
*/
const string CONTAINER_STORE_INVENTORY_TAG = "CONTAINER_STORE_INVENTORY";
/**
* PUBLIC: When a container receives user-defined events UE_GENERATE_TREASURE or
* UE_CONTAINER_DESTRUCTION, this local object will be set to the object that
* triggered the event (the character who opened or destroyed the container).
*/
const string CONTAINER_ACTOR_TAG = "CONTAINER_ACTOR";
/**
* PUBLIC: This user-defined event is fired whenever the respawn system needs
* variables to be set. Once you have set these variables, you must call
* {@link #callbackSetRespawnVariables(object)} or the container will not
* respawn.
*
* @see #RESPAWN_DURATION_TAG
* @see #RESPAWNER_RESREF_TAG
* @see #CONTAINER_STORE_INVENTORY_TAG
*/
const int UE_SET_RESPAWN_VARIABLES = 46;
/**
* PUBLIC: This user-defined event is fired whenever it is appropriate to
* generate treasure.
*/
const int UE_GENERATE_TREASURE = 47;
/**
* PUBLIC: This user-defined event is fired whenever a container is destroyed,
* to allow for custom behaviors such as the destruction of fragile treasure.
*/
const int UE_CONTAINER_DESTRUCTION = 48;
/**
* PRIVATE: the variable name which stores whether the respawn timer has elapsed
*/
const string RESPAWN_TIMER_TAG = "RESPAWN_TIMER";
/**
* PRIVATE: the variable name which stores the respawner instance containers
*/
const string RESPAWNER_TAG = "RESPAWNER";
/**
* PRIVATE: the variable name which stores the container instance for spawners
*/
const string INSTANCE_TAG = "SPAWN_INSTANCE";
/**
* PRIVATE: the variable name which stores the blueprint resref for the
* container to respawn
*/
const string CONTAINER_RESREF_TAG = "CONTAINER_RESREF";
/**
* PRIVATE: the name of the array to store item resrefs in for the initial
* container contents
*/
const string RESREF_ARRAY = "ITEM_RESREF";
/**
* PRIVATE: the name of the array to store item stack sizes in for the initial
* container contents
*/
const string STACKSIZE_ARRAY = "ITEM_STACKSIZE";
/**
* PRIVATE: Writes an error message to the log
*
* @param sMsg the message to write
*/
void logError(string sMsg) {
string sErrorMessage = "ERROR: " + sMsg;
WriteTimestampedLogEntry(sErrorMessage);
}
/**
* PRIVATE: Sets the respawner object associated with a container
*/
void setRespawner(object oContainer, object oSpawner) {
SetLocalObject(oContainer, RESPAWNER_TAG, oSpawner);
}
/**
* PRIVATE: Sets the container object associated with the spawner
*/
void setInstance(object oSpawner, object oContainer) {
SetLocalObject(oSpawner, INSTANCE_TAG, oContainer);
}
/**
* PRIVATE: Moves the respawn variables from the container to the spawner
*/
void moveRespawnVariables(object oContainer, object oSpawner) {
moveLocalFloat(oContainer, oSpawner, RESPAWN_DURATION_TAG);
moveLocalInt(oContainer, oSpawner, CONTAINER_STORE_INVENTORY_TAG);
DeleteLocalString(oContainer, RESPAWNER_RESREF_TAG);//no longer needed
}
/**
* Returns true if the inventory should be stored/restored
*/
int shouldStoreInventory(object oObj) {
return GetLocalInt(oObj, CONTAINER_STORE_INVENTORY_TAG);
}
/**
* PRIVATE: Returns the blueprint resref for the container that the spawner
* generates
*/
string getContainerResRef(object oSpawner) {
return GetLocalString(oSpawner, CONTAINER_RESREF_TAG);
}
/**
* PRIVATE: Sets the blueprint resref for the container that the spawner
* generates
*/
void setContainerResRef(object oSpawner, string sResRef) {
SetLocalString(oSpawner, CONTAINER_RESREF_TAG, sResRef);
}
/**
* PRIVATE: Stores the contents of the container on the spawner.
* Note that GetResRef(object) may return an empty string
*/
void storeInventory(object oContainer, object oSpawner) {
int nIndex = 0;
object oItem = GetFirstItemInInventory(oContainer);
while(GetIsObjectValid(oItem)) {
int nStackSize = GetNumStackedItems(oItem);
string sResRef = GetResRef(oItem);
//only store it if we'll be able to create it
if(nStackSize > 0 && sResRef != "") {
SetLocalArrayString(oSpawner, RESREF_ARRAY, nIndex, sResRef);
SetLocalArrayInt(oSpawner, STACKSIZE_ARRAY, nIndex, nStackSize);
nIndex++;
}
oItem = GetNextItemInInventory(oContainer);
}
}
/**
* PRIVATE: Returns the container object associated with the spawner, or
* OBJECT_INVALID if an instance doesn't exist
*/
object getInstance(object oSpawner) {
return GetLocalObject(oSpawner, INSTANCE_TAG);
}
/**
* PRIVATE: Returns TRUE if the specified area is empty
*/
int areaIsEmpty(object oArea) {
object oTarget = GetFirstObjectInArea(oArea);
while(GetIsObjectValid(oTarget)) {
if(GetIsPC(oTarget)) {
return FALSE;
}
oTarget = GetNextObjectInArea(oArea);
}
return TRUE;
}
/**
* PRIVATE: Clears a container's inventory
*/
void clearInventory(object oContainer) {
object oItem = GetFirstItemInInventory(oContainer);
while(GetIsObjectValid(oItem)) {
DestroyObject(oItem);
oItem = GetNextItemInInventory(oContainer);
}
}
/**
* PRIVATE: Clears the hold on treasure generation set by setRespawnTimer
*/
void clearRespawnTimer(object oContainer) {
SetLocalInt(oContainer, RESPAWN_TIMER_TAG, FALSE);
}
/**
* PRIVATE: Restores the contents of the container from the spawner
*/
void restoreInventory(object oSpawner, object oContainer) {
int nLength = GetLocalArrayLength(oSpawner, RESREF_ARRAY);
int nIndex;
for(nIndex = 0; nIndex < nLength; nIndex++) {
string sResRef = GetLocalArrayString(oSpawner, RESREF_ARRAY, nIndex);
int nStackSize = GetLocalArrayInt(oSpawner, STACKSIZE_ARRAY, nIndex);
if(sResRef == "") {
logError("invalid item resref");
continue;//ignore this one
}
if(nStackSize < 1) {//if the stack size wasn't properly set
logError("invalid stack size");
nStackSize = 1;//deal with it gracefully
}
CreateItemOnObject(sResRef, oContainer, nStackSize);
}
}
/**
* PRIVATE: If we need to create a new container and don't have a resref, fail.
* If there's an error when we try to create a new container, fail.
* If we make a new container, tell it that we're its respawner and store
* for our reference that it's our instance. If we don't need a new
* instance, clear the container and reset the variable so that treasure will
* respawn. If we stored the inventory for the container, restore it.
*/
void ActionRespawnContainer(object oSpawner) {
object oContainer = getInstance(oSpawner);
string sResRef = getContainerResRef(oSpawner);
location lLoc = GetLocation(oSpawner);
//Check to see if we could respawn the container
if(sResRef == "") {
logError("no resref for chest");
DestroyObject(oSpawner, 0.5);
return;
}
//Check to see if the area is empty so that we can destroy and respawn the
//container to reset the traps.
if(GetIsObjectValid(oContainer)) {
object oArea = GetArea(oContainer);
if(areaIsEmpty(oArea)) {
clearInventory(oContainer);
DestroyObject(oContainer);
oContainer = OBJECT_INVALID;
}
}
if(!GetIsObjectValid(oContainer)) {//if the container was destroyed, respawn
oContainer = CreateObject(OBJECT_TYPE_PLACEABLE, sResRef, lLoc);
setRespawner(oContainer, oSpawner);//Tell the container we're its respawner
setInstance(oSpawner, oContainer);//Set the respawner's instance
} else {//we need to reset the flag so that treasure is generated
if(GetLockLockable(oContainer) && !GetLocked(oContainer)) {
SetLocked(oContainer, TRUE);//reset the lock
}
clearRespawnTimer(oContainer);
}
if(!GetIsObjectValid(oContainer)) {//if we failed to respawn it
logError("could not create container");
DestroyObject(oSpawner, 0.5);
return;
}
if(shouldStoreInventory(oSpawner)) {
clearInventory(oContainer);
restoreInventory(oSpawner, oContainer);
}
}
/**
* PRIVATE: Returns the number of seconds between respawns
*/
float getRespawnDuration(object oObj) {
return GetLocalFloat(oObj, RESPAWN_DURATION_TAG);
}
/**
* PRIVATE: Adds a respawn event to the event queue for the spawner
*
* @param oSpawner the spawner to add an event for
*/
void addRespawnToEventQueue(object oSpawner) {
AssignCommand(oSpawner, ClearAllActions());
AssignCommand(oSpawner,
DelayCommand(getRespawnDuration(oSpawner), ActionRespawnContainer(oSpawner))
);
}
/**
* PRIVATE: Sets all appropriate variables on the spawner and adds a respawn
* event to its queue.
*/
void configureSpawner(object oSpawner, object oContainer) {
setRespawner(oContainer, oSpawner);
setInstance(oSpawner, oContainer);
moveRespawnVariables(oContainer, oSpawner);
if(shouldStoreInventory(oSpawner)) {
storeInventory(oContainer, oSpawner);
}
setContainerResRef(oSpawner, GetResRef(oContainer));
addRespawnToEventQueue(oSpawner);
}
/**
* PRIVATE: Returns the resref of the respawner blueprint
*/
string getRespawnerResRef(object oObj) {
return GetLocalString(oObj, RESPAWNER_RESREF_TAG);
}
/**
* PUBLIC: Called from the user-defined UE_SET_RESPAWN_VARIABLES event. Allows
* us to take actions, such as initialize the respawner, after the respawn
* variables have been set.
*
* @param oContainer the container to set up a respawner for
*/
void callbackSetRespawnVariables(object oContainer) {
location lLoc = GetLocation(oContainer);
string sResRef = getRespawnerResRef(oContainer);
//create a new spawner at the container's location
object oSpawner = CreateObject(OBJECT_TYPE_PLACEABLE, sResRef, lLoc);
if(!GetIsObjectValid(oSpawner)) {
logError("failed spawner creation: " + sResRef);
} else {
configureSpawner(oSpawner, oContainer);
}
}