PRC8/nwn/nwnprc/trunk/include/x3_inc_horse.nss
Jaysyn904 6ec137a24e Updated AMS marker feats
Updated AMS marker feats.  Removed arcane & divine marker feats.  Updated Dread Necromancer for epic progression. Updated weapon baseitem models.  Updated new weapons for crafting & npc equip.
 Updated prefix.  Updated release archive.
2024-02-11 14:01:05 -05:00

3435 lines
171 KiB
Plaintext

//::///////////////////////////////////////////////
//:: Horse Mounting and Control System
//:: x3_inc_horse
//:: Copyright (c) 2007-2008 Bioware Corp.
//:://////////////////////////////////////////////
/*
Holds horse specific functionality which can be used
in other scripts to support horses as present in patch 1.69.
*/
//:://////////////////////////////////////////////
//:: Created By: Deva B. Winblood
//:: Tweaking By: Martin "Azbest" Psikal
//:: Created On: Dec 14th, 2007
//:: Last Update: April 21th, 2008
//:: Help from community members: Proleric,
//:: The Krit, OldMansBeard, FriendlyFire,
//:: Bannor "JP" Bloodfist, Brian Chung,
//:: Craig Welburn, Barry_1066, Jahodnik,
//:: fluffyamoeba, Axe Murderer, and more.
//:: Thank you!
//:://////////////////////////////////////////////
#include "x0_i0_position"
#include "X0_INC_HENAI"
#include "x3_inc_skin"
/*
HOW TO GET THE MOST FROM THIS HORSE SCRIPT:
===========================================
This script has been setup to simply work with the default horses but, some
effort was made to provide you with some simple methods to hook your own
mounts, mounting scripts, etc. into this script.
[HORSE BLUEPRINT VARIABLES]
===========================
The following variables can be set on the BLUEPRINT of a creature to extend
this mounting system. These scripts could be used to add database
functionality, or extend the system however, you like.
bX3_IS_MOUNT = integer that should be set to 1 if this is a custom blueprint
mount that for some reason the script indicates is not a mount.
X3_HORSE_PREMOUNT_SCRIPT = script to fire before the horse is mounted. If
this script determines it cannot mount then it will set X3_HORSE_NOMOUNT to
TRUE on the mount. You can use a script like this to create support for
saddlebags or to
use this to extend the mounting system. This will be executed by the rider
and the variable oX3_TempHorse will be set pointing to the horse so that
the custom script knows which horse this relates to.
X3_HORSE_POSTMOUNT_SCRIPT = can be set to a script that can be fired after
this mount is mounted. This may be useful for adding things like feats and
other after mounting needs. If this
script does not exist then the standard Speed, Skill Decreases, Mounted
Archery Adjustments, etc. will be applied. If you want your own post mount
script then it is important to note you will have to apply these modifiers
if you also want them to be used within your own script. The horse that
was mounted will be referenced by oX3_TempHorse as stored on the Rider. If
you use such a custom hook script make sure to set bX3_IS_MOUNTED to TRUE by
using the SetSkinInt() function from x3_inc_skin.
on the rider after they mount. This is required in some cases for the
dismount radial to work. (mainly when working with custom mounts)
X3_HORSE_PREDISMOUNT_SCRIPT = can be set to a script that needs to be
executed before the dismount portion continues. If X3_HORSE_NODISMOUNT is
set to TRUE then the horse dismount will be aborted.
X3_HORSE_POSTDISMOUNT_SCRIPT = can be set to a script to be executed after
dismount to reverse steps that may have occurred with a POSTMOUNT script
such as adding feats. This script could be used to remove feats.
It is possible you might want to add a new horse/mount type that does not
fit neatly into the appearance.2da or tails.2da with the other horse types.
The following variables if set to any number greater than 0 will override
the default settings and use what you specifiy instead.
X3_HORSE_NULL_APPEARANCE = the appearance to use when scaling the horse for
the mounting animation.
X3_HORSE_TAIL = the tail to use with the tails.2da that defines this horse.
X3_HORSE_FOOTSTEP = the footstep number to use when this horse is mounted.
X3_HORSE_MOUNT_DURATION = is the duration in seconds 0.0f that the mount
animation should take with this horse. This only needs to be set if
the mounting animation for this blueprint is faster or longer than the
default animations.
X3_HORSE_MOUNT_SPEED = is the mount speed increase or decrease that should
be used with this mount. If the value is 0 then it will use the
HORSE_DEFAULT_SPEED_INCREASE constant value.
X3_HORSE_DISMOUNT_DURATION = is the duration in seconds 0.0f that the dis-
mount animation should take with this hores. This only needs to be set if
the dismounting animation for this blueprint is faster or longer than the
default animations.
X3_HORSE_RESTRICT_race = is a variable that set to TRUE this horse cannot
be mounted by the specified race. Supported races are ELF, HUMAN, HALFELF,
DWARF, HALFORC, HALFLING, and GNOME, CUSTOM# = racial type number.
X3_CUSTOM_RACE_APPEARANCE = a variable to set on a rider if they use a custom
racial appearance. This value should be set to what appearance number they
use from appearance.2da. This will prevent custom races from being denied
mounting rights due to the script thinking the rider is shape shifted.
X3_CUSTOM_RACE_MOUNTED_PHENO = This variable should be used on the rider if they
should use a special phenotype when mounted. This is provided to support
custom races.
X3_CUSTOM_RACE_JOUST_PHENO = This variable should be used on the rider if
they need a special phenotype when mounted in joust mode.
X3_CUSTOM_RACE_PHENOTYPE = This variable should be used on the rider if they
need a special phenotype when not mounted.
X3_CUSTOM_RACE_MOUNTED_APPEARANCE = a variable to set on the rider to
indicate which appearance they should use when mounted.
X3_HORSE_OWNER_TAG = a string that can be set on the horse that will tell
it to Add itself as a henchman to an NPC with the specified tag.
X3_HORSE_NOT_RIDEABLE_OWNER = if this integer is set to 1 on the Horse then
Mount will not be useable and the error it will return if asked is that it
is NOT rideable due to it being owned by someone else. This is useful if
you want horses around that the PCs and Henchmen cannot mount for reasons
such as they are owned by a store, etc.
bX3_IS_MOUNT = a variable that should be set to TRUE if a mount is added
as a henchman but, is still mountable.
X3_NO_MOUNT_ANIMATE = if this integer is set to 1 then this mount does not EVER
animate mounting or dismounting.
X3_TOTAL_MOUNT_ANIMATION_DELAY = a variable containing a time lot indicating
how much time the routine has before it needs to be finished. it is used for
the sake of synchronizing animation and the process running in the background,
exclusively used in mounting animation portion of the HorseMount() routine,
but can be used elsewhere. Note, that the variable is artificially set even
in case no animation is desired so that the code doesnt happen instantly. It
is not meant to be changed, unless something bad is happening timing-wise.
The value is precalculated and in our particular case it is supposed to hold
the total animation length.
The following make it easier to implement existing mount support systems
in conjunction with this one.
X3_HORSE_SCRIPT_MOUNT = Script to call for mounting instead of using the
default one called by the horse menu feat. This still checks to make sure
mounting in the area is legal first. This is intended for making the radial
menus do something different just for this horse and is not intended to
alter the HorseMount(), HorseDismount(), etc. functions. There are already
POSTMOUNT, POSTDISMOUNT, PREMOUNT, and PREDISMOUNT hooks which make that
functionality possible so, that is not what this variable is for.
X3_HORSE_SCRIPT_DISMOUNT = Script to call for dismounting instead of using
the default one called by the horse menu feat. This is intended for making
the radial menus do something different just for this horse and is not
intended to alter the HorseMount(), HorseDismount(), etc. functions. There
are already POSTMOUNT, POSTDISMOUNT, PREMOUNT, and PREDISMOUNT hooks which
make that functionality possible so, that is not what this variable is for.
X3_HORSE_SCRIPT_ASSIGN = Script to call for assign mount instead of using
the default one called by the horse menu feat. This is intended for making
the radial menus do something different just for this horse and is not
intended to alter the HorseMount(), HorseDismount(), etc. functions. There
are already POSTMOUNT, POSTDISMOUNT, PREMOUNT, and PREDISMOUNT hooks which
make that functionality possible so, that is not what this variable is for.
bX3_HAS_SADDLEBAGS = Integer that if set to 1 on the mount will indicate
the horse has saddle bags. It will support inventory control if it is
enabled (NOT: by default). You will also want to set the dialog
X3_DLG_SADDLEBAG on the horse blueprint. Or, create your own dialog that
handles what the saddlebags one does. You will only be able to access
saddlebags of associates in your party.
[AREA RELATED VARIABLES TO SET]
===============================
X3_NO_HORSES = Horses not allowed in this area
X3_NO_MOUNTING = Horses may not be ridden in this area and anyone attempting
to do so should be forcibly dismounted.
X3_HITCHING_POST = if a placeable or waypoint in the area a person was in
before entering a new area has a tag of this then it will move any horses
to this object and set them to STAND_GUARD mode.
X3_MOUNT_OK_EXCEPTION = is an integer that if set to 1 on this area will
override the external and underground restrictions for this area that may
be set module wide.
fX3_MOUNT_MULTIPLE = is a floating point value that is multiplied times
all delays if it exists on the area. It can be used to make an area perform
the mounting animations at a different speed if you have really busy areas
and want to make the animation faster or slower. If this is not defined
then it will always be a value of 1.0. If it is set to 0.5 then delays
will be shortened by half. If it is set to 2.0 then the delays will take
200% longer. This is something a module builder will need to be aware of
and can adjust. NOTE: This can be set on the PC as well and whichever number
is larger is the one that will be used.
fX3_DISMOUNT_MULTIPLE = is similar to fX3_MOUNT_MULTIPLE and should only be
supplied if you want the dismount to use a different speed than the mount
multiple. It can be set on the area or on the PC. The PC will take
priority over the area.
bX3_MOUNT_NO_ZAXIS = value to set to TRUE or 1 on the module, PC, or area
to indicate when calculating the proper mounting location you do not want
the Z Axis to be included in the measurement. This has been found to
work well in areas where you do not want the Z axis to be measured in terms
of whether to perform the mounting animation or not.
X3_ABORT_WHEN_STUCK = if set to TRUE, distance between player and horse is
recorded and checked in each cycle against the current one, when moving to
a horse during the mounting procedure. Should the two ever be equal, meaning
that the player got stuck on his way to horse, the mounting procedure will
be terminated. This is handy in cases when horses are behind obstacles that
are hard to overcome like walls, but the timer that ensures that rider can
mount his horse even in difficult terrain, would eventually force-mount the
potential rider, which could seem like an illogical act. This doesn't need
to be used when using X3_HORSE_ACT_VS_DELAY option, where the timer starts
ticking only after the player gets as close as 1.5m to a horse. The switch
can be set on an area or horses (ie. using an OnEnter script of a trigger).
[MODULE RELATED VARIABLES TO SET]
=================================
X3_HORSE_PALADIN_USE_PHB = integer that if set to 1 on the module object
will cause the script to use paladin mount summoning durations as specified
in the Player's Handbook 3.5 edition rather than just defaulting to 24 hours.
X3_HORSE_DISABLE_SPEED = integer that if set to 1 on the module object
will indicate you do not want a speed increase applied when a person mounts.
X3_HORSE_DISABLE_SKILL = integer that if set to 1 on the module object will
indicate you do not want the skill decreases to be applied when a person
mounts.
X3_HORSE_ENABLE_ACBOOST = integer that if set to 1 on the module object will
indicate that you want the PCs AC to be increased if need be to at least
match that of the horse that is being mounted.
X3_HORSE_ENABLE_HPBOOST = integer that if set to 1 on the module object will
indicate that you want the PCs Hit Points to be increased by half of the
hit points of the mount when it is mounted.
X3_NO_MOUNT_COMMANDABLE = integer that if set to 1 on the module object will
indicate that you do not want the SetCommandable() commands to be used with
the module.
X3_NO_MOUNTED_COMBAT_FEAT = integer that if set to 1 on the module object
will indicate that you do not want the special code added to Bioware scripts
to try to support Mounted Combat close to how it is in Player's Handbook to
be used.
X3_ENABLE_MOUNT_DAMAGE = integer that if set to 1 will attempt to transfer
some damage to the mount when a rider dismounts if damage occurred while
they were mounted.
X3_ENABLE_MOUNT_DB = integer that if set to 1 will enable database and
persistent world support with this script. You will want to modify the
HORSE_Support functions related to the database so, that they write and
read properly however you have the database setup in your module. You will
also want to plan on using something like the x3_mod_def_hb script for your
module heartbeat script if you are going to use the database. This is not
used on modules by default.
X3_NO_SHAPESHIFT_SPELL_CHECK = integer that if set to 1 on the module will
prevent the script from checking to see if a shapeshifted spell is targetted
on a mounted creature. If this variable is set to 1 then the
x2_inc_spellhook scripts will work exactly like they did before horses were
introduced with no concern whether the target is mounted or not.
X3_MOUNT_NO_REST_DISMOUNT = integer that if set to 1 on the module will make
it so, you are able to rest while mounted.
X3_MOUNT_NO_REST_DESPAWN = integer that if set to 1 on the module will make
it so, your paladin mount is not despawned when you rest and adheres
strictly to his summoned duration. If time is advanced by resting then
it is still possible it will despawn.
X3_MOUNTS_EXTERNAL_ONLY = integer that if set to 1 on the module will make
it so mounts can only be ridden in external areas. There is an exception
variable that can be set on an area to override this.
X3_MOUNTS_NO_UNDERGROUND = integer that if set to 1 on the module will make
it so mounts cannot be ridden in underground areas. There is an exception
variable that can be set on an area to override this.
X3_HORSE_ENABLE_SADDLEBAGS = integer that if set to 1 on the module will
enable inventory support for the horse. NOTE: If you want it to use a quick
non-database method for storing the inventory place a waypoint with the tag
X3_HORSE_INVENTORY_STORAGE somewhere in an area that a PC can never get to.
If this waypoint does not exist then it will assume that the database is to
be used. If you are using a database it is adviseable that you change the
support functions because, they use the standard databases and it will often
prove slower than you may like.
X3_SADDLEBAG_DATABASE = string to set to the name of the database to use
for storing saddlebag inventory. If no name is specified it will use the
module tag and a small modifier.
X3_HORSE_NO_HENCHMAN_INCREASE = integer that if set to 1 on the module will
prevent the henchmen from being increased to make room for the horse.
X3_HORSE_MAX_HENCHMEN = integer to set on the module to indicate the maximum
number of henchmen to allow it to be increased to in order to make room for
horses. By default there is no maximum.
X3_HORSE_NO_CORPSES = integer to set on the module to 1 to indicate you
do not want lootable horse corpses created when a mounted PC or NPC dies.
X3_RESTORE_HENCHMEN_LOCATIONS = integer to set to 1 on the module if you
want henchmen's henchmen to be restored to a location near the henchman that
is their master when a PC master of the henchman connects. This is NOT
enabled by default to prevent problems with older modules.
X3_EXTEND_PALMOUNT = string variable that if set on the module object
will allow a person to extend this paladin mount script to call other
scripts in a daisy chain type situation such as saddle bag handling ones.
X3_EXTEND_PALDMOUNT = string variable that if set on the module object
will allow a person to extend this paladin mount script to call other
scripts in a daisy chain type situation such as saddle bag handling ones.
X3_HORSE_ACT_VS_DELAY = integer to set on the module to 1 to indicate you
want the system to use Actions as opposed to delays in some portions of the
mounting sequence. Doing this might provide another way to handle inaccessible
horses besides using X3_HORSE_NOT_RIDEABLE_OWNER. It may result in not
being able to access accidentally poorly placed horses due to scripts or
other factors but, it may be desired by some module designers so, it has
been provided as an option.
[ON THE MODULE or THE PLAYER]
=============================
X3_PALMOUNT_SUMMONOVR = string if set on the module or the player (player has
priority) it will do all the checks to see if a summon is okay. Then if
this variable is set it will execute the script you define here rather than
using the standard summon paladin mount function. Beware: If you use this
then handling all other aspects of this mount become your responsability.
fX3_TIMEOUT_TO_MOUNT = value to set on the module or PC to indicate how long
the PC/NPC should attempt to move into a proper mounting animation to perform
the mounting animation. When this time is reached if it is still not in
position it will instant mount instead and will not animate. If this value is
lower than 6.0 or is not set, then the default value of 18.0 will be used.
fX3_FREQUENCY = frequency of recursive call of the HorseMount() function to
try and initiate new pathfinding to the horse everytime until the character
reaches the mounting position or until the time limit for mounting is up or
unless X3_HORSE_ACT_VS_DELAY is set to TRUE, in which case the actionqueue
is not locked and moving towards a horse is interruptable, ie. by clicking.
if set larger than 9.0 or less than 1.0 the value defaults to 2.0 seconds.
bX3_MOUNT_NO_ZAXIS = value to set to TRUE or 1 on the module, PC, or area
to indicate when calculating the proper mounting location you do not want
the Z Axis to be included in the measurement. This has been found to
work well in areas where you do not want the Z axis to be measured in terms
of whether to perform the mounting animation or not.
[SCRIPT TO OVERRIDE SKIN SYSTEM]
================================
The skin system is designed to supply the horse radial menus to older characters
and it also is used to track some of the information about the state of the
PC semi-persistently. The system is setup in such a way that PRC or HCR or
other systems that use skins will still use their own skin as long as they
run first. If the horse system does not detect a skin it will create one.
Thus, if it runs before PRC or HCR create their skins there might be a problem.
To make this less of an upgrade nightmare the horse script includes support
for a hook script. If PRC, or HCR or other scripting package people create
a script named [x3_mod_pre_enter] and make that script add their skin to the
PC then this script WILL execute ALWAYS before the horse scripts create
their skin. This means all you need to do is make such a script and have
it add your skin and your scripts should function as they did before (EVEN)
if the horse scripts execute other aspects before yours.
[DEBUGGING AND TWEAKING SCRIPTS]
================================
Press ~ in game to bring up console mode. Type DebugMode 1 and press enter.
Press ~ in game and this time type dm_runscript scriptname and press enter.
Press ~ in game and type DebugMode 0 when you want to disable debug mode.
SCRIPTS:
x3_fix_horse = use this when the PC is being treated as mounted when it is not
x3_fix_horseout = use this to make a module not designed with horses in mind
only allow horses in outside areas.
x3_fix_nocmd = set it not to use the SetCommandable with mounting toggle
x3_fix_act = set to use actions when mounting toggle
x3_fix_speed100 = mount/dismount speed multiple set to normal 100%
x3_fix_speed125 = mount/dismount speed multiple set to 1255 (25% slower)
x3_fix_speed150 = mount/dismount speed multiple set to 150% (50% slower)
x3_fix_speed200 = mount/dismount speed multiple set to 200% (100% slower)
*/
////////////////////////////////////////////////////////////////////////////////
/////////////////////////////// CONSTANTS //////////////////////////////////
////////////////////////////////////////////////////////////////////////////////
const string X3_HORSE_DATABASE = "X3HORSE";
// The following constants are provided to facilitate extending the system
// if need be. They are also provided in the event that a hak pack used
// in a module might rearrange the location of some of the aspects needed
// to handle horses.
// This constant points to the location in the appearance.2da where the
// unmounted horse appearances occur.
const int HORSE_APPEARANCE_OFFSET = 496;
// This constant points to the location in the tails.2da where the horse
// appearances begin.
const int HORSE_TAIL_OFFSET = 15;
// This constant lists how many horses are listed beginning at the location
// specified by the offset.
const int HORSE_NUMBER_OF_HORSES = 65;
// This is the prefix that should be used with the paladin mounts when spawning
// them.
const string HORSE_PALADIN_PREFIX = "x3_palhrs";
// The constants that follow here specify what appearance to use when scaling
// the horse as a tail during mounting process to make the animation be handled
// properly.
const int HORSE_NULL_RACE_DWARF = 562;
const int HORSE_NULL_RACE_ELF = 563;
const int HORSE_NULL_RACE_HALFLING = 565;
const int HORSE_NULL_RACE_HUMAN = 568;
const int HORSE_NULL_RACE_HALFELF = 566;
const int HORSE_NULL_RACE_HALFORC = 567;
const int HORSE_NULL_RACE_GNOME = 564;
// The constants that follow here specify the appearance that should be used
// when the specified mount is mounted. These appearances are often required
// to set the proper speeds, radiuses, etc. They also have the complete
// phenotypes and animations associated with them.
const int HORSE_RACE_MOUNTED_DWARFM = 483;
const int HORSE_RACE_MOUNTED_DWARFF = 482;
const int HORSE_RACE_MOUNTED_ELFM = 485;
const int HORSE_RACE_MOUNTED_ELFF = 484;
const int HORSE_RACE_MOUNTED_GNOMEM = 487;
const int HORSE_RACE_MOUNTED_GNOMEF = 486;
const int HORSE_RACE_MOUNTED_HALFLINGM= 489;
const int HORSE_RACE_MOUNTED_HALFLINGF= 488;
const int HORSE_RACE_MOUNTED_HALFELFM = 491;
const int HORSE_RACE_MOUNTED_HALFELFF = 490;
const int HORSE_RACE_MOUNTED_HALFORCM = 493;
const int HORSE_RACE_MOUNTED_HALFORCF = 492;
const int HORSE_RACE_MOUNTED_HUMANM = 495;
const int HORSE_RACE_MOUNTED_HUMANF = 494;
// The following constants indicate which phenotype numbers should be used by
// the mounting system. _N specifies the mounting race started as a normal
// phenotype, and _L specifies the race started as a large phenotype.
const int HORSE_PHENOTYPE_MOUNTED_N = 3;
const int HORSE_PHENOTYPE_MOUNTED_L = 5;
const int HORSE_PHENOTYPE_JOUSTING_N = 6;
const int HORSE_PHENOTYPE_JOUSTING_L = 8;
// The following constants indicate which animation numbers indicate which
// animations can be used well with the horse system.
const int HORSE_ANIMATION_MOUNT = 41;
const int HORSE_ANIMATION_DISMOUNT = 42;
const int HORSE_ANIMATION_LOOPING_JOUST_VIOLENT_FALL = ANIMATION_LOOPING_CUSTOM3;
const int HORSE_ANIMATION_LOOPING_JOUST_GLANCE = ANIMATION_LOOPING_CUSTOM4;
const int HORSE_ANIMATION_LOOPING_JOUST_FALL = ANIMATION_LOOPING_CUSTOM5;
const int HORSE_ANIMATION_LOOPING_JOUST_STAB = ANIMATION_LOOPING_CUSTOM10;
const int HORSE_ANIMATION_LOOPING_JOUST_HELMOFF = ANIMATION_LOOPING_CUSTOM4;
// The following constant defines what footstep sound should be used when
// the horse is mounted.
const int HORSE_FOOTSTEP_SOUND = 17;
// The following constant defines the duration in seconds that it should take
// to complete the default animation.
const float HORSE_MOUNT_DURATION = 2.0f;
// The following constant defines the duration in seconds that it should take
// to complete the default animation.
const float HORSE_DISMOUNT_DURATION = 3.0f;
// The following constants refer to specific feats that might be needed to
// handle horses
const int IP_CONST_HORSE_MENU = 40;
// The constant below designates the default speed increase that should be
// granted when a person mounts a horse.
const int HORSE_DEFAULT_SPEED_INCREASE= 99;
// The following constants were added to support the new feats. When these
// constants are added to the master constants list they can be removed from
// this part of the script.
//const int FEAT_MOUNTED_COMBAT = 1087;
//const int FEAT_MOUNTED_ARCHERY = 1088;
// Scripts called by the above feats
// x3_s3_horse
// x3_s3_palmount
// This constant is a safety to be inserted between the ClearAllActions and
// a command to play a dismount animation. If set to 0.0 or if missing, the
// dismount animation will not play properly when trying to dismount a horse
// while in motion. If you ever encounter problems with mounting animation,
// try inserting this small delay in the same fashion in the animated mounting
// procedure. All pending DelayCommands shall respect this delay accordingly
// for the sake of precise timing. Values of 0.8f and above are safe to use.
const float X3_ACTION_DELAY = 0.8f;
////////////////////////////////////////////////////////////////////////////////
//////////////////////////////// PROTOTYPES ////////////////////////////////
////////////////////////////////////////////////////////////////////////////////
// FILE: x3_inc_horse FUNCTION: HorseMount()
// This function will cause the calling object to attempt to mount oHorse.
// If bAnimate is set to TRUE this will cause the calling object and oHorse
// to be involved in the complete animating process. If you bAnimate is false
// and bInstant is false the caller will still walk to the horse before mounting.
// If bAnimate is false and bInstant is TRUE then the script will immediately
// mount oHorse without any moving (this is the quickest method if needed for
// situations like cutscenes). nState is a variable used by the function itself
// and need not be adjusted when this function is called.
void HorseMount(object oHorse,int bAnimate=TRUE,int bInstant=FALSE,int nState=0);
// FILE: x3_inc_horse FUNCTION: HorseDismount()
// This function will cause the calling object to dismount if it is mounted. If
// bAnimate is TRUE then it will animate the dismount. If bOwner is set to
// TRUE then it will set the object that dismounted as the owner. This function
// will return the object that is the dismounted horse.
object HorseDismount(int bAnimate=TRUE,int bSetOwner=TRUE);
// FILE: x3_inc_horse FUNCTION: HorseSetOwner()
// This function will set oOwner as the owner of oHorse. If bAssign is set to
// TRUE it will also set this horse to be oOwners assigned mount. This is not
// done by default so more than one mount can be assigned to a single target if
// need be.
void HorseSetOwner(object oHorse,object oOwner,int bAssign=FALSE);
// FILE: x3_inc_horse FUNCTION: HorseRemoveOwner()
// This function will remove the owner from oHorse. This will not work on
// Paladin mounts.
void HorseRemoveOwner(object oHorse);
// FILE: x3_inc_horse FUNCTION: HorseGetCanBeMounted()
// This function will return TRUE if oTarget can be mounted. If oRider is
// specified it will also make sure that oTarget can be ridden by oRider.
// The bAssignMount field should be set to TRUE if the call of this function
// should ignore whether someone can mount a horse in the area or not.
int HorseGetCanBeMounted(object oTarget,object oRider=OBJECT_INVALID,int bAssignMount=FALSE);
// FILE: x3_inc_horse FUNCTION: HorseSummonPaladinMount()
// This function can be used to cause the calling object to summon the paladin
// mount. If bPHBDuration is set to TRUE then the mount will use the Players
// Handbook 3.0 edition rules for the duration that the mount will stay around.
object HorseSummonPaladinMount(int bPHBDuration=FALSE);
// FILE: x3_inc_horse FUNCTION: HorseUnsummonPaladinMount()
// This function can be used to cause the calling object to unsummon its paladin
// mount.
void HorseUnsummonPaladinMount();
// FILE: x3_inc_horse FUNCTION: HorseGetIsMounted()
// This function will return TRUE if oTarget is mounted.
int HorseGetIsMounted(object oTarget);
// FILE: x3_inc_horse FUNCTION: HorseCreateHorse()
// This function will create a horse based on the blueprint sResRef at location
// lLoc and will set the owner of the horse to oOwner. If sTag is set to
// something other than "" it will set the horse to that tag. If nAppearance,
// nTail, and nFootstep are set to something other than -1 then it will set the
// horse to that appearance, tail, or footstep. The function will return the
// horse that is spawned. This function is setup the way it is so, that you
// could potentially use a single blueprint to store multiple appearance horses.
// sScript is a specific script that should be executed on the horse after it is
// spawned. This is again supplied to further support multiple horses with a
// single blueprint.
object HorseCreateHorse(string sResRef,location lLoc,object oOwner=OBJECT_INVALID,string sTag="",int nAppearance=-1,int nTail=-1,int nFootstep=-1,string sScript="");
// FILE: x3_inc_horse FUNCTION: HorseGetOwner()
// This function will return an object pointer to whom the owner of oHorse is
// if there is a valid horse. If there is no owner it will return
// OBJECT_INVALID.
object HorseGetOwner(object oHorse);
// FILE: x3_inc_horse FUNCTION: HorseSetPhenotype()
// This function will set oRider to the correct mounted phenotype for riding
// a horse. If bJoust is set to TRUE it will set oRider to the mounted jousting
// phenotype. This is a special phenotype with differing animation sets and
// designed to hold the lance in a very specific way.
void HorseSetPhenotype(object oRider,int bJoust=FALSE);
// FILE: x3_inc_horse FUNCTION: HorseInstantMount()
// This function is primarily supplied for cutscenes and other instances where
// you simply need oRider to be switched to mounted mode as quickly as possible
// without preserving any variables or anything. However, you can store a
// resref for the horse to dismount by setting sResRef. If bJoust is set to
// TRUE then it will use the Joust Phenotypes.
// WARNING: This does not care whether someone meets the criteria to mount,
// or even if they are already mounted. It will simply set them to the
// proper appearance and mode. The ResRef is provided only in case someone
// uses HorseDismount() instead of using HorseInstantDismount() with this
// rider. It is recommended this function only be used in conjunction with
// HorseInstantDismount().
void HorseInstantMount(object oRider,int nTail,int bJoust=FALSE,string sResRef="");
// FILE: x3_inc_horse FUNCTION: HorseInstantDismount()
// This function is used to rapidly dismount oRider and does not produce a horse
// object. It's intended usage is with cutscenes or other instances where
// having oRider dismounted as quickly as possible are required. This will not
// produce horse/mount as it is primarily intende for cutscene work and not
// intended to replace the HorseDismount() function in other cases.
// WARNING: This does not protect Saddlebags so, it is recommended this only
// be used in conjunction with HorseInstantMount. If you need to protect
// saddlebag contents use HorseDismount().
void HorseInstantDismount(object oRider);
// FILE: x3_inc_horse FUNCTION: HorseGetMountTail()
// This function will return the tail that should be used with the specified
// horse.
int HorseGetMountTail(object oHorse);
// FILE: x3_inc_horse FUNCTION: HorseGetMountFailureMessage()
// This is a companion function to HorseGetCanBeMounted. If you need a text
// message that explains why the horse cannot be mounted.
string HorseGetMountFailureMessage(object oTarget,object oRider=OBJECT_INVALID);
// FILE: x3_inc_horse FUNCTION: HorseAddHorseMenu()
// This function will add horse menus to the respective PC. This is only
// needed for PCs that were not created new using patch 1.69.
void HorseAddHorseMenu(object oPC);
// FILE: x3_inc_horse FUNCTION: HorseGetPaladinMount()
// If this rider has a paladin mount then it will be returned as the object.
// If the rider is currently riding their paladin mount then it will return
// oRider as the object. If no paladin mount currently exists for oRider it
// will return OBJECT_INVALID.
object HorseGetPaladinMount(object oRider);
// FILE: x3_inc_horse FUNCTION: HorseGetIsAMount()
// This will return TRUE if oTarget is a mountable creature.
int HorseGetIsAMount(object oTarget);
// FILE: x3_inc_horse FUNCTION: HorseGetIsDisabled()
// This function detects if oCreature is in a disabled state. Can be used to
// detect disabling effects such as death, fear, confusion, sleep, paralysis,
// petrify, stun, entangle and knockdown (also applied in a non-hostile way,
// where oCreature doesn't enter a combat state)
int HorseGetIsDisabled(object oCreature=OBJECT_SELF);
// FILE: x3_inc_horse FUNCTION: HorseReloadFromDatabase()
// This function is provided for use with an OnClientEnter script when you
// are using a persistent world type environment and need the PC's mounted
// state reloaded. This will also make sure that henchmen for the PC are also
// restored as they were.
void HorseReloadFromDatabase(object oPC,string sDatabase);
// FILE: x3_inc_horse FUNCTION: HorseSaveToDatabase()
// This will save all of the current status for the PC and the PCs henchmen.
void HorseSaveToDatabase(object oPC,string sDatabase);
// FILE: x3_inc_horse FUNCTION: HorseStoreInventory()
// This function is used to store inventory of the horse for later recovery.
// See x3_inc_horse for complete details.
void HorseStoreInventory(object oCreature,object oRider=OBJECT_INVALID);
// FILE: x3_inc_horse FUNCTION: HorseRestoreInventory()
// This function is used to return stored inventory back onto the horse
// See x3_inc_horse for complete details. If bDrop is set to TRUE then it
// will drop the contents where the horse is rather than putting them on the
// horse. bDrop was primarily intended for cases where a mounted PC dies.
void HorseRestoreInventory(object oCreature,int bDrop=FALSE);
// FILE: x3_inc_horse FUNCTION: HorseChangeToDefault()
// This function will set oCreature to the default racial appearance and is
// useful for reversing any situations where a creature or PC is stuck in some
// variation of a mounted appearance. This will also clear ANY information
// stored on the creature relating to mounting. NOTE: If the target had a
// tail when not mounted this function will not restore that.
void HorseChangeToDefault(object oCreature);
// FILE: x3_inc_horse FUNCTION: HorseIfNotDefaultAppearanceChange()
// This funtion will check the appearance of oCreature, If it is not set to
// the default racial appearance (see racialtypes.2da) then it will call the
// HorseChangeToDefault() function.
void HorseIfNotDefaultAppearanceChange(object oCreature);
// FILE: x3_inc_horse FUNCTION: HorseGetMyHorse()
// When oRider is dismounted, this function returns the horse currently assigned
// to oRider (OBJECT_INVALID if there is none). In certain script hooks
// (pre-mount, post-mount and post-dismount), the function returns the horse
// that is currently being mounted / dismounted. It should not be used when the
// rider is mounted, or in a pre-dismount script hook.
object HorseGetMyHorse(object oRider);
// FILE: x3_inc_horse FUNCTION: HorseGetHasAHorse()
// This function will return TRUE if oRider has a mount or horse.
int HorseGetHasAHorse(object oRider);
// FILE: x3_inc_horse FUNCTION: HorseGetHorse()
// This function will return the nNth horse that is owned by oRider.
object HorseGetHorse(object oRider,int nN=1);
// FILE: x3_inc_horse FUNCTION: HorseRestoreHenchmenLocations()
// This function will check the module variable X3_RESTORE_HENCHMEN_LOCATIONS
// which when set to TRUE on the module will cause this function to jump any
// npc henchmen to the PC and will also make sure their henchmen also jump.
// This function is provided so, that there exists a function which will remain
// true to whether henchmen are mounted in no mount areas and whether mounts are
// prevented from entering no mount areas. This is not the perfect function
// but, it will address some of the issues with the engine not restoring the
// locations of henchmen belonging to henchmen on a module reload. This will
// actually NOT jump henchmen to the PC. It's primary purpose is to make sure
// that henchmen belonging to the henchman are jumped to the proper location.
// It is assumed that the henchman that is their master will already be at the
// proper location due to a reload or due to a respawn from some special script
// for the server.
void HorseRestoreHenchmenLocations(object oPC);
// FILE: x3_inc_horse FUNCTION: HorseHitchHorses()
// This function is designed to be used in conjunction with a transition script.
// oHitch is an object to hitch horses/mounts near. If it is OBJECT_INVALID
// then a specific destination to hitch will not be used. oClicker is who
// clicked the area transistion. lPreJump is a pre-transition location which
// is used to determine where the horses should stay near (actually walk a
// small distance away) if they do not transition.
void HorseHitchHorses(object oHitch,object oClicker,location lPreJump);
// FILE: x3_inc_horse FUNCTION: HorseForceJump()
// This is used to make sure oJumper makes it to withing fRange of oDestination.
// The nTimeOut is used to set a maximum number of attempts that will be made.
void HorseForceJump(object oJumper,object oDestination,float fRange=2.0,int nTimeOut=10);
// FILE: x3_inc_horse FUNCTION: HorseMoveAssociates()
// This function is intended to be used after a transition has been made and
// oMaster has moved onto the target destination. This function will cause
// any associates that also transitioned to move away from oMaster a small
// amount to make sure they do not block movement for oMaster. This can be
// particularly important when you consider horses have a bit more personal
// space requirements than the traditional associate.
void HorseMoveAssociates(object oMaster);
// FILE: x3_inc_horse FUNCTION: HorsePreloadAnimations()
// The mount and dismount animations are rather complex and by default the
// aurora engine does not always preload these animations. This function was
// created to temporarily set the tail and appearance of a PC to that of one of
// the horse models that will thus, force preloading of the animations. After
// that is done the animations should played properly. This is a good function
// to call in the OnEnter script of the module or similar location.
void HorsePreloadAnimations(object oPC);
////////////////////////////////////////////////////////////////////////////////
// SUPPORT FUNCTIONS
//-----------------------------
// These functions are not intended to be called by external scripts to this
// file and are simply used as support functions called by other functions
// from within this script.
////////////////////////////////////////////////////////////////////////////////
int HORSE_SupportGetMountedAppearance(object oRider)
{ // PURPOSE: Return which appearance the rider should use when mounted
int nRace=GetRacialType(oRider);
int nGender=GetGender(oRider);
if (GetLocalInt(oRider,"X3_CUSTOM_RACE_MOUNTED_APPEARANCE")>0) return GetLocalInt(oRider,"X3_CUSTOM_RACE_MOUNTED_APPEARANCE");
if (nGender==GENDER_FEMALE)
{ // female
if (nRace==RACIAL_TYPE_DWARF) return HORSE_RACE_MOUNTED_DWARFF;
else if (nRace==RACIAL_TYPE_ELF) return HORSE_RACE_MOUNTED_ELFF;
else if (nRace==RACIAL_TYPE_GNOME) return HORSE_RACE_MOUNTED_GNOMEF;
else if (nRace==RACIAL_TYPE_HALFLING) return HORSE_RACE_MOUNTED_HALFLINGF;
else if (nRace==RACIAL_TYPE_HALFELF) return HORSE_RACE_MOUNTED_HALFELFF;
else if (nRace==RACIAL_TYPE_HALFORC) return HORSE_RACE_MOUNTED_HALFORCF;
else if (nRace==RACIAL_TYPE_HUMAN) return HORSE_RACE_MOUNTED_HUMANF;
} // female
else
{ // male
if (nRace==RACIAL_TYPE_DWARF) return HORSE_RACE_MOUNTED_DWARFM;
else if (nRace==RACIAL_TYPE_ELF) return HORSE_RACE_MOUNTED_ELFM;
else if (nRace==RACIAL_TYPE_GNOME) return HORSE_RACE_MOUNTED_GNOMEM;
else if (nRace==RACIAL_TYPE_HALFLING) return HORSE_RACE_MOUNTED_HALFLINGM;
else if (nRace==RACIAL_TYPE_HALFELF) return HORSE_RACE_MOUNTED_HALFELFM;
else if (nRace==RACIAL_TYPE_HALFORC) return HORSE_RACE_MOUNTED_HALFORCM;
else if (nRace==RACIAL_TYPE_HUMAN) return HORSE_RACE_MOUNTED_HUMANM;
} // male
return GetAppearanceType(oRider);
} // HORSE_SupportGetMountedAppearance()
string HORSE_SupportRaceRestrictString(object oRider)
{ // PURPOSE: This will return the race restriction string to use for oRider
string sRet="X3_HORSE_RESTRICT_";
int nRace=GetRacialType(oRider);
if (GetObjectType(oRider)==OBJECT_TYPE_CREATURE)
{ // valid
if (nRace==RACIAL_TYPE_DWARF) sRet=sRet+"DWARF";
else if (nRace==RACIAL_TYPE_ELF) sRet=sRet+"ELF";
else if (nRace==RACIAL_TYPE_GNOME) sRet=sRet+"GNOME";
else if (nRace==RACIAL_TYPE_HALFELF) sRet=sRet+"HALFELF";
else if (nRace==RACIAL_TYPE_HALFLING) sRet=sRet+"HALFLING";
else if (nRace==RACIAL_TYPE_HALFORC) sRet=sRet+"HALFORC";
else if (nRace==RACIAL_TYPE_HUMAN) sRet=sRet+"HUMAN";
else {sRet=sRet+"CUSTOM"+IntToString(nRace);}
} // valid
return sRet;
} // HORSE_SupportRaceRestrictString(oRider)
string HORSE_SupportMountResRef(object oRider)
{ // PURPOSE: Return the resref of the mount oRider is on
if (GetObjectType(oRider)==OBJECT_TYPE_CREATURE)
{ // valid parameter
return GetSkinString(oRider,"sX3_HorseResRef");
} // valid parameter
return "";
} // HORSE_SupportMountResRef()
int HORSE_SupportMountAppearance(object oRider)
{ // PURPOSE: Return the appearance of the mount oRider is on
if (GetObjectType(oRider)==OBJECT_TYPE_CREATURE)
{ // valid parameter
return GetSkinInt(oRider,"nX3_HorseAppearance");
} // valid parameter
return -1;
} // HORSE_SupportMountAppearance()
int HORSE_SupportMountTail(object oRider)
{ // PURPOSE: Return the tail of the mount oRider is on
if (GetObjectType(oRider)==OBJECT_TYPE_CREATURE)
{ // valid parameter
return GetCreatureTailType(oRider);
} // valid parameter
return -1;
} // HORSE_SupportMountTail()
int HORSE_SupportMountFootstep(object oRider)
{ // PURPOSE: Return the footstep sound of the rider is supposed to be when not
// mounted
if (GetObjectType(oRider)==OBJECT_TYPE_CREATURE)
{ // valid parameter
return GetSkinInt(oRider,"nX3_StoredFootstep");
} // valid parameter
return -1;
} // HORSE_SupportMountFootstep()
string HORSE_SupportMountTag(object oRider)
{ // PURPOSE: Return the tag of the mount that rider is riding
if (GetObjectType(oRider)==OBJECT_TYPE_CREATURE)
{ // valid parameter
return GetSkinString(oRider,"sX3_HorseMountTag");
} // valid parameter
return "";
} // HORSE_SupportMountTag()
string HORSE_SupportMountScript(object oRider)
{ // PURPOSE: Return the post spawn script to run for the horse oRider is riding
if (GetObjectType(oRider)==OBJECT_TYPE_CREATURE)
{ // valid parameter
return GetLocalString(oRider,"sX3_HorseMountString");
} // valid parameter
return "";
} // HORSE_SupportMountScript()
int HORSE_SupportRiderAppearance(object oRider)
{ // PURPOSE: Return the appearance of oRider when not mounted
if (GetObjectType(oRider)==OBJECT_TYPE_CREATURE)
{ // valid parameter
return GetSkinInt(oRider,"nX3_HorseRiderAppearance");
} // valid parameter
return -1;
} // HORSE_SupportRiderAppearance()
int HORSE_SupportRiderPhenotype(object oRider)
{ // PURPOSE: Return the phenotype of the rider when not mounted
if (GetObjectType(oRider)==OBJECT_TYPE_CREATURE)
{ // valid parameter
if (GetLocalInt(oRider,"X3_CUSTOM_RACE_PHENOTYPE")) return GetLocalInt(oRider,"X3_CUSTOM_RACE_PHENOTYPE");
return GetSkinInt(oRider,"nX3_HorseRiderPhenotype");
} // valid parameter
return -1;
} // HORSE_SupportRiderPhenotype()
void HORSE_SupportCleanVariables(object oTarget)
{ // PURPOSE: Remove any mount related variables after dismount
DeleteLocalInt(oTarget,"X3_HORSE_NULL_APPEARANCE");
//DeleteSkinInt(oTarget,"X3_HORSE_APPEARANCE");
DeleteLocalInt(oTarget,"X3_HORSE_TAIL");
//DeleteSkinInt(oTarget,"nX3_HorseRiderAppearance");
//DeleteSkinInt(oTarget,"nX3_HorseRiderPhenotype");
//DeleteSkinInt(oTarget,"nX3_StoredFootstep");
DeleteSkinString(oTarget,"X3_HORSE_PREDISMOUNT_SCRIPT");
DeleteSkinString(oTarget,"X3_HORSE_POSTDISMOUNT_SCRIPT");
DeleteLocalString(oTarget,"X3_HORSE_PREMOUNT_SCRIPT");
DeleteLocalString(oTarget,"X3_HORSE_POSTMOUNT_SCRIPT");
DeleteSkinString(oTarget,"sX3_HorseResRef");
DeleteSkinString(oTarget,"sX3_HorseMountTag");
DeleteLocalString(oTarget,"sX3_HorseMountScript");
DeleteLocalInt(oTarget,"X3_NO_MOUNT_CUTSCENE");
DeleteLocalInt(oTarget,"X3_NO_MOUNT_ANIMATE");
DeleteLocalInt(oTarget,"nX3_RiderHP");
DeleteSkinInt(oTarget,"nX3_HorseHP");
DeleteLocalInt(oTarget,"nX3_HorsePortrait");
DeleteLocalInt(oTarget,"bX3_HAS_SADDLEBAGS");
DeleteLocalString(oTarget,"sDB_Inv");
DeleteSkinInt(oTarget,"bX3_IS_MOUNTED");
DeleteLocalInt(oTarget,"nX3_StoredMountState");
DeleteLocalInt(oTarget,"X3_ABORT_WHEN_STUCK");
//DeleteSkinInt(oTarget,"nX3_HorseRiderTail");
} // HORSE_SupportCleanVariables()
void HORSE_SupportMountCleanVariables(object oRider)
{ // PURPOSE: Clean variables off of oRider related to the mount
if (GetObjectType(oRider)==OBJECT_TYPE_CREATURE)
{ // valid parameter
HORSE_SupportCleanVariables(oRider);
} // valid parameter
} // HORSE_SupportMountCleanVariables()
location HORSE_SupportGetMountLocation(object oCreature,object oRider,float fDirection=90.0)
{ // PURPOSE: To Locate the location to place the mount
object oArea=GetArea(oCreature);
vector vPos=GetPosition(oCreature);
float fFace=GetFacing(oCreature);
location lLoc;
float fDist=0.8*StringToFloat(Get2DAString("appearance","WING_TAIL_SCALE",HORSE_SupportGetMountedAppearance(oRider)));
lLoc=Location(oArea,vPos+fDist*AngleToVector(fFace-fDirection),fFace);
return lLoc;
} // HORSE_SupportGetMountLocation()
int HORSE_SupportNullAppearance(object oHorse,object oRider)
{ // PURPOSE: This will return which appearance should be used to handle animation
int nRace;
if (GetLocalInt(oHorse,"X3_HORSE_NULL_APPEARANCE")>0) return GetLocalInt(oHorse,"X3_HORSE_NULL_APPEARANCE");
nRace=GetRacialType(oRider);
if (nRace==RACIAL_TYPE_DWARF) return HORSE_NULL_RACE_DWARF;
else if (nRace==RACIAL_TYPE_ELF) return HORSE_NULL_RACE_ELF;
else if (nRace==RACIAL_TYPE_GNOME) return HORSE_NULL_RACE_GNOME;
else if (nRace==RACIAL_TYPE_HALFELF) return HORSE_NULL_RACE_HALFELF;
else if (nRace==RACIAL_TYPE_HALFLING) return HORSE_NULL_RACE_HALFLING;
else if (nRace==RACIAL_TYPE_HALFORC) return HORSE_NULL_RACE_HALFORC;
return HORSE_NULL_RACE_HUMAN;
} // HORSE_SupportNullAppearance()
int Horse_SupportRaceAppearance(int nAppearance)
{ // PURPOSE: This will return TRUE if the appearance passed as a parameter
// is an appearance used as part of the mounting process.
if (nAppearance==HORSE_RACE_MOUNTED_DWARFF) return TRUE;
else if (nAppearance==HORSE_RACE_MOUNTED_DWARFM) return TRUE;
else if (nAppearance==HORSE_RACE_MOUNTED_ELFF) return TRUE;
else if (nAppearance==HORSE_RACE_MOUNTED_ELFM) return TRUE;
else if (nAppearance==HORSE_RACE_MOUNTED_GNOMEF) return TRUE;
else if (nAppearance==HORSE_RACE_MOUNTED_GNOMEM) return TRUE;
else if (nAppearance==HORSE_RACE_MOUNTED_HALFELFF) return TRUE;
else if (nAppearance==HORSE_RACE_MOUNTED_HALFELFM) return TRUE;
else if (nAppearance==HORSE_RACE_MOUNTED_HALFLINGF) return TRUE;
else if (nAppearance==HORSE_RACE_MOUNTED_HALFLINGM) return TRUE;
else if (nAppearance==HORSE_RACE_MOUNTED_HALFORCF) return TRUE;
else if (nAppearance==HORSE_RACE_MOUNTED_HALFORCM) return TRUE;
else if (nAppearance==HORSE_RACE_MOUNTED_HUMANF) return TRUE;
else if (nAppearance==HORSE_RACE_MOUNTED_HUMANM) return TRUE;
return FALSE;
} // Horse_SupportRaceAppearance()
void HORSE_SupportIncreaseSpeed(object oRider,object oHorse)
{ // PURPOSE: Change the movement speed based on that of the horse (wont work unless EffectMovementSpeedIncrease() bug is fixed)
if (GetLocalInt(GetModule(),"X3_HORSE_DISABLE_SPEED")) return;
int nSpeed=GetLocalInt(oHorse,"X3_HORSE_MOUNT_SPEED");
effect eMovement;
float fSpeed;
int nMonkEpic;
int nMonk=GetLevelByClass(CLASS_TYPE_MONK,oRider);
if (nSpeed==0) nSpeed=HORSE_DEFAULT_SPEED_INCREASE;
if (nSpeed>50) nSpeed=50; // upper limit
if (nSpeed<-100) nSpeed=-100;
if (nMonk>2)
{ // find out monk adjustment
nSpeed-=10*nMonk/3;
//nSpeed-=GetBuggedSpeedAdjustment(oRider)+10*nMonk/3;
} // find out monk adjustment
if (GetLevelByClass(CLASS_TYPE_BARBARIAN,oRider)>0)
{ // barbarian class adjustment
nSpeed-=10;
//nSpeed-=GetBuggedSpeedAdjustment(oRider)+10;
} // barbarian class adjustment
if (nSpeed<-150) nSpeed=-150;
SetLocalInt(oRider,"nX3_SpeedIncrease",nSpeed);
eMovement=SupernaturalEffect(EffectMovementSpeedIncrease(nSpeed)); // EffectMovementSpeedIncrease() with negative numbers is the same as EffectMovementSpeedDecrease() with positive numbers
ApplyEffectToObject(DURATION_TYPE_PERMANENT,eMovement,oRider);
} // HORSE_SupportIncreaseSpeed()
void HORSE_SupportOriginalSpeed(object oRider)
{ // PURPOSE: Remove speed increases based upon mounted
if (GetLocalInt(GetModule(),"X3_HORSE_DISABLE_SPEED")) return;
effect eSearch=GetFirstEffect(oRider);
while(GetIsEffectValid(eSearch))
{ // cycle through effects
if((GetEffectType(eSearch) == EFFECT_TYPE_MOVEMENT_SPEED_INCREASE) &&
(GetEffectDurationType(eSearch) == DURATION_TYPE_PERMANENT) &&
(GetEffectSubType(eSearch) == SUBTYPE_SUPERNATURAL))
{ // check to see if matches conditions
RemoveEffect(oRider,eSearch);
} // check to see if matches conditions
eSearch=GetNextEffect(oRider);
} // cycle through effects
} // HORSE_SupportOriginalSpeed()
void HORSE_SupportAdjustMountedArcheryPenalty(object oRider)
{ // PURPOSE: Check for feats and adjust penalties to archery while mounted
int nMountedArcheryPenalty=GetLocalInt(oRider,"nX3_MountedArchery");
int nNewPenalty;
effect eEffect;
int bMatched;
//if (HorseGetIsMounted(oRider)&&GetWeaponRanged(GetItemInSlot(INVENTORY_SLOT_RIGHTHAND,oRider)))
if (HorseGetIsMounted(oRider)&&GetLocalInt(oRider,"bX3_M_ARCHERY"))
{ // Rider is mounted and has a ranged weapon - apply penalty
nNewPenalty=4;
if (GetHasFeat(FEAT_MOUNTED_ARCHERY,oRider)) nNewPenalty=2;
if (nNewPenalty!=nMountedArcheryPenalty&&nMountedArcheryPenalty>0)
{ // remove existing penalties before applying the new one
eEffect=GetFirstEffect(oRider);
bMatched=FALSE;
while(GetIsEffectValid(eEffect)&&!bMatched)
{ // traverse effects
if ((GetEffectType(eEffect)==EFFECT_TYPE_ATTACK_DECREASE)&&
(GetEffectSubType(eEffect)==SUBTYPE_SUPERNATURAL)&&
(GetEffectDurationType(eEffect)==DURATION_TYPE_PERMANENT))
{ // match
RemoveEffect(oRider,eEffect);
bMatched=TRUE;
} // match
eEffect=GetNextEffect(oRider);
} // traverse effects
} // remove existing penalties before applying the new one
ApplyEffectToObject(DURATION_TYPE_PERMANENT,SupernaturalEffect(EffectAttackDecrease(nNewPenalty)),oRider);
SetLocalInt(oRider,"nX3_MountedArchery",nNewPenalty);
} // Rider is mounted and has a ranged weapon - apply penalty
else if (nMountedArcheryPenalty)
{ // no penalty needed - remove it
eEffect=GetFirstEffect(oRider);
bMatched=FALSE;
while(GetIsEffectValid(eEffect)&&!bMatched)
{ // traverse effects
if ((GetEffectType(eEffect)==EFFECT_TYPE_ATTACK_DECREASE)&&
(GetEffectSubType(eEffect)==SUBTYPE_SUPERNATURAL)&&
(GetEffectDurationType(eEffect)==DURATION_TYPE_PERMANENT))
{ // match
RemoveEffect(oRider,eEffect);
bMatched=TRUE;
} // match
eEffect=GetNextEffect(oRider);
} // traverse effects
DeleteLocalInt(oRider,"nX3_MountedArchery");
} // no penalty needed - remove it
} // HORSE_SupportAdjustMountedArcheryPenalty()
void HORSE_SupportApplyMountedSkillDecreases(object oRider)
{ // PURPOSE: Applies decreases to skills while mounted
if (GetLocalInt(GetModule(),"X3_HORSE_DISABLE_SKILL")) return;
object oMod=GetModule();
effect eDisarmTrap = SupernaturalEffect(EffectSkillDecrease(SKILL_DISABLE_TRAP , 50));
effect eOpenLock = SupernaturalEffect(EffectSkillDecrease(SKILL_OPEN_LOCK , 50));
effect eHide = SupernaturalEffect(EffectSkillDecrease(SKILL_HIDE , 50));
effect eMove = SupernaturalEffect(EffectSkillDecrease(SKILL_MOVE_SILENTLY, 50));
effect ePickPocket = SupernaturalEffect(EffectSkillDecrease(SKILL_PICK_POCKET , 50));
effect eSetTrap = SupernaturalEffect(EffectSkillDecrease(SKILL_SET_TRAP , 50));
effect eTumble = SupernaturalEffect(EffectSkillDecrease(SKILL_TUMBLE , 50));
AssignCommand(oMod,ApplyEffectToObject(DURATION_TYPE_PERMANENT,eDisarmTrap,oRider));
AssignCommand(oMod,ApplyEffectToObject(DURATION_TYPE_PERMANENT,eOpenLock,oRider));
AssignCommand(oMod,ApplyEffectToObject(DURATION_TYPE_PERMANENT,eHide,oRider));
AssignCommand(oMod,ApplyEffectToObject(DURATION_TYPE_PERMANENT,eMove,oRider));
AssignCommand(oMod,ApplyEffectToObject(DURATION_TYPE_PERMANENT,ePickPocket,oRider));
AssignCommand(oMod,ApplyEffectToObject(DURATION_TYPE_PERMANENT,eSetTrap,oRider));
AssignCommand(oMod,ApplyEffectToObject(DURATION_TYPE_PERMANENT,eTumble,oRider));
} // HORSE_SupportApplyMountedSkillDecreases()
void HORSE_SupportRemoveMountedSkillDecreases(object oRider)
{ // PURPOSE: To remove any negative effects caused by being mounted
if (GetLocalInt(GetModule(),"X3_HORSE_DISABLE_SKILL")) return;
effect ePenalty=GetFirstEffect(oRider);
while (GetIsEffectValid(ePenalty))
{ // traverse effects
if ((GetEffectType(ePenalty)==EFFECT_TYPE_SKILL_DECREASE)&&
(GetEffectSubType(ePenalty)==SUBTYPE_SUPERNATURAL)&&
(GetEffectDurationType(ePenalty)==DURATION_TYPE_PERMANENT))
{ // match
RemoveEffect(oRider, ePenalty);
} // match
ePenalty=GetNextEffect(oRider);
} // traverse effects
} // HORSE_SupportRemoveMountedSkillDecreases()
void HORSE_SupportResetUnmountedAppearance(object oRider)
{ // PURPOSE: Reset oRider to an unmounted appearance
int nApp=GetSkinInt(oRider,"nX3_HorseRiderAppearance");
int nPheno=GetSkinInt(oRider,"nX3_HorseRiderPhenotype");
int nFoot=GetSkinInt(oRider,"nX3_StoredFootstep");
int nTail=GetSkinInt(oRider,"nX3_HorseRiderTail");
SetCreatureTailType(nTail,oRider);
SetPhenoType(nPheno,oRider);
SetCreatureAppearanceType(oRider,nApp);
SetFootstepType(nFoot,oRider);
} // HORSE_SupportResetUnmountedAppearance()
int HORSE_SupportAbsoluteMinute()
{ // PURPOSE: Returns the current absolute time expressed in minutes
int nTime=GetTimeMinute()+GetTimeHour()*60+GetCalendarDay()*60*24;
int nYear=GetCalendarYear();
if (nYear>20) nYear=nYear%20;
nTime=nTime+GetCalendarMonth()*60*24*30+nYear*60*24*30*12;
return nTime;
} // HORSE_SupportAbsoluteMinute()
void HORSE_SupportMonitorPaladinUnsummon(object oPaladin)
{ // PURPOSE: Monitor whether to unsummon the paladin mount
// see hook scripts x3_s2_palmount and x3_s2_paldmount
object oMount=GetLocalObject(oPaladin,"oX3_PALADIN_MOUNT");
int nTime;
if (GetIsObjectValid(oPaladin)&&oMount==oPaladin)
{ // valid
if (GetIsObjectValid(oMount))
{ // paladin mount exists
SetLocalObject(oPaladin,"oX3_PALADIN_MOUNT",oMount);
nTime=HORSE_SupportAbsoluteMinute();
if (nTime>=GetLocalInt(oPaladin,"nX3_PALADIN_UNSUMMON")&&!GetCutsceneMode(oPaladin))
{ // unsummon as long as not in a cutscene
AssignCommand(oPaladin,HorseUnsummonPaladinMount());
} // unsummon as long as not in a cutscene
DelayCommand(10.0,HORSE_SupportMonitorPaladinUnsummon(oPaladin));
} // paladin mount exists
} // valid
} // HORSE_SupportMonitorPaladinUnsummon()
void HORSE_SupportApplyACBonus(object oRider,object oHorse)
{ // PURPOSE: Apply AC bonus
if (!GetLocalInt(GetModule(),"X3_HORSE_ENABLE_ACBOOST")) return;
effect eEffect;
int nRiderAC=GetAC(oRider);
int nHorseAC=GetAC(oHorse);
int nDiff=nHorseAC-nRiderAC;
if (nDiff<1) return;
eEffect=SupernaturalEffect(EffectACIncrease(nDiff,AC_NATURAL_BONUS));
AssignCommand(GetModule(),ApplyEffectToObject(DURATION_TYPE_PERMANENT,eEffect,oRider));
} // HORSE_SupportApplyACBonus()
void HORSE_SupportRemoveACBonus(object oRider)
{ // PURPOSE: Remove AC Bonus
if (!GetLocalInt(GetModule(),"X3_HORSE_ENABLE_ACBOOST")) return;
effect eEffect;
eEffect=GetFirstEffect(oRider);
while(GetIsEffectValid(eEffect))
{ // traverse effects
if ((GetEffectType(eEffect)==EFFECT_TYPE_AC_INCREASE)&&
(GetEffectSubType(eEffect)==SUBTYPE_SUPERNATURAL)&&
(GetEffectDurationType(eEffect)==DURATION_TYPE_PERMANENT))
{ // remove
RemoveEffect(oRider,eEffect);
return;
} // remove
eEffect=GetNextEffect(oRider);
} // traverse effects
} // HORSE_SupportRemoveACBonus()
void HORSE_SupportApplyHPBonus(object oRider,object oHorse)
{ // PURPOSE: Apply HP bonus
if (!GetLocalInt(GetModule(),"X3_HORSE_ENABLE_HPBOOST")) return;
effect eEffect;
int nHP=GetCurrentHitPoints(oHorse);
nHP=nHP/2;
if (nHP<1) nHP=1;
eEffect=SupernaturalEffect(EffectTemporaryHitpoints(nHP));
AssignCommand(GetModule(),ApplyEffectToObject(DURATION_TYPE_PERMANENT,eEffect,oRider));
} // HORSE_SupportApplyHPBonus()
void HORSE_SupportRemoveHPBonus(object oRider)
{ // PURPOSE: Remove HP Bonus
if (!GetLocalInt(GetModule(),"X3_HORSE_ENABLE_HPBOOST")) return;
effect eEffect;
eEffect=GetFirstEffect(oRider);
while(GetIsEffectValid(eEffect))
{ // traverse effects
if ((GetEffectType(eEffect)==EFFECT_TYPE_TEMPORARY_HITPOINTS)&&
(GetEffectSubType(eEffect)==SUBTYPE_SUPERNATURAL)&&
(GetEffectDurationType(eEffect)==DURATION_TYPE_PERMANENT))
{ // remove
RemoveEffect(oRider,eEffect);
return;
} // remove
eEffect=GetNextEffect(oRider);
} // traverse effects
} // HORSE_SupportRemoveHPBonus()
void HORSE_SupportHandleDamage(object oRider,object oHorse)
{ // PURPOSE: Handle resetting damage to the horse back to its original setting
// and also support damage sharing with the horse if that feature is enabled.
int nRiderCHP=GetCurrentHitPoints(oRider);
int nHorseCHP=GetCurrentHitPoints(oHorse);
int nRiderSHP=GetLocalInt(oRider,"nX3_RiderHP");
int nHorseSHP=GetSkinInt(oRider,"nX3_HorseHP");
int nDamage;
effect eDamage;
effect eHeal;
effect ePCDamage;
int nAmount;
if (!GetIsObjectValid(oRider)||!GetIsObjectValid(oHorse)) return;
if (nRiderCHP<nRiderSHP) nDamage=nRiderSHP-nRiderCHP;
if (nHorseCHP>nHorseSHP&&nHorseSHP!=0)
{ // restore horse damage
eDamage=EffectDamage(nHorseCHP-nHorseSHP);
nHorseCHP=nHorseSHP;
AssignCommand(GetModule(),ApplyEffectToObject(DURATION_TYPE_INSTANT,eDamage,oHorse));
} // restore horse damage
if (nDamage>0)
{ // damage was received while mounted
if (GetLocalInt(GetModule(),"X3_ENABLE_MOUNT_DAMAGE"))
{ // share damage with the mount
nAmount=nDamage/2;
if (nAmount>(nHorseCHP-1)) nAmount=nHorseCHP-1;
eHeal=EffectHeal(nAmount);
eDamage=EffectDamage(nAmount);
ePCDamage=EffectDamage(nDamage-nAmount);
if (GetCurrentHitPoints(oRider)>(nRiderSHP-(nDamage-nAmount))) AssignCommand(GetModule(),ApplyEffectToObject(DURATION_TYPE_INSTANT,ePCDamage,oRider));
else if (GetCurrentHitPoints(oRider)<(nRiderSHP-(nDamage-nAmount))) AssignCommand(GetModule(),ApplyEffectToObject(DURATION_TYPE_INSTANT,eHeal,oRider));
AssignCommand(GetModule(),ApplyEffectToObject(DURATION_TYPE_INSTANT,eDamage,oHorse));
} // share damage with the mount
} // damage was received while mounted
} // HORSE_SupportHandleDamage()
void KillTheHorse(object oHorse,int bStart=TRUE)
{ // PURPOSE: To Kill the horse
effect eDeath=EffectDeath();
object oItem=GetItemInSlot(INVENTORY_SLOT_CARMOUR,oHorse);
//SendMessageToPC(GetFirstPC(),"KillTheHorse("+GetName(oHorse)+","+IntToString(bStart)+")");
if (GetBaseItemType(oItem)==BASE_ITEM_CREATUREITEM&&bStart)
{ // destroy
//SendMessageToPC(GetFirstPC(),"Destroy CARMOUR '"+GetName(oHorse)+"'");
SetPlotFlag(oItem,FALSE);
SetDroppableFlag(oItem,FALSE);
DestroyObject(oItem);
} // destroy
if (bStart) oItem=GetFirstItemInInventory(oHorse);
else { oItem=GetNextItemInInventory(oHorse); }
if (GetIsObjectValid(oItem))
{ // valid
//SendMessageToPC(GetFirstPC(),"Item '"+GetName(oItem)+"'");
if (GetBaseItemType(oItem)==BASE_ITEM_CREATUREITEM)
{ // destroy
//SendMessageToPC(GetFirstPC()," Destroy");
SetPlotFlag(oItem,FALSE);
SetDroppableFlag(oItem,FALSE);
DestroyObject(oItem,1.0);
} // destroy
DelayCommand(0.03,KillTheHorse(oHorse,FALSE));
} // valid
else
{ // kill
//SendMessageToPC(GetFirstPC(),"Kill '"+GetName(oHorse)+"'");
DelayCommand(1.0,ApplyEffectToObject(DURATION_TYPE_INSTANT,eDeath,oHorse));
} // kill
} // KillTheHorse()
void HORSE_SupportTransferInventory(object oFrom,object oTo,location lLoc,int bDestroyFrom=FALSE,int bFirst=TRUE)
{ // PURPOSE: Delay Command transfer to handle moving inventory
object oItem;
object oCopy;
if (bFirst) oItem=GetFirstItemInInventory(oFrom);
else { oItem=GetNextItemInInventory(oFrom); }
if (GetIsObjectValid(oItem))
{ // item found
if (GetIsObjectValid(oTo))
{ // to object
if (GetBaseItemType(oItem)!=BASE_ITEM_CREATUREITEM)
{ // no pchide transfers
oCopy=CopyItem(oItem,oTo,TRUE);
} // no pchide transfers
} // to object
else
{ // to location
if (GetBaseItemType(oItem)!=BASE_ITEM_CREATUREITEM)
{ // no pchide transfers
oCopy=CopyObject(oItem,lLoc);
if (GetItemStackSize(oCopy)!=GetItemStackSize(oItem)) SetItemStackSize(oCopy,GetItemStackSize(oItem));
if (GetItemCharges(oCopy)!=GetItemCharges(oItem)) SetItemCharges(oCopy,GetItemCharges(oItem));
} // no pchide transfers
} // to location
DestroyObject(oItem,2.0);
DelayCommand(0.02,HORSE_SupportTransferInventory(oFrom,oTo,lLoc,bDestroyFrom,FALSE));
} // item found
else
{ // transfer gold
if (GetIsObjectValid(oTo))
{ // to object
if (GetGold(oFrom)>0) oCopy=CreateItemOnObject("nw_it_gold001",oTo,GetGold(oFrom));
} // to object
else
{ // to location
if (GetGold(oFrom)>0) oCopy=CreateObject(OBJECT_TYPE_ITEM,"nw_it_gold001",lLoc,GetGold(oFrom));
} // to location
if (bDestroyFrom)
{ // destroy
DestroyObject(oFrom);
} // destroy
if (GetLocalInt(oTo,"bDie"))
{ // death effect
oItem=GetItemInSlot(INVENTORY_SLOT_CARMOUR,oTo);
if (GetBaseItemType(oItem)==BASE_ITEM_CREATUREITEM)
{ // destroy
SetDroppableFlag(oItem,FALSE);
SetPlotFlag(oItem,FALSE);
DestroyObject(oItem);
} // destroy
DelayCommand(0.1,KillTheHorse(oTo));
} // death effect
} // transfer gold
} // HORSE_SupportTransferInventory()
int HORSE_SupportCountHenchmen(object oPC)
{ // PURPOSE: Return the number of henchmen
int nC=0;
object oHench=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oPC,nC+1);
while(GetIsObjectValid(oHench))
{ // count
nC++;
oHench=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oPC,nC+1);
} // count
return nC;
} // HORSE_SupportCountHenchmen()
void HORSE_SupportTransferPreservedValues(object oRider,object oHorse)
{ // PURPOSE: This will transfer preserved values for the mounted horse to
// oHorse from oRider and then will remove them from oRider
string sS;
float fX3_MOUNT_MULTIPLE=GetLocalFloat(GetArea(oRider),"fX3_MOUNT_MULTIPLE");
float fX3_DISMOUNT_MULTIPLE=GetLocalFloat(GetArea(oRider),"fX3_DISMOUNT_MULTIPLE");
if (GetLocalFloat(oRider,"fX3_MOUNT_MULTIPLE")>fX3_MOUNT_MULTIPLE) fX3_MOUNT_MULTIPLE=GetLocalFloat(oRider,"fX3_MOUNT_MULTIPLE");
if (fX3_MOUNT_MULTIPLE<=0.0) fX3_MOUNT_MULTIPLE=1.0;
if (GetLocalFloat(oRider,"fX3_DISMOUNT_MULTIPLE")>0.0) fX3_DISMOUNT_MULTIPLE=GetLocalFloat(oRider,"fX3_DISMOUNT_MULTIPLE");
if (fX3_DISMOUNT_MULTIPLE>0.0) fX3_MOUNT_MULTIPLE=fX3_DISMOUNT_MULTIPLE; // use dismount multiple instead of mount multiple
if (GetIsObjectValid(oHorse))
{ // transfer values to horse
} // transfer values to horse
sS=GetSkinString(oRider,"X3_HORSE_POSTDISMOUNT_SCRIPT");
if (GetStringLength(sS)>0)
{ // run post dismount script
SetLocalObject(oRider,"oX3_TempHorse",oHorse);
ExecuteScript(sS,oRider);
DeleteLocalObject(oRider,"oX3_TempHorse");
} // run post dismount script
// delete variables
if (GetLocalInt(oRider,"bX3_HORSE_MODIFIERS")&&GetStringLength(sS)<1)
{ // remove horse modifiers
DeleteLocalInt(oRider,"bX3_HORSE_MODIFIERS");
DelayCommand(0.10*fX3_MOUNT_MULTIPLE,HORSE_SupportOriginalSpeed(oRider));
DelayCommand(0.30*fX3_MOUNT_MULTIPLE,HORSE_SupportAdjustMountedArcheryPenalty(oRider));
DelayCommand(0.15*fX3_MOUNT_MULTIPLE,HORSE_SupportRemoveMountedSkillDecreases(oRider));
DelayCommand(0.25*fX3_MOUNT_MULTIPLE,HORSE_SupportRemoveACBonus(oRider));
DelayCommand(0.25*fX3_MOUNT_MULTIPLE,HORSE_SupportRemoveHPBonus(oRider));
DelayCommand(0.20*fX3_MOUNT_MULTIPLE,HORSE_SupportHandleDamage(oRider,oHorse));
} // remove horse modifiers
//HORSE_SupportMountCleanVariables(oRider);
} // HORSE_SupportTransferPreservedValues()
void HORSE_SupportDismountWrapper(int bAnimate,int bSetOwner)
{ // Wrap
object oHorse=HorseDismount(bAnimate,bSetOwner);
object oNPC=OBJECT_SELF;
if (bSetOwner) SetLocalObject(oNPC,"oAssignedHorse",oHorse);
} // HORSE_SupportDismountWrapper()
void HORSE_SupportRestoreFromPreload(object oPC,int nApp,int nTail)
{ // PURPOSE: Restore to previous appearance before preload
SetCreatureAppearanceType(oPC,nApp);
SetCreatureTailType(nTail,oPC);
} // HORSE_SupportRestoreFromPreload()
void HORSE_SupportSetMountingSentinel(object oRider,object oHorse,float fTimeout=6.0, int nCount=0)
{ // PURPOSE: This will set a mounting process that will make sure the PC or NPC
// is returned to commandable. There may be a slight delay but, this will
// insure recovery from aborted mounts that could result in PC being stuck as
// not commandable.
if (!HorseGetIsMounted(oRider)&&!IsInConversation(oRider)&&!GetIsInCombat(oRider)&&!HorseGetIsDisabled(oRider))
{ // keep waiting
float fFreq=1.0; // frequency in seconds
if (nCount<FloatToInt(fTimeout/fFreq))
{ // keep trying
DelayCommand(fFreq,HORSE_SupportSetMountingSentinel(oRider,oHorse,fTimeout,nCount+1));
} // keep trying
else if (!GetCutsceneMode(oRider))
{ // make commandable
if (!GetCommandable(oRider))
{ // commandable
SetCommandable(TRUE,oRider);
SetCommandable(TRUE,oHorse);
//HORSE_SupportCleanVariables(oRider);
DeleteLocalInt(oRider,"X3_DOING_HORSE_ACTION");
DeleteLocalInt(oHorse,"X3_DOING_HORSE_ACTION");
DeleteLocalObject(oHorse,"oX3_TempRider");
DeleteLocalInt(oRider,"nX3_MovingMount");
DeleteLocalFloat(oRider,"fLastHorseDist");
DeleteLocalInt(oRider,"bPathIsBlocked");
} // commandable
} // make commandable
} // keep waiting
else if (!GetCutsceneMode(oRider))
{ // make sure commandable if not in a cutscene
if (!GetCommandable(oRider))
{ // commandable
SetCommandable(TRUE,oRider);
SetCommandable(TRUE,oHorse);
//HORSE_SupportCleanVariables(oRider);
DeleteLocalInt(oRider,"X3_DOING_HORSE_ACTION");
DeleteLocalInt(oHorse,"X3_DOING_HORSE_ACTION");
DeleteLocalObject(oHorse,"oX3_TempRider");
DeleteLocalInt(oRider,"nX3_MovingMount");
DeleteLocalFloat(oRider,"fLastHorseDist");
DeleteLocalInt(oRider,"bPathIsBlocked");
} // commandable
} // make sure commandable if not in a cutscene
} // HORSE_SupportSetMountingSentinel()
void HORSE_Support_AssignRemainingMount(object oOwner)
{ // PURPOSE: Make a link between the oOwner and one of the horses he may own
int nN=1;
object oAssociate=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oOwner,nN);
while(GetIsObjectValid(oAssociate))
{ // traverse associtates
if (HorseGetIsAMount(oAssociate)&&HorseGetCanBeMounted(oAssociate,oOwner,TRUE)&&oOwner==HorseGetOwner(oAssociate))
{ // pick first unmounted owned mount
SetLocalObject(oOwner,"oAssignedHorse",oAssociate);
return;
} // pick first unmounted owned mount
nN++;
oAssociate=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oOwner,nN);
} // traverse associtates
} // HORSE_Support_AssignRemainingMount()
////////////////////////////////////////////////////////////////////////////////
//////////////////////// DATABASE SUPPORT FUNCTIONS ////////////////////////
////////////////////////////////////////////////////////////////////////////////
/*
These functions were provided so, Persistent World and other multiplayer
gamers would have some functions to look at that they can modify to make
work with their own persistent world. People do these in so many ways that
while these are functional, they used the default Bioware database commands
and as a result are slow. This was known when they were designed but, it
was determined that having some sample functions that could be modified
might be beneficial to the community.
*/
void HORSE_SupportDeleteFromDatabase(object oPC,string sDatabase,int nN=1)
{ // PURPOSE: Delete Henchmen information from the database
string sN=IntToString(nN);
DeleteCampaignVariable(sDatabase,"sX3_RR_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"sX3_TAG_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"sX3_SCR_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_HAP_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_HTL_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_HNAP_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_HAP_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_APP_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_PHEN_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_FOOT_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_NOAN_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_CAPP_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_CPHE_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_CTAIL_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"sX3_SPRM_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"sX3_SPOM_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"sX3_SPRDM_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"sX3_SPODM_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"bX3_MOUNTED_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"sX3_HTAG_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"sX3_HRR_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_HPOR_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_HHP_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_RHP_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"bX3_ISM_"+sN,oPC);
DeleteCampaignVariable(sDatabase,"nX3_TAL_"+sN,oPC);
if (GetLocalInt(GetModule(),"X3_HORSE_ENABLE_SADDLEBAGS"))
{ // saddlebags enabled
DeleteCampaignVariable(sDatabase,"bX3_HSB_"+sN,oPC);
} // saddlebags enabled
} // HORSE_SupportDeleteFromDatabase()
void HORSE_SupportSaveToDatabase(object oPC,object oHenchman,string sDatabase,int nN=1)
{ // PURPOSE: Save henchman to database
string sN=IntToString(nN);
SetCampaignString(sDatabase,"sX3_RR_",GetSkinString(oHenchman,"sX3_HorseResRef"),oPC);
SetCampaignString(sDatabase,"sX3_TAG_",GetSkinString(oHenchman,"sX3_HorseMountTag"),oPC);
SetCampaignString(sDatabase,"sX3_SCR_",GetLocalString(oHenchman,"sX3_HorseMountScript"),oPC);
//SetCampaignInt(sDatabase,"nX3_HAP_",GetSkinInt(oHenchman,"X3_HORSE_APPEARANCE"),oPC);
SetCampaignInt(sDatabase,"nX3_HTL_",GetLocalInt(oHenchman,"X3_HORSE_TAIL"),oPC);
SetCampaignInt(sDatabase,"nX3_HNAP_",GetLocalInt(oHenchman,"X3_HORSE_NULL_APPEARANCE"),oPC);
SetCampaignInt(sDatabase,"nX3_APP_",GetSkinInt(oHenchman,"nX3_HorseRiderAppearance"),oPC);
SetCampaignInt(sDatabase,"nX3_PHEN_",GetSkinInt(oHenchman,"nX3_HorseRiderPhenotype"),oPC);
SetCampaignInt(sDatabase,"nX3_FOOT_",GetSkinInt(oHenchman,"nX3_StoredFootstep"),oPC);
SetCampaignInt(sDatabase,"nX3_NOAN_",GetLocalInt(oHenchman,"X3_NO_MOUNT_ANIMATE"),oPC);
SetCampaignInt(sDatabase,"nX3_CAPP_",GetAppearanceType(oHenchman),oPC);
SetCampaignInt(sDatabase,"nX3_CPHE_",GetPhenoType(oHenchman),oPC);
SetCampaignInt(sDatabase,"nX3_CTAIL_",GetCreatureTailType(oHenchman),oPC);
SetCampaignString(sDatabase,"sX3_SPRM_",GetLocalString(oHenchman,"X3_HORSE_PREMOUNT_SCRIPT"),oPC);
SetCampaignString(sDatabase,"sX3_SPOM_",GetLocalString(oHenchman,"X3_HORSE_POSTMOUNT_SCRIPT"),oPC);
SetCampaignString(sDatabase,"sX3_SPRDM_",GetSkinString(oHenchman,"X3_HORSE_PREDISMOUNT_SCRIPT"),oPC);
SetCampaignString(sDatabase,"sX3_SPODM_",GetSkinString(oHenchman,"X3_HORSE_POSTDISMOUNT_SCRIPT"),oPC);
SetCampaignInt(sDatabase,"bX3_MOUNTED_",HorseGetIsMounted(oHenchman),oPC);
SetCampaignString(sDatabase,"sX3_HTAG_"+sN,GetTag(oHenchman),oPC);
SetCampaignString(sDatabase,"sX3_HRR_"+sN,GetResRef(oHenchman),oPC);
SetCampaignInt(sDatabase,"nX3_HPOR_"+sN,GetLocalInt(oHenchman,"nX3_HorsePortrait"),oPC);
SetCampaignInt(sDatabase,"nX3_HHP_"+sN,GetSkinInt(oHenchman,"nX3_HorseHP"),oPC);
SetCampaignInt(sDatabase,"nX3_RHP_"+sN,GetLocalInt(oHenchman,"nX3_RiderHP"),oPC);
SetCampaignInt(sDatabase,"bX3_ISM_"+sN,GetSkinInt(oHenchman,"bX3_IS_MOUNTED"),oPC);
SetCampaignInt(sDatabase,"nX3_TAL_"+sN,GetSkinInt(oHenchman,"nX3_HorseRiderTail"),oPC);
if (GetLocalInt(GetModule(),"X3_HORSE_ENABLE_SADDLEBAGS"))
{ // saddlebags enabled
SetCampaignInt(sDatabase,"bX3_HSB_"+sN,GetLocalInt(oHenchman,"bX3_HAS_SADDLEBAGS"),oPC);
} // saddlebags enabled
} // HORSE_SupportSaveToDatabase()
int HORSE_SupportGetHenchmanExistsInDatabase(object oPC,string sDatabase,int nN=1)
{ // PURPOSE: Return TRUE if there is a henchman saved in the database
string sS=GetCampaignString(sDatabase,"sX3_HTAG_"+IntToString(nN),oPC);
if (GetStringLength(sS)>0) return TRUE;
return FALSE;
} // HORSE_SupportGetHenchmanExistsInDatabase()
void HORSE_SupportRestoreHenchmanFromDatabase(object oPC,string sDatabase,int nN=1)
{ // PURPOSE: This will reload and assign the henchman to the PC
object oHenchman=OBJECT_INVALID;
string sN=IntToString(nN);
string sTag=GetCampaignString(sDatabase,"sX3_HTAG_"+sN,oPC);
string sResRef=GetCampaignString(sDatabase,"sX3_HRR_"+sN,oPC);
int nNN=1;
object oOb;
oOb=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oPC,nNN);
while(!GetIsObjectValid(oHenchman)&&GetIsObjectValid(oOb))
{ // see if henchman is already in play
if (sResRef==GetResRef(oOb)&&GetTag(oOb)==sTag) oHenchman=oOb;
nNN++;
oOb=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oPC,nNN);
} // see if henchman is already in play
if (!GetIsObjectValid(oHenchman))
{ // create henchman
oHenchman=CreateObject(OBJECT_TYPE_CREATURE,sResRef,GetLocation(oPC),FALSE,sTag);
AssignCommand(oHenchman,SetAssociateState(NW_ASC_DISTANCE_6_METERS,TRUE));
DelayCommand(1.0,AddHenchman(oPC,oHenchman));
} // create henchman
// load and set henchman
SetCreatureAppearanceType(oHenchman,GetCampaignInt(sDatabase,"nX3_CAPP_"+sN,oPC));
SetPhenoType(GetCampaignInt(sDatabase,"nX3_CPHE_"+sN,oPC),oHenchman);
SetCreatureTailType(GetCampaignInt(sDatabase,"nX3_CTAIL_"+sN,oPC),oHenchman);
SetSkinString(oHenchman,"sX3_HorseResRef",GetCampaignString(sDatabase,"sX3_RR_"+sN,oPC));
SetSkinString(oHenchman,"sX3_HorseMountTag",GetCampaignString(sDatabase,"sX3_TAG_"+sN,oPC));
SetLocalString(oHenchman,"sX3_HorseMountScript",GetCampaignString(sDatabase,"sX3_SCR_"+sN,oPC));
SetLocalString(oHenchman,"X3_HORSE_PREMOUNT_SCRIPT",GetCampaignString(sDatabase,"sX3_SPRM_"+sN,oPC));
SetLocalString(oHenchman,"X3_HORSE_POSTMOUNT_SCRIPT",GetCampaignString(sDatabase,"sX3_SPOM_"+sN,oPC));
SetSkinString(oHenchman,"X3_HORSE_PREDISMOUNT_SCRIPT",GetCampaignString(sDatabase,"sX3_SPRDM_"+sN,oPC));
SetSkinString(oHenchman,"X3_HORSE_POSTDISMOUNT_SCRIPT",GetCampaignString(sDatabase,"sX3_SPODM_"+sN,oPC));
//SetSkinInt(oHenchman,"X3_HORSE_APPEARANCE",GetCampaignInt(sDatabase,"nX3_HAP_"+sN,oPC));
SetLocalInt(oHenchman,"X3_HORSE_TAIL",GetCampaignInt(sDatabase,"nX3_HTL_"+sN,oPC));
SetLocalInt(oHenchman,"X3_HORSE_NULL_APPEARANCE",GetCampaignInt(sDatabase,"nX3_HNAP_"+sN,oPC));
SetSkinInt(oHenchman,"nX3_HorseRiderAppearance",GetCampaignInt(sDatabase,"nX3_APP_"+sN,oPC));
SetSkinInt(oHenchman,"nX3_HorseRiderPhenotype",GetCampaignInt(sDatabase,"nX3_PHEN_"+sN,oPC));
SetSkinInt(oHenchman,"nX3_StoredFootstep",GetCampaignInt(sDatabase,"nX3_FOOT_"+sN,oPC));
SetLocalInt(oHenchman,"X3_NO_MOUNT_ANIMATE",GetCampaignInt(sDatabase,"nX3_NOAN_"+sN,oPC));
SetLocalInt(oHenchman,"nX3_HorsePortrait",GetCampaignInt(sDatabase,"nX3_HPOR_"+sN,oPC));
SetSkinInt(oHenchman,"nX3_HorseHP",GetCampaignInt(sDatabase,"nX3_HHP_"+sN,oPC));
SetLocalInt(oHenchman,"nX3_RiderHP",GetCampaignInt(sDatabase,"nX3_RHP_"+sN,oPC));
SetSkinInt(oHenchman,"bX3_IS_MOUNTED",GetCampaignInt(sDatabase,"bX3_ISM_"+sN,oPC));
SetSkinInt(oHenchman,"nX3_HorseRiderTail",GetCampaignInt(sDatabase,"nX3_TAL_"+sN,oPC));
if (GetLocalInt(GetModule(),"X3_HORSE_ENABLE_SADDLEBAGS"))
{ // saddlebags enabled
SetLocalInt(oHenchman,"bX3_HAS_SADDLEBAGS",GetCampaignInt(sDatabase,"bX3_HSB_"+sN,oPC));
} // saddlebags enabled
} // HORSE_SupportRestoreHenchmanFromDatabase()
void HORSE_SupportDeleteMountedPCFromDatabase(object oPC,string sDatabase)
{ // PURPOSE: This will remove the info about this PC being mounted
DeleteCampaignVariable(sDatabase,"sX3_RR",oPC);
DeleteCampaignVariable(sDatabase,"sX3_TAG",oPC);
DeleteCampaignVariable(sDatabase,"sX3_SCR",oPC);
DeleteCampaignVariable(sDatabase,"nX3_HAP",oPC);
DeleteCampaignVariable(sDatabase,"nX3_HTL",oPC);
DeleteCampaignVariable(sDatabase,"nX3_HNAP",oPC);
DeleteCampaignVariable(sDatabase,"nX3_HAP",oPC);
DeleteCampaignVariable(sDatabase,"nX3_APP",oPC);
DeleteCampaignVariable(sDatabase,"nX3_PHEN",oPC);
DeleteCampaignVariable(sDatabase,"nX3_FOOT",oPC);
DeleteCampaignVariable(sDatabase,"nX3_NOAN",oPC);
DeleteCampaignVariable(sDatabase,"nX3_CAPP",oPC);
DeleteCampaignVariable(sDatabase,"nX3_CPHE",oPC);
DeleteCampaignVariable(sDatabase,"nX3_CTAIL",oPC);
DeleteCampaignVariable(sDatabase,"sX3_SPRM",oPC);
DeleteCampaignVariable(sDatabase,"sX3_SPOM",oPC);
DeleteCampaignVariable(sDatabase,"sX3_SPRDM",oPC);
DeleteCampaignVariable(sDatabase,"sX3_SPODM",oPC);
DeleteCampaignVariable(sDatabase,"bX3_MOUNTED",oPC);
DeleteCampaignVariable(sDatabase,"nX3_PALUS",oPC);
DeleteCampaignVariable(sDatabase,"nX3_HPOR",oPC);
DeleteCampaignVariable(sDatabase,"nX3_HHP",oPC);
DeleteCampaignVariable(sDatabase,"nX3_RHP",oPC);
DeleteCampaignVariable(sDatabase,"bX3_ISM",oPC);
DeleteCampaignVariable(sDatabase,"nX3_TAL",oPC);
if (GetLocalInt(GetModule(),"X3_HORSE_ENABLE_SADDLEBAGS"))
{ // saddlebags enabled
DeleteCampaignVariable(sDatabase,"bX3_HSB",oPC);
} // saddlebags enabled
} // HORSE_SupportDeleteMountedPCFromDatabase()
void HORSE_SupportStoreMountedPCInDatabase(object oPC,string sDatabase)
{ // PURPOSE: This will store the PCs information about being mounted
SetCampaignString(sDatabase,"sX3_RR",GetSkinString(oPC,"sX3_HorseResRef"),oPC);
SetCampaignString(sDatabase,"sX3_TAG",GetSkinString(oPC,"sX3_HorseMountTag"),oPC);
SetCampaignString(sDatabase,"sX3_SCR",GetLocalString(oPC,"sX3_HorseMountScript"),oPC);
//SetCampaignInt(sDatabase,"nX3_HAP",GetSkinInt(oPC,"X3_HORSE_APPEARANCE"),oPC);
SetCampaignInt(sDatabase,"nX3_HTL",GetLocalInt(oPC,"X3_HORSE_TAIL"),oPC);
SetCampaignInt(sDatabase,"nX3_HNAP",GetLocalInt(oPC,"X3_HORSE_NULL_APPEARANCE"),oPC);
SetCampaignInt(sDatabase,"nX3_APP",GetSkinInt(oPC,"nX3_HorseRiderAppearance"),oPC);
SetCampaignInt(sDatabase,"nX3_PHEN",GetSkinInt(oPC,"nX3_HorseRiderPhenotype"),oPC);
SetCampaignInt(sDatabase,"nX3_FOOT",GetSkinInt(oPC,"nX3_StoredFootstep"),oPC);
SetCampaignInt(sDatabase,"nX3_NOAN",GetLocalInt(oPC,"X3_NO_MOUNT_ANIMATE"),oPC);
SetCampaignInt(sDatabase,"nX3_CAPP",GetAppearanceType(oPC),oPC);
SetCampaignInt(sDatabase,"nX3_CPHE",GetPhenoType(oPC),oPC);
SetCampaignInt(sDatabase,"nX3_CTAIL",GetCreatureTailType(oPC),oPC);
SetCampaignString(sDatabase,"sX3_SPRM",GetLocalString(oPC,"X3_HORSE_PREMOUNT_SCRIPT"),oPC);
SetCampaignString(sDatabase,"sX3_SPOM",GetLocalString(oPC,"X3_HORSE_POSTMOUNT_SCRIPT"),oPC);
SetCampaignString(sDatabase,"sX3_SPRDM",GetSkinString(oPC,"X3_HORSE_PREDISMOUNT_SCRIPT"),oPC);
SetCampaignString(sDatabase,"sX3_SPODM",GetSkinString(oPC,"X3_HORSE_POSTDISMOUNT_SCRIPT"),oPC);
SetCampaignInt(sDatabase,"nX3_HPOR",GetLocalInt(oPC,"nX3_HorsePortrait"),oPC);
SetCampaignInt(sDatabase,"nX3_RHP",GetLocalInt(oPC,"nX3_RiderHP"),oPC);
SetCampaignInt(sDatabase,"nX3_HHP",GetSkinInt(oPC,"nX3_HorseHP"),oPC);
SetCampaignInt(sDatabase,"bX3_MOUNTED",TRUE,oPC);
SetCampaignInt(sDatabase,"bX3_ISM",GetSkinInt(oPC,"bX3_IS_MOUNTED"),oPC);
SetCampaignInt(sDatabase,"nX3_PALUS",GetLocalInt(oPC,"nX3_PALADIN_UNSUMMON"),oPC);
SetCampaignInt(sDatabase,"nX3_TAL",GetSkinInt(oPC,"nX3_HorseRiderTail"),oPC);
if (GetLocalInt(GetModule(),"X3_HORSE_ENABLE_SADDLEBAGS"))
{ // saddlebags enabled
SetCampaignInt(sDatabase,"bX3_HSB",GetLocalInt(oPC,"bX3_HAS_SADDLEBAGS"),oPC);
} // saddlebags enabled
} // HORSE_SupportStoreMountedPCInDatabase()
void HORSE_SupportReloadMountedPCFromDatabase(object oPC,string sDatabase)
{ // PURPOSE: This will restore the mounted information about the PC
SetCreatureAppearanceType(oPC,GetCampaignInt(sDatabase,"nX3_CAPP",oPC));
SetPhenoType(GetCampaignInt(sDatabase,"nX3_CPHE",oPC),oPC);
SetCreatureTailType(GetCampaignInt(sDatabase,"nX3_CTAIL",oPC),oPC);
SetSkinString(oPC,"sX3_HorseResRef",GetCampaignString(sDatabase,"sX3_RR",oPC));
SetSkinString(oPC,"sX3_HorseMountTag",GetCampaignString(sDatabase,"sX3_TAG",oPC));
SetLocalString(oPC,"sX3_HorseMountScript",GetCampaignString(sDatabase,"sX3_SCR",oPC));
SetLocalString(oPC,"X3_HORSE_PREMOUNT_SCRIPT",GetCampaignString(sDatabase,"sX3_SPRM",oPC));
SetLocalString(oPC,"X3_HORSE_POSTMOUNT_SCRIPT",GetCampaignString(sDatabase,"sX3_SPOM",oPC));
SetSkinString(oPC,"X3_HORSE_PREDISMOUNT_SCRIPT",GetCampaignString(sDatabase,"sX3_SPRDM",oPC));
SetSkinString(oPC,"X3_HORSE_POSTDISMOUNT_SCRIPT",GetCampaignString(sDatabase,"sX3_SPODM",oPC));
//SetSkinInt(oPC,"X3_HORSE_APPEARANCE",GetCampaignInt(sDatabase,"nX3_HAP",oPC));
SetLocalInt(oPC,"X3_HORSE_TAIL",GetCampaignInt(sDatabase,"nX3_HTL",oPC));
SetLocalInt(oPC,"X3_HORSE_NULL_APPEARANCE",GetCampaignInt(sDatabase,"nX3_HNAP",oPC));
SetSkinInt(oPC,"nX3_HorseRiderAppearance",GetCampaignInt(sDatabase,"nX3_APP",oPC));
SetSkinInt(oPC,"nX3_HorseRiderPhenotype",GetCampaignInt(sDatabase,"nX3_PHEN",oPC));
SetSkinInt(oPC,"nX3_StoredFootstep",GetCampaignInt(sDatabase,"nX3_FOOT",oPC));
SetLocalInt(oPC,"X3_NO_MOUNT_ANIMATE",GetCampaignInt(sDatabase,"nX3_NOAN",oPC));
SetLocalInt(oPC,"nX3_PALADIN_UNSUMMON",GetCampaignInt(sDatabase,"nX3_PALUS",oPC));
SetLocalInt(oPC,"nX3_HorsePortrait",GetCampaignInt(sDatabase,"nX3_HPOR",oPC));
SetLocalInt(oPC,"nX3_RiderHP",GetCampaignInt(sDatabase,"nX3_RHP",oPC));
SetSkinInt(oPC,"nX3_HorseHP",GetCampaignInt(sDatabase,"nX3_HHP",oPC));
SetSkinInt(oPC,"bX3_IS_MOUNTED",GetCampaignInt(sDatabase,"bX3_ISM",oPC));
SetSkinInt(oPC,"nX3_HorseRiderTail",GetCampaignInt(sDatabase,"nX3_TAL",oPC));
if (GetLocalInt(GetModule(),"X3_HORSE_ENABLE_SADDLEBAGS"))
{ // saddlebags enabled
SetLocalInt(oPC,"bX3_HAS_SADDLEBAGS",GetCampaignInt(sDatabase,"bX3_HSB",oPC));
} // saddlebags enabled
if (HorseGetIsMounted(oPC)&&GetStringLeft(GetSkinString(oPC,"sX3_HorseResRef"),GetStringLength(HORSE_PALADIN_PREFIX))==HORSE_PALADIN_PREFIX) AssignCommand(oPC,HORSE_SupportMonitorPaladinUnsummon(oPC));
} // HORSE_SupportReloadMountedPCFromDatabase()
////////////////////////////////////////////////////////////////////////////////
// FUNCTIONS
//-----------------------------
// This is where the actual functions defined in the prototype section are
// completed.
////////////////////////////////////////////////////////////////////////////////
void HorseReloadFromDatabase(object oPC,string sDatabase)
{ // PURPOSE: Reload status from database and set it the same
int nN;
// Reload settings for oPC
HORSE_SupportReloadMountedPCFromDatabase(oPC,sDatabase);
nN=1;
while(HORSE_SupportGetHenchmanExistsInDatabase(oPC,sDatabase,nN))
{ // restore henchmen
HORSE_SupportRestoreHenchmanFromDatabase(oPC,sDatabase,nN);
nN++;
} // restore henchmen
} // HorseReloadFromDatabase()
void HorseSaveToDatabase(object oPC,string sDatabase)
{ // PURPOSE: Save Status to Database
int nN=1;
int nMax;
object oHenchman;
if (HorseGetIsMounted(oPC))
{ // Store mounted info
HORSE_SupportStoreMountedPCInDatabase(oPC,sDatabase);
} // Store mounted info
else
{ // Delete mounted info
HORSE_SupportDeleteMountedPCFromDatabase(oPC,sDatabase);
} // Delete mounted info
while(HORSE_SupportGetHenchmanExistsInDatabase(oPC,sDatabase,nN))
{ // count
nMax++;
nN++;
} // count
nN=1;
while(nN<=nMax)
{ // remove old
HORSE_SupportDeleteFromDatabase(oPC,sDatabase,nN);
nN++;
} // remove old
nN=1;
oHenchman=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oPC,nN);
while(GetIsObjectValid(oHenchman))
{ // traverse henchmen
HORSE_SupportSaveToDatabase(oPC,oHenchman,sDatabase,nN);
nN++;
oHenchman=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oPC,nN);
} // traverse henchmen
} // HorseSaveToDatabase()
object HorseCreateHorse(string sResRef,location lLoc,object oOwner=OBJECT_INVALID,string sTag="",int nAppearance=-1,int nTail=-1,int nFootstep=-1,string sScript="")
{ // PURPOSE: Spawn a horse with the specified information
object oHorse=OBJECT_INVALID;
int nHenchmen=HORSE_SupportCountHenchmen(oOwner);
int nMax=GetLocalInt(GetModule(),"X3_HORSE_MAX_HENCHMEN");
int bIncHenchmen=GetLocalInt(GetModule(),"X3_HORSE_NO_HENCHMAN_INCREASE");
int nCurr=GetMaxHenchmen();
if (GetStringLength(sResRef)>0)
{ // resref specified
oHorse=CreateObject(OBJECT_TYPE_CREATURE,sResRef,lLoc,FALSE,sTag);
if (GetIsObjectValid(oHorse))
{ // horse successfully created
SetLocalInt(oHorse,"bX3_IS_MOUNT",TRUE);
SetLocalString(oHorse,"sX3_OriginalName",GetName(oHorse));
if (GetIsObjectValid(oOwner)&&GetObjectType(oOwner)==OBJECT_TYPE_CREATURE)
{ // assign owner
SetLocalObject(oHorse,"oX3_HorseOwner",oOwner);
if (nHenchmen==nCurr)
{ // see if increase possible
if (!bIncHenchmen)
{ // increase is permissable
if (nMax==0||nMax>0)
{ // do the increase
SetMaxHenchmen(nCurr+1);
} // do the increase
} // increase is permissable
} // see if increase possible
AddHenchman(oOwner,oHorse);
//AssignCommand(oHorse,SetAssociateState(NW_ASC_DISTANCE_6_METERS,TRUE));
SetAssociateState(NW_ASC_DISTANCE_6_METERS,TRUE,oHorse);
if (GetMaster(oHorse)==oOwner) SetName(oHorse,GetName(oOwner)+"'s "+GetName(oHorse));
if (GetLocalInt(GetModule(),"X3_ENABLE_MOUNT_DB")&&GetIsPC(oOwner)) SetLocalInt(oOwner,"bX3_STORE_MOUNT_INFO",TRUE);
} // assign owner
if (nAppearance>-1)
{ // set appearance
SetCreatureAppearanceType(oHorse,nAppearance);
} // set appearance
if (nTail>-1)
{ // set tail
SetCreatureTailType(nTail,oHorse);
} // set tail
if (nFootstep>-1)
{ // footstep
SetFootstepType(nFootstep,oHorse);
} // footstep
if (GetStringLength(sScript)>0)
{ // post spawn script
SetLocalString(oHorse,"sX3_HORSE_CREATED_SCRIPT",sScript);
ExecuteScript(sScript,oHorse);
} // post spawn script
} // horse successfully created
} // resref specified
return oHorse;
} // HorseCreateHorse()
int HorseGetIsMounted(object oTarget)
{ // PURPOSE: Return whether oTarget is mounted
if (GetObjectType(oTarget)==OBJECT_TYPE_CREATURE)
{ // valid parameter
if (GetSkinInt(oTarget,"bX3_IS_MOUNTED")) return TRUE;
} // valid parameter
return FALSE;
} // HorseGetIsMounted()
int HorseGetCanBeMounted(object oTarget,object oRider=OBJECT_INVALID,int bAssignMount=FALSE)
{ // PURPOSE: This will return whether oTarget can be mounted
int nAppearance;
string sS;
object oOb;
if (GetObjectType(oTarget)==OBJECT_TYPE_CREATURE)
{ // valid oTarget type
if (GetIsDead(oTarget)) return FALSE;
if (GetIsPC(oTarget)) return FALSE;
if (GetIsObjectValid(oRider)&&GetObjectType(oRider)!=OBJECT_TYPE_CREATURE) return FALSE;
if (GetIsObjectValid(oRider)&&GetIsEnemy(oRider,oTarget)) return FALSE;
if (GetIsDMPossessed(oTarget)) return FALSE;
if (HorseGetIsMounted(oRider)) return FALSE;
//if (GetIsObjectValid(oRider)&&GetCreatureTailType(oRider)!=CREATURE_TAIL_TYPE_NONE) return FALSE;
nAppearance=GetAppearanceType(oRider);
if (GetIsPC(oRider)&&nAppearance>6&&GetLocalInt(oRider,"X3_CUSTOM_RACE_APPEARANCE")!=nAppearance&&!Horse_SupportRaceAppearance(nAppearance)) return FALSE; // PC is shape shifted
if (GetIsObjectValid(oRider)&&GetObjectType(oRider)==OBJECT_TYPE_CREATURE)
{ // Rider is a creature
sS=HORSE_SupportRaceRestrictString(oRider);
if (GetLocalInt(oTarget,sS)) return FALSE; // race restricted
} // Rider is a creature
oOb=GetMaster(oTarget);
if (GetIsObjectValid(oOb)&&GetIsObjectValid(oRider))
{ // has master make sure is part of party
if (oOb!=oRider&&oOb!=GetMaster(oRider)&&GetMaster(oOb)!=GetMaster(oRider))
{ // not part of party
return FALSE;
} // not part of party
} // has master make sure is part of party
if (GetLocalInt(oTarget,"X3_HORSE_NOT_RIDEABLE_OWNER"))
{ // not rideable due to owner
return FALSE;
} // not rideable due to owner
if (GetLocalInt(GetArea(oTarget),"X3_NO_MOUNTING")&&!bAssignMount)
{ // no mounting allowed in this area
return FALSE;
} // no mounting allowed in this area
sS=GetResRef(oTarget);
if (GetStringLeft(sS,GetStringLength(HORSE_PALADIN_PREFIX))==HORSE_PALADIN_PREFIX&&GetIsObjectValid(oRider))
{ // paladin mount
if (HorseGetOwner(oTarget)!=oRider) return FALSE;
} // paladin mount
if (!HorseGetIsAMount(oTarget)) return FALSE;
return TRUE;
} // valid oTarget type
return FALSE;
} // HorseGetCanBeMounted()
string HorseGetMountFailureMessage(object oTarget,object oRider=OBJECT_INVALID)
{ // PURPOSE: This will return the error message
int nAppearance;
string sS;
object oOb;
if (GetObjectType(oTarget)==OBJECT_TYPE_CREATURE)
{ // valid oTarget type
if (GetIsDead(oTarget)) return StringToRGBString(GetStringByStrRef(111993),STRING_COLOR_ROSE);
if (GetIsPC(oTarget)) return StringToRGBString(GetStringByStrRef(111994),STRING_COLOR_ROSE);
if (GetIsObjectValid(oRider)&&GetObjectType(oRider)!=OBJECT_TYPE_CREATURE) return StringToRGBString(GetStringByStrRef(111995),STRING_COLOR_ROSE);
if (GetIsObjectValid(oRider)&&GetIsEnemy(oRider,oTarget)) return StringToRGBString(GetStringByStrRef(111996),STRING_COLOR_ROSE);
if (GetIsDMPossessed(oTarget)) return StringToRGBString(GetStringByStrRef(111997),STRING_COLOR_ROSE);
if (HorseGetIsMounted(oRider)) return StringToRGBString(GetStringByStrRef(111998),STRING_COLOR_ROSE);
//if (GetIsObjectValid(oRider)&&GetCreatureTailType(oRider)!=CREATURE_TAIL_TYPE_NONE) return StringToRGBString(GetStringByStrRef(111998),STRING_COLOR_ROSE);
nAppearance=GetAppearanceType(oRider);
if (GetIsPC(oRider)&&nAppearance>6&&GetLocalInt(oRider,"X3_CUSTOM_RACE_APPEARANCE")!=nAppearance&&!Horse_SupportRaceAppearance(nAppearance)) return StringToRGBString(GetStringByStrRef(111999),STRING_COLOR_ROSE);
if (GetIsObjectValid(oRider)&&GetObjectType(oRider)==OBJECT_TYPE_CREATURE)
{ // Rider is a creature
sS=HORSE_SupportRaceRestrictString(oRider);
if (GetLocalInt(oTarget,sS)) return StringToRGBString(GetStringByStrRef(112000),STRING_COLOR_PINK);
} // Rider is a creature
oOb=GetMaster(oTarget);
if (GetIsObjectValid(oOb)&&GetIsObjectValid(oRider))
{ // has master make sure is part of party
if (oOb!=oRider&&GetMaster(oRider)!=oOb)
{ // not part of party
return StringToRGBString(GetStringByStrRef(112001),STRING_COLOR_ROSE);
} // not part of party
} // has master make sure is part of party
if (GetLocalInt(oTarget,"X3_HORSE_NOT_RIDEABLE_OWNER"))
{ // not rideable due to owner
return StringToRGBString(GetStringByStrRef(112002),STRING_COLOR_PINK);
} // not rideable due to owner
if (GetLocalInt(GetArea(oTarget),"X3_NO_MOUNTING"))
{ // no mounting allowed in this area
return StringToRGBString(GetStringByStrRef(112003),STRING_COLOR_PINK);
} // no mounting allowed in this area
sS=GetResRef(oTarget);
if (GetStringLeft(sS,GetStringLength(HORSE_PALADIN_PREFIX))==HORSE_PALADIN_PREFIX&&GetIsObjectValid(oRider))
{ // paladin mount
if (HorseGetOwner(oTarget)!=oRider) return StringToRGBString(GetStringByStrRef(112004),STRING_COLOR_PINK);
} // paladin mount
if (!HorseGetIsAMount(oTarget)) return StringToRGBString(GetStringByStrRef(112005),STRING_COLOR_PINK);
} // valid oTarget type
else
{
// The target is not a mount (as its not a creature).
return StringToRGBString(GetStringByStrRef(112005),STRING_COLOR_ROSE);
}
return "";
} // HorseGetMountFailureMessage()
void HorseSetOwner(object oHorse,object oOwner,int bAssign=FALSE)
{ // PURPOSE: Set oHorse to be owned by oOwner
object oPreviousOwner;
string sName;
int nHenchmen=HORSE_SupportCountHenchmen(oOwner);
int nMax=GetLocalInt(GetModule(),"X3_HORSE_MAX_HENCHMEN");
int bIncHenchmen=GetLocalInt(GetModule(),"X3_HORSE_NO_HENCHMAN_INCREASE");
int nCurr=GetMaxHenchmen();
if (GetObjectType(oHorse)==OBJECT_TYPE_CREATURE&&GetObjectType(oOwner)==OBJECT_TYPE_CREATURE)
{ // valid parameters
//oPreviousOwner=GetMaster(oHorse);
//if (oPreviousOwner==oOwner) return; // already is the owner
if ((HorseGetCanBeMounted(oHorse,oOwner,TRUE)||GetIsPC(oOwner))&&!HorseGetIsAMount(oOwner))
{ // horse can be mounted
oPreviousOwner=GetMaster(oHorse);
if (oPreviousOwner!=oOwner)
{ // new owner
if (GetObjectType(oPreviousOwner)==OBJECT_TYPE_CREATURE)
{ // remove as henchman from previous owner
RemoveHenchman(oPreviousOwner,oHorse);
if (GetLocalInt(GetModule(),"X3_ENABLE_MOUNT_DB")&&GetIsPC(oPreviousOwner)) SetLocalInt(oPreviousOwner,"bX3_STORE_MOUNT_INFO",TRUE);
} // remove as henchman from previous owner
if (nHenchmen==nCurr)
{ // see if increase possible
if (!bIncHenchmen)
{ // increase is permissable
if (nMax==0||nMax>0)
{ // do the increase
SetMaxHenchmen(nCurr+1);
} // do the increase
} // increase is permissable
} // see if increase possible
AssignCommand(oHorse,ClearAllActions());
AddHenchman(oOwner,oHorse);
SetLocalObject(oHorse,"oX3_HorseOwner",oOwner);
//AssignCommand(oHorse,SetAssociateState(NW_ASC_DISTANCE_6_METERS,TRUE));
DelayCommand(1.0,SetAssociateState(NW_ASC_DISTANCE_6_METERS,TRUE,oHorse));
if (bAssign) SetLocalObject(oOwner,"oAssignedHorse",oHorse);
} // new owner
else
{ // make sure variables on oHorse are correct
SetLocalObject(oHorse,"oX3_HorseOwner",oOwner);
//AssignCommand(oHorse,SetAssociateState(NW_ASC_DISTANCE_6_METERS,TRUE));
SetAssociateState(NW_ASC_DISTANCE_6_METERS,TRUE,oHorse);
if (bAssign) SetLocalObject(oOwner,"oAssignedHorse",oHorse);
if (GetLocalInt(GetModule(),"X3_ENABLE_MOUNT_DB")&&GetIsPC(oOwner)) SetLocalInt(oOwner,"bX3_STORE_MOUNT_INFO",TRUE);
} // make sure variables on oHorse are correct
sName=GetLocalString(oHorse,"sX3_OriginalName");
if (GetStringLength(sName)<1)
{ // define original name
sName=GetName(oHorse);
SetLocalString(oHorse,"sX3_OriginalName",sName);
} // define original name
if (GetMaster(oHorse)==oOwner)
{ // was set okay
if (GetStringLowerCase(GetStringRight(GetName(oOwner),1))=="s"||GetStringLowerCase(GetStringRight(GetName(oOwner),1))=="z")
SetName(oHorse,GetName(oOwner)+"' "+sName);
else { SetName(oHorse,GetName(oOwner)+"'s "+sName); }
} // was set okay
} // horse can be mounted
else
{ // not valid
if (GetIsPC(oOwner))
{ // PC
PrintString("X3 HORSE ERROR: Attempt made to set "+GetName(oOwner)+" as owner of "+GetName(oHorse)+" is an invalid assignment.");
} // PC
else
{ // error
PrintString("X3 HORSE ERROR: Attempt made to set "+GetName(oOwner)+" as owner of "+GetName(oHorse)+" is an invalid assignment.");
} // error
} // not valid
} // valid parameters
} // HorseSetOwner()
void HorseRemoveOwner(object oHorse)
{ // PURPOSE: Remove the owner from oHorse
object oOwner;
string sString;
if (GetObjectType(oHorse)==OBJECT_TYPE_CREATURE)
{ // valid parameter
oOwner=GetLocalObject(oHorse,"oX3_HorseOwner");
if (GetObjectType(oOwner)==OBJECT_TYPE_CREATURE)
{ // owner found
if (GetLocalInt(GetModule(),"X3_ENABLE_MOUNT_DB")&&GetIsPC(oOwner)) SetLocalInt(oOwner,"bX3_STORE_MOUNT_INFO",TRUE);
sString=GetResRef(oHorse);
// do not allow removing paladin horses from owner
if (GetStringLeft(sString,GetStringLength(HORSE_PALADIN_PREFIX))==HORSE_PALADIN_PREFIX) return;
if (GetMaster(oHorse)==oOwner) RemoveHenchman(oOwner,oHorse);
DeleteLocalObject(oHorse,"oX3_HorseOwner");
if (oHorse==GetLocalObject(oOwner,"oAssignedHorse"))
{
DeleteLocalObject(oOwner,"oAssignedHorse");
HORSE_Support_AssignRemainingMount(oOwner);
}
} // owner found
} // valid parameter
sString=GetLocalString(oHorse,"sX3_OriginalName");
if (GetStringLength(sString)>0) SetName(oHorse,sString);
} // HorseRemoveOwner()
object HorseGetOwner(object oHorse)
{ // PURPOSE: Return who the owner of oHorse is or return OBJECT_INVALID
object oOwner;
if (GetObjectType(oHorse)==OBJECT_TYPE_CREATURE)
{ // valid parameter
oOwner=GetLocalObject(oHorse,"oX3_HorseOwner");
if (GetObjectType(oOwner)==OBJECT_TYPE_CREATURE) return oOwner;
} // valid parameter
return OBJECT_INVALID;
} // HorseGetOwner()
void HorseAddHorseMenu(object oPC)
{ // PURPOSE: Add Horse Menu to the PC
object oSkin=SKIN_SupportGetSkin(oPC);
itemproperty iProp;
if (GetIsPC(oPC))
{ // valid parameter
iProp=ItemPropertyBonusFeat(IP_CONST_HORSE_MENU);
AddItemProperty(DURATION_TYPE_PERMANENT,iProp,oSkin);
} // valid parameter
} // HorseAddHorseMenu()
void HorseSetPhenotype(object oRider,int bJoust=FALSE)
{ // PURPOSE: To set the proper phenotype for oRider when mounted
int nCurrent=GetPhenoType(oRider);
int nPheno=-1;
if (bJoust)
{ // jousting
if (GetLocalInt(oRider,"X3_CUSTOM_RACE_JOUST_PHENO")>0) nPheno=GetLocalInt(oRider,"X3_CUSTOM_RACE_JOUST_PHENO");
else if (nCurrent==PHENOTYPE_NORMAL||nCurrent==HORSE_PHENOTYPE_MOUNTED_N) nPheno=HORSE_PHENOTYPE_JOUSTING_N;
else if (nCurrent==PHENOTYPE_BIG||nCurrent==HORSE_PHENOTYPE_MOUNTED_L) nPheno=HORSE_PHENOTYPE_JOUSTING_L;
} // jousting
else
{ // not jousting
if (GetLocalInt(oRider,"X3_CUSTOM_RACE_MOUNTED_PHENO")>0) nPheno=GetLocalInt(oRider,"X3_CUSTOM_RACE_MOUNTED_PHENO");
else if (nCurrent==PHENOTYPE_NORMAL||nCurrent==HORSE_PHENOTYPE_JOUSTING_N) nPheno=HORSE_PHENOTYPE_MOUNTED_N;
else if (nCurrent==PHENOTYPE_BIG||nCurrent==HORSE_PHENOTYPE_JOUSTING_L) nPheno=HORSE_PHENOTYPE_MOUNTED_L;
} // not jousting
if (nPheno!=-1) SetPhenoType(nPheno,oRider);
} // HorseSetPhenotype()
int HorseGetIsDisabled(object oCreature)
{ // PURPOSE: blocked path, death, entanglement, confusion, paralysis, stun, petrification, fear, turned, sleep, knockdown detection by Azbest
int bDisabled=FALSE;
if (GetIsDead(oCreature)||GetLocalInt(oCreature,"bPathIsBlocked"))
{
bDisabled=TRUE;
}
else
{
effect eEffect=GetFirstEffect(oCreature);
while(GetIsEffectValid(eEffect)&&!bDisabled)
{ // traverse effects
if ((GetEffectType(eEffect)==EFFECT_TYPE_INVALIDEFFECT&&(GetEffectDurationType(eEffect)==DURATION_TYPE_TEMPORARY||GetEffectDurationType(eEffect)==DURATION_TYPE_PERMANENT))||
GetEffectType(eEffect)==EFFECT_TYPE_ENTANGLE||
GetEffectType(eEffect)==EFFECT_TYPE_CONFUSED||
GetEffectType(eEffect)==EFFECT_TYPE_PARALYZE||
GetEffectType(eEffect)==EFFECT_TYPE_STUNNED||
GetEffectType(eEffect)==EFFECT_TYPE_PETRIFY||
GetEffectType(eEffect)==EFFECT_TYPE_FRIGHTENED||
GetEffectType(eEffect)==EFFECT_TYPE_TURNED||
GetEffectType(eEffect)==EFFECT_TYPE_SLEEP)
{ // match
bDisabled=TRUE;
} // match
eEffect=GetNextEffect(oCreature);
} // traverse effects
}
return bDisabled;
} // HorseGetIsDisabled()
void HorseMount(object oHorse,int bAnimate=TRUE,int bInstant=FALSE,int nState=0)
{ // PURPOSE: Handle the mounting of oHorse
// 0 = Initial state
// 1 = move to horse
// 2 = adjust for animation
// 3 = animate
// 4 = store horse and premount info
// 5 = set appearances
// 6 = destroy oHorse
object oRider=OBJECT_SELF;
float fMonitorSpeed=0.4f;
float fDenominator=3.0; // fraction denominator used to calculate synchronised timing of the process that runs concurently with animation
string sS;
string sTag;
string sResRef;
int nApp,nTail,nPheno,nN;
location lLoc;
float fF;
object oOb;
int nStoredState=GetLocalInt(oRider,"nX3_StoredMountState");
int bPostMount=FALSE;
float fTimeDelay; // used to hold delay common references and reduce some repeated math
int bAct=GetLocalInt(GetModule(),"X3_HORSE_ACT_VS_DELAY");
float fX3_MOUNT_MULTIPLE=GetLocalFloat(GetArea(oRider),"fX3_MOUNT_MULTIPLE");
float fX3_TIMEOUT_TO_MOUNT=GetLocalFloat(GetModule(),"fX3_TIMEOUT_TO_MOUNT");
if (GetLocalFloat(oRider,"fX3_MOUNT_MULTIPLE")>fX3_MOUNT_MULTIPLE) fX3_MOUNT_MULTIPLE=GetLocalFloat(oRider,"fX3_MOUNT_MULTIPLE");
if (fX3_MOUNT_MULTIPLE<=0.0) fX3_MOUNT_MULTIPLE=1.0;
if (GetLocalFloat(oRider,"fX3_TIMEOUT_TO_MOUNT")!=0.0) fX3_TIMEOUT_TO_MOUNT=GetLocalFloat(oRider,"fX3_TIMEOUT_TO_MOUNT");
if (fX3_TIMEOUT_TO_MOUNT<6.0) fX3_TIMEOUT_TO_MOUNT=18.0;
if (GetStringLength(GetLocalString(oHorse,"X3_HORSE_POSTMOUNT_SCRIPT"))>0) bPostMount=TRUE;
if (HorseGetIsDisabled()) return; // drop out if disabled
if (GetObjectType(oHorse)!=OBJECT_TYPE_CREATURE) return;
if (GetIsInCombat(oRider)||GetIsInCombat(oHorse)||IsInConversation(oRider)) return;
if (nState!=0&&nStoredState>nState) return; // abort recursion of this state
switch(nState)
{ // main mounting switch
case 0:
{ // ----- 0 - Initial State
DeleteLocalInt(oRider,"nX3_StoredMountState");
if (!HorseGetCanBeMounted(oHorse,oRider))
{ // not mountable
if (GetIsPC(oRider))
{ // provide error message
FloatingTextStrRefOnCreature(111983,oRider,FALSE);
} // provide error message
return;
} // not mountable
if (GetDistanceBetween(oHorse,oRider)>20.0)
{ // too far
ClearAllActions();
ActionMoveToObject(oHorse,TRUE,15.0);
return;
} // too far
sS=GetLocalString(oHorse,"X3_HORSE_PREMOUNT_SCRIPT");
if (GetStringLength(sS)>0)
{ // premount script
SetLocalObject(oRider,"oX3_TempHorse",oHorse);
ExecuteScript(sS,oRider);
if (GetLocalInt(oHorse,"X3_HORSE_NOMOUNT"))
{ // no mount set
DeleteLocalInt(oHorse,"X3_HORSE_NOMOUNT");
return;
} // no mount set
} // premount script
SetLocalInt(oHorse,"X3_DOING_HORSE_ACTION",TRUE);
SetLocalInt(oRider,"X3_DOING_HORSE_ACTION",TRUE);
if (bAct)
{ // make sure horse action can be cancelled if actions cancelled
fTimeDelay=8.0*fX3_MOUNT_MULTIPLE;
DelayCommand(fTimeDelay,DeleteLocalInt(oHorse,"X3_DOING_HORSE_ACTION"));
DelayCommand(fTimeDelay,DeleteLocalInt(oRider,"X3_DOING_HORSE_ACTION"));
} // make sure horse action can be cancelled if actions cancelled
SetSkinInt(oRider,"nX3_StoredFootstep",GetFootstepType(oRider));
SetSkinInt(oRider,"nX3_HorseRiderPhenotype",GetPhenoType(oRider));
SetLocalInt(oRider,"X3_HORSE_TAIL",HorseGetMountTail(oHorse));
SetSkinInt(oRider,"nX3_HorseRiderAppearance",GetAppearanceType(oRider));
SetSkinInt(oRider,"nX3_HorseTail",GetCreatureTailType(oHorse));
SetSkinInt(oRider,"nX3_HorseAppearance",GetAppearanceType(oHorse));
SetLocalInt(oRider,"nX3_HorsePortrait",GetPortraitId(oHorse));
SetLocalInt(oRider,"nX3_RiderHP",GetCurrentHitPoints(oRider));
SetSkinInt(oRider,"nX3_HorseHP",GetCurrentHitPoints(oHorse));
SetSkinInt(oRider,"nX3_HorseRiderTail",GetCreatureTailType(oRider));
SetLocalInt(oRider,"bX3_IS_RIDER",TRUE);
if (!GetIsPC(oRider))
{ // not a PC - set associate state
SetAssociateState(NW_ASC_IS_BUSY,TRUE,oRider);
} // not a PC - set associate state
AssignCommand(oHorse,ClearAllActions(TRUE));
fTimeDelay=(fX3_TIMEOUT_TO_MOUNT+HORSE_MOUNT_DURATION)*fX3_MOUNT_MULTIPLE;
ApplyEffectToObject(DURATION_TYPE_TEMPORARY,EffectCutsceneImmobilize(),oHorse,fTimeDelay);
ApplyEffectToObject(DURATION_TYPE_TEMPORARY,EffectCutsceneGhost(),oHorse,fTimeDelay);
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE")&&!bAct) AssignCommand(GetModule(),DelayCommand(2.0,HORSE_SupportSetMountingSentinel(oRider,oHorse,fX3_TIMEOUT_TO_MOUNT)));
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE")&&!bAct) SetCommandable(FALSE,oHorse);
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE")&&!bAct) SetCommandable(FALSE,oRider);
fTimeDelay=fMonitorSpeed*fX3_MOUNT_MULTIPLE;
SetLocalFloat(oRider,"X3_TOTAL_MOUNT_ANIMATION_DELAY",fDenominator*fTimeDelay); // sets a small time delay so that the following code doesnt happen at once
if (!bAnimate&&bInstant)
{ // rapid mount
DelayCommand(fTimeDelay,HorseMount(oHorse,bAnimate,bInstant,4));
} // rapid mount
else
{ // move
if (!bAct)
{ // not actions
DelayCommand(fTimeDelay,HorseMount(oHorse,bAnimate,bInstant,1));
} // not actions
else
{ // use actions - interruptable
ClearAllActions();
ActionMoveToObject(oHorse,TRUE,1.5);
ActionDoCommand(HorseMount(oHorse,bAnimate,bInstant,1));
} // use actions - interruptable
} // move
break;
} // ----- 0 - Initial State
case 1:
{ // ----- 1 - Move To Horse
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE")&&GetCommandable(oHorse)&&!bAct) SetCommandable(FALSE,oHorse);
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE")&&GetCommandable(oRider)&&!bAct) SetCommandable(FALSE,oRider);
oOb=GetLocalObject(oHorse,"oX3_TempRider");
if (oOb!=oRider&&GetIsObjectValid(oOb))
{ // someone else is mounting that
if (GetIsPC(oRider))
{ // someone else mounting
FloatingTextStringOnCreature(GetName(oOb)+GetStringByStrRef(111984),oRider,FALSE);
} // someone else mounting
DeleteLocalInt(oRider,"X3_DOING_HORSE_ACTION");
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE")&&!bAct) SetCommandable(TRUE,oRider);
return;
} // someone else is mounting that
SetLocalObject(oHorse,"oX3_TempRider",oRider);
lLoc=HORSE_SupportGetMountLocation(oHorse,oRider,-90.0);
if (GetLocalInt(GetArea(oRider),"bX3_MOUNT_NO_ZAXIS")||GetLocalInt(GetModule(),"bX3_MOUNT_NO_ZAXIS")||GetLocalInt(oRider,"bX3_MOUNT_NO_ZAXIS"))
{ // use vector without z axis - thanks Azbest
vector vLoc=GetPositionFromLocation(lLoc);
vector vRider=GetPosition(oRider);
vLoc-=Vector(0.0,0.0,vLoc.z);
vRider-=Vector(0.0,0.0,vRider.z);
fF=VectorMagnitude(vLoc-vRider);
} // use vector without z axis - thanks Azbest
else
{ // use location including Z axis
fF=GetDistanceBetweenLocations(GetLocation(oRider),lLoc);
} // use location including Z axis
fMonitorSpeed=GetLocalFloat(GetModule(),"fX3_FREQUENCY");
if (fMonitorSpeed<0.5||fMonitorSpeed>9.0) fMonitorSpeed=1.0; // frequency
fTimeDelay=fMonitorSpeed*fX3_MOUNT_MULTIPLE;
if (fX3_TIMEOUT_TO_MOUNT<fTimeDelay) fX3_TIMEOUT_TO_MOUNT=fTimeDelay;
if (fF>0.1&&GetLocalInt(oRider,"nX3_MovingMount")<=FloatToInt(fX3_TIMEOUT_TO_MOUNT/fTimeDelay))
{ // keep moving
nN=GetLocalInt(oRider,"nX3_MovingMount");
nN++; // used to support timing out if cannot reach destination
SetLocalInt(oRider,"nX3_MovingMount",nN);
if (nN>FloatToInt(fX3_TIMEOUT_TO_MOUNT/fTimeDelay))
{ // timed out
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE"))
{
SetCommandable(TRUE,oRider);
ClearAllActions();
ActionWait(X3_ACTION_DELAY*fX3_MOUNT_MULTIPLE);
ActionJumpToLocation(lLoc);
SetCommandable(FALSE,oRider);
}
bAnimate=FALSE;
} // timed out
else
{ // keep trying
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE"))
{
SetCommandable(TRUE,oRider);
ClearAllActions();
ActionMoveToLocation(lLoc,TRUE);
SetCommandable(FALSE,oRider);
}
SetLocalInt(oRider,"nX3_StoredMountState",1);
float fLastHorseDist=GetDistanceBetween(oRider,oHorse);
if (fLastHorseDist==GetLocalFloat(oRider,"fLastHorseDist"))
{
if (GetLocalInt(GetArea(oRider),"X3_ABORT_WHEN_STUCK")||GetLocalInt(oHorse,"X3_ABORT_WHEN_STUCK"))
{ // break next call if we are found twice within the same distance from horse, which means we are stuck
SetLocalInt(oRider,"bPathIsBlocked",TRUE);
SetLocalInt(oRider,"nX3_StoredMountState",2);
} // break next call if we are found twice within the same distance from horse, which means we are stuck
else
{ // if we are stuck, we want to mount
// no need to set bPathIsBlocked flag, because forced mount continues the proper chain of commands
SetLocalInt(oRider,"nX3_MovingMount",FloatToInt(fX3_TIMEOUT_TO_MOUNT/fTimeDelay)+1);
} // if we are stuck, we want to mount
}
SetLocalFloat(oRider,"fLastHorseDist",fLastHorseDist);
} // keep trying
DelayCommand(fTimeDelay,HorseMount(oHorse,bAnimate,bInstant,1));
} // keep moving
else
{ // next state
DeleteLocalInt(oRider,"nX3_MovingMount");
DeleteLocalFloat(oRider,"fLastHorseDist");
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE"))
{
SetCommandable(TRUE,oRider);
ClearAllActions();
SetFacing(GetFacing(oHorse));
SetCommandable(FALSE,oRider);
}
if (bAnimate&&!GetLocalInt(oHorse,"X3_NO_MOUNT_ANIMATE"))
{ // see about animate
DelayCommand(0.1*fX3_MOUNT_MULTIPLE,HorseMount(oHorse,bAnimate,bInstant,2));
} // see about animate
else
{ // no animate
DelayCommand(0.1*fX3_MOUNT_MULTIPLE,HorseMount(oHorse,bAnimate,bInstant,4));
} // no animate
} // next state
break;
} // ----- 1 - Move to Horse
case 2:
{ // ----- 2 - Adjust for Animation
SetLocalInt(oRider,"nX3_StoredMountState",3);
nTail=GetLocalInt(oRider,"X3_HORSE_TAIL");
SetLocalInt(oHorse,"X3_HORSE_TAIL",nTail);
nApp=HORSE_SupportNullAppearance(oHorse,oRider);
if (nApp>0) SetCreatureAppearanceType(oHorse,nApp);
SetCreatureTailType(nTail,oHorse);
DelayCommand(1.0*fX3_MOUNT_MULTIPLE,HorseMount(oHorse,bAnimate,bInstant,3)); // 1.0 second seems to be long enough to set up for animation
break;
} // ----- 2 - Adjust for Animation
case 3:
{ // ----- 3 - Animate
SetLocalInt(oRider,"nX3_StoredMountState",4);
fMonitorSpeed=HORSE_MOUNT_DURATION;
if (GetLocalFloat(oHorse,"X3_HORSE_MOUNT_DURATION")>0.0) fMonitorSpeed=GetLocalFloat(oHorse,"X3_HORSE_MOUNT_DURATION");
if (fMonitorSpeed<1.0) fMonitorSpeed=1.0;
fMonitorSpeed*=fX3_MOUNT_MULTIPLE;
ApplyEffectToObject(DURATION_TYPE_TEMPORARY,EffectCutsceneGhost(),oHorse,fMonitorSpeed+0.5);
ApplyEffectToObject(DURATION_TYPE_TEMPORARY,EffectCutsceneGhost(),oRider,fMonitorSpeed+0.5);
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE"))
{
SetCommandable(TRUE,oRider);
ClearAllActions();
ActionPlayAnimation(HORSE_ANIMATION_MOUNT,1.0,fMonitorSpeed);
SetCommandable(FALSE,oRider);
}
SetLocalFloat(oRider,"X3_TOTAL_MOUNT_ANIMATION_DELAY",fMonitorSpeed); // "case 4:" and "case 5:" need to happen concurrently and with the same time duration as the animation itself :)
HorseMount(oHorse,bAnimate,bInstant,4);
break;
} // ----- 3 - Animate
case 4:
{ // ----- 4 - Store Horse and Premount Info
SetLocalInt(oRider,"nX3_StoredMountState",5);
sResRef=GetResRef(oHorse);
if (GetStringLength(sResRef)<1) SendMessageToPC(oRider,"x3_inc_horse(HorseMount): Error finding horse ResRef in case 4.");
SetSkinString(oRider,"sX3_HorseResRef",sResRef);
SetSkinString(oRider,"sX3_HorseMountTag",GetTag(oHorse));
SetLocalString(oRider,"sX3_HorseMountScript",GetLocalString(oHorse,"sX3_HORSE_CREATED_SCRIPT"));
SetSkinString(oRider,"X3_HORSE_PREDISMOUNT_SCRIPT",GetLocalString(oHorse,"X3_HORSE_PREDISMOUNT_SCRIPT"));
SetSkinString(oRider,"X3_HORSE_POSTDISMOUNT_SCRIPT",GetLocalString(oHorse,"X3_HORSE_POSTDISMOUNT_SCRIPT"));
SetLocalInt(oRider,"X3_NO_MOUNT_ANIMATE",GetLocalInt(oHorse,"X3_NO_MOUNT_ANIMATE"));
if (GetLocalInt(oHorse,"X3_ABORT_WHEN_STUCK")) SetLocalInt(oRider,"X3_ABORT_WHEN_STUCK",TRUE);
if (GetLocalInt(GetModule(),"X3_HORSE_ENABLE_SADDLEBAGS"))
{ // saddlebags support enabled
if (GetLocalInt(oHorse,"bX3_HAS_SADDLEBAGS")&&GetLocalObject(oHorse,"oX3_HorseOwner")!=oRider) if (GetMaster(oHorse)!=oRider) HorseSetOwner(oHorse,oRider,TRUE);
SetLocalInt(oRider,"bX3_HAS_SADDLEBAGS",GetLocalInt(oHorse,"bX3_HAS_SADDLEBAGS"));
if (GetLocalInt(oHorse,"bX3_HAS_SADDLEBAGS"))
{ // transfer inventory
HorseStoreInventory(oHorse,oRider);
} // transfer inventory
} // saddlebads support enabled
fMonitorSpeed=GetLocalFloat(oRider,"X3_TOTAL_MOUNT_ANIMATION_DELAY")*(fDenominator-1.0)/fDenominator;
DelayCommand(fMonitorSpeed,HorseMount(oHorse,bAnimate,bInstant,5));
break;
} // ----- 4 - Store Horse and Premount Info
case 5:
{ // ----- 5 - Set Appearance
SetLocalInt(oRider,"nX3_StoredMountState",6);
nApp=GetAppearanceType(oHorse);
nTail=GetLocalInt(oRider,"X3_HORSE_TAIL");
SetCreatureAppearanceType(oRider,HORSE_SupportGetMountedAppearance(oRider));
HorseSetPhenotype(oRider);
SetCreatureTailType(nTail,oRider);
nApp=GetLocalInt(oHorse,"X3_HORSE_FOOTSTEP");
if (nApp>0) SetFootstepType(nApp,oRider);
else { SetFootstepType(HORSE_FOOTSTEP_SOUND,oRider); }
fMonitorSpeed=GetLocalFloat(oRider,"X3_TOTAL_MOUNT_ANIMATION_DELAY")/fDenominator;
if (!bAnimate||bInstant) fTimeDelay=fMonitorSpeed;
else fTimeDelay=0.6; // changing appearance, pheno and tail takes about 0.6 second, hide the horse after that amount of time
DelayCommand(fTimeDelay,ApplyEffectToObject(DURATION_TYPE_TEMPORARY,EffectVisualEffect(VFX_DUR_CUTSCENE_INVISIBILITY),oHorse,fMonitorSpeed*fDenominator));
DelayCommand(fMonitorSpeed,HorseMount(oHorse,bAnimate,bInstant,6));
break;
} // ----- 5 - Set Appearance
case 6:
{ // ----- 6 - Destroy oHorse
SetLocalInt(oRider,"nX3_StoredMountState",7);
SetSkinInt(oRider,"bX3_IS_MOUNTED",TRUE);
SetLocalObject(oRider,"oX3_Saddlebags",GetLocalObject(oHorse,"oX3_Saddlebags"));
sS=GetLocalString(oHorse,"X3_HORSE_POSTMOUNT_SCRIPT");
if (bPostMount)
{ // run post mount script
SetLocalObject(oRider,"oX3_TempHorse",oHorse);
ExecuteScript(sS,oRider);
DeleteLocalObject(oRider,"oX3_TempHorse");
DeleteLocalInt(oRider,"bX3_HORSE_MODIFIERS");
} // run post mount script
else
{ // no post mount script
HORSE_SupportIncreaseSpeed(oRider,oHorse);
HORSE_SupportAdjustMountedArcheryPenalty(oRider);
DelayCommand(0.4,HORSE_SupportApplyMountedSkillDecreases(oRider));
HORSE_SupportApplyACBonus(oRider,oHorse);
HORSE_SupportApplyHPBonus(oRider,oHorse);
} // no post mount script
SetLocalInt(oRider,"bX3_HORSE_MODIFIERS",TRUE);
if (GetLocalInt(GetModule(),"X3_ENABLE_MOUNT_DB")) SetLocalInt(oRider,"bX3_STORE_MOUNT_INFO",TRUE);
DeleteLocalFloat(oRider,"X3_TOTAL_MOUNT_ANIMATION_DELAY");
AssignCommand(oHorse,SetIsDestroyable(TRUE,FALSE,FALSE));
DestroyObject(oHorse,0.3);
DelayCommand(0.5*fX3_MOUNT_MULTIPLE,DeleteLocalInt(oRider,"X3_DOING_HORSE_ACTION"));
if (!GetIsPC(oRider))
{ // not a PC - set associate state
DelayCommand(0.5*fX3_MOUNT_MULTIPLE,SetAssociateState(NW_ASC_IS_BUSY,FALSE,oRider));
} // not a PC - set associate state
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE")) DelayCommand(0.5*fX3_MOUNT_MULTIPLE,SetCommandable(TRUE,oRider));
break;
} // ----- 6 - Destroy oHorse
default: break;
} // main mounting switch
} // HorseMount()
object HorseDismount(int bAnimate=TRUE,int bSetOwner=TRUE)
{ // PURPOSE: Dismount the horse
object oRider=OBJECT_SELF;
object oHorse=OBJECT_INVALID;
object oOb;
string sS,sRR,sT;
int nN,nApp,nTail,nFootstep;
int bPostDismount=FALSE;
location lLoc;
float fX3_MOUNT_MULTIPLE=GetLocalFloat(GetArea(oRider),"fX3_MOUNT_MULTIPLE");
float fX3_DISMOUNT_MULTIPLE=GetLocalFloat(GetArea(oRider),"fX3_DISMOUNT_MULTIPLE");
if (GetLocalFloat(oRider,"fX3_MOUNT_MULTIPLE")>fX3_MOUNT_MULTIPLE) fX3_MOUNT_MULTIPLE=GetLocalFloat(oRider,"fX3_MOUNT_MULTIPLE");
if (fX3_MOUNT_MULTIPLE<=0.0) fX3_MOUNT_MULTIPLE=1.0;
if (GetLocalFloat(oRider,"fX3_DISMOUNT_MULTIPLE")>0.0) fX3_DISMOUNT_MULTIPLE=GetLocalFloat(oRider,"fX3_DISMOUNT_MULTIPLE");
if (fX3_DISMOUNT_MULTIPLE>0.0) fX3_MOUNT_MULTIPLE=fX3_DISMOUNT_MULTIPLE; // use dismount multiple instead of mount multiple
float fDelay=1.0*fX3_MOUNT_MULTIPLE; // base delay for non-animated dismount
if (GetStringLength(GetLocalString(oHorse,"X3_HORSE_POSTDISMOUNT_SCRIPT"))>0) bPostDismount=TRUE;
if (GetIsObjectValid(oRider))
{ // oRider is a valid object
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE")) SetCommandable(FALSE,oRider);
if (HorseGetIsMounted(oRider))
{ // Mounted
if (!GetIsPC(oRider))
{ // not PC - set associate state busy
SetAssociateState(NW_ASC_IS_BUSY,TRUE,oRider);
} // not PC - set associate state busy
sS=GetSkinString(oRider,"X3_HORSE_PREDISMOUNT_SCRIPT");
if (GetStringLength(sS)>0)
{ // PREDISMOUNT SCRIPT
ExecuteScript(sS,oRider);
if (GetLocalInt(oRider,"X3_HORSE_NOMOUNT"))
{ // abort
DeleteLocalInt(oRider,"X3_HORSE_NOMOUNT");
return OBJECT_INVALID;
} // abort
} // PREDISMOUNT SCRIPT
DeleteSkinString(oRider,"X3_HORSE_PREDISMOUNT_SCRIPT");
SetLocalInt(oRider,"X3_DOING_HORSE_ACTION",TRUE);
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE"))
{
SetCommandable(TRUE,oRider);
ClearAllActions();
SetCommandable(FALSE,oRider);
}
if (bAnimate&&!GetLocalInt(oRider,"X3_NO_MOUNT_ANIMATE"))
{ // animated dismount
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE"))
{
SetCommandable(TRUE,oRider);
ActionWait(X3_ACTION_DELAY*fX3_MOUNT_MULTIPLE);
ActionPlayAnimation(HORSE_ANIMATION_DISMOUNT,1.0,HORSE_DISMOUNT_DURATION*fX3_MOUNT_MULTIPLE);
SetCommandable(FALSE,oRider);
}
fDelay=(X3_ACTION_DELAY+HORSE_DISMOUNT_DURATION)*fX3_MOUNT_MULTIPLE; // delay for animated dismount
}
sRR=GetSkinString(oRider,"sX3_HorseResRef");
if (GetStringLength(sRR)>0)
{ // create dismounted horse
if (bSetOwner) oOb=oRider;
sT=GetSkinString(oRider,"sX3_HorseMountTag");
nApp=GetSkinInt(oRider,"nX3_HorseAppearance");
nTail=GetSkinInt(oRider,"nX3_HorseTail"); // add this to be preserved
if (nApp<7) nApp=-1;
if (nTail<3) nTail=-1;
if (nFootstep<2) nFootstep=-1;
nFootstep=GetFootstepType(oRider);
sS=GetLocalString(oRider,"sX3_HorseMountScript");
lLoc=HORSE_SupportGetMountLocation(oRider,oRider,-90.0);
ApplyEffectToObject(DURATION_TYPE_TEMPORARY,EffectCutsceneGhost(),oRider,fDelay+1.0*fX3_MOUNT_MULTIPLE);
oHorse=HorseCreateHorse(sRR,GetLocation(oRider),oOb,sT,nApp,nTail,nFootstep,sS);
SetLocalInt(oHorse,"X3_DOING_HORSE_ACTION",TRUE);
if (GetLocalInt(oRider,"X3_ABORT_WHEN_STUCK")) SetLocalInt(oHorse,"X3_ABORT_WHEN_STUCK",TRUE);
if (!GetIsObjectValid(oHorse))
{ // failed to create
SendMessageToPC(oRider,"x3_inc_horse(HorseDismount): Failed to create horse.");
} // failed to create
if (oOb==oRider) SetLocalObject(oRider,"oAssignedHorse",oHorse);
if (GetLocalInt(oRider,"nX3_HorsePortrait")>0) SetPortraitId(oHorse,GetLocalInt(oRider,"nX3_HorsePortrait"));
ApplyEffectToObject(DURATION_TYPE_TEMPORARY,EffectVisualEffect(VFX_DUR_CUTSCENE_INVISIBILITY),oHorse,fDelay+0.5*fX3_MOUNT_MULTIPLE);
if (GetLocalInt(GetModule(),"X3_HORSE_ENABLE_SADDLEBAGS")&&GetIsObjectValid(oHorse))
{ // saddlebags support enabled
SetLocalInt(oHorse,"bX3_HAS_SADDLEBAGS",GetLocalInt(oRider,"bX3_HAS_SADDLEBAGS"));
if (GetLocalInt(oRider,"bX3_HAS_SADDLEBAGS"))
{ // transfer contents
HorseRestoreInventory(oHorse);
} // transfer contents
} // saddlebads support enabled
} // create dismounted horse
else
{ // resref not defined
SendMessageToPC(oRider,"x3_inc_horse(HorseDismount): Error resref missing.");
} // resref not defined
DeleteSkinInt(oRider,"bX3_IS_MOUNTED");
if (GetLocalInt(GetModule(),"X3_ENABLE_MOUNT_DB")) SetLocalInt(oRider,"bX3_STORE_MOUNT_INFO",TRUE);
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE"))
{
SetCommandable(TRUE,oRider);
if (!bAnimate||GetLocalInt(oRider,"X3_NO_MOUNT_ANIMATE")) ActionWait(fDelay);
ActionMoveToLocation(lLoc,FALSE);
SetCommandable(FALSE,oRider);
}
DelayCommand(fDelay-0.8*fX3_MOUNT_MULTIPLE,ApplyEffectToObject(DURATION_TYPE_TEMPORARY,EffectCutsceneGhost(),oHorse,1.7*fX3_MOUNT_MULTIPLE));
//DelayCommand(fDelay-0.8*fX3_MOUNT_MULTIPLE+1.7*fX3_MOUNT_MULTIPLE, FloatingTextStringOnCreature("ghost off",oRider)); // this tells us when the ghost effect from the above line wears off
DelayCommand(fDelay-0.7*fX3_MOUNT_MULTIPLE,HORSE_SupportResetUnmountedAppearance(oRider)); // keep in mind: changing mounted appearances takes about 0.6 seconds
DelayCommand(fDelay+0.0*fX3_MOUNT_MULTIPLE,HORSE_SupportTransferPreservedValues(oRider,oHorse));
DelayCommand(fDelay+0.7*fX3_MOUNT_MULTIPLE,HORSE_SupportCleanVariables(oRider));
DelayCommand(fDelay+1.0*fX3_MOUNT_MULTIPLE,DeleteLocalInt(oRider,"X3_DOING_HORSE_ACTION"));
DelayCommand(fDelay+1.0*fX3_MOUNT_MULTIPLE,DeleteLocalInt(oHorse,"X3_DOING_HORSE_ACTION"));
if (!GetIsPC(oRider))
{ // not PC - set associate state not busy
DelayCommand(fDelay+1.0*fX3_MOUNT_MULTIPLE,SetAssociateState(NW_ASC_IS_BUSY,FALSE,oRider));
} // not PC - set associate state not busy
if (!GetLocalInt(GetModule(),"X3_NO_MOUNT_COMMANDABLE")) DelayCommand(fDelay+1.0*fX3_MOUNT_MULTIPLE,SetCommandable(TRUE,oRider));
} // Mounted
} // oRider is a valid object
return oHorse;
} // HorseDismount()
int HorseGetMountTail(object oHorse)
{ // PURPOSE: Determine the tail that is needed to represent this horse
int nTail=GetCreatureTailType(oHorse);
int nApp=GetAppearanceType(oHorse);
if (GetLocalInt(oHorse,"X3_HORSE_TAIL")>0) return GetLocalInt(oHorse,"X3_HORSE_TAIL");
if (nApp>=HORSE_APPEARANCE_OFFSET&&nApp<=HORSE_APPEARANCE_OFFSET+HORSE_NUMBER_OF_HORSES)
{ // default horses
return (nApp-HORSE_APPEARANCE_OFFSET)+HORSE_TAIL_OFFSET;
} // default horses
else if (nApp>=HORSE_NULL_RACE_DWARF&&nApp<=HORSE_NULL_RACE_HUMAN) return nTail;
return CREATURE_TAIL_TYPE_NONE;
} // HorseGetMountTail()
void HorseInstantDismount(object oRider)
{ // PURPOSE: Instantly dismount oRider
float fX3_MOUNT_MULTIPLE=GetLocalFloat(GetArea(oRider),"fX3_MOUNT_MULTIPLE");
float fX3_DISMOUNT_MULTIPLE=GetLocalFloat(GetArea(oRider),"fX3_DISMOUNT_MULTIPLE");
if (GetLocalFloat(oRider,"fX3_MOUNT_MULTIPLE")>fX3_MOUNT_MULTIPLE) fX3_MOUNT_MULTIPLE=GetLocalFloat(oRider,"fX3_MOUNT_MULTIPLE");
if (fX3_MOUNT_MULTIPLE<=0.0) fX3_MOUNT_MULTIPLE=1.0;
if (GetLocalFloat(oRider,"fX3_DISMOUNT_MULTIPLE")>0.0) fX3_DISMOUNT_MULTIPLE=GetLocalFloat(oRider,"fX3_DISMOUNT_MULTIPLE");
if (fX3_DISMOUNT_MULTIPLE>0.0) fX3_MOUNT_MULTIPLE=fX3_DISMOUNT_MULTIPLE; // use dismount multiple instead of mount multiple
HORSE_SupportResetUnmountedAppearance(oRider);
DeleteLocalInt(oRider,"bX3_HORSE_MODIFIERS");
DelayCommand(0.1*fX3_MOUNT_MULTIPLE,HORSE_SupportOriginalSpeed(oRider));
DelayCommand(0.2*fX3_MOUNT_MULTIPLE,HORSE_SupportRemoveMountedSkillDecreases(oRider));
DelayCommand(0.2*fX3_MOUNT_MULTIPLE,HORSE_SupportAdjustMountedArcheryPenalty(oRider));
DelayCommand(0.3*fX3_MOUNT_MULTIPLE,HORSE_SupportRemoveACBonus(oRider));
DelayCommand(0.3*fX3_MOUNT_MULTIPLE,HORSE_SupportRemoveHPBonus(oRider));
if (GetLocalInt(GetModule(),"X3_ENABLE_MOUNT_DB")) SetLocalInt(oRider,"bX3_STORE_MOUNT_INFO",TRUE);
DelayCommand(0.4*fX3_MOUNT_MULTIPLE,HORSE_SupportMountCleanVariables(oRider));
DeleteSkinInt(oRider,"bX3_IS_MOUNTED");
} // HorseInstantDismount()
void HorseInstantMount(object oRider,int nTail,int bJoust=FALSE,string sResRef="")
{ // PURPOSE: Instantly mount oRider
string sRR=sResRef;
if (GetStringLength(sRR)<1) sRR="x3_horse001";
SetSkinInt(oRider,"nX3_StoredFootstep",GetFootstepType(oRider));
SetSkinInt(oRider,"nX3_HorseRiderPhenotype",GetPhenoType(oRider));
SetSkinInt(oRider,"nX3_HorseRiderAppearance",GetAppearanceType(oRider));
SetLocalInt(oRider,"nX3_RiderHP",GetCurrentHitPoints(oRider));
SetSkinInt(oRider,"nX3_HorseRiderTail",GetCreatureTailType(oRider));
SetCreatureAppearanceType(oRider,HORSE_SupportGetMountedAppearance(oRider));
HorseSetPhenotype(oRider,bJoust);
SetCreatureTailType(nTail,oRider);
SetFootstepType(HORSE_FOOTSTEP_SOUND,oRider);
SetSkinString(oRider,"sX3_HorseResRef",sRR);
HORSE_SupportIncreaseSpeed(oRider,OBJECT_INVALID);
HORSE_SupportAdjustMountedArcheryPenalty(oRider);
SetSkinInt(oRider,"bX3_IS_MOUNTED",TRUE);
DelayCommand(0.5,HORSE_SupportApplyMountedSkillDecreases(oRider));
if (GetLocalInt(GetModule(),"X3_ENABLE_MOUNT_DB")) SetLocalInt(oRider,"bX3_STORE_MOUNT_INFO",TRUE);
SetLocalInt(oRider,"bX3_HORSE_MODIFIERS",TRUE);
} // HorseInstantMount()
object HorseGetPaladinMount(object oRider)
{ // PURPOSE: Return the paladin mount for oRider
string sS;
object oNPC;
int nN=1;
if (GetIsObjectValid(oRider)&&GetObjectType(oRider)==OBJECT_TYPE_CREATURE)
{ // valid parameter
sS=GetResRef(oRider);
if (GetStringLeft(sS,GetStringLength(HORSE_PALADIN_PREFIX))==HORSE_PALADIN_PREFIX) return oRider; // oRider IS a paladin mount
sS=GetSkinString(oRider,"sX3_HorseResRef");
if (GetStringLeft(sS,GetStringLength(HORSE_PALADIN_PREFIX))==HORSE_PALADIN_PREFIX) return oRider; // oRider is riding a paladin mount
oNPC=GetLocalObject(oRider,"oX3PaladinMount");
if (GetIsObjectValid(oNPC)&&GetStringLeft(GetResRef(oNPC),GetStringLength(HORSE_PALADIN_PREFIX))==HORSE_PALADIN_PREFIX) return oNPC;
oNPC=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oRider,nN);
while(GetIsObjectValid(oNPC))
{ // check henchmen
sS=GetResRef(oNPC);
if (GetStringLeft(sS,GetStringLength(HORSE_PALADIN_PREFIX))==HORSE_PALADIN_PREFIX) return oNPC;
nN++;
oNPC=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oRider,nN);
} // check henchmen
return GetLocalObject(oRider,"oX3PaladinMount");
} // valid parameter
return OBJECT_INVALID;
} // HorseGetPaladinMount()
object HorseSummonPaladinMount(int bPHBDuration=FALSE)
{ // PURPOSE: Summon Paladin Mount
object oSummoner=OBJECT_SELF;
object oMount;
location lLoc;
int nLevel=GetLevelByClass(CLASS_TYPE_PALADIN,oSummoner);
int nDespawnTime;
int nCurrentTime;
int nMountNum=1;
string sResRef=HORSE_PALADIN_PREFIX;
effect eVFX;
oMount=HorseGetPaladinMount(oSummoner);
if (!GetIsObjectValid(oMount)&&nLevel>4&&GetObjectType(oSummoner)==OBJECT_TYPE_CREATURE)
{ // okay to summon - only one paladin mount at a time
if ((GetIsPC(oSummoner)||GetIsDM(oSummoner))&&!GetHasFeat(FEAT_HORSE_MENU,oSummoner)) HorseAddHorseMenu(oSummoner);
if (nLevel>7&&nLevel<11) nMountNum=2;
else if (nLevel>10&&nLevel<15) nMountNum=3;
else if (nLevel>14&&nLevel<25) nMountNum=4;
else if (nLevel>24&&nLevel<30) nMountNum=5;
else if (nLevel>29&&nLevel<35) nMountNum=6;
else if (nLevel>34&&nLevel<40) nMountNum=7;
else if (nLevel>39) nMountNum=8;
lLoc=HORSE_SupportGetMountLocation(oSummoner,oSummoner);
oMount=HorseCreateHorse(sResRef+IntToString(nMountNum),lLoc,oSummoner);
if (!GetIsObjectValid(oMount)) oMount=HorseCreateHorse(sResRef+IntToString(nMountNum),GetLocation(oSummoner),oSummoner);
if (GetIsObjectValid(oMount))
{ // oMount created
eVFX=EffectVisualEffect(VFX_DUR_CUTSCENE_INVISIBILITY);
ApplyEffectToObject(DURATION_TYPE_TEMPORARY,eVFX,oMount,3.0);
eVFX=EffectVisualEffect(VFX_FNF_SUMMON_MONSTER_2);
if (nMountNum>3) eVFX=EffectVisualEffect(VFX_FNF_SUMMON_MONSTER_3);
ApplyEffectAtLocation(DURATION_TYPE_INSTANT,eVFX,GetLocation(oMount));
if (bPHBDuration)
{ // Players Handbook 3.5 edition durations
nCurrentTime=HORSE_SupportAbsoluteMinute();
nDespawnTime=(2*nLevel*60)+nCurrentTime;
SetLocalInt(oSummoner,"nX3_PALADIN_UNSUMMON",nDespawnTime);
} // Players Handbook 3.5 edition durations
else
{ // 24 hour - popular bioware
nCurrentTime=HORSE_SupportAbsoluteMinute();
nDespawnTime=nCurrentTime+(60*24);
SetLocalInt(oSummoner,"nX3_PALADIN_UNSUMMON",nDespawnTime);
} // 24 hour - popular bioware
if (GetLocalInt(GetModule(),"X3_ENABLE_MOUNT_DB")&&GetIsPC(oSummoner)) SetLocalInt(oSummoner,"bX3_STORE_MOUNT_INFO",TRUE);
SetLocalObject(oSummoner,"oX3PaladinMount",oMount);
} // oMount created
} // okay to summon - only one paladin mount at a time
else { oMount=OBJECT_INVALID; }
return oMount;
} // HorseSummonPaladinMount()
object HorseSummonPhantomSteed(int nCasterLvl, int nDuration)
{ // PURPOSE: Summon Phantom Steed
object oSummoner=OBJECT_SELF;
object oMount;
location lLoc;
int nDespawnTime;
int nCurrentTime;
int nMountNum=1;
string sResRef=HORSE_PALADIN_PREFIX;
effect eVFX;
oMount=HorseGetPaladinMount(oSummoner);
if (!GetIsObjectValid(oMount) && GetObjectType(oSummoner) == OBJECT_TYPE_CREATURE)
{ // okay to summon - only one mount at a time
if ((GetIsPC(oSummoner) || GetIsDM(oSummoner))&&!GetHasFeat(FEAT_HORSE_MENU,oSummoner)) HorseAddHorseMenu(oSummoner);
if (nCasterLvl < 11) nMountNum = 2;
else if (nCasterLvl > 10 && nCasterLvl < 15) nMountNum = 3;
else if (nCasterLvl > 14 && nCasterLvl < 25) nMountNum = 4;
else if (nCasterLvl > 24 && nCasterLvl < 30) nMountNum = 5;
else if (nCasterLvl > 29 && nCasterLvl < 35) nMountNum = 6;
else if (nCasterLvl > 34 && nCasterLvl < 40) nMountNum = 7;
else if (nCasterLvl > 39) nMountNum = 8;
lLoc=HORSE_SupportGetMountLocation(oSummoner,oSummoner);
oMount=HorseCreateHorse(sResRef+IntToString(nMountNum),lLoc,oSummoner);
if (!GetIsObjectValid(oMount)) oMount=HorseCreateHorse(sResRef+IntToString(nMountNum),GetLocation(oSummoner),oSummoner);
if (GetIsObjectValid(oMount))
{ // oMount created
eVFX=EffectVisualEffect(VFX_DUR_CUTSCENE_INVISIBILITY);
ApplyEffectToObject(DURATION_TYPE_TEMPORARY,eVFX,oMount,3.0);
eVFX=EffectVisualEffect(VFX_FNF_SUMMON_MONSTER_2);
if (nMountNum>3) eVFX=EffectVisualEffect(VFX_FNF_SUMMON_MONSTER_3);
ApplyEffectAtLocation(DURATION_TYPE_INSTANT,eVFX,GetLocation(oMount));
nCurrentTime=HORSE_SupportAbsoluteMinute();
nDespawnTime=(nDuration*60)+nCurrentTime;
SetLocalInt(oSummoner,"nX3_PALADIN_UNSUMMON",nDespawnTime);
if (GetLocalInt(GetModule(),"X3_ENABLE_MOUNT_DB")&&GetIsPC(oSummoner)) SetLocalInt(oSummoner,"bX3_STORE_MOUNT_INFO",TRUE);
SetLocalObject(oSummoner,"oX3PaladinMount",oMount);
} // oMount created
} // okay to summon - only one paladin mount at a time
else { oMount=OBJECT_INVALID; }
return oMount;
} // HorseSummonPaladinMount()
void HorseUnsummonPaladinMount()
{ // PURPOSE: Unsummon Paladin Mount
object oPaladin=OBJECT_SELF;
object oMount=HorseGetPaladinMount(oPaladin);
effect eVFX;
if (!GetIsObjectValid(oMount)) oMount=GetLocalObject(oPaladin,"oX3PaladinMount");
if (GetIsObjectValid(oMount)&&oMount!=oPaladin)
{ // Paladin Mount exists
if (GetLocalInt(GetModule(),"X3_ENABLE_MOUNT_DB")&&GetIsPC(oPaladin)) SetLocalInt(oPaladin,"bX3_STORE_MOUNT_INFO",TRUE);
if (oMount==oPaladin)
{ // Mounted - must dismount first
ClearAllActions(TRUE);
oMount=HorseDismount(FALSE,TRUE);
DelayCommand(3.0,HorseUnsummonPaladinMount());
} // Mounted - must dismount first
else
{ // is the mount
if (GetIsPC(oPaladin))
{
SendMessageToPCByStrRef(oPaladin,111985);
}
DeleteLocalInt(oPaladin,"nX3_PALADIN_UNSUMMON");
eVFX=EffectVisualEffect(VFX_IMP_UNSUMMON);
ApplyEffectAtLocation(DURATION_TYPE_INSTANT,eVFX,GetLocation(oMount));
DestroyObject(oMount);
} // is the mount
} // Paladin Mount exists
else
{ // perhaps the command is being called by the mount itself
if (GetStringLeft(GetResRef(oPaladin),GetStringLength(HORSE_PALADIN_PREFIX))==HORSE_PALADIN_PREFIX)
{ // is a paladin mount despawning itself
oMount=oPaladin;
eVFX=EffectVisualEffect(VFX_IMP_UNSUMMON);
ApplyEffectAtLocation(DURATION_TYPE_INSTANT,eVFX,GetLocation(oMount));
DestroyObject(oMount);
} // is a paladin mount despawning itself
} // perhaps the command is being called by the mount itself
} // HorseUnsummonPaladinMount()
int HorseGetIsAMount(object oTarget)
{ // PURPOSE: Return TRUE if oTarget is a mountable creature
int nApp;
int nTail;
if (GetSkinInt(oTarget,"bX3_IS_MOUNTED")) return FALSE;
if (GetLocalInt(oTarget,"bX3_IS_MOUNT")) return TRUE;
else if (GetLocalInt(oTarget,"bX3_IS_RIDER")) return FALSE;
else if (GetStringLength(GetLocalString(oTarget,"X3_HORSE_PREMOUNT_SCRIPT"))>0) return TRUE;
else if (GetStringLength(GetLocalString(oTarget,"X3_HORSE_PREDISMOUNT_SCRIPT"))>0) return TRUE;
else if (GetStringLength(GetLocalString(oTarget,"X3_HORSE_POSTDISMOUNT_SCRIPT"))>0) return TRUE;
else if (GetStringLength(GetLocalString(oTarget,"X3_HORSE_POSTMOUNT_SCRIPT"))>0) return TRUE;
else if (GetStringLength(GetLocalString(oTarget,"X3_HORSE_OWNER_TAG"))>0) return TRUE;
else if (GetLocalInt(oTarget,"X3_HORSE_NULL_APPEARANCE")>0) return TRUE;
//else if (GetSkinInt(oTarget,"X3_HORSE_APPEARANCE")>0) return TRUE;
else if (GetLocalInt(oTarget,"X3_HORSE_TAIL")>0) return TRUE;
nApp=GetAppearanceType(oTarget);
nTail=GetCreatureTailType(oTarget);
if (nApp>=HORSE_APPEARANCE_OFFSET&&nApp<=HORSE_APPEARANCE_OFFSET+HORSE_NUMBER_OF_HORSES) return TRUE;
if (nApp==HORSE_NULL_RACE_GNOME||nApp==HORSE_NULL_RACE_ELF||nApp==HORSE_NULL_RACE_GNOME||nApp==HORSE_NULL_RACE_HALFELF||
nApp==HORSE_NULL_RACE_HALFLING||nApp==HORSE_NULL_RACE_HALFORC||nApp==HORSE_NULL_RACE_HUMAN)
{ // might be a tail always based mount
if (nTail>=HORSE_TAIL_OFFSET&&nTail<=HORSE_TAIL_OFFSET+HORSE_NUMBER_OF_HORSES&&nTail!=14) return TRUE;
} // might be a tail always based mount
return FALSE;
} // HorseGetIsAMount()
void HorseStoreInventory(object oCreature,object oRider=OBJECT_INVALID)
{ // PURPOSE: To store inventory being carried by oCreature
object oWP=GetWaypointByTag("X3_HORSE_INVENTORY_STORAGE");
object oOwner=GetMaster(oCreature);
object oOwnersMaster=GetMaster(oOwner);
object oItem;
object oChest;
object oCont;
int nC;
int nN;
string sR;
string sT;
int nST;
int nCH;
string sID=GetTag(oCreature); // ID to uniquely identify for inventory storage
string sDB="X3SADDLEBAG"+GetTag(GetModule());
if (GetStringLength(GetLocalString(GetModule(),"X3_SADDLEBAG_DATABASE"))>0) sDB=GetLocalString(GetModule(),"X3_SADDLEBAG_DATABASE");
if (GetStringLength(sID)>6) sID=GetStringLeft(sID,6);
sID=sID+GetStringRight(GetResRef(oCreature),2);
if (oOwner!=oCreature&&GetIsObjectValid(oOwner))
{ // owner specified
if (!GetIsPC(oOwner))
{ // henchman owner
sT=GetTag(oOwner);
if (GetStringLength(sT)>8) sT=GetStringLeft(sT,6)+GetStringRight(sT,2);
sID=sID+sT;
if (GetIsObjectValid(oOwnersMaster)&&oOwner!=oOwnersMaster&&GetIsPC(oOwnersMaster))
{ // PC
sID=sID+GetPCPublicCDKey(oOwnersMaster)+GetStringLeft(GetName(oOwnersMaster),4);
} // PC
} // henchman owner
else
{ // PC owner
sID=sID+GetPCPublicCDKey(oOwner)+GetStringLeft(GetName(oOwner),4);
} // PC owner
} // owner specified
if (GetIsObjectValid(oWP))
{ // do not use database
// I am using a placeable that has inventory but, by default does not
// have scripts that produce extra inventory.
oChest=CreateObject(OBJECT_TYPE_PLACEABLE,"x3_plc_jars001",GetLocation(oWP),FALSE,"X3_"+sID);
SetName(oChest,GetName(oCreature)+"'s Inventory");
SetLocalObject(oCreature,"oX3_Saddlebags",oChest);
HORSE_SupportTransferInventory(oCreature,oChest,GetLocation(oChest));
} // do not use database
else
{ // use database
nC=0;
oItem=GetFirstItemInInventory(oCreature);
while(GetIsObjectValid(oItem))
{ // store inventory
nC++;
sR=GetResRef(oItem);
sT=GetTag(oItem);
nST=GetItemStackSize(oItem);
nCH=GetItemCharges(oItem);
SetCampaignString(sDB,"sR"+sID+IntToString(nC),sR);
SetCampaignString(sDB,"sT"+sID+IntToString(nC),sT);
SetCampaignInt(sDB,"nS"+sID+IntToString(nC),nST);
SetCampaignInt(sDB,"nC"+sID+IntToString(nC),nCH);
DestroyObject(oItem,0.1);
oItem=GetNextItemInInventory(oCreature);
} // store inventory
SetCampaignInt(sDB,"nCO_"+sID,nC);
if (GetIsObjectValid(oRider)) SetLocalString(oRider,"sDB_Inv",sID);
} // use database
} // HorseStoreInventory()
void HorseRestoreInventory(object oCreature,int bDrop=FALSE)
{ // PURPOSE: To restore inventory that was stored while mounted
object oWP=GetWaypointByTag("X3_HORSE_INVENTORY_STORAGE");
object oOwner=GetMaster(oCreature);
object oOwnersMaster=GetMaster(oOwner);
object oItem;
object oChest;
int nC;
int nN;
string sR;
string sT;
int nST;
int nCH;
string sID=GetTag(oCreature);// ID to uniquely identify for inventory storage
string sDB="X3SADDLEBAG"+GetTag(GetModule());
if (GetStringLength(GetLocalString(GetModule(),"X3_SADDLEBAG_DATABASE"))>0) sDB=GetLocalString(GetModule(),"X3_SADDLEBAG_DATABASE");
if (GetStringLength(sID)>6) sID=GetStringLeft(sID,6);
sID=sID+GetStringRight(GetResRef(oCreature),2);
if (oOwner!=oCreature&&GetIsObjectValid(oOwner))
{ // owner specified
if (!GetIsPC(oOwner))
{ // henchman owner
sT=GetTag(oOwner);
if (GetStringLength(sT)>8) sT=GetStringLeft(sT,6)+GetStringRight(sT,2);
sID=sID+sT;
if (GetIsObjectValid(oOwnersMaster)&&oOwner!=oOwnersMaster&&GetIsPC(oOwnersMaster))
{ // PC
sID=sID+GetPCPublicCDKey(oOwnersMaster)+GetStringLeft(GetName(oOwnersMaster),4);
} // PC
} // henchman owner
else
{ // PC owner
sID=sID+GetPCPublicCDKey(oOwner)+GetStringLeft(GetName(oOwner),4);
} // PC owner
} // owner specified
if (GetIsObjectValid(oWP))
{ // do not use database
oChest=GetObjectByTag("X3_"+sID);
if (!GetIsObjectValid(oChest)) oChest=GetLocalObject(oOwner,"oX3_Saddlebags");
if (GetIsObjectValid(oChest))
{ // chest found
HORSE_SupportTransferInventory(oChest,oCreature,GetLocation(oCreature),TRUE);
} // chest found
else
{ // error
PrintString("ERROR: x3_inc_horse 'HorseRestoreInventory()' Could not find chest 'X3_"+sID+"'!");
} // error
} // do not use database
else
{ // use database
nC=GetCampaignInt(sDB,"nCO_"+sID);
while(nC>0)
{ // restore inventory
sR=GetCampaignString(sDB,"sR"+sID+IntToString(nC));
sT=GetCampaignString(sDB,"sT"+sID+IntToString(nC));
nST=GetCampaignInt(sDB,"nS"+sID+IntToString(nC));
nCH=GetCampaignInt(sDB,"nC"+sID+IntToString(nC));
DeleteCampaignVariable(sDB,"sR"+sID+IntToString(nC));
DeleteCampaignVariable(sDB,"sT"+sID+IntToString(nC));
DeleteCampaignVariable(sDB,"nS"+sID+IntToString(nC));
DeleteCampaignVariable(sDB,"nC"+sID+IntToString(nC));
if (!bDrop) oItem=CreateItemOnObject(sR,oCreature,nST,sT);
else
{ // drop
oItem=CreateObject(OBJECT_TYPE_ITEM,sR,GetLocation(oCreature),FALSE,sT);
if (GetItemStackSize(oItem)!=nST&&nST!=0) SetItemStackSize(oItem,nST);
} // drop
if (nCH>0) SetItemCharges(oItem,nCH);
nC--;
} // restore inventory
DeleteCampaignVariable(sDB,"nCO_"+sID);
} // use database
} // HorseRestoreInventory()
void HorseChangeToDefault(object oCreature)
{ // PURPOSE: HorseChangeToDefault
int nRace=GetRacialType(oCreature);
int nDefApp=StringToInt(Get2DAString("racialtypes","Appearance",nRace));
int nDefPhen;
int nPheno=GetPhenoType(oCreature);
if (nPheno==HORSE_PHENOTYPE_JOUSTING_L||nPheno==HORSE_PHENOTYPE_MOUNTED_L) nDefPhen=PHENOTYPE_BIG;
else { nDefPhen=PHENOTYPE_NORMAL; }
SetCreatureAppearanceType(oCreature,nDefApp);
SetPhenoType(nDefPhen,oCreature);
SetCreatureTailType(CREATURE_TAIL_TYPE_NONE,oCreature);
SetFootstepType(FOOTSTEP_TYPE_NORMAL,oCreature);
DeleteLocalInt(oCreature,"bX3_HORSE_MODIFIERS");
DelayCommand(0.1,HORSE_SupportOriginalSpeed(oCreature));
DelayCommand(0.2,HORSE_SupportRemoveMountedSkillDecreases(oCreature));
DelayCommand(0.2,HORSE_SupportAdjustMountedArcheryPenalty(oCreature));
DelayCommand(0.4,HORSE_SupportRemoveACBonus(oCreature));
DelayCommand(0.4,HORSE_SupportRemoveHPBonus(oCreature));
DelayCommand(1.0,HORSE_SupportCleanVariables(oCreature));
if (GetLocalInt(GetModule(),"X3_ENABLE_MOUNT_DB")&&GetIsPC(oCreature)) SetLocalInt(oCreature,"bX3_STORE_MOUNT_INFO",TRUE);
} // HorseChangeToDefault()
void HorseIfNotDefaultAppearanceChange(object oCreature)
{ // PURPOSE: See if not default appearance
int nRace=GetRacialType(oCreature);
int nDefApp=StringToInt(Get2DAString("racialtypes","Appearance",nRace));
if (GetAppearanceType(oCreature)!=nDefApp) HorseChangeToDefault(oCreature);
} // HorseIfNotDefaultAppearanceChange()
object HorseGetMyHorse(object oRider)
{ // PURPOSE: Return active horse
object oRet=GetLocalObject(oRider,"oX3_TempHorse");
if (GetIsObjectValid(oRet)) return oRet;
return GetLocalObject(oRider,"oAssignedHorse");
} // HorseGetMyHorse()
int HorseGetHasAHorse(object oRider)
{ // PURPOSE: Return if oRider has a horse
int nN=1;
object oHench=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oRider,nN);
while(GetIsObjectValid(oHench))
{ // traverse henchmen
if (HorseGetIsAMount(oHench)) return TRUE;
nN++;
oHench=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oRider,nN);
} // traverse henchmen
return FALSE;
} // HorseGetHasAHorse()
object HorseGetHorse(object oRider,int nN=1)
{ // PURPOSE: Return the nNth horse of oRider
int nC=1;
int nH=0;
object oHench=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oRider,nC);
while(GetIsObjectValid(oHench))
{ // traverse henchmen
if (HorseGetIsAMount(oHench))
{ // is a horse
nH++;
if (nH==nN) return oHench;
} // is a horse
nC++;
oHench=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oRider,nC);
} // traverse henchmen
return OBJECT_INVALID;
} // HorseGetHorse()
void HorseRestoreHenchmenLocations(object oPC)
{ // PURPOSE: Restore the locations of the henchmen of henchmen
object oHench;
int nN;
int nC;
object oAssoc;
int bNoMounts;
int bNoMounting;
int bRunAgain=FALSE;
float fDelay=0.2;
object oArea;
if (GetLocalInt(GetModule(),"X3_RESTORE_HENCHMEN_LOCATIONS"))
{ // restore location of henchmen belonging to henchmen
nN=1;
oHench=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oPC,nN);
while(GetIsObjectValid(oHench))
{ // hench
oArea=GetArea(oHench);
bNoMounts=FALSE;
bNoMounting=FALSE;
if (!GetLocalInt(oArea,"X3_MOUNT_OK_EXCEPTION"))
{ // find out if horses ok
if (GetLocalInt(GetModule(),"X3_MOUNTS_EXTERNAL_ONLY")&&GetIsAreaInterior(oArea)) bNoMounts=TRUE;
else if (GetLocalInt(GetModule(),"X3_MOUNTS_NO_UNDERGROUND")&&!GetIsAreaAboveGround(oArea)) bNoMounts=TRUE;
} // find out if horses ok
if (GetLocalInt(oArea,"X3_NO_MOUNTING")||GetLocalInt(oArea,"X3_NO_HORSES")||bNoMounts) bNoMounting=TRUE;
nC=1;
oAssoc=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oHench,nC);
while(GetIsObjectValid(oAssoc))
{ // check each associate
if (HorseGetIsMounted(oAssoc)&&(bNoMounts||bNoMounting))
{ // dismount
bRunAgain=TRUE;
DelayCommand(fDelay,AssignCommand(oAssoc,HORSE_SupportDismountWrapper(FALSE,TRUE)));
fDelay=fDelay+0.2;
} // dismount
else if (HorseGetIsAMount(oAssoc)&&bNoMounts)
{ // no mounts
DelayCommand(fDelay,RemoveHenchman(oHench,oAssoc));
DelayCommand(fDelay+0.01,SetAssociateState(NW_ASC_MODE_STAND_GROUND,TRUE,oAssoc));
fDelay=fDelay+0.2;
} // no mounts
else if (GetArea(oAssoc)!=oArea)
{ // jump
AssignCommand(oAssoc,ClearAllActions(TRUE));
AssignCommand(oAssoc,ActionJumpToObject(oHench));
} // jump
nC++;
oAssoc=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oHench,nC);
} // check each associate
nN++;
oHench=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oPC,nN);
} // hench
nN=1;
oHench=GetNearestCreature(CREATURE_TYPE_PLAYER_CHAR,PLAYER_CHAR_NOT_PC,oPC,nN);
while(GetIsObjectValid(oHench))
{ // test non-henchmen
if (!GetIsObjectValid(GetMaster(oHench))||GetMaster(oHench)==oHench)
{ // test associates
oArea=GetArea(oHench);
bNoMounts=FALSE;
bNoMounting=FALSE;
if (!GetLocalInt(oArea,"X3_MOUNT_OK_EXCEPTION"))
{ // find out if horses ok
if (GetLocalInt(GetModule(),"X3_MOUNTS_EXTERNAL_ONLY")&&GetIsAreaInterior(oArea)) bNoMounts=TRUE;
else if (GetLocalInt(GetModule(),"X3_MOUNTS_NO_UNDERGROUND")&&!GetIsAreaAboveGround(oArea)) bNoMounts=TRUE;
} // find out if horses ok
if (GetLocalInt(oArea,"X3_NO_MOUNTING")||GetLocalInt(oArea,"X3_NO_HORSES")||bNoMounts) bNoMounting=TRUE;
nC=1;
oAssoc=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oHench,nC);
while(GetIsObjectValid(oAssoc))
{ // check each associate
if (HorseGetIsMounted(oAssoc)&&(bNoMounts||bNoMounting))
{ // dismount
bRunAgain=TRUE;
DelayCommand(fDelay,AssignCommand(oAssoc,HORSE_SupportDismountWrapper(FALSE,TRUE)));
fDelay=fDelay+0.2;
} // dismount
else if (HorseGetIsAMount(oAssoc)&&bNoMounts)
{ // no mounts
DelayCommand(fDelay,RemoveHenchman(oHench,oAssoc));
DelayCommand(fDelay+0.01,SetAssociateState(NW_ASC_MODE_STAND_GROUND,TRUE,oAssoc));
fDelay=fDelay+0.2;
} // no mounts
else if (GetArea(oAssoc)!=oArea)
{ // jump
AssignCommand(oAssoc,ClearAllActions(TRUE));
AssignCommand(oAssoc,ActionJumpToObject(oHench));
} // jump
nC++;
oAssoc=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oHench,nC);
} // check each associate
} // test associates
nN++;
oHench=GetNearestCreature(CREATURE_TYPE_PLAYER_CHAR,PLAYER_CHAR_NOT_PC,oPC,nN);
} // test non-henchmen
if (bRunAgain) DelayCommand(5.0+fDelay,HorseRestoreHenchmenLocations(oPC));
} // restore location of henchmen belonging to henchmen
} // HorseRestoreHenchmenLocations()
////////////////////////////////////////////////////////////////////////////////
/////////////////////////////// TRANSITIONS ////////////////////////////////
////////////////////////////////////////////////////////////////////////////////
void HorseHitchHorses(object oHitch,object oClicker,location lPreJump)
{ // PURPOSE: Hitch all associates to clicker
object oOb;
object oHench;
int nN=1;
int nNN;
int bNotAllowed=FALSE;
float fX3_MOUNT_MULTIPLE=GetLocalFloat(GetArea(oClicker),"fX3_MOUNT_MULTIPLE");
float fX3_DISMOUNT_MULTIPLE=GetLocalFloat(GetArea(oClicker),"fX3_DISMOUNT_MULTIPLE");
if (GetLocalFloat(oClicker,"fX3_MOUNT_MULTIPLE")>fX3_MOUNT_MULTIPLE) fX3_MOUNT_MULTIPLE=GetLocalFloat(oClicker,"fX3_MOUNT_MULTIPLE");
if (fX3_MOUNT_MULTIPLE<=0.0) fX3_MOUNT_MULTIPLE=1.0;
if (GetLocalFloat(oClicker,"fX3_DISMOUNT_MULTIPLE")>0.0) fX3_DISMOUNT_MULTIPLE=GetLocalFloat(oClicker,"fX3_DISMOUNT_MULTIPLE");
if (fX3_DISMOUNT_MULTIPLE>0.0) fX3_MOUNT_MULTIPLE=fX3_DISMOUNT_MULTIPLE; // use dismount multiple instead of mount multiple
float fDelay=0.2*fX3_MOUNT_MULTIPLE;
oOb=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oClicker,nN);
while(GetIsObjectValid(oOb))
{ // traverse henchmen
if (HorseGetIsAMount(oOb)&&(!GetLocalInt(oOb,"X3_DOING_HORSE_ACTION")))
{ // is a mount
bNotAllowed=TRUE;
if (GetStringLeft(GetResRef(oOb),GetStringLength(HORSE_PALADIN_PREFIX))==HORSE_PALADIN_PREFIX) SetLocalObject(oClicker,"oX3PaladinMount",oOb);
if (GetIsObjectValid(oHitch)&&GetDistanceBetween(oClicker,oHitch)<15.0)
{ // jump to hitch
DelayCommand(0.1,RemoveHenchman(oClicker,oOb));
AssignCommand(oOb,ClearAllActions(TRUE));
//AssignCommand(oOb,ActionWait(X3_ACTION_DELAY/2*fX3_MOUNT_MULTIPLE));
AssignCommand(oOb,ActionJumpToObject(oHitch));
AssignCommand(oOb,ActionDoCommand(SetFacingPoint(GetPosition(oHitch))));
SetAssociateState(NW_ASC_MODE_STAND_GROUND,TRUE,oOb);
} // jump to hitch
else
{ // stand where you are and make way
DelayCommand(0.1,RemoveHenchman(oClicker,oOb));
SetAssociateState(NW_ASC_MODE_STAND_GROUND,TRUE,oOb);
AssignCommand(oOb,ClearAllActions(TRUE));
if (GetDistanceBetween(oOb,oClicker)<3.0) DelayCommand(1.0*fX3_MOUNT_MULTIPLE,AssignCommand(oOb,ActionMoveAwayFromLocation(lPreJump,FALSE,3.0+IntToFloat(Random(15))/10.0)));
else if (GetDistanceBetween(oOb,oClicker)>4.0) DelayCommand(1.0*fX3_MOUNT_MULTIPLE,AssignCommand(oOb,ActionMoveToLocation(GetBehindLocation(oClicker))));
} // stand where you are and make way
} // is a mount
else
{ // check for mounts for this henchman
oHench=oOb;
nNN=1;
oOb=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oHench,nNN);
while(GetIsObjectValid(oOb))
{ // traverse henchmen
if (HorseGetIsAMount(oOb)&&(!GetLocalInt(oOb,"X3_DOING_HORSE_ACTION")))
{ // is a mount
bNotAllowed=TRUE;
if (GetStringLeft(GetResRef(oOb),GetStringLength(HORSE_PALADIN_PREFIX))==HORSE_PALADIN_PREFIX) SetLocalObject(oHench,"oX3PaladinMount",oOb);
if (GetIsObjectValid(oHitch)&&GetDistanceBetween(oClicker,oHitch)<15.0)
{ // jump to hitch
DelayCommand(0.1,RemoveHenchman(oHench,oOb));
AssignCommand(oOb,ClearAllActions(TRUE));
//AssignCommand(oOb,ActionWait(X3_ACTION_DELAY/2*fX3_MOUNT_MULTIPLE));
AssignCommand(oOb,ActionJumpToObject(oHitch));
AssignCommand(oOb,ActionDoCommand(SetFacingPoint(GetPosition(oHitch))));
SetAssociateState(NW_ASC_MODE_STAND_GROUND,TRUE,oOb);
} // jump to hitch
else
{ // stand where you are and make way
DelayCommand(0.1,RemoveHenchman(oHench,oOb));
AssignCommand(oOb,ClearAllActions(TRUE));
SetAssociateState(NW_ASC_MODE_STAND_GROUND,TRUE,oOb);
fDelay+=0.2*fX3_MOUNT_MULTIPLE;
//if (GetDistanceBetween(oOb,oClicker)<4.0) DelayCommand(fDelay,AssignCommand(oOb,ActionMoveAwayFromLocation(lPreJump,FALSE,4.0+IntToFloat(Random(15))/10.0)));
if (GetDistanceBetween(oOb,OBJECT_SELF)<4.0) DelayCommand(fDelay,AssignCommand(oOb,ActionMoveAwayFromLocation(lPreJump,FALSE,4.0+IntToFloat(Random(15))/10.0)));
//else if (GetDistanceBetween(oOb,oClicker)>5.0) DelayCommand(fDelay,AssignCommand(oOb,ActionMoveToLocation(GetBehindLocation(oClicker))));
else if (GetDistanceBetween(oOb,OBJECT_SELF)>5.0) DelayCommand(fDelay,AssignCommand(oOb,ActionMoveToObject(OBJECT_SELF,FALSE,4.0+IntToFloat(Random(15))/10.0)));
//SendMessageToPC(oClicker, FloatToString(GetDistanceBetween(oOb,OBJECT_SELF)));
} // stand where you are and make way
} // is a mount
nNN++;
oOb=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oHench,nNN);
} // traverse henchmen
} // check for mounts for this henchman
nN++;
oOb=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oClicker,nN);
} // traverse henchmen
if (bNotAllowed) SendMessageToPCByStrRef(oClicker,111990);
} // HorseHitchHorses()
void HorseForceJump(object oJumper,object oDestination,float fRange=2.0,int nTimeOut=10)
{ // PURPOSE: Make sure jump
//SendMessageToPC(oJumper,"nw_g0_transition:ForceJump("+IntToString(nTimeOut)+")");
if (nTimeOut>0&&(GetArea(oJumper)!=GetArea(oDestination)||GetDistanceBetween(oJumper,oDestination)>fRange))
{ // jump
AssignCommand(oJumper,ClearAllActions(TRUE));
AssignCommand(oJumper,ActionJumpToObject(oDestination));
DelayCommand(1.0,HorseForceJump(oJumper,oDestination,fRange,nTimeOut-1));
} // jump
} // HorseForceJump()
void HorseMoveAssociates(object oMaster)
{ // PURPOSE: Give the PC some breathing room
int nN,nNN;
float fDelay=0.2;
object oHench,oAssociate,oSummon;
nN=1;
oAssociate=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oMaster,nN);
while(GetIsObjectValid(oAssociate))
{ // move associates
if (HorseGetIsAMount(oAssociate)||HorseGetIsMounted(oAssociate))
{ // only move mounts or mounted associates
if (GetArea(oAssociate)==GetArea(oMaster)&&GetDistanceBetween(oMaster,oAssociate)<3.0)
{ // move away
AssignCommand(oAssociate,ClearAllActions());
DelayCommand(1.0,AssignCommand(oAssociate,ActionMoveAwayFromObject(oMaster,FALSE,3.0+IntToFloat(Random(10))/10.0)));
} // move away
} // only move mounts or mounted associates
else
{ // henchmen of henchman
oHench=oAssociate;
nNN=1;
oAssociate=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oHench,nNN);
while(GetIsObjectValid(oAssociate))
{ // traverse henchmen
if (HorseGetIsAMount(oAssociate)||HorseGetIsMounted(oAssociate))
{ // only move mounts or mounted associates
if (GetArea(oAssociate)==GetArea(oMaster)&&GetDistanceBetween(oMaster,oAssociate)<3.0)
{ // move away
fDelay+=0.2;
AssignCommand(oAssociate,ClearAllActions());
DelayCommand(fDelay,AssignCommand(oAssociate,ActionMoveAwayFromObject(oMaster,FALSE,3.0+IntToFloat(Random(15))/10.0)));
} // move away
} // only move mounts or mounted associates
nNN++;
oAssociate=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oHench,nNN);
oSummon=GetAssociate(ASSOCIATE_TYPE_SUMMONED,oHench,nNN);
} // traverse henchmen
nNN=1;
oSummon=GetAssociate(ASSOCIATE_TYPE_SUMMONED,oHench,nNN);
while(GetIsObjectValid(oSummon))
{ // traverse summons
if (HorseGetIsAMount(oSummon)||HorseGetIsMounted(oSummon))
{ // only move mounts or mounted associates
if (GetArea(oSummon)==GetArea(oMaster)&&GetDistanceBetween(oMaster,oSummon)<3.0)
{ // move away
fDelay+=0.2;
AssignCommand(oSummon,ClearAllActions());
DelayCommand(fDelay,AssignCommand(oSummon,ActionMoveAwayFromObject(oMaster,FALSE,3.0+IntToFloat(Random(15))/10.0)));
} // move away
} // only move mounts or mounted associates
nNN++;
oSummon=GetAssociate(ASSOCIATE_TYPE_SUMMONED,oHench,nNN);
} // traverse summons
} // henchmen of henchman
nN++;
oAssociate=GetAssociate(ASSOCIATE_TYPE_HENCHMAN,oMaster,nN);
} // move associates
nN=1;
oAssociate=GetAssociate(ASSOCIATE_TYPE_SUMMONED,oMaster,nN);
while(GetIsObjectValid(oAssociate))
{ // move associates
if (HorseGetIsAMount(oAssociate)||HorseGetIsMounted(oAssociate))
{ // only move mounts or mounted associates
if (GetArea(oAssociate)==GetArea(oMaster)&&GetDistanceBetween(oMaster,oAssociate)<3.0)
{ // move away
AssignCommand(oAssociate,ClearAllActions());
DelayCommand(1.0,AssignCommand(oAssociate,ActionMoveAwayFromObject(oMaster,FALSE,3.0+IntToFloat(Random(10))/10.0)));
} // move away
} // only move mounts or mounted associates
nN++;
oAssociate=GetAssociate(ASSOCIATE_TYPE_SUMMONED,oMaster,nN);
} // move associates
} // HorseMoveAssociates()
////////////////////////////////////////////////////////////////////////////////
///////////////////////////// DEATH HANDLING ///////////////////////////////
////////////////////////////////////////////////////////////////////////////////
void HorseDismountWrapper()
{ // Wrap
object oNPC=OBJECT_SELF;
if (HorseGetIsMounted(oNPC))
{ // make sure dismount
DelayCommand(0.1,HORSE_SupportResetUnmountedAppearance(oNPC));
DelayCommand(0.8,HORSE_SupportCleanVariables(oNPC));
DelayCommand(1.0,HORSE_SupportRemoveACBonus(oNPC));
DelayCommand(1.0,HORSE_SupportRemoveHPBonus(oNPC));
DelayCommand(1.0,HORSE_SupportRemoveMountedSkillDecreases(oNPC));
DelayCommand(1.0,HORSE_SupportAdjustMountedArcheryPenalty(oNPC));
DelayCommand(1.0,HORSE_SupportOriginalSpeed(oNPC));
DelayCommand(2.0,HorseDismountWrapper());
} // make sure dismount
} // HorseDismountWrapper()
void HorseReassign(object oHorse,object oHench,object oMaster)
{ // PURPOSE: To handle horse reassign
if (GetIsObjectValid(oMaster)&&oMaster!=oHench)
{ // reassign
HorseSetOwner(oHorse,oMaster);
DeleteLocalObject(oHench,"oAssignedHorse");
} // reassign
else
{ // free
HorseRemoveOwner(oHorse);
DeleteLocalObject(oHench,"oAssignedHorse");
} // free
} // HorseReassign()
int HorseHandleDeath()
{ // PURPOSE: Handle horses, re-assigning dying henchman's horse, handle mounted henchman's death (dismount, transfer saddlebag content to horse, re-assign horse to PC or free it)
object oDied=OBJECT_SELF;
object oHorse=HorseGetHorse(oDied);
object oMaster=GetMaster(oDied);
if (GetLocalInt(GetModule(),"X3_ENABLE_MOUNT_DB")&&GetIsObjectValid(GetMaster(oDied))) SetLocalInt(GetMaster(oDied),"bX3_STORE_MOUNT_INFO",TRUE);
if (HorseGetIsAMount(oDied))
{ // check dying horse for saddlebags and eventual re-assigning to master
if (GetLocalInt(GetModule(),"X3_HORSE_ENABLE_SADDLEBAGS")&&GetLocalInt(oDied,"bX3_HAS_SADDLEBAGS"))
{ // might have saddle bags
object oOwner=HorseGetOwner(oDied);
if (!GetIsPC(oOwner)&&GetIsObjectValid(oOwner)&&oOwner!=oDied)
{ // npc
oOwner=GetMaster(oOwner);
if (GetIsPC(oOwner))
{ // set to PC then drop
HorseSetOwner(oDied,oOwner);
} // set to PC then drop
} // npc
else if (GetIsPC(oOwner))
{ // PC
} // PC
DeleteLocalInt(oDied,"bX3_HAS_SADDLEBAGS"); // if you don't want to remove saddlebags for horse resurrection purpose, make sure that the corpse's inventory content is moved to the resurrected living horse again (if you do that, you can possibly remove the check for saddlebags including the repeated call of the death script)
DelayCommand(0.1,ExecuteScript(GetLocalString(oDied,"sX3_DEATH_SCRIPT"),oDied));
return TRUE; // no resurrection for horses by default
} // might have saddle bags
return TRUE; // horse should not respawn like other henchmen
} // check dying horse for saddlebags and eventual re-assigning to master
if (GetIsObjectValid(oHorse)) HorseReassign(oHorse,oDied,GetMaster(oDied)); // check henchman's horses for possible re-assigning to master or free them
// * Bob Minors Jan 2008: added support for X3 mounts:
// * If killed while mounted, we are dismounted. This takes about 0.5s so drop out and rerun this script again afterward
if (HorseGetIsMounted(oDied))
{ // check if creature died in mounted state
oHorse=HorseDismount(FALSE,TRUE);
SetLocalObject(oDied,"oAssignedHorse",oHorse); // Kludge
HorseReassign(oHorse,oDied,oMaster);
ApplyEffectToObject(DURATION_TYPE_TEMPORARY,EffectKnockdown(),oHorse,1.8);
DelayCommand(2.0,AssignCommand(oHorse,ActionMoveAwayFromObject(oDied,TRUE,3.0))); // drop rider and jump aside
DelayCommand(1.2,ExecuteScript(GetLocalString(oDied,"sX3_DEATH_SCRIPT"),oDied)); // give dismount and potential inventory transfer some space before rerun
//return TRUE; // this return makes the dead mounted henchman non-resurrectable, so its commented out so that they can be resurrected by default
} // check if creature died in mounted state
return FALSE;
} //HorseHandleDeath()
////////////////////////////////////////////////////////////////////////////////
/////////////////////////// PRELOAD ANIMATIONS /////////////////////////////
////////////////////////////////////////////////////////////////////////////////
void HorsePreloadAnimations(object oPC)
{ // PURPOSE: Force a preload of mount animations
int nApp=GetAppearanceType(oPC);
int nTail=GetCreatureTailType(oPC);
//SetCreatureAppearanceType(oPC,HORSE_SupportGetMountedAppearance(oPC));
if (nTail==0) nTail=14;
HORSE_SupportRestoreFromPreload(oPC,nApp,nTail);
} // HorsePreloadAnimations()
//void main(){}