Updated Release Archive. Fixed Mage-killer prereqs. Removed old LETO & ConvoCC related files. Added organized spell scroll store. Fixed Gloura spellbook. Various TLK fixes. Reorganized Repo. Removed invalid user folders. Added DocGen back in.
408 lines
17 KiB
Plaintext
408 lines
17 KiB
Plaintext
//:://////////////////////////////////////////////
|
|
//:: Shadowcasting Mystery gain conversation script
|
|
//:: shd_mystconv
|
|
//:://////////////////////////////////////////////
|
|
/** @file
|
|
@author Stratovarius - 2019.02.08
|
|
*/
|
|
//:://////////////////////////////////////////////
|
|
//:://////////////////////////////////////////////
|
|
|
|
#include "prc_inc_function"
|
|
#include "shd_inc_shdfunc"
|
|
#include "inc_dynconv"
|
|
|
|
//////////////////////////////////////////////////
|
|
/* Constant defintions */
|
|
//////////////////////////////////////////////////
|
|
|
|
const int STAGE_SELECT_LEVEL = 0;
|
|
const int STAGE_SELECT_MYSTERY = 1;
|
|
const int STAGE_CONFIRM_SELECTION = 2;
|
|
const int STAGE_ALL_MYSTERIES_SELECTED = 3;
|
|
|
|
const int CHOICE_BACK_TO_LSELECT = -1;
|
|
|
|
const int STRREF_BACK_TO_LSELECT = 16836035; // "Return to level selection."
|
|
const int STRREF_LEVELLIST_HEADER = 16836036; // "Select level of mystery to gain.\n\nNOTE:\nThis may take a while when first browsing a particular level's mysteries."
|
|
const int STRREF_MYSTLIST_HEADER1 = 16836037; // "Select a mystery to gain.\nYou can select"
|
|
const int STRREF_MYSTLIST_HEADER2 = 16836038; // "more mysteries"
|
|
const int STRREF_SELECTED_HEADER1 = 16824209; // "You have selected:"
|
|
const int STRREF_SELECTED_HEADER2 = 16824210; // "Is this correct?"
|
|
const int STRREF_END_HEADER = 16836039; // "You will be able to select more mysteries after you gain another level in a shadowcasting class."
|
|
const int STRREF_END_CONVO_SELECT = 16824212; // "Finish"
|
|
const int LEVEL_STRREF_START = 16824809;
|
|
const int STRREF_YES = 4752; // "Yes"
|
|
const int STRREF_NO = 4753; // "No"
|
|
|
|
const int SORT = TRUE; // If the sorting takes too much CPU, set to FALSE
|
|
const int DEBUG_LIST = FALSE;
|
|
|
|
//////////////////////////////////////////////////
|
|
/* Function defintions */
|
|
//////////////////////////////////////////////////
|
|
|
|
void PrintList(object oPC)
|
|
{
|
|
string tp = "Printing list:\n";
|
|
string s = GetLocalString(oPC, "PRC_MystConvo_List_Head");
|
|
if(s == ""){
|
|
tp += "Empty\n";
|
|
}
|
|
else{
|
|
tp += s + "\n";
|
|
s = GetLocalString(oPC, "PRC_MystConvo_List_Next_" + s);
|
|
while(s != ""){
|
|
tp += "=> " + s + "\n";
|
|
s = GetLocalString(oPC, "PRC_MystConvo_List_Next_" + s);
|
|
}
|
|
}
|
|
|
|
DoDebug(tp);
|
|
}
|
|
|
|
/**
|
|
* Creates a linked list of entries that is sorted into alphabetical order
|
|
* as it is built.
|
|
* Assumption: mystery names are unique.
|
|
*
|
|
* @param oPC The storage object aka whomever is gaining powers in this conversation
|
|
* @param sChoice The choice string
|
|
* @param nChoice The choice value
|
|
*/
|
|
void AddToTempList(object oPC, string sChoice, int nChoice)
|
|
{
|
|
if(DEBUG_LIST) DoDebug("\nAdding to temp list: '" + sChoice + "' - " + IntToString(nChoice));
|
|
if(DEBUG_LIST) PrintList(oPC);
|
|
// If there is nothing yet
|
|
if(!GetLocalInt(oPC, "PRC_MystConvo_ListInited"))
|
|
{
|
|
SetLocalString(oPC, "PRC_MystConvo_List_Head", sChoice);
|
|
SetLocalInt(oPC, "PRC_MystConvo_List_" + sChoice, nChoice);
|
|
|
|
SetLocalInt(oPC, "PRC_MystConvo_ListInited", TRUE);
|
|
}
|
|
else
|
|
{
|
|
// Find the location to instert into
|
|
string sPrev = "", sNext = GetLocalString(oPC, "PRC_MystConvo_List_Head");
|
|
while(sNext != "" && StringCompare(sChoice, sNext) >= 0)
|
|
{
|
|
if(DEBUG_LIST) DoDebug("Comparison between '" + sChoice + "' and '" + sNext + "' = " + IntToString(StringCompare(sChoice, sNext)));
|
|
sPrev = sNext;
|
|
sNext = GetLocalString(oPC, "PRC_MystConvo_List_Next_" + sNext);
|
|
}
|
|
|
|
// Insert the new entry
|
|
// Does it replace the head?
|
|
if(sPrev == "")
|
|
{
|
|
if(DEBUG_LIST) DoDebug("New head");
|
|
SetLocalString(oPC, "PRC_MystConvo_List_Head", sChoice);
|
|
}
|
|
else
|
|
{
|
|
if(DEBUG_LIST) DoDebug("Inserting into position between '" + sPrev + "' and '" + sNext + "'");
|
|
SetLocalString(oPC, "PRC_MystConvo_List_Next_" + sPrev, sChoice);
|
|
}
|
|
|
|
SetLocalString(oPC, "PRC_MystConvo_List_Next_" + sChoice, sNext);
|
|
SetLocalInt(oPC, "PRC_MystConvo_List_" + sChoice, nChoice);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reads the linked list built with AddToTempList() to AddChoice() and
|
|
* deletes it.
|
|
*
|
|
* @param oPC A PC gaining powers at the moment
|
|
*/
|
|
void TransferTempList(object oPC)
|
|
{
|
|
string sChoice = GetLocalString(oPC, "PRC_MystConvo_List_Head");
|
|
int nChoice = GetLocalInt (oPC, "PRC_MystConvo_List_" + sChoice);
|
|
|
|
DeleteLocalString(oPC, "PRC_MystConvo_List_Head");
|
|
string sPrev;
|
|
|
|
if(DEBUG_LIST) DoDebug("Head is: '" + sChoice + "' - " + IntToString(nChoice));
|
|
|
|
while(sChoice != "")
|
|
{
|
|
// Add the choice
|
|
AddChoice(sChoice, nChoice, oPC);
|
|
|
|
// Get next
|
|
sChoice = GetLocalString(oPC, "PRC_MystConvo_List_Next_" + (sPrev = sChoice));
|
|
nChoice = GetLocalInt (oPC, "PRC_MystConvo_List_" + sChoice);
|
|
|
|
if(DEBUG_LIST) DoDebug("Next is: '" + sChoice + "' - " + IntToString(nChoice) + "; previous = '" + sPrev + "'");
|
|
|
|
// Delete the already handled data
|
|
DeleteLocalString(oPC, "PRC_MystConvo_List_Next_" + sPrev);
|
|
DeleteLocalInt (oPC, "PRC_MystConvo_List_" + sPrev);
|
|
}
|
|
|
|
DeleteLocalInt(oPC, "PRC_MystConvo_ListInited");
|
|
}
|
|
|
|
void main()
|
|
{
|
|
object oPC = GetPCSpeaker();
|
|
int nValue = GetLocalInt(oPC, DYNCONV_VARIABLE);
|
|
int nStage = GetStage(oPC);
|
|
|
|
// This can be Shadowcaster or Shadowsmith
|
|
int nClass = GetLocalInt(oPC, "nClass");
|
|
string sPowerFile = GetAMSDefinitionFileName(nClass);
|
|
|
|
// 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)
|
|
{
|
|
if(DEBUG) DoDebug("shd_mystconv: Running setup stage for stage " + IntToString(nStage));
|
|
// Check if this stage is marked as already set up
|
|
// This stops list duplication when scrolling
|
|
if(!GetIsStageSetUp(nStage, oPC))
|
|
{
|
|
if(DEBUG) DoDebug("shd_mystconv: Stage was not set up already");
|
|
// Level selection stage
|
|
if(nStage == STAGE_SELECT_LEVEL)
|
|
{
|
|
if(DEBUG) DoDebug("shd_mystconv: Building level selection");
|
|
SetHeader(GetStringByStrRef(STRREF_LEVELLIST_HEADER));
|
|
|
|
// Determine maximum mystery level that we should display
|
|
// We do this three times for Apprentice, Initiate, and Master mysteries
|
|
int nMaxLevel = GetMaxMysteryLevelLearnable(oPC, nClass, 1);
|
|
|
|
// Set the tokens
|
|
int i;
|
|
for(i = 0; i < nMaxLevel; i++){
|
|
AddChoice(GetStringByStrRef(LEVEL_STRREF_START - i), // The minus is correct, these are stored in inverse order in the TLK. Whoops
|
|
i + 1
|
|
);
|
|
}
|
|
|
|
nMaxLevel = GetMaxMysteryLevelLearnable(oPC, nClass, 2);
|
|
|
|
// Set the tokens
|
|
for(i = 3; i < nMaxLevel; i++){
|
|
AddChoice(GetStringByStrRef(LEVEL_STRREF_START - i), // The minus is correct, these are stored in inverse order in the TLK. Whoops
|
|
i + 1
|
|
);
|
|
}
|
|
|
|
nMaxLevel = GetMaxMysteryLevelLearnable(oPC, nClass, 3);
|
|
|
|
// Set the tokens
|
|
for(i = 6; i < nMaxLevel; i++){
|
|
AddChoice(GetStringByStrRef(LEVEL_STRREF_START - i), // The minus is correct, these are stored in inverse order in the TLK. Whoops
|
|
i + 1
|
|
);
|
|
}
|
|
|
|
// Set the next, previous and wait tokens to default values
|
|
SetDefaultTokens();
|
|
// Set the convo quit text to "Abort"
|
|
SetCustomToken(DYNCONV_TOKEN_EXIT, GetStringByStrRef(DYNCONV_STRREF_ABORT_CONVO));
|
|
}
|
|
// mystery selection stage
|
|
if(nStage == STAGE_SELECT_MYSTERY)
|
|
{
|
|
if(DEBUG) DoDebug("shd_mystconv: Building mystery selection");
|
|
|
|
int nCurrentMysts = GetMysteryCount(oPC, nClass);
|
|
int nMaxMysts = GetMaxMysteryCount(oPC, nClass);
|
|
string sToken = GetStringByStrRef(STRREF_MYSTLIST_HEADER1) + " " + //"Select a mystery to gain.\n You can select "
|
|
IntToString(nMaxMysts-nCurrentMysts) + " " +
|
|
GetStringByStrRef(STRREF_MYSTLIST_HEADER2); //" more mysteries"
|
|
SetHeader(sToken);
|
|
|
|
// Set the first choice to be return to level selection stage
|
|
AddChoice(GetStringByStrRef(STRREF_BACK_TO_LSELECT), CHOICE_BACK_TO_LSELECT, oPC);
|
|
|
|
int nMystLevelToBrowse = GetLocalInt(oPC, "nMystLevelToBrowse");
|
|
|
|
if(DEBUG)
|
|
{
|
|
DoDebug("shd_mystconv: Current Mysterys: " + IntToString(nCurrentMysts));
|
|
DoDebug("shd_mystconv: Max Mysterys: " + IntToString(nMaxMysts));
|
|
DoDebug("shd_mystconv: Mystery Level To Browse: " + IntToString(nMystLevelToBrowse));
|
|
}
|
|
|
|
|
|
int i, nMystLevel;
|
|
string sFeatID;
|
|
for(i = 0; i < GetPRCSwitch(FILE_END_CLASS_POWER) ; i++)
|
|
{
|
|
nMystLevel = StringToInt(Get2DACache(sPowerFile, "Level", i));
|
|
// Skip any powers of too low level
|
|
if(nMystLevel < nMystLevelToBrowse){
|
|
continue;
|
|
}
|
|
//Due to the way the mystery list 2das are structured, we know that once
|
|
//the level of a read mystery is greater than the maximum castable
|
|
//it'll never be lower again. Therefore, we can skip reading the
|
|
//mysteries that wouldn't be shown anyway.
|
|
|
|
if(nMystLevel > nMystLevelToBrowse){
|
|
break;
|
|
}
|
|
sFeatID = Get2DACache(sPowerFile, "FeatID", i);
|
|
if(sFeatID != "" // Non-blank row
|
|
&& !GetHasFeat(StringToInt(sFeatID), oPC) // PC does not already posses the power
|
|
)
|
|
{
|
|
if(SORT) AddToTempList(oPC, GetStringByStrRef(StringToInt(Get2DACache(sPowerFile, "Name", i))), i);
|
|
else AddChoice(GetStringByStrRef(StringToInt(Get2DACache(sPowerFile, "Name", i))), i, oPC);
|
|
}
|
|
}
|
|
|
|
if(SORT) TransferTempList(oPC);
|
|
|
|
// Hack - In the mystery selection stage, on returning from
|
|
// confirmation dialog where the answer was "No", restore the
|
|
// offset to be the same as on entering the confirmation dialog.
|
|
if(GetLocalInt(oPC, "MYSTLISTChoiceOffset"))
|
|
{
|
|
if(DEBUG) DoDebug("shd_mystconv: Running offset restoration hack");
|
|
SetLocalInt(oPC, DYNCONV_CHOICEOFFSET, GetLocalInt(oPC, "MYSTLISTChoiceOffset") - 1);
|
|
DeleteLocalInt(oPC, "MYSTLISTChoiceOffset");
|
|
}
|
|
|
|
MarkStageSetUp(STAGE_SELECT_MYSTERY, oPC);
|
|
}
|
|
// Selection confirmation stage
|
|
else if(nStage == STAGE_CONFIRM_SELECTION)
|
|
{
|
|
if(DEBUG) DoDebug("shd_mystconv: Building selection confirmation");
|
|
// Build the confirmantion query
|
|
string sToken = GetStringByStrRef(STRREF_SELECTED_HEADER1) + "\n\n"; // "You have selected:"
|
|
int nMyst = GetLocalInt(oPC, "nMyst");
|
|
int nFeatID = StringToInt(Get2DAString(sPowerFile, "FeatID", nMyst));
|
|
if(DEBUG) DoDebug("STAGE_CONFIRM_SELECTION nMyst: " + IntToString(nMyst));
|
|
if(DEBUG) DoDebug("STAGE_CONFIRM_SELECTION nFeatID: " + IntToString(nFeatID));
|
|
sToken += GetStringByStrRef(StringToInt(Get2DACache("feat", "FEAT", nFeatID)))+"\n";
|
|
sToken += GetStringByStrRef(StringToInt(Get2DACache("feat", "DESCRIPTION", nFeatID)))+"\n\n";
|
|
sToken += GetStringByStrRef(STRREF_SELECTED_HEADER2); // "Is this correct?"
|
|
SetHeader(sToken);
|
|
|
|
AddChoice(GetStringByStrRef(STRREF_YES), TRUE, oPC); // "Yes"
|
|
AddChoice(GetStringByStrRef(STRREF_NO), FALSE, oPC); // "No"
|
|
}
|
|
// Conversation finished stage
|
|
else if(nStage == STAGE_ALL_MYSTERIES_SELECTED)
|
|
{
|
|
if(DEBUG) DoDebug("shd_mystconv: Building finish note");
|
|
SetHeader(GetStringByStrRef(STRREF_END_HEADER));
|
|
// Set the convo quit text to "Finish"
|
|
SetCustomToken(DYNCONV_TOKEN_EXIT, GetStringByStrRef(STRREF_END_CONVO_SELECT));
|
|
AllowExit(DYNCONV_EXIT_ALLOWED_SHOW_CHOICE, FALSE, oPC);
|
|
}
|
|
}
|
|
|
|
// Do token setup
|
|
SetupTokens();
|
|
}
|
|
else if(nValue == DYNCONV_EXITED)
|
|
{
|
|
if(DEBUG) DoDebug("shd_mystconv: Running exit handler");
|
|
// End of conversation cleanup
|
|
DeleteLocalInt(oPC, "nClass");
|
|
DeleteLocalInt(oPC, "nMyst");
|
|
DeleteLocalInt(oPC, "nMystLevelToBrowse");
|
|
DeleteLocalInt(oPC, "MYSTLISTChoiceOffset");
|
|
|
|
// Restart the convo to pick next mystery if needed
|
|
// done via EvalPRCFeats to avoid conflicts with new spellbooks
|
|
DelayCommand(1.0, EvalPRCFeats(oPC));
|
|
|
|
if (GetCompletedPaths(oPC) > GetPathBonusFeats(oPC))
|
|
DelayCommand(1.0, StartDynamicConversation("shd_pathconv", oPC, DYNCONV_EXIT_NOT_ALLOWED, FALSE, TRUE, oPC));
|
|
}
|
|
else if(nValue == DYNCONV_ABORTED)
|
|
{
|
|
// This section should never be run, since aborting this conversation should
|
|
// always be forbidden and as such, any attempts to abort the conversation
|
|
// should be handled transparently by the system
|
|
if(DEBUG) DoDebug("shd_mystconv: ERROR: Conversation abort section run");
|
|
DelayCommand(1.0, EvalPRCFeats(oPC));
|
|
|
|
if (GetCompletedPaths(oPC) > GetPathBonusFeats(oPC))
|
|
DelayCommand(1.0, StartDynamicConversation("shd_pathconv", oPC, DYNCONV_EXIT_NOT_ALLOWED, FALSE, TRUE, oPC));
|
|
}
|
|
// Handle PC response
|
|
else
|
|
{
|
|
int nChoice = GetChoice(oPC);
|
|
if(DEBUG) DoDebug("shd_mystconv: Handling PC response, stage = " + IntToString(nStage) + "; nChoice = " + IntToString(nChoice) + "; choice text = '" + GetChoiceText(oPC) + "'");
|
|
if(nStage == STAGE_SELECT_LEVEL)
|
|
{
|
|
if(DEBUG) DoDebug("shd_mystconv: Level selected");
|
|
SetLocalInt(oPC, "nMystLevelToBrowse", nChoice);
|
|
nStage = STAGE_SELECT_MYSTERY;
|
|
|
|
MarkStageNotSetUp(STAGE_SELECT_LEVEL, oPC);
|
|
}
|
|
else if(nStage == STAGE_SELECT_MYSTERY)
|
|
{
|
|
if(nChoice == CHOICE_BACK_TO_LSELECT)
|
|
{
|
|
if(DEBUG) DoDebug("shd_mystconv: Returning to level selection");
|
|
nStage = STAGE_SELECT_LEVEL;
|
|
// Clean up
|
|
DeleteLocalInt(oPC, "nMystLevelToBrowse");
|
|
}
|
|
else
|
|
{
|
|
if(DEBUG) DoDebug("shd_mystconv: Entering mystery confirmation");
|
|
SetLocalInt(oPC, "nMyst", nChoice);
|
|
// Store offset so that if the user decides not to take the mystery,
|
|
// we can return to the same page in the mystery list instead of resetting to the beginning
|
|
// Store the value +1 in order to be able to differentiate between offset 0 and undefined
|
|
SetLocalInt(oPC, "MYSTLISTChoiceOffset", GetLocalInt(oPC, DYNCONV_CHOICEOFFSET) + 1);
|
|
nStage = STAGE_CONFIRM_SELECTION;
|
|
}
|
|
MarkStageNotSetUp(STAGE_SELECT_MYSTERY, oPC);
|
|
}
|
|
else if(nStage == STAGE_CONFIRM_SELECTION)
|
|
{
|
|
if(DEBUG) DoDebug("shd_mystconv: Handling mystery confirmation");
|
|
if(nChoice == TRUE)
|
|
{
|
|
if(DEBUG) DoDebug("shd_mystconv: Adding power");
|
|
int nMyst = GetLocalInt(oPC, "nMyst");
|
|
|
|
AddMysteryKnown(oPC, nClass, nMyst, GetHitDice(oPC));
|
|
|
|
// Delete the stored offset
|
|
DeleteLocalInt(oPC, "MYSTLISTChoiceOffset");
|
|
}
|
|
|
|
int nMyst = GetMysteryCount(oPC, nClass);
|
|
int nMaxMyst = GetMaxMysteryCount(oPC, nClass);
|
|
|
|
if(DEBUG)
|
|
{
|
|
DoDebug("shd_mystconv: Checking for being full");
|
|
DoDebug("shd_mystconv: Mystery Count: " + IntToString(nMyst));
|
|
DoDebug("shd_mystconv: Mystery Max: " + IntToString(nMaxMyst));
|
|
}
|
|
|
|
//Check all three lexicons for being full
|
|
if(nMyst >= nMaxMyst)
|
|
nStage = STAGE_ALL_MYSTERIES_SELECTED;
|
|
else
|
|
nStage = STAGE_SELECT_LEVEL;
|
|
}
|
|
|
|
if(DEBUG) DoDebug("shd_mystconv: New stage: " + IntToString(nStage));
|
|
|
|
// Store the stage value. If it has been changed, this clears out the choices
|
|
SetStage(nStage, oPC);
|
|
}
|
|
}
|