/*
* Copyright (C) 2010 - 2024 Eluna Lua Engine
* This program is free software licensed under GPL version 3
* Please see the included DOCS/LICENSE.md for more information
*/
#ifndef _LUA_ENGINE_H
#define _LUA_ENGINE_H
#include "Common.h"
#include "ElunaUtility.h"
#include "Hooks.h"
#if !defined ELUNA_CMANGOS
#include "DBCEnums.h"
#include "Group.h"
#include "Item.h"
#include "Map.h"
#include "SharedDefines.h"
#include "Weather.h"
#include "World.h"
#if defined ELUNA_VMANGOS
#include "Player.h"
#endif
#else
#include "Entities/Item.h"
#include "Globals/SharedDefines.h"
#include "Groups/Group.h"
#include "Maps/Map.h"
#include "Server/DBCEnums.h"
#include "Weather/Weather.h"
#include "World/World.h"
#include "Entities/Player.h"
#endif
#include
#include
extern "C"
{
#include "lua.h"
#include "lualib.h"
#include "lauxlib.h"
};
class AuctionHouseObject;
class Channel;
class Corpse;
class Creature;
class CreatureAI;
class ElunaInstanceAI;
class GameObject;
class Group;
class Guild;
class Item;
class Pet;
class Player;
class Quest;
class Spell;
class SpellCastTargets;
class Unit;
class Weather;
class WorldPacket;
struct AreaTriggerEntry;
struct AuctionEntry;
#if defined ELUNA_TRINITY
class Battleground;
class GameObjectAI;
class InstanceScript;
class TempSummon;
class Vehicle;
struct ItemTemplate;
typedef Battleground BattleGround;
typedef BattlegroundTypeId BattleGroundTypeId;
typedef InstanceScript InstanceData;
#else
class InstanceData;
struct ItemPrototype;
struct SpellEntry;
typedef ItemPrototype ItemTemplate;
typedef SpellEffectIndex SpellEffIndex;
typedef SpellEntry SpellInfo;
#if defined ELUNA_CMANGOS
class TemporarySpawn;
typedef TemporarySpawn TempSummon;
#endif
#if defined ELUNA_VMANGOS || ELUNA_MANGOS
class TemporarySummon;
typedef TemporarySummon TempSummon;
#endif
#if ELUNA_EXPANSION == EXP_CLASSIC
typedef int Difficulty;
#endif
#if ELUNA_EXPANSION >= EXP_WOTLK
class VehicleInfo;
typedef VehicleInfo Vehicle;
#endif
#endif
struct lua_State;
class EventMgr;
class ElunaObject;
class BaseBindingMap;
template class ElunaTemplate;
template class BindingMap;
template struct EventKey;
template struct EntryKey;
template struct UniqueObjectKey;
struct LuaScript
{
std::string fileext;
std::string filename;
std::string filepath;
std::string modulepath;
BytecodeBuffer bytecode;
int32 mapId;
};
enum MethodRegisterState
{
METHOD_REG_NONE = 0,
METHOD_REG_MAP,
METHOD_REG_WORLD,
METHOD_REG_ALL
};
enum MethodFlags : uint32
{
METHOD_FLAG_NONE = 0x0,
METHOD_FLAG_UNSAFE = 0x1,
METHOD_FLAG_DEPRECATED = 0x2
};
#define ELUNA_STATE_PTR "Eluna State Ptr"
#if defined ELUNA_TRINITY
#define ELUNA_GAME_API TC_GAME_API
#define TRACKABLE_PTR_NAMESPACE ::Trinity::
#else
#define ELUNA_GAME_API
#if defined ELUNA_CMANGOS
#define TRACKABLE_PTR_NAMESPACE ::MaNGOS::
#endif
#endif
class ELUNA_GAME_API Eluna
{
public:
void ReloadEluna() { reload = true; }
bool ExecuteCall(int params, int res);
private:
// Indicates that the lua state should be reloaded
bool reload = false;
#if !defined TRACKABLE_PTR_NAMESPACE
// A counter for lua event stacks that occur (see event_level).
// This is used to determine whether an object belongs to the current call stack or not.
// 0 is reserved for always belonging to the call stack
// 1 is reserved for a non valid callstackid
uint64 callstackid = 2;
#endif
// A counter for the amount of nested events. When the event_level
// reaches 0 we are about to return back to C++. At this point the
// objects used during the event stack are invalidated.
uint32 event_level;
// When a hook pushes arguments to be passed to event handlers,
// this is used to keep track of how many arguments were pushed.
uint8 push_counter;
Map* const boundMap;
// Map from instance ID -> Lua table ref
std::unordered_map instanceDataRefs;
// Map from map ID -> Lua table ref
std::unordered_map continentDataRefs;
std::array, Hooks::REGTYPE_COUNT> bindingMaps;
template
void CreateBinding(Hooks::RegisterTypes type)
{
auto index = static_cast>(type);
bindingMaps[index] = std::make_unique>(L);
}
void OpenLua();
void CloseLua();
void DestroyBindStores();
void CreateBindStores();
void RegisterHookGlobals(lua_State* _L);
#if !defined TRACKABLE_PTR_NAMESPACE
void InvalidateObjects();
#endif
// Use ReloadEluna() to make eluna reload
// This is called on world update to reload eluna
void _ReloadEluna();
// Some helpers for hooks to call event handlers.
// The bodies of the templates are in HookHelpers.h, so if you want to use them you need to #include "HookHelpers.h".
template int SetupStack(BindingMap* bindings1, BindingMap* bindings2, const K1& key1, const K2& key2, int number_of_arguments);
int CallOneFunction(int number_of_functions, int number_of_arguments, int number_of_results);
void CleanUpStack(int number_of_arguments);
template void ReplaceArgument(T value, uint8 index);
template void CallAllFunctions(BindingMap* bindings1, BindingMap* bindings2, const K1& key1, const K2& key2);
template bool CallAllFunctionsBool(BindingMap* bindings1, BindingMap* bindings2, const K1& key1, const K2& key2, bool default_value = false);
// Same as above but for only one binding instead of two.
// `key` is passed twice because there's no NULL for references, but it's not actually used if `bindings2` is NULL.
template int SetupStack(BindingMap* bindings, const K& key, int number_of_arguments)
{
return SetupStack(bindings, NULL, key, key, number_of_arguments);
}
template void CallAllFunctions(BindingMap* bindings, const K& key)
{
CallAllFunctions(bindings, NULL, key, key);
}
template bool CallAllFunctionsBool(BindingMap* bindings, const K& key, bool default_value = false)
{
return CallAllFunctionsBool(bindings, NULL, key, key, default_value);
}
// Non-static pushes, to be used in hooks.
// They up the pushed value counter for hook helper functions.
void HookPush() { Push(); ++push_counter; }
void HookPush(const long long value) { Push(value); ++push_counter; }
void HookPush(const unsigned long long value) { Push(value); ++push_counter; }
void HookPush(const long value) { Push(value); ++push_counter; }
void HookPush(const unsigned long value) { Push(value); ++push_counter; }
void HookPush(const int value) { Push(value); ++push_counter; }
void HookPush(const unsigned int value) { Push(value); ++push_counter; }
void HookPush(const bool value) { Push(value); ++push_counter; }
void HookPush(const float value) { Push(value); ++push_counter; }
void HookPush(const double value) { Push(value); ++push_counter; }
void HookPush(const std::string& value) { Push(value); ++push_counter; }
void HookPush(const char* value) { Push(value); ++push_counter; }
void HookPush(ObjectGuid const value) { Push(value); ++push_counter; }
template
void HookPush(T const* ptr) { Push(ptr); ++push_counter; }
#if defined ELUNA_TRINITY
QueryCallbackProcessor queryProcessor;
#endif
public:
lua_State* L;
std::unique_ptr eventMgr;
#if defined ELUNA_TRINITY
QueryCallbackProcessor& GetQueryProcessor() { return queryProcessor; }
#endif
static int StackTrace(lua_State* _L);
static void Report(lua_State* _L);
// Never returns nullptr
static Eluna* GetEluna(lua_State* L)
{
lua_pushstring(L, ELUNA_STATE_PTR);
lua_rawget(L, LUA_REGISTRYINDEX);
ASSERT(lua_islightuserdata(L, -1));
Eluna* E = static_cast(lua_touserdata(L, -1));
lua_pop(L, 1);
ASSERT(E);
return E;
}
// can be used by anything, including methods.
void Push(); // nil
void Push(const long long);
void Push(const unsigned long long);
void Push(const long);
void Push(const unsigned long);
void Push(const int);
void Push(const unsigned int);
void Push(const bool);
void Push(const float);
void Push(const double);
void Push(const std::string&);
void Push(const char*);
void Push(Object const* obj);
void Push(WorldObject const* obj);
void Push(Unit const* unit);
void Push(Pet const* pet);
void Push(TempSummon const* summon);
void Push(ObjectGuid const guid);
template
void Push(T const* ptr)
{
ElunaTemplate::Push(this, ptr);
}
/*
* Returns `true` if Eluna has instance data for `map`.
*/
bool HasInstanceData(Map const* map);
/*
* Use the top element of the stack as the instance data table for `map`,
* then pops it off the stack.
*/
void CreateInstanceData(Map const* map);
/*
* Retrieve the instance data for the `Map` scripted by `ai` and push it
* onto the stack.
*
* An `ElunaInstanceAI` is needed because the instance data might
* not exist (i.e. Eluna has been reloaded).
*
* In that case, the AI is "reloaded" (new instance data table is created
* and loaded with the last known save state, and `Load`/`Initialize`
* hooks are called).
*/
void PushInstanceData(ElunaInstanceAI* ai, bool incrementCounter = true);
void RunScripts();
bool HasLuaState() const { return L != NULL; }
#if !defined TRACKABLE_PTR_NAMESPACE
uint64 GetCallstackId() const { return callstackid; }
#endif
int Register(std::underlying_type_t regtype, uint32 entry, ObjectGuid guid, uint32 instanceId, uint32 event_id, int functionRef, uint32 shots);
void UpdateEluna(uint32 diff);
// Checks
template T CHECKVAL(int narg);
template T CHECKVAL(int narg, T def)
{
return lua_isnoneornil(L, narg) ? def : CHECKVAL(narg);
}
template T* CHECKOBJ(int narg, bool error = true)
{
return ElunaTemplate::Check(this, narg, error);
}
ElunaObject* CHECKTYPE(int narg, const char* tname, bool error = true);
CreatureAI* GetAI(Creature* creature);
InstanceData* GetInstanceData(Map* map);
void FreeInstanceId(uint32 instanceId);
Map* GetBoundMap() const { return boundMap; }
int32 GetBoundMapId() const
{
if(const Map * map = GetBoundMap())
return map->GetId();
return -1;
}
uint32 GetBoundInstanceId() const
{
if(const Map * map = GetBoundMap())
return map->GetInstanceId();
return 0;
}
template
BindingMap* GetBinding(std::underlying_type_t type)
{
if (type >= Hooks::REGTYPE_COUNT)
return nullptr;
auto& binding = bindingMaps[type];
if (!binding)
return nullptr;
return dynamic_cast*>(binding.get());
}
template
BindingMap* GetBinding(Hooks::RegisterTypes type)
{
return GetBinding(static_cast>(type));
}
Eluna(Map * map);
~Eluna();
// Prevent copy
Eluna(Eluna const&) = delete;
Eluna& operator=(const Eluna&) = delete;
/* Custom */
void OnTimedEvent(int funcRef, uint32 delay, uint32 calls, WorldObject* obj);
bool OnCommand(Player* player, const char* text);
void OnWorldUpdate(uint32 diff);
void OnLootItem(Player* pPlayer, Item* pItem, uint32 count, ObjectGuid guid);
void OnLootMoney(Player* pPlayer, uint32 amount);
void OnFirstLogin(Player* pPlayer);
void OnEquip(Player* pPlayer, Item* pItem, uint8 bag, uint8 slot);
void OnRepop(Player* pPlayer);
void OnResurrect(Player* pPlayer);
void OnQuestAbandon(Player* pPlayer, uint32 questId);
void OnQuestStatusChanged(Player* pPlayer, uint32 questId, uint8 status);
void OnLearnTalents(Player* pPlayer, uint32 talentId, uint32 talentRank, uint32 spellid);
void OnSkillChange(Player* pPlayer, uint32 skillId, uint32 skillValue);
void OnLearnSpell(Player* pPlayer, uint32 spellid);
InventoryResult OnCanUseItem(const Player* pPlayer, uint32 itemEntry);
void OnLuaStateClose();
void OnLuaStateOpen();
bool OnAddonMessage(Player* sender, uint32 type, std::string& msg, Player* receiver, Guild* guild, Group* group, Channel* channel);
bool OnTradeInit(Player* trader, Player* tradee);
bool OnTradeAccept(Player* trader, Player* tradee);
bool OnSendMail(Player* sender, ObjectGuid recipientGuid);
void OnDiscoverArea(Player* player, uint32 area);
/* Item */
void OnDummyEffect(WorldObject* pCaster, uint32 spellId, SpellEffIndex effIndex, Item* pTarget);
bool OnQuestAccept(Player* pPlayer, Item* pItem, Quest const* pQuest);
bool OnUse(Player* pPlayer, Item* pItem, SpellCastTargets const& targets);
bool OnItemUse(Player* pPlayer, Item* pItem, SpellCastTargets const& targets);
bool OnItemGossip(Player* pPlayer, Item* pItem, SpellCastTargets const& targets);
bool OnExpire(Player* pPlayer, ItemTemplate const* pProto);
bool OnRemove(Player* pPlayer, Item* item);
void OnAdd(Player* pPlayer, Item* item);
void HandleGossipSelectOption(Player* pPlayer, Item* item, uint32 sender, uint32 action, const std::string& code);
void OnItemEquip(Player* pPlayer, Item* pItem, uint8 slot);
void OnItemUnEquip(Player* pPlayer, Item* pItem, uint8 slot);
/* Creature */
void OnDummyEffect(WorldObject* pCaster, uint32 spellId, SpellEffIndex effIndex, Creature* pTarget);
bool OnGossipHello(Player* pPlayer, Creature* pCreature);
bool OnGossipSelect(Player* pPlayer, Creature* pCreature, uint32 sender, uint32 action);
bool OnGossipSelectCode(Player* pPlayer, Creature* pCreature, uint32 sender, uint32 action, const char* code);
bool OnQuestAccept(Player* pPlayer, Creature* pCreature, Quest const* pQuest);
bool OnQuestReward(Player* pPlayer, Creature* pCreature, Quest const* pQuest, uint32 opt);
void GetDialogStatus(const Player* pPlayer, const Creature* pCreature);
bool OnSummoned(Creature* creature, Unit* summoner);
bool UpdateAI(Creature* me, const uint32 diff);
bool EnterCombat(Creature* me, Unit* target);
bool DamageTaken(Creature* me, Unit* attacker, uint32& damage);
bool JustDied(Creature* me, Unit* killer);
bool KilledUnit(Creature* me, Unit* victim);
bool JustSummoned(Creature* me, Creature* summon);
bool SummonedCreatureDespawn(Creature* me, Creature* summon);
bool MovementInform(Creature* me, uint32 type, uint32 id);
bool AttackStart(Creature* me, Unit* target);
bool EnterEvadeMode(Creature* me);
bool JustRespawned(Creature* me);
bool JustReachedHome(Creature* me);
bool ReceiveEmote(Creature* me, Player* player, uint32 emoteId);
bool CorpseRemoved(Creature* me, uint32& respawnDelay);
bool MoveInLineOfSight(Creature* me, Unit* who);
bool SpellHit(Creature* me, WorldObject* caster, SpellInfo const* spell);
bool SpellHitTarget(Creature* me, WorldObject* target, SpellInfo const* spell);
bool SummonedCreatureDies(Creature* me, Creature* summon, Unit* killer);
bool OwnerAttackedBy(Creature* me, Unit* attacker);
bool OwnerAttacked(Creature* me, Unit* target);
void On_Reset(Creature* me);
/* GameObject */
void OnDummyEffect(WorldObject* pCaster, uint32 spellId, SpellEffIndex effIndex, GameObject* pTarget);
bool OnGameObjectUse(Player* pPlayer, GameObject* pGameObject);
bool OnGossipHello(Player* pPlayer, GameObject* pGameObject);
bool OnGossipSelect(Player* pPlayer, GameObject* pGameObject, uint32 sender, uint32 action);
bool OnGossipSelectCode(Player* pPlayer, GameObject* pGameObject, uint32 sender, uint32 action, const char* code);
bool OnQuestAccept(Player* pPlayer, GameObject* pGameObject, Quest const* pQuest);
bool OnQuestReward(Player* pPlayer, GameObject* pGameObject, Quest const* pQuest, uint32 opt);
void GetDialogStatus(const Player* pPlayer, const GameObject* pGameObject);
#if ELUNA_EXPANSION >= EXP_WOTLK
void OnDestroyed(GameObject* pGameObject, WorldObject* attacker);
void OnDamaged(GameObject* pGameObject, WorldObject* attacker);
#endif
void OnLootStateChanged(GameObject* pGameObject, uint32 state);
void OnGameObjectStateChanged(GameObject* pGameObject, uint32 state);
void UpdateAI(GameObject* pGameObject, uint32 diff);
void OnSpawn(GameObject* gameobject);
/* Packet */
bool OnPacketSend(WorldSession* session, const WorldPacket& packet);
void OnPacketSendAny(Player* player, const WorldPacket& packet, bool& result);
void OnPacketSendOne(Player* player, const WorldPacket& packet, bool& result);
bool OnPacketReceive(WorldSession* session, WorldPacket& packet);
void OnPacketReceiveAny(Player* player, WorldPacket& packet, bool& result);
void OnPacketReceiveOne(Player* player, WorldPacket& packet, bool& result);
/* Player */
void OnPlayerEnterCombat(Player* pPlayer, Unit* pEnemy);
void OnPlayerLeaveCombat(Player* pPlayer);
void OnPVPKill(Player* pKiller, Player* pKilled);
void OnCreatureKill(Player* pKiller, Creature* pKilled);
void OnPlayerKilledByCreature(Creature* pKiller, Player* pKilled);
void OnPlayerKilledByEnvironment(Player* pKilled, uint8 damageType);
void OnLevelChanged(Player* pPlayer, uint8 oldLevel);
void OnFreeTalentPointsChanged(Player* pPlayer, uint32 newPoints);
void OnTalentsReset(Player* pPlayer, bool noCost);
#if ELUNA_EXPANSION < EXP_CATA
void OnMoneyChanged(Player* pPlayer, int32& amount);
#else
void OnMoneyChanged(Player* pPlayer, int64& amount);
#endif
void OnGiveXP(Player* pPlayer, uint32& amount, Unit* pVictim);
void OnReputationChange(Player* pPlayer, uint32 factionID, int32& standing, bool incremental);
void OnDuelRequest(Player* pTarget, Player* pChallenger);
void OnDuelStart(Player* pStarter, Player* pChallenger);
void OnDuelEnd(Player* pWinner, Player* pLoser, DuelCompleteType type);
bool OnChat(Player* pPlayer, uint32 type, uint32 lang, std::string& msg);
bool OnChat(Player* pPlayer, uint32 type, uint32 lang, std::string& msg, Group* pGroup);
bool OnChat(Player* pPlayer, uint32 type, uint32 lang, std::string& msg, Guild* pGuild);
bool OnChat(Player* pPlayer, uint32 type, uint32 lang, std::string& msg, Channel* pChannel);
bool OnChat(Player* pPlayer, uint32 type, uint32 lang, std::string& msg, Player* pReceiver);
void OnEmote(Player* pPlayer, uint32 emote);
void OnTextEmote(Player* pPlayer, uint32 textEmote, uint32 emoteNum, ObjectGuid guid);
void OnSpellCast(Player* pPlayer, Spell* pSpell, bool skipCheck);
void OnLogin(Player* pPlayer);
void OnLogout(Player* pPlayer);
void OnCreate(Player* pPlayer);
void OnDelete(uint32 guid);
void OnSave(Player* pPlayer);
void OnBindToInstance(Player* pPlayer, Difficulty difficulty, uint32 mapid, bool permanent);
void OnUpdateZone(Player* pPlayer, uint32 newZone, uint32 newArea);
void OnUpdateArea(Player* pPlayer, uint32 oldArea, uint32 newArea);
void OnMapChanged(Player* pPlayer);
void HandleGossipSelectOption(Player* pPlayer, uint32 menuId, uint32 sender, uint32 action, const std::string& code);
void OnAchievementComplete(Player* pPlayer, uint32 achievementId);
#if ELUNA_EXPANSION >= EXP_WOTLK
/* Vehicle */
void OnInstall(Vehicle* vehicle);
void OnUninstall(Vehicle* vehicle);
void OnInstallAccessory(Vehicle* vehicle, Creature* accessory);
void OnAddPassenger(Vehicle* vehicle, Unit* passenger, int8 seatId);
void OnRemovePassenger(Vehicle* vehicle, Unit* passenger);
#endif
/* AreaTrigger */
bool OnAreaTrigger(Player* pPlayer, AreaTriggerEntry const* pTrigger);
/* Weather */
void OnChange(Weather* weather, uint32 zone, WeatherState state, float grade);
#if ELUNA_EXPANSION < EXP_RETAIL
/* Auction House */
void OnAdd(AuctionHouseObject* ah, AuctionEntry* entry);
void OnRemove(AuctionHouseObject* ah, AuctionEntry* entry);
void OnSuccessful(AuctionHouseObject* ah, AuctionEntry* entry);
void OnExpire(AuctionHouseObject* ah, AuctionEntry* entry);
#endif
/* Guild */
void OnAddMember(Guild* guild, Player* player, uint32 plRank);
#if ELUNA_EXPANSION == EXP_RETAIL
void OnRemoveMember(Guild* guild, ObjectGuid guid, bool isDisbanding);
#else
void OnRemoveMember(Guild* guild, Player* player, bool isDisbanding);
#endif
void OnMOTDChanged(Guild* guild, const std::string& newMotd);
void OnInfoChanged(Guild* guild, const std::string& newInfo);
void OnCreate(Guild* guild, Player* leader, const std::string& name);
void OnDisband(Guild* guild);
#if ELUNA_EXPANSION < EXP_CATA
void OnMemberWitdrawMoney(Guild* guild, Player* player, uint32& amount, bool isRepair);
void OnMemberDepositMoney(Guild* guild, Player* player, uint32& amount);
#else
void OnMemberWitdrawMoney(Guild* guild, Player* player, uint64& amount, bool isRepair);
void OnMemberDepositMoney(Guild* guild, Player* player, uint64& amount);
#endif
void OnItemMove(Guild* guild, Player* player, Item* pItem, bool isSrcBank, uint8 srcContainer, uint8 srcSlotId, bool isDestBank, uint8 destContainer, uint8 destSlotId);
void OnEvent(Guild* guild, uint8 eventType, uint32 playerGuid1, uint32 playerGuid2, uint8 newRank);
void OnBankEvent(Guild* guild, uint8 eventType, uint8 tabId, uint32 playerGuid, uint32 itemOrMoney, uint16 itemStackCount, uint8 destTabId);
/* Group */
void OnAddMember(Group* group, ObjectGuid guid);
void OnInviteMember(Group* group, ObjectGuid guid);
void OnRemoveMember(Group* group, ObjectGuid guid, uint8 method);
void OnChangeLeader(Group* group, ObjectGuid newLeaderGuid, ObjectGuid oldLeaderGuid);
void OnDisband(Group* group);
#if ELUNA_EXPANSION < EXP_RETAIL
void OnCreate(Group* group, ObjectGuid leaderGuid, GroupType groupType);
#else
void OnCreate(Group* group, ObjectGuid leaderGuid, GroupFlags groupFlags);
#endif
bool OnMemberAccept(Group* group, Player* player);
/* Map */
void OnCreate(Map* map);
void OnDestroy(Map* map);
void OnPlayerEnter(Map* map, Player* player);
void OnPlayerLeave(Map* map, Player* player);
void OnMapUpdate(Map* map, uint32 diff);
void OnAddToWorld(Creature* creature);
void OnRemoveFromWorld(Creature* creature);
void OnAddToWorld(GameObject* gameobject);
void OnRemoveFromWorld(GameObject* gameobject);
void OnRemove(Creature* creature);
void OnRemove(GameObject* gameobject);
/* Instance */
void OnInitialize(ElunaInstanceAI* ai);
void OnLoad(ElunaInstanceAI* ai);
void OnUpdateInstance(ElunaInstanceAI* ai, uint32 diff);
void OnPlayerEnterInstance(ElunaInstanceAI* ai, Player* player);
void OnCreatureCreate(ElunaInstanceAI* ai, Creature* creature);
void OnGameObjectCreate(ElunaInstanceAI* ai, GameObject* gameobject);
bool OnCheckEncounterInProgress(ElunaInstanceAI* ai);
/* World */
void OnOpenStateChange(bool open);
void OnConfigLoad(bool reload);
void OnShutdownInitiate(ShutdownExitCode code, ShutdownMask mask);
void OnShutdownCancel();
void OnStartup();
void OnShutdown();
void OnGameEventStart(uint32 eventid);
void OnGameEventStop(uint32 eventid);
/* Battle Ground */
void OnBGStart(BattleGround* bg, BattleGroundTypeId bgId, uint32 instanceId);
void OnBGEnd(BattleGround* bg, BattleGroundTypeId bgId, uint32 instanceId, Team winner);
void OnBGCreate(BattleGround* bg, BattleGroundTypeId bgId, uint32 instanceId);
void OnBGDestroy(BattleGround* bg, BattleGroundTypeId bgId, uint32 instanceId);
/* Spell */
void OnSpellCast(Spell* pSpell, bool skipCheck);
};
template<> Unit* Eluna::CHECKOBJ(int narg, bool error);
template<> Object* Eluna::CHECKOBJ