2024-02-11 14:01:05 -05:00
|
|
|
/*
|
|
|
|
----------------
|
|
|
|
Pearl of Black Doubt
|
|
|
|
|
|
|
|
tob_dmnd_prlbdt
|
|
|
|
----------------
|
|
|
|
|
|
|
|
15/07/07 by Stratovarius
|
|
|
|
*/ /** @file
|
|
|
|
|
|
|
|
Pearl of Black Doubt
|
|
|
|
|
|
|
|
Diamond Mind (Stance)
|
|
|
|
Level: Swordsage 3, Warmage 3
|
|
|
|
Prerequisite: One Diamond Mind maneuver.
|
|
|
|
Initiation Action: 1 Swift Action
|
|
|
|
Range: Personal
|
|
|
|
Target: You
|
|
|
|
Duration: Stance
|
|
|
|
|
2025-06-10 17:38:59 -04:00
|
|
|
With every miss, your opponents become more uncertain, their doubt growing
|
|
|
|
like an irritating pearl in the mouth of a helpless oyster.
|
|
|
|
|
|
|
|
Whenever a foe swings and misses you, you gain +2 AC.
|
2025-05-27 22:29:19 -04:00
|
|
|
*/
|
2024-02-11 14:01:05 -05:00
|
|
|
#include "tob_inc_move"
|
|
|
|
#include "tob_movehook"
|
|
|
|
////#include "prc_alterations"
|
|
|
|
|
2025-05-27 22:29:19 -04:00
|
|
|
int GetApproximateAPR(object oCreature)
|
|
|
|
{
|
|
|
|
int nBAB = GetBaseAttackBonus(oCreature);
|
|
|
|
int nAttacks = 1;
|
|
|
|
|
2025-06-10 17:38:59 -04:00
|
|
|
if (nBAB >= 6) nAttacks++;
|
2025-05-27 22:29:19 -04:00
|
|
|
if (nBAB >= 11) nAttacks++;
|
|
|
|
if (nBAB >= 16) nAttacks++;
|
|
|
|
|
2025-06-10 17:38:59 -04:00
|
|
|
if (PRCGetHasEffect(EFFECT_TYPE_HASTE, oCreature))
|
2025-05-27 22:29:19 -04:00
|
|
|
{
|
|
|
|
nAttacks++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nAttacks;
|
|
|
|
}
|
|
|
|
|
2025-06-10 17:38:59 -04:00
|
|
|
void main()
|
2024-02-11 14:01:05 -05:00
|
|
|
{
|
2025-06-10 17:38:59 -04:00
|
|
|
//Declare main variables.
|
|
|
|
int nEvent = GetRunningEvent();
|
|
|
|
|
|
|
|
object oItem;
|
|
|
|
object oInitiator;
|
|
|
|
object oTarget = PRCGetSpellTargetObject();
|
|
|
|
|
|
|
|
switch(nEvent)
|
|
|
|
{
|
|
|
|
|
|
|
|
case EVENT_ONHEARTBEAT: oInitiator = OBJECT_SELF; break;
|
|
|
|
case EVENT_ONPLAYERREST_FINISHED: oInitiator = GetLastBeingRested(); break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
oInitiator = OBJECT_SELF;
|
2025-05-27 22:29:19 -04:00
|
|
|
|
2025-06-10 17:38:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if(nEvent == FALSE)
|
|
|
|
{
|
|
|
|
if (!PreManeuverCastCode())
|
|
|
|
{
|
|
|
|
// If code within the PreManeuverCastCode (i.e. UMD) reports FALSE, do not run this spell
|
|
|
|
return;
|
2024-02-11 14:01:05 -05:00
|
|
|
}
|
2025-06-10 17:38:59 -04:00
|
|
|
// End of Spell Cast Hook
|
|
|
|
|
|
|
|
struct maneuver move = EvaluateManeuver(oInitiator, oTarget);
|
|
|
|
|
|
|
|
if(move.bCanManeuver)
|
2025-05-27 22:29:19 -04:00
|
|
|
{
|
2025-06-10 17:38:59 -04:00
|
|
|
object oItem = GetItemInSlot(INVENTORY_SLOT_CHEST, oInitiator);
|
|
|
|
|
|
|
|
IPSafeAddItemProperty(oItem, ItemPropertyOnHitCastSpell(IP_CONST_ONHIT_CASTSPELL_ONHIT_UNIQUEPOWER, 1),
|
|
|
|
9999.0, X2_IP_ADDPROP_POLICY_KEEP_EXISTING, FALSE, FALSE);
|
|
|
|
|
|
|
|
effect eDur = EffectVisualEffect(VFX_DUR_BLUESHIELDPROTECT);
|
|
|
|
eDur = TagEffect(eDur, "PEARL_OF_BLACK_DOUBT");
|
|
|
|
eDur = ExtraordinaryEffect(eDur);
|
|
|
|
|
|
|
|
SPApplyEffectToObject(DURATION_TYPE_TEMPORARY, eDur, oTarget, 9999.0);
|
|
|
|
|
|
|
|
if(DEBUG) DoDebug("PoBD stance activated");
|
|
|
|
|
|
|
|
AddEventScript(oInitiator, EVENT_ONHEARTBEAT, "tob_dmnd_prlbdt", TRUE, FALSE);
|
|
|
|
AddEventScript(oInitiator, EVENT_ONPLAYERREST_FINISHED, "tob_dmnd_prlbdt", TRUE, FALSE);
|
2025-05-27 22:29:19 -04:00
|
|
|
}
|
|
|
|
|
2025-06-10 17:38:59 -04:00
|
|
|
}
|
|
|
|
if(nEvent == EVENT_ONHEARTBEAT)
|
|
|
|
{
|
|
|
|
int nBonus = 0;
|
|
|
|
location lLoc = GetLocation(oTarget);
|
|
|
|
object oEnemy = GetFirstObjectInShape(SHAPE_SPHERE, 5.0, lLoc, TRUE, OBJECT_TYPE_CREATURE);
|
2024-02-11 14:01:05 -05:00
|
|
|
|
2025-06-10 17:38:59 -04:00
|
|
|
while (GetIsObjectValid(oEnemy))
|
|
|
|
{
|
|
|
|
if (GetIsEnemy(oTarget, oEnemy) &&
|
|
|
|
GetIsInCombat(oEnemy) &&
|
|
|
|
GetDistanceBetween(oEnemy, oTarget) <= 5.0 &&
|
|
|
|
GetCurrentAction(oEnemy) == ACTION_ATTACKOBJECT && // Must be attacking
|
|
|
|
GetAttackTarget(oEnemy) == oTarget) // Must be attacking this PC
|
|
|
|
{
|
|
|
|
int nAPR = GetApproximateAPR(oEnemy);
|
|
|
|
nBonus += 2 * nAPR;
|
|
|
|
|
|
|
|
string s = "Enemy: " + GetName(oEnemy) + " APR: " + IntToString(nAPR);
|
|
|
|
if(DEBUG) DoDebug(s, oTarget);
|
|
|
|
}
|
|
|
|
|
|
|
|
oEnemy = GetNextObjectInShape(SHAPE_SPHERE, 5.0, lLoc);
|
|
|
|
}
|
2024-02-11 14:01:05 -05:00
|
|
|
|
2025-06-10 17:38:59 -04:00
|
|
|
if (nBonus > 0)
|
|
|
|
{
|
|
|
|
if(DEBUG) DoDebug("Applying AC Bonus: " + IntToString(nBonus));
|
|
|
|
|
|
|
|
effect eAC = EffectACIncrease(nBonus);
|
|
|
|
eAC = ExtraordinaryEffect(eAC);
|
|
|
|
eAC = TagEffect(eAC, "PEARL_OF_BLACK_DOUBT_BONUS");
|
|
|
|
|
|
|
|
ApplyEffectToObject(DURATION_TYPE_TEMPORARY, eAC, oTarget, 5.9);
|
|
|
|
SetLocalInt(oTarget, "PearlOfBlackDoubtBonus", nBonus);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if(DEBUG) DoDebug("PoBD: No enemies in melee");
|
|
|
|
DeleteLocalInt(oTarget, "PearlOfBlackDoubtBonus");
|
2025-05-27 22:29:19 -04:00
|
|
|
|
2025-06-10 17:38:59 -04:00
|
|
|
effect eEffect = GetFirstEffect(oInitiator);
|
|
|
|
while(GetIsEffectValid(eEffect))
|
|
|
|
{
|
|
|
|
if(GetEffectTag(eEffect) == "PEARL_OF_BLACK_DOUBT_BONUS")
|
|
|
|
RemoveEffect(oInitiator, eEffect);
|
|
|
|
|
|
|
|
eEffect = GetNextEffect(oInitiator);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(nEvent == EVENT_ONPLAYERREST_FINISHED)
|
|
|
|
{
|
|
|
|
effect eEffect = GetFirstEffect(oInitiator);
|
|
|
|
while(GetIsEffectValid(eEffect))
|
|
|
|
{
|
|
|
|
if(GetEffectTag(eEffect) == "PEARL_OF_BLACK_DOUBT")
|
|
|
|
RemoveEffect(oInitiator, eEffect);
|
|
|
|
|
|
|
|
eEffect = GetNextEffect(oInitiator);
|
|
|
|
}
|
2025-05-27 22:29:19 -04:00
|
|
|
|
2025-06-10 17:38:59 -04:00
|
|
|
if(DEBUG) DoDebug("Removing Pearl of Black Doubt");
|
2025-05-27 22:29:19 -04:00
|
|
|
|
2025-06-10 17:38:59 -04:00
|
|
|
RemoveEventScript(oInitiator, EVENT_ONHEARTBEAT, "tob_dmnd_prlbdt", TRUE, FALSE);
|
|
|
|
RemoveEventScript(oInitiator, EVENT_ONPLAYERREST_FINISHED, "tob_dmnd_prlbdt", TRUE, FALSE);
|
|
|
|
}
|
2024-02-11 14:01:05 -05:00
|
|
|
}
|