mirror of
https://github.com/ZhengPeiRu21/mod-individual-progression
synced 2025-11-29 23:44:51 +08:00
add new Bang a Gong quest to onPlayerCompleteQuest player progresses to AQ war phase on completion.
880 lines
42 KiB
C++
880 lines
42 KiB
C++
#include "IndividualProgression.h"
|
|
|
|
class IndividualPlayerProgression : public PlayerScript
|
|
{
|
|
|
|
private:
|
|
static bool IsTBCRaceStartingZone(uint32 mapid, float x, float y, float z)
|
|
{
|
|
Map const *map = sMapMgr->FindMap(mapid, 0);
|
|
uint32 zoneId = map->GetZoneId(0, x, y, z);
|
|
return (zoneId == ZONE_AZUREMYST || zoneId == ZONE_BLOODMYST || zoneId == ZONE_GHOSTLANDS ||
|
|
zoneId == ZONE_EVERSONG || zoneId == ZONE_EXODAR || zoneId == ZONE_SILVERMOON ||
|
|
zoneId == ZONE_VEILED_SEA);
|
|
}
|
|
|
|
public:
|
|
IndividualPlayerProgression() : PlayerScript("IndividualProgression") { }
|
|
|
|
void OnPlayerLogin(Player* player) override
|
|
{
|
|
if (player->getClass() == CLASS_DEATH_KNIGHT && sIndividualProgression->deathKnightStartingProgression && !sIndividualProgression->hasPassedProgression(player, static_cast<ProgressionState>(sIndividualProgression->deathKnightStartingProgression)))
|
|
{
|
|
sIndividualProgression->UpdateProgressionState(player, static_cast<ProgressionState>(sIndividualProgression->deathKnightStartingProgression));
|
|
}
|
|
if (sIndividualProgression->startingProgression && !sIndividualProgression->hasPassedProgression(player, static_cast<ProgressionState>(sIndividualProgression->startingProgression)))
|
|
{
|
|
sIndividualProgression->UpdateProgressionState(player, static_cast<ProgressionState>(sIndividualProgression->startingProgression));
|
|
}
|
|
sIndividualProgression->CheckAdjustments(player);
|
|
}
|
|
|
|
void OnPlayerSetMaxLevel(Player* player, uint32& maxPlayerLevel) override
|
|
{
|
|
if (!sIndividualProgression->enabled || isExcludedFromProgression(player))
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_NAXX40))
|
|
{
|
|
if (sWorld->getIntConfig(CONFIG_MAX_PLAYER_LEVEL) > 60)
|
|
{
|
|
maxPlayerLevel = 60;
|
|
}
|
|
}
|
|
else if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_TBC_TIER_5))
|
|
{
|
|
if (sWorld->getIntConfig(CONFIG_MAX_PLAYER_LEVEL) > 70)
|
|
{
|
|
maxPlayerLevel = 70;
|
|
}
|
|
}
|
|
}
|
|
|
|
void OnPlayerMapChanged(Player* player) override
|
|
{
|
|
sIndividualProgression->CheckAdjustments(player);
|
|
}
|
|
|
|
void OnPlayerLevelChanged(Player* player, uint8 /*oldLevel*/) override
|
|
{
|
|
sIndividualProgression->CheckAdjustments(player);
|
|
}
|
|
|
|
void OnPlayerEquip(Player* player, Item* /*it*/, uint8 /*bag*/, uint8 /*slot*/, bool /*update*/) override
|
|
{
|
|
sIndividualProgression->CheckAdjustments(player);
|
|
}
|
|
|
|
void OnPlayerResurrect(Player* player, float /*restore_percent*/, bool /*applySickness*/) override
|
|
{
|
|
sIndividualProgression->CheckAdjustments(player);
|
|
}
|
|
|
|
bool OnPlayerShouldBeRewardedWithMoneyInsteadOfExp(Player* player) override
|
|
{
|
|
if (!sIndividualProgression->questMoneyAtLevelCap)
|
|
{
|
|
return false;
|
|
}
|
|
// Player is still in Vanilla content - give money at 60 level cap
|
|
return ((!sIndividualProgression->hasPassedProgression(player, PROGRESSION_NAXX40) && player->GetLevel() == 60) ||
|
|
// Player is in TBC content - give money at 70 level cap
|
|
(!sIndividualProgression->hasPassedProgression(player, PROGRESSION_TBC_TIER_5) && player->GetLevel() == 70));
|
|
}
|
|
|
|
void OnPlayerAfterUpdateMaxHealth(Player* player, float& value) override
|
|
{
|
|
// TODO: This should be adjust to use an aura like damage adjustment. This is more robust to update when changing equipment, etc.
|
|
if (!sIndividualProgression->enabled)
|
|
{
|
|
return;
|
|
}
|
|
float gearAdjustment = 0.0;
|
|
for (uint8 i = EQUIPMENT_SLOT_START; i < EQUIPMENT_SLOT_END; ++i)
|
|
{
|
|
if (Item* item = player->GetItemByPos(INVENTORY_SLOT_BAG_0, i))
|
|
sIndividualProgression->ComputeGearTuning(player, gearAdjustment, item->GetTemplate());
|
|
}
|
|
// Player is still in Vanilla content - give Vanilla health adjustment
|
|
if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_NAXX40) || (isExcludedFromProgression(player) && (player->GetLevel() < 61)))
|
|
{
|
|
float adjustmentAmount = 1.0f - sIndividualProgression->vanillaHealthAdjustment;
|
|
float applyPercent = ((player->GetLevel() - 10.0f) / 50.0f);
|
|
float computedAdjustment = player->GetLevel() > 10 ? 1.0f - applyPercent * adjustmentAmount : 1.0f;
|
|
value *= computedAdjustment;
|
|
}
|
|
// Player is in TBC content - give TBC health adjustment
|
|
else if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_TBC_TIER_5) || (isExcludedFromProgression(player) && (player->GetLevel() < 71)))
|
|
{
|
|
value *= (sIndividualProgression->tbcHealthAdjustment - gearAdjustment);
|
|
}
|
|
// Player is in WotLK content - only need to check gear adjustment
|
|
else
|
|
{
|
|
value *= 1 - gearAdjustment;
|
|
}
|
|
}
|
|
|
|
void OnPlayerQuestComputeXP(Player* player, Quest const* quest, uint32& xpValue) override
|
|
{
|
|
if (!sIndividualProgression->enabled || !sIndividualProgression->questXpFix || isExcludedFromProgression(player))
|
|
{
|
|
return;
|
|
}
|
|
if (sIndividualProgression->questXpMap.count(quest->GetQuestId()))
|
|
{
|
|
uint32 vanillaXpValue = sIndividualProgression->questXpMap[quest->GetQuestId()];
|
|
if (player)
|
|
{
|
|
uint32 originalXpValue = quest->XPValue(quest->GetQuestLevel() == -1 ? player->GetLevel() : quest->GetQuestLevel());
|
|
xpValue *= vanillaXpValue * 1.0 / originalXpValue;
|
|
}
|
|
}
|
|
}
|
|
|
|
void OnPlayerGiveXP(Player* player, uint32& amount, Unit* /*victim*/, uint8 xpSource) override
|
|
{
|
|
if (!sIndividualProgression->enabled || isExcludedFromProgression(player))
|
|
{
|
|
return;
|
|
}
|
|
// Player is still in Vanilla content - do not give XP past level 60
|
|
if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_NAXX40) && player->GetLevel() >= 60)
|
|
{
|
|
// Still award XP to pets - they won't be able to pass the player's level
|
|
Pet* pet = player->GetPet();
|
|
if (pet && xpSource == XPSOURCE_KILL)
|
|
pet->GivePetXP(player->GetGroup() ? amount / 2 : amount);
|
|
amount = 0;
|
|
}
|
|
// Player is in TBC content - do not give XP past level 70
|
|
else if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_TBC_TIER_5) && player->GetLevel() >= 70)
|
|
{
|
|
// Still award XP to pets - they won't be able to pass the player's level
|
|
Pet* pet = player->GetPet();
|
|
if (pet && xpSource == XPSOURCE_KILL)
|
|
pet->GivePetXP(player->GetGroup() ? amount / 2 : amount);
|
|
amount = 0;
|
|
}
|
|
}
|
|
|
|
bool isExcludedFromProgression(Player* player)
|
|
{
|
|
if(!sIndividualProgression->excludeAccounts) {
|
|
return false;
|
|
}
|
|
std::string accountName;
|
|
bool accountNameFound = AccountMgr::GetName(player->GetSession()->GetAccountId(), accountName);
|
|
std::regex excludedAccountsRegex (sIndividualProgression->excludedAccountsRegex);
|
|
return (accountNameFound && std::regex_match(accountName, excludedAccountsRegex));
|
|
}
|
|
|
|
bool OnPlayerBeforeTeleport(Player* player, uint32 mapid, float x, float y, float z, float /*orientation*/, uint32 /*options*/, Unit* /*target*/) override
|
|
{
|
|
if (!sIndividualProgression->enabled || player->IsGameMaster() || isExcludedFromProgression(player))
|
|
{
|
|
return true;
|
|
}
|
|
if (mapid == MAP_BLACKWING_LAIR && !sIndividualProgression->hasPassedProgression(player, PROGRESSION_MOLTEN_CORE))
|
|
{
|
|
return false;
|
|
}
|
|
if (mapid == MAP_ZUL_GURUB && !sIndividualProgression->hasPassedProgression(player, PROGRESSION_BLACKWING_LAIR))
|
|
{
|
|
return false;
|
|
}
|
|
if (mapid == MAP_AHN_QIRAJ_TEMPLE && !sIndividualProgression->hasPassedProgression(player, PROGRESSION_PRE_AQ))
|
|
{
|
|
return false;
|
|
}
|
|
if (mapid == MAP_RUINS_OF_AHN_QIRAJ && !sIndividualProgression->hasPassedProgression(player, PROGRESSION_PRE_AQ))
|
|
{
|
|
return false;
|
|
}
|
|
if (mapid == MAP_OUTLAND)
|
|
{
|
|
if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_NAXX40))
|
|
{
|
|
// The player may be in the Azuremyst area which is on the outlands map - check the area ID
|
|
return IsTBCRaceStartingZone(mapid, x, y, z);
|
|
}
|
|
Map const *map = sMapMgr->FindMap(mapid, 0);
|
|
uint32 zoneId = map->GetZoneId(0, x, y, z);
|
|
if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_TBC_TIER_4) && zoneId == ZONE_QUELDANAS)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
if (mapid == MAP_ZUL_AMAN && !sIndividualProgression->hasPassedProgression(player, PROGRESSION_TBC_TIER_3))
|
|
{
|
|
return false;
|
|
}
|
|
if (mapid == MAP_NORTHREND && !sIndividualProgression->hasPassedProgression(player, PROGRESSION_TBC_TIER_5))
|
|
{
|
|
return false;
|
|
}
|
|
if (mapid == MAP_ULDUAR && !sIndividualProgression->hasPassedProgression(player, PROGRESSION_WOTLK_TIER_1))
|
|
{
|
|
return false;
|
|
}
|
|
if ((mapid == MAP_TRIAL_OF_THE_CHAMPION || mapid == MAP_TRIAL_OF_THE_CRUSADER) && !sIndividualProgression->hasPassedProgression(player, PROGRESSION_WOTLK_TIER_2)){
|
|
return false;
|
|
}
|
|
// This will also restrict other Frozen Halls dungeons, because Forge of Souls must be completed first to access them
|
|
if ((mapid == MAP_ICECROWN_CITADEL || mapid == MAP_THE_FORGE_OF_SOULS) && !sIndividualProgression->hasPassedProgression(player, PROGRESSION_WOTLK_TIER_3))
|
|
{
|
|
return false;
|
|
}
|
|
if (mapid == MAP_THE_RUBY_SANCTUM && !sIndividualProgression->hasPassedProgression(player, PROGRESSION_WOTLK_TIER_4))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
InstanceTemplate const* instanceTemplate = sObjectMgr->GetInstanceTemplate(mapid);
|
|
if (instanceTemplate)
|
|
{
|
|
if (instanceTemplate->Parent == MAP_OUTLAND && !sIndividualProgression->hasPassedProgression(player, PROGRESSION_NAXX40))
|
|
{
|
|
return false;
|
|
}
|
|
if (instanceTemplate->Parent == MAP_NORTHREND && mapid != MAP_NAXXRAMAS && !sIndividualProgression->hasPassedProgression(player, PROGRESSION_TBC_TIER_5))
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void OnPlayerCompleteQuest(Player* player, Quest const* quest) override
|
|
{
|
|
if (!sIndividualProgression->enabled || isExcludedFromProgression(player))
|
|
{
|
|
return;
|
|
}
|
|
switch (quest->GetQuestId())
|
|
{
|
|
case MIGHT_OF_KALIMDOR:
|
|
if (!sIndividualProgression->disableDefaultProgression)
|
|
{
|
|
sIndividualProgression->UpdateProgressionState(player, PROGRESSION_PRE_AQ);
|
|
}
|
|
break;
|
|
case BANG_A_GONG:
|
|
if (!sIndividualProgression->disableDefaultProgression)
|
|
{
|
|
sIndividualProgression->UpdateProgressionState(player, PROGRESSION_PRE_AQ);
|
|
}
|
|
break;
|
|
case QUEST_MORROWGRAIN:
|
|
case QUEST_TROLL_NECKLACE:
|
|
case QUEST_DEADWOOD:
|
|
case QUEST_WINTERFALL:
|
|
if (sIndividualProgression->repeatableVanillaQuestsXp)
|
|
{
|
|
// Reset the quest status so the player can take it and receive rewards again
|
|
player->RemoveRewardedQuest(quest->GetQuestId());
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
bool OnPlayerCanGroupInvite(Player* player, std::string& membername) override
|
|
{
|
|
if (!sIndividualProgression->enabled || !sIndividualProgression->enforceGroupRules || isExcludedFromProgression(player))
|
|
{
|
|
return true;
|
|
}
|
|
Player* otherPlayer = ObjectAccessor::FindPlayerByName(membername, false);
|
|
uint8 currentState = player->GetPlayerSetting("mod-individual-progression", SETTING_PROGRESSION_STATE).value;
|
|
uint8 otherPlayerState = otherPlayer->GetPlayerSetting("mod-individual-progression", SETTING_PROGRESSION_STATE).value;
|
|
return (currentState == otherPlayerState);
|
|
}
|
|
|
|
bool OnPlayerCanGroupAccept(Player* player, Group* group) override
|
|
{
|
|
if (!sIndividualProgression->enabled || !sIndividualProgression->enforceGroupRules || isExcludedFromProgression(player))
|
|
{
|
|
return true;
|
|
}
|
|
Player* groupLeader = ObjectAccessor::FindPlayerByLowGUID(group->GetLeaderGUID().GetCounter());
|
|
uint8 currentState = player->GetPlayerSetting("mod-individual-progression", SETTING_PROGRESSION_STATE).value;
|
|
uint8 otherPlayerState = groupLeader->GetPlayerSetting("mod-individual-progression", SETTING_PROGRESSION_STATE).value;
|
|
return (currentState == otherPlayerState);
|
|
}
|
|
|
|
|
|
|
|
void OnPlayerCreatureKill(Player* killer, Creature* killed) override
|
|
{
|
|
sIndividualProgression->checkKillProgression(killer, killed);
|
|
Group* group = killer->GetGroup();
|
|
if (!group)
|
|
{
|
|
return;
|
|
}
|
|
for (GroupReference* itr = group->GetFirstMember(); itr != nullptr; itr = itr->next())
|
|
{
|
|
Player* member = itr->GetSource();
|
|
if (!member)
|
|
continue;
|
|
|
|
if (killer->IsAtLootRewardDistance(member))
|
|
sIndividualProgression->checkKillProgression(member, killed);
|
|
}
|
|
}
|
|
|
|
bool OnPlayerUpdateFishingSkill(Player* player, int32 /*skill*/, int32 /*zone_skill*/, int32 chance, int32 roll) override
|
|
{
|
|
if (!sIndividualProgression->enabled || !sIndividualProgression->fishingFix || isExcludedFromProgression(player))
|
|
return true;
|
|
if (chance < roll)
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
void OnPlayerUpdateArea(Player* player, uint32 /*oldArea*/, uint32 newArea) override
|
|
{
|
|
switch (newArea) {
|
|
case AREA_DARKSHORE:
|
|
if ((sIndividualProgression->hasPassedProgression(player, PROGRESSION_PRE_AQ)) && (sIndividualProgression->isBeforeProgression(player, PROGRESSION_AQ)))
|
|
{
|
|
player->RemoveAura(IPP_PHASE);
|
|
player->CastSpell(player, IPP_PHASE, false);
|
|
}
|
|
break;
|
|
case AREA_GROVE_OF_THE_ANCIENTS:
|
|
if ((sIndividualProgression->hasPassedProgression(player, PROGRESSION_PRE_AQ)) && (sIndividualProgression->isBeforeProgression(player, PROGRESSION_AQ)))
|
|
{
|
|
player->RemoveAura(IPP_PHASE);
|
|
player->CastSpell(player, IPP_PHASE, false);
|
|
}
|
|
break;
|
|
case AREA_WILDBEND_RIVER:
|
|
if ((sIndividualProgression->hasPassedProgression(player, PROGRESSION_PRE_AQ)) && (sIndividualProgression->isBeforeProgression(player, PROGRESSION_AQ)))
|
|
{
|
|
player->RemoveAura(IPP_PHASE);
|
|
player->CastSpell(player, IPP_PHASE, false);
|
|
}
|
|
break;
|
|
case AREA_TWILIGHT_VALE:
|
|
if ((sIndividualProgression->hasPassedProgression(player, PROGRESSION_PRE_AQ)) && (sIndividualProgression->isBeforeProgression(player, PROGRESSION_AQ)))
|
|
{
|
|
player->RemoveAura(IPP_PHASE);
|
|
player->CastSpell(player, IPP_PHASE, false);
|
|
}
|
|
break;
|
|
case AREA_SILITHUS:
|
|
if ((sIndividualProgression->hasPassedProgression(player, PROGRESSION_PRE_AQ)) && (sIndividualProgression->isBeforeProgression(player, PROGRESSION_AQ)))
|
|
{
|
|
player->RemoveAura(IPP_PHASE);
|
|
player->CastSpell(player, IPP_PHASE, false);
|
|
}
|
|
break;
|
|
case AREA_HIVE_ASHI:
|
|
if ((sIndividualProgression->hasPassedProgression(player, PROGRESSION_PRE_AQ)) && (sIndividualProgression->isBeforeProgression(player, PROGRESSION_AQ)))
|
|
{
|
|
player->RemoveAura(IPP_PHASE);
|
|
player->CastSpell(player, IPP_PHASE, false);
|
|
}
|
|
break;
|
|
case AREA_HIVE_ZORA:
|
|
if ((sIndividualProgression->hasPassedProgression(player, PROGRESSION_PRE_AQ)) && (sIndividualProgression->isBeforeProgression(player, PROGRESSION_AQ)))
|
|
{
|
|
player->RemoveAura(IPP_PHASE);
|
|
player->CastSpell(player, IPP_PHASE, false);
|
|
}
|
|
break;
|
|
case AREA_HIVE_REGAL:
|
|
if ((sIndividualProgression->hasPassedProgression(player, PROGRESSION_PRE_AQ)) && (sIndividualProgression->isBeforeProgression(player, PROGRESSION_AQ)))
|
|
{
|
|
player->RemoveAura(IPP_PHASE);
|
|
player->CastSpell(player, IPP_PHASE, false);
|
|
}
|
|
break;
|
|
case AREA_LIGHTS_HOPE:
|
|
case AREA_ARGENT_TOURNAMENT_GROUNDS:
|
|
case AREA_ARGENT_SUNREAVER_PAVILION:
|
|
case AREA_ARGENT_SILVER_COVENANT_PAVILION:
|
|
case AREA_THE_RING_OF_CHAMPIONS:
|
|
case AREA_THE_ASPIRANTS_RING:
|
|
case AREA_THE_ARGENT_VALIANTS_RING:
|
|
case AREA_THE_ALLIANCE_VALIANTS_RING:
|
|
case AREA_THE_HORDE_VALIANTS_RING:
|
|
case AREA_ARGENT_PAVILION:
|
|
if (sIndividualProgression->hasPassedProgression(player, PROGRESSION_WOTLK_TIER_2))
|
|
{
|
|
player->RemoveAura(IPP_PHASE);
|
|
player->CastSpell(player, IPP_PHASE, false);
|
|
}
|
|
break;
|
|
default:
|
|
player->RemoveAura(IPP_PHASE);
|
|
}
|
|
}
|
|
|
|
void OnPlayerQueueRandomDungeon(Player* player, uint32& rDungeonId) override
|
|
{
|
|
// List of exceptions for seasonal event dungeons
|
|
std::set<uint32> seasonalEventDungeons = { 285, 286, 287, 288 };
|
|
if (seasonalEventDungeons.find(rDungeonId) != seasonalEventDungeons.end())
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Check if RDF is disabled in the context of Individual Progression
|
|
if (sConfigMgr->GetOption<bool>("IndividualProgression.DisableRDF", false))
|
|
{
|
|
ChatHandler(player->GetSession()).SendNotification("The Random Dungeon feature is currently disabled by the Individual Progression module.");
|
|
rDungeonId = 1000; // Set dungeon ID to an invalid value to cancel the queuing
|
|
return;
|
|
}
|
|
|
|
if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_NAXX40) || (isExcludedFromProgression(player) && (player->GetLevel() < 61)))
|
|
{
|
|
rDungeonId = RDF_CLASSIC;
|
|
}
|
|
else if (rDungeonId == RDF_WRATH_OF_THE_LICH_KING && !sIndividualProgression->hasPassedProgression(player, PROGRESSION_TBC_TIER_5) || (isExcludedFromProgression(player) && (player->GetLevel() < 71)))
|
|
{
|
|
rDungeonId = RDF_THE_BURNING_CRUSADE;
|
|
}
|
|
else if (rDungeonId == RDF_WRATH_OF_THE_LICH_KING_HEROIC && !sIndividualProgression->hasPassedProgression(player, PROGRESSION_TBC_TIER_5) || (isExcludedFromProgression(player) && (player->GetLevel() < 71)))
|
|
{
|
|
rDungeonId = RDF_THE_BURNING_CRUSADE_HEROIC;
|
|
}
|
|
}
|
|
|
|
bool OnPlayerCanEquipItem(Player* player, uint8 /*slot*/, uint16& /*dest*/, Item* pItem, bool /*swap*/, bool /*not_loading*/) override
|
|
{
|
|
if (sIndividualProgression->pvpGearRequirements)
|
|
{
|
|
switch (pItem->GetTemplate()->RequiredHonorRank)
|
|
{
|
|
case 5:
|
|
if (!(player->HasTitle(PRIVATE) || player->HasTitle(SCOUT) ||
|
|
player->HasTitle(CORPORAL) || player->HasTitle(GRUNT) ||
|
|
player->HasTitle(SERGEANT) || player->HasTitle(SERGEANT_H) ||
|
|
player->HasTitle(MASTER_SERGEANT) || player->HasTitle(SENIOR_SERGEANT) ||
|
|
player->HasTitle(SERGEANT_MAJOR) || player->HasTitle(FIRST_SERGEANT) ||
|
|
player->HasTitle(KNIGHT) || player->HasTitle(STONE_GUARD) ||
|
|
player->HasTitle(KNIGHT_LIEUTENANT) || player->HasTitle(BLOOD_GUARD) ||
|
|
player->HasTitle(KNIGHT_CAPTAIN) || player->HasTitle(LEGIONNAIRE) ||
|
|
player->HasTitle(KNIGHT_CHAMPION) || player->HasTitle(CENTURION) ||
|
|
player->HasTitle(LIEUTENANT_COMMANDER) || player->HasTitle(CHAMPION) ||
|
|
player->HasTitle(COMMANDER) || player->HasTitle(LIEUTENANT_GENERAL) ||
|
|
player->HasTitle(MARSHAL) || player->HasTitle(GENERAL) ||
|
|
player->HasTitle(FIELD_MARSHAL) || player->HasTitle(WARLORD) ||
|
|
player->HasTitle(GRAND_MARSHAL) || player->HasTitle(HIGH_WARLORD)))
|
|
return false;
|
|
break;
|
|
case 6:
|
|
if (!(player->HasTitle(CORPORAL) || player->HasTitle(GRUNT) ||
|
|
player->HasTitle(SERGEANT) || player->HasTitle(SERGEANT_H) ||
|
|
player->HasTitle(MASTER_SERGEANT) || player->HasTitle(SENIOR_SERGEANT) ||
|
|
player->HasTitle(SERGEANT_MAJOR) || player->HasTitle(FIRST_SERGEANT) ||
|
|
player->HasTitle(KNIGHT) || player->HasTitle(STONE_GUARD) ||
|
|
player->HasTitle(KNIGHT_LIEUTENANT) || player->HasTitle(BLOOD_GUARD) ||
|
|
player->HasTitle(KNIGHT_CAPTAIN) || player->HasTitle(LEGIONNAIRE) ||
|
|
player->HasTitle(KNIGHT_CHAMPION) || player->HasTitle(CENTURION) ||
|
|
player->HasTitle(LIEUTENANT_COMMANDER) || player->HasTitle(CHAMPION) ||
|
|
player->HasTitle(COMMANDER) || player->HasTitle(LIEUTENANT_GENERAL) ||
|
|
player->HasTitle(MARSHAL) || player->HasTitle(GENERAL) ||
|
|
player->HasTitle(FIELD_MARSHAL) || player->HasTitle(WARLORD) ||
|
|
player->HasTitle(GRAND_MARSHAL) || player->HasTitle(HIGH_WARLORD)))
|
|
return false;
|
|
break;
|
|
case 7:
|
|
if (!(player->HasTitle(SERGEANT) || player->HasTitle(SERGEANT_H) ||
|
|
player->HasTitle(MASTER_SERGEANT) || player->HasTitle(SENIOR_SERGEANT) ||
|
|
player->HasTitle(SERGEANT_MAJOR) || player->HasTitle(FIRST_SERGEANT) ||
|
|
player->HasTitle(KNIGHT) || player->HasTitle(STONE_GUARD) ||
|
|
player->HasTitle(KNIGHT_LIEUTENANT) || player->HasTitle(BLOOD_GUARD) ||
|
|
player->HasTitle(KNIGHT_CAPTAIN) || player->HasTitle(LEGIONNAIRE) ||
|
|
player->HasTitle(KNIGHT_CHAMPION) || player->HasTitle(CENTURION) ||
|
|
player->HasTitle(LIEUTENANT_COMMANDER) || player->HasTitle(CHAMPION) ||
|
|
player->HasTitle(COMMANDER) || player->HasTitle(LIEUTENANT_GENERAL) ||
|
|
player->HasTitle(MARSHAL) || player->HasTitle(GENERAL) ||
|
|
player->HasTitle(FIELD_MARSHAL) || player->HasTitle(WARLORD) ||
|
|
player->HasTitle(GRAND_MARSHAL) || player->HasTitle(HIGH_WARLORD)))
|
|
return false;
|
|
break;
|
|
case 8:
|
|
if (!(player->HasTitle(MASTER_SERGEANT) || player->HasTitle(SENIOR_SERGEANT) ||
|
|
player->HasTitle(SERGEANT_MAJOR) || player->HasTitle(FIRST_SERGEANT) ||
|
|
player->HasTitle(KNIGHT) || player->HasTitle(STONE_GUARD) ||
|
|
player->HasTitle(KNIGHT_LIEUTENANT) || player->HasTitle(BLOOD_GUARD) ||
|
|
player->HasTitle(KNIGHT_CAPTAIN) || player->HasTitle(LEGIONNAIRE) ||
|
|
player->HasTitle(KNIGHT_CHAMPION) || player->HasTitle(CENTURION) ||
|
|
player->HasTitle(LIEUTENANT_COMMANDER) || player->HasTitle(CHAMPION) ||
|
|
player->HasTitle(COMMANDER) || player->HasTitle(LIEUTENANT_GENERAL) ||
|
|
player->HasTitle(MARSHAL) || player->HasTitle(GENERAL) ||
|
|
player->HasTitle(FIELD_MARSHAL) || player->HasTitle(WARLORD) ||
|
|
player->HasTitle(GRAND_MARSHAL) || player->HasTitle(HIGH_WARLORD)))
|
|
return false;
|
|
break;
|
|
case 9:
|
|
if (!(player->HasTitle(SERGEANT_MAJOR) || player->HasTitle(FIRST_SERGEANT) ||
|
|
player->HasTitle(KNIGHT) || player->HasTitle(STONE_GUARD) ||
|
|
player->HasTitle(KNIGHT_LIEUTENANT) || player->HasTitle(BLOOD_GUARD) ||
|
|
player->HasTitle(KNIGHT_CAPTAIN) || player->HasTitle(LEGIONNAIRE) ||
|
|
player->HasTitle(KNIGHT_CHAMPION) || player->HasTitle(CENTURION) ||
|
|
player->HasTitle(LIEUTENANT_COMMANDER) || player->HasTitle(CHAMPION) ||
|
|
player->HasTitle(COMMANDER) || player->HasTitle(LIEUTENANT_GENERAL) ||
|
|
player->HasTitle(MARSHAL) || player->HasTitle(GENERAL) ||
|
|
player->HasTitle(FIELD_MARSHAL) || player->HasTitle(WARLORD) ||
|
|
player->HasTitle(GRAND_MARSHAL) || player->HasTitle(HIGH_WARLORD)))
|
|
return false;
|
|
break;
|
|
case 10:
|
|
if (!(player->HasTitle(KNIGHT) || player->HasTitle(STONE_GUARD) ||
|
|
player->HasTitle(KNIGHT_LIEUTENANT) || player->HasTitle(BLOOD_GUARD) ||
|
|
player->HasTitle(KNIGHT_CAPTAIN) || player->HasTitle(LEGIONNAIRE) ||
|
|
player->HasTitle(KNIGHT_CHAMPION) || player->HasTitle(CENTURION) ||
|
|
player->HasTitle(LIEUTENANT_COMMANDER) || player->HasTitle(CHAMPION) ||
|
|
player->HasTitle(COMMANDER) || player->HasTitle(LIEUTENANT_GENERAL) ||
|
|
player->HasTitle(MARSHAL) || player->HasTitle(GENERAL) ||
|
|
player->HasTitle(FIELD_MARSHAL) || player->HasTitle(WARLORD) ||
|
|
player->HasTitle(GRAND_MARSHAL) || player->HasTitle(HIGH_WARLORD)))
|
|
return false;
|
|
break;
|
|
case 11:
|
|
if (!(player->HasTitle(KNIGHT_LIEUTENANT) || player->HasTitle(BLOOD_GUARD) ||
|
|
player->HasTitle(KNIGHT_CAPTAIN) || player->HasTitle(LEGIONNAIRE) ||
|
|
player->HasTitle(KNIGHT_CHAMPION) || player->HasTitle(CENTURION) ||
|
|
player->HasTitle(LIEUTENANT_COMMANDER) || player->HasTitle(CHAMPION) ||
|
|
player->HasTitle(COMMANDER) || player->HasTitle(LIEUTENANT_GENERAL) ||
|
|
player->HasTitle(MARSHAL) || player->HasTitle(GENERAL) ||
|
|
player->HasTitle(FIELD_MARSHAL) || player->HasTitle(WARLORD) ||
|
|
player->HasTitle(GRAND_MARSHAL) || player->HasTitle(HIGH_WARLORD)))
|
|
return false;
|
|
break;
|
|
case 12:
|
|
if (!(player->HasTitle(KNIGHT_CAPTAIN) || player->HasTitle(LEGIONNAIRE) ||
|
|
player->HasTitle(KNIGHT_CHAMPION) || player->HasTitle(CENTURION) ||
|
|
player->HasTitle(LIEUTENANT_COMMANDER) || player->HasTitle(CHAMPION) ||
|
|
player->HasTitle(COMMANDER) || player->HasTitle(LIEUTENANT_GENERAL) ||
|
|
player->HasTitle(MARSHAL) || player->HasTitle(GENERAL) ||
|
|
player->HasTitle(FIELD_MARSHAL) || player->HasTitle(WARLORD) ||
|
|
player->HasTitle(GRAND_MARSHAL) || player->HasTitle(HIGH_WARLORD)))
|
|
return false;
|
|
break;
|
|
case 13:
|
|
if (!(player->HasTitle(KNIGHT_CHAMPION) || player->HasTitle(CENTURION) ||
|
|
player->HasTitle(LIEUTENANT_COMMANDER) || player->HasTitle(CHAMPION) ||
|
|
player->HasTitle(COMMANDER) || player->HasTitle(LIEUTENANT_GENERAL) ||
|
|
player->HasTitle(MARSHAL) || player->HasTitle(GENERAL) ||
|
|
player->HasTitle(FIELD_MARSHAL) || player->HasTitle(WARLORD) ||
|
|
player->HasTitle(GRAND_MARSHAL) || player->HasTitle(HIGH_WARLORD)))
|
|
return false;
|
|
break;
|
|
case 14:
|
|
if (!(player->HasTitle(LIEUTENANT_COMMANDER) || player->HasTitle(CHAMPION) ||
|
|
player->HasTitle(COMMANDER) || player->HasTitle(LIEUTENANT_GENERAL) ||
|
|
player->HasTitle(MARSHAL) || player->HasTitle(GENERAL) ||
|
|
player->HasTitle(FIELD_MARSHAL) || player->HasTitle(WARLORD) ||
|
|
player->HasTitle(GRAND_MARSHAL) || player->HasTitle(HIGH_WARLORD)))
|
|
return false;
|
|
break;
|
|
case 15:
|
|
if (!(player->HasTitle(COMMANDER) || player->HasTitle(LIEUTENANT_GENERAL) ||
|
|
player->HasTitle(MARSHAL) || player->HasTitle(GENERAL) ||
|
|
player->HasTitle(FIELD_MARSHAL) || player->HasTitle(WARLORD) ||
|
|
player->HasTitle(GRAND_MARSHAL) || player->HasTitle(HIGH_WARLORD)))
|
|
return false;
|
|
break;
|
|
case 16:
|
|
if (!(player->HasTitle(MARSHAL) || player->HasTitle(GENERAL) ||
|
|
player->HasTitle(FIELD_MARSHAL) || player->HasTitle(WARLORD) ||
|
|
player->HasTitle(GRAND_MARSHAL) || player->HasTitle(HIGH_WARLORD)))
|
|
return false;
|
|
break;
|
|
case 17:
|
|
if (!(player->HasTitle(FIELD_MARSHAL) || player->HasTitle(WARLORD) ||
|
|
player->HasTitle(GRAND_MARSHAL) || player->HasTitle(HIGH_WARLORD)))
|
|
return false;
|
|
break;
|
|
case 18:
|
|
if (!(player->HasTitle(GRAND_MARSHAL) || player->HasTitle(HIGH_WARLORD)))
|
|
return false;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
};
|
|
|
|
class IndividualPlayerProgression_AccountScript: public AccountScript
|
|
{
|
|
public:
|
|
IndividualPlayerProgression_AccountScript() : AccountScript("IndividualProgression_AccountScript")
|
|
{}
|
|
|
|
bool CanAccountCreateCharacter(uint32 accountId, uint8 charRace, uint8 charClass) override
|
|
{
|
|
if ((!sIndividualProgression->enabled) ||
|
|
(charRace != RACE_DRAENEI && charRace != RACE_BLOODELF && charClass != CLASS_DEATH_KNIGHT) ||
|
|
(!sIndividualProgression->tbcRacesProgressionLevel && !sIndividualProgression->deathKnightProgressionLevel))
|
|
{
|
|
return true;
|
|
}
|
|
uint8 highestProgression = sIndividualProgression->GetAccountProgression(accountId);
|
|
if (charRace == RACE_DRAENEI || charRace == RACE_BLOODELF)
|
|
{
|
|
if (sIndividualProgression->tbcRacesProgressionLevel)
|
|
{
|
|
if (highestProgression < sIndividualProgression->tbcRacesProgressionLevel)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
if (charClass == CLASS_DEATH_KNIGHT && sIndividualProgression->deathKnightProgressionLevel)
|
|
{
|
|
if (highestProgression < sIndividualProgression->deathKnightProgressionLevel)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
};
|
|
|
|
|
|
class IndividualPlayerProgression_PetScript : public PetScript
|
|
{
|
|
private:
|
|
static void CheckAdjustments(Pet* pet)
|
|
{
|
|
if (!sIndividualProgression->enabled)
|
|
{
|
|
return;
|
|
}
|
|
if (!pet || !pet->GetOwner())
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (!sIndividualProgression->hasPassedProgression(pet->GetOwner(), PROGRESSION_NAXX40))
|
|
{
|
|
AdjustVanillaStats(pet);
|
|
}
|
|
else if (!sIndividualProgression->hasPassedProgression(pet->GetOwner(), PROGRESSION_TBC_TIER_5))
|
|
{
|
|
AdjustTBCStats(pet);
|
|
}
|
|
}
|
|
|
|
static void AdjustVanillaStats(Pet* pet)
|
|
{
|
|
float adjustmentValue = -100.0f * (1.0f - sIndividualProgression->vanillaPowerAdjustment);
|
|
float adjustmentApplyPercent = (pet->GetLevel() - 10.0f) / 50.0f;
|
|
float computedAdjustment = pet->GetLevel() > 10 ? (adjustmentValue * adjustmentApplyPercent) : 0;
|
|
float hpAdjustmentValue = -100.0f * (1.0f - sIndividualProgression->vanillaHealthAdjustment);
|
|
float hpAdjustment = pet->GetLevel() > 10 ? (hpAdjustmentValue * adjustmentApplyPercent) : 0;
|
|
AdjustStats(pet, computedAdjustment, hpAdjustment);
|
|
}
|
|
|
|
static void AdjustTBCStats(Pet* pet)
|
|
{
|
|
float adjustmentValue = -100.0f * (1.0f - sIndividualProgression->tbcPowerAdjustment);
|
|
float adjustmentApplyPercent = 1;
|
|
float computedAdjustment = pet->GetLevel() > 10 ? (adjustmentValue * adjustmentApplyPercent) : 0;
|
|
float hpAdjustmentValue = -100.0f * (1.0f - sIndividualProgression->tbcHealthAdjustment);
|
|
float hpAdjustment = pet->GetLevel() > 10 ? (hpAdjustmentValue * adjustmentApplyPercent) : 0;
|
|
AdjustStats(pet, computedAdjustment, hpAdjustment);
|
|
}
|
|
|
|
static void AdjustStats(Pet* pet, float computedAdjustment, float hpAdjustment)
|
|
{
|
|
// int32 bp0 = 0; // This would be the damage taken adjustment value, but we are already adjusting health
|
|
auto bp1 = static_cast<int32>(computedAdjustment);
|
|
auto bp2 = static_cast<int32>(hpAdjustment);
|
|
|
|
pet->RemoveAura(ABSORB_SPELL);
|
|
pet->CastCustomSpell(pet, ABSORB_SPELL, &bp1, nullptr, nullptr, false);
|
|
|
|
pet->RemoveAura(HP_AURA_SPELL);
|
|
pet->CastCustomSpell(pet, HP_AURA_SPELL, &bp2, nullptr, nullptr, false);
|
|
}
|
|
|
|
public:
|
|
IndividualPlayerProgression_PetScript() : PetScript("IndividualProgression_PetScript") { }
|
|
|
|
void OnPetAddToWorld(Pet* pet) override
|
|
{
|
|
CheckAdjustments(pet);
|
|
}
|
|
};
|
|
|
|
class IndividualPlayerProgression_UnitScript : public UnitScript
|
|
{
|
|
private:
|
|
static float computeTotalGearTuning(Player* player)
|
|
{
|
|
float gearAdjustment = 0.0;
|
|
for (uint8 i = EQUIPMENT_SLOT_START; i < EQUIPMENT_SLOT_END; ++i)
|
|
{
|
|
if (Item* item = player->GetItemByPos(INVENTORY_SLOT_BAG_0, i))
|
|
sIndividualProgression->ComputeGearTuning(player, gearAdjustment, item->GetTemplate());
|
|
}
|
|
return gearAdjustment;
|
|
}
|
|
|
|
public:
|
|
IndividualPlayerProgression_UnitScript() : UnitScript("IndividualPlayerProgression_UnitScript") { }
|
|
|
|
bool isExcludedFromProgression(Player* player)
|
|
{
|
|
if(!sIndividualProgression->excludeAccounts) {
|
|
return false;
|
|
}
|
|
std::string accountName;
|
|
bool accountNameFound = AccountMgr::GetName(player->GetSession()->GetAccountId(), accountName);
|
|
std::regex excludedAccountsRegex (sIndividualProgression->excludedAccountsRegex);
|
|
return (accountNameFound && std::regex_match(accountName, excludedAccountsRegex));
|
|
}
|
|
|
|
void ModifyHealReceived(Unit* /*target*/, Unit *healer, uint32 &heal, SpellInfo const *spellInfo) override
|
|
{
|
|
// Skip potions, bandages, percentage based heals like Rune Tap, etc.
|
|
if (!sIndividualProgression->enabled || spellInfo->HasAttribute(SPELL_ATTR0_NO_IMMUNITIES) || spellInfo->Mechanic == MECHANIC_BANDAGE)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Skip percentage based heals or spells already nerfed by damage reduction
|
|
for (uint8 i = 0; i < 3; i++)
|
|
{
|
|
if (spellInfo->Effects[i].Effect == SPELL_EFFECT_HEAL_MAX_HEALTH)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
if (spellInfo->Id == SPELL_RUNE_TAP || spellInfo->Id == SPELL_LIFE_STEAL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (!healer)
|
|
return;
|
|
|
|
bool isPet = healer->GetOwner() && healer->GetOwner()->GetTypeId() == TYPEID_PLAYER;
|
|
if (!isPet && healer->GetTypeId() != TYPEID_PLAYER)
|
|
{
|
|
return;
|
|
}
|
|
Player* player = isPet ? healer->GetOwner()->ToPlayer() : healer->ToPlayer();
|
|
float gearAdjustment = computeTotalGearTuning(player);
|
|
if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_NAXX40) || (isExcludedFromProgression(player) && (player->GetLevel() < 61)))
|
|
{
|
|
heal *= (sIndividualProgression->ComputeVanillaAdjustment(player->GetLevel(), sIndividualProgression->vanillaHealingAdjustment) - gearAdjustment);
|
|
}
|
|
else if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_TBC_TIER_5) || (isExcludedFromProgression(player) && (player->GetLevel() < 71)))
|
|
{
|
|
heal *= (sIndividualProgression->tbcHealingAdjustment - gearAdjustment);
|
|
}
|
|
else
|
|
{
|
|
heal *= 1.0f - gearAdjustment;
|
|
}
|
|
}
|
|
|
|
void ModifySpellDamageTaken(Unit* /*target*/, Unit* attacker, int32& damage, SpellInfo const* /*spellInfo*/) override
|
|
{
|
|
if (!sIndividualProgression->enabled || !attacker)
|
|
return;
|
|
bool isPet = attacker->GetOwner() && attacker->GetOwner()->GetTypeId() == TYPEID_PLAYER;
|
|
if (!isPet && attacker->GetTypeId() != TYPEID_PLAYER)
|
|
{
|
|
return;
|
|
}
|
|
Player* player = isPet ? attacker->GetOwner()->ToPlayer() : attacker->ToPlayer();
|
|
float gearAdjustment = computeTotalGearTuning(player);
|
|
if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_NAXX40) || (isExcludedFromProgression(player) && (player->GetLevel() < 61)))
|
|
{
|
|
damage *= (sIndividualProgression->ComputeVanillaAdjustment(player->GetLevel(), sIndividualProgression->vanillaPowerAdjustment) - gearAdjustment);
|
|
}
|
|
else if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_TBC_TIER_5) || (isExcludedFromProgression(player) && (player->GetLevel() < 71)))
|
|
{
|
|
damage *= (sIndividualProgression->tbcPowerAdjustment - gearAdjustment);
|
|
}
|
|
else
|
|
{
|
|
damage *= 1.0f - gearAdjustment;
|
|
}
|
|
}
|
|
|
|
void ModifyMeleeDamage(Unit* /*target*/, Unit* attacker, uint32& damage) override
|
|
{
|
|
if (!sIndividualProgression->enabled || !attacker)
|
|
return;
|
|
|
|
bool isPet = attacker->GetOwner() && attacker->GetOwner()->GetTypeId() == TYPEID_PLAYER;
|
|
if (!isPet && attacker->GetTypeId() != TYPEID_PLAYER)
|
|
{
|
|
return;
|
|
}
|
|
Player* player = isPet ? attacker->GetOwner()->ToPlayer() : attacker->ToPlayer();
|
|
float gearAdjustment = computeTotalGearTuning(player);
|
|
if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_NAXX40) || (isExcludedFromProgression(player) && (player->GetLevel() < 61)))
|
|
{
|
|
damage *= (sIndividualProgression->ComputeVanillaAdjustment(player->GetLevel(), sIndividualProgression->vanillaPowerAdjustment) - gearAdjustment);
|
|
}
|
|
else if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_TBC_TIER_5) || (isExcludedFromProgression(player) && (player->GetLevel() < 71)))
|
|
{
|
|
damage *= (sIndividualProgression->tbcPowerAdjustment - gearAdjustment);
|
|
}
|
|
else
|
|
{
|
|
damage *= 1.0f - gearAdjustment;
|
|
}
|
|
}
|
|
|
|
void ModifyPeriodicDamageAurasTick(Unit* /*target*/, Unit* attacker, uint32& damage, SpellInfo const* spellInfo) override
|
|
{
|
|
if (!sIndividualProgression->enabled || !attacker)
|
|
return;
|
|
|
|
// Do not apply reductions to healing auras - these are already modified in the ModifyHeal hook
|
|
for (uint8 j = 0; j < MAX_SPELL_EFFECTS; ++j)
|
|
{
|
|
if (spellInfo->Effects[j].Effect == SPELL_EFFECT_APPLY_AURA && spellInfo->Effects[j].ApplyAuraName == SPELL_AURA_PERIODIC_HEAL)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
|
|
bool isPet = attacker->GetOwner() && attacker->GetOwner()->GetTypeId() == TYPEID_PLAYER;
|
|
if (!isPet && attacker->GetTypeId() != TYPEID_PLAYER)
|
|
{
|
|
return;
|
|
}
|
|
Player* player = isPet ? attacker->GetOwner()->ToPlayer() : attacker->ToPlayer();
|
|
float gearAdjustment = computeTotalGearTuning(player);
|
|
if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_NAXX40) || (isExcludedFromProgression(player) && (player->GetLevel() < 61)))
|
|
{
|
|
damage *= (sIndividualProgression->ComputeVanillaAdjustment(player->GetLevel(), sIndividualProgression->vanillaPowerAdjustment) - gearAdjustment);
|
|
}
|
|
else if (!sIndividualProgression->hasPassedProgression(player, PROGRESSION_TBC_TIER_5) || (isExcludedFromProgression(player) && (player->GetLevel() < 71)))
|
|
{
|
|
damage *= (sIndividualProgression->tbcPowerAdjustment - gearAdjustment);
|
|
}
|
|
else
|
|
{
|
|
damage *= 1.0f - gearAdjustment;
|
|
}
|
|
}
|
|
|
|
};
|
|
|
|
void AddSC_mod_individual_progression_player()
|
|
{
|
|
new IndividualPlayerProgression();
|
|
new IndividualPlayerProgression_PetScript();
|
|
new IndividualPlayerProgression_AccountScript();
|
|
new IndividualPlayerProgression_UnitScript();
|
|
}
|