PRC8/nwn/nwnprc/trunk/scripts/prc_cohort_convo.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

1082 lines
49 KiB
Plaintext

//:://////////////////////////////////////////////
//:: Conhort dynamic conversation
//:: filename
//:://////////////////////////////////////////////
/** @file
Dynamic conversation file for cohorts
Handles all the normal cohort options
plus:
item useage
feat useage
spell useage
interactions with nearby stuff
Note: This is one of the few convos where
the PC is not talking to themselves
@author Primogenitor
@date Created - 2006.07.12
*/
//:://////////////////////////////////////////////
//:://////////////////////////////////////////////
#include "inc_dynconv"
#include "prc_inc_leadersh"
#include "x0_inc_henai"
//#include "x0_i0_henchman"
//////////////////////////////////////////////////
/* Constant defintions */
//////////////////////////////////////////////////
const int STAGE_ENTRY = 0;
const int STAGE_REENTRY = 1;
const int STAGE_SPELL = 100;
const int STAGE_SPELL_TARGET = 101;
const int STAGE_FEAT = 200;
const int STAGE_FEAT_TARGET = 201;
const int STAGE_ITEM = 300;
const int STAGE_ITEM_SPELL = 301;
const int STAGE_ITEM_TARGET = 302;
const int STAGE_IDENTIFY = 400;
const int STAGE_IDENTIFY_YES = 401;
const int STAGE_IDENTIFY_NO = 402;
const int STAGE_TACTICS = 500;
const int STAGE_TACTICS_EQUIP = 501;
const int STAGE_TACTICS_DEFEND = 502;
const int STAGE_TACTICS_ATTACK = 503;
const int STAGE_TACTICS_DISTANCE = 504;
const int STAGE_TACTICS_DISTANCE_CLOSE = 541;
const int STAGE_TACTICS_DISTANCE_MEDIUM = 542;
const int STAGE_TACTICS_DISTANCE_LONG = 542;
const int STAGE_TACTICS_LOCK_HELP = 505;
const int STAGE_TACTICS_LOCK_NOHELP = 506;
const int STAGE_TACTICS_STEALTH_ALWAYS = 507;
const int STAGE_TACTICS_STEALTH_COMBAT = 507;
const int STAGE_TACTICS_STEALTH_NEVER = 508;
const int STAGE_LEAVE = 600;
const int STAGE_LEAVE_YES = 601;
const int STAGE_LEAVE_NO = 602;
//////////////////////////////////////////////////
/* Aid functions */
//////////////////////////////////////////////////
void AddUseableFeats(int nMin, int nMax, object oPC, object oCohort)
{
int i;
for(i=nMin;i<nMax;i++)
{
if(GetHasFeat(i, oCohort))
{
//test if its useable
int nSpellID = StringToInt(Get2DACache("feat", "SPELLID", i));
//0 is Aid so this is a safe comparison
if(nSpellID)
{
//test if it has a sucessor
string sSucessor = Get2DACache("feat", "SUCCESSOR", i);
if(sSucessor == ""
|| (sSucessor != ""
&& !GetHasFeat(StringToInt(sSucessor), oCohort)))
{
string sName = GetStringByStrRef(StringToInt(Get2DACache("feat", "FEAT", i)));
//test for subradials
if(Get2DACache("spells", "SubRadSpell1", nSpellID) != "")
{
int j;
for(j=1;j<=5;j++)
{
string sSubName;
int nSubSpellID = StringToInt(Get2DACache("spells", "SubRadSpell"+IntToString(j), nSpellID));
if(nSubSpellID)
{
sSubName = sName+" : "+GetStringByStrRef(StringToInt(Get2DACache("spells", "Name", nSubSpellID)));
//set them less than zero so we know its a subradial
AddChoice(sSubName, 0-nSubSpellID, oPC);
//store the feat to decrement
SetLocalInt(oCohort, "SubradialSpellFeatID"+IntToString(nSubSpellID), i);
}
}
}
else
{
AddChoice(sName, i, oPC);
}
}
}
}
}
if(i<GetPRCSwitch(FILE_END_FEAT))
{
DelayCommand(0.0, AddUseableFeats(nMax, nMax+(nMax-nMin), oPC, oCohort));
}
}
void AddUseableSpells(int nMin, int nMax, object oPC, object oCohort)
{
int i;
for(i=nMin;i<nMax;i++)
{
if(GetHasSpell(i, oCohort))
{
string sName = GetStringByStrRef(StringToInt(Get2DACache("spells", "Name", i)));
//test for subradials
if(Get2DACache("spells", "SubRadSpell1", i) != "")
{
int j;
for(j=1;j<=5;j++)
{
string sSubName;
int nSubSpellID = StringToInt(Get2DACache("spells", "SubRadSpell"+IntToString(j), i));
if(nSubSpellID)
{
sSubName = sName+" : "+GetStringByStrRef(StringToInt(Get2DACache("spells", "Name", nSubSpellID)));
//set them less than zero so we know its a subradial
AddChoice(sSubName, 0-nSubSpellID, oPC);
}
}
}
else
{
AddChoice(sName, i, oPC);
}
}
}
if(i<GetPRCSwitch(FILE_END_SPELLS))
{
DelayCommand(0.0, AddUseableSpells(nMax, nMax+(nMax-nMin), oPC, oCohort));
}
}
//////////////////////////////////////////////////
/* Main function */
//////////////////////////////////////////////////
void main()
{
object oPC = GetPCSpeaker();
object oCohort = OBJECT_SELF;
/* Get the value of the local variable set by the conversation script calling
* this script. Values:
* DYNCONV_ABORTED Conversation aborted
* DYNCONV_EXITED Conversation exited via the exit node
* DYNCONV_SETUP_STAGE System's reply turn
* 0 Error - something else called the script
* Other The user made a choice
*/
int nValue = GetLocalInt(oPC, DYNCONV_VARIABLE);
// The stage is used to determine the active conversation node.
// 0 is the entry node.
int nStage = GetStage(oPC);
// Check which of the conversation scripts called the scripts
if(nValue == 0) // All of them set the DynConv_Var to non-zero value, so something is wrong -> abort
return;
if(nValue == DYNCONV_SETUP_STAGE)
{
// Check if this stage is marked as already set up
// This stops list duplication when scrolling
if(!GetIsStageSetUp(nStage, oPC))
{
// variable named nStage determines the current conversation node
// Function SetHeader to set the text displayed to the PC
// Function AddChoice to add a response option for the PC. The responses are show in order added
if(nStage == STAGE_ENTRY
|| nStage == STAGE_REENTRY)
{
if(oPC == GetMaster(oCohort))
{
if(nStage == STAGE_ENTRY)
SetHeader("What do you want?");
else if(nStage == STAGE_REENTRY)
SetHeader("What do you *really* want?");
AddChoice("I need you to cast a spell", 1, oPC);
AddChoice("I need you to use a feat", 2, oPC);
AddChoice("I need you to use an item", 3, oPC);
AddChoice("I need you to identify my equipment", 4, oPC);
AddChoice("I need you to change your tactics", 5, oPC);
AddChoice("I think we need to part ways", 6, oPC);
}
else
{
if(!GetIsObjectValid(GetMaster(oCohort)))
if(DEBUG) DoDebug("Master not valid!");
if(!GetIsObjectValid(oCohort))
if(DEBUG) DoDebug("Cohort not valid!");
//speaker not their master
SetHeader("Sorry, I can't talk to you right now. Ask "+GetName(GetMaster(oCohort))+" and see if they can help.");
//no responces
}
//SetHeader("Foo.");
//AddChoice("Bar", 1, oPC);
//AddChoice("Baz!", 2, oPC);
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
//using spellss
else if(nStage == STAGE_SPELL)
{
SetHeader("If I must. Which spell do you want me to cast?");
AddUseableSpells(0, 1000, oPC, oCohort);
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
//targetting spells is combined with feats
//using feats
else if(nStage == STAGE_FEAT)
{
SetHeader("If I must. Which feat do you want me to use?");
AddUseableFeats(0, 1000, oPC, oCohort);
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
else if(nStage == STAGE_FEAT_TARGET
|| nStage == STAGE_SPELL_TARGET)
{
int nFeat = GetLocalInt(oCohort, "PRC_FeatToUse");
int nSpellID = StringToInt(Get2DACache("feat", "SPELLID", nFeat));
if(nFeat < 0)
{
//subradial feat
nSpellID = 0-nFeat;
nFeat = GetLocalInt(oCohort, "SubradialSpellFeatID"+IntToString(nSpellID));
}
if(nStage == STAGE_SPELL_TARGET)
nSpellID = GetLocalInt(oCohort, "PRC_SpellToUse");
int nTargetType = HexToInt(Get2DACache("spells", "TargetType", nSpellID));
int nHostileSpell = StringToInt(Get2DACache("spells", "HostileSetting", nSpellID));
string sRangeType = Get2DACache("spells", "Range", nSpellID);
float fRange = 50.0;
if(sRangeType == "S"
|| sRangeType == "T"
|| sRangeType == "P")
fRange = 8.0;
else if(sRangeType == "M")
fRange = 20.0;
else if(sRangeType == "L")
fRange = 40.0;
/*
# 0x01 = 1 = Self
# 0x02 = 2 = Creature
# 0x04 = 4 = Area/Ground
# 0x08 = 8 = Items
# 0x10 = 16 = Doors
# 0x20 = 32 = Placeables
*/
int nCaster = nTargetType & 1;
int nCreature = nTargetType & 2;
int nLocation = nTargetType & 4;
int nItem = nTargetType & 8;
int nDoor = nTargetType & 16;
int nPlaceable = nTargetType & 32;
int nCount;
if(array_exists(oCohort, "PRC_ItemsToUse_Target"))
array_delete(oCohort, "PRC_ItemsToUse_Target");
array_create(oCohort, "PRC_ItemsToUse_Target");
//self
if(nCaster)
{
AddChoice("Self ("+GetName(oCohort)+")", array_get_size(oCohort, "PRC_ItemsToUse_Target"));
array_set_object(oCohort, "PRC_ItemsToUse_Target",
array_get_size(oCohort, "PRC_ItemsToUse_Target"), oCohort);
}
//nearby objects or locations of those objects
if(nCreature
|| nDoor
|| nPlaceable
|| nLocation)
{
object oTest = GetFirstObjectInShape(SHAPE_SPHERE, fRange, GetLocation(oCohort));
while(GetIsObjectValid(oTest))
{
int nType = GetObjectType(oTest);
if(((nType == OBJECT_TYPE_CREATURE && nCreature)
|| (nType == OBJECT_TYPE_DOOR && nDoor)
|| (nType == OBJECT_TYPE_PLACEABLE && nPlaceable)
|| nLocation)
&& oTest != oCohort)//self-casting is covered earlier
{
AddChoice(GetName(oTest), array_get_size(oCohort, "PRC_ItemsToUse_Target"));
array_set_object(oCohort, "PRC_ItemsToUse_Target",
array_get_size(oCohort, "PRC_ItemsToUse_Target"), oTest);
}
oTest = GetNextObjectInShape(SHAPE_SPHERE, fRange, GetLocation(oCohort));
}
}
//items in inventory
if(nItem)
{
object oTest = GetFirstItemInInventory(oCohort);
while(GetIsObjectValid(oTest))
{
AddChoice("(in inventory) "+GetName(oTest), array_get_size(oCohort, "PRC_ItemsToUse_Target"));
array_set_object(oCohort, "PRC_ItemsToUse_Target",
array_get_size(oCohort, "PRC_ItemsToUse_Target"), oTest);
oTest = GetNextItemInInventory(oCohort);
}
}
if(nStage == STAGE_FEAT_TARGET)
{
SetHeader("Who or what do you want me to use "+GetStringByStrRef(StringToInt(Get2DACache("spells", "Name", nSpellID)))+" on?");
}
else if(nStage == STAGE_SPELL_TARGET)
SetHeader("Who or what do you want me to cast "+GetStringByStrRef(StringToInt(Get2DACache("spells", "Name", nSpellID)))+" on?");
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
//using items
else if(nStage == STAGE_ITEM)
{
SetHeader("If I must. Which item do you want me to activate?");
if(!array_exists(oCohort, "PRC_ItemsToUse"))
{
array_create(oCohort, "PRC_ItemsToUse");
int nSlot;
for(nSlot = 0; nSlot < 14; nSlot++)
{
object oItem = GetItemInSlot(nSlot, oCohort);
if(GetIsObjectValid(oItem)
&& GetIdentified(oItem))
{
AddChoice(GetName(oItem), array_get_size(oCohort, "PRC_ItemsToUse"));
array_set_object(oCohort, "PRC_ItemsToUse", array_get_size(oCohort, "PRC_ItemsToUse"), oItem);
}
}
object oItem = GetFirstItemInInventory(oCohort);
while(GetIsObjectValid(oItem))
{
if(GetIdentified(oItem))
{
itemproperty ipTest = GetFirstItemProperty(oItem);
itemproperty ipInvalid;
while(GetIsItemPropertyValid(ipTest))
{
if(GetItemPropertyType(ipTest) == ITEM_PROPERTY_CAST_SPELL)
{
AddChoice(GetName(oItem), array_get_size(oCohort, "PRC_ItemsToUse"));
array_set_object(oCohort, "PRC_ItemsToUse", array_get_size(oCohort, "PRC_ItemsToUse"), oItem);
ipTest = ipInvalid;
}
else
ipTest = GetNextItemProperty(oItem);
}
}
oItem = GetNextItemInInventory(oCohort);
}
}
else
{
int i;
for(i=0;i<array_get_size(oCohort, "PRC_ItemsToUse");i++)
{
object oItem = array_get_object(oCohort, "PRC_ItemsToUse", i);
AddChoice(GetName(oItem), i);
}
}
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
else if(nStage == STAGE_ITEM_SPELL)
{
object oItem = GetLocalObject(oCohort, "PRC_ItemToUse");
itemproperty ipTest = GetFirstItemProperty(oItem);
while(GetIsItemPropertyValid(ipTest))
{
if(GetItemPropertyType(ipTest) == ITEM_PROPERTY_CAST_SPELL)
{
int nSpellID = GetItemPropertySubType(ipTest);
//convert that to a real ID
nSpellID = StringToInt(Get2DACache("iprp_spells", "SpellIndex", nSpellID));
AddChoice(ItemPropertyToString(ipTest), nSpellID);
}
ipTest = GetNextItemProperty(oItem);
}
SetHeader("What do you want me to do with "+GetName(oItem)+"?");
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
else if(nStage == STAGE_ITEM_TARGET)
{
object oItem = GetLocalObject(oCohort, "PRC_ItemToUse");
int nSpellID = GetLocalInt(oCohort, "PRC_ItemToUse_Spell");
int nTargetType = HexToInt(Get2DACache("spells", "TargetType", nSpellID));
int nHostileSpell = StringToInt(Get2DACache("spells", "HostileSetting", nSpellID));
string sRangeType = Get2DACache("spells", "Range", nSpellID);
float fRange = 50.0;
if(sRangeType == "S"
|| sRangeType == "T"
|| sRangeType == "P")
fRange = 8.0;
else if(sRangeType == "M")
fRange = 20.0;
else if(sRangeType == "L")
fRange = 40.0;
/*
# 0x01 = 1 = Self
# 0x02 = 2 = Creature
# 0x04 = 4 = Area/Ground
# 0x08 = 8 = Items
# 0x10 = 16 = Doors
# 0x20 = 32 = Placeables
*/
int nCaster = nTargetType & 1;
int nCreature = nTargetType & 2;
int nLocation = nTargetType & 4;
int nItem = nTargetType & 8;
int nDoor = nTargetType & 16;
int nPlaceable = nTargetType & 32;
//potions are caster only
if(GetBaseItemType(oItem) == BASE_ITEM_POTIONS
|| GetBaseItemType(oItem) == BASE_ITEM_ENCHANTED_POTION)
{
nCaster = TRUE;
nCreature = FALSE;
nLocation = FALSE;
nItem = FALSE;
nDoor = FALSE;
nPlaceable = FALSE;
}
int nCount;
if(array_exists(oCohort, "PRC_ItemsToUse_Target"))
array_delete(oCohort, "PRC_ItemsToUse_Target");
array_create(oCohort, "PRC_ItemsToUse_Target");
//self
if(nCaster)
{
AddChoice("Self ("+GetName(oCohort)+")", array_get_size(oCohort, "PRC_ItemsToUse_Target"));
array_set_object(oCohort, "PRC_ItemsToUse_Target",
array_get_size(oCohort, "PRC_ItemsToUse_Target"), oCohort);
}
//nearby objects or locations of those objects
if(nCreature
|| nDoor
|| nPlaceable
|| nLocation)
{
object oTest = GetFirstObjectInShape(SHAPE_SPHERE, fRange, GetLocation(oCohort));
while(GetIsObjectValid(oTest))
{
int nType = GetObjectType(oTest);
if(((nType == OBJECT_TYPE_CREATURE && nCreature)
|| (nType == OBJECT_TYPE_DOOR && nDoor)
|| (nType == OBJECT_TYPE_PLACEABLE && nPlaceable)
|| nLocation)
&& oTest != oCohort)
{
AddChoice(GetName(oTest), array_get_size(oCohort, "PRC_ItemsToUse_Target"));
array_set_object(oCohort, "PRC_ItemsToUse_Target",
array_get_size(oCohort, "PRC_ItemsToUse_Target"), oTest);
}
oTest = GetNextObjectInShape(SHAPE_SPHERE, fRange, GetLocation(oCohort));
}
}
//items in inventory
if(nItem)
{
int nSlot;
for(nSlot = 0; nSlot < 14; nSlot++)
{
object oItem = GetItemInSlot(nSlot, oCohort);
if(GetIsObjectValid(oItem))
{
AddChoice("(in inventory) "+GetName(oItem), array_get_size(oCohort, "PRC_ItemsToUse_Target"));
array_set_object(oPC, "PRC_ItemsToUse_Target", array_get_size(oPC, "PRC_ItemsToUse_Target"), oItem);
}
}
object oTest = GetFirstItemInInventory(oCohort);
while(GetIsObjectValid(oTest))
{
AddChoice("(in inventory) "+GetName(oTest), array_get_size(oCohort, "PRC_ItemsToUse_Target"));
array_set_object(oCohort, "PRC_ItemsToUse_Target",
array_get_size(oCohort, "PRC_ItemsToUse_Target"), oTest);
oTest = GetNextItemInInventory(oCohort);
}
}
SetHeader("Who or what do you want me to use "+GetName(oItem)+" on?");
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
//Identification of stuff
else if(nStage == STAGE_IDENTIFY)
{
SetHeader("Eh. Maybe. Allow me to rummage through your things. If something is found that is identifiable, I will tell you. Good enough?");
AddChoice("Good enough, go ahead.", 1, oPC);
AddChoice("Maybe later then.", 2, oPC);
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
else if(nStage == STAGE_IDENTIFY_YES)
{
string sList;
sList += "Okay, I have identified:\n";
//taken from TryToIDItems() in inc_utility
int nLore = GetSkillRank(SKILL_LORE, oCohort);
int nGP;
string sMax = Get2DACache("SkillVsItemCost", "DeviceCostMax", nLore);
int nMax = StringToInt(sMax);
if (sMax == "") nMax = 120000000;
object oItem = GetFirstItemInInventory(oPC);
while(oItem != OBJECT_INVALID)
{
if(!GetIdentified(oItem))
{
// Check for the value of the item first.
SetIdentified(oItem, TRUE);
nGP = GetGoldPieceValue(oItem);
SetIdentified(oItem, FALSE);
// If oPC has enough Lore skill to ID the item, then do so.
if(nMax >= nGP)
{
SetIdentified(oItem, TRUE);
sList += GetName(oItem)+"\n";
}
}
oItem = GetNextItemInInventory(oPC);
}
SetHeader(sList);
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
else if(nStage == STAGE_IDENTIFY_NO)
{
SetHeader("Then why did you bother asking me if you didnt want me to have a look!");
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
//tactics
else if(nStage == STAGE_TACTICS)
{
SetHeader("*You* would advise *me* on tactics?");
AddChoice("I want to adjust your equipment.", 1, oPC);
//x0_d2_hen_attck
if(!GetAssociateState(NW_ASC_MODE_DEFEND_MASTER, oCohort))
AddChoice("Defend me and don't attack until I do.", 2, oPC);
//x0_d2_hen_defnd
if(GetAssociateState(NW_ASC_MODE_DEFEND_MASTER, oCohort))
AddChoice("Go ahead and attack as soon as you see enemies.", 3, oPC);
AddChoice("I want to change the distance you stay away from me.", 4, oPC);
//nw_ch_no_locks
if(!GetAssociateState(NW_ASC_RETRY_OPEN_LOCKS, oCohort))
AddChoice("Help me if I fail to open a locked door or chest.", 5, oPC);
//nw_ch_yes_locks
if(GetAssociateState(NW_ASC_RETRY_OPEN_LOCKS, oCohort))
AddChoice("Don't help me if I fail to open a locked door or chest.", 6, oPC);
//follows the HotU stealth system, as in xp2_hen_dae conversation
AddChoice("Try to remain stealthy all the time.", 7, oPC);
AddChoice("Stay stealthy, but only until the next fight.", 8, oPC);
//x2_d1_instealth
//if(GetActionMode(oCohort, ACTION_MODE_STEALTH))
//not used, cohort drops stealth during conversation
AddChoice("Don't try to be stealthy anymore.", 9, oPC);
AddChoice("I don't want to change anything else right now.", 10, oPC);
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
else if(nStage == STAGE_TACTICS_EQUIP)
{
SetHeader("*sigh* Very well, then. Here is my equipment.");
OpenInventory(oCohort, oPC);
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
else if(nStage == STAGE_TACTICS_DEFEND
|| nStage == STAGE_TACTICS_ATTACK
|| nStage == STAGE_TACTICS_LOCK_HELP
|| nStage == STAGE_TACTICS_LOCK_NOHELP
|| nStage == STAGE_TACTICS_STEALTH_ALWAYS
|| nStage == STAGE_TACTICS_STEALTH_COMBAT
|| nStage == STAGE_TACTICS_STEALTH_NEVER
|| nStage == STAGE_TACTICS_DISTANCE_CLOSE
|| nStage == STAGE_TACTICS_DISTANCE_MEDIUM
|| nStage == STAGE_TACTICS_DISTANCE_LONG)
{
string sMessage;
if(nStage == STAGE_TACTICS_DEFEND)
sMessage = "Very well, then.";
else if(nStage == STAGE_TACTICS_ATTACK)
sMessage = "None will stand against us!";
else if(nStage == STAGE_TACTICS_LOCK_HELP)
sMessage = "Indeed. Where you have failed, I shall succeed!";
else if(nStage == STAGE_TACTICS_LOCK_NOHELP)
sMessage = "You can attempt them on your own then.";
else if(nStage == STAGE_TACTICS_STEALTH_ALWAYS)
sMessage = "Very well, I shall strike from the shadows whenever I am able.";
else if(nStage == STAGE_TACTICS_STEALTH_COMBAT)
sMessage = "I shall remain hidden until the battle begins, then I will burst forth to slaughter our foes.";
else if(nStage == STAGE_TACTICS_STEALTH_NEVER)
sMessage = "I do not like to hide in the shadows, anyway.";
else if(nStage == STAGE_TACTICS_DISTANCE_CLOSE
|| nStage == STAGE_TACTICS_DISTANCE_MEDIUM
|| nStage == STAGE_TACTICS_DISTANCE_LONG)
sMessage = "As you wish.";
SetHeader(sMessage+" Do you want to change anything else?");
AddChoice("No, that is all.", 2, oPC);
AddChoice("Yes.", 1, oPC);
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
else if(nStage == STAGE_TACTICS_DISTANCE)
{
SetHeader("How far apart should we remain, then?");
AddChoice("Stay close.", 1, oPC);
AddChoice("Keep a medium distance.", 2, oPC);
AddChoice("Stay a long distance away.", 3, oPC);
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
//leaving the party
else if(nStage == STAGE_LEAVE)
{
SetHeader("What?! You would turn me aside and make your way without me?");
AddChoice("Leave me. Now.", 1, oPC);
AddChoice("Alright, fine. Stay with me, then.", 2, oPC);
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
else if(nStage == STAGE_LEAVE_YES)
{
SetHeader("Bah! Leave if you will, but you shall not get far without me!");
DelayCommand(3.0, RemoveCohortFromPlayer(oCohort, oPC));
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
else if(nStage == STAGE_LEAVE_NO)
{
SetHeader("Ah, so you have not lost your mind completely.");
MarkStageSetUp(nStage, oPC); // This prevents the setup being run for this stage again until MarkStageNotSetUp is called for it
SetDefaultTokens(); // Set the next, previous, exit and wait tokens to default values
}
//add more stages for more nodes with Else If clauses
}
// Do token setup
SetupTokens();
}
// End of conversation cleanup
else if(nValue == DYNCONV_EXITED)
{
// Add any locals set through this conversation
array_delete(oCohort, "PRC_ItemsToUse");
array_delete(oCohort, "PRC_ItemsToUse_Target");
DeleteLocalObject(oCohort, "PRC_ItemToUse");
DeleteLocalObject(oCohort, "PRC_ItemToUse_Spell");
DeleteLocalInt(oCohort, "PRC_FeatToUse");
DeleteLocalInt(oCohort, "PRC_SpellToUse");
DeleteLocalInt(oCohort, "PRC_InCohortConvoMarker");
}
// Abort conversation cleanup.
// NOTE: This section is only run when the conversation is aborted
// while aborting is allowed. When it isn't, the dynconvo infrastructure
// handles restoring the conversation in a transparent manner
else if(nValue == DYNCONV_ABORTED)
{
// Add any locals set through this conversation
array_delete(oCohort, "PRC_ItemsToUse");
array_delete(oCohort, "PRC_ItemsToUse_Target");
DeleteLocalObject(oCohort, "PRC_ItemToUse");
DeleteLocalObject(oCohort, "PRC_ItemToUse_Spell");
DeleteLocalInt(oCohort, "PRC_FeatToUse");
DeleteLocalInt(oCohort, "PRC_SpellToUse");
DeleteLocalInt(oCohort, "PRC_InCohortConvoMarker");
}
// Handle PC responses
else
{
// variable named nChoice is the value of the player's choice as stored when building the choice list
// variable named nStage determines the current conversation node
int nChoice = GetChoice(oPC);
int nOldStage = nStage;
if(nStage == STAGE_ENTRY
|| nStage == STAGE_REENTRY)
{
switch(nChoice)
{
case 1: nStage = STAGE_SPELL; break;
case 2: nStage = STAGE_FEAT; break;
case 3: nStage = STAGE_ITEM; break;
case 4: nStage = STAGE_IDENTIFY; break;
case 5: nStage = STAGE_TACTICS; break;
case 6: nStage = STAGE_LEAVE; break;
}
}
else if(nStage == STAGE_LEAVE)
{
switch(nChoice)
{
case 1: nStage = STAGE_LEAVE_YES; break;
case 2: nStage = STAGE_LEAVE_NO; break;
}
}
else if(nStage == STAGE_SPELL)
{
SetLocalInt(oCohort, "PRC_SpellToUse", nChoice);
nStage = STAGE_SPELL_TARGET;
}
else if(nStage == STAGE_SPELL_TARGET)
{
int nSpellID = GetLocalInt(oCohort, "PRC_SpellToUse");
object oTarget = array_get_object(oCohort, "PRC_ItemsToUse_Target", nChoice);
//test if location or object
//use object by preference
int nTargetType = HexToInt(Get2DACache("spells", "TargetType", nSpellID));
/*
# 0x01 = 1 = Self
# 0x02 = 2 = Creature
# 0x04 = 4 = Area/Ground
# 0x08 = 8 = Items
# 0x10 = 16 = Doors
# 0x20 = 32 = Placeables
*/
int nCaster = nTargetType & 1;
int nCreature = nTargetType & 2;
int nLocation = nTargetType & 4;
int nItem = nTargetType & 8;
int nDoor = nTargetType & 16;
int nPlaceable = nTargetType & 32;
int nType = GetObjectType(oTarget);
if((oTarget == oCohort && nCaster)
|| (nType == OBJECT_TYPE_CREATURE && nCreature)
|| (nType == OBJECT_TYPE_DOOR && nDoor)
|| (nType == OBJECT_TYPE_PLACEABLE && nPlaceable)
|| (nType == OBJECT_TYPE_ITEM && nItem))
{
AssignCommand(oCohort, ClearAllActions());
AssignCommand(oCohort, ActionCastSpellAtObject(nSpellID, oTarget));
}
else if(nLocation)
{
AssignCommand(oCohort, ClearAllActions());
AssignCommand(oCohort, ActionCastSpellAtLocation(nSpellID, GetLocation(oTarget)));
}
AllowExit(DYNCONV_EXIT_FORCE_EXIT);
}
else if(nStage == STAGE_FEAT)
{
SetLocalInt(oCohort, "PRC_FeatToUse", nChoice);
//a few hardcoded feats are fuggly
if(nChoice == FEAT_ANIMAL_COMPANION)
{
DecrementRemainingFeatUses(oCohort, FEAT_ANIMAL_COMPANION);
AssignCommand(oCohort, SummonAnimalCompanion());
AllowExit(DYNCONV_EXIT_FORCE_EXIT);
}
else if(nChoice == FEAT_SUMMON_FAMILIAR)
{
DecrementRemainingFeatUses(oCohort, FEAT_SUMMON_FAMILIAR);
AssignCommand(oCohort, SummonFamiliar());
AllowExit(DYNCONV_EXIT_FORCE_EXIT);
}
//if its self only, use it
if(StringToInt(Get2DACache("feat", "TARGETSELF", nChoice)))
{
AssignCommand(oCohort, ClearAllActions());
AssignCommand(oCohort, ActionUseFeat(nChoice, oCohort));
//feat used, end conversation
AllowExit(DYNCONV_EXIT_FORCE_EXIT);
}
nStage = STAGE_FEAT_TARGET;
}
else if(nStage == STAGE_FEAT_TARGET)
{
int nFeat = GetLocalInt(oCohort, "PRC_FeatToUse");
object oTarget = array_get_object(oCohort, "PRC_ItemsToUse_Target", nChoice);
if(nFeat < 0)
{
//subradial feat
int nSpellID;
nSpellID = 0-nFeat;
nFeat = GetLocalInt(oCohort, "SubradialSpellFeatID"+IntToString(nSpellID));
DecrementRemainingFeatUses(oCohort, nFeat);
AssignCommand(oCohort, ClearAllActions());
AssignCommand(oCohort, ActionCastSpellAtObject(nSpellID, oTarget, METAMAGIC_ANY, TRUE));
}
else
{
AssignCommand(oCohort, ClearAllActions());
AssignCommand(oCohort, ActionUseFeat(nFeat, oTarget));
}
AllowExit(DYNCONV_EXIT_FORCE_EXIT);
}
else if(nStage == STAGE_ITEM)
{
object oItem = array_get_object(oCohort, "PRC_ItemsToUse", nChoice);
SetLocalObject(oCohort, "PRC_ItemToUse", oItem);
//if its multiple spells
itemproperty ipTest = GetFirstItemProperty(oItem);
int nSpellCount;
while(GetIsItemPropertyValid(ipTest))
{
if(GetItemPropertyType(ipTest) == ITEM_PROPERTY_CAST_SPELL)
nSpellCount++;
ipTest = GetNextItemProperty(oItem);
}
if(nSpellCount > 1)
{
nStage = STAGE_ITEM_SPELL;
}
else
{
ipTest = GetFirstItemProperty(oItem);
while(GetIsItemPropertyValid(ipTest))
{
if(GetItemPropertyType(ipTest) == ITEM_PROPERTY_CAST_SPELL)
break;
ipTest = GetNextItemProperty(oItem);
}
int nSpellID = GetItemPropertySubType(ipTest);
//convert that to a real ID
nSpellID = StringToInt(Get2DACache("iprp_spells", "SpellIndex", nSpellID));
//store it
SetLocalInt(oCohort, "PRC_ItemToUse_Spell", nSpellID);
nStage = STAGE_ITEM_TARGET;
}
}
else if(nStage == STAGE_ITEM_SPELL)
{
SetLocalInt(oCohort, "PRC_ItemToUse_Spell", nChoice);
//check if its self-only, if so use it
object oItem = GetLocalObject(oCohort, "PRC_ItemToUse");
int nSpellID = GetLocalInt(oCohort, "PRC_ItemToUse_Spell");
object oTarget = array_get_object(oCohort, "PRC_ItemsToUse_Target", nChoice);
itemproperty ipIP;
ipIP = GetFirstItemProperty(oItem);
while(GetIsItemPropertyValid(ipIP))
{
if(GetItemPropertyType(ipIP) == ITEM_PROPERTY_CAST_SPELL)
{
int nipSpellID = GetItemPropertySubType(ipIP);
//convert that to a real ID
nipSpellID = StringToInt(Get2DACache("iprp_spells", "SpellIndex", nipSpellID));
if(nipSpellID == nSpellID)
{
if(DEBUG) DoDebug("Ending itemprop loop "+IntToString(nipSpellID));
break;//end while loop
}
}
ipIP = GetNextItemProperty(oItem);
}
//test if location or object
//use object by preference
int nTargetType = HexToInt(Get2DACache("spells", "TargetType", nSpellID));
if(nTargetType == 1)
{
//self only item, use it
AssignCommand(oCohort, ClearAllActions());
AssignCommand(oCohort,
ActionUseItemPropertyAtObject(oItem, ipIP, oCohort));
if(DEBUG) DoDebug("Running ActionUseItemPropertyAtObject() at "+GetName(oCohort));
//item used, end conversation
AllowExit(DYNCONV_EXIT_FORCE_EXIT);
}
nStage = STAGE_ITEM_TARGET;
}
else if(nStage == STAGE_ITEM_TARGET)
{
object oItem = GetLocalObject(oCohort, "PRC_ItemToUse");
int nSpellID = GetLocalInt(oCohort, "PRC_ItemToUse_Spell");
object oTarget = array_get_object(oCohort, "PRC_ItemsToUse_Target", nChoice);
itemproperty ipIP;
ipIP = GetFirstItemProperty(oItem);
while(GetIsItemPropertyValid(ipIP))
{
if(GetItemPropertyType(ipIP) == ITEM_PROPERTY_CAST_SPELL)
{
int nipSpellID = GetItemPropertySubType(ipIP);
//convert that to a real ID
nipSpellID = StringToInt(Get2DACache("iprp_spells", "SpellIndex", nipSpellID));
if(nipSpellID == nSpellID)
{
if(DEBUG) DoDebug("Ending itemprop loop "+IntToString(nipSpellID));
break;//end while loop
}
}
ipIP = GetNextItemProperty(oItem);
}
//test if location or object
//use object by preference
int nTargetType = HexToInt(Get2DACache("spells", "TargetType", nSpellID));
/*
# 0x01 = 1 = Self
# 0x02 = 2 = Creature
# 0x04 = 4 = Area/Ground
# 0x08 = 8 = Items
# 0x10 = 16 = Doors
# 0x20 = 32 = Placeables
*/
int nCaster = nTargetType & 1;
int nCreature = nTargetType & 2;
int nLocation = nTargetType & 4;
int nItem = nTargetType & 8;
int nDoor = nTargetType & 16;
int nPlaceable = nTargetType & 32;
int nType = GetObjectType(oTarget);
if((oTarget == oCohort && nCaster)
|| (nType == OBJECT_TYPE_CREATURE && nCreature)
|| (nType == OBJECT_TYPE_DOOR && nDoor)
|| (nType == OBJECT_TYPE_PLACEABLE && nPlaceable)
|| (nType == OBJECT_TYPE_ITEM && nItem))
{
AssignCommand(oCohort, ClearAllActions());
AssignCommand(oCohort,
ActionUseItemPropertyAtObject(oItem, ipIP, oTarget));
if(DEBUG) DoDebug("Running ActionUseItemPropertyAtObject() at "+GetName(oTarget));
}
else if(nLocation)
{
AssignCommand(oCohort, ClearAllActions());
AssignCommand(oCohort,
ActionUseItemPropertyAtLocation(oItem, ipIP, GetLocation(oTarget)));
if(DEBUG) DoDebug("Running ActionUseItemPropertyAtLocation() at "+GetName(oTarget));
}
AllowExit(DYNCONV_EXIT_FORCE_EXIT);
}
else if(nStage == STAGE_IDENTIFY)
{
switch(nChoice)
{
case 1: nStage = STAGE_IDENTIFY_YES; break;
case 2: nStage = STAGE_IDENTIFY_NO; break;
}
}
else if(nStage == STAGE_TACTICS)
{
switch(nChoice)
{
case 1: nStage = STAGE_TACTICS_EQUIP; break;
case 2: nStage = STAGE_TACTICS_DEFEND;
SetAssociateState(NW_ASC_MODE_DEFEND_MASTER, TRUE, oCohort);
break;
case 3: nStage = STAGE_TACTICS_ATTACK;
SetAssociateState(NW_ASC_MODE_DEFEND_MASTER, FALSE, oCohort);
break;
case 4: nStage = STAGE_TACTICS_DISTANCE; break;
case 5: nStage = STAGE_TACTICS_LOCK_HELP;
SetAssociateState(NW_ASC_RETRY_OPEN_LOCKS, TRUE, oCohort);
break;
case 6: nStage = STAGE_TACTICS_LOCK_NOHELP;
SetAssociateState(NW_ASC_RETRY_OPEN_LOCKS, FALSE, oCohort);
break;
case 7: nStage = STAGE_TACTICS_STEALTH_ALWAYS;
AssignCommand(oCohort, ClearAllActions());
SetLocalInt(oCohort, "X2_HENCH_STEALTH_MODE", 1);
break;
case 8: nStage = STAGE_TACTICS_STEALTH_COMBAT;
AssignCommand(oCohort, ClearAllActions());
SetLocalInt(oCohort, "X2_HENCH_STEALTH_MODE", 2);
break;
case 9: nStage = STAGE_TACTICS_STEALTH_NEVER;
AssignCommand(oCohort, ClearAllActions());
SetLocalInt(oCohort, "X2_HENCH_STEALTH_MODE", 0);
SetActionMode(oCohort, ACTION_MODE_STEALTH, FALSE);
break;
case 10: nStage = STAGE_REENTRY; break;
}
}
else if(nStage == STAGE_TACTICS_DISTANCE)
{
switch(nChoice)
{
case 1: nStage = STAGE_TACTICS_DISTANCE_CLOSE;
//nw_ch_dist_6
SetAssociateState(NW_ASC_DISTANCE_2_METERS, TRUE, oCohort);
SetAssociateState(NW_ASC_DISTANCE_4_METERS, FALSE, oCohort);
SetAssociateState(NW_ASC_DISTANCE_6_METERS, FALSE, oCohort);
break;
case 2: nStage = STAGE_TACTICS_DISTANCE_MEDIUM;
//nw_ch_dist_12
SetAssociateState(NW_ASC_DISTANCE_2_METERS, FALSE, oCohort);
SetAssociateState(NW_ASC_DISTANCE_4_METERS, TRUE, oCohort);
SetAssociateState(NW_ASC_DISTANCE_6_METERS, FALSE, oCohort);
break;
case 3: nStage = STAGE_TACTICS_DISTANCE_LONG;
//nw_ch_dist_18
SetAssociateState(NW_ASC_DISTANCE_2_METERS, FALSE, oCohort);
SetAssociateState(NW_ASC_DISTANCE_4_METERS, FALSE, oCohort);
SetAssociateState(NW_ASC_DISTANCE_6_METERS, TRUE, oCohort);
break;
}
}
else if(nStage == STAGE_TACTICS_DEFEND
|| nStage == STAGE_TACTICS_ATTACK
|| nStage == STAGE_TACTICS_LOCK_HELP
|| nStage == STAGE_TACTICS_LOCK_NOHELP
|| nStage == STAGE_TACTICS_STEALTH_ALWAYS
|| nStage == STAGE_TACTICS_STEALTH_COMBAT
|| nStage == STAGE_TACTICS_STEALTH_NEVER
|| nStage == STAGE_TACTICS_DISTANCE_CLOSE
|| nStage == STAGE_TACTICS_DISTANCE_MEDIUM
|| nStage == STAGE_TACTICS_DISTANCE_LONG)
{
switch(nChoice)
{
case 1: nStage = STAGE_TACTICS; break;
case 2: AllowExit(DYNCONV_EXIT_FORCE_EXIT); break;
}
}
else if(nStage == STAGE_LEAVE)
{
switch(nChoice)
{
case 1: nStage = STAGE_LEAVE_YES; break;
case 2: nStage = STAGE_LEAVE_NO; break;
}
}
// Store the stage value. If it has been changed, this clears out the choices
SetStage(nStage, oPC);
//if stage changed, mark the new one as not set up
if(nStage != nOldStage)
MarkStageNotSetUp(nStage, oPC);
}
}