Fixes FS #297 and part of FS #403. Added sizes to all entities. Moved all damage-related functions from cPawn to cEntity API change: renamed cPawn:TeleportTo() to cEntity:TeleportToCoords() git-svn-id: http://mc-server.googlecode.com/svn/trunk@1635 0a769ca7-a7f5-676a-18bf-c427514a06d6
		
			
				
	
	
		
			1384 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1384 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
 | 
						|
#include "Globals.h"  // NOTE: MSVC stupidness requires this to be the same across all modules
 | 
						|
 | 
						|
#include "PluginManager.h"
 | 
						|
#include "Plugin.h"
 | 
						|
#include "Plugin_NewLua.h"
 | 
						|
#include "WebAdmin.h"
 | 
						|
#include "Item.h"
 | 
						|
#include "Root.h"
 | 
						|
#include "Server.h"
 | 
						|
#include "CommandOutput.h"
 | 
						|
 | 
						|
#include "../iniFile/iniFile.h"
 | 
						|
#include "tolua++.h"
 | 
						|
#include "Player.h"
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
cPluginManager * cPluginManager::Get(void)
 | 
						|
{
 | 
						|
	return cRoot::Get()->GetPluginManager();
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
cPluginManager::cPluginManager(void) :
 | 
						|
	m_bReloadPlugins(false)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
cPluginManager::~cPluginManager()
 | 
						|
{
 | 
						|
	UnloadPluginsNow();
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
void cPluginManager::ReloadPlugins(void)
 | 
						|
{
 | 
						|
	m_bReloadPlugins = true;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
void cPluginManager::FindPlugins(void)
 | 
						|
{
 | 
						|
	AString PluginsPath = FILE_IO_PREFIX + AString( "Plugins/" );
 | 
						|
 | 
						|
	// First get a clean list of only the currently running plugins, we don't want to mess those up
 | 
						|
	for (PluginMap::iterator itr = m_Plugins.begin(); itr != m_Plugins.end();)
 | 
						|
	{
 | 
						|
		if( itr->second == NULL )
 | 
						|
		{
 | 
						|
			PluginMap::iterator thiz = itr;
 | 
						|
			++thiz;
 | 
						|
			m_Plugins.erase( itr );
 | 
						|
			itr = thiz;
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		++itr;
 | 
						|
	}
 | 
						|
 | 
						|
	AStringList Files = GetDirectoryContents(PluginsPath.c_str());
 | 
						|
	for (AStringList::const_iterator itr = Files.begin(); itr != Files.end(); ++itr)
 | 
						|
	{
 | 
						|
		if (itr->rfind(".") != AString::npos)
 | 
						|
		{
 | 
						|
			// Ignore files, we only want directories
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
 | 
						|
		// Add plugin name/directory to the list
 | 
						|
		if (m_Plugins.find( *itr ) == m_Plugins.end())
 | 
						|
		{
 | 
						|
			m_Plugins[ *itr ] = NULL;
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
void cPluginManager::ReloadPluginsNow(void)
 | 
						|
{
 | 
						|
	LOG("Loading plugins");
 | 
						|
	m_bReloadPlugins = false;
 | 
						|
	UnloadPluginsNow();
 | 
						|
 | 
						|
	FindPlugins();
 | 
						|
 | 
						|
	cServer::BindBuiltInConsoleCommands();
 | 
						|
 | 
						|
	cIniFile IniFile("settings.ini");
 | 
						|
	if (!IniFile.ReadFile())
 | 
						|
	{
 | 
						|
		LOGWARNING("cPluginManager: Can't find settings.ini, so can't load any plugins.");
 | 
						|
	}
 | 
						|
		
 | 
						|
	unsigned int KeyNum = IniFile.FindKey("Plugins");
 | 
						|
	unsigned int NumPlugins = IniFile.GetNumValues(KeyNum);
 | 
						|
	if (NumPlugins > 0)
 | 
						|
	{
 | 
						|
		for(unsigned int i = 0; i < NumPlugins; i++)
 | 
						|
		{
 | 
						|
			AString ValueName = IniFile.GetValueName(KeyNum, i );
 | 
						|
			if (
 | 
						|
				(ValueName.compare("NewPlugin") == 0) || 
 | 
						|
				(ValueName.compare("Plugin") == 0)
 | 
						|
			)
 | 
						|
			{
 | 
						|
				AString PluginFile = IniFile.GetValue(KeyNum, i);
 | 
						|
				if (!PluginFile.empty())
 | 
						|
				{
 | 
						|
					if (m_Plugins.find(PluginFile) != m_Plugins.end())
 | 
						|
					{
 | 
						|
						LoadPlugin( PluginFile );
 | 
						|
					}
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (GetNumPlugins() == 0)
 | 
						|
	{
 | 
						|
		LOG("No plugins loaded");
 | 
						|
	}
 | 
						|
	else
 | 
						|
	{
 | 
						|
		LOG("Loaded %i plugin(s)", GetNumPlugins());
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
void cPluginManager::Tick(float a_Dt)
 | 
						|
{
 | 
						|
	while (!m_DisablePluginList.empty())
 | 
						|
	{
 | 
						|
		RemovePlugin(m_DisablePluginList.front());
 | 
						|
		m_DisablePluginList.pop_front();
 | 
						|
	}
 | 
						|
 | 
						|
	if (m_bReloadPlugins)
 | 
						|
	{
 | 
						|
		ReloadPluginsNow();
 | 
						|
	}
 | 
						|
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_TICK);
 | 
						|
	if (Plugins != m_Hooks.end())
 | 
						|
	{
 | 
						|
		for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
		{
 | 
						|
			(*itr)->Tick(a_Dt);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookBlockToPickups(
 | 
						|
	cWorld * a_World, cEntity * a_Digger,
 | 
						|
	int a_BlockX, int a_BlockY, int a_BlockZ, BLOCKTYPE a_BlockType, NIBBLETYPE a_BlockMeta, 
 | 
						|
	cItems & a_Pickups
 | 
						|
)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_BLOCK_TO_PICKUPS);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnBlockToPickups(a_World, a_Digger, a_BlockX, a_BlockY, a_BlockZ, a_BlockType, a_BlockMeta, a_Pickups))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookChat(cPlayer * a_Player, AString & a_Message)
 | 
						|
{
 | 
						|
	if (ExecuteCommand(a_Player, a_Message))
 | 
						|
	{
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
 | 
						|
	// Check if it was a standard command (starts with a slash)
 | 
						|
	if (!a_Message.empty() && (a_Message[0] == '/'))
 | 
						|
	{
 | 
						|
		AStringVector Split(StringSplit(a_Message, " "));
 | 
						|
		ASSERT(!Split.empty());  // This should not happen - we know there's at least one char in the message so the split needs to be at least one item long
 | 
						|
		a_Player->SendMessage(Printf("Unknown Command: \"%s\"", Split[0].c_str()));
 | 
						|
		LOGINFO("Player \"%s\" issued an unknown command: \"%s\"", a_Player->GetName().c_str(), a_Message.c_str());
 | 
						|
		return true;	// Cancel sending
 | 
						|
	}
 | 
						|
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_CHAT);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnChat(a_Player, a_Message))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookChunkAvailable(cWorld * a_World, int a_ChunkX, int a_ChunkZ)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_CHUNK_AVAILABLE);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnChunkAvailable(a_World, a_ChunkX, a_ChunkZ))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookChunkGenerated(cWorld * a_World, int a_ChunkX, int a_ChunkZ, cChunkDesc * a_ChunkDesc)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_CHUNK_GENERATED);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnChunkGenerated(a_World, a_ChunkX, a_ChunkZ, a_ChunkDesc))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookChunkGenerating(cWorld * a_World, int a_ChunkX, int a_ChunkZ, cChunkDesc * a_ChunkDesc)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_CHUNK_GENERATING);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnChunkGenerating(a_World, a_ChunkX, a_ChunkZ, a_ChunkDesc))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookChunkUnloaded(cWorld * a_World, int a_ChunkX, int a_ChunkZ)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_CHUNK_UNLOADED);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnChunkUnloaded(a_World, a_ChunkX, a_ChunkZ))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookChunkUnloading(cWorld * a_World, int a_ChunkX, int a_ChunkZ)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_CHUNK_UNLOADING);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnChunkUnloading(a_World, a_ChunkX, a_ChunkZ))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookCollectingPickup(cPlayer * a_Player, cPickup & a_Pickup)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_COLLECTING_PICKUP);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnCollectingPickup(a_Player, &a_Pickup))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookCraftingNoRecipe(const cPlayer * a_Player, const cCraftingGrid * a_Grid, cCraftingRecipe * a_Recipe)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_CRAFTING_NO_RECIPE);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnCraftingNoRecipe(a_Player, a_Grid, a_Recipe))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookDisconnect(cPlayer * a_Player, const AString & a_Reason)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_DISCONNECT);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnDisconnect(a_Player, a_Reason))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookExecuteCommand(cPlayer * a_Player, const AStringVector & a_Split)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_EXECUTE_COMMAND);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnExecuteCommand(a_Player, a_Split))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookHandshake(cClientHandle * a_ClientHandle, const AString & a_Username)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_HANDSHAKE);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnHandshake(a_ClientHandle, a_Username))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookKilling(cEntity & a_Victim, cEntity * a_Killer)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_KILLING);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnKilling(a_Victim, a_Killer))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookLogin(cClientHandle * a_Client, int a_ProtocolVersion, const AString & a_Username)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_LOGIN);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnLogin(a_Client, a_ProtocolVersion, a_Username))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerBreakingBlock(cPlayer & a_Player, int a_BlockX, int a_BlockY, int a_BlockZ, char a_BlockFace, BLOCKTYPE a_BlockType, NIBBLETYPE a_BlockMeta)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_BREAKING_BLOCK);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerBreakingBlock(a_Player, a_BlockX, a_BlockY, a_BlockZ, a_BlockFace, a_BlockType, a_BlockMeta))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerBrokenBlock(cPlayer & a_Player, int a_BlockX, int a_BlockY, int a_BlockZ, char a_BlockFace, BLOCKTYPE a_BlockType, NIBBLETYPE a_BlockMeta)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_BROKEN_BLOCK);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerBrokenBlock(a_Player, a_BlockX, a_BlockY, a_BlockZ, a_BlockFace, a_BlockType, a_BlockMeta))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerEating(cPlayer & a_Player)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_EATING);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerEating(a_Player))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerJoined(cPlayer & a_Player)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_JOINED);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerJoined(a_Player))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerLeftClick(cPlayer & a_Player, int a_BlockX, int a_BlockY, int a_BlockZ, char a_BlockFace, char a_Status)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_LEFT_CLICK);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerLeftClick(a_Player, a_BlockX, a_BlockY, a_BlockZ, a_BlockFace, a_Status))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerMoving(cPlayer & a_Player)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_MOVING);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerMoved(a_Player))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerPlacedBlock(cPlayer & a_Player, int a_BlockX, int a_BlockY, int a_BlockZ, char a_BlockFace, int a_CursorX, int a_CursorY, int a_CursorZ, BLOCKTYPE a_BlockType, NIBBLETYPE a_BlockMeta)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_PLACED_BLOCK);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerPlacedBlock(a_Player, a_BlockX, a_BlockY, a_BlockZ, a_BlockFace, a_CursorX, a_CursorY, a_CursorZ, a_BlockType, a_BlockMeta))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerPlacingBlock(cPlayer & a_Player, int a_BlockX, int a_BlockY, int a_BlockZ, char a_BlockFace, int a_CursorX, int a_CursorY, int a_CursorZ, BLOCKTYPE a_BlockType, NIBBLETYPE a_BlockMeta)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_PLACING_BLOCK);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerPlacingBlock(a_Player, a_BlockX, a_BlockY, a_BlockZ, a_BlockFace, a_CursorX, a_CursorY, a_CursorZ, a_BlockType, a_BlockMeta))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerRightClick(cPlayer & a_Player, int a_BlockX, int a_BlockY, int a_BlockZ, char a_BlockFace, int a_CursorX, int a_CursorY, int a_CursorZ)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_RIGHT_CLICK);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerRightClick(a_Player, a_BlockX, a_BlockY, a_BlockZ, a_BlockFace, a_CursorX, a_CursorY, a_CursorZ))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerShooting(cPlayer & a_Player)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_SHOOTING);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerShooting(a_Player))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerSpawned(cPlayer & a_Player)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_SPAWNED);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerSpawned(a_Player))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerTossingItem(cPlayer & a_Player)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_TOSSING_ITEM);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerTossingItem(a_Player))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerUsedBlock(cPlayer & a_Player, int a_BlockX, int a_BlockY, int a_BlockZ, char a_BlockFace, int a_CursorX, int a_CursorY, int a_CursorZ, BLOCKTYPE a_BlockType, NIBBLETYPE a_BlockMeta)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_USED_BLOCK);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerUsedBlock(a_Player, a_BlockX, a_BlockY, a_BlockZ, a_BlockFace, a_CursorX, a_CursorY, a_CursorZ, a_BlockType, a_BlockMeta))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerUsedItem(cPlayer & a_Player, int a_BlockX, int a_BlockY, int a_BlockZ, char a_BlockFace, int a_CursorX, int a_CursorY, int a_CursorZ)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_USED_ITEM);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerUsedItem(a_Player, a_BlockX, a_BlockY, a_BlockZ, a_BlockFace, a_CursorX, a_CursorY, a_CursorZ))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerUsingBlock(cPlayer & a_Player, int a_BlockX, int a_BlockY, int a_BlockZ, char a_BlockFace, int a_CursorX, int a_CursorY, int a_CursorZ, BLOCKTYPE a_BlockType, NIBBLETYPE a_BlockMeta)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_USING_BLOCK);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerUsingBlock(a_Player, a_BlockX, a_BlockY, a_BlockZ, a_BlockFace, a_CursorX, a_CursorY, a_CursorZ, a_BlockType, a_BlockMeta))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPlayerUsingItem(cPlayer & a_Player, int a_BlockX, int a_BlockY, int a_BlockZ, char a_BlockFace, int a_CursorX, int a_CursorY, int a_CursorZ)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PLAYER_USING_ITEM);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPlayerUsingItem(a_Player, a_BlockX, a_BlockY, a_BlockZ, a_BlockFace, a_CursorX, a_CursorY, a_CursorZ))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPostCrafting(const cPlayer * a_Player, const cCraftingGrid * a_Grid, cCraftingRecipe * a_Recipe)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_POST_CRAFTING);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPostCrafting(a_Player, a_Grid, a_Recipe))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookPreCrafting(const cPlayer * a_Player, const cCraftingGrid * a_Grid, cCraftingRecipe * a_Recipe)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_PRE_CRAFTING);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnPreCrafting(a_Player, a_Grid, a_Recipe))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookTakeDamage(cEntity & a_Receiver, TakeDamageInfo & a_TDI)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_TAKE_DAMAGE);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnTakeDamage(a_Receiver, a_TDI))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookUpdatingSign(cWorld * a_World, int a_BlockX, int a_BlockY, int a_BlockZ, AString & a_Line1, AString & a_Line2, AString & a_Line3, AString & a_Line4, cPlayer * a_Player)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_UPDATING_SIGN);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnUpdatingSign(a_World, a_BlockX, a_BlockY, a_BlockZ, a_Line1, a_Line2, a_Line3, a_Line4, a_Player))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookUpdatedSign(cWorld * a_World, int a_BlockX, int a_BlockY, int a_BlockZ, const AString & a_Line1, const AString & a_Line2, const AString & a_Line3, const AString & a_Line4, cPlayer * a_Player)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_UPDATED_SIGN);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnUpdatedSign(a_World, a_BlockX, a_BlockY, a_BlockZ, a_Line1, a_Line2, a_Line3, a_Line4, a_Player))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookWeatherChanged(cWorld & a_World)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_WEATHER_CHANGED);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnWeatherChanged(a_World))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::CallHookWeatherChanging(cWorld & a_World, eWeather & a_NewWeather)
 | 
						|
{
 | 
						|
	HookMap::iterator Plugins = m_Hooks.find(HOOK_WEATHER_CHANGING);
 | 
						|
	if (Plugins == m_Hooks.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	for (PluginList::iterator itr = Plugins->second.begin(); itr != Plugins->second.end(); ++itr)
 | 
						|
	{
 | 
						|
		if ((*itr)->OnWeatherChanging(a_World, a_NewWeather))
 | 
						|
		{
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::HandleCommand(cPlayer * a_Player, const AString & a_Command, bool a_ShouldCheckPermissions)
 | 
						|
{
 | 
						|
	ASSERT(a_Player != NULL);
 | 
						|
	
 | 
						|
	AStringVector Split(StringSplit(a_Command, " "));
 | 
						|
	if (Split.empty())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	
 | 
						|
	CommandMap::iterator cmd = m_Commands.find(Split[0]);
 | 
						|
	if (cmd == m_Commands.end())
 | 
						|
	{
 | 
						|
		// Command not found
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	
 | 
						|
	// Ask plugins first if a command is okay to execute the command:
 | 
						|
	if (CallHookExecuteCommand(a_Player, Split))
 | 
						|
	{
 | 
						|
		LOGINFO("Player \"%s\" tried executing command \"%s\" that was stopped by the HOOK_EXECUTE_COMMAND hook", a_Player->GetName().c_str(), Split[0].c_str());
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	
 | 
						|
	if (
 | 
						|
		a_ShouldCheckPermissions &&
 | 
						|
		!cmd->second.m_Permission.empty() &&
 | 
						|
		!a_Player->HasPermission(cmd->second.m_Permission)
 | 
						|
	)
 | 
						|
	{
 | 
						|
		LOGINFO("Player \"%s\" tried to execute forbidden command \"%s\".", a_Player->GetName().c_str(), Split[0].c_str());
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	ASSERT(cmd->second.m_Plugin != NULL);
 | 
						|
	
 | 
						|
	return cmd->second.m_Plugin->HandleCommand(Split, a_Player);	
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
cPlugin * cPluginManager::GetPlugin( const AString & a_Plugin ) const
 | 
						|
{
 | 
						|
	for( PluginMap::const_iterator itr = m_Plugins.begin(); itr != m_Plugins.end(); ++itr )
 | 
						|
	{
 | 
						|
		if (itr->second == NULL ) continue;
 | 
						|
		if (itr->second->GetName().compare(a_Plugin) == 0)
 | 
						|
		{
 | 
						|
			return itr->second;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
const cPluginManager::PluginMap & cPluginManager::GetAllPlugins() const
 | 
						|
{
 | 
						|
	return m_Plugins;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
void cPluginManager::UnloadPluginsNow()
 | 
						|
{
 | 
						|
	m_Hooks.clear();
 | 
						|
 | 
						|
	while (!m_Plugins.empty())
 | 
						|
	{
 | 
						|
		RemovePlugin(m_Plugins.begin()->second);
 | 
						|
	}
 | 
						|
 | 
						|
	m_Commands.clear();
 | 
						|
	m_ConsoleCommands.clear();
 | 
						|
	//SquirrelVM::Shutdown(); // This breaks shit
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::DisablePlugin(const AString & a_PluginName)
 | 
						|
{
 | 
						|
	PluginMap::iterator itr = m_Plugins.find(a_PluginName);
 | 
						|
	if (itr == m_Plugins.end())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	
 | 
						|
	if (itr->first.compare(a_PluginName) == 0)  // _X 2013_02_01: wtf? Isn't this supposed to be what find() does?
 | 
						|
	{
 | 
						|
		m_DisablePluginList.push_back(itr->second);
 | 
						|
		itr->second = NULL;	// Get rid of this thing right away
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::LoadPlugin(const AString & a_PluginName)
 | 
						|
{
 | 
						|
	return AddPlugin(new cPlugin_NewLua(a_PluginName.c_str()));
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
void cPluginManager::RemoveHooks(cPlugin * a_Plugin)
 | 
						|
{
 | 
						|
	for (HookMap::iterator itr = m_Hooks.begin(), end = m_Hooks.end(); itr != end; ++itr)
 | 
						|
	{
 | 
						|
		itr->second.remove(a_Plugin);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
void cPluginManager::RemovePlugin(cPlugin * a_Plugin)
 | 
						|
{
 | 
						|
	for (PluginMap::iterator itr = m_Plugins.begin(); itr != m_Plugins.end(); ++itr)
 | 
						|
	{
 | 
						|
		if (itr->second == a_Plugin)
 | 
						|
		{
 | 
						|
			m_Plugins.erase(itr);
 | 
						|
			break;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	
 | 
						|
	RemovePluginCommands(a_Plugin);
 | 
						|
	RemovePluginConsoleCommands(a_Plugin);
 | 
						|
	RemoveHooks(a_Plugin);
 | 
						|
	if (a_Plugin != NULL)
 | 
						|
	{
 | 
						|
		a_Plugin->OnDisable();
 | 
						|
	}
 | 
						|
	delete a_Plugin;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
void cPluginManager::RemovePluginCommands(cPlugin * a_Plugin)
 | 
						|
{
 | 
						|
	if (a_Plugin != NULL)
 | 
						|
	{
 | 
						|
		a_Plugin->ClearCommands();
 | 
						|
	}
 | 
						|
	
 | 
						|
	for (CommandMap::iterator itr = m_Commands.begin(); itr != m_Commands.end();)
 | 
						|
	{
 | 
						|
		if (itr->second.m_Plugin == a_Plugin)
 | 
						|
		{
 | 
						|
			CommandMap::iterator EraseMe = itr;  // Stupid GCC doesn't have a std::map::erase() that would return the next iterator
 | 
						|
			++itr;
 | 
						|
			m_Commands.erase(EraseMe);
 | 
						|
		}
 | 
						|
		else
 | 
						|
		{
 | 
						|
			++itr;
 | 
						|
		}
 | 
						|
	}  // for itr - m_Commands[]
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::BindCommand(const AString & a_Command, cPlugin * a_Plugin, const AString & a_Permission, const AString & a_HelpString)
 | 
						|
{
 | 
						|
	CommandMap::iterator cmd = m_Commands.find(a_Command);
 | 
						|
	if (cmd != m_Commands.end())
 | 
						|
	{
 | 
						|
		LOGWARNING("Command \"%s\" is already bound to plugin \"%s\".", a_Command.c_str(), cmd->second.m_Plugin->GetName().c_str());
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	
 | 
						|
	m_Commands[a_Command].m_Plugin     = a_Plugin;
 | 
						|
	m_Commands[a_Command].m_Permission = a_Permission;
 | 
						|
	m_Commands[a_Command].m_HelpString = a_HelpString;
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::ForEachCommand(cCommandEnumCallback & a_Callback)
 | 
						|
{
 | 
						|
	for (CommandMap::iterator itr = m_Commands.begin(), end = m_Commands.end(); itr != end; ++itr)
 | 
						|
	{
 | 
						|
		if (a_Callback.Command(itr->first, itr->second.m_Plugin, itr->second.m_Permission, itr->second.m_HelpString))
 | 
						|
		{
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
	}  // for itr - m_Commands[]
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::IsCommandBound(const AString & a_Command)
 | 
						|
{
 | 
						|
	return (m_Commands.find(a_Command) != m_Commands.end());
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
AString cPluginManager::GetCommandPermission(const AString & a_Command)
 | 
						|
{
 | 
						|
	CommandMap::iterator cmd = m_Commands.find(a_Command);
 | 
						|
	return (cmd == m_Commands.end()) ? "" : cmd->second.m_Permission;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::ExecuteCommand(cPlayer * a_Player, const AString & a_Command)
 | 
						|
{
 | 
						|
	return HandleCommand(a_Player, a_Command, true);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::ForceExecuteCommand(cPlayer * a_Player, const AString & a_Command)
 | 
						|
{
 | 
						|
	return HandleCommand(a_Player, a_Command, false);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
void cPluginManager::RemovePluginConsoleCommands(cPlugin * a_Plugin)
 | 
						|
{
 | 
						|
	if (a_Plugin != NULL)
 | 
						|
	{
 | 
						|
		a_Plugin->ClearConsoleCommands();
 | 
						|
	}
 | 
						|
	
 | 
						|
	for (CommandMap::iterator itr = m_ConsoleCommands.begin(); itr != m_ConsoleCommands.end();)
 | 
						|
	{
 | 
						|
		if (itr->second.m_Plugin == a_Plugin)
 | 
						|
		{
 | 
						|
			CommandMap::iterator EraseMe = itr;  // Stupid GCC doesn't have a std::map::erase() that would return the next iterator
 | 
						|
			++itr;
 | 
						|
			m_ConsoleCommands.erase(EraseMe);
 | 
						|
		}
 | 
						|
		else
 | 
						|
		{
 | 
						|
			++itr;
 | 
						|
		}
 | 
						|
	}  // for itr - m_Commands[]
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::BindConsoleCommand(const AString & a_Command, cPlugin * a_Plugin, const AString & a_HelpString)
 | 
						|
{
 | 
						|
	CommandMap::iterator cmd = m_ConsoleCommands.find(a_Command);
 | 
						|
	if (cmd != m_ConsoleCommands.end())
 | 
						|
	{
 | 
						|
		if (cmd->second.m_Plugin == NULL)
 | 
						|
		{
 | 
						|
			LOGWARNING("Console command \"%s\" is already bound internally by MCServer.", a_Command.c_str());
 | 
						|
		}
 | 
						|
		else
 | 
						|
		{
 | 
						|
			LOGWARNING("Console command \"%s\" is already bound to plugin \"%s\".", a_Command.c_str(), cmd->second.m_Plugin->GetName().c_str());
 | 
						|
		}
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	
 | 
						|
	m_ConsoleCommands[a_Command].m_Plugin     = a_Plugin;
 | 
						|
	m_ConsoleCommands[a_Command].m_Permission = "";
 | 
						|
	m_ConsoleCommands[a_Command].m_HelpString = a_HelpString;
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::ForEachConsoleCommand(cCommandEnumCallback & a_Callback)
 | 
						|
{
 | 
						|
	for (CommandMap::iterator itr = m_ConsoleCommands.begin(), end = m_ConsoleCommands.end(); itr != end; ++itr)
 | 
						|
	{
 | 
						|
		if (a_Callback.Command(itr->first, itr->second.m_Plugin, "", itr->second.m_HelpString))
 | 
						|
		{
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
	}  // for itr - m_Commands[]
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::IsConsoleCommandBound(const AString & a_Command)
 | 
						|
{
 | 
						|
	return (m_ConsoleCommands.find(a_Command) != m_ConsoleCommands.end());
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::ExecuteConsoleCommand(const AStringVector & a_Split, cCommandOutputCallback & a_Output)
 | 
						|
{
 | 
						|
	if (a_Split.empty())
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	
 | 
						|
	CommandMap::iterator cmd = m_ConsoleCommands.find(a_Split[0]);
 | 
						|
	if (cmd == m_ConsoleCommands.end())
 | 
						|
	{
 | 
						|
		// Command not found
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	
 | 
						|
	if (cmd->second.m_Plugin == NULL)
 | 
						|
	{
 | 
						|
		// This is a built-in command
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	
 | 
						|
	// Ask plugins first if a command is okay to execute the console command:
 | 
						|
	if (CallHookExecuteCommand(NULL, a_Split))
 | 
						|
	{
 | 
						|
		a_Output.Out("Command \"%s\" was stopped by the HOOK_EXECUTE_COMMAND hook", a_Split[0].c_str());
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	return cmd->second.m_Plugin->HandleConsoleCommand(a_Split, a_Output);	
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool cPluginManager::AddPlugin(cPlugin * a_Plugin)
 | 
						|
{
 | 
						|
	m_Plugins[a_Plugin->GetDirectory()] = a_Plugin;
 | 
						|
	if (a_Plugin->Initialize())
 | 
						|
	{
 | 
						|
		// Initialization OK
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
 | 
						|
	// Initialization failed
 | 
						|
	RemovePlugin(a_Plugin);  // Also undoes any registrations that Initialize() might have made
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
void cPluginManager::AddHook(cPlugin * a_Plugin, PluginHook a_Hook)
 | 
						|
{
 | 
						|
	if (!a_Plugin)
 | 
						|
	{
 | 
						|
		LOGWARN("Called cPluginManager::AddHook() with a_Plugin == NULL");
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	if (!a_Plugin->CanAddHook(a_Hook))
 | 
						|
	{
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	PluginList & Plugins = m_Hooks[a_Hook];
 | 
						|
	Plugins.remove(a_Plugin);
 | 
						|
	Plugins.push_back(a_Plugin);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
unsigned int cPluginManager::GetNumPlugins() const
 | 
						|
{
 | 
						|
	return m_Plugins.size(); 
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 |