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.
2913 lines
108 KiB
Plaintext
2913 lines
108 KiB
Plaintext
//::///////////////////////////////////////////////
|
|
//:: prc_x2_craft
|
|
//:: Copyright (c) 2003 Bioare Corp.
|
|
//:://////////////////////////////////////////////
|
|
/*
|
|
|
|
Central include for crafting feat and
|
|
crafting skill system.
|
|
|
|
*/
|
|
//:://////////////////////////////////////////////
|
|
//:: Created By: Georg Zoeller
|
|
//:: Created On: 2003-05-09
|
|
//:: Last Updated On: 2003-10-14
|
|
//:://////////////////////////////////////////////
|
|
|
|
|
|
struct craft_struct
|
|
{
|
|
int nRow;
|
|
string sResRef;
|
|
int nDC;
|
|
int nCost;
|
|
string sLabel;
|
|
};
|
|
|
|
struct craft_receipe_struct
|
|
{
|
|
int nMode;
|
|
object oMajor;
|
|
object oMinor;
|
|
};
|
|
|
|
struct craft_cost_struct
|
|
{
|
|
int nGoldCost;
|
|
int nXPCost;
|
|
int nTimeCost;
|
|
};
|
|
|
|
const string X2_CI_CRAFTSKILL_CONV ="x2_p_craftskills";
|
|
|
|
// Brew Potion related Constants
|
|
/* moved to be code switches
|
|
|
|
const int X2_CI_BREWPOTION_MAXLEVEL = 3; // Max Level for potions
|
|
const int X2_CI_BREWPOTION_COSTMODIFIER = 50; // gp Brew Potion XPCost Modifier
|
|
|
|
// Scribe Scroll related constants
|
|
const int X2_CI_SCRIBESCROLL_COSTMODIFIER = 25; // Scribescroll Cost Modifier
|
|
|
|
// Craft Wand related constants
|
|
const int X2_CI_CRAFTWAND_MAXLEVEL = 4;
|
|
const int X2_CI_CRAFTWAND_COSTMODIFIER = 750;
|
|
*/
|
|
const int X2_CI_BREWPOTION_FEAT_ID = 944; // Brew Potion feat simulation
|
|
const int X2_CI_SCRIBESCROLL_FEAT_ID = 945;
|
|
const int X2_CI_CRAFTWAND_FEAT_ID = 946;
|
|
const int X2_CI_CRAFTROD_FEAT_ID = 2927;
|
|
const int X2_CI_CRAFTROD_EPIC_FEAT_ID = 3490;
|
|
const int X2_CI_CRAFTSTAFF_FEAT_ID = 2928;
|
|
const int X2_CI_CRAFTSTAFF_EPIC_FEAT_ID = 3491;
|
|
const string X2_CI_BREWPOTION_NEWITEM_RESREF = "x2_it_pcpotion"; // ResRef for new potion item
|
|
const string X2_CI_SCRIBESCROLL_NEWITEM_RESREF = "x2_it_pcscroll"; // ResRef for new scroll item
|
|
const string X2_CI_CRAFTWAND_NEWITEM_RESREF = "x2_it_pcwand";
|
|
//const string X2_CI_CRAFTROD_NEWITEM_RESREF = "x2_it_pcwand";
|
|
//const string X2_CI_CRAFTSTAFF_NEWITEM_RESREF = "x2_it_pcwand";
|
|
|
|
// 2da for the craftskills
|
|
const string X2_CI_CRAFTING_WP_2DA = "des_crft_weapon" ;
|
|
const string X2_CI_CRAFTING_AR_2DA = "des_crft_armor" ;
|
|
const string X2_CI_CRAFTING_MAT_2DA = "des_crft_mat";
|
|
|
|
|
|
// 2da for matching spells to properties
|
|
const string X2_CI_CRAFTING_SP_2DA = "des_crft_spells" ;
|
|
// Base custom token for item modification conversations (do not change unless you want to change the conversation too)
|
|
const int X2_CI_CRAFTINGSKILL_CTOKENBASE = 13220;
|
|
|
|
// Base custom token for DC item modification conversations (do not change unless you want to change the conversation too)
|
|
const int X2_CI_CRAFTINGSKILL_DC_CTOKENBASE = 14220;
|
|
|
|
// Base custom token for DC item modification conversations (do not change unless you want to change the conversation too)
|
|
const int X2_CI_CRAFTINGSKILL_GP_CTOKENBASE = 14320;
|
|
|
|
// Base custom token for DC item modification conversations (do not change unless you want to change the conversation too)
|
|
const int X2_CI_MODIFYARMOR_GP_CTOKENBASE = 14420;
|
|
|
|
//How many items per 2da row in X2_IP_CRAFTING_2DA, do not change>4 until you want to create more conversation condition scripts as well
|
|
const int X2_CI_CRAFTING_ITEMS_PER_ROW = 5;
|
|
|
|
// name of the scroll 2da
|
|
const string X2_CI_2DA_SCROLLS = "des_crft_scroll";
|
|
|
|
const int X2_CI_CRAFTMODE_INVALID = 0;
|
|
const int X2_CI_CRAFTMODE_CONTAINER = 1; // no longer used, but left in for the community to reactivate
|
|
const int X2_CI_CRAFTMODE_BASE_ITEM = 2;
|
|
const int X2_CI_CRAFTMODE_ASSEMBLE = 3;
|
|
|
|
const int X2_CI_MAGICTYPE_INVALID = 0;
|
|
const int X2_CI_MAGICTYPE_ARCANE = 1;
|
|
const int X2_CI_MAGICTYPE_DIVINE = 2;
|
|
|
|
const int X2_CI_MODMODE_INVALID = 0;
|
|
const int X2_CI_MODMODE_ARMOR = 1;
|
|
const int X2_CI_MODMODE_WEAPON = 2;
|
|
|
|
// Runecrafting constants
|
|
const int PRC_RUNE_BASECOST = 0;
|
|
const int PRC_RUNE_CHARGES = 1;
|
|
const int PRC_RUNE_PERDAY = 2;
|
|
const int PRC_RUNE_MAXCHARGES = 3;
|
|
const int PRC_RUNE_MAXUSESPERDAY = 4;
|
|
// Attune Gem constants
|
|
const int PRC_GEM_BASECOST = 5;
|
|
const int PRC_GEM_PERLEVEL = 6;
|
|
// Craft Skull Talisman constants
|
|
const int PRC_SKULL_BASECOST = 7;
|
|
|
|
// * Returns TRUE if an item is a Craft Base Item
|
|
// * to be used in spellscript that can be cast on items - i.e light
|
|
int CIGetIsCraftFeatBaseItem( object oItem );
|
|
|
|
// * Checks if the last spell cast was used to brew potion and will do the brewing process.
|
|
// * Returns TRUE if the spell was indeed used to brew a potion (regardless of the actual outcome of the brewing process)
|
|
// * Meant to be used in spellscripts only
|
|
int CICraftCheckBrewPotion(object oSpellTarget, object oCaster, int nID = 0);
|
|
|
|
// * Checks if the last spell cast was used to scribe a scroll and handles the scribe scroll process
|
|
// * Returns TRUE if the spell was indeed used to scribe a scroll (regardless of the actual outcome)
|
|
// * Meant to be used in spellscripts only
|
|
int CICraftCheckScribeScroll(object oSpellTarget, object oCaster, int nID = 0);
|
|
|
|
// * Create a new potion item based on the spell nSpellID on the creator
|
|
object CICraftBrewPotion(object oCreator, int nSpellID );
|
|
|
|
// * Create a new scroll item based on the spell nSpellID on the creator
|
|
object CICraftScribeScroll(object oCreator, int nSpellID);
|
|
|
|
|
|
// * Checks if the caster intends to use his item creation feats and
|
|
// * calls appropriate item creation subroutine if conditions are met (spell cast on correct item, etc).
|
|
// * Returns TRUE if the spell was used for an item creation feat
|
|
int CIGetSpellWasUsedForItemCreation(object oSpellTarget);
|
|
|
|
// This function checks whether Inscribe Rune is turned on
|
|
// and if so, deducts the appropriate experience and gold
|
|
// then creates the rune in the caster's inventory.
|
|
// This will also cause the spell to fail if turned on.
|
|
int InscribeRune(object oTarget = OBJECT_INVALID, object oCaster = OBJECT_INVALID, int nSpell = 0);
|
|
|
|
// This function checks whether Attune Gem is turned on
|
|
// and if so, deducts the appropriate experience and gold
|
|
// then creates the gem in the caster's inventory.
|
|
// This will also cause the spell to fail if turned on.
|
|
int AttuneGem(object oTarget = OBJECT_INVALID, object oCaster = OBJECT_INVALID, int nSpell = 0);
|
|
|
|
// Gets the Magical Artisan feat given a particular crafting feat
|
|
int GetMagicalArtisanFeat(int nCraftingFeat);
|
|
|
|
// Gets the modified gold cost taking cost reduction feats and cost
|
|
// scaling switches into account
|
|
int GetModifiedGoldCost(int nCost, object oPC, int nCraftingFeat);
|
|
|
|
// Gets the modified xp cost taking cost reduction feats and cost
|
|
// scaling switches into account
|
|
int GetModifiedXPCost(int nCost, object oPC, int nCraftingFeat);
|
|
|
|
// Gets the modified time cost taking cost reduction feats and cost
|
|
// scaling switches into account
|
|
int GetModifiedTimeCost(int nCost, object oPC, int nCraftingFeat);
|
|
|
|
// Imbue item check for warlocks, returns TRUE if a spell requirement is met
|
|
int CheckImbueItem(object oPC, int nSpell);
|
|
|
|
// Gets PnP xp cost given a gold cost and whether the item is epic
|
|
int GetPnPItemXPCost(int nCost, int bEpic);
|
|
|
|
// Returns a struct containing gold, xp and time costs given the base cost and other arguments
|
|
struct craft_cost_struct GetModifiedCostsFromBase(int nCost, object oPC, int nCraftingFeat, int bEpic);
|
|
|
|
// Additional checking for emulating spells during crafting
|
|
int CheckAlternativeCrafting(object oPC, int nSpell, struct craft_cost_struct costs);
|
|
|
|
// Returns the maximum of caster level used and other effective levels from emulating spells
|
|
int GetAlternativeCasterLevel(object oPC, int nLevel);
|
|
|
|
//////////////////////////////////////////////////
|
|
/* Include section */
|
|
//////////////////////////////////////////////////
|
|
|
|
//#include "prc_x2_itemprop"
|
|
//#include "x2_inc_switches"
|
|
#include "prc_inc_newip"
|
|
#include "prc_inc_spells"
|
|
#include "prc_add_spell_dc"
|
|
|
|
//////////////////////////////////////////////////
|
|
/* Function definitions */
|
|
//////////////////////////////////////////////////
|
|
|
|
|
|
// * Returns the innate level of a spell. If bDefaultZeroToOne is given
|
|
// * Level 0 spell will be returned as level 1 spells
|
|
int CIGetSpellInnateLevel(int nSpellID, int bDefaultZeroToOne = FALSE)
|
|
{
|
|
int nRet = StringToInt(Get2DACache(X2_CI_CRAFTING_SP_2DA, "Level", nSpellID));
|
|
if (nRet == 0 && bDefaultZeroToOne == TRUE) // Was missing the "bDefaultZeroToOne == TRUE" check, fixed to match specification - Ornedan
|
|
nRet = 1;
|
|
|
|
return nRet;
|
|
}
|
|
|
|
// * Makes oPC do a Craft check using nSkill to create the item supplied in sResRe
|
|
// * If oContainer is specified, the item will be created there.
|
|
// * Throwing weapons are created with stack sizes of 10, ammo with 20
|
|
// * oPC - The player crafting
|
|
// * nSkill - SKILL_CRAFT_WEAPON or SKILL_CRAFT_ARMOR,
|
|
// * sResRef - ResRef of the item to be crafted
|
|
// * nDC - DC to beat to succeed
|
|
// * oContainer - if a container is specified, create item inside
|
|
object CIUseCraftItemSkill(object oPC, int nSkill, string sResRef, int nDC, object oContainer = OBJECT_INVALID);
|
|
|
|
// * Returns TRUE if a spell is prevented from being used with one of the crafting feats
|
|
int CIGetIsSpellRestrictedFromCraftFeat(int nSpellID, int nFeatID);
|
|
|
|
// * Return craftitemstructdata
|
|
struct craft_struct CIGetCraftItemStructFrom2DA(string s2DA, int nRow, int nItemNo);
|
|
|
|
// * Return the type of magic as one of the following constants
|
|
// * const int X2_CI_MAGICTYPE_INVALID = 0;
|
|
// * const int X2_CI_MAGICTYPE_ARCANE = 1;
|
|
// * const int X2_CI_MAGICTYPE_DIVINE = 2;
|
|
// * Parameters:
|
|
// * nClass - CLASS_TYPE_* constant
|
|
int CI_GetClassMagicType(int nClass)
|
|
{
|
|
if(GetIsArcaneClass(nClass))
|
|
return X2_CI_MAGICTYPE_ARCANE;
|
|
else if(GetIsDivineClass(nClass))
|
|
return X2_CI_MAGICTYPE_DIVINE;
|
|
|
|
return X2_CI_MAGICTYPE_INVALID;
|
|
}
|
|
|
|
string GetMaterialComponentTag(int nPropID)
|
|
{
|
|
string sRet = Get2DACache("des_matcomp","comp_tag",nPropID);
|
|
return sRet;
|
|
}
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Return true if oItem is a crafting target item
|
|
// -----------------------------------------------------------------------------
|
|
int CIGetIsCraftFeatBaseItem(object oItem)
|
|
{
|
|
int nBt = GetBaseItemType(oItem);
|
|
// blank scroll, empty potion, wand
|
|
if (nBt == BASE_ITEM_BLANK_POTION ||
|
|
nBt == BASE_ITEM_BLANK_SCROLL ||
|
|
nBt == BASE_ITEM_BLANK_WAND ||
|
|
nBt == BASE_ITEM_CRAFTED_ROD ||
|
|
nBt == BASE_ITEM_CRAFTED_STAFF)
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Georg, 2003-06-12
|
|
// Create a new playermade potion object with properties matching nSpellID and return it
|
|
// -----------------------------------------------------------------------------
|
|
object CICraftBrewPotion(object oCreator, int nSpellID )
|
|
{
|
|
|
|
int nPropID = IPGetIPConstCastSpellFromSpellID(nSpellID);
|
|
|
|
object oTarget;
|
|
// * GZ 2003-09-11: If the current spell cast is not acid fog, and
|
|
// * returned property ID is 0, bail out to prevent
|
|
// * creation of acid fog items.
|
|
if (nPropID == 0 && nSpellID != 0)
|
|
{
|
|
FloatingTextStrRefOnCreature(84544,oCreator);
|
|
return OBJECT_INVALID;
|
|
}
|
|
|
|
/* //just a tad retarded, don't you think? other crafting feats are not similarly restricted
|
|
//Uses per day
|
|
int nUsesAllowed;
|
|
|
|
if(GetHasFeat(FEAT_BREW_4PERDAY,oCreator)) nUsesAllowed = 4;
|
|
|
|
else if(GetHasFeat(FEAT_BREW_3PERDAY, oCreator)) nUsesAllowed = 3;
|
|
|
|
else if(GetHasFeat(FEAT_BREW_2PERDAY, oCreator)) nUsesAllowed = 2;
|
|
|
|
else nUsesAllowed = 1;
|
|
|
|
int nUsed = GetLocalInt(oCreator, "PRC_POTIONS_BREWED");
|
|
|
|
if(nUsed >= nUsesAllowed)
|
|
{
|
|
SendMessageToPC(oCreator, "You must rest before you can brew any more potions");
|
|
return OBJECT_INVALID;
|
|
}
|
|
*/
|
|
|
|
int nCasterLevel = GetAlternativeCasterLevel(oCreator, PRCGetCasterLevel(oCreator));
|
|
|
|
if (nPropID != -1)
|
|
{
|
|
itemproperty ipProp = ItemPropertyCastSpell(nPropID,IP_CONST_CASTSPELL_NUMUSES_SINGLE_USE);
|
|
oTarget = CreateItemOnObject(X2_CI_BREWPOTION_NEWITEM_RESREF,oCreator);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipProp,oTarget);
|
|
if(GetPRCSwitch(PRC_BREW_POTION_CASTER_LEVEL))
|
|
{
|
|
itemproperty ipLevel = ItemPropertyCastSpellCasterLevel(nSpellID, nCasterLevel);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipLevel,oTarget);
|
|
itemproperty ipMeta = ItemPropertyCastSpellMetamagic(nSpellID, PRCGetMetaMagicFeat());
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipMeta,oTarget);
|
|
itemproperty ipDC = ItemPropertyCastSpellDC(nSpellID, PRCGetSaveDC(PRCGetSpellTargetObject(), OBJECT_SELF));
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipDC,oTarget);
|
|
}
|
|
|
|
//Increment usage
|
|
//SetLocalInt(oCreator, "PRC_POTIONS_BREWED", nUsed++);
|
|
}
|
|
return oTarget;
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Wrapper for the crafting cost calculation, returns GP required
|
|
// -----------------------------------------------------------------------------
|
|
int CIGetCraftGPCost(int nLevel, int nMod, string sCasterLevelSwitch)
|
|
{
|
|
int nLvlRow = IPGetIPConstCastSpellFromSpellID(PRCGetSpellId());
|
|
int nCLevel;// = StringToInt(Get2DACache("iprp_spells","CasterLvl",nLvlRow));
|
|
//PRC modification
|
|
if(GetPRCSwitch(sCasterLevelSwitch))
|
|
{
|
|
nCLevel = PRCGetCasterLevel();
|
|
}
|
|
else
|
|
{
|
|
nCLevel = StringToInt(Get2DACache("iprp_spells","CasterLvl",nLvlRow));
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// in case we don't get a valid CLevel, use spell level instead
|
|
// -------------------------------------------------------------------------
|
|
if (nCLevel ==0)
|
|
{
|
|
nCLevel = nLevel;
|
|
}
|
|
int nRet = nCLevel * nLevel * nMod;
|
|
return nRet;
|
|
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Georg, 2003-06-12
|
|
// Create a new playermade wand object with properties matching nSpellID
|
|
// and return it
|
|
// -----------------------------------------------------------------------------
|
|
object CICraftCraftWand(object oCreator, int nSpellID )
|
|
{
|
|
int nPropID = IPGetIPConstCastSpellFromSpellID(nSpellID);
|
|
|
|
object oTarget;
|
|
// * GZ 2003-09-11: If the current spell cast is not acid fog, and
|
|
// * returned property ID is 0, bail out to prevent
|
|
// * creation of acid fog items.
|
|
if (nPropID == 0 && nSpellID != 0)
|
|
{
|
|
FloatingTextStrRefOnCreature(84544,oCreator);
|
|
return OBJECT_INVALID;
|
|
}
|
|
|
|
|
|
//int nClass = PRCGetLastSpellCastClass();
|
|
int nCasterLevel = GetAlternativeCasterLevel(oCreator, PRCGetCasterLevel(oCreator));
|
|
|
|
if (nPropID != -1)
|
|
{
|
|
itemproperty ipProp = ItemPropertyCastSpell(nPropID,IP_CONST_CASTSPELL_NUMUSES_1_CHARGE_PER_USE);
|
|
oTarget = CreateItemOnObject(X2_CI_CRAFTWAND_NEWITEM_RESREF,oCreator);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipProp,oTarget);
|
|
|
|
if(GetPRCSwitch(PRC_CRAFT_WAND_CASTER_LEVEL))
|
|
{
|
|
itemproperty ipLevel = ItemPropertyCastSpellCasterLevel(nSpellID, nCasterLevel);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipLevel,oTarget);
|
|
itemproperty ipMeta = ItemPropertyCastSpellMetamagic(nSpellID, PRCGetMetaMagicFeat());
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipMeta,oTarget);
|
|
itemproperty ipDC = ItemPropertyCastSpellDC(nSpellID, PRCGetSaveDC(PRCGetSpellTargetObject(), OBJECT_SELF));
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipDC,oTarget);
|
|
}
|
|
|
|
//int nType = CI_GetClassMagicType(nClass);
|
|
//itemproperty ipLimit;
|
|
|
|
/* //this is a bit silly, really, removed in line with other crafting types
|
|
if (nType == X2_CI_MAGICTYPE_DIVINE)
|
|
{
|
|
ipLimit = ItemPropertyLimitUseByClass(CLASS_TYPE_PALADIN);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipLimit,oTarget);
|
|
ipLimit = ItemPropertyLimitUseByClass(CLASS_TYPE_RANGER);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipLimit,oTarget);
|
|
ipLimit = ItemPropertyLimitUseByClass(CLASS_TYPE_DRUID);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipLimit,oTarget);
|
|
ipLimit = ItemPropertyLimitUseByClass(CLASS_TYPE_CLERIC);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipLimit,oTarget);
|
|
}
|
|
else if (nType == X2_CI_MAGICTYPE_ARCANE)
|
|
{
|
|
ipLimit = ItemPropertyLimitUseByClass(CLASS_TYPE_WIZARD);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipLimit,oTarget);
|
|
ipLimit = ItemPropertyLimitUseByClass(CLASS_TYPE_SORCERER);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipLimit,oTarget);
|
|
ipLimit = ItemPropertyLimitUseByClass(CLASS_TYPE_BARD);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipLimit,oTarget);
|
|
}
|
|
|
|
if(nClass != CLASS_TYPE_WARLOCK)
|
|
{
|
|
ipLimit = ItemPropertyLimitUseByClass(nClass);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipLimit,oTarget);
|
|
}
|
|
*/
|
|
|
|
int nCharges = nCasterLevel + d20();
|
|
|
|
if (nCharges == 0) // stupi cheaters
|
|
{
|
|
nCharges = 10+d20();
|
|
}
|
|
// Hard core rule mode enabled
|
|
if (GetModuleSwitchValue(MODULE_SWITCH_ENABLE_CRAFT_WAND_50_CHARGES))
|
|
{
|
|
SetItemCharges(oTarget,50);
|
|
}
|
|
else
|
|
{
|
|
SetItemCharges(oTarget,nCharges);
|
|
}
|
|
// TODOL Add use restrictions there when item becomes available
|
|
}
|
|
return oTarget;
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Georg, 2003-06-12
|
|
// Create and Return a magic wand with an item property
|
|
// matching nSpellID. Charges are set to d20 + casterlevel
|
|
// capped at 50 max
|
|
// -----------------------------------------------------------------------------
|
|
object CICraftScribeScroll(object oCreator, int nSpellID)
|
|
{
|
|
int nPropID = IPGetIPConstCastSpellFromSpellID(nSpellID);
|
|
object oTarget;
|
|
// Handle optional material components
|
|
string sMat = GetMaterialComponentTag(nPropID);
|
|
if (sMat != "")
|
|
{
|
|
object oMat = GetItemPossessedBy(oCreator,sMat);
|
|
if (oMat== OBJECT_INVALID)
|
|
{
|
|
FloatingTextStrRefOnCreature(83374, oCreator); // Missing material component
|
|
return OBJECT_INVALID;
|
|
}
|
|
else
|
|
{
|
|
DestroyObject (oMat);
|
|
}
|
|
}
|
|
|
|
// get scroll resref from scrolls lookup 2da
|
|
int nClass =PRCGetLastSpellCastClass ();
|
|
string sClass = "";
|
|
switch (nClass)
|
|
{
|
|
case CLASS_TYPE_WIZARD:
|
|
sClass = "Wiz_Sorc";
|
|
break;
|
|
|
|
case CLASS_TYPE_SORCERER:
|
|
sClass = "Wiz_Sorc";
|
|
break;
|
|
case CLASS_TYPE_CLERIC:
|
|
case CLASS_TYPE_UR_PRIEST:
|
|
sClass = "Cleric";
|
|
break;
|
|
case CLASS_TYPE_PALADIN:
|
|
sClass = "Paladin";
|
|
break;
|
|
case CLASS_TYPE_DRUID:
|
|
case CLASS_TYPE_BLIGHTER:
|
|
sClass = "Druid";
|
|
break;
|
|
case CLASS_TYPE_RANGER:
|
|
sClass = "Ranger";
|
|
break;
|
|
case CLASS_TYPE_BARD:
|
|
sClass = "Bard";
|
|
break;
|
|
}
|
|
string sResRef;
|
|
if (sClass != "")
|
|
{
|
|
sResRef = Get2DACache(X2_CI_2DA_SCROLLS,sClass,nSpellID);
|
|
if (sResRef != "")
|
|
{
|
|
oTarget = CreateItemOnObject(sResRef,oCreator);
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
sResRef = "craft_scroll";
|
|
oTarget = CreateItemOnObject(sResRef ,oCreator);
|
|
RemoveItemProperty(oTarget, GetFirstItemProperty(oTarget));
|
|
itemproperty ipSpell = ItemPropertyCastSpell(nPropID, IP_CONST_CASTSPELL_NUMUSES_SINGLE_USE);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipSpell,oTarget);
|
|
}
|
|
if(GetPRCSwitch(PRC_SCRIBE_SCROLL_CASTER_LEVEL))
|
|
{
|
|
int nCasterLevel = GetAlternativeCasterLevel(oCreator, PRCGetCasterLevel(oCreator));
|
|
itemproperty ipLevel = ItemPropertyCastSpellCasterLevel(nSpellID, nCasterLevel);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipLevel,oTarget);
|
|
itemproperty ipMeta = ItemPropertyCastSpellMetamagic(nSpellID, PRCGetMetaMagicFeat());
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipMeta,oTarget);
|
|
itemproperty ipDC = ItemPropertyCastSpellDC(nSpellID, PRCGetSaveDC(PRCGetSpellTargetObject(), OBJECT_SELF));
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipDC,oTarget);
|
|
}
|
|
|
|
if (oTarget == OBJECT_INVALID)
|
|
{
|
|
WriteTimestampedLogEntry("prc_x2_craft::CICraftScribeScroll failed - Resref: " + sResRef + " Class: " + sClass + "(" +IntToString(nClass) +") " + " SpellID " + IntToString (nSpellID));
|
|
}
|
|
return oTarget;
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Returns TRUE if the player used the last spell to brew a potion
|
|
// -----------------------------------------------------------------------------
|
|
int CICraftCheckBrewPotion(object oSpellTarget, object oCaster, int nID = 0)
|
|
{
|
|
|
|
if(nID == 0) nID = PRCGetSpellId();
|
|
|
|
object oSpellTarget = PRCGetSpellTargetObject();
|
|
object oCaster = OBJECT_SELF;
|
|
int nLevel = CIGetSpellInnateLevel(nID,TRUE);
|
|
if(GetPRCSwitch(PRC_BREW_POTION_CASTER_LEVEL))
|
|
{
|
|
int nMetaMagic = PRCGetMetaMagicFeat();
|
|
switch(nMetaMagic)
|
|
{
|
|
case METAMAGIC_EMPOWER:
|
|
nLevel += 2;
|
|
break;
|
|
case METAMAGIC_EXTEND:
|
|
nLevel += 1;
|
|
break;
|
|
case METAMAGIC_MAXIMIZE:
|
|
nLevel += 3;
|
|
break;
|
|
/* case METAMAGIC_QUICKEN:
|
|
nLevel += 1;
|
|
break;
|
|
case METAMAGIC_SILENT:
|
|
nLevel += 5;
|
|
break;
|
|
case METAMAGIC_STILL:
|
|
nLevel += 6;
|
|
break;
|
|
These dont work as IPs since they are hardcoded */
|
|
}
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// check if brew potion feat is there
|
|
// -------------------------------------------------------------------------
|
|
if (GetHasFeat(X2_CI_BREWPOTION_FEAT_ID, oCaster) != TRUE)
|
|
{
|
|
FloatingTextStrRefOnCreature(40487, oCaster); // Item Creation Failed - Don't know how to create that type of item
|
|
return TRUE;
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// check if spell is below maxlevel for brew potions
|
|
// -------------------------------------------------------------------------
|
|
int nPotionMaxLevel = GetPRCSwitch(X2_CI_BREWPOTION_MAXLEVEL);
|
|
if(nPotionMaxLevel == 0)
|
|
nPotionMaxLevel = 3;
|
|
|
|
//Master Alchemist
|
|
|
|
if(GetHasFeat(FEAT_BREW_POTION_9TH, oCaster)) nPotionMaxLevel = 9;
|
|
else if(GetHasFeat(FEAT_BREW_POTION_8TH, oCaster)) nPotionMaxLevel = 8;
|
|
else if(GetHasFeat(FEAT_BREW_POTION_7TH, oCaster)) nPotionMaxLevel = 7;
|
|
else if(GetHasFeat(FEAT_BREW_POTION_6TH, oCaster)) nPotionMaxLevel = 6;
|
|
else if(GetHasFeat(FEAT_BREW_POTION_5TH, oCaster)) nPotionMaxLevel = 5;
|
|
else if(GetHasFeat(FEAT_BREW_POTION_4TH, oCaster)) nPotionMaxLevel = 4;
|
|
|
|
if (nLevel > nPotionMaxLevel)
|
|
{
|
|
FloatingTextStrRefOnCreature(76416, oCaster);
|
|
return TRUE;
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Check if the spell is allowed to be used with Brew Potions
|
|
// -------------------------------------------------------------------------
|
|
if (CIGetIsSpellRestrictedFromCraftFeat(nID, X2_CI_BREWPOTION_FEAT_ID))
|
|
{
|
|
FloatingTextStrRefOnCreature(83450, oCaster);
|
|
return TRUE;
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// XP/GP Cost Calculation
|
|
// -------------------------------------------------------------------------
|
|
int nCostModifier = GetPRCSwitch(X2_CI_BREWPOTION_COSTMODIFIER);
|
|
if(nCostModifier == 0)
|
|
nCostModifier = 50;
|
|
int nCost = CIGetCraftGPCost(nLevel, nCostModifier, PRC_BREW_POTION_CASTER_LEVEL);
|
|
|
|
struct craft_cost_struct costs = GetModifiedCostsFromBase(nCost, oCaster, FEAT_BREW_POTION, FALSE);
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Does Player have enough gold?
|
|
// -------------------------------------------------------------------------
|
|
//if (GetGold(oCaster) < nGoldCost)
|
|
if(!GetHasGPToSpend(oCaster, costs.nGoldCost))
|
|
{
|
|
FloatingTextStrRefOnCreature(3786, oCaster); // Item Creation Failed - not enough gold!
|
|
return TRUE;
|
|
}
|
|
|
|
int nHD = GetHitDice(oCaster);
|
|
int nMinXPForLevel = ((nHD * (nHD - 1)) / 2) * 1000;
|
|
int nNewXP = GetXP(oCaster) - costs.nXPCost;
|
|
|
|
|
|
// -------------------------------------------------------------------------
|
|
// check for sufficient XP to cast spell
|
|
// -------------------------------------------------------------------------
|
|
//if (nMinXPForLevel > nNewXP || nNewXP == 0 )
|
|
if (!GetHasXPToSpend(oCaster, costs.nXPCost))
|
|
{
|
|
FloatingTextStrRefOnCreature(3785, oCaster); // Item Creation Failed - Not enough XP
|
|
return TRUE;
|
|
}
|
|
|
|
//check spell emulation
|
|
if(!CheckAlternativeCrafting(oCaster, nID, costs))
|
|
{
|
|
FloatingTextStringOnCreature("*Crafting failed!*", oCaster, FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Here we brew the new potion
|
|
// -------------------------------------------------------------------------
|
|
object oPotion = CICraftBrewPotion(oCaster, nID);
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Verify Results
|
|
// -------------------------------------------------------------------------
|
|
if (GetIsObjectValid(oPotion))
|
|
{
|
|
//TakeGoldFromCreature(nGoldCost, oCaster, TRUE);
|
|
//SetXP(oCaster, nNewXP);
|
|
SpendXP(oCaster, costs.nXPCost);
|
|
SpendGP(oCaster, costs.nGoldCost);
|
|
DestroyObject (oSpellTarget);
|
|
FloatingTextStrRefOnCreature(8502, oCaster); // Item Creation successful
|
|
|
|
//advance time here
|
|
if(!costs.nTimeCost) costs.nTimeCost = 1;
|
|
AdvanceTimeForPlayer(oCaster, RoundsToSeconds(costs.nTimeCost));
|
|
string sName;
|
|
sName = Get2DACache("spells", "Name", nID);
|
|
sName = "Potion of "+GetStringByStrRef(StringToInt(sName));
|
|
SetName(oPotion, sName);
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
FloatingTextStrRefOnCreature(76417, oCaster); // Item Creation Failed
|
|
return TRUE;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Returns TRUE if the player used the last spell to create a scroll
|
|
// -----------------------------------------------------------------------------
|
|
int CICraftCheckScribeScroll(object oSpellTarget, object oCaster, int nID = 0)
|
|
{
|
|
if(nID == 0) nID = PRCGetSpellId();
|
|
|
|
// -------------------------------------------------------------------------
|
|
// check if scribe scroll feat is there
|
|
// -------------------------------------------------------------------------
|
|
if (GetHasFeat(X2_CI_SCRIBESCROLL_FEAT_ID, oCaster) != TRUE)
|
|
{
|
|
FloatingTextStrRefOnCreature(40487, oCaster); // Item Creation Failed - Don't know how to create that type of item
|
|
return TRUE;
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Check if the spell is allowed to be used with Scribe Scroll
|
|
// -------------------------------------------------------------------------
|
|
if (CIGetIsSpellRestrictedFromCraftFeat(nID, X2_CI_SCRIBESCROLL_FEAT_ID))
|
|
{
|
|
FloatingTextStrRefOnCreature(83451, oCaster); // can not be used with this feat
|
|
return TRUE;
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// XP/GP Cost Calculation
|
|
// -------------------------------------------------------------------------
|
|
int nLevel = CIGetSpellInnateLevel(nID,TRUE);
|
|
int nCostModifier = GetPRCSwitch(X2_CI_SCRIBESCROLL_COSTMODIFIER);
|
|
if(nCostModifier == 0)
|
|
nCostModifier = 25;
|
|
int nCost = CIGetCraftGPCost(nLevel, nCostModifier, PRC_SCRIBE_SCROLL_CASTER_LEVEL);
|
|
|
|
struct craft_cost_struct costs = GetModifiedCostsFromBase(nCost, oCaster, FEAT_SCRIBE_SCROLL, FALSE);
|
|
|
|
if(GetPRCSwitch(PRC_SCRIBE_SCROLL_CASTER_LEVEL))
|
|
{
|
|
int nMetaMagic = PRCGetMetaMagicFeat();
|
|
switch(nMetaMagic)
|
|
{
|
|
case METAMAGIC_EMPOWER:
|
|
nLevel += 2;
|
|
break;
|
|
case METAMAGIC_EXTEND:
|
|
nLevel += 1;
|
|
break;
|
|
case METAMAGIC_MAXIMIZE:
|
|
nLevel += 3;
|
|
break;
|
|
/* case METAMAGIC_QUICKEN:
|
|
nLevel += 1;
|
|
break;
|
|
case METAMAGIC_SILENT:
|
|
nLevel += 5;
|
|
break;
|
|
case METAMAGIC_STILL:
|
|
nLevel += 6;
|
|
break;
|
|
These dont work as IPs since they are hardcoded */
|
|
}
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Does Player have enough gold?
|
|
// -------------------------------------------------------------------------
|
|
if(!GetHasGPToSpend(oCaster, costs.nGoldCost))
|
|
{
|
|
FloatingTextStrRefOnCreature(3786, oCaster); // Item Creation Failed - not enough gold!
|
|
return TRUE;
|
|
}
|
|
|
|
int nHD = GetHitDice(oCaster);
|
|
int nMinXPForLevel = ((nHD * (nHD - 1)) / 2) * 1000;
|
|
int nNewXP = GetXP(oCaster) - costs.nXPCost;
|
|
|
|
// -------------------------------------------------------------------------
|
|
// check for sufficient XP to cast spell
|
|
// -------------------------------------------------------------------------
|
|
//if (nMinXPForLevel > nNewXP || nNewXP == 0 )
|
|
if (!GetHasXPToSpend(oCaster, costs.nXPCost))
|
|
{
|
|
FloatingTextStrRefOnCreature(3785, oCaster); // Item Creation Failed - Not enough XP
|
|
return TRUE;
|
|
}
|
|
|
|
//check spell emulation
|
|
if(!CheckAlternativeCrafting(oCaster, nID, costs))
|
|
{
|
|
FloatingTextStringOnCreature("*Crafting failed!*", oCaster, FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Here we scribe the scroll
|
|
// -------------------------------------------------------------------------
|
|
object oScroll = CICraftScribeScroll(oCaster, nID);
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Verify Results
|
|
// -------------------------------------------------------------------------
|
|
if (GetIsObjectValid(oScroll))
|
|
{
|
|
//----------------------------------------------------------------------
|
|
// Some scrollsare ar not identified ... fix that here
|
|
//----------------------------------------------------------------------
|
|
SetIdentified(oScroll,TRUE);
|
|
ActionPlayAnimation (ANIMATION_FIREFORGET_READ,1.0);
|
|
SpendXP(oCaster, costs.nXPCost);
|
|
SpendGP(oCaster, costs.nGoldCost);
|
|
DestroyObject (oSpellTarget);
|
|
FloatingTextStrRefOnCreature(8502, oCaster); // Item Creation successful
|
|
|
|
//advance time here
|
|
if(!costs.nTimeCost) costs.nTimeCost = 1;
|
|
AdvanceTimeForPlayer(oCaster, RoundsToSeconds(costs.nTimeCost));
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
FloatingTextStrRefOnCreature(76417, oCaster); // Item Creation Failed
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Returns TRUE if the player used the last spell to craft a wand
|
|
// -----------------------------------------------------------------------------
|
|
int CICraftCheckCraftWand(object oSpellTarget, object oCaster, int nID = 0)
|
|
{
|
|
|
|
if(nID == 0) nID = PRCGetSpellId();
|
|
|
|
// -------------------------------------------------------------------------
|
|
// check if craft wand feat is there
|
|
// -------------------------------------------------------------------------
|
|
if (GetHasFeat(X2_CI_CRAFTWAND_FEAT_ID, oCaster) != TRUE)
|
|
{
|
|
FloatingTextStrRefOnCreature(40487, oCaster); // Item Creation Failed - Don't know how to create that type of item
|
|
return TRUE; // tried item creation but do not know how to do it
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Check if the spell is allowed to be used with Craft Wand
|
|
// -------------------------------------------------------------------------
|
|
if (CIGetIsSpellRestrictedFromCraftFeat(nID, X2_CI_CRAFTWAND_FEAT_ID))
|
|
{
|
|
FloatingTextStrRefOnCreature(83452, oCaster); // can not be used with this feat
|
|
return TRUE;
|
|
}
|
|
|
|
int nLevel = CIGetSpellInnateLevel(nID,TRUE);
|
|
if(GetPRCSwitch(PRC_CRAFT_WAND_CASTER_LEVEL))
|
|
{
|
|
int nMetaMagic = PRCGetMetaMagicFeat();
|
|
switch(nMetaMagic)
|
|
{
|
|
case METAMAGIC_EMPOWER:
|
|
nLevel += 2;
|
|
break;
|
|
case METAMAGIC_EXTEND:
|
|
nLevel += 1;
|
|
break;
|
|
case METAMAGIC_MAXIMIZE:
|
|
nLevel += 3;
|
|
break;
|
|
/* case METAMAGIC_QUICKEN:
|
|
nLevel += 1;
|
|
break;
|
|
case METAMAGIC_SILENT:
|
|
nLevel += 5;
|
|
break;
|
|
case METAMAGIC_STILL:
|
|
nLevel += 6;
|
|
break;
|
|
These dont work as IPs since they are hardcoded */
|
|
}
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// check if spell is below maxlevel for craft want
|
|
// -------------------------------------------------------------------------
|
|
int nMaxLevel = GetPRCSwitch(X2_CI_CRAFTWAND_MAXLEVEL);
|
|
if(nMaxLevel == 0)
|
|
nMaxLevel = 4;
|
|
if (nLevel > nMaxLevel)
|
|
{
|
|
FloatingTextStrRefOnCreature(83623, oCaster);
|
|
return TRUE;
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// XP/GP Cost Calculation
|
|
// -------------------------------------------------------------------------
|
|
int nCostMod = GetPRCSwitch(X2_CI_CRAFTWAND_COSTMODIFIER);
|
|
if(nCostMod == 0)
|
|
nCostMod = 750;
|
|
int nCost = CIGetCraftGPCost(nLevel, nCostMod, PRC_CRAFT_WAND_CASTER_LEVEL);
|
|
|
|
struct craft_cost_struct costs = GetModifiedCostsFromBase(nCost, oCaster, FEAT_CRAFT_WAND, FALSE);
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Does Player have enough gold?
|
|
// -------------------------------------------------------------------------
|
|
if(!GetHasGPToSpend(oCaster, costs.nGoldCost))
|
|
{
|
|
FloatingTextStrRefOnCreature(3786, oCaster); // Item Creation Failed - not enough gold!
|
|
return TRUE;
|
|
}
|
|
|
|
// more calculations on XP cost
|
|
int nHD = GetHitDice(oCaster);
|
|
int nMinXPForLevel = ((nHD * (nHD - 1)) / 2) * 1000;
|
|
int nNewXP = GetXP(oCaster) - costs.nXPCost;
|
|
|
|
// -------------------------------------------------------------------------
|
|
// check for sufficient XP to cast spell
|
|
// -------------------------------------------------------------------------
|
|
if (!GetHasXPToSpend(oCaster, costs.nXPCost))
|
|
{
|
|
FloatingTextStrRefOnCreature(3785, oCaster); // Item Creation Failed - Not enough XP
|
|
return TRUE;
|
|
}
|
|
|
|
//check spell emulation
|
|
if(!CheckAlternativeCrafting(oCaster, nID, costs))
|
|
{
|
|
FloatingTextStringOnCreature("*Crafting failed!*", oCaster, FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Here we craft the wand
|
|
// -------------------------------------------------------------------------
|
|
object oWand = CICraftCraftWand(oCaster, nID);
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Verify Results
|
|
// -------------------------------------------------------------------------
|
|
if (GetIsObjectValid(oWand))
|
|
{
|
|
SpendXP(oCaster, costs.nXPCost);
|
|
SpendGP(oCaster, costs.nGoldCost);
|
|
DestroyObject (oSpellTarget);
|
|
FloatingTextStrRefOnCreature(8502, oCaster); // Item Creation successful
|
|
|
|
//advance time here
|
|
if(!costs.nTimeCost) costs.nTimeCost = 1;
|
|
AdvanceTimeForPlayer(oCaster, RoundsToSeconds(costs.nTimeCost));
|
|
string sName;
|
|
sName = Get2DACache("spells", "Name", nID);
|
|
sName = "Wand of "+GetStringByStrRef(StringToInt(sName));
|
|
SetName(oWand, sName);
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
FloatingTextStrRefOnCreature(76417, oCaster); // Item Creation Failed
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
int CICraftCheckCraftStaff(object oSpellTarget, object oCaster, int nSpellID = 0)
|
|
{
|
|
|
|
if(nSpellID == 0) nSpellID = PRCGetSpellId();
|
|
int nCasterLevel = GetAlternativeCasterLevel(oCaster, PRCGetCasterLevel(oCaster));
|
|
int bSuccess = TRUE;
|
|
int nCount = 0;
|
|
itemproperty ip = GetFirstItemProperty(oSpellTarget);
|
|
while(GetIsItemPropertyValid(ip))
|
|
{
|
|
if(GetItemPropertyType(ip) == ITEM_PROPERTY_CAST_SPELL)
|
|
nCount++;
|
|
ip = GetNextItemProperty(oSpellTarget);
|
|
}
|
|
if(nCount >= 8)
|
|
{
|
|
FloatingTextStringOnCreature("* Failure - Too many castspell itemproperties *", oCaster);
|
|
return TRUE;
|
|
}
|
|
if(!GetHasFeat(X2_CI_CRAFTSTAFF_FEAT_ID, oCaster))
|
|
{
|
|
FloatingTextStrRefOnCreature(40487, oCaster); // Item Creation Failed - Don't know how to create that type of item
|
|
return TRUE; // tried item creation but do not know how to do it
|
|
}
|
|
int nMetaMagic = PRCGetMetaMagicFeat();
|
|
if(nMetaMagic && !GetHasFeat(X2_CI_CRAFTSTAFF_EPIC_FEAT_ID, oCaster))
|
|
{
|
|
FloatingTextStringOnCreature("* Failure - You must be able to craft epic staffs to apply metamagic *", oCaster);
|
|
return TRUE; // tried item creation but do not know how to do it
|
|
}
|
|
if(CIGetIsSpellRestrictedFromCraftFeat(nSpellID, X2_CI_CRAFTSTAFF_FEAT_ID))
|
|
{
|
|
FloatingTextStrRefOnCreature(16829169, oCaster); // can not be used with this feat
|
|
return TRUE;
|
|
}
|
|
int nLevel = CIGetSpellInnateLevel(nSpellID,TRUE);
|
|
if(GetPRCSwitch(PRC_CRAFT_STAFF_CASTER_LEVEL))
|
|
{
|
|
switch(nMetaMagic)
|
|
{
|
|
case METAMAGIC_EMPOWER:
|
|
nLevel += 2;
|
|
break;
|
|
case METAMAGIC_EXTEND:
|
|
nLevel += 1;
|
|
break;
|
|
case METAMAGIC_MAXIMIZE:
|
|
nLevel += 3;
|
|
break;
|
|
/* case METAMAGIC_QUICKEN:
|
|
nLevel += 1;
|
|
break;
|
|
case METAMAGIC_SILENT:
|
|
nLevel += 5;
|
|
break;
|
|
case METAMAGIC_STILL:
|
|
nLevel += 6;
|
|
break;
|
|
These dont work as IPs since they are hardcoded */
|
|
}
|
|
}
|
|
int nCostMod = GetPRCSwitch(X2_CI_CRAFTSTAFF_COSTMODIFIER);
|
|
if(!nCostMod) nCostMod = 750;
|
|
int nLvlRow = IPGetIPConstCastSpellFromSpellID(nSpellID);
|
|
int nCLevel = StringToInt(Get2DACache("iprp_spells","CasterLvl",nLvlRow));
|
|
int nCost = CIGetCraftGPCost(nLevel, nCostMod, PRC_CRAFT_STAFF_CASTER_LEVEL);
|
|
|
|
//discount for second or 3+ spells
|
|
if(nCount+1 == 2)
|
|
nCost = (nCost*3)/4;
|
|
else if(nCount+1 >= 3)
|
|
nCost = nCost/2;
|
|
|
|
//takes epic xp costs into account
|
|
struct craft_cost_struct costs = GetModifiedCostsFromBase(nCost, oCaster, FEAT_CRAFT_STAFF, (nMetaMagic > 0));
|
|
|
|
if(costs.nGoldCost < 1) costs.nXPCost = 1;
|
|
if(costs.nXPCost < 1) costs.nXPCost = 1;
|
|
//if(GetGold(oCaster) < nGoldCost) // enough gold?
|
|
if (!GetHasGPToSpend(oCaster, costs.nGoldCost))
|
|
{
|
|
FloatingTextStrRefOnCreature(3786, oCaster); // Item Creation Failed - not enough gold!
|
|
return TRUE;
|
|
}
|
|
int nHD = GetHitDice(oCaster);
|
|
int nMinXPForLevel = (nHD * (nHD - 1)) * 500;
|
|
int nNewXP = GetXP(oCaster) - costs.nXPCost;
|
|
//if (nMinXPForLevel > nNewXP || nNewXP == 0 )
|
|
if (!GetHasXPToSpend(oCaster, costs.nXPCost))
|
|
{
|
|
FloatingTextStrRefOnCreature(3785, oCaster); // Item Creation Failed - Not enough XP
|
|
return TRUE;
|
|
}
|
|
//check spell emulation
|
|
if(!CheckAlternativeCrafting(oCaster, nSpellID, costs))
|
|
{
|
|
FloatingTextStringOnCreature("*Crafting failed!*", oCaster, FALSE);
|
|
return TRUE;
|
|
}
|
|
int nPropID = IPGetIPConstCastSpellFromSpellID(nSpellID);
|
|
if (nPropID == 0 && nSpellID != 0)
|
|
{
|
|
FloatingTextStrRefOnCreature(84544,oCaster);
|
|
return TRUE;
|
|
}
|
|
if (nPropID != -1)
|
|
{
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ItemPropertyCastSpell(nPropID,IP_CONST_CASTSPELL_NUMUSES_1_CHARGE_PER_USE),oSpellTarget);
|
|
|
|
if(GetPRCSwitch(PRC_CRAFT_STAFF_CASTER_LEVEL))
|
|
{
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ItemPropertyCastSpellCasterLevel(nSpellID, nCasterLevel),oSpellTarget);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ItemPropertyCastSpellMetamagic(nSpellID, PRCGetMetaMagicFeat()),oSpellTarget);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ItemPropertyCastSpellDC(nSpellID, PRCGetSaveDC(PRCGetSpellTargetObject(), OBJECT_SELF)),oSpellTarget);
|
|
}
|
|
}
|
|
else
|
|
bSuccess = FALSE;
|
|
|
|
if(bSuccess)
|
|
{
|
|
//TakeGoldFromCreature(nGoldCost, oCaster, TRUE);
|
|
//SetXP(oCaster, nNewXP);
|
|
SpendXP(oCaster, costs.nXPCost);
|
|
SpendGP(oCaster, costs.nGoldCost);
|
|
//DestroyObject (oSpellTarget);
|
|
FloatingTextStrRefOnCreature(8502, oCaster); // Item Creation successful
|
|
|
|
//advance time here
|
|
if(!costs.nTimeCost) costs.nTimeCost = 1;
|
|
AdvanceTimeForPlayer(oCaster, RoundsToSeconds(costs.nTimeCost));
|
|
string sName;
|
|
sName = GetName(oCaster)+"'s Magic Staff";
|
|
//sName = Get2DACache("spells", "Name", nID);
|
|
//sName = "Wand of "+GetStringByStrRef(StringToInt(sName));
|
|
SetName(oSpellTarget, sName);
|
|
SetItemCursedFlag(oSpellTarget, FALSE);
|
|
SetDroppableFlag(oSpellTarget, TRUE);
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
FloatingTextStrRefOnCreature(76417, oCaster); // Item Creation Failed
|
|
return TRUE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
int CICraftCheckCraftRod(object oSpellTarget, object oCaster, int nSpellID = 0)
|
|
{
|
|
|
|
if(nSpellID == 0) nSpellID = PRCGetSpellId();
|
|
int nCasterLevel = GetAlternativeCasterLevel(oCaster, PRCGetCasterLevel(oCaster));
|
|
int bSuccess = TRUE;
|
|
int nCount = 0;
|
|
itemproperty ip = GetFirstItemProperty(oSpellTarget);
|
|
while(GetIsItemPropertyValid(ip))
|
|
{
|
|
if(GetItemPropertyType(ip) == ITEM_PROPERTY_CAST_SPELL)
|
|
nCount++;
|
|
ip = GetNextItemProperty(oSpellTarget);
|
|
}
|
|
if(nCount >= 8)
|
|
{
|
|
FloatingTextStringOnCreature("* Failure - Too many castspell itemproperties *", oCaster);
|
|
return TRUE;
|
|
}
|
|
if(!GetHasFeat(X2_CI_CRAFTROD_FEAT_ID, oCaster))
|
|
{
|
|
FloatingTextStrRefOnCreature(40487, oCaster); // Item Creation Failed - Don't know how to create that type of item
|
|
return TRUE; // tried item creation but do not know how to do it
|
|
}
|
|
int nMetaMagic = PRCGetMetaMagicFeat();
|
|
if(nMetaMagic && !GetHasFeat(X2_CI_CRAFTROD_EPIC_FEAT_ID, oCaster))
|
|
{
|
|
FloatingTextStringOnCreature("* Failure - You must be able to craft epic rods to apply metamagic *", oCaster);
|
|
return TRUE; // tried item creation but do not know how to do it
|
|
}
|
|
if(CIGetIsSpellRestrictedFromCraftFeat(nSpellID, X2_CI_CRAFTROD_FEAT_ID))
|
|
{
|
|
FloatingTextStrRefOnCreature(16829169, oCaster); // can not be used with this feat
|
|
return TRUE;
|
|
}
|
|
int nLevel = CIGetSpellInnateLevel(nSpellID,TRUE);
|
|
if(GetPRCSwitch(PRC_CRAFT_ROD_CASTER_LEVEL))
|
|
{
|
|
switch(nMetaMagic)
|
|
{
|
|
case METAMAGIC_EMPOWER:
|
|
nLevel += 2;
|
|
break;
|
|
case METAMAGIC_EXTEND:
|
|
nLevel += 1;
|
|
break;
|
|
case METAMAGIC_MAXIMIZE:
|
|
nLevel += 3;
|
|
break;
|
|
/* case METAMAGIC_QUICKEN:
|
|
nLevel += 1;
|
|
break;
|
|
case METAMAGIC_SILENT:
|
|
nLevel += 5;
|
|
break;
|
|
case METAMAGIC_STILL:
|
|
nLevel += 6;
|
|
break;
|
|
These dont work as IPs since they are hardcoded */
|
|
}
|
|
}
|
|
int nCostMod = GetPRCSwitch(X2_CI_CRAFTROD_COSTMODIFIER);
|
|
if(!nCostMod) nCostMod = 750;
|
|
int nLvlRow = IPGetIPConstCastSpellFromSpellID(nSpellID);
|
|
int nCLevel = StringToInt(Get2DACache("iprp_spells","CasterLvl",nLvlRow));
|
|
int nCost = CIGetCraftGPCost(nLevel, nCostMod, PRC_CRAFT_ROD_CASTER_LEVEL);
|
|
|
|
//discount for second or 3+ spells
|
|
if(nCount+1 == 2)
|
|
nCost = (nCost*3)/4;
|
|
else if(nCount+1 >= 3)
|
|
nCost = nCost/2;
|
|
|
|
//takes epic xp costs into account
|
|
struct craft_cost_struct costs = GetModifiedCostsFromBase(nCost, oCaster, FEAT_CRAFT_ROD, (nMetaMagic > 0));
|
|
|
|
if(costs.nGoldCost < 1) costs.nXPCost = 1;
|
|
if(costs.nXPCost < 1) costs.nXPCost = 1;
|
|
//if(GetGold(oCaster) < nGoldCost) // enough gold?
|
|
if (!GetHasGPToSpend(oCaster, costs.nGoldCost))
|
|
{
|
|
FloatingTextStrRefOnCreature(3786, oCaster); // Item Creation Failed - not enough gold!
|
|
return TRUE;
|
|
}
|
|
int nHD = GetHitDice(oCaster);
|
|
int nMinXPForLevel = (nHD * (nHD - 1)) * 500;
|
|
int nNewXP = GetXP(oCaster) - costs.nXPCost;
|
|
//if (nMinXPForLevel > nNewXP || nNewXP == 0 )
|
|
if (!GetHasXPToSpend(oCaster, costs.nXPCost))
|
|
{
|
|
FloatingTextStrRefOnCreature(3785, oCaster); // Item Creation Failed - Not enough XP
|
|
return TRUE;
|
|
}
|
|
//check spell emulation
|
|
if(!CheckAlternativeCrafting(oCaster, nSpellID, costs))
|
|
{
|
|
FloatingTextStringOnCreature("*Crafting failed!*", oCaster, FALSE);
|
|
return TRUE;
|
|
}
|
|
int nPropID = IPGetIPConstCastSpellFromSpellID(nSpellID);
|
|
if (nPropID == 0 && nSpellID != 0)
|
|
{
|
|
FloatingTextStrRefOnCreature(84544,oCaster);
|
|
return TRUE;
|
|
}
|
|
if (nPropID != -1)
|
|
{
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ItemPropertyCastSpell(nPropID,IP_CONST_CASTSPELL_NUMUSES_1_CHARGE_PER_USE),oSpellTarget);
|
|
|
|
if(GetPRCSwitch(PRC_CRAFT_ROD_CASTER_LEVEL))
|
|
{
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ItemPropertyCastSpellMetamagic(nSpellID, PRCGetMetaMagicFeat()),oSpellTarget);
|
|
}
|
|
}
|
|
else
|
|
bSuccess = FALSE;
|
|
|
|
if(bSuccess)
|
|
{
|
|
//TakeGoldFromCreature(nGoldCost, oCaster, TRUE);
|
|
//SetXP(oCaster, nNewXP);
|
|
SpendXP(oCaster, costs.nXPCost);
|
|
SpendGP(oCaster, costs.nGoldCost);
|
|
//DestroyObject (oSpellTarget);
|
|
FloatingTextStrRefOnCreature(8502, oCaster); // Item Creation successful
|
|
|
|
//advance time here
|
|
if(!costs.nTimeCost) costs.nTimeCost = 1;
|
|
AdvanceTimeForPlayer(oCaster, RoundsToSeconds(costs.nTimeCost));
|
|
string sName;
|
|
sName = GetName(oCaster)+"'s Magic Rod";
|
|
//sName = Get2DACache("spells", "Name", nID);
|
|
//sName = "Wand of "+GetStringByStrRef(StringToInt(sName));
|
|
SetName(oSpellTarget, sName);
|
|
SetItemCursedFlag(oSpellTarget, FALSE);
|
|
SetDroppableFlag(oSpellTarget, TRUE);
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
FloatingTextStrRefOnCreature(76417, oCaster); // Item Creation Failed
|
|
return TRUE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
int InscribeRune(object oTarget = OBJECT_INVALID, object oCaster = OBJECT_INVALID, int nSpell = 0)
|
|
{
|
|
if(!GetIsObjectValid(oCaster)) oCaster = OBJECT_SELF;
|
|
// Get the item used to cast the spell
|
|
object oItem = GetSpellCastItem();
|
|
if(GetResRef(oItem) == "prc_rune_1")
|
|
{
|
|
string sName = GetName(GetItemPossessor(oItem));
|
|
if (DEBUG) FloatingTextStringOnCreature(sName + " has just cast a rune spell", oCaster, FALSE);
|
|
|
|
if(DEBUG) DoDebug("Checking for One Use runes");
|
|
// This check is used to clear up the one use runes
|
|
itemproperty ip = GetFirstItemProperty(oItem);
|
|
while(GetIsItemPropertyValid(ip))
|
|
{
|
|
if(GetItemPropertyType(ip) == ITEM_PROPERTY_CAST_SPELL)
|
|
{
|
|
if(DEBUG) DoDebug("Rune can cast spells");
|
|
if (GetItemPropertyCostTableValue(ip) == 5) // Only one use runes have 2 charges per use
|
|
{
|
|
if(DEBUG) DoDebug("Rune has 2 charges a use, marking it a one use rune");
|
|
// Give it enough time for the spell to finish casting
|
|
DestroyObject(oItem, 1.0);
|
|
if(DEBUG) DoDebug("Rune destroyed.");
|
|
}
|
|
}
|
|
|
|
ip = GetNextItemProperty(oItem);
|
|
}
|
|
}
|
|
|
|
// If Inscribing is turned off, the spell functions as normal
|
|
if(!GetLocalInt(oCaster, "InscribeRune")) return TRUE;
|
|
|
|
// No point being in here if you don't have runes.
|
|
if(!GetHasFeat(FEAT_INSCRIBE_RUNE, oCaster))
|
|
{
|
|
FloatingTextStrRefOnCreature(40487, oCaster); // Item Creation Failed - Don't know how to create that type of item
|
|
return TRUE; // tried item creation but do not know how to do it
|
|
}
|
|
|
|
// No point scribing runes from items, and its not allowed.
|
|
if (oItem != OBJECT_INVALID)
|
|
{
|
|
FloatingTextStringOnCreature("You cannot scribe a rune from an item.", oCaster, FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
if(!GetIsObjectValid(oTarget)) oTarget = PRCGetSpellTargetObject();
|
|
int nCaster = GetAlternativeCasterLevel(oCaster, PRCGetCasterLevel(oCaster));
|
|
int nDC = PRCGetSaveDC(oTarget, oCaster);
|
|
if(!nSpell) nSpell = PRCGetSpellId();
|
|
int nSpellLevel = 0;
|
|
int nClass = GetLevelByClass(CLASS_TYPE_RUNECASTER, oCaster);
|
|
|
|
// This accounts for the fact that there is no bonus to runecraft at level 10
|
|
// Also adjusts it to fit the epic progression, which starts at 13
|
|
if (nClass >= 10) nClass -= 3;
|
|
// Bonus to Runecrafting checks from the Runecaster class
|
|
int nRuneCraft = (nClass + 2)/3;
|
|
// Runecraft local int that counts uses/charges
|
|
int nCount = GetLocalInt(oCaster, "RuneCounter");
|
|
|
|
int nLastClass = PRCGetLastSpellCastClass();
|
|
if (nLastClass == CLASS_TYPE_CLERIC || nLastClass == CLASS_TYPE_UR_PRIEST) nSpellLevel = StringToInt(lookup_spell_cleric_level(nSpell));
|
|
else if (nLastClass == CLASS_TYPE_DRUID) nSpellLevel = StringToInt(lookup_spell_druid_level(nSpell));
|
|
else if (nLastClass == CLASS_TYPE_WIZARD || nLastClass == CLASS_TYPE_SORCERER) nSpellLevel = StringToInt(lookup_spell_level(nSpell));
|
|
// If none of these work, check the innate level of the spell
|
|
if (nSpellLevel == 0) nSpellLevel = StringToInt(lookup_spell_innate(nSpell));
|
|
// Minimum level.
|
|
if (nSpellLevel == 0) nSpellLevel = 1;
|
|
|
|
// This will be modified with Runecaster code later.
|
|
int nCharges = 1;
|
|
if (GetLocalInt(oCaster, "RuneCharges")) nCharges = nCount;
|
|
if (GetLocalInt(oCaster, "RuneUsesPerDay"))
|
|
{
|
|
// 5 is the max uses per day
|
|
if (nCount > 5) nCount = 5;
|
|
int nMaxUses = StringToInt(Get2DACache("prc_rune_craft", "Cost", PRC_RUNE_MAXUSESPERDAY));
|
|
if (nCount > nMaxUses) nCount = nMaxUses;
|
|
nCharges = nCount;
|
|
}
|
|
// Can't have no charges
|
|
if (nCharges == 0) nCharges = 1;
|
|
int nMaxCharges = StringToInt(Get2DACache("prc_rune_craft", "Cost", PRC_RUNE_MAXCHARGES));
|
|
if (nCount > nMaxCharges) nCharges = nMaxCharges;
|
|
|
|
FloatingTextStringOnCreature("Spell Level: " + IntToString(nSpellLevel), OBJECT_SELF, FALSE);
|
|
FloatingTextStringOnCreature("Caster Level: " + IntToString(nCaster), OBJECT_SELF, FALSE);
|
|
FloatingTextStringOnCreature("Number of Charges: " + IntToString(nCharges), OBJECT_SELF, FALSE);
|
|
|
|
// Gold cost multipler, varies depending on the ability used to craft
|
|
int nMultiplier = StringToInt(Get2DACache("prc_rune_craft", "Cost", PRC_RUNE_BASECOST));
|
|
if (nClass > 0) nMultiplier /= 2;
|
|
if (GetLocalInt(oCaster, "RuneCharges")) nMultiplier = StringToInt(Get2DACache("prc_rune_craft", "Cost", PRC_RUNE_CHARGES));
|
|
if (GetLocalInt(oCaster, "RuneUsesPerDay")) nMultiplier = StringToInt(Get2DACache("prc_rune_craft", "Cost", PRC_RUNE_PERDAY));
|
|
|
|
// Cost of the rune in gold and XP
|
|
int nCost = nSpellLevel * nCaster * nCharges * nMultiplier;
|
|
|
|
struct craft_cost_struct costs = GetModifiedCostsFromBase(nCost, oCaster, FEAT_INSCRIBE_RUNE, FALSE);
|
|
|
|
FloatingTextStringOnCreature("Gold Cost: " + IntToString(costs.nGoldCost), OBJECT_SELF, FALSE);
|
|
FloatingTextStringOnCreature("XP Cost: " + IntToString(costs.nXPCost), OBJECT_SELF, FALSE);
|
|
|
|
// See if the caster has enough gold and XP to scribe a rune, and that he passes the skill check.
|
|
int nHD = GetHitDice(oCaster);
|
|
int nMinXPForLevel = ((nHD * (nHD - 1)) / 2) * 1000;
|
|
int nNewXP = GetXP(oCaster) - costs.nXPCost;
|
|
int nGold = GetGold(oCaster);
|
|
int nNewGold = nGold - costs.nGoldCost;
|
|
int nCheck = FALSE;
|
|
// Does the PC have Maximize Rune turned on?
|
|
int nMaximize = 0;
|
|
if (GetLocalInt(oCaster, "MaximizeRune")) nMaximize = 5;
|
|
// The check does not use GetIsSkillSuccessful so it doesn't show on the PC
|
|
if ((GetSkillRank(SKILL_CRAFT_ARMOR, oCaster) + d20() + nRuneCraft) >= (20 + nSpellLevel + nMaximize)) nCheck = TRUE;
|
|
|
|
|
|
if (!GetHasGPToSpend(oCaster, costs.nGoldCost))
|
|
{
|
|
FloatingTextStringOnCreature("You do not have enough gold to scribe this rune.", oCaster, FALSE);
|
|
// Since they don't have enough, the spell casts normally
|
|
return TRUE;
|
|
}
|
|
if (!GetHasXPToSpend(oCaster, costs.nXPCost) )
|
|
{
|
|
FloatingTextStringOnCreature("You do not have enough experience to scribe this rune.", oCaster, FALSE);
|
|
// Since they don't have enough, the spell casts normally
|
|
return TRUE;
|
|
}
|
|
if (!nCheck)
|
|
{
|
|
FloatingTextStringOnCreature("You have failed the craft check to scribe this rune.", oCaster, FALSE);
|
|
// Since they don't have enough, the spell casts normally
|
|
return TRUE;
|
|
}
|
|
|
|
// Steal all the code from craft wand.
|
|
// The reason craft wand is used is because it is possible to create runes with charges using the Runecaster class.
|
|
int nPropID = IPGetIPConstCastSpellFromSpellID(nSpell);
|
|
|
|
// * GZ 2003-09-11: If the current spell cast is not acid fog, and
|
|
// * returned property ID is 0, bail out to prevent
|
|
// * creation of acid fog items.
|
|
if (nPropID == 0 && nSpell != 0)
|
|
{
|
|
FloatingTextStrRefOnCreature(84544,oCaster);
|
|
return TRUE;
|
|
}
|
|
|
|
//check spell emulation
|
|
if(!CheckAlternativeCrafting(oCaster, nSpell, costs))
|
|
{
|
|
FloatingTextStringOnCreature("*Crafting failed!*", oCaster, FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
if (nPropID != -1)
|
|
{
|
|
// This part is always done
|
|
int nRuneChant = 0;
|
|
if (nClass >= 30) nRuneChant = 10;
|
|
else if (nClass >= 27) nRuneChant = 9;
|
|
else if (nClass >= 24) nRuneChant = 8;
|
|
else if (nClass >= 21) nRuneChant = 7;
|
|
else if (nClass >= 18) nRuneChant = 6;
|
|
else if (nClass >= 15) nRuneChant = 5;
|
|
else if (nClass >= 12) nRuneChant = 4;
|
|
else if (nClass >= 9) nRuneChant = 3;
|
|
else if (nClass >= 5) nRuneChant = 2;
|
|
else if (nClass >= 2) nRuneChant = 1;
|
|
|
|
// Since we know they can now pay for it, create the rune stone
|
|
object oRune = CreateItemOnObject("prc_rune_1", oCaster, 1, IntToString(nRuneChant));
|
|
|
|
// If they have this active, the bonuses are already added, so skip
|
|
// If they don't, add the bonuses down below
|
|
if(GetHasSpellEffect(SPELL_RUNE_CHANT))
|
|
nRuneChant = 0;
|
|
|
|
itemproperty ipLevel = ItemPropertyCastSpellCasterLevel(nSpell, PRCGetCasterLevel());
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipLevel,oRune);
|
|
itemproperty ipMeta = ItemPropertyCastSpellMetamagic(nSpell, PRCGetMetaMagicFeat());
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipMeta,oRune);
|
|
itemproperty ipDC = ItemPropertyCastSpellDC(nSpell, PRCGetSaveDC(PRCGetSpellTargetObject(), OBJECT_SELF) + nRuneChant);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipDC,oRune);
|
|
// If Maximize Rune is turned on and we pass the check, add the Maximize IProp
|
|
if (GetLocalInt(oCaster, "MaximizeRune"))
|
|
{
|
|
itemproperty ipMax = ItemPropertyCastSpellMetamagic(nSpell, METAMAGIC_MAXIMIZE);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipMax,oRune);
|
|
}
|
|
|
|
// If its uses per day instead of charges, we do some different stuff here
|
|
if (GetLocalInt(oCaster, "RuneUsesPerDay"))
|
|
{
|
|
int nIPUses;
|
|
if (nCount == 1) nIPUses = IP_CONST_CASTSPELL_NUMUSES_1_USE_PER_DAY;
|
|
else if (nCount == 2) nIPUses = IP_CONST_CASTSPELL_NUMUSES_2_USES_PER_DAY;
|
|
else if (nCount == 3) nIPUses = IP_CONST_CASTSPELL_NUMUSES_3_USES_PER_DAY;
|
|
else if (nCount == 4) nIPUses = IP_CONST_CASTSPELL_NUMUSES_4_USES_PER_DAY;
|
|
// Caps out at 5 per day
|
|
else if (nCount >= 5) nIPUses = IP_CONST_CASTSPELL_NUMUSES_5_USES_PER_DAY;
|
|
|
|
itemproperty ipProp = ItemPropertyCastSpell(nPropID,nIPUses);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipProp,oRune);
|
|
}
|
|
else if (nCharges == 1) // This is to handle one use runes so the spellhooking works
|
|
{
|
|
itemproperty ipProp = ItemPropertyCastSpell(nPropID,IP_CONST_CASTSPELL_NUMUSES_2_CHARGES_PER_USE);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipProp,oRune);
|
|
// This is done so the item exists when it is used for the game to read data off of
|
|
nCharges = 3;
|
|
}
|
|
else // Do the normal charges
|
|
{
|
|
itemproperty ipProp = ItemPropertyCastSpell(nPropID,IP_CONST_CASTSPELL_NUMUSES_1_CHARGE_PER_USE);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipProp,oRune);
|
|
}
|
|
SetItemCharges(oRune,nCharges);
|
|
SetXP(oCaster,nNewXP);
|
|
TakeGoldFromCreature(costs.nGoldCost, oCaster, TRUE);
|
|
|
|
//advance time here
|
|
if(!costs.nTimeCost) costs.nTimeCost = 1;
|
|
AdvanceTimeForPlayer(oCaster, RoundsToSeconds(costs.nTimeCost));
|
|
string sName;
|
|
sName = Get2DACache("spells", "Name", nSpell);
|
|
sName = "Rune of "+GetStringByStrRef(StringToInt(sName));
|
|
if(GetLocalInt(oCaster, "MaximizeRune"))
|
|
sName = "Maximized "+sName;
|
|
SetName(oRune, sName);
|
|
}
|
|
|
|
// If we have made it this far, they have crafted the rune and the spell has been used up, so it returns false.
|
|
return FALSE;
|
|
}
|
|
|
|
int AttuneGem(object oTarget = OBJECT_INVALID, object oCaster = OBJECT_INVALID, int nSpell = 0)
|
|
{
|
|
if(!GetIsObjectValid(oCaster)) oCaster = OBJECT_SELF;
|
|
// Get the item used to cast the spell
|
|
object oItem = GetSpellCastItem();
|
|
if (GetTag(oItem) == "prc_attunegem")
|
|
{
|
|
string sName = GetName(GetItemPossessor(oItem));
|
|
if (DEBUG) FloatingTextStringOnCreature(sName + " has just cast a gem spell", oCaster, FALSE);
|
|
|
|
if(DEBUG) DoDebug("Checking for One Use Gems");
|
|
// This check is used to clear up the one use Gems
|
|
itemproperty ip = GetFirstItemProperty(oItem);
|
|
while(GetIsItemPropertyValid(ip))
|
|
{
|
|
if(GetItemPropertyType(ip) == ITEM_PROPERTY_CAST_SPELL)
|
|
{
|
|
if(DEBUG) DoDebug("Gem can cast spells");
|
|
if (GetItemPropertyCostTableValue(ip) == 5) // Only one use Gems have 2 charges per use
|
|
{
|
|
if(DEBUG) DoDebug("Gem has 2 charges a use, marking it a one use Gem");
|
|
// Give it enough time for the spell to finish casting
|
|
DestroyObject(oItem, 1.0);
|
|
if(DEBUG) DoDebug("Gem destroyed.");
|
|
}
|
|
}
|
|
|
|
ip = GetNextItemProperty(oItem);
|
|
}
|
|
}
|
|
|
|
// If Inscribing is turned off, the spell functions as normal
|
|
if(!GetLocalInt(oCaster, "AttuneGem")) return TRUE;
|
|
|
|
// No point being in here if you don't have Gems.
|
|
if(!GetHasFeat(FEAT_ATTUNE_GEM, oCaster))
|
|
{
|
|
FloatingTextStrRefOnCreature(40487, oCaster); // Item Creation Failed - Don't know how to create that type of item
|
|
return TRUE; // tried item creation but do not know how to do it
|
|
}
|
|
|
|
// No point scribing Gems from items, and its not allowed.
|
|
if (oItem != OBJECT_INVALID)
|
|
{
|
|
FloatingTextStringOnCreature("You cannot scribe a Gem from an item.", oCaster, FALSE);
|
|
return TRUE;
|
|
}
|
|
// oTarget here should be the gem. If it's not, fail.
|
|
if(!GetIsObjectValid(oTarget)) oTarget = PRCGetSpellTargetObject();
|
|
// Only accepts bioware gems
|
|
if (GetStringLeft(GetResRef(oTarget), 5) == "it_gem")
|
|
{
|
|
FloatingTextStringOnCreature("Spell target is not a valid gem.", oCaster, FALSE);
|
|
// And out we go
|
|
return TRUE;
|
|
}
|
|
|
|
int nCaster = GetAlternativeCasterLevel(oCaster, PRCGetCasterLevel(oCaster));
|
|
int nDC = PRCGetSaveDC(oTarget, oCaster);
|
|
if(!nSpell) nSpell = PRCGetSpellId();
|
|
int nSpellLevel;
|
|
|
|
if (PRCGetLastSpellCastClass() == CLASS_TYPE_CLERIC || PRCGetLastSpellCastClass() == CLASS_TYPE_UR_PRIEST) nSpellLevel = StringToInt(lookup_spell_cleric_level(nSpell));
|
|
else if (PRCGetLastSpellCastClass() == CLASS_TYPE_DRUID) nSpellLevel = StringToInt(lookup_spell_druid_level(nSpell));
|
|
else if (PRCGetLastSpellCastClass() == CLASS_TYPE_WIZARD || PRCGetLastSpellCastClass() == CLASS_TYPE_SORCERER) nSpellLevel = StringToInt(lookup_spell_level(nSpell));
|
|
// If none of these work, check the innate level of the spell
|
|
if (nSpellLevel == 0) nSpellLevel = StringToInt(lookup_spell_innate(nSpell));
|
|
// Minimum level.
|
|
if (nSpellLevel == 0) nSpellLevel = 1;
|
|
|
|
int nCharges = 1;
|
|
|
|
FloatingTextStringOnCreature("Spell Level: " + IntToString(nSpellLevel), OBJECT_SELF, FALSE);
|
|
FloatingTextStringOnCreature("Caster Level: " + IntToString(nCaster), OBJECT_SELF, FALSE);
|
|
FloatingTextStringOnCreature("Number of Charges: " + IntToString(nCharges), OBJECT_SELF, FALSE);
|
|
|
|
// Gold cost multipler, varies depending on the ability used to craft
|
|
int nMultiplier = StringToInt(Get2DACache("prc_rune_craft", "Cost", PRC_GEM_BASECOST));
|
|
|
|
// Cost of the Gem in gold and XP
|
|
int nCost = nSpellLevel * nCaster * nMultiplier;
|
|
|
|
struct craft_cost_struct costs = GetModifiedCostsFromBase(nCost, oCaster, FEAT_ATTUNE_GEM, FALSE);
|
|
|
|
FloatingTextStringOnCreature("Gold Cost: " + IntToString(costs.nGoldCost), OBJECT_SELF, FALSE);
|
|
FloatingTextStringOnCreature("XP Cost: " + IntToString(costs.nXPCost), OBJECT_SELF, FALSE);
|
|
|
|
// See if the caster has enough gold and XP to scribe a Gem, and that he passes the skill check.
|
|
int nHD = GetHitDice(oCaster);
|
|
int nMinXPForLevel = ((nHD * (nHD - 1)) / 2) * 1000;
|
|
int nNewXP = GetXP(oCaster) - costs.nXPCost;
|
|
int nGold = GetGold(oCaster);
|
|
int nNewGold = nGold - costs.nGoldCost;
|
|
int nCheck = FALSE;
|
|
|
|
if (!GetHasGPToSpend(oCaster, costs.nGoldCost))
|
|
{
|
|
FloatingTextStringOnCreature("You do not have enough gold to scribe this Gem.", oCaster, FALSE);
|
|
// Since they don't have enough, the spell casts normally
|
|
return TRUE;
|
|
}
|
|
if (!GetHasXPToSpend(oCaster, costs.nXPCost) )
|
|
{
|
|
FloatingTextStringOnCreature("You do not have enough experience to scribe this Gem.", oCaster, FALSE);
|
|
// Since they don't have enough, the spell casts normally
|
|
return TRUE;
|
|
}
|
|
|
|
// Is the gem worth enough?
|
|
int nGemGold = GetGoldPieceValue(oTarget);
|
|
int nGemLevel = nGemGold / StringToInt(Get2DACache("prc_rune_craft", "Cost", PRC_GEM_PERLEVEL));
|
|
if (nGemLevel > 9) nGemLevel = 9;
|
|
if (nSpellLevel > nGemLevel)
|
|
{
|
|
FloatingTextStringOnCreature("Gem is not high enough level for this spell", oCaster, FALSE);
|
|
// The spell casts normally
|
|
return TRUE;
|
|
}
|
|
|
|
// Steal all the code from craft wand.
|
|
int nPropID = IPGetIPConstCastSpellFromSpellID(nSpell);
|
|
|
|
// * GZ 2003-09-11: If the current spell cast is not acid fog, and
|
|
// * returned property ID is 0, bail out to prevent
|
|
// * creation of acid fog items.
|
|
if (nPropID == 0 && nSpell != 0)
|
|
{
|
|
FloatingTextStrRefOnCreature(84544,oCaster);
|
|
return TRUE;
|
|
}
|
|
|
|
//check spell emulation
|
|
if(!CheckAlternativeCrafting(oCaster, nSpell, costs))
|
|
{
|
|
FloatingTextStringOnCreature("*Crafting failed!*", oCaster, FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
if (nPropID != -1)
|
|
{
|
|
itemproperty ipLevel = ItemPropertyCastSpellCasterLevel(nSpell, PRCGetCasterLevel());
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipLevel,oTarget);
|
|
itemproperty ipMeta = ItemPropertyCastSpellMetamagic(nSpell, PRCGetMetaMagicFeat());
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipMeta,oTarget);
|
|
itemproperty ipDC = ItemPropertyCastSpellDC(nSpell, PRCGetSaveDC(PRCGetSpellTargetObject(), OBJECT_SELF));
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipDC,oTarget);
|
|
|
|
if (nCharges == 1) // This is to handle one use Gems so the spellhooking works
|
|
{
|
|
itemproperty ipProp = ItemPropertyCastSpell(nPropID,IP_CONST_CASTSPELL_NUMUSES_2_CHARGES_PER_USE);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipProp,oTarget);
|
|
// This is done so the item exists when it is used for the game to read data off of
|
|
nCharges = 3;
|
|
}
|
|
|
|
SetItemCharges(oTarget, nCharges);
|
|
SetXP(oCaster, nNewXP);
|
|
TakeGoldFromCreature(costs.nGoldCost, oCaster, TRUE);
|
|
|
|
string sName;
|
|
sName = Get2DACache("spells", "Name", nSpell);
|
|
sName = "Gem of "+GetStringByStrRef(StringToInt(sName));
|
|
SetName(oTarget, sName);
|
|
|
|
// This is done to allow the item to be set properly, and then alter the tag
|
|
CopyObject(oTarget, GetLocation(oCaster), oCaster, "prc_attunegem");
|
|
DestroyObject(oTarget, 0.1);
|
|
}
|
|
|
|
// If we have made it this far, they have crafted the Gem and the spell has been used up, so it returns false.
|
|
return FALSE;
|
|
}
|
|
|
|
int CraftSkullTalisman(object oTarget = OBJECT_INVALID, object oCaster = OBJECT_INVALID, int nSpell = 0)
|
|
{
|
|
if(!GetIsObjectValid(oCaster)) oCaster = OBJECT_SELF;
|
|
// Get the item used to cast the spell
|
|
object oItem = GetSpellCastItem();
|
|
if (GetTag(oItem) == "prc_skulltalis")
|
|
{
|
|
string sName = GetName(GetItemPossessor(oItem));
|
|
if (DEBUG) FloatingTextStringOnCreature(sName + " has just cast a skull talisman spell", oCaster, FALSE);
|
|
|
|
if (DEBUG) DoDebug("Checking for One Use Skulls");
|
|
// This check is used to clear up the one use SkullTalismans
|
|
itemproperty ip = GetFirstItemProperty(oItem);
|
|
while(GetIsItemPropertyValid(ip))
|
|
{
|
|
if(GetItemPropertyType(ip) == ITEM_PROPERTY_CAST_SPELL)
|
|
{
|
|
if (DEBUG) DoDebug("Skull Talisman can cast spells");
|
|
if (GetItemPropertyCostTableValue(ip) == 5) // Only one use Skull Talismans have 2 charges per use
|
|
{
|
|
if(DEBUG) DoDebug("Skull Talisman has 2 charges a use, marking it a one use Skull Talisman");
|
|
// Give it enough time for the spell to finish casting
|
|
DestroyObject(oItem, 1.0);
|
|
if(DEBUG) DoDebug("Skull Talisman destroyed.");
|
|
}
|
|
}
|
|
|
|
ip = GetNextItemProperty(oItem);
|
|
}
|
|
}
|
|
|
|
// If Inscribing is turned off, the spell functions as normal
|
|
if(!GetLocalInt(oCaster, "CraftSkullTalisman")) return TRUE;
|
|
|
|
// No point being in here if you don't have SkullTalismans.
|
|
if(!GetHasFeat(FEAT_CRAFT_SKULL_TALISMAN, oCaster))
|
|
{
|
|
FloatingTextStrRefOnCreature(40487, oCaster); // Item Creation Failed - Don't know how to create that type of item
|
|
return TRUE; // tried item creation but do not know how to do it
|
|
}
|
|
|
|
// No point scribing SkullTalismans from items, and its not allowed.
|
|
if (oItem != OBJECT_INVALID)
|
|
{
|
|
FloatingTextStringOnCreature("You cannot scribe a Skull Talisman from an item.", oCaster, FALSE);
|
|
return TRUE;
|
|
}
|
|
// oTarget here should be the Caster.
|
|
if(!GetIsObjectValid(oTarget)) oTarget = PRCGetSpellTargetObject();
|
|
|
|
int nCaster = GetAlternativeCasterLevel(oCaster, PRCGetCasterLevel(oCaster));
|
|
int nDC = PRCGetSaveDC(oTarget, oCaster);
|
|
if(!nSpell) nSpell = PRCGetSpellId();
|
|
int nSpellLevel;
|
|
|
|
if (PRCGetLastSpellCastClass() == CLASS_TYPE_CLERIC || PRCGetLastSpellCastClass() == CLASS_TYPE_UR_PRIEST) nSpellLevel = StringToInt(lookup_spell_cleric_level(nSpell));
|
|
else if (PRCGetLastSpellCastClass() == CLASS_TYPE_DRUID) nSpellLevel = StringToInt(lookup_spell_druid_level(nSpell));
|
|
else if (PRCGetLastSpellCastClass() == CLASS_TYPE_WIZARD || PRCGetLastSpellCastClass() == CLASS_TYPE_SORCERER) nSpellLevel = StringToInt(lookup_spell_level(nSpell));
|
|
// If none of these work, check the innate level of the spell
|
|
if (nSpellLevel == 0) nSpellLevel = StringToInt(lookup_spell_innate(nSpell));
|
|
// Minimum level.
|
|
if (nSpellLevel == 0) nSpellLevel = 1;
|
|
|
|
int nCharges = 1;
|
|
|
|
FloatingTextStringOnCreature("Spell Level: " + IntToString(nSpellLevel), OBJECT_SELF, FALSE);
|
|
FloatingTextStringOnCreature("Caster Level: " + IntToString(nCaster), OBJECT_SELF, FALSE);
|
|
|
|
// Gold cost multipler, varies depending on the ability used to craft
|
|
int nMultiplier = StringToInt(Get2DACache("prc_rune_craft", "Cost", PRC_SKULL_BASECOST));
|
|
|
|
// Cost of the Skull Talisman in gold and XP
|
|
int nCost = nSpellLevel * nCaster * nMultiplier;
|
|
|
|
struct craft_cost_struct costs = GetModifiedCostsFromBase(nCost, oCaster, FEAT_CRAFT_SKULL_TALISMAN, FALSE);
|
|
|
|
FloatingTextStringOnCreature("Gold Cost: " + IntToString(costs.nGoldCost), OBJECT_SELF, FALSE);
|
|
FloatingTextStringOnCreature("XP Cost: " + IntToString(costs.nXPCost), OBJECT_SELF, FALSE);
|
|
|
|
// See if the caster has enough gold and XP to scribe a Skull Talisman, and that he passes the skill check.
|
|
int nHD = GetHitDice(oCaster);
|
|
int nMinXPForLevel = ((nHD * (nHD - 1)) / 2) * 1000;
|
|
int nNewXP = GetXP(oCaster) - costs.nXPCost;
|
|
int nGold = GetGold(oCaster);
|
|
int nNewGold = nGold - costs.nGoldCost;
|
|
int nCheck = FALSE;
|
|
|
|
if (!GetHasGPToSpend(oCaster, costs.nGoldCost))
|
|
{
|
|
FloatingTextStringOnCreature("You do not have enough gold to scribe this SkullTalisman.", oCaster, FALSE);
|
|
// Since they don't have enough, the spell casts normally
|
|
return TRUE;
|
|
}
|
|
if (!GetHasXPToSpend(oCaster, costs.nXPCost) )
|
|
{
|
|
FloatingTextStringOnCreature("You do not have enough experience to scribe this SkullTalisman.", oCaster, FALSE);
|
|
// Since they don't have enough, the spell casts normally
|
|
return TRUE;
|
|
}
|
|
|
|
// Create the item to have all the effects applied to
|
|
oTarget = CreateItemOnObject("prc_skulltalis", oCaster, 1);
|
|
|
|
// Steal all the code from craft wand.
|
|
int nPropID = IPGetIPConstCastSpellFromSpellID(nSpell);
|
|
|
|
// * GZ 2003-09-11: If the current spell cast is not acid fog, and
|
|
// * returned property ID is 0, bail out to prevent
|
|
// * creation of acid fog items.
|
|
if (nPropID == 0 && nSpell != 0)
|
|
{
|
|
FloatingTextStrRefOnCreature(84544,oCaster);
|
|
return TRUE;
|
|
}
|
|
|
|
//check spell emulation
|
|
if(!CheckAlternativeCrafting(oCaster, nSpell, costs))
|
|
{
|
|
FloatingTextStringOnCreature("*Crafting failed!*", oCaster, FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
if (nPropID != -1)
|
|
{
|
|
itemproperty ipLevel = ItemPropertyCastSpellCasterLevel(nSpell, PRCGetCasterLevel());
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipLevel,oTarget);
|
|
itemproperty ipMeta = ItemPropertyCastSpellMetamagic(nSpell, PRCGetMetaMagicFeat());
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipMeta,oTarget);
|
|
itemproperty ipDC = ItemPropertyCastSpellDC(nSpell, PRCGetSaveDC(PRCGetSpellTargetObject(), OBJECT_SELF));
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipDC,oTarget);
|
|
|
|
if (nCharges == 1) // This is to handle one use Skull Talismans so the spellhooking works
|
|
{
|
|
itemproperty ipProp = ItemPropertyCastSpell(nPropID,IP_CONST_CASTSPELL_NUMUSES_2_CHARGES_PER_USE);
|
|
AddItemProperty(DURATION_TYPE_PERMANENT,ipProp,oTarget);
|
|
// This is done so the item exists when it is used for the game to read data off of
|
|
nCharges = 3;
|
|
}
|
|
|
|
SetItemCharges(oTarget, nCharges);
|
|
SetXP(oCaster, nNewXP);
|
|
TakeGoldFromCreature(costs.nGoldCost, oCaster, TRUE);
|
|
|
|
string sName;
|
|
sName = Get2DACache("spells", "Name", nSpell);
|
|
sName = "Skull Talisman of "+ GetStringByStrRef(StringToInt(sName));
|
|
SetName(oTarget, sName);
|
|
|
|
// This is done to allow the item to be set properly, and then alter the tag
|
|
CopyObject(oTarget, GetLocation(oCaster), oCaster, "prc_skulltalis");
|
|
DestroyObject(oTarget, 0.1);
|
|
}
|
|
|
|
// If we have made it this far, they have crafted the Skull Talisman and the spell has been used up, so it returns false.
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Georg, July 2003
|
|
// Checks if the caster intends to use his item creation feats and
|
|
// calls appropriate item creation subroutine if conditions are met
|
|
// (spell cast on correct item, etc).
|
|
// Returns TRUE if the spell was used for an item creation feat
|
|
// -----------------------------------------------------------------------------
|
|
int CIGetSpellWasUsedForItemCreation(object oSpellTarget)
|
|
{
|
|
object oCaster = OBJECT_SELF;
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Spell cast on crafting base item (blank scroll, etc) ?
|
|
// -------------------------------------------------------------------------
|
|
if (!CIGetIsCraftFeatBaseItem(oSpellTarget))
|
|
{
|
|
return FALSE; // not blank scroll baseitem
|
|
}
|
|
else
|
|
{
|
|
// ---------------------------------------------------------------------
|
|
// Check Item Creation Feats were disabled through x2_inc_switches
|
|
// ---------------------------------------------------------------------
|
|
if (GetModuleSwitchValue(MODULE_SWITCH_DISABLE_ITEM_CREATION_FEATS) == TRUE)
|
|
{
|
|
FloatingTextStrRefOnCreature(83612, oCaster); // * Item creation feats are not enabled in this module *
|
|
return FALSE;
|
|
}
|
|
if (GetLocalInt(GetArea(oCaster), PRC_AREA_DISABLE_CRAFTING))
|
|
{
|
|
FloatingTextStrRefOnCreature(16832014, oCaster); // * Item creation feats are not enabled in this area *
|
|
return FALSE;
|
|
}
|
|
|
|
// ---------------------------------------------------------------------
|
|
// Ensure that item creation does not work one item was cast on another
|
|
// ---------------------------------------------------------------------
|
|
if (GetSpellCastItem() != OBJECT_INVALID)
|
|
{
|
|
FloatingTextStrRefOnCreature(83373, oCaster); // can not use one item to enchant another
|
|
return TRUE;
|
|
}
|
|
|
|
// ---------------------------------------------------------------------
|
|
// Ok, what kind of feat the user wants to use by examining the base itm
|
|
// ---------------------------------------------------------------------
|
|
int nBt = GetBaseItemType(oSpellTarget);
|
|
int nRet = FALSE;
|
|
switch (nBt)
|
|
{
|
|
case BASE_ITEM_BLANK_POTION :
|
|
// -------------------------------------------------
|
|
// Brew Potion
|
|
// -------------------------------------------------
|
|
nRet = CICraftCheckBrewPotion(oSpellTarget,oCaster);
|
|
break;
|
|
|
|
|
|
case BASE_ITEM_BLANK_SCROLL :
|
|
// -------------------------------------------------
|
|
// Scribe Scroll
|
|
// -------------------------------------------------
|
|
nRet = CICraftCheckScribeScroll(oSpellTarget,oCaster);
|
|
break;
|
|
|
|
|
|
case BASE_ITEM_BLANK_WAND :
|
|
// -------------------------------------------------
|
|
// Craft Wand
|
|
// -------------------------------------------------
|
|
nRet = CICraftCheckCraftWand(oSpellTarget,oCaster);
|
|
break;
|
|
|
|
case BASE_ITEM_CRAFTED_ROD :
|
|
// -------------------------------------------------
|
|
// Craft Rod
|
|
// -------------------------------------------------
|
|
nRet = CICraftCheckCraftRod(oSpellTarget,oCaster);
|
|
break;
|
|
|
|
case BASE_ITEM_CRAFTED_STAFF :
|
|
// -------------------------------------------------
|
|
// Craft Staff
|
|
// -------------------------------------------------
|
|
nRet = CICraftCheckCraftStaff(oSpellTarget,oCaster);
|
|
break;
|
|
|
|
// you could add more crafting basetypes here....
|
|
}
|
|
|
|
return nRet;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Makes oPC do a Craft check using nSkill to create the item supplied in sResRe
|
|
// If oContainer is specified, the item will be created there.
|
|
// Throwing weapons are created with stack sizes of 10, ammo with 20
|
|
// -----------------------------------------------------------------------------
|
|
object CIUseCraftItemSkill(object oPC, int nSkill, string sResRef, int nDC, object oContainer = OBJECT_INVALID)
|
|
{
|
|
int bSuccess = GetIsSkillSuccessful(oPC, nSkill, nDC);
|
|
object oNew;
|
|
if (bSuccess)
|
|
{
|
|
// actual item creation
|
|
// if a crafting container was specified, create inside
|
|
int bFix;
|
|
if (oContainer == OBJECT_INVALID)
|
|
{
|
|
//------------------------------------------------------------------
|
|
// We create the item in the work container to get rid of the
|
|
// stackable item problems that happen when we create the item
|
|
// directly on the player
|
|
//------------------------------------------------------------------
|
|
if (GetLevelByClass(CLASS_TYPE_IRONSOUL_FORGEMASTER, oPC))
|
|
oNew = CreateItemOnObject(sResRef,IPGetIPWorkContainer(oPC),1,GetName(oPC));
|
|
else
|
|
oNew = CreateItemOnObject(sResRef,IPGetIPWorkContainer(oPC));
|
|
bFix = TRUE;
|
|
}
|
|
else
|
|
{
|
|
if (GetLevelByClass(CLASS_TYPE_IRONSOUL_FORGEMASTER, oPC))
|
|
oNew = CreateItemOnObject(sResRef,oContainer,1,GetName(oPC));
|
|
else
|
|
oNew = CreateItemOnObject(sResRef,oContainer);
|
|
}
|
|
|
|
int nBase = GetBaseItemType(oNew);
|
|
if (nBase == BASE_ITEM_BOLT || nBase == BASE_ITEM_ARROW || nBase == BASE_ITEM_BULLET)
|
|
{
|
|
SetItemStackSize(oNew, 20);
|
|
}
|
|
else if (nBase == BASE_ITEM_THROWINGAXE || nBase == BASE_ITEM_SHURIKEN || nBase == BASE_ITEM_DART)
|
|
{
|
|
SetItemStackSize(oNew, 10);
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
// Get around the whole stackable item mess...
|
|
//----------------------------------------------------------------------
|
|
if (bFix)
|
|
{
|
|
object oRet = CopyObject(oNew,GetLocation(oPC),oPC);
|
|
DestroyObject(oNew);
|
|
oNew = oRet;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
oNew = OBJECT_INVALID;
|
|
}
|
|
|
|
return oNew;
|
|
}
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// georg, 2003-06-13 (
|
|
// Craft an item. This is only to be called from the crafting conversation
|
|
// spawned by x2_s2_crafting!!!
|
|
// -----------------------------------------------------------------------------
|
|
int CIDoCraftItemFromConversation(int nNumber)
|
|
{
|
|
string sNumber = IntToString(nNumber);
|
|
object oPC = GetPCSpeaker();
|
|
//object oMaterial = GetLocalObject(oPC,"X2_CI_CRAFT_MATERIAL");
|
|
object oMajor = GetLocalObject(oPC,"X2_CI_CRAFT_MAJOR");
|
|
object oMinor = GetLocalObject(oPC,"X2_CI_CRAFT_MINOR");
|
|
int nSkill = GetLocalInt(oPC,"X2_CI_CRAFT_SKILL");
|
|
int nMode = GetLocalInt(oPC,"X2_CI_CRAFT_MODE");
|
|
string sResult;
|
|
string s2DA;
|
|
int nDC;
|
|
|
|
|
|
DeleteLocalObject(oPC,"X2_CI_CRAFT_MAJOR");
|
|
DeleteLocalObject(oPC,"X2_CI_CRAFT_MINOR");
|
|
|
|
if (!GetIsObjectValid(oMajor))
|
|
{
|
|
FloatingTextStrRefOnCreature(83374,oPC); //"Invalid target"
|
|
DeleteLocalInt(oPC,"X2_CRAFT_SUCCESS");
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
if (GetItemPossessor(oMajor) != oPC)
|
|
{
|
|
FloatingTextStrRefOnCreature(83354,oPC); //"Invalid target"
|
|
DeleteLocalInt(oPC,"X2_CRAFT_SUCCESS");
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
// If we are in container mode,
|
|
if (nMode == X2_CI_CRAFTMODE_CONTAINER)
|
|
{
|
|
if (!GetIsObjectValid(oMinor))
|
|
{
|
|
FloatingTextStrRefOnCreature(83374,oPC); //"Invalid target"
|
|
DeleteLocalInt(oPC,"X2_CRAFT_SUCCESS");
|
|
return FALSE;
|
|
}
|
|
else if (GetItemPossessor(oMinor) != oPC)
|
|
{
|
|
FloatingTextStrRefOnCreature(83354,oPC); //"Invalid target"
|
|
DeleteLocalInt(oPC,"X2_CRAFT_SUCCESS");
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
if (nSkill == 26) // craft weapon
|
|
{
|
|
s2DA = X2_CI_CRAFTING_WP_2DA;
|
|
}
|
|
else if (nSkill == 25)
|
|
{
|
|
s2DA = X2_CI_CRAFTING_AR_2DA;
|
|
}
|
|
|
|
int nRow = GetLocalInt(oPC,"X2_CI_CRAFT_RESULTROW");
|
|
struct craft_struct stItem = CIGetCraftItemStructFrom2DA(s2DA,nRow,nNumber);
|
|
object oContainer = OBJECT_INVALID;
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// We once used a crafting container, but found it too complicated. Code is still
|
|
// left in here for the community
|
|
// ---------------------------------------------------------------------------
|
|
if (nMode == X2_CI_CRAFTMODE_CONTAINER)
|
|
{
|
|
oContainer = GetItemPossessedBy(oPC,"x2_it_craftcont");
|
|
}
|
|
|
|
// Do the crafting...
|
|
object oRet = CIUseCraftItemSkill( oPC, nSkill, stItem.sResRef, stItem.nDC, oContainer) ;
|
|
|
|
// * If you made an item, it should always be identified;
|
|
SetIdentified(oRet,TRUE);
|
|
|
|
if (GetIsObjectValid(oRet))
|
|
{
|
|
// -----------------------------------------------------------------------
|
|
// Copy all item properties from the major object on the resulting item
|
|
// Through we problably won't use this, its a neat thing to have for the
|
|
// community
|
|
// to enable magic item creation from the crafting system
|
|
// -----------------------------------------------------------------------
|
|
//if (GetGold(oPC)<stItem.nCost)
|
|
if (!GetHasGPToSpend(oPC, stItem.nCost))
|
|
{
|
|
DeleteLocalInt(oPC,"X2_CRAFT_SUCCESS");
|
|
FloatingTextStrRefOnCreature(86675,oPC);
|
|
DestroyObject(oRet);
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
//TakeGoldFromCreature(stItem.nCost, oPC,TRUE);
|
|
SpendGP(oPC, stItem.nCost);
|
|
IPCopyItemProperties(oMajor,oRet);
|
|
}
|
|
// set success variable for conversation
|
|
SetLocalInt(oPC,"X2_CRAFT_SUCCESS",TRUE);
|
|
}
|
|
else
|
|
{
|
|
//TakeGoldFromCreature(stItem.nCost / 4, oPC,TRUE);
|
|
SpendGP(oPC, stItem.nCost/4);
|
|
// make sure there is no success
|
|
DeleteLocalInt(oPC,"X2_CRAFT_SUCCESS");
|
|
}
|
|
|
|
// Destroy first material component
|
|
DestroyObject (oMajor);
|
|
|
|
// if we are running in a container, destroy the second material component as well
|
|
if (nMode == X2_CI_CRAFTMODE_CONTAINER || nMode == X2_CI_CRAFTMODE_ASSEMBLE)
|
|
{
|
|
DestroyObject (oMinor);
|
|
}
|
|
int nRet = (oRet != OBJECT_INVALID);
|
|
return nRet;
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Retrieve craft information on a certain item
|
|
// -----------------------------------------------------------------------------
|
|
struct craft_struct CIGetCraftItemStructFrom2DA(string s2DA, int nRow, int nItemNo)
|
|
{
|
|
struct craft_struct stRet;
|
|
string sNumber = IntToString(nItemNo);
|
|
|
|
stRet.nRow = nRow;
|
|
string sLabel = Get2DACache(s2DA,"Label"+ sNumber, nRow);
|
|
if (sLabel == "")
|
|
{
|
|
return stRet; // empty, no need to read further
|
|
}
|
|
int nStrRef = StringToInt(sLabel);
|
|
if (nStrRef != 0) // Handle bioware StrRefs
|
|
{
|
|
sLabel = GetStringByStrRef(nStrRef);
|
|
}
|
|
stRet.sLabel = sLabel;
|
|
stRet.nDC = StringToInt(Get2DACache(s2DA,"DC"+ sNumber, nRow));
|
|
stRet.nCost = StringToInt(Get2DACache(s2DA,"CostGP"+ sNumber, nRow));
|
|
stRet.sResRef = Get2DACache(s2DA,"ResRef"+ sNumber, nRow);
|
|
|
|
return stRet;
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Return the cost
|
|
// -----------------------------------------------------------------------------
|
|
int CIGetItemPartModificationCost(object oOldItem, int nPart)
|
|
{
|
|
int nRet = StringToInt(Get2DACache(X2_IP_ARMORPARTS_2DA,"CraftCost",nPart));
|
|
nRet = (GetGoldPieceValue(oOldItem) / 100 * nRet);
|
|
|
|
// minimum cost for modification is 1 gp
|
|
if (nRet == 0)
|
|
{
|
|
nRet =1;
|
|
}
|
|
return nRet;
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Return the DC for modifying a certain armor part on oOldItem
|
|
// -----------------------------------------------------------------------------
|
|
int CIGetItemPartModificationDC(object oOldItem, int nPart)
|
|
{
|
|
int nRet = StringToInt(Get2DACache(X2_IP_ARMORPARTS_2DA,"CraftDC",nPart));
|
|
// minimum cost for modification is 1 gp
|
|
return nRet;
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// returns the dc
|
|
// dc to modify oOlditem to look like oNewItem
|
|
// -----------------------------------------------------------------------------
|
|
int CIGetArmorModificationCost(object oOldItem, object oNewItem)
|
|
{
|
|
int nTotal = 0;
|
|
int nPart;
|
|
for (nPart = 0; nPart<ITEM_APPR_ARMOR_NUM_MODELS; nPart++)
|
|
{
|
|
|
|
if (GetItemAppearance(oOldItem,ITEM_APPR_TYPE_ARMOR_MODEL, nPart) !=GetItemAppearance(oNewItem,ITEM_APPR_TYPE_ARMOR_MODEL, nPart))
|
|
{
|
|
nTotal+= CIGetItemPartModificationCost(oOldItem,nPart);
|
|
}
|
|
}
|
|
|
|
// Modification Cost should not exceed value of old item +1 GP
|
|
if (nTotal > GetGoldPieceValue(oOldItem))
|
|
{
|
|
nTotal = GetGoldPieceValue(oOldItem)+1;
|
|
}
|
|
return nTotal;
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// returns the cost in gold piece that it would
|
|
// cost to modify oOlditem to look like oNewItem
|
|
// -----------------------------------------------------------------------------
|
|
int CIGetArmorModificationDC(object oOldItem, object oNewItem)
|
|
{
|
|
int nTotal = 0;
|
|
int nPart;
|
|
int nDC =0;
|
|
for (nPart = 0; nPart<ITEM_APPR_ARMOR_NUM_MODELS; nPart++)
|
|
{
|
|
|
|
if (GetItemAppearance(oOldItem,ITEM_APPR_TYPE_ARMOR_MODEL, nPart) !=GetItemAppearance(oNewItem,ITEM_APPR_TYPE_ARMOR_MODEL, nPart))
|
|
{
|
|
nDC = CIGetItemPartModificationDC(oOldItem,nPart);
|
|
if (nDC>nTotal)
|
|
{
|
|
nTotal = nDC;
|
|
}
|
|
}
|
|
}
|
|
|
|
nTotal = GetItemACValue(oOldItem) + nTotal + 5;
|
|
|
|
return nTotal;
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// returns TRUE if the spell matching nSpellID is prevented from being used
|
|
// with the CraftFeat matching nFeatID
|
|
// This is controlled in des_crft_spells.2da
|
|
// -----------------------------------------------------------------------------
|
|
int CIGetIsSpellRestrictedFromCraftFeat(int nSpellID, int nFeatID)
|
|
{
|
|
string sCol;
|
|
switch(nFeatID)
|
|
{
|
|
case X2_CI_BREWPOTION_FEAT_ID: sCol = "NoPotion"; break;
|
|
case X2_CI_SCRIBESCROLL_FEAT_ID: sCol = "NoScroll"; break;
|
|
case X2_CI_CRAFTWAND_FEAT_ID:
|
|
case X2_CI_CRAFTROD_FEAT_ID:
|
|
case X2_CI_CRAFTSTAFF_FEAT_ID: sCol = "NoWand"; break;
|
|
}
|
|
return !(!StringToInt(Get2DACache(X2_CI_CRAFTING_SP_2DA,sCol,nSpellID)));
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Retrieve the row in des_crft_bmat too look up receipe
|
|
// -----------------------------------------------------------------------------
|
|
int CIGetCraftingReceipeRow(int nMode, object oMajor, object oMinor, int nSkill)
|
|
{
|
|
if (nMode == X2_CI_CRAFTMODE_CONTAINER || nMode == X2_CI_CRAFTMODE_ASSEMBLE )
|
|
{
|
|
int nMinorId = StringToInt(Get2DACache("des_crft_amat",GetTag(oMinor),1));
|
|
int nMajorId = StringToInt(Get2DACache("des_crft_bmat",GetTag(oMajor),nMinorId));
|
|
return nMajorId;
|
|
}
|
|
else if (nMode == X2_CI_CRAFTMODE_BASE_ITEM)
|
|
{
|
|
int nLookUpRow;
|
|
string sTag = GetTag(oMajor);
|
|
switch (nSkill)
|
|
{
|
|
case 26: nLookUpRow =1 ; break;
|
|
case 25: nLookUpRow= 2 ; break;
|
|
}
|
|
int nRet = StringToInt(Get2DACache(X2_CI_CRAFTING_MAT_2DA,sTag,nLookUpRow));
|
|
return nRet;
|
|
}
|
|
else
|
|
{
|
|
return 0; // error
|
|
}
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// used to set all variable required for the crafting conversation
|
|
// (Used materials, number of choises, 2da row, skill and mode)
|
|
// -----------------------------------------------------------------------------
|
|
void CISetupCraftingConversation(object oPC, int nNumber, int nSkill, int nReceipe, object oMajor, object oMinor, int nMode)
|
|
{
|
|
|
|
SetLocalObject(oPC,"X2_CI_CRAFT_MAJOR",oMajor);
|
|
if (nMode == X2_CI_CRAFTMODE_CONTAINER || nMode == X2_CI_CRAFTMODE_ASSEMBLE )
|
|
{
|
|
SetLocalObject(oPC,"X2_CI_CRAFT_MINOR", oMinor);
|
|
}
|
|
SetLocalInt(oPC,"X2_CI_CRAFT_NOOFITEMS",nNumber); // number of crafting choises for this material
|
|
SetLocalInt(oPC,"X2_CI_CRAFT_SKILL",nSkill); // skill used (craft armor or craft waeapon)
|
|
SetLocalInt(oPC,"X2_CI_CRAFT_RESULTROW",nReceipe); // number of crafting choises for this material
|
|
SetLocalInt(oPC,"X2_CI_CRAFT_MODE",nMode);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// oItem - The item used for crafting
|
|
// -----------------------------------------------------------------------------
|
|
struct craft_receipe_struct CIGetCraftingModeFromTarget(object oPC,object oTarget, object oItem = OBJECT_INVALID)
|
|
{
|
|
struct craft_receipe_struct stStruct;
|
|
|
|
|
|
if (GetBaseItemType(oItem) == 112 ) // small
|
|
{
|
|
stStruct.oMajor = oItem;
|
|
stStruct.nMode = X2_CI_CRAFTMODE_BASE_ITEM;
|
|
return stStruct;
|
|
}
|
|
|
|
if (!GetIsObjectValid(oTarget))
|
|
{
|
|
stStruct.nMode = X2_CI_CRAFTMODE_INVALID;
|
|
return stStruct;
|
|
}
|
|
|
|
|
|
// A small craftitem was used on a large one
|
|
if (GetBaseItemType(oItem) == 110 ) // small
|
|
{
|
|
if (GetBaseItemType(oTarget) == 109) // large
|
|
{
|
|
stStruct.nMode = X2_CI_CRAFTMODE_ASSEMBLE; // Mode is ASSEMBLE
|
|
stStruct.oMajor = oTarget;
|
|
stStruct.oMinor = oItem;
|
|
return stStruct;
|
|
}
|
|
else
|
|
{
|
|
FloatingTextStrRefOnCreature(84201,oPC);
|
|
}
|
|
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// *** CONTAINER IS NO LONGER USED IN OFFICIAL CAMPAIGN
|
|
// BUT CODE LEFT IN FOR COMMUNITY.
|
|
// THE FOLLOWING CONDITION IS NEVER TRUE FOR THE OC (no crafting container)
|
|
// To reactivate, create a container with tag x2_it_craftcont
|
|
int bCraftCont = (GetTag(oTarget) == "x2_it_craftcont");
|
|
|
|
|
|
if (bCraftCont == TRUE)
|
|
{
|
|
// First item in container is baseitem .. mode = baseitem
|
|
if ( GetBaseItemType(GetFirstItemInInventory(oTarget)) == 112)
|
|
{
|
|
stStruct.nMode = X2_CI_CRAFTMODE_BASE_ITEM;
|
|
stStruct.oMajor = GetFirstItemInInventory(oTarget);
|
|
return stStruct;
|
|
}
|
|
else
|
|
{
|
|
object oTest = GetFirstItemInInventory(oTarget);
|
|
int nCount =1;
|
|
int bMajor = FALSE;
|
|
int bMinor = FALSE;
|
|
// No item in inventory ... mode = fail
|
|
if (!GetIsObjectValid(oTest))
|
|
{
|
|
FloatingTextStrRefOnCreature(84200,oPC);
|
|
stStruct.nMode = X2_CI_CRAFTMODE_INVALID;
|
|
return stStruct;
|
|
}
|
|
else
|
|
{
|
|
while (GetIsObjectValid(oTest) && nCount <3)
|
|
{
|
|
if (GetBaseItemType(oTest) == 109)
|
|
{
|
|
stStruct.oMajor = oTest;
|
|
bMajor = TRUE;
|
|
}
|
|
else if (GetBaseItemType(oTest) == 110)
|
|
{
|
|
stStruct.oMinor = oTest;
|
|
bMinor = TRUE;
|
|
}
|
|
else if ( GetBaseItemType(oTest) == 112)
|
|
{
|
|
stStruct.nMode = X2_CI_CRAFTMODE_BASE_ITEM;
|
|
stStruct.oMajor = oTest;
|
|
return stStruct;
|
|
}
|
|
oTest = GetNextItemInInventory(oTarget);
|
|
if (GetIsObjectValid(oTest))
|
|
{
|
|
nCount ++;
|
|
}
|
|
}
|
|
|
|
if (nCount >2)
|
|
{
|
|
FloatingTextStrRefOnCreature(84356,oPC);
|
|
stStruct.nMode = X2_CI_CRAFTMODE_INVALID;
|
|
return stStruct;
|
|
}
|
|
else if (nCount <2)
|
|
{
|
|
FloatingTextStrRefOnCreature(84356,oPC);
|
|
stStruct.nMode = X2_CI_CRAFTMODE_INVALID;
|
|
return stStruct;
|
|
}
|
|
|
|
if (bMajor && bMinor)
|
|
{
|
|
stStruct.nMode = X2_CI_CRAFTMODE_CONTAINER;
|
|
return stStruct;
|
|
}
|
|
else
|
|
{
|
|
FloatingTextStrRefOnCreature(84356,oPC);
|
|
//FloatingTextStringOnCreature("Temp: Wrong combination of items in the crafting container",oPC);
|
|
stStruct.nMode = X2_CI_CRAFTMODE_INVALID;
|
|
return stStruct;
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// not a container but a baseitem
|
|
if (GetBaseItemType(oTarget) == 112)
|
|
{
|
|
stStruct.nMode = X2_CI_CRAFTMODE_BASE_ITEM;
|
|
stStruct.oMajor = oTarget;
|
|
return stStruct;
|
|
|
|
}
|
|
else
|
|
{
|
|
if (GetBaseItemType(oTarget) == 109 || GetBaseItemType(oTarget) == 110)
|
|
{
|
|
FloatingTextStrRefOnCreature(84357,oPC);
|
|
stStruct.nMode = X2_CI_CRAFTMODE_INVALID;
|
|
return stStruct;
|
|
}
|
|
else
|
|
{
|
|
FloatingTextStrRefOnCreature(84357,oPC);
|
|
// not a valid item
|
|
stStruct.nMode = X2_CI_CRAFTMODE_INVALID;
|
|
return stStruct;
|
|
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// *** Crafting Conversation Functions ***
|
|
// -----------------------------------------------------------------------------
|
|
int CIGetInModWeaponOrArmorConv(object oPC)
|
|
{
|
|
return GetLocalInt(oPC,"X2_L_CRAFT_MODIFY_CONVERSATION");
|
|
}
|
|
|
|
|
|
void CISetCurrentModMode(object oPC, int nMode)
|
|
{
|
|
if (nMode == X2_CI_MODMODE_INVALID)
|
|
{
|
|
DeleteLocalInt(oPC,"X2_L_CRAFT_MODIFY_MODE");
|
|
}
|
|
else
|
|
{
|
|
SetLocalInt(oPC,"X2_L_CRAFT_MODIFY_MODE",nMode);
|
|
}
|
|
}
|
|
|
|
int CIGetCurrentModMode(object oPC)
|
|
{
|
|
return GetLocalInt(oPC,"X2_L_CRAFT_MODIFY_MODE");
|
|
}
|
|
|
|
|
|
object CIGetCurrentModBackup(object oPC)
|
|
{
|
|
return GetLocalObject(GetPCSpeaker(),"X2_O_CRAFT_MODIFY_BACKUP");
|
|
}
|
|
|
|
object CIGetCurrentModItem(object oPC)
|
|
{
|
|
return GetLocalObject(GetPCSpeaker(),"X2_O_CRAFT_MODIFY_ITEM");
|
|
}
|
|
|
|
|
|
void CISetCurrentModBackup(object oPC, object oBackup)
|
|
{
|
|
SetLocalObject(GetPCSpeaker(),"X2_O_CRAFT_MODIFY_BACKUP",oBackup);
|
|
}
|
|
|
|
void CISetCurrentModItem(object oPC, object oItem)
|
|
{
|
|
SetLocalObject(GetPCSpeaker(),"X2_O_CRAFT_MODIFY_ITEM",oItem);
|
|
}
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// * This does multiple things:
|
|
// - store the part currently modified
|
|
// - setup the custom token for the conversation
|
|
// - zoom the camera to that part
|
|
// -----------------------------------------------------------------------------
|
|
void CISetCurrentModPart(object oPC, int nPart, int nStrRef)
|
|
{
|
|
SetLocalInt(oPC,"X2_TAILOR_CURRENT_PART",nPart);
|
|
|
|
if (CIGetCurrentModMode(oPC) == X2_CI_MODMODE_ARMOR)
|
|
{
|
|
|
|
// * Make the camera float near the PC
|
|
float fFacing = GetFacing(oPC) + 180.0;
|
|
|
|
if (nPart == ITEM_APPR_ARMOR_MODEL_LSHOULDER || nPart == ITEM_APPR_ARMOR_MODEL_LFOREARM ||
|
|
nPart == ITEM_APPR_ARMOR_MODEL_LHAND || nPart == ITEM_APPR_ARMOR_MODEL_LBICEP)
|
|
{
|
|
fFacing += 80.0;
|
|
}
|
|
|
|
if (nPart == ITEM_APPR_ARMOR_MODEL_RSHOULDER || nPart == ITEM_APPR_ARMOR_MODEL_RFOREARM ||
|
|
nPart == ITEM_APPR_ARMOR_MODEL_RHAND || nPart == ITEM_APPR_ARMOR_MODEL_RBICEP)
|
|
{
|
|
fFacing -= 80.0;
|
|
}
|
|
|
|
float fPitch = 75.0;
|
|
if (fFacing > 359.0)
|
|
{
|
|
fFacing -=359.0;
|
|
}
|
|
|
|
float fDistance = 3.5f;
|
|
if (nPart == ITEM_APPR_ARMOR_MODEL_PELVIS || nPart == ITEM_APPR_ARMOR_MODEL_BELT )
|
|
{
|
|
fDistance = 2.0f;
|
|
}
|
|
|
|
if (nPart == ITEM_APPR_ARMOR_MODEL_LSHOULDER || nPart == ITEM_APPR_ARMOR_MODEL_RSHOULDER )
|
|
{
|
|
fPitch = 50.0f;
|
|
fDistance = 3.0f;
|
|
}
|
|
else if (nPart == ITEM_APPR_ARMOR_MODEL_LFOREARM || nPart == ITEM_APPR_ARMOR_MODEL_LHAND)
|
|
{
|
|
fDistance = 2.0f;
|
|
fPitch = 60.0f;
|
|
}
|
|
else if (nPart == ITEM_APPR_ARMOR_MODEL_NECK)
|
|
{
|
|
fPitch = 90.0f;
|
|
}
|
|
else if (nPart == ITEM_APPR_ARMOR_MODEL_RFOOT || nPart == ITEM_APPR_ARMOR_MODEL_LFOOT )
|
|
{
|
|
fDistance = 3.5f;
|
|
fPitch = 47.0f;
|
|
}
|
|
else if (nPart == ITEM_APPR_ARMOR_MODEL_LTHIGH || nPart == ITEM_APPR_ARMOR_MODEL_RTHIGH )
|
|
{
|
|
fDistance = 2.5f;
|
|
fPitch = 65.0f;
|
|
}
|
|
else if ( nPart == ITEM_APPR_ARMOR_MODEL_RSHIN || nPart == ITEM_APPR_ARMOR_MODEL_LSHIN )
|
|
{
|
|
fDistance = 3.5f;
|
|
fPitch = 95.0f;
|
|
}
|
|
|
|
if (GetRacialType(oPC) == RACIAL_TYPE_HALFORC)
|
|
{
|
|
fDistance += 1.0f;
|
|
}
|
|
|
|
SetCameraFacing(fFacing, fDistance, fPitch,CAMERA_TRANSITION_TYPE_VERY_FAST) ;
|
|
}
|
|
|
|
int nCost = GetLocalInt(oPC,"X2_TAILOR_CURRENT_COST");
|
|
int nDC = GetLocalInt(oPC,"X2_TAILOR_CURRENT_DC");
|
|
|
|
SetCustomToken(X2_CI_MODIFYARMOR_GP_CTOKENBASE,IntToString(nCost));
|
|
SetCustomToken(X2_CI_MODIFYARMOR_GP_CTOKENBASE+1,IntToString(nDC));
|
|
|
|
|
|
SetCustomToken(XP_IP_ITEMMODCONVERSATION_CTOKENBASE,GetStringByStrRef(nStrRef));
|
|
}
|
|
|
|
int CIGetCurrentModPart(object oPC)
|
|
{
|
|
return GetLocalInt(oPC,"X2_TAILOR_CURRENT_PART");
|
|
}
|
|
|
|
|
|
void CISetDefaultModItemCamera(object oPC)
|
|
{
|
|
float fDistance = 3.5f;
|
|
float fPitch = 75.0f;
|
|
float fFacing;
|
|
|
|
if (CIGetCurrentModMode(oPC) == X2_CI_MODMODE_ARMOR)
|
|
{
|
|
fFacing = GetFacing(oPC) + 180.0;
|
|
if (fFacing > 359.0)
|
|
{
|
|
fFacing -=359.0;
|
|
}
|
|
}
|
|
else if (CIGetCurrentModMode(oPC) == X2_CI_MODMODE_WEAPON)
|
|
{
|
|
fFacing = GetFacing(oPC) + 180.0;
|
|
fFacing -= 90.0;
|
|
if (fFacing > 359.0)
|
|
{
|
|
fFacing -=359.0;
|
|
}
|
|
}
|
|
|
|
SetCameraFacing(fFacing, fDistance, fPitch,CAMERA_TRANSITION_TYPE_VERY_FAST) ;
|
|
}
|
|
|
|
void CIUpdateModItemCostDC(object oPC, int nDC, int nCost)
|
|
{
|
|
SetLocalInt(oPC,"X2_TAILOR_CURRENT_COST", nCost);
|
|
SetLocalInt(oPC,"X2_TAILOR_CURRENT_DC",nDC);
|
|
SetCustomToken(X2_CI_MODIFYARMOR_GP_CTOKENBASE,IntToString(nCost));
|
|
SetCustomToken(X2_CI_MODIFYARMOR_GP_CTOKENBASE+1,IntToString(nDC));
|
|
}
|
|
|
|
|
|
// dc to modify oOlditem to look like oNewItem
|
|
int CIGetWeaponModificationCost(object oOldItem, object oNewItem)
|
|
{
|
|
int nTotal = 0;
|
|
int nPart;
|
|
for (nPart = 0; nPart<=2; nPart++)
|
|
{
|
|
if (GetItemAppearance(oOldItem,ITEM_APPR_TYPE_WEAPON_MODEL, nPart) !=GetItemAppearance(oNewItem,ITEM_APPR_TYPE_WEAPON_MODEL, nPart))
|
|
{
|
|
nTotal+= (GetGoldPieceValue(oOldItem)/4)+1;
|
|
}
|
|
}
|
|
|
|
// Modification Cost should not exceed value of old item +1 GP
|
|
if (nTotal > GetGoldPieceValue(oOldItem))
|
|
{
|
|
nTotal = GetGoldPieceValue(oOldItem)+1;
|
|
}
|
|
return nTotal;
|
|
}
|
|
|
|
int GetMagicalArtisanFeat(int nCraftingFeat)
|
|
{
|
|
int nReturn = 0;
|
|
switch(nCraftingFeat)
|
|
{
|
|
case FEAT_BREW_POTION:
|
|
{
|
|
nReturn = FEAT_MAGICAL_ARTISAN_BREW_POTION;
|
|
break;
|
|
}
|
|
case FEAT_SCRIBE_SCROLL:
|
|
{
|
|
nReturn = FEAT_MAGICAL_ARTISAN_SCRIBE_SCROLL;
|
|
break;
|
|
}
|
|
case FEAT_INSCRIBE_RUNE:
|
|
{
|
|
nReturn = FEAT_MAGICAL_ARTISAN_INSCRIBE_RUNE;
|
|
break;
|
|
}
|
|
case FEAT_ATTUNE_GEM:
|
|
{
|
|
nReturn = FEAT_MAGICAL_ARTISAN_ATTUNE_GEM;
|
|
break;
|
|
}
|
|
case FEAT_CRAFT_ARMS_ARMOR:
|
|
{
|
|
nReturn = FEAT_MAGICAL_ARTISAN_CRAFT_MAGIC_ARMS;
|
|
break;
|
|
}
|
|
case FEAT_CRAFT_ROD:
|
|
{
|
|
nReturn = FEAT_MAGICAL_ARTISAN_CRAFT_ROD;
|
|
break;
|
|
}
|
|
case FEAT_CRAFT_STAFF:
|
|
{
|
|
nReturn = FEAT_MAGICAL_ARTISAN_CRAFT_STAFF;
|
|
break;
|
|
}
|
|
case FEAT_CRAFT_WAND:
|
|
{
|
|
nReturn = FEAT_MAGICAL_ARTISAN_CRAFT_WAND;
|
|
break;
|
|
}
|
|
case FEAT_CRAFT_WONDROUS:
|
|
{
|
|
nReturn = FEAT_MAGICAL_ARTISAN_CRAFT_WONDROUS;
|
|
break;
|
|
}
|
|
case FEAT_FORGE_RING:
|
|
{
|
|
nReturn = FEAT_MAGICAL_ARTISAN_FORGE_RING;
|
|
break;
|
|
}
|
|
case FEAT_CRAFT_SKULL_TALISMAN:
|
|
{
|
|
nReturn = FEAT_MAGICAL_ARTISAN_CRAFT_SKULL_TALISMAN;
|
|
break;
|
|
}
|
|
default:
|
|
{
|
|
if(DEBUG) DoDebug("GetMagicalArtisanFeat: invalid crafting feat");
|
|
break;
|
|
}
|
|
}
|
|
return nReturn;
|
|
}
|
|
|
|
int GetPnPItemXPCost(int nCost, int bEpic)
|
|
{
|
|
int nXP = nCost / 25;
|
|
if(bEpic) nXP = (nCost / 100) + 10000;
|
|
return nXP;
|
|
}
|
|
|
|
int GetCraftingTime(int nCost)
|
|
{
|
|
int nTemp = nCost / 1000;
|
|
if(nCost % 1000) nTemp++;
|
|
float fDelay;
|
|
switch(GetPRCSwitch(PRC_CRAFTING_TIME_SCALE))
|
|
{
|
|
case 0: fDelay = HoursToSeconds(nTemp); break; //1 hour/1000gp, default
|
|
case 1: fDelay = 0.0; break; //off, no delay
|
|
case 2: fDelay = RoundsToSeconds(nTemp); break; //1 round/1000gp
|
|
case 3: fDelay = TurnsToSeconds(nTemp); break; //1 turn/1000gp
|
|
case 4: fDelay = HoursToSeconds(nTemp); break; //1 hour/1000gp
|
|
case 5: fDelay = 24 * HoursToSeconds(nTemp); break; //1 day/1000gp
|
|
}
|
|
int nMultiplyer = GetPRCSwitch(PRC_CRAFT_TIMER_MULTIPLIER);
|
|
if(nMultiplyer)
|
|
fDelay *= (IntToFloat(nMultiplyer) / 100.0);
|
|
return FloatToInt(fDelay / 6);
|
|
}
|
|
|
|
int GetModifiedGoldCost(int nCost, object oPC, int nCraftingFeat)
|
|
{
|
|
if(nCost == 0)
|
|
return nCost;
|
|
float fCost = IntToFloat(nCost);
|
|
if(GetHasFeat(FEAT_EXTRAORDINARY_ARTISAN_I , oPC)) fCost *= 0.75;
|
|
if(GetHasFeat(FEAT_EXTRAORDINARY_ARTISAN_II , oPC)) fCost *= 0.75;
|
|
if(GetHasFeat(FEAT_EXTRAORDINARY_ARTISAN_III , oPC)) fCost *= 0.75;
|
|
if(GetHasFeat(GetMagicalArtisanFeat(nCraftingFeat), oPC)) fCost *= 0.75;
|
|
if(GetHasFeat(FEAT_SHIELD_DWARF_WARDER, oPC) && FEAT_CRAFT_ARMS_ARMOR == nCraftingFeat) fCost *= 0.95;
|
|
int nScale = GetPRCSwitch(PRC_CRAFTING_COST_SCALE);
|
|
if(nScale > 0)
|
|
{ //you're not getting away with negative values that easily :P
|
|
fCost = fCost * IntToFloat(nScale) / 100.0;
|
|
}
|
|
return FloatToInt(fCost);
|
|
}
|
|
|
|
int GetModifiedXPCost(int nCost, object oPC, int nCraftingFeat)
|
|
{
|
|
if(nCost == 0)
|
|
return nCost;
|
|
float fCost = IntToFloat(nCost);
|
|
if(GetHasFeat(FEAT_LEGENDARY_ARTISAN_I , oPC)) fCost *= 0.75;
|
|
if(GetHasFeat(FEAT_LEGENDARY_ARTISAN_II , oPC)) fCost *= 0.75;
|
|
if(GetHasFeat(FEAT_LEGENDARY_ARTISAN_III , oPC)) fCost *= 0.75;
|
|
if(GetHasFeat(GetMagicalArtisanFeat(nCraftingFeat), oPC)) fCost *= 0.75;
|
|
int nScale = GetPRCSwitch(PRC_CRAFTING_COST_SCALE);
|
|
if(nScale > 0)
|
|
{ //you're not getting away with negative values that easily :P
|
|
fCost = fCost * IntToFloat(nScale) / 100.0;
|
|
}
|
|
return FloatToInt(fCost);
|
|
}
|
|
|
|
int GetModifiedTimeCost(int nCost, object oPC, int nCraftingFeat)
|
|
{
|
|
if(nCost == 0)
|
|
return nCost;
|
|
float fCost = IntToFloat(nCost);
|
|
if(GetLevelByClass(CLASS_TYPE_MAESTER, oPC)) fCost /= 2;
|
|
if(GetHasFeat(FEAT_EXCEPTIONAL_ARTISAN_I , oPC)) fCost *= 0.75;
|
|
if(GetHasFeat(FEAT_EXCEPTIONAL_ARTISAN_II , oPC)) fCost *= 0.75;
|
|
if(GetHasFeat(FEAT_EXCEPTIONAL_ARTISAN_III , oPC)) fCost *= 0.75;
|
|
if(GetHasFeat(GetMagicalArtisanFeat(nCraftingFeat), oPC)) fCost *= 0.75;
|
|
if(nCraftingFeat == FEAT_BREW_POTION)
|
|
{ //master alchemist stuff here
|
|
if(GetHasFeat(FEAT_BREW_4PERDAY , oPC)) fCost /= 4;
|
|
else if(GetHasFeat(FEAT_BREW_3PERDAY , oPC)) fCost /= 3;
|
|
else if(GetHasFeat(FEAT_BREW_2PERDAY , oPC)) fCost /= 2;
|
|
}
|
|
int nScale = GetPRCSwitch(PRC_CRAFTING_COST_SCALE);
|
|
if(nScale > 0)
|
|
{ //you're not getting away with negative values that easily :P
|
|
fCost = fCost * IntToFloat(nScale) / 100.0;
|
|
}
|
|
return FloatToInt(fCost);
|
|
}
|
|
|
|
struct craft_cost_struct GetModifiedCostsFromBase(int nCost, object oPC, int nCraftingFeat, int bEpic)
|
|
{
|
|
struct craft_cost_struct costs;
|
|
|
|
costs.nGoldCost = GetModifiedGoldCost(nCost / 2, oPC, nCraftingFeat);
|
|
costs.nXPCost = GetModifiedXPCost(GetPnPItemXPCost(nCost, bEpic), oPC, nCraftingFeat);
|
|
costs.nTimeCost = GetModifiedTimeCost(GetCraftingTime(nCost), oPC, nCraftingFeat);
|
|
|
|
return costs;
|
|
}
|
|
|
|
//Checks crafting prereqs for warlocks
|
|
int CheckImbueItem(object oPC, int nSpell)
|
|
{
|
|
if(!GetHasFeat(FEAT_IMBUE_ITEM, oPC)) return FALSE;
|
|
int nImbueDC;
|
|
int bArcane = TRUE;
|
|
int nLevel;
|
|
int nArcaneSpellLevel;
|
|
int nDivineSpellLevel;
|
|
string sTemp;
|
|
|
|
sTemp = Get2DACache("spells", "Wiz_Sorc", nSpell);
|
|
if(sTemp == "")
|
|
{
|
|
sTemp = Get2DACache("spells", "Bard", nSpell);
|
|
if(sTemp == "")
|
|
{
|
|
bArcane = FALSE; //now checking the divine classes
|
|
sTemp = Get2DACache("spells", "Cleric", nSpell);
|
|
if(sTemp == "")
|
|
{
|
|
sTemp = Get2DACache("spells", "Druid", nSpell);
|
|
if(sTemp == "")
|
|
{
|
|
sTemp = Get2DACache("spells", "Paladin", nSpell);
|
|
if(sTemp == "")
|
|
{
|
|
sTemp = Get2DACache("spells", "Ranger", nSpell);
|
|
if(sTemp == "")
|
|
{
|
|
if(DEBUG) DoDebug("CheckImbueItem: ERROR - spell is neither arcane nor divine");
|
|
return FALSE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
//warlocks with deceive item get to take 10
|
|
return GetPRCIsSkillSuccessful(oPC, SKILL_USE_MAGIC_DEVICE, StringToInt(sTemp) + (bArcane ? 15 : 25), GetHasFeat(FEAT_DECEIVE_ITEM, oPC) ? 10 : -1);
|
|
}
|
|
|
|
// checks alternative crafting, eg. warlock, artificer
|
|
int CheckAlternativeCrafting(object oPC, int nSpell, struct craft_cost_struct costs)
|
|
{
|
|
//nSpell1 = (PRCGetHasSpell(nTemp, oPC)) ? -1 : StringToInt(Get2DACache("spells", "Innate", nTemp)) + 20;
|
|
//if(nSpell1 == -1) nSpell1 = (GetPRCIsSkillSuccessful(oPC, SKILL_USE_MAGIC_DEVICE, nSpell1, bTake10)) ? -1 : nSpell1;
|
|
|
|
int bChecked = FALSE;
|
|
int bSuccess = FALSE;
|
|
int i;
|
|
int bTake10 = GetHasFeat(FEAT_SKILL_MASTERY_ARTIFICER, oPC) ? 10 : -1;
|
|
|
|
//artificer crafting check
|
|
if(!bSuccess && GetLocalInt(oPC, "ArtificerCrafting"))
|
|
{
|
|
bChecked = TRUE;
|
|
//bSuccess = CheckImbueItem(oPC, nSpell);
|
|
for(i = 0; i < costs.nTimeCost; i++)
|
|
{
|
|
bSuccess = GetPRCIsSkillSuccessful(oPC, SKILL_USE_MAGIC_DEVICE, StringToInt(Get2DACache("spells", "Innate", nSpell)) + 20, bTake10);
|
|
if(bSuccess)
|
|
break;
|
|
}
|
|
}
|
|
//warlock crafting check
|
|
if(!bSuccess && GetLocalInt(oPC, "UsingImbueItem"))
|
|
{
|
|
bChecked = TRUE;
|
|
bSuccess = CheckImbueItem(oPC, nSpell);
|
|
}
|
|
|
|
|
|
if(!bChecked)
|
|
return TRUE; //we never checked because we had the actual spell, so successful
|
|
else
|
|
return bSuccess;
|
|
}
|
|
|
|
int GetAlternativeCasterLevel(object oPC, int nLevel)
|
|
{
|
|
// Battlesmith adds 3x class level to caster level for casting
|
|
nLevel += GetLevelByClass(CLASS_TYPE_BATTLESMITH) * 3;
|
|
nLevel += GetLevelByClass(CLASS_TYPE_IRONSOUL_FORGEMASTER) * 3;
|
|
if(GetLocalInt(oPC, "UsingImbueItem"))
|
|
{
|
|
nLevel = max(GetLocalInt(oPC, "InvokerLevel"), nLevel);
|
|
}
|
|
if(GetLocalInt(oPC, "ArtificerCrafting"))
|
|
{
|
|
nLevel = max(GetLevelByClass(CLASS_TYPE_ARTIFICER, oPC), nLevel);
|
|
}
|
|
return nLevel;
|
|
}
|
|
|
|
|
|
// Test main
|
|
//void main(){}
|