Ir para conteúdo
  • Cadastre-se

Scripting Queria um HEALING igual ao do global, na cor azul, e mostrando a numeração apenas do que falta pra enxer o life


Posts Recomendados

Em meu ot quando eu healo ou uso potion, aparece na cor verde feia e mostra o valor total do heal, tipo eu healo 1000 com 1000 de life, mesmo acim aparece 1000 na cor verde

Eu queria um igual ao do global qe vcs devem conhecer bem, na cor azul claro linda e que mostra a numeração apenas do que encheu do life, tipo tenho 1000 de life e falta 200 pra encher, ela mostra o numero 200. Grato.

Link para o post
Compartilhar em outros sites

Preciso analizar o script das potions que fica em actions\scripts\liquids.

 

Um abraço! 

se ajudei reP+

 

Link para o post
Compartilhar em outros sites

actions/scripts/liquids/potions.lua

local config = { 
        removeOnUse = "no", 
        usableOnTarget = "yes", -- can be used on target? (fe. healing friend) 
        splashable = "no", 
        realAnimation = "no", -- make text effect visible only for players in range 1x1 
        healthMultiplier = 1.0, 
        manaMultiplier = 1.0 
} 
 
config.removeOnUse = getBooleanFromString(config.removeOnUse) 
config.usableOnTarget = getBooleanFromString(config.usableOnTarget) 
config.splashable = getBooleanFromString(config.splashable) 
config.realAnimation = getBooleanFromString(config.realAnimation) 
 
local POTIONS = { 
        [8704] = {empty = 7636, splash = 2, health = {150, 200}}, -- small health potion 
        [7618] = {empty = 7636, splash = 2, health = {200, 300}}, -- health potion 
        [7588] = {empty = 7634, splash = 2, health = {200, 400}, level = 50, vocations = {3, 4, 7, 8}, vocStr = "knights and paladins"}, -- strong health potion 
        [7591] = {empty = 7635, splash = 2, health = {700, 900}, level = 80, vocations = {4, 8}, vocStr = "knights"}, -- great health potion 
        [8473] = {empty = 7635, splash = 2, health = {1200, 1400}, level = 130, vocations = {4, 8}, vocStr = "knights"}, -- ultimate health potion 
        [7620] = {empty = 7636, splash = 7, mana = {150, 350}}, -- mana potion 
        [7589] = {empty = 7634, splash = 7, mana = {200, 400}, level = 50, vocations = {1, 2, 3, 5, 6, 7}, vocStr = "sorcerers, druids and paladins"}, -- strong mana potion 
        [7590] = {empty = 7635, splash = 7, mana = {1000, 1200}, level = 80, vocations = {1, 2, 5, 6}, vocStr = "sorcerers and druids"}, -- great mana potion 
        [8472] = {empty = 7635, splash = 3, health = {900, 1100}, mana = {400, 400}, level = 80, vocations = {3, 7}, vocStr = "paladins"}, -- great spirit potion 
} 
 
local exhaust = createConditionObject(CONDITION_EXHAUST) 
setConditionParam(exhaust, CONDITION_PARAM_TICKS, (getConfigInfo('timeBetweenExActions') - 100)) 
 
function onUse(cid, item, fromPosition, itemEx, toPosition) 
        local potion = POTIONS[item.itemid] 
        if(not potion) then 
                return false 
        end 
        if(not isPlayer(itemEx.uid) or (not config.usableOnTarget and cid ~= itemEx.uid)) then 
                if(not config.splashable) then 
                        return false 
                end 
                if(toPosition.x == CONTAINER_POSITION) then 
                        toPosition = getThingPos(item.uid) 
                end 
                doDecayItem(doCreateItem(2016, potion.splash, toPosition)) 
                doTransformItem(item.uid, potion.empty) 
                return TRUE 
        end 
        if(hasCondition(cid, CONDITION_EXHAUST_HEAL)) then 
                doPlayerSendDefaultCancel(cid, RETURNVALUE_YOUAREEXHAUSTED) 
                return TRUE 
        end 
        if(((potion.level and getPlayerLevel(cid) < potion.level) or (potion.vocations and not isInArray(potion.vocations, getPlayerVocation(cid)))) and 
                not getPlayerCustomFlagValue(cid, PLAYERCUSTOMFLAG_GAMEMASTERPRIVILEGES)) 
        then 
                doCreatureSay(itemEx.uid, "Only " .. potion.vocStr .. (potion.level and (" of level " .. potion.level) or "") .. " or above may drink this fluid.", TALKTYPE_ORANGE_1) 
                return TRUE 
        end 
        local health = potion.health 
        if(health and not doCreatureAddHealth(itemEx.uid, math.ceil(math.random(health[1], health[2]) * config.healthMultiplier))) then 
                return false 
        end 
        local mana = potion.mana 
        if(mana and not doPlayerAddMana(itemEx.uid, math.ceil(math.random(mana[1], mana[2]) * config.manaMultiplier))) then 
                return false 
        end  
        doAddCondition(cid, exhaust)
        doSendMagicEffect(getThingPos(itemEx.uid), 30)
        doSendAnimatedText(fromPosition, "Aaaah...", 17)
        if(not potion.empty or config.removeOnUse) then 
        doRemoveItem(item.uid, 1) 
        return TRUE 
        end 
        doRemoveItem(item.uid, 0) 
        doPlayerAddItem(cid, potion.empty, 0) 
        doPlayerRemoveItem(cid, potion.empty, getPlayerItemCount(cid, potion.empty)) 
        doPlayerAddItem(cid, potion.empty, getPlayerItemCount(cid, potion.empty)) 
        return TRUE 
end

Em vez de deixar todas as potions *que seria mto trabalhoso* vou deixar apenas a great spirit pra tomar como base.

/liquids/great_mana.lua:

local MIN = 1050
local MAX = 1200
local EMPTY_POTION = 7635

local exhaust = createConditionObject(CONDITION_EXHAUST)
setConditionParam(exhaust, CONDITION_PARAM_TICKS, (getConfigInfo('timeBetweenExActions') - 100))

function onUse(cid, item, fromPosition, itemEx, toPosition)
	if isPlayer(itemEx.uid) == FALSE then
		return FALSE
	end

	if hasCondition(cid, CONDITION_EXHAUST_HEAL) == TRUE then
		doPlayerSendDefaultCancel(cid, RETURNVALUE_YOUAREEXHAUSTED)
		return TRUE
	end

	if((not(isSorcerer(itemEx.uid) or isDruid(itemEx.uid)) or getPlayerLevel(itemEx.uid) < 80) and getPlayerCustomFlagValue(itemEx.uid, PlayerCustomFlag_GamemasterPrivileges) == FALSE) then
		doCreatureSay(itemEx.uid, "Only sorcerers and druids of level 80 or above may drink this fluid.", TALKTYPE_ORANGE_1)
		return TRUE
	end

	if doPlayerAddMana(itemEx.uid, math.random(MIN, MAX)) == LUA_ERROR then
		return FALSE
	end

	doAddCondition(cid, exhaust)
	doSendMagicEffect(getThingPos(itemEx.uid), CONST_ME_MAGIC_BLUE)
	doCreatureSay(itemEx.uid, "Aaaah...", TALKTYPE_ORANGE_1)
	doRemoveItem(item.uid, 0)
     doPlayerAddItem(cid, EMPTY_POTION, 0)
	return TRUE
end

vou deixar também o game.cpp que é um arquivo da source que mexe diretamente com as potions.

/source/game.cpp:

Spoiler

////////////////////////////////////////////////////////////////////////
// OpenTibia - an opensource roleplaying game
////////////////////////////////////////////////////////////////////////
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program.  If not, see <http://www.gnu.org/licenses/>.
////////////////////////////////////////////////////////////////////////
#include "otpch.h"
#include "game.h"

#include "configmanager.h"
#ifdef __LOGIN_SERVER__
#include "gameservers.h"
#endif
#include "server.h"
#include "chat.h"

#include "luascript.h"
#include "creature.h"
#include "combat.h"
#include "tile.h"

#include "database.h"
#include "iologindata.h"
#include "ioban.h"
#include "ioguild.h"

#include "items.h"
#include "trashholder.h"
#include "container.h"
#include "monsters.h"

#include "house.h"
#include "quests.h"

#include "actions.h"
#include "globalevent.h"
#include "movement.h"
#include "raids.h"
#include "scriptmanager.h"
#include "spells.h"
#include "talkaction.h"
#include "weapons.h"

#include "vocation.h"
#include "group.h"

#ifdef __EXCEPTION_TRACER__
#include "exception.h"
#endif

extern ConfigManager g_config;
extern Actions* g_actions;
extern Monsters g_monsters;
extern Npcs g_npcs;
extern Chat g_chat;
extern TalkActions* g_talkActions;
extern Spells* g_spells;
extern MoveEvents* g_moveEvents;
extern Weapons* g_weapons;
extern CreatureEvents* g_creatureEvents;
extern GlobalEvents* g_globalEvents;

Game::Game()
{
    gameState = GAMESTATE_NORMAL;
    worldType = WORLDTYPE_OPEN;
    map = NULL;
    playersRecord = lastStageLevel = 0;
    for(int32_t i = 0; i < 3; i++)
        globalSaveMessage = false;

    //(1440 minutes/day) * 10 seconds event interval / (3600 seconds/day)
    lightHourDelta = 1440 * 10 / 3600;
    lightHour = SUNRISE + (SUNSET - SUNRISE) / 2;
    lightLevel = LIGHT_LEVEL_DAY;
    lightState = LIGHT_STATE_DAY;

    lastBucket = checkCreatureLastIndex = checkLightEvent = checkCreatureEvent = checkDecayEvent = saveEvent = 0;
}

Game::~Game()
{
    if(map)
        delete map;
}

void Game::start(ServiceManager* servicer)
{
    checkDecayEvent = Scheduler::getInstance().addEvent(createSchedulerTask(EVENT_DECAYINTERVAL,
        boost::bind(&Game::checkDecay, this)));
    checkCreatureEvent = Scheduler::getInstance().addEvent(createSchedulerTask(EVENT_CREATURE_THINK_INTERVAL,
        boost::bind(&Game::checkCreatures, this)));
    checkLightEvent = Scheduler::getInstance().addEvent(createSchedulerTask(EVENT_LIGHTINTERVAL,
        boost::bind(&Game::checkLight, this)));

    services = servicer;
    if(!g_config.getBool(ConfigManager::GLOBALSAVE_ENABLED) || g_config.getNumber(ConfigManager::GLOBALSAVE_H) < 1 ||
        g_config.getNumber(ConfigManager::GLOBALSAVE_H) > 24 || g_config.getNumber(ConfigManager::GLOBALSAVE_M) < 0
        || g_config.getNumber(ConfigManager::GLOBALSAVE_M) > 59)
        return;

    time_t timeNow = time(NULL);
    const tm* theTime = localtime(&timeNow);

    int32_t prepareHour = g_config.getNumber(ConfigManager::GLOBALSAVE_H),
        prepareMinute = g_config.getNumber(ConfigManager::GLOBALSAVE_M) - 5,
        hoursLeft = 0, minutesLeft = 0;
    if(theTime->tm_hour > prepareHour)
    {
        hoursLeft = 23 - (theTime->tm_hour - prepareHour);
        if(theTime->tm_min > prepareMinute)
        {
            minutesLeft = 60 - (theTime->tm_min - prepareMinute);
            hoursLeft--;
        }
        else if(theTime->tm_min != prepareMinute)
            minutesLeft = prepareMinute - theTime->tm_min;
    }
    else if(theTime->tm_hour == prepareHour)
    {
        if(theTime->tm_min > (prepareMinute - 1) && theTime->tm_min < (prepareMinute + 6))
        {
            if(theTime->tm_min > (prepareMinute + 1))
                setGlobalSaveMessage(0, true);

            if(theTime->tm_min > (prepareMinute + 3))
                setGlobalSaveMessage(1, true);

            prepareGlobalSave();
        }
        else if(theTime->tm_min > prepareMinute)
        {
            minutesLeft = 60 - (theTime->tm_min - prepareMinute);
            hoursLeft = 23;
        }
        else
            minutesLeft = prepareMinute - theTime->tm_min;
    }
    else
    {
        hoursLeft = prepareHour - theTime->tm_hour;
        if(theTime->tm_min > prepareMinute)
        {
            minutesLeft = 60 - (theTime->tm_min - prepareMinute);
            hoursLeft--;
        }
        else if(theTime->tm_min != prepareMinute)
            minutesLeft = prepareMinute - theTime->tm_min;
    }

    if(!hoursLeft || !minutesLeft)
        return;

    uint32_t timeLeft = (hoursLeft * 3600000) + minutesLeft * 60000;
    saveEvent = Scheduler::getInstance().addEvent(createSchedulerTask(timeLeft,
        boost::bind(&Game::prepareGlobalSave, this)));
}

void Game::loadGameState()
{
    ScriptEnviroment::loadGameState();
    loadMotd();
    loadPlayersRecord();
    checkHighscores();
}

void Game::setGameState(GameState_t newState)
{
    if(gameState == GAMESTATE_SHUTDOWN)
        return; //this cannot be stopped

    if(gameState != newState)
    {
        gameState = newState;
        switch(newState)
        {
            case GAMESTATE_INIT:
            {
                Spawns::getInstance()->startup();
                Raids::getInstance()->loadFromXml();
                Raids::getInstance()->startup();
                Quests::getInstance()->loadFromXml();

                loadGameState();
                g_globalEvents->startup();

                IOBan::getInstance()->clearTemporials();
                if(g_config.getBool(ConfigManager::INIT_PREMIUM_UPDATE))
                    IOLoginData::getInstance()->updatePremiumDays();
                break;
            }

            case GAMESTATE_SHUTDOWN:
            {
                g_globalEvents->execute(GLOBALEVENT_SHUTDOWN);
                AutoList<Player>::iterator it = Player::autoList.begin();
                while(it != Player::autoList.end()) //kick all players that are still online
                {
                    it->second->kickPlayer(true, true);
                    it = Player::autoList.begin();
                }

                Houses::getInstance()->payHouses();
                saveGameState(false);
                Dispatcher::getInstance().addTask(createTask(boost::bind(&Game::shutdown, this)));

                Scheduler::getInstance().stop();
                Dispatcher::getInstance().stop();
                break;
            }

            case GAMESTATE_CLOSED:
            {
                AutoList<Player>::iterator it = Player::autoList.begin();
                while(it != Player::autoList.end()) //kick all players who not allowed to stay
                {
                    if(!it->second->hasFlag(PlayerFlag_CanAlwaysLogin))
                    {
                        it->second->kickPlayer(true, true);
                        it = Player::autoList.begin();
                    }
                    else
                        ++it;
                }

                saveGameState(false);
                break;
            }

            case GAMESTATE_NORMAL:
            case GAMESTATE_MAINTAIN:
            case GAMESTATE_STARTUP:
            case GAMESTATE_CLOSING:
            default:
                break;
        }
    }
}

void Game::saveGameState(bool shallow)
{
    std::clog << "> Saving server..." << std::endl;
    uint64_t start = OTSYS_TIME();
    if(gameState == GAMESTATE_NORMAL)
        setGameState(GAMESTATE_MAINTAIN);

    IOLoginData* io = IOLoginData::getInstance();
    for(AutoList<Player>::iterator it = Player::autoList.begin(); it != Player::autoList.end(); ++it)
    {
        it->second->loginPosition = it->second->getPosition();
        io->savePlayer(it->second, false, shallow);
    }

    std::string storage = "relational";
    if(g_config.getBool(ConfigManager::HOUSE_STORAGE))
        storage = "binary";

    map->saveMap();
    ScriptEnviroment::saveGameState();
    if(gameState == GAMESTATE_MAINTAIN)
        setGameState(GAMESTATE_NORMAL);

    std::clog << "> SAVE: Complete in " << (OTSYS_TIME() - start) / (1000.) << " seconds using "
        << storage << " house storage." << std::endl;
}

int32_t Game::loadMap(std::string filename)
{
    if(!map)
        map = new Map;

    std::string file = getFilePath(FILE_TYPE_CONFIG, "world/" + filename);
    if(!fileExists(file.c_str()))
        file = getFilePath(FILE_TYPE_OTHER, "world/" + filename);

    return map->loadMap(file);
}

void Game::cleanMapEx(uint32_t& count)
{
    uint64_t start = OTSYS_TIME();
    uint32_t tiles = 0; count = 0;

    int32_t marked = -1;
    if(gameState == GAMESTATE_NORMAL)
        setGameState(GAMESTATE_MAINTAIN);

    Tile* tile = NULL;
    ItemVector::iterator tit;
    if(g_config.getBool(ConfigManager::STORE_TRASH))
    {
        marked = trash.size();
        Trash::iterator it = trash.begin();
        if(g_config.getBool(ConfigManager::CLEAN_PROTECTED_ZONES))
        {
            for(; it != trash.end(); ++it)
            {
                if(!(tile = getTile(*it)))
                    continue;

                tile->resetFlag(TILESTATE_TRASHED);
                if(tile->hasFlag(TILESTATE_HOUSE) || !tile->getItemList())
                    continue;

                ++tiles;
                tit = tile->getItemList()->begin();
                while(tile->getItemList() && tit != tile->getItemList()->end())
                {
                    if((*tit)->isMoveable() && !(*tit)->isLoadedFromMap()
                        && !(*tit)->isScriptProtected())
                    {
                        internalRemoveItem(NULL, *tit);
                        if(tile->getItemList())
                            tit = tile->getItemList()->begin();

                        ++count;
                    }
                    else
                        ++tit;
                }
            }

            trash.clear();
        }
        else
        {
            for(; it != trash.end(); ++it)
            {
                if(!(tile = getTile(*it)))
                    continue;

                tile->resetFlag(TILESTATE_TRASHED);
                if(tile->hasFlag(TILESTATE_PROTECTIONZONE) || !tile->getItemList())
                    continue;

                ++tiles;
                tit = tile->getItemList()->begin();
                while(tile->getItemList() && tit != tile->getItemList()->end())
                {
                    if((*tit)->isMoveable() && !(*tit)->isLoadedFromMap()
                        && !(*tit)->isScriptProtected())
                    {
                        internalRemoveItem(NULL, *tit);
                        if(tile->getItemList())
                            tit = tile->getItemList()->begin();

                        ++count;
                    }
                    else
                        ++tit;
                }
            }

            trash.clear();
        }
    }
    else if(g_config.getBool(ConfigManager::CLEAN_PROTECTED_ZONES))
    {
        for(uint16_t z = 0; z < (uint16_t)MAP_MAX_LAYERS; z++)
        {
            for(uint16_t y = 1; y <= map->mapHeight; y++)
            {
                for(uint16_t x = 1; x <= map->mapWidth; x++)
                {
                    if(!(tile = getTile(x, y, z)) || tile->hasFlag(TILESTATE_HOUSE) || !tile->getItemList())
                        continue;

                    ++tiles;
                    tit = tile->getItemList()->begin();
                    while(tile->getItemList() && tit != tile->getItemList()->end())
                    {
                        if((*tit)->isMoveable() && !(*tit)->isLoadedFromMap()
                            && !(*tit)->isScriptProtected())
                        {
                            internalRemoveItem(NULL, *tit);
                            if(tile->getItemList())
                                tit = tile->getItemList()->begin();

                            ++count;
                        }
                        else
                            ++tit;
                    }
                }
            }
        }
    }
    else
    {
        for(uint16_t z = 0; z < (uint16_t)MAP_MAX_LAYERS; z++)
        {
            for(uint16_t y = 1; y <= map->mapHeight; y++)
            {
                for(uint16_t x = 1; x <= map->mapWidth; x++)
                {
                    if(!(tile = getTile(x, y, z)) || tile->hasFlag(TILESTATE_PROTECTIONZONE) || !tile->getItemList())
                        continue;

                    ++tiles;
                    tit = tile->getItemList()->begin();
                    while(tile->getItemList() && tit != tile->getItemList()->end())
                    {
                        if((*tit)->isMoveable() && !(*tit)->isLoadedFromMap()
                            && !(*tit)->isScriptProtected())
                        {
                            internalRemoveItem(NULL, *tit);
                            if(tile->getItemList())
                                tit = tile->getItemList()->begin();

                            ++count;
                        }
                        else
                            ++tit;
                    }
                }
            }
        }
    }

    if(gameState == GAMESTATE_MAINTAIN)
        setGameState(GAMESTATE_NORMAL);

    std::clog << "> CLEAN: Removed " << count << " item" << (count != 1 ? "s" : "")
        << " from " << tiles << " tile" << (tiles != 1 ? "s" : "");
    if(marked >= 0)
        std::clog << " (" << marked << " were marked)";

    std::clog << " in " << (OTSYS_TIME() - start) / (1000.) << " seconds." << std::endl;
}

void Game::cleanMap()
{
    uint32_t dummy;
    cleanMapEx(dummy);
}

void Game::proceduralRefresh(RefreshTiles::iterator* it/* = NULL*/)
{
    if(!it)
        it = new RefreshTiles::iterator(refreshTiles.begin());

    // Refresh 250 tiles each cycle
    refreshMap(it, 250);
    if((*it) != refreshTiles.end())
    {
        delete it;
        return;
    }

    // Refresh some items every 100 ms until all tiles has been checked
    // For 100k tiles, this would take 100000/2500 = 40s = half a minute
    Scheduler::getInstance().addEvent(createSchedulerTask(100,
        boost::bind(&Game::proceduralRefresh, this, it)));
}

void Game::refreshMap(RefreshTiles::iterator* it/* = NULL*/, uint32_t limit/* = 0*/)
{
    RefreshTiles::iterator end = refreshTiles.end();
    if(!it)
    {
        RefreshTiles::iterator begin = refreshTiles.begin();
        it = &begin;
    }

    Tile* tile = NULL;
    TileItemVector* items = NULL;

    Item* item = NULL;
    for(uint32_t cleaned = 0, downItemsSize = 0; (*it) != end &&
        (limit ? (cleaned < limit) : true); ++(*it), ++cleaned)
    {
        if(!(tile = (*it)->first))
            continue;

        if((items = tile->getItemList()))
        {
            downItemsSize = tile->getDownItemCount();
            for(uint32_t i = downItemsSize - 1; i; --i)
            {
                if((item = items->at(i)))
                {
                    #ifndef __DEBUG__
                    internalRemoveItem(NULL, item);
                    #else
                    if(internalRemoveItem(NULL, item) != RET_NOERROR)
                    {
                        std::clog << "> WARNING: Could not refresh item: " << item->getID();
                        std::clog << " at position: " << tile->getPosition() << std::endl;
                    }
                    #endif
                }
            }
        }

        cleanup();
        TileItemVector list = (*it)->second.list;
        for(ItemVector::reverse_iterator it = list.rbegin(); it != list.rend(); ++it)
        {
            Item* item = (*it)->clone();
            if(internalAddItem(NULL, tile, item , INDEX_WHEREEVER, FLAG_NOLIMIT) == RET_NOERROR)
            {
                if(item->getUniqueId() != 0)
                    ScriptEnviroment::addUniqueThing(item);

                startDecay(item);
            }
            else
            {
                std::clog << "> WARNING: Could not refresh item: " << item->getID()
                    << " at position: " << tile->getPosition() << std::endl;
                delete item;
            }
        }
    }
}

bool Game::isSwimmingPool(Item* item, const Tile* tile, bool checkProtection) const
{
    if(!tile)
        return false;

    TrashHolder* trashHolder = NULL;
    if(!item)
        trashHolder = tile->getTrashHolder();
    else
        trashHolder = item->getTrashHolder();

    return trashHolder && trashHolder->getEffect() == MAGIC_EFFECT_LOSE_ENERGY && (!checkProtection
        || tile->getZone() == ZONE_PROTECTION || tile->getZone() == ZONE_OPTIONAL);
}

Cylinder* Game::internalGetCylinder(Player* player, const Position& pos)
{
    if(pos.x != 0xFFFF)
        return getTile(pos);

    //container
    if(pos.y & 0x40)
        return player->getContainer((uint8_t)(pos.y & 0x0F));

    return player;
}

Thing* Game::internalGetThing(Player* player, const Position& pos, int32_t index,
    uint32_t spriteId/* = 0*/, stackposType_t type/* = STACKPOS_NORMAL*/)
{
    if(pos.x != 0xFFFF)
    {
        Tile* tile = getTile(pos);
        if(!tile)
            return NULL;

        if(type == STACKPOS_LOOK)
            return tile->getTopVisibleThing(player);

        Thing* thing = NULL;
        switch(type)
        {
            case STACKPOS_MOVE:
            {
                Item* item = tile->getTopDownItem();
                if(item && item->isMoveable())
                    thing = item;
                else
                    thing = tile->getTopVisibleCreature(player);

                break;
            }

            case STACKPOS_USE:
            {
                thing = tile->getTopDownItem();
                break;
            }

            case STACKPOS_USEITEM:
            {
                Item* downItem = tile->getTopDownItem();
                Item* item = tile->getItemByTopOrder(2);
                if(item && g_actions->hasAction(item))
                {
                    const ItemType& it = Item::items[item->getID()];
                    if(!downItem || (!it.hasHeight && !it.allowPickupable))
                        thing = item;
                }

                if(!thing)
                    thing = downItem;

                if(!thing)
                    thing = tile->getTopTopItem();

                if(!thing)
                    thing = tile->ground;

                break;
            }

            default:
                thing = tile->__getThing(index);
                break;
        }

        if(player && thing && thing->getItem())
        {
            if(tile->hasProperty(ISVERTICAL))
            {
                if(player->getPosition().x + 1 == tile->getPosition().x)
                    thing = NULL;
            }
            else if(tile->hasProperty(ISHORIZONTAL) && player->getPosition().y + 1 == tile->getPosition().y)
                thing = NULL;
        }

        return thing;
    }
    else if(pos.y & 0x40)
    {
        uint8_t fromCid = pos.y & 0x0F, slot = pos.z;
        if(Container* parentcontainer = player->getContainer(fromCid))
            return parentcontainer->getItem(slot);
    }
    else if(!pos.y && !pos.z)
    {
        const ItemType& it = Item::items.getItemIdByClientId(spriteId);
        if(!it.id)
            return NULL;

        int32_t subType = -1;
        if(it.isFluidContainer() && index < int32_t(sizeof(reverseFluidMap) / sizeof(int8_t)))
            subType = reverseFluidMap[index];

        return findItemOfType(player, it.id, true, subType);
    }

    return player->getInventoryItem((slots_t)static_cast<uint8_t>(pos.y));
}

void Game::internalGetPosition(Item* item, Position& pos, int16_t& stackpos)
{
    pos.x = pos.y = pos.z = stackpos = 0;
    if(Cylinder* topParent = item->getTopParent())
    {
        if(Player* player = dynamic_cast<Player*>(topParent))
        {
            pos.x = 0xFFFF;

            Container* container = dynamic_cast<Container*>(item->getParent());
            if(container)
            {
                pos.y = ((uint16_t) ((uint16_t)0x40) | ((uint16_t)player->getContainerID(container)) );
                pos.z = container->__getIndexOfThing(item);
                stackpos = pos.z;
            }
            else
            {
                pos.y = player->__getIndexOfThing(item);
                stackpos = pos.y;
            }
        }
        else if(Tile* tile = topParent->getTile())
        {
            pos = tile->getPosition();
            stackpos = tile->__getIndexOfThing(item);
        }
    }
}

Creature* Game::getCreatureByID(uint32_t id)
{
    if(!id)
        return NULL;

    AutoList<Creature>::iterator it = autoList.find(id);
    if(it != autoList.end() && !it->second->isRemoved())
        return it->second;

    return NULL; //just in case the player doesnt exist
}

Player* Game::getPlayerByID(uint32_t id)
{
    if(!id)
        return NULL;

    AutoList<Player>::iterator it = Player::autoList.find(id);
    if(it != Player::autoList.end() && !it->second->isRemoved())
        return it->second;

    return NULL; //just in case the player doesnt exist
}

Creature* Game::getCreatureByName(std::string s)
{
    if(s.empty())
        return NULL;

    toLowerCaseString(s);
    for(AutoList<Creature>::iterator it = autoList.begin(); it != autoList.end(); ++it)
    {
        if(!it->second->isRemoved() && asLowerCaseString(it->second->getName()) == s)
            return it->second;
    }

    return NULL; //just in case the creature doesnt exist
}

Player* Game::getPlayerByName(std::string s)
{
    if(s.empty())
        return NULL;

    toLowerCaseString(s);
    for(AutoList<Player>::iterator it = Player::autoList.begin(); it != Player::autoList.end(); ++it)
    {
        if(!it->second->isRemoved() && asLowerCaseString(it->second->getName()) == s)
            return it->second;
    }

    return NULL;
}

Player* Game::getPlayerByNameEx(const std::string& s)
{
    Player* player = getPlayerByName(s);
    if(player)
        return player;

    std::string name = s;
    if(!IOLoginData::getInstance()->playerExists(name))
        return NULL;

    player = new Player(name, NULL);
    if(IOLoginData::getInstance()->loadPlayer(player, name))
        return player;

#ifdef __DEBUG__
    std::clog << "[Failure - Game::getPlayerByNameEx] Cannot load player: " << name << std::endl;
#endif
    delete player;
    return NULL;
}

Player* Game::getPlayerByGuid(uint32_t guid)
{
    if(!guid)
        return NULL;

    for(AutoList<Player>::iterator it = Player::autoList.begin(); it != Player::autoList.end(); ++it)
    {
        if(!it->second->isRemoved() && it->second->getGUID() == guid)
            return it->second;
    }

    return NULL;
}

Player* Game::getPlayerByGuidEx(uint32_t guid)
{
    Player* player = getPlayerByGuid(guid);
    if(player)
        return player;

    std::string name;
    if(!IOLoginData::getInstance()->getNameByGuid(guid, name))
        return NULL;

    player = new Player(name, NULL);
    if(IOLoginData::getInstance()->loadPlayer(player, name))
        return player;

#ifdef __DEBUG__
    std::clog << "[Failure - Game::getPlayerByGuidEx] Cannot load player: " << name << std::endl;
#endif
    delete player;
    return NULL;
}

ReturnValue Game::getPlayerByNameWildcard(std::string s, Player*& player)
{
    player = NULL;
    if(s.empty())
        return RET_PLAYERWITHTHISNAMEISNOTONLINE;

    char tmp = *s.rbegin();
    if(tmp != '~' && tmp != '*')
    {
        player = getPlayerByName(s);
        if(!player)
            return RET_PLAYERWITHTHISNAMEISNOTONLINE;

        return RET_NOERROR;
    }

    Player* last = NULL;
    s = s.substr(0, s.length() - 1);

    toLowerCaseString(s);
    for(AutoList<Player>::iterator it = Player::autoList.begin(); it != Player::autoList.end(); ++it)
    {
        if(it->second->isRemoved())
            continue;

        std::string name = asLowerCaseString(it->second->getName());
        if(name.substr(0, s.length()) != s)
            continue;

        if(last)
            return RET_NAMEISTOOAMBIGUOUS;

        last = it->second;
    }

    if(!last)
        return RET_PLAYERWITHTHISNAMEISNOTONLINE;

    player = last;
    return RET_NOERROR;
}

Player* Game::getPlayerByAccount(uint32_t acc)
{
    for(AutoList<Player>::iterator it = Player::autoList.begin(); it != Player::autoList.end(); ++it)
    {
        if(!it->second->isRemoved() && it->second->getAccount() == acc)
            return it->second;
    }

    return NULL;
}

PlayerVector Game::getPlayersByName(std::string s)
{
    toLowerCaseString(s);
    PlayerVector players;
    for(AutoList<Player>::iterator it = Player::autoList.begin(); it != Player::autoList.end(); ++it)
    {
        if(!it->second->isRemoved() && asLowerCaseString(it->second->getName()) == s)
            players.push_back(it->second);
    }

    return players;
}

PlayerVector Game::getPlayersByAccount(uint32_t acc)
{
    PlayerVector players;
    for(AutoList<Player>::iterator it = Player::autoList.begin(); it != Player::autoList.end(); ++it)
    {
        if(!it->second->isRemoved() && it->second->getAccount() == acc)
            players.push_back(it->second);
    }

    return players;
}

PlayerVector Game::getPlayersByIP(uint32_t ip, uint32_t mask)
{
    PlayerVector players;
    for(AutoList<Player>::iterator it = Player::autoList.begin(); it != Player::autoList.end(); ++it)
    {
        if(!it->second->isRemoved() && (it->second->getIP() & mask) == (ip & mask))
            players.push_back(it->second);
    }

    return players;
}

bool Game::internalPlaceCreature(Creature* creature, const Position& pos, bool extendedPos /*= false*/, bool forced /*= false*/)
{
    if(creature->getParent())
        return false;

    if(!map->placeCreature(pos, creature, extendedPos, forced))
        return false;

    creature->addRef();
    creature->setID();

    autoList[creature->getID()] = creature;
    creature->addList();
    return true;
}

bool Game::placeCreature(Creature* creature, const Position& pos, bool extendedPos /*= false*/, bool forced /*= false*/)
{
    Player* tmpPlayer = NULL;
    if((tmpPlayer = creature->getPlayer()) && !tmpPlayer->storedConditionList.empty())
    {
        for(ConditionList::iterator it = tmpPlayer->storedConditionList.begin(); it != tmpPlayer->storedConditionList.end(); ++it)
        {
            if((*it)->getType() == CONDITION_MUTED && ((*it)->getTicks() - (
                (time(NULL) - tmpPlayer->getLastLogout()) * 1000)) <= 0)
                continue;

            tmpPlayer->addCondition(*it);
        }

        tmpPlayer->storedConditionList.clear();
    }

    if(!internalPlaceCreature(creature, pos, extendedPos, forced))
        return false;

    SpectatorVec::iterator it;
    SpectatorVec list;

    getSpectators(list, creature->getPosition(), false, true);
    for(it = list.begin(); it != list.end(); ++it)
    {
        if((tmpPlayer = (*it)->getPlayer()))
            tmpPlayer->sendCreatureAppear(creature);
    }

    for(it = list.begin(); it != list.end(); ++it)
        (*it)->onCreatureAppear(creature);

    creature->setLastPosition(pos);
    creature->getParent()->postAddNotification(NULL, creature, NULL, creature->getParent()->__getIndexOfThing(creature));
    addCreatureCheck(creature);

    creature->onPlacedCreature();
    return true;
}

ReturnValue Game::placeSummon(Creature* creature, const std::string& name)
{
    Monster* monster = Monster::createMonster(name);
    if(!monster)
        return RET_NOTPOSSIBLE;

    // Place the monster
    creature->addSummon(monster);
    if(placeCreature(monster, creature->getPosition(), true))
        return RET_NOERROR;

    creature->removeSummon(monster);
    return RET_NOTENOUGHROOM;
}

bool Game::removeCreature(Creature* creature, bool isLogout /*= true*/)
{
    if(creature->isRemoved())
        return false;

    Tile* tile = creature->getTile();
    SpectatorVec list;

    SpectatorVec::iterator it;
    getSpectators(list, tile->getPosition(), false, true);

    Player* player = NULL;
    std::vector<uint32_t> oldStackPosVector;
    for(it = list.begin(); it != list.end(); ++it)
    {
        if((player = (*it)->getPlayer()) && player->canSeeCreature(creature))
            oldStackPosVector.push_back(tile->getClientIndexOfThing(player, creature));
    }

    int32_t oldIndex = tile->__getIndexOfThing(creature);
    if(!map->removeCreature(creature))
        return false;

    //send to client
    uint32_t i = 0;
    for(it = list.begin(); it != list.end(); ++it)
    {
        if(!(player = (*it)->getPlayer()) || !player->canSeeCreature(creature))
            continue;

        player->sendCreatureDisappear(creature, oldStackPosVector);
        ++i;
    }

    //event method
    for(it = list.begin(); it != list.end(); ++it)
        (*it)->onCreatureDisappear(creature, isLogout);

    creature->getParent()->postRemoveNotification(NULL, creature, NULL, oldIndex, true);
    creature->onRemovedCreature();

    autoList.erase(creature->getID());
    freeThing(creature);

    removeCreatureCheck(creature);
    for(std::list<Creature*>::iterator it = creature->summons.begin(); it != creature->summons.end(); ++it)
        removeCreature(*it);

    return true;
}

bool Game::playerMoveThing(uint32_t playerId, const Position& fromPos,
    uint16_t spriteId, int16_t fromStackpos, const Position& toPos, uint8_t count)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    uint8_t fromIndex = 0;
    if(fromPos.x == 0xFFFF)
    {
        if(fromPos.y & 0x40)
            fromIndex = static_cast<uint8_t>(fromPos.z);
        else
            fromIndex = static_cast<uint8_t>(fromPos.y);
    }
    else
        fromIndex = fromStackpos;

    Thing* thing = internalGetThing(player, fromPos, fromIndex, spriteId, STACKPOS_MOVE);
    Cylinder* toCylinder = internalGetCylinder(player, toPos);
    if(!thing || !toCylinder)
    {
        player->sendCancelMessage(RET_NOTPOSSIBLE);
        return false;
    }

    if(Creature* movingCreature = thing->getCreature())
    {
        uint32_t delay = g_config.getNumber(ConfigManager::PUSH_CREATURE_DELAY);
        if(Position::areInRange<1,1,0>(movingCreature->getPosition(), player->getPosition()) && delay > 0
            && !player->hasCustomFlag(PlayerCustomFlag_CanThrowAnywhere))
        {
            SchedulerTask* task = createSchedulerTask(delay, boost::bind(&Game::playerMoveCreature, this,
                player->getID(), movingCreature->getID(), movingCreature->getPosition(), toCylinder->getPosition()));
            player->setNextActionTask(task);
        }
        else
            playerMoveCreature(playerId, movingCreature->getID(), movingCreature->getPosition(), toCylinder->getPosition());
    }
    else if(thing->getItem())
        playerMoveItem(playerId, fromPos, spriteId, fromStackpos, toPos, count);

    return true;
}

bool Game::playerMoveCreature(uint32_t playerId, uint32_t movingCreatureId,
    const Position& movingCreaturePos, const Position& toPos)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved() || player->hasFlag(PlayerFlag_CannotMoveCreatures))
        return false;

    if(!player->canDoAction())
    {
        uint32_t delay = player->getNextActionTime();
        SchedulerTask* task = createSchedulerTask(delay, boost::bind(&Game::playerMoveCreature,
            this, playerId, movingCreatureId, movingCreaturePos, toPos));

        player->setNextActionTask(task);
        return false;
    }

    Creature* movingCreature = getCreatureByID(movingCreatureId);
    if(!movingCreature || movingCreature->isRemoved() || movingCreature->getNoMove())
        return false;

    player->setNextActionTask(NULL);
    if(!Position::areInRange<1,1,0>(movingCreaturePos, player->getPosition()) && !player->hasCustomFlag(PlayerCustomFlag_CanMoveFromFar))
    {
        //need to walk to the creature first before moving it
        std::list<Direction> listDir;
        if(getPathToEx(player, movingCreaturePos, listDir, 0, 1, true, true))
        {
            Dispatcher::getInstance().addTask(createTask(boost::bind(&Game::playerAutoWalk,
                this, player->getID(), listDir)));
            SchedulerTask* task = createSchedulerTask(std::max((int32_t)SCHEDULER_MINTICKS, player->getStepDuration()),
                boost::bind(&Game::playerMoveCreature, this, playerId, movingCreatureId, movingCreaturePos, toPos));

            player->setNextWalkActionTask(task);
            return true;
        }

        player->sendCancelMessage(RET_THEREISNOWAY);
        return false;
    }

    Tile* toTile = map->getTile(toPos);
    if(!toTile)
    {
        player->sendCancelMessage(RET_NOTPOSSIBLE);
        return false;
    }

    if((!movingCreature->isPushable() && !player->hasFlag(PlayerFlag_CanPushAllCreatures)) || !player->canSeeCreature(movingCreature))
    {
        player->sendCancelMessage(RET_NOTMOVEABLE);
        return false;
    }

    //check throw distance
    const Position& pos = movingCreature->getPosition();
    if(!player->hasCustomFlag(PlayerCustomFlag_CanThrowAnywhere) && ((std::abs(pos.x - toPos.x) > movingCreature->getThrowRange()) ||
        (std::abs(pos.y - toPos.y) > movingCreature->getThrowRange()) || (std::abs(pos.z - toPos.z) * 4 > movingCreature->getThrowRange())))
    {
        player->sendCancelMessage(RET_DESTINATIONOUTOFREACH);
        return false;
    }

    if(player != movingCreature)
    {
        if(toTile->hasProperty(BLOCKPATH))
        {
            player->sendCancelMessage(RET_NOTENOUGHROOM);
            return false;
        }

        if((movingCreature->getZone() == ZONE_PROTECTION || movingCreature->getZone() == ZONE_OPTIONAL)
            && !toTile->hasFlag(TILESTATE_OPTIONALZONE) && !toTile->hasFlag(TILESTATE_PROTECTIONZONE)
            && !player->hasFlag(PlayerFlag_IgnoreProtectionZone))
        {
            player->sendCancelMessage(RET_NOTPOSSIBLE);
            return false;
        }

        if(!player->hasFlag(PlayerFlag_CanPushAllCreatures))
        {
            if(toTile->getCreatures() && !toTile->getCreatures()->empty())
            {
                player->sendCancelMessage(RET_NOTPOSSIBLE);
                return false;
            }

            uint32_t protectionLevel = g_config.getNumber(ConfigManager::PROTECTION_LEVEL);
            if(player->getLevel() < protectionLevel && player->getVocation()->isAttackable())
            {
                Player* movingPlayer = movingCreature->getPlayer();
                if(movingPlayer && movingPlayer->getLevel() >= protectionLevel
                    && movingPlayer->getVocation()->isAttackable())
                {
                    player->sendCancelMessage(RET_PLAYERISNOTREACHABLE);
                    return false;
                }
            }
        }
    }

    bool deny = false;
    CreatureEventList pushEvents = player->getCreatureEvents(CREATURE_EVENT_PUSH);
    for(CreatureEventList::iterator it = pushEvents.begin(); it != pushEvents.end(); ++it)
    {
        if(!(*it)->executePush(player, movingCreature) && !deny)
            deny = true;
    }

    if(deny)
        return false;

    ReturnValue ret = internalMoveCreature(player, movingCreature, movingCreature->getTile(), toTile);
    if(ret != RET_NOERROR)
    {
        if(!player->hasCustomFlag(PlayerCustomFlag_CanMoveFromFar) || !player->hasCustomFlag(PlayerCustomFlag_CanMoveAnywhere))
        {
            player->sendCancelMessage(ret);
            return false;
        }

        if(!toTile->ground)
        {
            player->sendCancelMessage(RET_NOTPOSSIBLE);
            return true;
        }

        internalTeleport(movingCreature, toTile->getPosition(), false);
        return true;
    }

    if(Player* movingPlayer = movingCreature->getPlayer())
    {
        uint64_t delay = OTSYS_TIME() + movingPlayer->getStepDuration();
        if(delay > movingPlayer->getNextActionTime())
            movingPlayer->setNextAction(delay);
    }

    return true;
}

ReturnValue Game::internalMoveCreature(Creature* creature, Direction direction, uint32_t flags/* = 0*/)
{
    const Position& currentPos = creature->getPosition();
    Cylinder* fromTile = creature->getTile();
    Cylinder* toTile = NULL;

    Position destPos = getNextPosition(direction, currentPos);
    if(direction < SOUTHWEST && creature->getPlayer())
    {
        Tile* tmpTile = NULL;
        if(currentPos.z != 8 && creature->getTile()->hasHeight(3)) //try go up
        {
            if((!(tmpTile = map->getTile(Position(currentPos.x, currentPos.y, currentPos.z - 1)))
                || (!tmpTile->ground && !tmpTile->hasProperty(BLOCKSOLID))) &&
                (tmpTile = map->getTile(Position(destPos.x, destPos.y, destPos.z - 1)))
                && tmpTile->ground && !tmpTile->hasProperty(BLOCKSOLID))
            {
                flags = flags | FLAG_IGNOREBLOCKITEM | FLAG_IGNOREBLOCKCREATURE;
                destPos.z--;
            }
        }
        else if(currentPos.z != 7 && (!(tmpTile = map->getTile(destPos)) || (!tmpTile->ground &&
            !tmpTile->hasProperty(BLOCKSOLID))) && (tmpTile = map->getTile(Position(
            destPos.x, destPos.y, destPos.z + 1))) && tmpTile->hasHeight(3)) //try go down
        {
            flags = flags | FLAG_IGNOREBLOCKITEM | FLAG_IGNOREBLOCKCREATURE;
            destPos.z++;
        }
    }

    ReturnValue ret = RET_NOTPOSSIBLE;
    if((toTile = map->getTile(destPos)))
        ret = internalMoveCreature(NULL, creature, fromTile, toTile, flags);

    if(ret == RET_NOERROR)
        return ret;

    Player* player = creature->getPlayer();
    if(!player)
        return ret;

    player->sendCancelMessage(ret);
    player->sendCancelWalk();
    return ret;
}

ReturnValue Game::internalMoveCreature(Creature* actor, Creature* creature, Cylinder* fromCylinder,
    Cylinder* toCylinder, uint32_t flags/* = 0*/, bool forceTeleport/* = false*/)
{
    //check if we can move the creature to the destination
    ReturnValue ret = toCylinder->__queryAdd(0, creature, 1, flags);
    if(ret != RET_NOERROR)
        return ret;

    fromCylinder->getTile()->moveCreature(actor, creature, toCylinder, forceTeleport);
    if(creature->getParent() != toCylinder)
        return RET_NOERROR;

    Item* toItem = NULL;
    Cylinder* subCylinder = NULL;

    int32_t n = 0, tmp = 0;
    while((subCylinder = toCylinder->__queryDestination(tmp, creature, &toItem, flags)) != toCylinder)
    {
        toCylinder->getTile()->moveCreature(actor, creature, subCylinder);
        if(creature->getParent() != subCylinder) //could happen if a script move the creature
             break;

        toCylinder = subCylinder;
        flags = 0;
        if(++n >= MAP_MAX_LAYERS) //to prevent infinite loop
            break;
    }

    return RET_NOERROR;
}

bool Game::playerMoveItem(uint32_t playerId, const Position& fromPos,
    uint16_t spriteId, int16_t fromStackpos, const Position& toPos, uint8_t count)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved() || player->hasFlag(PlayerFlag_CannotMoveItems))
        return false;

    if(!player->canDoAction())
    {
        uint32_t delay = player->getNextActionTime();
        SchedulerTask* task = createSchedulerTask(delay, boost::bind(&Game::playerMoveItem, this,
            playerId, fromPos, spriteId, fromStackpos, toPos, count));

        player->setNextActionTask(task);
        return false;
    }

    player->setNextActionTask(NULL);
    Cylinder* fromCylinder = internalGetCylinder(player, fromPos);

    uint8_t fromIndex = 0;
    if(fromPos.x == 0xFFFF)
    {
        if(fromPos.y & 0x40)
            fromIndex = static_cast<uint8_t>(fromPos.z);
        else
            fromIndex = static_cast<uint8_t>(fromPos.y);
    }
    else
        fromIndex = fromStackpos;

    Thing* thing = internalGetThing(player, fromPos, fromIndex, spriteId, STACKPOS_MOVE);
    if(!thing || !thing->getItem())
    {
        player->sendCancelMessage(RET_NOTPOSSIBLE);
        return false;
    }

    Item* item = thing->getItem();
    Cylinder* toCylinder = internalGetCylinder(player, toPos);

    uint8_t toIndex = 0;
    if(toPos.x == 0xFFFF)
    {
        if(toPos.y & 0x40)
            toIndex = static_cast<uint8_t>(toPos.z);
        else
            toIndex = static_cast<uint8_t>(toPos.y);
    }

    if(!fromCylinder || !toCylinder || !item || item->getClientID() != spriteId)
    {
        player->sendCancelMessage(RET_NOTPOSSIBLE);
        return false;
    }

    if(!player->hasCustomFlag(PlayerCustomFlag_CanPushAllItems) && (!item->isPushable() || (item->isLoadedFromMap() &&
        (item->getUniqueId() != 0 || (item->getActionId() != 0 && item->getContainer())))))
    {
        player->sendCancelMessage(RET_NOTMOVEABLE);
        return false;
    }

    const Position &mapToPos = toCylinder->getTile()->getPosition(), &playerPos = player->getPosition(),
        &mapFromPos = fromCylinder->getTile()->getPosition();
    if(playerPos.z > mapFromPos.z && !player->hasCustomFlag(PlayerCustomFlag_CanThrowAnywhere))
    {
        player->sendCancelMessage(RET_FIRSTGOUPSTAIRS);
        return false;
    }

    if(playerPos.z < mapFromPos.z && !player->hasCustomFlag(PlayerCustomFlag_CanThrowAnywhere))
    {
        player->sendCancelMessage(RET_FIRSTGODOWNSTAIRS);
        return false;
    }

    if(!Position::areInRange<1,1,0>(playerPos, mapFromPos) && !player->hasCustomFlag(PlayerCustomFlag_CanMoveFromFar))
    {
        //need to walk to the item first before using it
        std::list<Direction> listDir;
        if(getPathToEx(player, item->getPosition(), listDir, 0, 1, true, true))
        {
            Dispatcher::getInstance().addTask(createTask(boost::bind(&Game::playerAutoWalk,
                this, player->getID(), listDir)));
            SchedulerTask* task = createSchedulerTask(std::max((int32_t)SCHEDULER_MINTICKS, player->getStepDuration()),
                boost::bind(&Game::playerMoveItem, this, playerId, fromPos, spriteId, fromStackpos, toPos, count));

            player->setNextWalkActionTask(task);
            return true;
        }

        player->sendCancelMessage(RET_THEREISNOWAY);
        return false;
    }

    //hangable item specific code
    if(item->isHangable() && toCylinder->getTile()->hasProperty(SUPPORTHANGABLE))
    {
        //destination supports hangable objects so need to move there first
        if(toCylinder->getTile()->hasProperty(ISVERTICAL))
        {
            if(player->getPosition().x + 1 == mapToPos.x)
            {
                player->sendCancelMessage(RET_NOTPOSSIBLE);
                return false;
            }
        }
        else if(toCylinder->getTile()->hasProperty(ISHORIZONTAL))
        {
            if(player->getPosition().y + 1 == mapToPos.y)
            {
                player->sendCancelMessage(RET_NOTPOSSIBLE);
                return false;
            }
        }

        if(!Position::areInRange<1,1,0>(playerPos, mapToPos) && !player->hasCustomFlag(PlayerCustomFlag_CanMoveFromFar))
        {
            Position walkPos = mapToPos;
            if(toCylinder->getTile()->hasProperty(ISVERTICAL))
                walkPos.x -= -1;

            if(toCylinder->getTile()->hasProperty(ISHORIZONTAL))
                walkPos.y -= -1;

            Position itemPos = fromPos;
            int16_t itemStackpos = fromStackpos;
            if(fromPos.x != 0xFFFF && Position::areInRange<1,1,0>(mapFromPos, player->getPosition())
                && !Position::areInRange<1,1,0>(mapFromPos, walkPos))
            {
                //need to pickup the item first
                Item* moveItem = NULL;
                ReturnValue ret = internalMoveItem(player, fromCylinder, player, INDEX_WHEREEVER, item, count, &moveItem);
                if(ret != RET_NOERROR)
                {
                    player->sendCancelMessage(ret);
                    return false;
                }

                //changing the position since its now in the inventory of the player
                internalGetPosition(moveItem, itemPos, itemStackpos);
            }

            std::list<Direction> listDir;
            if(map->getPathTo(player, walkPos, listDir))
            {
                Dispatcher::getInstance().addTask(createTask(boost::bind(&Game::playerAutoWalk,
                    this, player->getID(), listDir)));
                SchedulerTask* task = createSchedulerTask(std::max((int32_t)SCHEDULER_MINTICKS, player->getStepDuration()),
                    boost::bind(&Game::playerMoveItem, this, playerId, itemPos, spriteId, itemStackpos, toPos, count));

                player->setNextWalkActionTask(task);
                return true;
            }

            player->sendCancelMessage(RET_THEREISNOWAY);
            return false;
        }
    }

    if(!player->hasCustomFlag(PlayerCustomFlag_CanThrowAnywhere))
    {
        if((std::abs(playerPos.x - mapToPos.x) > item->getThrowRange()) ||
            (std::abs(playerPos.y - mapToPos.y) > item->getThrowRange()) ||
            (std::abs(mapFromPos.z - mapToPos.z) * 4 > item->getThrowRange()))
        {
            player->sendCancelMessage(RET_DESTINATIONOUTOFREACH);
            return false;
        }
    }

    if(!canThrowObjectTo(mapFromPos, mapToPos) && !player->hasCustomFlag(PlayerCustomFlag_CanThrowAnywhere))
    {
        player->sendCancelMessage(RET_CANNOTTHROW);
        return false;
    }

    ReturnValue ret = internalMoveItem(player, fromCylinder, toCylinder, toIndex, item, count, NULL);
    if(ret == RET_NOERROR)
        return true;

    player->sendCancelMessage(ret);
    return false;
}

ReturnValue Game::internalMoveItem(Creature* actor, Cylinder* fromCylinder, Cylinder* toCylinder,
    int32_t index, Item* item, uint32_t count, Item** _moveItem, uint32_t flags /*= 0*/)
{
    if(!toCylinder)
        return RET_NOTPOSSIBLE;

    if(!item->getParent())
    {
        assert(fromCylinder == item->getParent());
        return internalAddItem(actor, toCylinder, item, INDEX_WHEREEVER, FLAG_NOLIMIT);
    }

    Item* toItem = NULL;
    Cylinder* subCylinder = NULL;

    int32_t floor = 0;
    while((subCylinder = toCylinder->__queryDestination(index, item, &toItem, flags)) != toCylinder)
    {
        toCylinder = subCylinder;
        flags = 0;
        //to prevent infinite loop
        if(++floor >= MAP_MAX_LAYERS)
            break;
    }

    //destination is the same as the source?
    if(item == toItem)
        return RET_NOERROR; //silently ignore move

    //check if we can add this item
    ReturnValue ret = toCylinder->__queryAdd(index, item, count, flags);
    if(ret == RET_NEEDEXCHANGE)
    {
        //check if we can add it to source cylinder
        int32_t fromIndex = fromCylinder->__getIndexOfThing(item);

        ret = fromCylinder->__queryAdd(fromIndex, toItem, toItem->getItemCount(), 0);
        if(ret == RET_NOERROR)
        {
            //check how much we can move
            uint32_t maxExchangeQueryCount = 0;
            ReturnValue retExchangeMaxCount = fromCylinder->__queryMaxCount(-1, toItem, toItem->getItemCount(), maxExchangeQueryCount, 0);

            if(retExchangeMaxCount != RET_NOERROR && maxExchangeQueryCount == 0)
                return retExchangeMaxCount;

            if((toCylinder->__queryRemove(toItem, toItem->getItemCount(), flags) == RET_NOERROR) && ret == RET_NOERROR)
            {
                int32_t oldToItemIndex = toCylinder->__getIndexOfThing(toItem);
                toCylinder->__removeThing(toItem, toItem->getItemCount());

                fromCylinder->__addThing(actor, toItem);
                if(oldToItemIndex != -1)
                    toCylinder->postRemoveNotification(actor, toItem, fromCylinder, oldToItemIndex, true);

                int32_t newToItemIndex = fromCylinder->__getIndexOfThing(toItem);
                if(newToItemIndex != -1)
                    fromCylinder->postAddNotification(actor, toItem, toCylinder, newToItemIndex);

                ret = toCylinder->__queryAdd(index, item, count, flags);
                toItem = NULL;
            }
        }
    }

    if(ret != RET_NOERROR)
        return ret;

    //check how much we can move
    uint32_t maxQueryCount = 0;
    ReturnValue retMaxCount = toCylinder->__queryMaxCount(index, item, count, maxQueryCount, flags);
    if(retMaxCount != RET_NOERROR && !maxQueryCount)
        return retMaxCount;

    uint32_t m = maxQueryCount, n = 0;
    if(item->isStackable())
        m = std::min((uint32_t)count, m);

    Item* moveItem = item;
    //check if we can remove this item
    ret = fromCylinder->__queryRemove(item, m, flags);
    if(ret != RET_NOERROR)
        return ret;

    //remove the item
    int32_t itemIndex = fromCylinder->__getIndexOfThing(item);
    fromCylinder->__removeThing(item, m);

    bool isCompleteRemoval = item->isRemoved();
    Item* updateItem = NULL;
    //update item(s)
    if(item->isStackable())
    {
        if(toItem && toItem->getID() == item->getID())
        {
            n = std::min((uint32_t)100 - toItem->getItemCount(), m);
            toCylinder->__updateThing(toItem, toItem->getID(), toItem->getItemCount() + n);
            updateItem = toItem;
        }

        if(m - n > 0)
            moveItem = Item::CreateItem(item->getID(), m - n);
        else
            moveItem = NULL;

        if(item->isRemoved())
            freeThing(item);
    }

    //add item
    if(moveItem /*m - n > 0*/)
        toCylinder->__addThing(actor, index, moveItem);

    if(itemIndex != -1)
        fromCylinder->postRemoveNotification(actor, item, toCylinder, itemIndex, isCompleteRemoval);

    if(moveItem)
    {
        int32_t moveItemIndex = toCylinder->__getIndexOfThing(moveItem);
        if(moveItemIndex != -1)
            toCylinder->postAddNotification(actor, moveItem, fromCylinder, moveItemIndex);
    }

    if(updateItem)
    {
        int32_t updateItemIndex = toCylinder->__getIndexOfThing(updateItem);
        if(updateItemIndex != -1)
            toCylinder->postAddNotification(actor, updateItem, fromCylinder, updateItemIndex);
    }

    if(_moveItem)
    {
        if(moveItem)
            *_moveItem = moveItem;
        else
            *_moveItem = item;
    }

    //we could not move all, inform the player
    if(item->isStackable() && maxQueryCount < count)
        return retMaxCount;

    return ret;
}

ReturnValue Game::internalAddItem(Creature* actor, Cylinder* toCylinder, Item* item, int32_t index /*= INDEX_WHEREEVER*/,
    uint32_t flags /*= 0*/, bool test /*= false*/)
{
    uint32_t remainderCount = 0;
    return internalAddItem(actor, toCylinder, item, index, flags, test, remainderCount);
}

ReturnValue Game::internalAddItem(Creature* actor, Cylinder* toCylinder, Item* item, int32_t index,
    uint32_t flags, bool test, uint32_t& remainderCount)
{
    remainderCount = 0;
    if(!toCylinder || !item)
        return RET_NOTPOSSIBLE;

    Cylinder* destCylinder = toCylinder;
    Item* toItem = NULL;
    toCylinder = toCylinder->__queryDestination(index, item, &toItem, flags);

    //check if we can add this item
    ReturnValue ret = toCylinder->__queryAdd(index, item, item->getItemCount(), flags);
    if(ret != RET_NOERROR)
        return ret;

    uint32_t maxQueryCount = 0;
    ret = destCylinder->__queryMaxCount(INDEX_WHEREEVER, item, item->getItemCount(), maxQueryCount, flags);
    if(ret != RET_NOERROR)
        return ret;

    if(test)
        return RET_NOERROR;

    if(item->isStackable() && toItem)
    {
        uint32_t m = std::min((uint32_t)item->getItemCount(), maxQueryCount), n = 0;
        if(toItem->getID() == item->getID())
        {
            n = std::min((uint32_t)100 - toItem->getItemCount(), m);
            toCylinder->__updateThing(toItem, toItem->getID(), toItem->getItemCount() + n);
        }

        if(m - n > 0)
        {
            if(m - n != item->getItemCount())
            {
                Item* remainderItem = Item::CreateItem(item->getID(), m - n);
                if(internalAddItem(NULL, destCylinder, remainderItem, INDEX_WHEREEVER, flags, false) != RET_NOERROR)
                {
                    freeThing(remainderItem);
                    remainderCount = m - n;
                }
            }
        }
        else if(item->getParent() != VirtualCylinder::virtualCylinder)
        {
            item->onRemoved();
            freeThing(item);
        }
    }
    else
    {
        toCylinder->__addThing(NULL, index, item);
        int32_t itemIndex = toCylinder->__getIndexOfThing(item);
        if(itemIndex != -1)
            toCylinder->postAddNotification(actor, item, NULL, itemIndex);
    }

    return RET_NOERROR;
}

ReturnValue Game::internalRemoveItem(Creature* actor, Item* item, int32_t count /*= -1*/, bool test /*= false*/, uint32_t flags /*= 0*/)
{
    Cylinder* cylinder = item->getParent();
    if(!cylinder)
        return RET_NOTPOSSIBLE;

    if(count == -1)
        count = item->getItemCount();

    //check if we can remove this item
    ReturnValue ret = cylinder->__queryRemove(item, count, flags | FLAG_IGNORENOTMOVEABLE);
    if(ret != RET_NOERROR)
        return ret;

    if(!item->canRemove())
        return RET_NOTPOSSIBLE;

    if(!test)
    {
        //remove the item
        int32_t index = cylinder->__getIndexOfThing(item);
        cylinder->__removeThing(item, count);

        bool isCompleteRemoval = false;
        if(item->isRemoved())
        {
            isCompleteRemoval = true;
            freeThing(item);
        }

        cylinder->postRemoveNotification(actor, item, NULL, index, isCompleteRemoval);
    }

    item->onRemoved();
    return RET_NOERROR;
}

ReturnValue Game::internalPlayerAddItem(Creature* actor, Player* player, Item* item,
    bool dropOnMap/* = true*/, slots_t slot/* = SLOT_WHEREEVER*/)
{
    uint32_t remainderCount = 0;
    ReturnValue ret = internalAddItem(actor, player, item, (int32_t)slot, 0, false, remainderCount);
    if(remainderCount > 0)
    {
        Item* remainderItem = Item::CreateItem(item->getID(), remainderCount);
        ReturnValue remainderRet = internalAddItem(actor, player->getTile(), remainderItem, INDEX_WHEREEVER, FLAG_NOLIMIT);
        if(remainderRet != RET_NOERROR)
            freeThing(remainderItem);
    }

    if(ret != RET_NOERROR && dropOnMap)
        ret = internalAddItem(actor, player->getTile(), item, (int32_t)slot, FLAG_NOLIMIT);

    return ret;
}

Item* Game::findItemOfType(Cylinder* cylinder, uint16_t itemId,
    bool depthSearch /*= true*/, int32_t subType /*= -1*/)
{
    if(!cylinder)
        return false;

    std::list<Container*> listContainer;
    Container* tmpContainer = NULL;

    Thing* thing = NULL;
    Item* item = NULL;
    for(int32_t i = cylinder->__getFirstIndex(); i < cylinder->__getLastIndex();)
    {
        if((thing = cylinder->__getThing(i)) && (item = thing->getItem()))
        {
            if(item->getID() == itemId && (subType == -1 || subType == item->getSubType()))
                return item;
            else
            {
                ++i;
                if(depthSearch && (tmpContainer = item->getContainer()))
                    listContainer.push_back(tmpContainer);
            }
        }
        else
            ++i;
    }

    while(listContainer.size() > 0)
    {
        Container* container = listContainer.front();
        listContainer.pop_front();
        for(int32_t i = 0; i < (int32_t)container->size();)
        {
            Item* item = container->getItem(i);
            if(item->getID() == itemId && (subType == -1 || subType == item->getSubType()))
                return item;
            else
            {
                ++i;
                if((tmpContainer = item->getContainer()))
                    listContainer.push_back(tmpContainer);
            }
        }
    }

    return NULL;
}

bool Game::removeItemOfType(Cylinder* cylinder, uint16_t itemId, int32_t count, int32_t subType /*= -1*/)
{
    if(!cylinder || ((int32_t)cylinder->__getItemTypeCount(itemId, subType) < count))
        return false;

    if(count <= 0)
        return true;

    std::list<Container*> listContainer;
    Container* tmpContainer = NULL;
    Item* item = NULL;

    Thing* thing = NULL;
    for(int32_t i = cylinder->__getFirstIndex(); i < cylinder->__getLastIndex() && count > 0;)
    {
        if((thing = cylinder->__getThing(i)) && (item = thing->getItem()))
        {
            if(item->getID() == itemId)
            {
                if(item->isStackable())
                {
                    if(item->getItemCount() > count)
                    {
                        internalRemoveItem(NULL, item, count);
                        count = 0;
                    }
                    else
                    {
                        count -= item->getItemCount();
                        internalRemoveItem(NULL, item);
                    }
                }
                else if(subType == -1 || subType == item->getSubType())
                {
                    --count;
                    internalRemoveItem(NULL, item);
                }
                else
                    ++i;
            }
            else
            {
                ++i;
                if((tmpContainer = item->getContainer()))
                    listContainer.push_back(tmpContainer);
            }
        }
        else
            ++i;
    }

    while(listContainer.size() > 0 && count > 0)
    {
        Container* container = listContainer.front();
        listContainer.pop_front();
        for(int32_t i = 0; i < (int32_t)container->size() && count > 0;)
        {
            Item* item = container->getItem(i);
            if(item->getID() == itemId)
            {
                if(item->isStackable())
                {
                    if(item->getItemCount() > count)
                    {
                        internalRemoveItem(NULL, item, count);
                        count = 0;
                    }
                    else
                    {
                        count-= item->getItemCount();
                        internalRemoveItem(NULL, item);
                    }
                }
                else if(subType == -1 || subType == item->getSubType())
                {
                    --count;
                    internalRemoveItem(NULL, item);
                }
                else
                    ++i;
            }
            else
            {
                ++i;
                if((tmpContainer = item->getContainer()))
                    listContainer.push_back(tmpContainer);
            }
        }
    }

    return (count == 0);
}

uint64_t Game::getMoney(const Cylinder* cylinder)
{
    if(!cylinder)
        return 0;

    std::list<Container*> listContainer;
    Container* tmpContainer = NULL;
    Item* item = NULL;

    Thing* thing = NULL;
    uint64_t moneyCount = 0;
    for(int32_t i = cylinder->__getFirstIndex(); i < cylinder->__getLastIndex(); ++i)
    {
        if(!(thing = cylinder->__getThing(i)) || !(item = thing->getItem()))
            continue;

        if((tmpContainer = item->getContainer()))
            listContainer.push_back(tmpContainer);
        else if(item->getWorth() != 0)
            moneyCount += item->getWorth();
    }

    while(listContainer.size() > 0)
    {
        Container* container = listContainer.front();
        listContainer.pop_front();
        for(ItemList::const_iterator it = container->getItems(); it != container->getEnd(); ++it)
        {
            item = *it;
            if((tmpContainer = item->getContainer()))
                listContainer.push_back(tmpContainer);
            else if(item->getWorth() != 0)
                moneyCount += item->getWorth();
        }
    }

    return moneyCount;
}

bool Game::removeMoney(Cylinder* cylinder, int64_t money, uint32_t flags /*= 0*/)
{
    if(!cylinder)
        return false;

    if(money <= 0)
        return true;

    typedef std::multimap<int32_t, Item*, std::less<int32_t> > MoneyMultiMap;
    MoneyMultiMap moneyMap;

    std::list<Container*> listContainer;
    Container* tmpContainer = NULL;

    Thing* thing = NULL;
    Item* item = NULL;

    int64_t moneyCount = 0;
    for(int32_t i = cylinder->__getFirstIndex(); i < cylinder->__getLastIndex() && money > 0; ++i)
    {
        if(!(thing = cylinder->__getThing(i)) || !(item = thing->getItem()))
            continue;

        if((tmpContainer = item->getContainer()))
            listContainer.push_back(tmpContainer);
        else if(item->getWorth() != 0)
        {
            moneyCount += item->getWorth();
            moneyMap.insert(std::make_pair(item->getWorth(), item));
        }
    }

    while(listContainer.size() > 0 && money > 0)
    {
        Container* container = listContainer.front();
        listContainer.pop_front();
        for(int32_t i = 0; i < (int32_t)container->size() && money > 0; i++)
        {
            Item* item = container->getItem(i);
            if((tmpContainer = item->getContainer()))
                listContainer.push_back(tmpContainer);
            else if(item->getWorth() != 0)
            {
                moneyCount += item->getWorth();
                moneyMap.insert(std::make_pair(item->getWorth(), item));
            }
        }
    }

    // Not enough money
    if(moneyCount < money)
        return false;

    for(MoneyMultiMap::iterator mit = moneyMap.begin(); mit != moneyMap.end() && money > 0; ++mit)
    {
        Item* item = mit->second;
        if(!item)
            continue;

        internalRemoveItem(NULL, item);
        if(mit->first > money)
        {
            // Remove a monetary value from an item
            addMoney(cylinder, (int64_t)(item->getWorth() - money), flags);
            money = 0;
        }
        else
            money -= mit->first;

        mit->second = NULL;
    }

    moneyMap.clear();
    return money == 0;
}

void Game::addMoney(Cylinder* cylinder, int64_t money, uint32_t flags /*= 0*/)
{
    IntegerMap moneyMap = Item::items.getMoneyMap();
    for(IntegerMap::reverse_iterator it = moneyMap.rbegin(); it != moneyMap.rend(); ++it)
    {
        int64_t tmp = money / it->first;
        money -= tmp * it->first;
        if(!tmp)
            continue;

        do
        {
            Item* remainItem = Item::CreateItem(it->second, std::min((int64_t)100, tmp));
            if(internalAddItem(NULL, cylinder, remainItem, INDEX_WHEREEVER, flags) != RET_NOERROR)
                internalAddItem(NULL, cylinder->getTile(), remainItem, INDEX_WHEREEVER, FLAG_NOLIMIT);

            tmp -= std::min((int64_t)100, tmp);
        }
        while(tmp > 0);
    }
}

Item* Game::transformItem(Item* item, uint16_t newId, int32_t newCount /*= -1*/)
{
    if(item->getID() == newId && (newCount == -1 || (newCount == item->getSubType() && newCount != 0)))
        return item;

    Cylinder* cylinder = item->getParent();
    if(!cylinder)
        return NULL;

    int32_t itemIndex = cylinder->__getIndexOfThing(item);
    if(itemIndex == -1)
    {
#ifdef __DEBUG__
        std::clog << "Error: transformItem, itemIndex == -1" << std::endl;
#endif
        return item;
    }

    if(!item->canTransform())
        return item;

    const ItemType& curType = Item::items[item->getID()];
    const ItemType& newType = Item::items[newId];
    if(curType.alwaysOnTop != newType.alwaysOnTop)
    {
        //This only occurs when you transform items on tiles from a downItem to a topItem (or vice versa)
        //Remove the old, and add the new
        ReturnValue ret = internalRemoveItem(NULL, item);
        if(ret != RET_NOERROR)
            return item;

        Item* newItem = NULL;
        if(newCount == -1)
            newItem = Item::CreateItem(newId);
        else
            newItem = Item::CreateItem(newId, newCount);

        if(!newItem)
            return NULL;

        newItem->copyAttributes(item);
        if(internalAddItem(NULL, cylinder, newItem, INDEX_WHEREEVER, FLAG_NOLIMIT) == RET_NOERROR)
            return newItem;

        delete newItem;
        return NULL;
    }

    if(curType.type == newType.type)
    {
        //Both items has the same type so we can safely change id/subtype
        if(!newCount && (item->isStackable() || item->hasCharges()))
        {
            if(!item->isStackable() && (!item->getDefaultDuration() || item->getDuration() <= 0))
            {
                int32_t tmpId = newId;
                if(curType.id == newType.id)
                    tmpId = curType.decayTo;

                if(tmpId != -1)
                {
                    item = transformItem(item, tmpId);
                    return item;
                }
            }

            internalRemoveItem(NULL, item);
            return NULL;
        }

        uint16_t itemId = item->getID();
        int32_t count = item->getSubType();

        cylinder->postRemoveNotification(NULL, item, cylinder, itemIndex, false);
        if(curType.id != newType.id)
        {
            itemId = newId;
            if(newType.group != curType.group)
                item->setDefaultSubtype();
        }

        if(newCount != -1 && newType.hasSubType())
            count = newCount;

        cylinder->__updateThing(item, itemId, count);
        cylinder->postAddNotification(NULL, item, cylinder, itemIndex);
        return item;
    }

    //Replacing the the old item with the new while maintaining the old position
    Item* newItem = NULL;
    if(newCount == -1)
        newItem = Item::CreateItem(newId);
    else
        newItem = Item::CreateItem(newId, newCount);

    if(!newItem)
    {
        #ifdef __DEBUG__
        std::clog << "Error: [Game::transformItem] Item of type " << item->getID() << " transforming into invalid type " << newId << std::endl;
        #endif
        return NULL;
    }

    cylinder->__replaceThing(itemIndex, newItem);
    cylinder->postAddNotification(NULL, newItem, cylinder, itemIndex);

    item->setParent(NULL);
    cylinder->postRemoveNotification(NULL, item, cylinder, itemIndex, true);

    freeThing(item);
    return newItem;
}

ReturnValue Game::internalTeleport(Thing* thing, const Position& newPos, bool forceTeleport, uint32_t flags/* = 0*/, bool fullTeleport/* = true*/)
{
    if(newPos == thing->getPosition())
        return RET_NOERROR;

    if(thing->isRemoved())
        return RET_NOTPOSSIBLE;

    if(Tile* toTile = map->getTile(newPos))
    {
        if(Creature* creature = thing->getCreature())
        {
            if(fullTeleport)
                return internalMoveCreature(NULL, creature, creature->getParent(), toTile, flags, forceTeleport);

            creature->getTile()->moveCreature(NULL, creature, toTile, forceTeleport);
            return RET_NOERROR;
        }

        if(Item* item = thing->getItem())
            return internalMoveItem(NULL, item->getParent(), toTile, INDEX_WHEREEVER, item, item->getItemCount(), NULL, flags);
    }

    return RET_NOTPOSSIBLE;
}

//Implementation of player invoked events
bool Game::playerMove(uint32_t playerId, Direction dir)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    player->setIdleTime(0);
    if(player->getNoMove())
    {
        player->sendCancelWalk();
        return false;
    }

    std::list<Direction> dirs;
    dirs.push_back(dir);
    return player->startAutoWalk(dirs);
}

bool Game::playerBroadcastMessage(Player* player, SpeakClasses type, const std::string& text)
{
    if(!player->hasFlag(PlayerFlag_CanBroadcast) || type < SPEAK_CLASS_FIRST || type > SPEAK_CLASS_LAST)
        return false;

    for(AutoList<Player>::iterator it = Player::autoList.begin(); it != Player::autoList.end(); ++it)
        it->second->sendCreatureSay(player, type, text);

    //TODO: event handling - onCreatureSay
    std::clog << "> " << player->getName() << " broadcasted: \"" << text << "\"." << std::endl;
    return true;
}

bool Game::playerCreatePrivateChannel(uint32_t playerId, ProtocolGame* pg) //CA
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved() || !player->isPremium())
        return false;

    ChatChannel* channel = NULL;
    if(pg != NULL && pg->getIsCast()) { //CA
        channel = g_chat.getPrivateChannel(player);
        if(channel) {
            pg->publicSendCreatePrivateChannel(channel->getId(), channel->getName());
            return true;
        } else
            pg->publicSendMessage(player, SPEAK_PRIVATE, "The cast owner disabled communication.");

        return false;
    } else {
        channel = g_chat.createChannel(player, 0xFFFF);
        if(!channel || !channel->addUser(player))
            return false;
    }

    for(AutoList<ProtocolGame>::const_iterator it = Player::cSpectators.begin(); it != Player::cSpectators.end(); ++it) if(it->second->getPlayer() == player)
        it->second->publicSendMessage(player, SPEAK_PRIVATE, "The cast owner turned communication on.");

    player->sendCreatePrivateChannel(channel->getId(), channel->getName());
    return true;
}

bool Game::playerChannelInvite(uint32_t playerId, const std::string& name)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    PrivateChatChannel* channel = g_chat.getPrivateChannel(player);
    if(!channel)
        return false;

    Player* invitePlayer = getPlayerByName(name);
    if(!invitePlayer)
        return false;

    channel->invitePlayer(player, invitePlayer);
    return true;
}

bool Game::playerChannelExclude(uint32_t playerId, const std::string& name)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    PrivateChatChannel* channel = g_chat.getPrivateChannel(player);
    if(!channel)
        return false;

    Player* excludePlayer = getPlayerByName(name);
    if(!excludePlayer)
        return false;

    channel->excludePlayer(player, excludePlayer);
    return true;
}

bool Game::playerRequestChannels(uint32_t playerId, ProtocolGame* pg) //CA
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    if(pg != NULL && pg->getIsCast())
        pg->publicSendChannelsDialog();
    else
        player->sendChannelsDialog();
        
    return true;
}

bool Game::playerOpenChannel(uint32_t playerId, uint16_t channelId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    ChatChannel* channel = g_chat.addUserToChannel(player, channelId);
    if(!channel)
    {
        #ifdef __DEBUG_CHAT__
        std::clog << "Game::playerOpenChannel - failed adding user to channel." << std::endl;
        #endif
        return false;
    }

    if(channel->getId() != CHANNEL_RVR)
        player->sendChannel(channel->getId(), channel->getName());
    else
        player->sendRuleViolationsChannel(channel->getId());

    return true;
}

bool Game::playerCloseChannel(uint32_t playerId, uint16_t channelId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    ChatChannel* channel = g_chat.getPrivateChannel(player); //CA
    if(channel && channel->getId() == channelId) {
        for(AutoList<ProtocolGame>::const_iterator it = Player::cSpectators.begin(); it != Player::cSpectators.end(); ++it) if(it->second->getPlayer() == player)
            it->second->publicSendMessage(player, SPEAK_PRIVATE, "The cast owner turned communication off.");
    }

    g_chat.removeUserFromChannel(player, channelId);
    return true;
}

bool Game::playerOpenPrivateChannel(uint32_t playerId, std::string& receiver)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    if(IOLoginData::getInstance()->playerExists(receiver))
        player->sendOpenPrivateChannel(receiver);
    else
        player->sendCancel("A player with this name does not exist.");

    return true;
}

bool Game::playerProcessRuleViolation(uint32_t playerId, const std::string& name)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    if(!player->hasFlag(PlayerFlag_CanAnswerRuleViolations))
        return false;

    Player* reporter = getPlayerByName(name);
    if(!reporter)
        return false;

    RuleViolationsMap::iterator it = ruleViolations.find(reporter->getID());
    if(it == ruleViolations.end())
        return false;

    RuleViolation& rvr = *it->second;
    if(!rvr.isOpen)
        return false;

    rvr.isOpen = false;
    rvr.gamemaster = player;
    if(ChatChannel* channel = g_chat.getChannelById(CHANNEL_RVR))
    {
        UsersMap tmpMap = channel->getUsers();
        for(UsersMap::iterator tit = tmpMap.begin(); tit != tmpMap.end(); ++tit)
            tit->second->sendRemoveReport(reporter->getName());
    }

    return true;
}

bool Game::playerCloseRuleViolation(uint32_t playerId, const std::string& name)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    Player* reporter = getPlayerByName(name);
    if(!reporter)
        return false;

    return closeRuleViolation(reporter);
}

bool Game::playerCancelRuleViolation(uint32_t playerId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    return cancelRuleViolation(player);
}

bool Game::playerCloseNpcChannel(uint32_t playerId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    SpectatorVec list;
    SpectatorVec::iterator it;

    getSpectators(list, player->getPosition());
    Npc* npc = NULL;
    for(it = list.begin(); it != list.end(); ++it)
    {
        if((npc = (*it)->getNpc()))
            npc->onPlayerCloseChannel(player);
    }

    return true;
}

bool Game::playerReceivePing(uint32_t playerId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    player->receivePing();
    return true;
}

bool Game::playerAutoWalk(uint32_t playerId, std::list<Direction>& listDir)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    player->setIdleTime(0);
    if(player->hasCondition(CONDITION_GAMEMASTER, GAMEMASTER_TELEPORT))
    {
        Position pos = player->getPosition();
        for(std::list<Direction>::iterator it = listDir.begin(); it != listDir.end(); ++it)
            pos = getNextPosition((*it), pos);

        pos = getClosestFreeTile(player, pos, true, false);
        if(!pos.x || !pos.y)
        {
            player->sendCancelWalk();
            return false;
        }

        internalCreatureTurn(player, getDirectionTo(player->getPosition(), pos, false));
        internalTeleport(player, pos, true);
        return true;
    }

    player->setNextWalkTask(NULL);
    return player->startAutoWalk(listDir);
}

bool Game::playerStopAutoWalk(uint32_t playerId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    player->stopWalk();
    return true;
}

bool Game::playerUseItemEx(uint32_t playerId, const Position& fromPos, int16_t fromStackpos, uint16_t fromSpriteId,
    const Position& toPos, int16_t toStackpos, uint16_t toSpriteId, bool isHotkey)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    if(isHotkey && !g_config.getBool(ConfigManager::AIMBOT_HOTKEY_ENABLED))
        return false;

    Thing* thing = internalGetThing(player, fromPos, fromStackpos, fromSpriteId, STACKPOS_USEITEM);
    if(!thing)
    {
        player->sendCancelMessage(RET_NOTPOSSIBLE);
        return false;
    }

    Item* item = thing->getItem();
    if(!item || !item->isUseable())
    {
        player->sendCancelMessage(RET_CANNOTUSETHISOBJECT);
        return false;
    }

    Position walkToPos = fromPos;
    ReturnValue ret = g_actions->canUse(player, fromPos);
    if(ret == RET_NOERROR)
    {
        ret = g_actions->canUse(player, toPos, item);
        if(ret == RET_TOOFARAWAY)
            walkToPos = toPos;
    }

    if(ret != RET_NOERROR)
    {
        if(ret == RET_TOOFARAWAY)
        {
            Position itemPos = fromPos;
            int16_t itemStackpos = fromStackpos;
            if(fromPos.x != 0xFFFF && toPos.x != 0xFFFF && Position::areInRange<1,1,0>(fromPos,
                player->getPosition()) && !Position::areInRange<1,1,0>(fromPos, toPos))
            {
                Item* moveItem = NULL;
                ReturnValue retTmp = internalMoveItem(player, item->getParent(), player,
                    INDEX_WHEREEVER, item, item->getItemCount(), &moveItem);
                if(retTmp != RET_NOERROR)
                {
                    player->sendCancelMessage(retTmp);
                    return false;
                }

                //changing the position since its now in the inventory of the player
                internalGetPosition(moveItem, itemPos, itemStackpos);
            }

            std::list<Direction> listDir;
            if(getPathToEx(player, walkToPos, listDir, 0, 1, true, true, 10))
            {
                Dispatcher::getInstance().addTask(createTask(boost::bind(&Game::playerAutoWalk,
                    this, player->getID(), listDir)));

                SchedulerTask* task = createSchedulerTask(400, boost::bind(&Game::playerUseItemEx, this,
                    playerId, itemPos, itemStackpos, fromSpriteId, toPos, toStackpos, toSpriteId, isHotkey));

                player->setNextWalkActionTask(task);
                return true;
            }

            ret = RET_THEREISNOWAY;
        }

        player->sendCancelMessage(ret);
        return false;
    }

    if(isHotkey)
        showHotkeyUseMessage(player, item);

    if(!player->canDoAction())
    {
        uint32_t delay = player->getNextActionTime();
        SchedulerTask* task = createSchedulerTask(delay, boost::bind(&Game::playerUseItemEx, this,
            playerId, fromPos, fromStackpos, fromSpriteId, toPos, toStackpos, toSpriteId, isHotkey));

        player->setNextActionTask(task);
        return false;
    }

    player->setIdleTime(0);
    player->setNextActionTask(NULL);
    return g_actions->useItemEx(player, fromPos, toPos, toStackpos, item, isHotkey);
}

bool Game::playerUseItem(uint32_t playerId, const Position& pos, int16_t stackpos,
    uint8_t index, uint16_t spriteId, bool isHotkey)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    if(isHotkey && !g_config.getBool(ConfigManager::AIMBOT_HOTKEY_ENABLED))
        return false;

    Thing* thing = internalGetThing(player, pos, stackpos, spriteId, STACKPOS_USEITEM);
    if(!thing)
    {
        player->sendCancelMessage(RET_NOTPOSSIBLE);
        return false;
    }

    Item* item = thing->getItem();
    if(!item || item->isUseable())
    {
        player->sendCancelMessage(RET_CANNOTUSETHISOBJECT);
        return false;
    }

    ReturnValue ret = g_actions->canUse(player, pos);
    if(ret != RET_NOERROR)
    {
        if(ret == RET_TOOFARAWAY)
        {
            std::list<Direction> listDir;
            if(getPathToEx(player, pos, listDir, 0, 1, true, true))
            {
                Dispatcher::getInstance().addTask(createTask(boost::bind(&Game::playerAutoWalk,
                    this, player->getID(), listDir)));

                SchedulerTask* task = createSchedulerTask(400, boost::bind(&Game::playerUseItem, this,
                    playerId, pos, stackpos, index, spriteId, isHotkey));

                player->setNextWalkActionTask(task);
                return true;
            }

            ret = RET_THEREISNOWAY;
        }

        player->sendCancelMessage(ret);
        return false;
    }

    if(isHotkey)
        showHotkeyUseMessage(player, item);

    if(!player->canDoAction())
    {
        uint32_t delay = player->getNextActionTime();
        SchedulerTask* task = createSchedulerTask(delay, boost::bind(&Game::playerUseItem, this,
            playerId, pos, stackpos, index, spriteId, isHotkey));

        player->setNextActionTask(task);
        return false;
    }

    player->setIdleTime(0);
    player->setNextActionTask(NULL);
    return g_actions->useItem(player, pos, index, item);
}

bool Game::playerUseBattleWindow(uint32_t playerId, const Position& fromPos, int16_t fromStackpos,
    uint32_t creatureId, uint16_t spriteId, bool isHotkey)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    Creature* creature = getCreatureByID(creatureId);
    if(!creature)
        return false;

    if(!Position::areInRange<7,5,0>(creature->getPosition(), player->getPosition()))
        return false;

    if(!g_config.getBool(ConfigManager::AIMBOT_HOTKEY_ENABLED) && (creature->getPlayer() || isHotkey))
    {
        player->sendCancelMessage(RET_DIRECTPLAYERSHOOT);
        return false;
    }

    Thing* thing = internalGetThing(player, fromPos, fromStackpos, spriteId, STACKPOS_USE);
    if(!thing)
    {
        player->sendCancelMessage(RET_NOTPOSSIBLE);
        return false;
    }

    Item* item = thing->getItem();
    if(!item || item->getClientID() != spriteId)
    {
        player->sendCancelMessage(RET_CANNOTUSETHISOBJECT);
        return false;
    }

    ReturnValue ret = g_actions->canUse(player, fromPos);
    if(ret != RET_NOERROR)
    {
        if(ret == RET_TOOFARAWAY)
        {
            std::list<Direction> listDir;
            if(getPathToEx(player, item->getPosition(), listDir, 0, 1, true, true))
            {
                Dispatcher::getInstance().addTask(createTask(boost::bind(&Game::playerAutoWalk,
                    this, player->getID(), listDir)));

                SchedulerTask* task = createSchedulerTask(400, boost::bind(&Game::playerUseBattleWindow, this,
                    playerId, fromPos, fromStackpos, creatureId, spriteId, isHotkey));

                player->setNextWalkActionTask(task);
                return true;
            }

            ret = RET_THEREISNOWAY;
        }

        player->sendCancelMessage(ret);
        return false;
    }

    if(isHotkey)
        showHotkeyUseMessage(player, item);

    if(!player->canDoAction())
    {
        uint32_t delay = player->getNextActionTime();
        SchedulerTask* task = createSchedulerTask(delay, boost::bind(&Game::playerUseBattleWindow, this,
            playerId, fromPos, fromStackpos, creatureId, spriteId, isHotkey));

        player->setNextActionTask(task);
        return false;
    }

    player->setIdleTime(0);
    player->setNextActionTask(NULL);
    return g_actions->useItemEx(player, fromPos, creature->getPosition(),
        creature->getParent()->__getIndexOfThing(creature), item, isHotkey, creatureId);
}

bool Game::playerCloseContainer(uint32_t playerId, uint8_t cid)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    player->closeContainer(cid);
    player->sendCloseContainer(cid);
    return true;
}

bool Game::playerMoveUpContainer(uint32_t playerId, uint8_t cid)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    Container* container = player->getContainer(cid);
    if(!container)
        return false;

    Container* parentContainer = dynamic_cast<Container*>(container->getParent());
    if(!parentContainer)
        return false;

    player->addContainer(cid, parentContainer);
    player->sendContainer(cid, parentContainer, (dynamic_cast<const Container*>(parentContainer->getParent()) != NULL));
    return true;
}

bool Game::playerUpdateTile(uint32_t playerId, const Position& pos)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    if(!player->canSee(pos))
        return false;

    if(Tile* tile = getTile(pos))
        player->sendUpdateTile(tile, pos);

    return true;
}

bool Game::playerUpdateContainer(uint32_t playerId, uint8_t cid)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    Container* container = player->getContainer(cid);
    if(!container)
        return false;

    player->sendContainer(cid, container, (dynamic_cast<const Container*>(container->getParent()) != NULL));
    return true;
}

bool Game::playerRotateItem(uint32_t playerId, const Position& pos, int16_t stackpos, const uint16_t spriteId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    Thing* thing = internalGetThing(player, pos, stackpos);
    if(!thing)
        return false;

    Item* item = thing->getItem();
    if(!item || item->getClientID() != spriteId || !item->isRoteable() || (item->isLoadedFromMap() &&
        (item->getUniqueId() != 0 || (item->getActionId() != 0 && item->getContainer()))))
    {
        player->sendCancelMessage(RET_NOTPOSSIBLE);
        return false;
    }

    if(pos.x != 0xFFFF && !Position::areInRange<1,1,0>(pos, player->getPosition()))
    {
        std::list<Direction> listDir;
        if(getPathToEx(player, pos, listDir, 0, 1, true, true))
        {
            Dispatcher::getInstance().addTask(createTask(boost::bind(&Game::playerAutoWalk,
                this, player->getID(), listDir)));

            SchedulerTask* task = createSchedulerTask(400, boost::bind(&Game::playerRotateItem, this,
                playerId, pos, stackpos, spriteId));

            player->setNextWalkActionTask(task);
            return true;
        }

        player->sendCancelMessage(RET_THEREISNOWAY);
        return false;
    }

    uint16_t newId = Item::items[item->getID()].rotateTo;
    if(newId != 0)
        transformItem(item, newId);

    player->setIdleTime(0);
    return true;
}

bool Game::playerWriteItem(uint32_t playerId, uint32_t windowTextId, const std::string& text)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    uint16_t maxTextLength = 0;
    uint32_t internalWindowTextId = 0;

    Item* writeItem = player->getWriteItem(internalWindowTextId, maxTextLength);
    if(text.length() > maxTextLength || windowTextId != internalWindowTextId)
        return false;

    if(!writeItem || writeItem->isRemoved())
    {
        player->sendCancelMessage(RET_NOTPOSSIBLE);
        return false;
    }

    Cylinder* topParent = writeItem->getTopParent();
    Player* owner = dynamic_cast<Player*>(topParent);
    if(owner && owner != player)
    {
        player->sendCancelMessage(RET_NOTPOSSIBLE);
        return false;
    }

    if(!Position::areInRange<1,1,0>(writeItem->getPosition(), player->getPosition()))
    {
        player->sendCancelMessage(RET_NOTPOSSIBLE);
        return false;
    }

    bool deny = false;
    CreatureEventList textEditEvents = player->getCreatureEvents(CREATURE_EVENT_TEXTEDIT);
    for(CreatureEventList::iterator it = textEditEvents.begin(); it != textEditEvents.end(); ++it)
    {
        if(!(*it)->executeTextEdit(player, writeItem, text))
            deny = true;
    }

    if(deny)
        return false;

    if(!text.empty())
    {
        if(writeItem->getText() != text)
        {
            writeItem->setText(text);
            writeItem->setWriter(player->getName());
            writeItem->setDate(std::time(NULL));
        }
    }
    else
    {
        writeItem->resetText();
        writeItem->resetWriter();
        writeItem->resetDate();
    }

    uint16_t newId = Item::items[writeItem->getID()].writeOnceItemId;
    if(newId != 0)
        transformItem(writeItem, newId);

    player->setWriteItem(NULL);
    return true;
}

bool Game::playerUpdateHouseWindow(uint32_t playerId, uint8_t listId, uint32_t windowTextId, const std::string& text)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    uint32_t internalWindowTextId = 0;
    uint32_t internalListId = 0;

    House* house = player->getEditHouse(internalWindowTextId, internalListId);
    if(house && internalWindowTextId == windowTextId && listId == 0)
    {
        house->setAccessList(internalListId, text);
        player->setEditHouse(NULL);
    }

    return true;
}

bool Game::playerRequestTrade(uint32_t playerId, const Position& pos, int16_t stackpos,
    uint32_t tradePlayerId, uint16_t spriteId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    Player* tradePartner = getPlayerByID(tradePlayerId);
    if(!tradePartner || tradePartner == player)
    {
        player->sendTextMessage(MSG_INFO_DESCR, "Sorry, not possible.");
        return false;
    }

    if(!Position::areInRange<2,2,0>(tradePartner->getPosition(), player->getPosition()))
    {
        std::stringstream ss;
        ss << tradePartner->getName() << " tells you to move closer.";
        player->sendTextMessage(MSG_INFO_DESCR, ss.str());
        return false;
    }

    if(!canThrowObjectTo(tradePartner->getPosition(), player->getPosition())
        && !player->hasCustomFlag(PlayerCustomFlag_CanThrowAnywhere))
    {
        player->sendCancelMessage(RET_CREATUREISNOTREACHABLE);
        return false;
    }

    Item* tradeItem = dynamic_cast<Item*>(internalGetThing(player, pos, stackpos, spriteId, STACKPOS_USE));
    if(!tradeItem || tradeItem->getClientID() != spriteId || !tradeItem->isPickupable() || (tradeItem->isLoadedFromMap() &&
        (tradeItem->getUniqueId() != 0 || (tradeItem->getActionId() != 0 && tradeItem->getContainer()))))
    {
        player->sendCancelMessage(RET_NOTPOSSIBLE);
        return false;
    }

    if(player->getPosition().z > tradeItem->getPosition().z)
    {
        player->sendCancelMessage(RET_FIRSTGOUPSTAIRS);
        return false;
    }

    if(player->getPosition().z < tradeItem->getPosition().z)
    {
        player->sendCancelMessage(RET_FIRSTGODOWNSTAIRS);
        return false;
    }

    if(!Position::areInRange<1,1,0>(tradeItem->getPosition(), player->getPosition()))
    {
        std::list<Direction> listDir;
        if(getPathToEx(player, pos, listDir, 0, 1, true, true))
        {
            Dispatcher::getInstance().addTask(createTask(boost::bind(&Game::playerAutoWalk,
                this, player->getID(), listDir)));

            SchedulerTask* task = createSchedulerTask(400, boost::bind(&Game::playerRequestTrade, this,
                playerId, pos, stackpos, tradePlayerId, spriteId));

            player->setNextWalkActionTask(task);
            return true;
        }

        player->sendCancelMessage(RET_THEREISNOWAY);
        return false;
    }

    const Container* container = NULL;
    for(std::map<Item*, uint32_t>::const_iterator it = tradeItems.begin(); it != tradeItems.end(); it++)
    {
        if(tradeItem == it->first ||
            ((container = dynamic_cast<const Container*>(tradeItem)) && container->isHoldingItem(it->first)) ||
            ((container = dynamic_cast<const Container*>(it->first)) && container->isHoldingItem(tradeItem)))
        {
            player->sendTextMessage(MSG_INFO_DESCR, "This item is already being traded.");
            return false;
        }
    }

    Container* tradeContainer = tradeItem->getContainer();
    if(tradeContainer && tradeContainer->getItemHoldingCount() + 1 > 100)
    {
        player->sendTextMessage(MSG_INFO_DESCR, "You cannot trade more than 100 items.");
        return false;
    }

    bool deny = false;
    CreatureEventList tradeEvents = player->getCreatureEvents(CREATURE_EVENT_TRADE_REQUEST);
    for(CreatureEventList::iterator it = tradeEvents.begin(); it != tradeEvents.end(); ++it)
    {
        if(!(*it)->executeTradeRequest(player, tradePartner, tradeItem))
            deny = true;
    }

    if(deny)
        return false;

    return internalStartTrade(player, tradePartner, tradeItem);
}

bool Game::internalStartTrade(Player* player, Player* tradePartner, Item* tradeItem)
{
    if(player->tradeState != TRADE_NONE && !(player->tradeState == TRADE_ACKNOWLEDGE && player->tradePartner == tradePartner))
    {
        player->sendCancelMessage(RET_YOUAREALREADYTRADING);
        return false;
    }
    else if(tradePartner->tradeState != TRADE_NONE && tradePartner->tradePartner != player)
    {
        player->sendCancelMessage(RET_THISPLAYERISALREADYTRADING);
        return false;
    }

    player->tradePartner = tradePartner;
    player->tradeItem = tradeItem;
    player->tradeState = TRADE_INITIATED;

    tradeItem->addRef();
    tradeItems[tradeItem] = player->getID();

    player->sendTradeItemRequest(player, tradeItem, true);
    if(tradePartner->tradeState == TRADE_NONE)
    {
        char buffer[100];
        sprintf(buffer, "%s wants to trade with you", player->getName().c_str());
        tradePartner->sendTextMessage(MSG_INFO_DESCR, buffer);

        tradePartner->tradeState = TRADE_ACKNOWLEDGE;
        tradePartner->tradePartner = player;
    }
    else
    {
        Item* counterItem = tradePartner->tradeItem;
        player->sendTradeItemRequest(tradePartner, counterItem, false);
        tradePartner->sendTradeItemRequest(player, tradeItem, false);
    }

    return true;
}

bool Game::playerAcceptTrade(uint32_t playerId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved() || (player->getTradeState() != TRADE_ACKNOWLEDGE
        && player->getTradeState() != TRADE_INITIATED))
        return false;

    Player* tradePartner = player->tradePartner;
    if(!tradePartner)
        return false;

    if(!canThrowObjectTo(tradePartner->getPosition(), player->getPosition())
        && !player->hasCustomFlag(PlayerCustomFlag_CanThrowAnywhere))
    {
        player->sendCancelMessage(RET_CREATUREISNOTREACHABLE);
        return false;
    }

    player->setTradeState(TRADE_ACCEPT);
    if(tradePartner->getTradeState() != TRADE_ACCEPT)
        return false;

    Item* tradeItem1 = player->tradeItem;
    Item* tradeItem2 = tradePartner->tradeItem;

    bool deny = false;
    CreatureEventList tradeEvents = player->getCreatureEvents(CREATURE_EVENT_TRADE_ACCEPT);
    for(CreatureEventList::iterator it = tradeEvents.begin(); it != tradeEvents.end(); ++it)
    {
        if(!(*it)->executeTradeAccept(player, tradePartner, tradeItem1, tradeItem2))
            deny = true;
    }

    if(deny)
        return false;

    player->setTradeState(TRADE_TRANSFER);
    tradePartner->setTradeState(TRADE_TRANSFER);

    std::map<Item*, uint32_t>::iterator it = tradeItems.find(tradeItem1);
    if(it != tradeItems.end())
    {
        freeThing(it->first);
        tradeItems.erase(it);
    }

    it = tradeItems.find(tradeItem2);
    if(it != tradeItems.end())
    {
        freeThing(it->first);
        tradeItems.erase(it);
    }

    ReturnValue ret1 = internalAddItem(player, tradePartner, tradeItem1, INDEX_WHEREEVER, 0, true);
    ReturnValue ret2 = internalAddItem(tradePartner, player, tradeItem2, INDEX_WHEREEVER, 0, true);

    bool success = false;
    if(ret1 == RET_NOERROR && ret2 == RET_NOERROR)
    {
        ret1 = internalRemoveItem(tradePartner, tradeItem1, tradeItem1->getItemCount(), true);
        ret2 = internalRemoveItem(player, tradeItem2, tradeItem2->getItemCount(), true);
        if(ret1 == RET_NOERROR && ret2 == RET_NOERROR)
        {
            Cylinder *cylinder1 = tradeItem1->getParent(), *cylinder2 = tradeItem2->getParent();
            internalMoveItem(player, cylinder1, tradePartner, INDEX_WHEREEVER, tradeItem1, tradeItem1->getItemCount(), NULL);
            internalMoveItem(tradePartner, cylinder2, player, INDEX_WHEREEVER, tradeItem2, tradeItem2->getItemCount(), NULL);

            tradeItem1->onTradeEvent(ON_TRADE_TRANSFER, tradePartner, player);
            tradeItem2->onTradeEvent(ON_TRADE_TRANSFER, player, tradePartner);
            success = true;
        }
    }

    if(!success)
    {
        std::string error;
        if(tradeItem2)
        {
            error = getTradeErrorDescription(ret1, tradeItem1);
            tradePartner->sendTextMessage(MSG_INFO_DESCR, error);
            tradeItem2->onTradeEvent(ON_TRADE_CANCEL, tradePartner, NULL);
        }

        if(tradeItem1)
        {
            error = getTradeErrorDescription(ret2, tradeItem2);
            player->sendTextMessage(MSG_INFO_DESCR, error);
            tradeItem1->onTradeEvent(ON_TRADE_CANCEL, player, NULL);
        }
    }

    player->setTradeState(TRADE_NONE);
    player->tradeItem = NULL;
    player->tradePartner = NULL;

    tradePartner->setTradeState(TRADE_NONE);
    tradePartner->tradeItem = NULL;
    tradePartner->tradePartner = NULL;

    player->sendTradeClose();
    tradePartner->sendTradeClose();
    return success;
}

std::string Game::getTradeErrorDescription(ReturnValue ret, Item* item)
{
    if(!item)
        return std::string();

    std::stringstream ss;
    if(ret == RET_NOTENOUGHCAPACITY)
    {
        ss << "You do not have enough capacity to carry";
        if(item->isStackable() && item->getItemCount() > 1)
            ss << " these objects.";
        else
            ss << " this object." ;

        ss << std::endl << " " << item->getWeightDescription();
    }
    else if(ret == RET_NOTENOUGHROOM || ret == RET_CONTAINERNOTENOUGHROOM)
    {
        ss << "You do not have enough room to carry";
        if(item->isStackable() && item->getItemCount() > 1)
            ss << " these objects.";
        else
            ss << " this object.";
    }
    else
        ss << "Trade could not be completed.";

    return ss.str().c_str();
}

bool Game::playerLookInTrade(uint32_t playerId, bool lookAtCounterOffer, int32_t index)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    Player* tradePartner = player->tradePartner;
    if(!tradePartner)
        return false;

    Item* tradeItem = NULL;
    if(lookAtCounterOffer)
        tradeItem = tradePartner->getTradeItem();
    else
        tradeItem = player->getTradeItem();

    if(!tradeItem)
        return false;

    std::stringstream ss;
    ss << "You see ";

    int32_t lookDistance = std::max(std::abs(player->getPosition().x - tradeItem->getPosition().x),
        std::abs(player->getPosition().y - tradeItem->getPosition().y));
    if(!index)
    {
        ss << tradeItem->getDescription(lookDistance);
        if(player->hasCustomFlag(PlayerCustomFlag_CanSeeItemDetails))
        {
            ss << std::endl << "ItemID: [" << tradeItem->getID() << "]";
            if(tradeItem->getActionId() > 0)
                ss << ", ActionID: [" << tradeItem->getActionId() << "]";

            if(tradeItem->getUniqueId() > 0)
                ss << ", UniqueID: [" << tradeItem->getUniqueId() << "]";

            ss << ".";
            const ItemType& it = Item::items[tradeItem->getID()];
            if(it.transformEquipTo)
                ss << std::endl << "TransformTo (onEquip): [" << it.transformEquipTo << "]";
            else if(it.transformDeEquipTo)
                ss << std::endl << "TransformTo (onDeEquip): [" << it.transformDeEquipTo << "]";
            else if(it.decayTo != -1)
                ss << std::endl << "DecayTo: [" << it.decayTo << "]";
        }

        player->sendTextMessage(MSG_INFO_DESCR, ss.str());
        return false;
    }

    Container* tradeContainer = tradeItem->getContainer();
    if(!tradeContainer || index > (int32_t)tradeContainer->getItemHoldingCount())
        return false;

    std::list<const Container*> listContainer;
    listContainer.push_back(tradeContainer);

    ItemList::const_iterator it;
    Container* tmpContainer = NULL;
    while(listContainer.size() > 0)
    {
        const Container* container = listContainer.front();
        listContainer.pop_front();
        for(it = container->getItems(); it != container->getEnd(); ++it)
        {
            if((tmpContainer = (*it)->getContainer()))
                listContainer.push_back(tmpContainer);

            --index;
            if(index != 0)
                continue;

            ss << (*it)->getDescription(lookDistance);
            if(player->hasCustomFlag(PlayerCustomFlag_CanSeeItemDetails))
            {
                ss << std::endl << "ItemID: [" << (*it)->getID() << "]";
                if((*it)->getActionId() > 0)
                    ss << ", ActionID: [" << (*it)->getActionId() << "]";

                if((*it)->getUniqueId() > 0)
                    ss << ", UniqueID: [" << (*it)->getUniqueId() << "]";

                ss << ".";
                const ItemType& iit = Item::items[(*it)->getID()];
                if(iit.transformEquipTo)
                    ss << std::endl << "TransformTo: [" << iit.transformEquipTo << "] (onEquip).";
                else if(iit.transformDeEquipTo)
                    ss << std::endl << "TransformTo: [" << iit.transformDeEquipTo << "] (onDeEquip).";
                else if(iit.decayTo != -1)
                    ss << std::endl << "DecayTo: [" << iit.decayTo << "].";
            }

            player->sendTextMessage(MSG_INFO_DESCR, ss.str());
            return true;
        }
    }

    return false;
}

bool Game::playerCloseTrade(uint32_t playerId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    return internalCloseTrade(player);
}

bool Game::internalCloseTrade(Player* player)
{
    Player* tradePartner = player->tradePartner;
    if((tradePartner && tradePartner->getTradeState() == TRADE_TRANSFER) || player->getTradeState() == TRADE_TRANSFER)
    {
        std::clog << "[Warning - Game::internalCloseTrade] TradeState == TRADE_TRANSFER, " <<
            player->getName() << " " << player->getTradeState() << ", " <<
            tradePartner->getName() << " " << tradePartner->getTradeState() << std::endl;
        return true;
    }

    std::vector<Item*>::iterator it;
    if(player->getTradeItem())
    {
        std::map<Item*, uint32_t>::iterator it = tradeItems.find(player->getTradeItem());
        if(it != tradeItems.end())
        {
            freeThing(it->first);
            tradeItems.erase(it);
        }

        player->tradeItem->onTradeEvent(ON_TRADE_CANCEL, player, NULL);
        player->tradeItem = NULL;
    }

    player->setTradeState(TRADE_NONE);
    player->tradePartner = NULL;

    player->sendTextMessage(MSG_STATUS_SMALL, "Trade cancelled.");
    player->sendTradeClose();
    if(tradePartner)
    {
        if(tradePartner->getTradeItem())
        {
            std::map<Item*, uint32_t>::iterator it = tradeItems.find(tradePartner->getTradeItem());
            if(it != tradeItems.end())
            {
                freeThing(it->first);
                tradeItems.erase(it);
            }

            tradePartner->tradeItem->onTradeEvent(ON_TRADE_CANCEL, tradePartner, NULL);
            tradePartner->tradeItem = NULL;
        }

        tradePartner->setTradeState(TRADE_NONE);
        tradePartner->tradePartner = NULL;

        tradePartner->sendTextMessage(MSG_STATUS_SMALL, "Trade cancelled.");
        tradePartner->sendTradeClose();
    }

    return true;
}

bool Game::playerPurchaseItem(uint32_t playerId, uint16_t spriteId, uint8_t count, uint8_t amount,
    bool ignoreCap/* = false*/, bool inBackpacks/* = false*/)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    int32_t onBuy, onSell;
    Npc* merchant = player->getShopOwner(onBuy, onSell);
    if(!merchant)
        return false;

    const ItemType& it = Item::items.getItemIdByClientId(spriteId);
    if(!it.id)
        return false;

    uint8_t subType = count;
    if(it.isFluidContainer() && count < uint8_t(sizeof(reverseFluidMap) / sizeof(int8_t)))
        subType = reverseFluidMap[count];

    if(!player->canShopItem(it.id, subType, SHOPEVENT_BUY))
        return false;

    merchant->onPlayerTrade(player, SHOPEVENT_BUY, onBuy, it.id, subType, amount, ignoreCap, inBackpacks);
    return true;
}

bool Game::playerSellItem(uint32_t playerId, uint16_t spriteId, uint8_t count, uint8_t amount)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    int32_t onBuy, onSell;
    Npc* merchant = player->getShopOwner(onBuy, onSell);
    if(!merchant)
        return false;

    const ItemType& it = Item::items.getItemIdByClientId(spriteId);
    if(!it.id)
        return false;

    uint8_t subType = count;
    if(it.isFluidContainer() && count < uint8_t(sizeof(reverseFluidMap) / sizeof(int8_t)))
        subType = reverseFluidMap[count];

    if(!player->canShopItem(it.id, subType, SHOPEVENT_SELL))
        return false;

    merchant->onPlayerTrade(player, SHOPEVENT_SELL, onSell, it.id, subType, amount);
    return true;
}

bool Game::playerCloseShop(uint32_t playerId)
{
    Player* player = getPlayerByID(playerId);
    if(player == NULL || player->isRemoved())
        return false;

    player->closeShopWindow();
    return true;
}

bool Game::playerLookInShop(uint32_t playerId, uint16_t spriteId, uint8_t count)
{
    Player* player = getPlayerByID(playerId);
    if(player == NULL || player->isRemoved())
        return false;

    const ItemType& it = Item::items.getItemIdByClientId(spriteId);
    if(it.id == 0)
        return false;

    int32_t subType = count;
    if(it.isFluidContainer())
    {
        if(subType == 3) // FIXME: hack
            subType = 11;
        else if(count < uint8_t(sizeof(reverseFluidMap) / sizeof(int8_t)))
            subType = reverseFluidMap[count];
    }

    std::stringstream ss;
    ss << "You see " << Item::getDescription(it, 1, NULL, subType);
    if(player->hasCustomFlag(PlayerCustomFlag_CanSeeItemDetails))
        ss << std::endl << "ItemID: [" << it.id << "].";

    player->sendTextMessage(MSG_INFO_DESCR, ss.str());
    return true;
}

bool Game::playerLookAt(uint32_t playerId, const Position& pos, uint16_t spriteId, int16_t stackpos)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    Thing* thing = internalGetThing(player, pos, stackpos, spriteId, STACKPOS_LOOK);
    if(!thing)
    {
        player->sendCancelMessage(RET_NOTPOSSIBLE);
        return false;
    }

    Position thingPos = pos;
    if(pos.x == 0xFFFF)
        thingPos = thing->getPosition();

    if(!player->canSee(thingPos))
    {
        player->sendCancelMessage(RET_NOTPOSSIBLE);
        return false;
    }

    Position playerPos = player->getPosition();
    int32_t lookDistance = -1;
    if(thing != player)
    {
        lookDistance = std::max(std::abs(playerPos.x - thingPos.x), std::abs(playerPos.y - thingPos.y));
        if(playerPos.z != thingPos.z)
            lookDistance = lookDistance + 9 + 6;
    }

    bool deny = false;
    CreatureEventList lookEvents = player->getCreatureEvents(CREATURE_EVENT_LOOK);
    for(CreatureEventList::iterator it = lookEvents.begin(); it != lookEvents.end(); ++it)
    {
        if(!(*it)->executeLook(player, thing, thingPos, stackpos, lookDistance))
            deny = true;
    }

    if(deny)
        return false;

    std::stringstream ss;
    ss << "You see " << thing->getDescription(lookDistance);
    if(player->hasCustomFlag(PlayerCustomFlag_CanSeeItemDetails))
    {
        if(Item* item = thing->getItem())
        {
            ss << std::endl << "ItemID: [" << item->getID() << "]";
            if(item->getActionId() > 0)
                ss << ", ActionID: [" << item->getActionId() << "]";

            if(item->getUniqueId() > 0)
                ss << ", UniqueID: [" << item->getUniqueId() << "]";

            ss << ".";
            const ItemType& it = Item::items[item->getID()];
            if(it.transformEquipTo)
                ss << std::endl << "TransformTo: [" << it.transformEquipTo << "] (onEquip).";
            else if(it.transformDeEquipTo)
                ss << std::endl << "TransformTo: [" << it.transformDeEquipTo << "] (onDeEquip).";
            else if(it.decayTo != -1)
                ss << std::endl << "DecayTo: [" << it.decayTo << "].";
        }
    }

    if(player->hasCustomFlag(PlayerCustomFlag_CanSeeCreatureDetails))
    {
        if(const Creature* creature = thing->getCreature())
        {
            ss << std::endl << "Health: [" << creature->getHealth() << " / " << creature->getMaxHealth() << "]";
            if(creature->getMaxMana() > 0)
                ss << ", Mana: [" << creature->getMana() << " / " << creature->getMaxMana() << "]";

            ss << ".";
            if(const Player* destPlayer = creature->getPlayer())
            {
                ss << std::endl << "IP: " << convertIPAddress(destPlayer->getIP()) << ", Client: " << destPlayer->getClientVersion() << ".";
                if(destPlayer->isGhost())
                    ss << std::endl << "* Ghost mode *";
            }
        }
    }

    if(player->hasCustomFlag(PlayerCustomFlag_CanSeePosition))
        ss << std::endl << "Position: [X: " << thingPos.x << "] [Y: " << thingPos.y << "] [Z: " << thingPos.z << "].";

    player->sendTextMessage(MSG_INFO_DESCR, ss.str());
    return true;
}

bool Game::playerQuests(uint32_t playerId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    player->sendQuests();
    return true;
}

bool Game::playerQuestInfo(uint32_t playerId, uint16_t questId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    Quest* quest = Quests::getInstance()->getQuestById(questId);
    if(!quest)
        return false;

    player->sendQuestInfo(quest);
    return true;
}

bool Game::playerCancelAttackAndFollow(uint32_t playerId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    playerSetAttackedCreature(playerId, 0);
    playerFollowCreature(playerId, 0);

    player->stopWalk();
    return true;
}

bool Game::playerSetAttackedCreature(uint32_t playerId, uint32_t creatureId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    if(player->getAttackedCreature() && !creatureId)
    {
        player->setAttackedCreature(NULL);
        player->sendCancelTarget();
        return true;
    }

    Creature* attackCreature = getCreatureByID(creatureId);
    if(!attackCreature)
    {
        player->setAttackedCreature(NULL);
        player->sendCancelTarget();
        return false;
    }

    ReturnValue ret = Combat::canTargetCreature(player, attackCreature);
    if(ret != RET_NOERROR)
    {
        player->sendCancelMessage(ret);
        player->sendCancelTarget();
        player->setAttackedCreature(NULL);
        return false;
    }

    player->setAttackedCreature(attackCreature);
    Dispatcher::getInstance().addTask(createTask(boost::bind(
        &Game::updateCreatureWalk, this, player->getID())));
    return true;
}

bool Game::playerFollowCreature(uint32_t playerId, uint32_t creatureId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    Creature* followCreature = NULL;
    if(creatureId)
        followCreature = getCreatureByID(creatureId);

    player->setAttackedCreature(NULL);
    Dispatcher::getInstance().addTask(createTask(boost::bind(
        &Game::updateCreatureWalk, this, player->getID())));
    return player->setFollowCreature(followCreature);
}

bool Game::playerSetFightModes(uint32_t playerId, fightMode_t fightMode, chaseMode_t chaseMode, secureMode_t secureMode)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    player->setFightMode(fightMode);
    player->setChaseMode(chaseMode);
    player->setSecureMode(secureMode);

    return true;
}

bool Game::playerRequestAddVip(uint32_t playerId, const std::string& vipName)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    uint32_t guid;
    bool specialVip;

    std::string name = vipName;
    if(!IOLoginData::getInstance()->getGuidByNameEx(guid, specialVip, name))
    {
        player->sendTextMessage(MSG_STATUS_SMALL, "A player with that name does not exist.");
        return false;
    }

    if(specialVip && !player->hasFlag(PlayerFlag_SpecialVIP))
    {
        player->sendTextMessage(MSG_STATUS_SMALL, "You cannot add this player.");
        return false;
    }

    bool online = false;
    if(Player* target = getPlayerByName(name))
        online = player->canSeeCreature(target);

    return player->addVIP(guid, name, online);
}

bool Game::playerRequestRemoveVip(uint32_t playerId, uint32_t guid)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    player->removeVIP(guid);
    return true;
}

bool Game::playerTurn(uint32_t playerId, Direction dir)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    if(internalCreatureTurn(player, dir))
    {
        player->setIdleTime(0);
        return true;
    }

    if(player->getDirection() != dir || !player->hasCustomFlag(PlayerCustomFlag_CanTurnhop))
        return false;

    Position pos = getNextPosition(dir, player->getPosition());
    Tile* tile = map->getTile(pos);
    if(!tile || !tile->ground)
        return false;

    player->setIdleTime(0);
    ReturnValue ret = tile->__queryAdd(0, player, 1, FLAG_IGNOREBLOCKITEM);
    if(ret != RET_NOTENOUGHROOM && (ret != RET_NOTPOSSIBLE || player->hasCustomFlag(PlayerCustomFlag_CanMoveAnywhere))
        && (ret != RET_PLAYERISNOTINVITED || player->hasFlag(PlayerFlag_CanEditHouses)))
        return internalTeleport(player, pos, false, FLAG_NOLIMIT, false);

    player->sendCancelMessage(ret);
    return false;
}

bool Game::playerRequestOutfit(uint32_t playerId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    player->sendOutfitWindow();
    return true;
}

bool Game::playerChangeOutfit(uint32_t playerId, Outfit_t outfit)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    if(!player->changeOutfit(outfit, true))
        return false;

    player->setIdleTime(0);
    if(!player->hasCondition(CONDITION_OUTFIT, -1))
        internalCreatureChangeOutfit(player, outfit);

    return true;
}

bool Game::playerSay(uint32_t playerId, uint16_t channelId, SpeakClasses type, const std::string& receiver, const std::string& text, ProtocolGame* pg) //CA
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    if (pg != NULL && pg->getIsCast() && !player->isAccountManager()) {  //CA
        if(g_talkActions->onPlayerSay(player, type == SPEAK_SAY ? (unsigned)CHANNEL_DEFAULT : channelId, text, false, pg))
            return true;

        PlayerCast pc = player->getCast();
        for(std::list<CastBan>::iterator it = pc.muted.begin(); it != pc.muted.end(); ++it)
            if(it->ip == pg->getIP()) {
                pg->publicSendMessage(player, SPEAK_PRIVATE, "You are muted on this cast.");
                return false;
            }

        if(playerTalkToChannel(player, type, text, channelId, pg))
            return true;
        else
            return false;
    }

    uint32_t muted = 0;
    bool mute = player->isMuted(channelId, type, muted);
    if(muted && mute)
    {
        char buffer[75];
        sprintf(buffer, "You are still muted for %d seconds.", muted);
        player->sendTextMessage(MSG_STATUS_SMALL, buffer);
        return false;
    }

    if(player->isAccountManager())
    {
        if(mute)
            player->removeMessageBuffer();

        return internalCreatureSay(player, SPEAK_SAY, text, false);
    }

    if(g_talkActions->onPlayerSay(player, type == SPEAK_SAY ? (unsigned)CHANNEL_DEFAULT : channelId, text, false))
        return true;

    ReturnValue ret = RET_NOERROR;
    if(!muted)
    {
        ret = g_spells->onPlayerSay(player, text);
        if(ret == RET_NOERROR || (ret == RET_NEEDEXCHANGE && !g_config.getBool(ConfigManager::BUFFER_SPELL_FAILURE)))
            return true;
    }

    if(mute)
        player->removeMessageBuffer();

    if(ret == RET_NEEDEXCHANGE)
        return true;

    switch(type)
    {
        case SPEAK_SAY:
            return internalCreatureSay(player, SPEAK_SAY, text, false);
        case SPEAK_WHISPER:
            return playerWhisper(player, text);
        case SPEAK_YELL:
            return playerYell(player, text);
        case SPEAK_PRIVATE:
        case SPEAK_PRIVATE_RED:
        case SPEAK_RVR_ANSWER:
            return playerSpeakTo(player, type, receiver, text);
        case SPEAK_CHANNEL_O:
        case SPEAK_CHANNEL_Y:
        case SPEAK_CHANNEL_RN:
        case SPEAK_CHANNEL_RA:
        case SPEAK_CHANNEL_W:
        {
            if(playerTalkToChannel(player, type, text, channelId))
                return true;

            return playerSay(playerId, 0, SPEAK_SAY, receiver, text);
        }
        case SPEAK_PRIVATE_PN:
            return playerSpeakToNpc(player, text);
        case SPEAK_BROADCAST:
            return playerBroadcastMessage(player, SPEAK_BROADCAST, text);
        case SPEAK_RVR_CHANNEL:
            return playerReportRuleViolation(player, text);
        case SPEAK_RVR_CONTINUE:
            return playerContinueReport(player, text);

        default:
            break;
    }

    return false;
}

bool Game::playerWhisper(Player* player, const std::string& text)
{
    SpectatorVec list;
    SpectatorVec::const_iterator it;
    getSpectators(list, player->getPosition(), false, false,
        Map::maxClientViewportX, Map::maxClientViewportX,
        Map::maxClientViewportY, Map::maxClientViewportY);

    //send to client
    Player* tmpPlayer = NULL;
    for(it = list.begin(); it != list.end(); ++it)
    {
        if((tmpPlayer = (*it)->getPlayer()))
            tmpPlayer->sendCreatureSay(player, SPEAK_WHISPER, text);
    }

    //event method
    for(it = list.begin(); it != list.end(); ++it)
        (*it)->onCreatureSay(player, SPEAK_WHISPER, text);

    return true;
}

bool Game::playerYell(Player* player, const std::string& text)
{
    if(player->getLevel() <= 1 && !player->hasFlag(PlayerFlag_CannotBeMuted))
    {
        player->sendTextMessage(MSG_STATUS_SMALL, "You may not yell as long as you are on level 1.");
        return true;
    }

    if(player->hasCondition(CONDITION_MUTED, 1))
    {
        player->sendCancelMessage(RET_YOUAREEXHAUSTED);
        return true;
    }

    if(!player->hasFlag(PlayerFlag_CannotBeMuted))
    {
        if(Condition* condition = Condition::createCondition(CONDITIONID_DEFAULT, CONDITION_MUTED, 30000, 0, false, 1))
            player->addCondition(condition);
    }

    internalCreatureSay(player, SPEAK_YELL, asUpperCaseString(text), false);
    return true;
}

bool Game::playerSpeakTo(Player* player, SpeakClasses type, const std::string& receiver,
    const std::string& text)
{
    Player* toPlayer = getPlayerByName(receiver);
    if(!toPlayer || toPlayer->isRemoved())
    {
        player->sendTextMessage(MSG_STATUS_SMALL, "A player with this name is not online.");
        return false;
    }

    bool canSee = player->canSeeCreature(toPlayer);
    if(toPlayer->hasCondition(CONDITION_GAMEMASTER, GAMEMASTER_IGNORE)
        && !player->hasFlag(PlayerFlag_CannotBeMuted))
    {
        char buffer[70];
        if(!canSee)
            sprintf(buffer, "A player with this name is not online.");
        else
            sprintf(buffer, "Sorry, %s is currently ignoring private messages.", toPlayer->getName().c_str());

        player->sendTextMessage(MSG_STATUS_SMALL, buffer);
        return false;
    }

    if(type == SPEAK_PRIVATE_RED && !player->hasFlag(PlayerFlag_CanTalkRedPrivate))
        type = SPEAK_PRIVATE;

    toPlayer->sendCreatureSay(player, type, text);
    toPlayer->onCreatureSay(player, type, text);
    if(!canSee)
    {
        player->sendTextMessage(MSG_STATUS_SMALL, "A player with this name is not online.");
        return false;
    }

    char buffer[80];
    sprintf(buffer, "Message sent to %s.", toPlayer->getName().c_str());
    player->sendTextMessage(MSG_STATUS_SMALL, buffer);
    return true;
}

bool Game::playerTalkToChannel(Player* player, SpeakClasses type, const std::string& text, uint16_t channelId, ProtocolGame* pg) //CA
{
    switch(type)
    {
        case SPEAK_CHANNEL_Y:
        {
            if(channelId == CHANNEL_HELP && player->hasFlag(PlayerFlag_TalkOrangeHelpChannel))
                type = SPEAK_CHANNEL_O;
                
            if(g_chat.getPrivateChannel(player) != NULL && channelId == g_chat.getPrivateChannel(player)->getId() && (pg == NULL || pg != NULL && !pg->getIsCast())) //CA
                type = SPEAK_CHANNEL_O; //CA
                
            break;
        }

        case SPEAK_CHANNEL_O:
        {
            if(channelId != CHANNEL_HELP || !player->hasFlag(PlayerFlag_TalkOrangeHelpChannel))
                type = SPEAK_CHANNEL_Y;
            break;
        }

        case SPEAK_CHANNEL_RN:
        {
            if(!player->hasFlag(PlayerFlag_CanTalkRedChannel))
                type = SPEAK_CHANNEL_Y;
            break;
        }

        case SPEAK_CHANNEL_RA:
        {
            if(!player->hasFlag(PlayerFlag_CanTalkRedChannelAnonymous))
                type = SPEAK_CHANNEL_Y;
            break;
        }

        default:
            break;
    }

    return g_chat.talkToChannel(player, type, text, channelId, pg); //CA
}

bool Game::playerSpeakToNpc(Player* player, const std::string& text)
{
    SpectatorVec list;
    SpectatorVec::iterator it;
    getSpectators(list, player->getPosition());

    //send to npcs only
    Npc* tmpNpc = NULL;
    for(it = list.begin(); it != list.end(); ++it)
    {
        if((tmpNpc = (*it)->getNpc()))
            (*it)->onCreatureSay(player, SPEAK_PRIVATE_PN, text);
    }
    return true;
}

bool Game::playerReportRuleViolation(Player* player, const std::string& text)
{
    //Do not allow reports on multiclones worlds since reports are name-based
    if(g_config.getNumber(ConfigManager::ALLOW_CLONES))
    {
        player->sendTextMessage(MSG_INFO_DESCR, "Rule violation reports are disabled.");
        return false;
    }

    cancelRuleViolation(player);
    boost::shared_ptr<RuleViolation> rvr(new RuleViolation(player, text, time(NULL)));
    ruleViolations[player->getID()] = rvr;

    ChatChannel* channel = g_chat.getChannelById(CHANNEL_RVR);
    if(!channel)
        return false;

    for(UsersMap::const_iterator it = channel->getUsers().begin(); it != channel->getUsers().end(); ++it)
        it->second->sendToChannel(player, SPEAK_RVR_CHANNEL, text, CHANNEL_RVR, rvr->time);

    return true;
}

bool Game::playerContinueReport(Player* player, const std::string& text)
{
    RuleViolationsMap::iterator it = ruleViolations.find(player->getID());
    if(it == ruleViolations.end())
        return false;

    RuleViolation& rvr = *it->second;
    Player* toPlayer = rvr.gamemaster;
    if(!toPlayer)
        return false;

    toPlayer->sendCreatureSay(player, SPEAK_RVR_CONTINUE, text);
    player->sendTextMessage(MSG_STATUS_SMALL, "Message sent to Gamemaster.");
    return true;
}

bool Game::canThrowObjectTo(const Position& fromPos, const Position& toPos, bool checkLineOfSight /*= true*/,
    int32_t rangex/* = Map::maxClientViewportX*/, int32_t rangey/* = Map::maxClientViewportY*/)
{
    return map->canThrowObjectTo(fromPos, toPos, checkLineOfSight, rangex, rangey);
}

bool Game::isSightClear(const Position& fromPos, const Position& toPos, bool floorCheck)
{
    return map->isSightClear(fromPos, toPos, floorCheck);
}

bool Game::internalCreatureTurn(Creature* creature, Direction dir)
{
    bool deny = false;
    CreatureEventList directionEvents = creature->getCreatureEvents(CREATURE_EVENT_DIRECTION);
    for(CreatureEventList::iterator it = directionEvents.begin(); it != directionEvents.end(); ++it)
    {
        if(!(*it)->executeDirection(creature, creature->getDirection(), dir) && !deny)
            deny = true;
    }

    if(deny || creature->getDirection() == dir)
        return false;

    creature->setDirection(dir);
    const SpectatorVec& list = getSpectators(creature->getPosition());
    SpectatorVec::const_iterator it;

    //send to client
    Player* tmpPlayer = NULL;
    for(it = list.begin(); it != list.end(); ++it)
    {
        if((tmpPlayer = (*it)->getPlayer()))
            tmpPlayer->sendCreatureTurn(creature);
    }

    //event method
    for(it = list.begin(); it != list.end(); ++it)
        (*it)->onCreatureTurn(creature);

    return true;
}

bool Game::internalCreatureSay(Creature* creature, SpeakClasses type, const std::string& text,
    bool ghostMode, SpectatorVec* spectators/* = NULL*/, Position* pos/* = NULL*/)
{
    Player* player = creature->getPlayer();
    if(player && player->isAccountManager())
    {
        player->manageAccount(text);
        return true;
    }

    Position destPos = creature->getPosition();
    if(pos)
        destPos = (*pos);

    SpectatorVec list;
    SpectatorVec::const_iterator it;
    if(!spectators || !spectators->size())
    {
        // This somewhat complex construct ensures that the cached SpectatorVec
        // is used if available and if it can be used, else a local vector is
        // used (hopefully the compiler will optimize away the construction of
        // the temporary when it's not used).
        if(type != SPEAK_YELL && type != SPEAK_MONSTER_YELL)
            getSpectators(list, destPos, false, false,
                Map::maxClientViewportX, Map::maxClientViewportX,
                Map::maxClientViewportY, Map::maxClientViewportY);
        else
            getSpectators(list, destPos, false, true, 18, 18, 14, 14);
    }
    else
        list = (*spectators);

    //send to client
    Player* tmpPlayer = NULL;
    for(it = list.begin(); it != list.end(); ++it)
    {
        if(!(tmpPlayer = (*it)->getPlayer()))
            continue;

        if(!ghostMode || tmpPlayer->canSeeCreature(creature))
            tmpPlayer->sendCreatureSay(creature, type, text, &destPos);
    }

    //event method
    for(it = list.begin(); it != list.end(); ++it)
        (*it)->onCreatureSay(creature, type, text, &destPos);

    return true;
}

bool Game::getPathTo(const Creature* creature, const Position& destPos,
    std::list<Direction>& listDir, int32_t maxSearchDist /*= -1*/)
{
    return map->getPathTo(creature, destPos, listDir, maxSearchDist);
}

bool Game::getPathToEx(const Creature* creature, const Position& targetPos,
    std::list<Direction>& dirList, const FindPathParams& fpp)
{
    return map->getPathMatching(creature, dirList, FrozenPathingConditionCall(targetPos), fpp);
}

bool Game::getPathToEx(const Creature* creature, const Position& targetPos, std::list<Direction>& dirList,
    uint32_t minTargetDist, uint32_t maxTargetDist, bool fullPathSearch /*= true*/,
    bool clearSight /*= true*/, int32_t maxSearchDist /*= -1*/)
{
    FindPathParams fpp;
    fpp.fullPathSearch = fullPathSearch;
    fpp.maxSearchDist = maxSearchDist;
    fpp.clearSight = clearSight;
    fpp.minTargetDist = minTargetDist;
    fpp.maxTargetDist = maxTargetDist;
    return getPathToEx(creature, targetPos, dirList, fpp);
}

void Game::checkCreatureWalk(uint32_t creatureId)
{
    Creature* creature = getCreatureByID(creatureId);
    if(!creature || creature->getHealth() < 1)
        return;

    creature->onWalk();
    cleanup();
}

void Game::updateCreatureWalk(uint32_t creatureId)
{
    Creature* creature = getCreatureByID(creatureId);
    if(creature && creature->getHealth() > 0)
        creature->goToFollowCreature();
}

void Game::checkCreatureAttack(uint32_t creatureId)
{
    Creature* creature = getCreatureByID(creatureId);
    if(creature && creature->getHealth() > 0)
        creature->onAttacking(0);
}

void Game::addCreatureCheck(Creature* creature)
{
    if(creature->isRemoved())
        return;

    creature->checked = true;
    if(creature->checkVector >= 0) //already in a vector, or about to be added
        return;

    toAddCheckCreatureVector.push_back(creature);
    creature->checkVector = random_range(0, EVENT_CREATURECOUNT - 1);
    creature->addRef();
}

void Game::removeCreatureCheck(Creature* creature)
{
    if(creature->checkVector == -1) //not in any vector
        return;

    creature->checked = false;
}

void Game::checkCreatures()
{
    Scheduler::getInstance().addEvent(createSchedulerTask(
        EVENT_CHECK_CREATURE_INTERVAL, boost::bind(&Game::checkCreatures, this)));
    checkCreatureLastIndex++;
    if(checkCreatureLastIndex == EVENT_CREATURECOUNT)
        checkCreatureLastIndex = 0;

    std::vector<Creature*>::iterator it;
    for(it = toAddCheckCreatureVector.begin(); it != toAddCheckCreatureVector.end(); ++it)
        checkCreatureVectors[(*it)->checkVector].push_back(*it);

    toAddCheckCreatureVector.clear();
    std::vector<Creature*>& checkCreatureVector = checkCreatureVectors[checkCreatureLastIndex];
    for(it = checkCreatureVector.begin(); it != checkCreatureVector.end();)
    {
        if((*it)->checked)
        {
            if((*it)->getHealth() > 0 || !(*it)->onDeath())
                (*it)->onThink(EVENT_CREATURE_THINK_INTERVAL);

            ++it;
        }
        else
        {
            (*it)->checkVector = -1;
            freeThing(*it);
            it = checkCreatureVector.erase(it);
        }
    }

    cleanup();
}

void Game::changeSpeed(Creature* creature, int32_t varSpeedDelta)
{
    int32_t varSpeed = creature->getSpeed() - creature->getBaseSpeed();
    varSpeed += varSpeedDelta;
    creature->setSpeed(varSpeed);

    const SpectatorVec& list = getSpectators(creature->getPosition());
    SpectatorVec::const_iterator it;

    //send to client
    Player* tmpPlayer = NULL;
    for(it = list.begin(); it != list.end(); ++it)
    {
        if((tmpPlayer = (*it)->getPlayer()))
            tmpPlayer->sendChangeSpeed(creature, creature->getStepSpeed());
    }
}

void Game::internalCreatureChangeOutfit(Creature* creature, const Outfit_t& outfit, bool forced/* = false*/)
{
    if(!forced)
    {
        bool deny = false;
        CreatureEventList outfitEvents = creature->getCreatureEvents(CREATURE_EVENT_OUTFIT);
        for(CreatureEventList::iterator it = outfitEvents.begin(); it != outfitEvents.end(); ++it)
        {
            if(!(*it)->executeOutfit(creature, creature->getCurrentOutfit(), outfit) && !deny)
                deny = true;
        }

        if(deny || creature->getCurrentOutfit() == outfit)
            return;
    }

    creature->setCurrentOutfit(outfit);
    const SpectatorVec& list = getSpectators(creature->getPosition());
    SpectatorVec::const_iterator it;

    //send to client
    Player* tmpPlayer = NULL;
    for(it = list.begin(); it != list.end(); ++it)
    {
        if((tmpPlayer = (*it)->getPlayer()))
            tmpPlayer->sendCreatureChangeOutfit(creature, outfit);
    }

    //event method
    for(it = list.begin(); it != list.end(); ++it)
        (*it)->onCreatureChangeOutfit(creature, outfit);
}

void Game::internalCreatureChangeVisible(Creature* creature, Visible_t visible)
{
    const SpectatorVec& list = getSpectators(creature->getPosition());
    SpectatorVec::const_iterator it;

    //send to client
    Player* tmpPlayer = NULL;
    for(it = list.begin(); it != list.end(); ++it)
    {
        if((tmpPlayer = (*it)->getPlayer()))
            tmpPlayer->sendCreatureChangeVisible(creature, visible);
    }

    //event method
    for(it = list.begin(); it != list.end(); ++it)
        (*it)->onCreatureChangeVisible(creature, visible);
}


void Game::changeLight(const Creature* creature)
{
    const SpectatorVec& list = getSpectators(creature->getPosition());

    //send to client
    Player* tmpPlayer = NULL;
    for(SpectatorVec::const_iterator it = list.begin(); it != list.end(); ++it)
    {
        if((tmpPlayer = (*it)->getPlayer()))
            tmpPlayer->sendCreatureLight(creature);
    }
}

bool Game::combatBlockHit(CombatType_t combatType, Creature* attacker, Creature* target,
    int32_t& healthChange, bool checkDefense, bool checkArmor)
{
    if(healthChange > 0)
        return false;

    const Position& targetPos = target->getPosition();
    const SpectatorVec& list = getSpectators(targetPos);
    if(!target->isAttackable() || Combat::canDoCombat(attacker, target) != RET_NOERROR)
    {
        addMagicEffect(list, targetPos, MAGIC_EFFECT_POFF, target->isGhost());
        return true;
    }

    int32_t damage = -healthChange;
    BlockType_t blockType = target->blockHit(attacker, combatType, damage, checkDefense, checkArmor);

    healthChange = -damage;
    if(blockType == BLOCK_DEFENSE)
    {
        addMagicEffect(list, targetPos, MAGIC_EFFECT_POFF);
        return true;
    }
    else if(blockType == BLOCK_ARMOR)
    {
        addMagicEffect(list, targetPos, MAGIC_EFFECT_BLOCKHIT);
        return true;
    }
    else if(blockType != BLOCK_IMMUNITY)
        return false;

    MagicEffect_t effect = MAGIC_EFFECT_NONE;
    switch(combatType)
    {
        case COMBAT_UNDEFINEDDAMAGE:
            break;

        case COMBAT_ENERGYDAMAGE:
        case COMBAT_FIREDAMAGE:
        case COMBAT_PHYSICALDAMAGE:
        case COMBAT_ICEDAMAGE:
        case COMBAT_DEATHDAMAGE:
        case COMBAT_EARTHDAMAGE:
        case COMBAT_HOLYDAMAGE:
        {
            effect = MAGIC_EFFECT_BLOCKHIT;
            break;
        }

        default:
        {
            effect = MAGIC_EFFECT_POFF;
            break;
        }
    }

    addMagicEffect(list, targetPos, effect);
    return true;
}

bool Game::combatChangeHealth(CombatType_t combatType, Creature* attacker, Creature* target, int32_t healthChange,
    MagicEffect_t hitEffect/* = MAGIC_EFFECT_UNKNOWN*/, Color_t hitColor/* = COLOR_UNKNOWN*/, bool force/* = false*/)
{
    const Position& targetPos = target->getPosition();
    if(healthChange > 0)
    {
        if(!force && target->getHealth() <= 0)
            return false;

        bool deny = false;
        CreatureEventList statsChangeEvents = target->getCreatureEvents(CREATURE_EVENT_STATSCHANGE);
        for(CreatureEventList::iterator it = statsChangeEvents.begin(); it != statsChangeEvents.end(); ++it)
        {
            if(!(*it)->executeStatsChange(target, attacker, STATSCHANGE_HEALTHGAIN, combatType, healthChange))
                deny = true;
        }

        if(deny)
            return false;

if(g_config.getBool(ConfigManager::SHOW_HEALING_DAMAGE) && !target->isGhost() &&
    (g_config.getBool(ConfigManager::SHOW_HEALING_DAMAGE_MONSTER) || !target->getMonster()))
{
    int32_t realHeal = healthChange;
    if((realHeal + target->getHealth()) > target->getMaxHealth())
        realHeal = (target->getMaxHealth() - target->getHealth());
    
    if(realHeal > 0)
    {
        char buffer[20];
        sprintf(buffer, "+%d", realHeal);

        const SpectatorVec& list = getSpectators(targetPos);
        if(combatType != COMBAT_HEALING)
            addMagicEffect(list, targetPos, MAGIC_EFFECT_WRAPS_BLUE);

        addAnimatedText(list, targetPos, COLOR_BLUE, buffer);
    }
}
    else
    {
        const SpectatorVec& list = getSpectators(targetPos);
        if(!target->isAttackable() || Combat::canDoCombat(attacker, target) != RET_NOERROR)
        {
            addMagicEffect(list, targetPos, MAGIC_EFFECT_POFF);
            return true;
        }

        int32_t damage = -healthChange;
        if(damage != 0)
        {
            if(target->hasCondition(CONDITION_MANASHIELD) && combatType != COMBAT_UNDEFINEDDAMAGE)
            {
                int32_t manaDamage = std::min(target->getMana(), damage);
                damage = std::max((int32_t)0, damage - manaDamage);
                if(manaDamage != 0)
                {
                    bool deny = false;
                    CreatureEventList statsChangeEvents = target->getCreatureEvents(CREATURE_EVENT_STATSCHANGE);
                    for(CreatureEventList::iterator it = statsChangeEvents.begin(); it != statsChangeEvents.end(); ++it)
                    {
                        if(!(*it)->executeStatsChange(target, attacker, STATSCHANGE_MANALOSS, combatType, manaDamage))
                            deny = true;
                    }

                    if(deny)
                        return false;

                    target->drainMana(attacker, combatType, manaDamage);
                    char buffer[20];
                    sprintf(buffer, "%d", manaDamage);

                    addMagicEffect(list, targetPos, MAGIC_EFFECT_LOSE_ENERGY);
                    addAnimatedText(list, targetPos, COLOR_BLUE, buffer);
                }
            }

            damage = std::min(target->getHealth(), damage);
            if(damage > 0)
            {
                bool deny = false;
                CreatureEventList statsChangeEvents = target->getCreatureEvents(CREATURE_EVENT_STATSCHANGE);
                for(CreatureEventList::iterator it = statsChangeEvents.begin(); it != statsChangeEvents.end(); ++it)
                {
                    if(!(*it)->executeStatsChange(target, attacker, STATSCHANGE_HEALTHLOSS, combatType, damage))
                        deny = true;
                }

                if(deny)
                    return false;

                target->drainHealth(attacker, combatType, damage);
                addCreatureHealth(list, target);

                Color_t textColor = COLOR_NONE;
                MagicEffect_t magicEffect = MAGIC_EFFECT_NONE;
                switch(combatType)
                {
                    case COMBAT_PHYSICALDAMAGE:
                    {
                        Item* splash = NULL;
                        switch(target->getRace())
                        {
                            case RACE_VENOM:
                                textColor = COLOR_LIGHTGREEN;
                                magicEffect = MAGIC_EFFECT_POISON;
                                splash = Item::CreateItem(ITEM_SMALLSPLASH, FLUID_GREEN);
                                break;

                            case RACE_BLOOD:
                                textColor = COLOR_RED;
                                magicEffect = MAGIC_EFFECT_DRAW_BLOOD;
                                splash = Item::CreateItem(ITEM_SMALLSPLASH, FLUID_BLOOD);
                                break;

                            case RACE_UNDEAD:
                                textColor = COLOR_GREY;
                                magicEffect = MAGIC_EFFECT_HIT_AREA;
                                break;

                            case RACE_FIRE:
                                textColor = COLOR_ORANGE;
                                magicEffect = MAGIC_EFFECT_DRAW_BLOOD;
                                break;

                            case RACE_ENERGY:
                                textColor = COLOR_PURPLE;
                                magicEffect = MAGIC_EFFECT_PURPLEENERGY;
                                break;

                            default:
                                break;
                        }

                        if(splash)
                        {
                            internalAddItem(NULL, target->getTile(), splash, INDEX_WHEREEVER, FLAG_NOLIMIT);
                            startDecay(splash);
                        }
                        break;
                    }

                    case COMBAT_ENERGYDAMAGE:
                    {
                        textColor = COLOR_PURPLE;
                        magicEffect = MAGIC_EFFECT_ENERGY_DAMAGE;
                        break;
                    }

                    case COMBAT_EARTHDAMAGE:
                    {
                        textColor = COLOR_LIGHTGREEN;
                        magicEffect = MAGIC_EFFECT_POISON_RINGS;
                        break;
                    }

                    case COMBAT_DROWNDAMAGE:
                    {
                        textColor = COLOR_LIGHTBLUE;
                        magicEffect = MAGIC_EFFECT_LOSE_ENERGY;
                        break;
                    }

                    case COMBAT_FIREDAMAGE:
                    {
                        textColor = COLOR_ORANGE;
                        magicEffect = MAGIC_EFFECT_HITBY_FIRE;
                        break;
                    }

                    case COMBAT_ICEDAMAGE:
                    {
                        textColor = COLOR_TEAL;
                        magicEffect = MAGIC_EFFECT_ICEATTACK;
                        break;
                    }

                    case COMBAT_HOLYDAMAGE:
                    {
                        textColor = COLOR_YELLOW;
                        magicEffect = MAGIC_EFFECT_HOLYDAMAGE;
                        break;
                    }

                    case COMBAT_DEATHDAMAGE:
                    {
                        textColor = COLOR_DARKRED;
                        magicEffect = MAGIC_EFFECT_SMALLCLOUDS;
                        break;
                    }

                    case COMBAT_LIFEDRAIN:
                    {
                        textColor = COLOR_RED;
                        magicEffect = MAGIC_EFFECT_WRAPS_RED;
                        break;
                    }

                    default:
                        break;
                }

                if(hitEffect != MAGIC_EFFECT_UNKNOWN)
                    magicEffect = hitEffect;

                if(hitColor != COLOR_UNKNOWN)
                    textColor = hitColor;

                if(textColor < COLOR_NONE && magicEffect < MAGIC_EFFECT_NONE)
                {
                    char buffer[20];
                    sprintf(buffer, "%d", damage);

                    addMagicEffect(list, targetPos, magicEffect);
                    addAnimatedText(list, targetPos, textColor, buffer);
                }
            }
        }
    }

    return true;
}

bool Game::combatChangeMana(Creature* attacker, Creature* target, int32_t manaChange)
{
    const Position& targetPos = target->getPosition();
    if(manaChange > 0)
    {
        bool deny = false;
        CreatureEventList statsChangeEvents = target->getCreatureEvents(CREATURE_EVENT_STATSCHANGE);
        for(CreatureEventList::iterator it = statsChangeEvents.begin(); it != statsCh angeEvents.end(); ++it)
        {
            if(!(*it)->executeStatsChange(target, attacker, STATSCHANGE_MANAGAIN, COMBAT_HEALING, manaChange))
                deny = true;
        }

        if(deny)
            return false;

        if(g_config.getBool(ConfigManager::SHOW_HEALING_DAMAGE) && !target->isGhost() &&
        (g_config.getBool(ConfigManager::SHOW_HEALING_DAMAGE_MONSTER) || !target->getMonster()))
        {
            int32_t realHeal = manaChange;
            if((realHeal + target->getMana()) > target->getMaxMana())
                realHeal = (target->getMaxMana() - target->getMana());
            
            if(realHeal > 0)
            {
                char buffer[20];
                sprintf(buffer, "+%d", realHeal);
                
                const SpectatorVec& list = getSpectators(targetPos);
                addAnimatedText(list, targetPos, COLOR_DARKPURPLE, buffer);
            }
        }
        target->changeMana(manaChange);
    }
    else
    {
        const SpectatorVec& list = getSpectators(targetPos);
        if(!target->isAttackable() || Combat::canDoCombat(attacker, target) != RET_NOERROR)
        {
            addMagicEffect(list, targetPos, MAGIC_EFFECT_POFF);
            return false;
        }

        int32_t manaLoss = std::min(target->getMana(), -manaChange);
        BlockType_t blockType = target->blockHit(attacker, COMBAT_MANADRAIN, manaLoss);
        if(blockType != BLOCK_NONE)
        {
            addMagicEffect(list, targetPos, MAGIC_EFFECT_POFF);
            return false;
        }

        if(manaLoss > 0)
        {
            bool deny = false;
            CreatureEventList statsChangeEvents = target->getCreatureEvents(CREATURE_EVENT_STATSCHANGE);
            for(CreatureEventList::iterator it = statsChangeEvents.begin(); it != statsChangeEvents.end(); ++it)
            {
                if(!(*it)->executeStatsChange(target, attacker, STATSCHANGE_MANALOSS, COMBAT_UNDEFINEDDAMAGE, manaChange))
                    deny = true;
            }

            if(deny)
                return false;

            target->drainMana(attacker, COMBAT_MANADRAIN, manaLoss);
            char buffer[20];
            sprintf(buffer, "%d", manaLoss);

            addAnimatedText(list, targetPos, COLOR_BLUE, buffer);
        }
    }

    return true;
}
}

void Game::addCreatureHealth(const Creature* target)
{
    const SpectatorVec& list = getSpectators(target->getPosition());
    addCreatureHealth(list, target);
}

void Game::addCreatureHealth(const SpectatorVec& list, const Creature* target)
{
    Player* player = NULL;
    for(SpectatorVec::const_iterator it = list.begin(); it != list.end(); ++it)
    {
        if((player = (*it)->getPlayer()))
            player->sendCreatureHealth(target);
    }
}

void Game::addCreatureSquare(const Creature* target, uint8_t squareColor)
{
    const SpectatorVec& list = getSpectators(target->getPosition());
    addCreatureSquare(list, target, squareColor);
}

void Game::addCreatureSquare(const SpectatorVec& list, const Creature* target, uint8_t squareColor)
{
    Player* player = NULL;
    for(SpectatorVec::const_iterator it = list.begin(); it != list.end(); ++it)
    {
        if((player = (*it)->getPlayer()))
            player->sendCreatureSquare(target, squareColor);
    }
}

void Game::addAnimatedText(const Position& pos, uint8_t textColor, const std::string& text)
{
    const SpectatorVec& list = getSpectators(pos);
    addAnimatedText(list, pos, textColor, text);
}

void Game::addAnimatedText(const SpectatorVec& list, const Position& pos, uint8_t textColor,
    const std::string& text)
{
    Player* player = NULL;
    for(SpectatorVec::const_iterator it = list.begin(); it != list.end(); ++it)
    {
        if((player = (*it)->getPlayer()))
            player->sendAnimatedText(pos, textColor, text);
    }
}

void Game::addMagicEffect(const Position& pos, uint8_t effect, bool ghostMode/* = false*/)
{
    if(ghostMode)
        return;

    const SpectatorVec& list = getSpectators(pos);
    addMagicEffect(list, pos, effect);
}

void Game::addMagicEffect(const SpectatorVec& list, const Position& pos, uint8_t effect,
    bool ghostMode/* = false*/)
{
    if(ghostMode)
        return;

    Player* player = NULL;
    for(SpectatorVec::const_iterator it = list.begin(); it != list.end(); ++it)
    {
        if((player = (*it)->getPlayer()))
            player->sendMagicEffect(pos, effect);
    }
}

void Game::addDistanceEffect(const Position& fromPos, const Position& toPos, uint8_t effect)
{
    SpectatorVec list;
    getSpectators(list, fromPos, false);
    getSpectators(list, toPos, true);
    addDistanceEffect(list, fromPos, toPos, effect);
}

void Game::addDistanceEffect(const SpectatorVec& list, const Position& fromPos,
    const Position& toPos, uint8_t effect)
{
    Player* player = NULL;
    for(SpectatorVec::const_iterator it = list.begin(); it != list.end(); ++it)
    {
        if((player = (*it)->getPlayer()))
            player->sendDistanceShoot(fromPos, toPos, effect);
    }
}

void Game::startDecay(Item* item)
{
    if(!item || !item->canDecay() || item->getDecaying() == DECAYING_TRUE)
        return;

    if(item->getDuration() > 0)
    {
        item->addRef();
        item->setDecaying(DECAYING_TRUE);
        toDecayItems.push_back(item);
    }
    else
        internalDecayItem(item);
}

void Game::internalDecayItem(Item* item)
{
    const ItemType& it = Item::items.getItemType(item->getID());
    if(it.decayTo)
    {
        Item* newItem = transformItem(item, it.decayTo);
        startDecay(newItem);
    }
    else
    {
        ReturnValue ret = internalRemoveItem(NULL, item);
        if(ret != RET_NOERROR)
            std::clog << "> DEBUG: internalDecayItem failed, error code: " << (int32_t)ret << ", item id: " << item->getID() << std::endl;
    }
}

void Game::checkDecay()
{
    Scheduler::getInstance().addEvent(createSchedulerTask(EVENT_DECAYINTERVAL,
        boost::bind(&Game::checkDecay, this)));

    size_t bucket = (lastBucket + 1) % EVENT_DECAYBUCKETS;
    for(DecayList::iterator it = decayItems[bucket].begin(); it != decayItems[bucket].end();)
    {
        Item* item = *it;
        int32_t decreaseTime = EVENT_DECAYINTERVAL * EVENT_DECAYBUCKETS;
        if(item->getDuration() - decreaseTime < 0)
            decreaseTime = item->getDuration();

        item->decreaseDuration(decreaseTime);
        if(!item->canDecay())
        {
            item->setDecaying(DECAYING_FALSE);
            freeThing(item);
            it = decayItems[bucket].erase(it);
            continue;
        }

        int32_t dur = item->getDuration();
        if(dur <= 0)
        {
            it = decayItems[bucket].erase(it);
            internalDecayItem(item);
            freeThing(item);
        }
        else if(dur < EVENT_DECAYINTERVAL * EVENT_DECAYBUCKETS)
        {
            it = decayItems[bucket].erase(it);
            size_t newBucket = (bucket + ((dur + EVENT_DECAYINTERVAL / 2) / 1000)) % EVENT_DECAYBUCKETS;
            if(newBucket == bucket)
            {
                internalDecayItem(item);
                freeThing(item);
            }
            else
                decayItems[newBucket].push_back(item);
        }
        else
            ++it;
    }

    lastBucket = bucket;
    cleanup();
}

void Game::checkLight()
{
    Scheduler::getInstance().addEvent(createSchedulerTask(EVENT_LIGHTINTERVAL,
        boost::bind(&Game::checkLight, this)));

    lightHour = lightHour + lightHourDelta;
    if(lightHour > 1440)
        lightHour = lightHour - 1440;

    if(std::abs(lightHour - SUNRISE) < 2 * lightHourDelta)
        lightState = LIGHT_STATE_SUNRISE;
    else if(std::abs(lightHour - SUNSET) < 2 * lightHourDelta)
        lightState = LIGHT_STATE_SUNSET;

    int32_t newLightLevel = lightLevel;
    bool lightChange = false;
    switch(lightState)
    {
        case LIGHT_STATE_SUNRISE:
        {
            newLightLevel += (LIGHT_LEVEL_DAY - LIGHT_LEVEL_NIGHT) / 30;
            lightChange = true;
            break;
        }
        case LIGHT_STATE_SUNSET:
        {
            newLightLevel -= (LIGHT_LEVEL_DAY - LIGHT_LEVEL_NIGHT) / 30;
            lightChange = true;
            break;
        }
        default:
            break;
    }

    if(newLightLevel <= LIGHT_LEVEL_NIGHT)
    {
        lightLevel = LIGHT_LEVEL_NIGHT;
        lightState = LIGHT_STATE_NIGHT;
    }
    else if(newLightLevel >= LIGHT_LEVEL_DAY)
    {
        lightLevel = LIGHT_LEVEL_DAY;
        lightState = LIGHT_STATE_DAY;
    }
    else
        lightLevel = newLightLevel;

    if(lightChange)
    {
        LightInfo lightInfo;
        getWorldLightInfo(lightInfo);
        for(AutoList<Player>::iterator it = Player::autoList.begin(); it != Player::autoList.end(); ++it)
            it->second->sendWorldLight(lightInfo);
    }
}

void Game::getWorldLightInfo(LightInfo& lightInfo)
{
    lightInfo.level = lightLevel;
    lightInfo.color = 0xD7;
}

bool Game::cancelRuleViolation(Player* player)
{
    RuleViolationsMap::iterator it = ruleViolations.find(player->getID());
    if(it == ruleViolations.end())
        return false;

    Player* gamemaster = it->second->gamemaster;
    if(!it->second->isOpen && gamemaster) //Send to the responser
        gamemaster->sendRuleViolationCancel(player->getName());
    else if(ChatChannel* channel = g_chat.getChannelById(CHANNEL_RVR))
    {
        UsersMap tmpMap = channel->getUsers();
        for(UsersMap::iterator tit = tmpMap.begin(); tit != tmpMap.end(); ++tit)
            tit->second->sendRemoveReport(player->getName());
    }

    //Now erase it
    ruleViolations.erase(it);
    return true;
}

bool Game::closeRuleViolation(Player* player)
{
    RuleViolationsMap::iterator it = ruleViolations.find(player->getID());
    if(it == ruleViolations.end())
        return false;

    ruleViolations.erase(it);
    player->sendLockRuleViolation();
    if(ChatChannel* channel = g_chat.getChannelById(CHANNEL_RVR))
    {
        UsersMap tmpMap = channel->getUsers();
        for(UsersMap::iterator tit = tmpMap.begin(); tit != tmpMap.end(); ++tit)
            tit->second->sendRemoveReport(player->getName());
    }

    return true;
}

void Game::updateCreatureSkull(Creature* creature)
{
    const SpectatorVec& list = getSpectators(creature->getPosition());

    //send to client
    Player* tmpPlayer = NULL;
    for(SpectatorVec::const_iterator it = list.begin(); it != list.end(); ++it)
    {
         if((tmpPlayer = (*it)->getPlayer()))
            tmpPlayer->sendCreatureSkull(creature);
    }
}

void Game::updateCreatureShield(Creature* creature)
{
    const SpectatorVec& list = getSpectators(creature->getPosition());

    //send to client
    Player* tmpPlayer = NULL;
    for(SpectatorVec::const_iterator it = list.begin(); it != list.end(); ++it)
    {
        if((tmpPlayer = (*it)->getPlayer()))
            tmpPlayer->sendCreatureShield(creature);
    }
}

void Game::updateCreatureEmblem(Creature* creature)
{
    const SpectatorVec& list = getSpectators(creature->getPosition());

    //send to client
    Player* tmpPlayer = NULL;
    for(SpectatorVec::const_iterator it = list.begin(); it != list.end(); ++it)
    {
        if((tmpPlayer = (*it)->getPlayer()))
            tmpPlayer->sendCreatureEmblem(creature);
    }
}

void Game::updateCreatureImpassable(Creature* creature)
{
    const SpectatorVec& list = getSpectators(creature->getPosition());

    //send to client
    Player* tmpPlayer = NULL;
    for(SpectatorVec::const_iterator it = list.begin(); it != list.end(); ++it)
    {
        if((tmpPlayer = (*it)->getPlayer()))
            tmpPlayer->sendCreatureImpassable(creature);
    }
}

bool Game::playerInviteToParty(uint32_t playerId, uint32_t invitedId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    Player* invitedPlayer = getPlayerByID(invitedId);
    if(!invitedPlayer || invitedPlayer->isRemoved() || invitedPlayer->isInviting(player))
        return false;

    if(invitedPlayer->getParty())
    {
        char buffer[90];
        sprintf(buffer, "%s is already in a party.", invitedPlayer->getName().c_str());
        player->sendTextMessage(MSG_INFO_DESCR, buffer);
        return false;
    }

    Party* party = player->getParty();
    if(!party)
        party = new Party(player);
    else if(party->getLeader() != player)
        return false;

    return party->invitePlayer(invitedPlayer);
}

bool Game::playerJoinParty(uint32_t playerId, uint32_t leaderId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    Player* leader = getPlayerByID(leaderId);
    if(!leader || leader->isRemoved() || !leader->isInviting(player))
        return false;

    if(!player->getParty())
        return leader->getParty()->join(player);

    player->sendTextMessage(MSG_INFO_DESCR, "You are already in a party.");
    return false;
}

bool Game::playerRevokePartyInvitation(uint32_t playerId, uint32_t invitedId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved() || !player->getParty() || player->getParty()->getLeader() != player)
        return false;

    Player* invitedPlayer = getPlayerByID(invitedId);
    if(!invitedPlayer || invitedPlayer->isRemoved() || !player->isInviting(invitedPlayer))
        return false;

    player->getParty()->revokeInvitation(invitedPlayer);
    return true;
}

bool Game::playerPassPartyLeadership(uint32_t playerId, uint32_t newLeaderId)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved() || !player->getParty() || player->getParty()->getLeader() != player)
        return false;

    Player* newLeader = getPlayerByID(newLeaderId);
    if(!newLeader || newLeader->isRemoved() || !newLeader->getParty() || newLeader->getParty() != player->getParty())
        return false;

    return player->getParty()->passLeadership(newLeader);
}

bool Game::playerLeaveParty(uint32_t playerId, bool forced/* = false*/)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved() || !player->getParty() || (player->hasCondition(CONDITION_INFIGHT) && !forced))
        return false;

    return player->getParty()->leave(player);
}

bool Game::playerSharePartyExperience(uint32_t playerId, bool activate, uint8_t)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    if(!player->getParty() || (!player->hasFlag(PlayerFlag_NotGainInFight)
        && player->hasCondition(CONDITION_INFIGHT)))
        return false;

    return player->getParty()->setSharedExperience(player, activate);
}

bool Game::playerReportBug(uint32_t playerId, std::string comment)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    if(!player->hasFlag(PlayerFlag_CanReportBugs))
        return false;

    CreatureEventList reportBugEvents = player->getCreatureEvents(CREATURE_EVENT_REPORTBUG);
    for(CreatureEventList::iterator it = reportBugEvents.begin(); it != reportBugEvents.end(); ++it)
        (*it)->executeReportBug(player, comment);

    return true;
}

bool Game::playerViolationWindow(uint32_t playerId, std::string name, uint8_t reason, ViolationAction_t action,
    std::string comment, std::string statement, uint32_t statementId, bool ipBanishment)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return false;

    Group* group = player->getGroup();
    if(!group)
        return false;

    std::string::size_type start = comment.find("{"), end = std::string::npos;
    if(start != std::string::npos)
        end = comment.find("}", start + 1);

    time_t length[3] = {0, 0, 0};
    if(end != std::string::npos)
    {
        int32_t i = 0;
        StringVec vec = explodeString(comment.substr(start + 1, end - 1), ",");
        for(StringVec::iterator it = vec.begin(); it != vec.end() && i < 4; ++it, ++i)
        {
            if((*it) == "delete")
            {
                i--;
                action = ACTION_DELETION;
                continue;
            }

            time_t banTime = time(NULL);
            StringVec tec = explodeString((*it), "+");
            for(StringVec::iterator tit = tec.begin(); tit != tec.end(); ++tit)
            {
                std::string tmp = (*tit);
                if(tmp[0] != 's' && tmp[0] != 'm' && tmp[0] != 'h' && tmp[0] != 'd'
                    && tmp[0] != 'w' && tmp[0] != 'o' && tmp[0] != 'y')
                    continue;

                uint32_t count = 1;
                if(tmp.size() > 1)
                {
                    count = atoi(tmp.substr(1).c_str());
                    if(!count)
                        count = 1;
                }

                if(tmp[0] == 's')
                    banTime += count;

                if(tmp[0] == 'm')
                    banTime += count * 60;

                if(tmp[0] == 'h')
                    banTime += count * 3600;

                if(tmp[0] == 'd')
                    banTime += count * 86400;

                if(tmp[0] == 'w')
                    banTime += count * 604800;

                if(tmp[0] == 'o')
                    banTime += count * 2592000;

                if(tmp[0] == 'y')
                    banTime += count * 31536000;

                length = banTime;
            }
        }

        comment = comment.substr(end + 1);
    }

    int16_t nameFlags = group->getNameViolationFlags(), statementFlags = group->getStatementViolationFlags();
    if((ipBanishment && ((nameFlags & IPBAN_FLAG) != IPBAN_FLAG || (statementFlags & IPBAN_FLAG) != IPBAN_FLAG)) ||
        !(nameFlags & (1 << action) || statementFlags & (1 << action)) || reason > group->getViolationReasons())
    {
        player->sendCancel("You do not have authorization for this action.");
        return false;
    }

    uint32_t commentSize = g_config.getNumber(ConfigManager::MAX_VIOLATIONCOMMENT_SIZE);
    if(comment.size() > commentSize)
    {
        char buffer[90];
        sprintf(buffer, "The comment may not exceed limit of %d characters.", commentSize);

        player->sendCancel(buffer);
        return false;
    }

    toLowerCaseString(name);
    Player* target = getPlayerByNameEx(name);
    if(!target || name == "account manager")
    {
        player->sendCancel("A player with this name does not exist.");
        return false;
    }

    if(target->hasFlag(PlayerFlag_CannotBeBanned))
    {
        player->sendCancel("You do not have authorization for this action.");
        return false;
    }

    Account account = IOLoginData::getInstance()->loadAccount(target->getAccount(), true);
    enum KickAction {
        NONE = 1,
        KICK = 2,
        FULL_KICK = 3,
    } kickAction = FULL_KICK;
    switch(action)
    {
        case ACTION_STATEMENT:
        {
            StatementMap::iterator it = g_chat.statementMap.find(statementId);
            if(it == g_chat.statementMap.end())
            {
                player->sendCancel("Statement has been already reported.");
                return false;
            }

            IOBan::getInstance()->addStatement(target->getGUID(), reason, comment,
                player->getGUID(), -1, statement);
            g_chat.statementMap.erase(it);

            kickAction = NONE;
            break;
        }

        case ACTION_NAMEREPORT:
        {
            int64_t banTime = -1;
            PlayerBan_t tmp = (PlayerBan_t)g_config.getNumber(ConfigManager::NAME_REPORT_TYPE);
            if(tmp == PLAYERBAN_BANISHMENT)
            {
                if(!length[0])
                    banTime = time(NULL) + g_config.getNumber(ConfigManager::BAN_LENGTH);
                else
                    banTime = length[0];
            }

            if(!IOBan::getInstance()->addPlayerBanishment(target->getGUID(), banTime, reason, action,
                comment, player->getGUID(), tmp))
            {
                player->sendCancel("Player has been already reported.");
                return false;
            }
            else if(tmp == PLAYERBAN_BANISHMENT)
                account.warnings++;

            kickAction = (KickAction)tmp;
            break;
        }

        case ACTION_NOTATION:
        case ACTION_BANISHMENT:
        case ACTION_BANREPORT:
        {
            if(action == ACTION_NOTATION)
            {
                if(!IOBan::getInstance()->addNotation(account.number, reason,
                    comment, player->getGUID(), target->getGUID()))
                {
                    player->sendCancel("Unable to perform action.");
                    return false;
                }

                if(IOBan::getInstance()->getNotationsCount(account.number) < (uint32_t)
                    g_config.getNumber(ConfigManager::NOTATIONS_TO_BAN))
                {
                    kickAction = NONE;
                    break;
                }

                action = ACTION_BANISHMENT;
            }

            bool deny = action != ACTION_BANREPORT;
            int64_t banTime = -1;

            account.warnings++;
            if(account.warnings >= g_config.getNumber(ConfigManager::WARNINGS_TO_DELETION))
                action = ACTION_DELETION;
            else if(length[0])
                banTime = length[0];
            else if(account.warnings >= g_config.getNumber(ConfigManager::WARNINGS_TO_FINALBAN))
                banTime = time(NULL) + g_config.getNumber(ConfigManager::FINALBAN_LENGTH);
            else
                banTime = time(NULL) + g_config.getNumber(ConfigManager::BAN_LENGTH);

            if(!IOBan::getInstance()->addAccountBanishment(account.number, banTime, reason, action,
                comment, player->getGUID(), target->getGUID()))
            {
                account.warnings--;
                player->sendCancel("Account is already banned.");
                return false;
            }

            if(deny)
                break;

            banTime = -1;
            PlayerBan_t tmp = (PlayerBan_t)g_config.getNumber(ConfigManager::NAME_REPORT_TYPE);
            if(tmp == PLAYERBAN_BANISHMENT)
            {
                if(!length[1])
                    banTime = time(NULL) + g_config.getNumber(ConfigManager::FINALBAN_LENGTH);
                else
                    banTime = length[1];
            }

            IOBan::getInstance()->addPlayerBanishment(target->getGUID(), banTime, reason, action, comment,
                player->getGUID(), tmp);
            break;
        }

        case ACTION_BANFINAL:
        case ACTION_BANREPORTFINAL:
        {
            bool allow = action == ACTION_BANREPORTFINAL;
            int64_t banTime = -1;

            account.warnings++;
            if(account.warnings >= g_config.getNumber(ConfigManager::WARNINGS_TO_DELETION))
                action = ACTION_DELETION;
            else if(length[0])
                banTime = length[0];
            else
                banTime = time(NULL) + g_config.getNumber(ConfigManager::FINALBAN_LENGTH);

            if(!IOBan::getInstance()->addAccountBanishment(account.number, banTime, reason, action,
                comment, player->getGUID(), target->getGUID()))
            {
                account.warnings--;
                player->sendCancel("Account is already banned.");
                return false;
            }

            if(action != ACTION_DELETION)
                account.warnings += (g_config.getNumber(ConfigManager::WARNINGS_TO_FINALBAN) - 1);

            if(allow)
                IOBan::getInstance()->addPlayerBanishment(target->getGUID(), -1, reason, action, comment,
                    player->getGUID(), (PlayerBan_t)g_config.getNumber(
                    ConfigManager::NAME_REPORT_TYPE));

            break;
        }

        case ACTION_DELETION:
        {
            //completely internal
            account.warnings++;
            if(!IOBan::getInstance()->addAccountBanishment(account.number, -1, reason, ACTION_DELETION,
                comment, player->getGUID(), target->getGUID()))
            {
                account.warnings--;
                player->sendCancel("Account is currently banned or already deleted.");
                return false;
            }

            break;
        }

        default:
            // these just shouldn't occur in rvw
            return false;
    }

    if(ipBanishment && target->getIP())
    {
        if(!length[2])
            length[2] = time(NULL) + g_config.getNumber(ConfigManager::IPBANISHMENT_LENGTH);

        IOBan::getInstance()->addIpBanishment(target->getIP(), length[2], reason, comment, player->getGUID(), 0xFFFFFFFF);
    }

    if(kickAction == FULL_KICK)
        IOBan::getInstance()->removeNotations(account.number);

    std::stringstream ss;
    if(g_config.getBool(ConfigManager::BROADCAST_BANISHMENTS))
        ss << player->getName() << " has";
    else
        ss << "You have";

    ss << " taken the action \"" << getAction(action, ipBanishment) << "\"";
    switch(action)
    {
        case ACTION_NOTATION:
        {
            ss << " (" << (g_config.getNumber(ConfigManager::NOTATIONS_TO_BAN) - IOBan::getInstance()->getNotationsCount(
                account.number)) << " left to banishment)";
            break;
        }
        case ACTION_STATEMENT:
        {
            ss << " for the statement: \"" << statement << "\"";
            break;
        }
        default:
            break;
    }

    ss << " against: " << name << " (Warnings: " << account.warnings << "), with reason: \"" << getReason(
        reason) << "\", and comment: \"" << comment << "\".";
    if(g_config.getBool(ConfigManager::BROADCAST_BANISHMENTS))
        broadcastMessage(ss.str(), MSG_STATUS_WARNING);
    else
        player->sendTextMessage(MSG_STATUS_CONSOLE_RED, ss.str());

    if(target->isVirtual())
    {
        delete target;
        target = NULL;
    }
    else if(kickAction > NONE)
    {
        char buffer[30];
        sprintf(buffer, "You have been %s.", (kickAction > KICK ? "banished" : "namelocked"));
        target->sendTextMessage(MSG_INFO_DESCR, buffer);

        addMagicEffect(target->getPosition(), MAGIC_EFFECT_WRAPS_GREEN);
        Scheduler::getInstance().addEvent(createSchedulerTask(1000, boost::bind(
            &Game::kickPlayer, this, target->getID(), false)));
    }

    IOLoginData::getInstance()->saveAccount(account);
    return true;
}

void Game::kickPlayer(uint32_t playerId, bool displayEffect)
{
    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return;

    player->kickPlayer(displayEffect, true);
}

bool Game::broadcastMessage(const std::string& text, MessageClasses type)
{
    if(type < MSG_CLASS_FIRST || type > MSG_CLASS_LAST)
        return false;

    std::clog << "> Broadcasted message: \"" << text << "\"." << std::endl;
    for(AutoList<Player>::iterator it = Player::autoList.begin(); it != Player::autoList.end(); ++it)
        it->second->sendTextMessage(type, text);

    return true;
}

Position Game::getClosestFreeTile(Creature* creature, Position pos, bool extended/* = false*/, bool ignoreHouse/* = true*/)
{
    PairVector relList;
    relList.push_back(PositionPair(0, 0));
    relList.push_back(PositionPair(-1, -1));
    relList.push_back(PositionPair(-1, 0));
    relList.push_back(PositionPair(-1, 1));
    relList.push_back(PositionPair(0, -1));
    relList.push_back(PositionPair(0, 1));
    relList.push_back(PositionPair(1, -1));
    relList.push_back(PositionPair(1, 0));
    relList.push_back(PositionPair(1, 1));

    if(extended)
    {
        relList.push_back(PositionPair(-2, 0));
        relList.push_back(PositionPair(0, -2));
        relList.push_back(PositionPair(0, 2));
        relList.push_back(PositionPair(2, 0));
    }

    std::random_shuffle(relList.begin() + 1, relList.end());
    if(Player* player = creature->getPlayer())
    {
        for(PairVector::iterator it = relList.begin(); it != relList.end(); ++it)
        {
            Tile* tile = map->getTile(Position((pos.x + it->first), (pos.y + it->second), pos.z));
            if(!tile || !tile->ground)
                continue;

            ReturnValue ret = tile->__queryAdd(0, player, 1, FLAG_IGNOREBLOCKITEM);
            if(ret == RET_NOTENOUGHROOM || (ret == RET_NOTPOSSIBLE && !player->hasCustomFlag(PlayerCustomFlag_CanMoveAnywhere))
                || (ret == RET_PLAYERISNOTINVITED && !ignoreHouse && !player->hasFlag(PlayerFlag_CanEditHouses)))
                continue;

            return tile->getPosition();
        }
    }
    else
    {
        for(PairVector::iterator it = relList.begin(); it != relList.end(); ++it)
        {
            Tile* tile = NULL;
            if((tile = map->getTile(Position((pos.x + it->first), (pos.y + it->second), pos.z)))
                && tile->__queryAdd(0, creature, 1, FLAG_IGNOREBLOCKITEM) == RET_NOERROR)
                return tile->getPosition();
        }
    }

    return Position(0, 0, 0);
}

std::string Game::getSearchString(const Position& fromPos, const Position& toPos, bool fromIsCreature/* = false*/, bool toIsCreature/* = false*/)
{
    /*
     * When the position is on same level and 0 to 4 squares away, they are "[toIsCreature: standing] next to you"
     * When the position is on same level and 5 to 100 squares away they are "to the north/west/south/east."
     * When the position is on any level and 101 to 274 squares away they are "far to the north/west/south/east."
     * When the position is on any level and 275+ squares away they are "very far to the north/west/south/east."
     * When the position is not directly north/west/south/east of you they are "((very) far) to the north-west/south-west/south-east/north-east."
     * When the position is on a lower or higher level and 5 to 100 squares away they are "on a lower (or) higher level to the north/west/south/east."
     * When the position is on a lower or higher level and 0 to 4 squares away they are "below (or) above you."
     */

    enum distance_t
    {
        DISTANCE_BESIDE,
        DISTANCE_CLOSE,
        DISTANCE_FAR,
        DISTANCE_VERYFAR
    };

    enum direction_t
    {
        DIR_N, DIR_S, DIR_E, DIR_W,
        DIR_NE, DIR_NW, DIR_SE, DIR_SW
    };

    enum level_t
    {
        LEVEL_HIGHER,
        LEVEL_LOWER,
        LEVEL_SAME
    };

    distance_t distance;
    direction_t direction;
    level_t level;

    int32_t dx = fromPos.x - toPos.x, dy = fromPos.y - toPos.y, dz = fromPos.z - toPos.z;
    if(dz > 0)
        level = LEVEL_HIGHER;
    else if(dz < 0)
        level = LEVEL_LOWER;
    else
        level = LEVEL_SAME;

    if(std::abs(dx) < 5 && std::abs(dy) < 5)
        distance = DISTANCE_BESIDE;
    else
    {
        int32_t tmp = dx * dx + dy * dy;
        if(tmp < 10000)
            distance = DISTANCE_CLOSE;
        else if(tmp < 75625)
            distance = DISTANCE_FAR;
        else
            distance = DISTANCE_VERYFAR;
    }

    float tan;
    if(dx != 0)
        tan = (float)dy / (float)dx;
    else
        tan = 10.;

    if(std::abs(tan) < 0.4142)
    {
        if(dx > 0)
            direction = DIR_W;
        else
            direction = DIR_E;
    }
    else if(std::abs(tan) < 2.4142)
    {
        if(tan > 0)
        {
            if(dy > 0)
                direction = DIR_NW;
            else
                direction = DIR_SE;
        }
        else
        {
            if(dx > 0)
                direction = DIR_SW;
            else
                direction = DIR_NE;
        }
    }
    else
    {
        if(dy > 0)
            direction = DIR_N;
        else
            direction = DIR_S;
    }

    std::stringstream ss;
    switch(distance)
    {
        case DISTANCE_BESIDE:
        {
            switch(level)
            {
                case LEVEL_SAME:
                {
                    ss << "is ";
                    if(toIsCreature)
                        ss << "standing ";

                    ss << "next to you";
                    break;
                }

                case LEVEL_HIGHER:
                {
                    ss << "is above ";
                    if(fromIsCreature)
                        ss << "you";

                    break;
                }

                case LEVEL_LOWER:
                {
                    ss << "is below ";
                    if(fromIsCreature)
                        ss << "you";

                    break;
                }

                default:
                    break;
            }

            break;
        }

        case DISTANCE_CLOSE:
        {
            switch(level)
            {
                case LEVEL_SAME:
                    ss << "is to the";
                    break;
                case LEVEL_HIGHER:
                    ss << "is on a higher level to the";
                    break;
                case LEVEL_LOWER:
                    ss << "is on a lower level to the";
                    break;
                default:
                    break;
            }

            break;
        }

        case DISTANCE_FAR:
            ss << "is far to the";
            break;

        case DISTANCE_VERYFAR:
            ss << "is very far to the";
            break;

        default:
            break;
    }

    if(distance != DISTANCE_BESIDE)
    {
        ss << " ";
        switch(direction)
        {
            case DIR_N:
                ss << "north";
                break;

            case DIR_S:
                ss << "south";
                break;

            case DIR_E:
                ss << "east";
                break;

            case DIR_W:
                ss << "west";
                break;

            case DIR_NE:
                ss << "north-east";
                break;

            case DIR_NW:
                ss << "north-west";
                break;

            case DIR_SE:
                ss << "south-east";
                break;

            case DIR_SW:
                ss << "south-west";
                break;

            default:
                break;
        }
    }

    return ss.str();
}

double Game::getExperienceStage(uint32_t level, double divider/* = 1.*/)
{
    if(!g_config.getBool(ConfigManager::EXPERIENCE_STAGES))
        return g_config.getDouble(ConfigManager::RATE_EXPERIENCE) * divider;

    if(lastStageLevel && level >= lastStageLevel)
        return stages[lastStageLevel] * divider;

    return stages[level] * divider;
}

bool Game::loadExperienceStages()
{
    if(!g_config.getBool(ConfigManager::EXPERIENCE_STAGES))
        return true;

    xmlDocPtr doc = xmlParseFile(getFilePath(FILE_TYPE_XML, "stages.xml").c_str());
    if(!doc)
    {
        std::clog << "[Warning - Game::loadExperienceStages] Cannot load stages file." << std::endl;
        std::clog << getLastXMLError() << std::endl;
        return false;
    }

    xmlNodePtr q, p, root = xmlDocGetRootElement(doc);
    if(xmlStrcmp(root->name, (const xmlChar*)"stages"))
    {
        std::clog << "[Error - Game::loadExperienceStages] Malformed stages file" << std::endl;
        xmlFreeDoc(doc);
        return false;
    }

    int32_t intValue, low = 0, high = 0;
    float floatValue, mul = 1.0f, defStageMultiplier;
    std::string strValue;

    lastStageLevel = 0;
    stages.clear();

    q = root->children;
    while(q)
    {
        if(!xmlStrcmp(q->name, (const xmlChar*)"world"))
        {
            if(readXMLString(q, "id", strValue))
            {
                IntegerVec intVector;
                if(!parseIntegerVec(strValue, intVector) || std::find(intVector.begin(),
                    intVector.end(), g_config.getNumber(ConfigManager::WORLD_ID)) == intVector.end())
                {
                    q = q->next;
                    continue;
                }
            }

            defStageMultiplier = 1.0f;
            if(readXMLFloat(q, "multiplier", floatValue))
                defStageMultiplier = floatValue;

            p = q->children;
            while(p)
            {
                if(!xmlStrcmp(p->name, (const xmlChar*)"stage"))
                {
                    low = 1;
                    if(readXMLInteger(p, "minlevel", intValue) || readXMLInteger(p, "minLevel", intValue))
                        low = intValue;

                    high = 0;
                    if(readXMLInteger(p, "maxlevel", intValue) || readXMLInteger(p, "maxLevel", intValue))
                        high = intValue;
                    else
                        lastStageLevel = low;

                    mul = 1.0f;
                    if(readXMLFloat(p, "multiplier", floatValue))
                        mul = floatValue;

                    mul *= defStageMultiplier;
                    if(lastStageLevel && lastStageLevel == (uint32_t)low)
                        stages[lastStageLevel] = mul;
                    else
                    {
                        for(int32_t i = low; i <= high; i++)
                            stages = mul;
                    }
                }

                p = p->next;
            }
        }

        if(!xmlStrcmp(q->name, (const xmlChar*)"stage"))
        {
            low = 1;
            if(readXMLInteger(q, "minlevel", intValue))
                low = intValue;
            else

            high = 0;
            if(readXMLInteger(q, "maxlevel", intValue))
                high = intValue;
            else
                lastStageLevel = low;

            mul = 1.0f;
            if(readXMLFloat(q, "multiplier", floatValue))
                mul = floatValue;

            if(lastStageLevel && lastStageLevel == (uint32_t)low)
                stages[lastStageLevel] = mul;
            else
            {
                for(int32_t i = low; i <= high; i++)
                    stages = mul;
            }
        }

        q = q->next;
    }

    xmlFreeDoc(doc);
    return true;
}

bool Game::reloadHighscores()
{
    lastHighscoreCheck = time(NULL);
    for(int16_t i = 0; i < 9; ++i)
        highscoreStorage = getHighscore(i);

    return true;
}

void Game::checkHighscores()
{
    reloadHighscores();
    uint32_t tmp = g_config.getNumber(ConfigManager::HIGHSCORES_UPDATETIME) * 60 * 1000;
    if(tmp <= 0)
        return;

    Scheduler::getInstance().addEvent(createSchedulerTask(tmp, boost::bind(&Game::checkHighscores, this)));
}

std::string Game::getHighscoreString(uint16_t skill)
{
    Highscore hs = highscoreStorage[skill];
    std::stringstream ss;
    ss << "Highscore for " << getSkillName(skill) << "\n\nRank Level - Player Name";
    for(uint32_t i = 0; i < hs.size(); i++)
        ss << "\n" << (i + 1) << ".  " << hs.second << "  -  " << hs.first;

    ss << "\n\nLast updated on:\n" << std::ctime(&lastHighscoreCheck);
    return ss.str();
}

Highscore Game::getHighscore(uint16_t skill)
{
    Highscore hs;

    Database* db = Database::getInstance();
    DBResult* result;

    DBQuery query;
    if(skill >= SKILL__MAGLEVEL)
    {
        if(skill == SKILL__MAGLEVEL)
            query << "SELECT `maglevel`, `name` FROM `players` ORDER BY `maglevel` DESC, `manaspent` DESC LIMIT " << g_config.getNumber(ConfigManager::HIGHSCORES_TOP);
        else
            query << "SELECT `level`, `name` FROM `players` ORDER BY `level` DESC, `experience` DESC LIMIT " << g_config.getNumber(ConfigManager::HIGHSCORES_TOP);

        if(!(result = db->storeQuery(query.str())))
            return hs;

        do
        {
            uint32_t level;
            if(skill == SKILL__MAGLEVEL)
                level = result->getDataInt("maglevel");
            else
                level = result->getDataInt("level");

            std::string name = result->getDataString("name");
            if(name.length() > 0)
                hs.push_back(std::make_pair(name, level));
        }
        while(result->next());
        result->free();
    }
    else
    {
        query << "SELECT `player_skills`.`value`, `players`.`name` FROM `player_skills`,`players` WHERE `player_skills`.`skillid`=" << skill << " AND `player_skills`.`player_id`=`players`.`id` ORDER BY `player_skills`.`value` DESC, `player_skills`.`count` DESC LIMIT " << g_config.getNumber(ConfigManager::HIGHSCORES_TOP);
        if(!(result = db->storeQuery(query.str())))
            return hs;

        do
        {
            std::string name = result->getDataString("name");
            if(name.length() > 0)
                hs.push_back(std::make_pair(name, result->getDataInt("value")));
        }
        while(result->next());
        result->free();
    }

    return hs;
}

int32_t Game::getMotdId()
{
    if(lastMotd == g_config.getString(ConfigManager::MOTD))
        return lastMotdId;

    lastMotd = g_config.getString(ConfigManager::MOTD);
    Database* db = Database::getInstance();

    DBQuery query;
    query << "INSERT INTO `server_motd` (`id`, `world_id`, `text`) VALUES (" << ++lastMotdId << ", " << g_config.getNumber(ConfigManager::WORLD_ID) << ", " << db->escapeString(lastMotd) << ")";
    if(db->query(query.str()))
        return lastMotdId;

    return --lastMotdId;
}

void Game::loadMotd()
{
    Database* db = Database::getInstance();
    DBQuery query;
    query << "SELECT `id`, `text` FROM `server_motd` WHERE `world_id` = " << g_config.getNumber(ConfigManager::WORLD_ID) << " ORDER BY `id` DESC LIMIT 1";

    DBResult* result;
    if(!(result = db->storeQuery(query.str())))
    {
        std::clog << "> ERROR: Failed to load motd!" << std::endl;
        lastMotdId = random_range(5, 500);
        return;
    }

    lastMotdId = result->getDataInt("id");
    lastMotd = result->getDataString("text");
    result->free();
}

void Game::checkPlayersRecord(Player* player)
{
    uint32_t count = getPlayersOnline();
    if(count <= playersRecord)
        return;

    GlobalEventMap recordEvents = g_globalEvents->getEventMap(GLOBALEVENT_RECORD);
    for(GlobalEventMap::iterator it = recordEvents.begin(); it != recordEvents.end(); ++it)
        it->second->executeRecord(count, playersRecord, player);

    playersRecord = count;
}

void Game::loadPlayersRecord()
{
    Database* db = Database::getInstance();
    DBQuery query;
    query << "SELECT `record` FROM `server_record` WHERE `world_id` = " << g_config.getNumber(ConfigManager::WORLD_ID) << " ORDER BY `timestamp` DESC LIMIT 1";

    DBResult* result;
    if(!(result = db->storeQuery(query.str())))
    {
        std::clog << "> ERROR: Failed to load players record!" << std::endl;
        return;
    }

    playersRecord = result->getDataInt("record");
    result->free();
}

bool Game::reloadInfo(ReloadInfo_t reload, uint32_t playerId/* = 0*/)
{
    bool done = false;
    switch(reload)
    {
        case RELOAD_ACTIONS:
        {
            if(g_actions->reload())
                done = true;
            else
                std::clog << "[Error - Game::reloadInfo] Failed to reload actions." << std::endl;

            break;
        }

        case RELOAD_CHAT:
        {
            if(g_chat.reload())
                done = true;
            else
                std::clog << "[Error - Game::reloadInfo] Failed to reload chat." << std::endl;

            break;
        }

        case RELOAD_CONFIG:
        {
            if(g_config.reload())
                done = true;
            else
                std::clog << "[Error - Game::reloadInfo] Failed to reload config." << std::endl;

            break;
        }

        case RELOAD_CREATUREEVENTS:
        {
            if(g_creatureEvents->reload())
                done = true;
            else
                std::clog << "[Error - Game::reloadInfo] Failed to reload creature events." << std::endl;

            break;
        }

        case RELOAD_GAMESERVERS:
        {
            #ifdef __LOGIN_SERVER__
            if(GameServers::getInstance()->reload())
                done = true;
            else
                std::clog << "[Error - Game::reloadInfo] Failed to reload game servers." << std::endl;

            #endif
            break;
        }

        case RELOAD_GLOBALEVENTS:
        {
            if(g_globalEvents->reload())
                done = true;
            else
                std::clog << "[Error - Game::reloadInfo] Failed to reload global events." << std::endl;

            break;
        }

        case RELOAD_GROUPS:
        {
            /*if(Groups::getInstance()->reload())
                done = true;
            else
                std::clog << "[Error - Game::reloadInfo] Failed to reload groups." << std::endl;*/

            break;
        }

        case RELOAD_HIGHSCORES:
        {
            if(reloadHighscores())
                done = true;
            else
                std::clog << "[Error - Game::reloadInfo] Failed to reload highscores." << std::endl;

            break;
        }

        case RELOAD_ITEMS:
        {
            //TODO
            std::clog << "[Notice - Game::reloadInfo] Reload type does not work." << std::endl;
            done = true;
            break;
        }

        case RELOAD_MODS:
        {
            if(ScriptManager::getInstance()->reloadMods())
                done = true;
            else
                std::clog << "[Error - Game::reloadInfo] Failed to reload mods." << std::endl;

            break;
        }

        case RELOAD_MONSTERS:
        {
            if(g_monsters.reload())
                done = true;
            else
                std::clog << "[Error - Game::reloadInfo] Failed to reload monsters." << std::endl;

            break;
        }

        case RELOAD_MOVEEVENTS:
        {
            if(g_moveEvents->reload())
                done = true;
            else
                std::clog << "[Error - Game::reloadInfo] Failed to reload move events." << std::endl;

            break;
        }

        case RELOAD_NPCS:
        {
            g_npcs.reload();
            done = true;
            break;
        }

        case RELOAD_OUTFITS:
        {
            //TODO
            std::clog << "[Notice - Game::reloadInfo] Reload type does not work." << std::endl;
            done = true;
            break;
        }

        case RELOAD_QUESTS:
        {
            if(Quests::getInstance()->reload())
                done = true;
            else
                std::clog << "[Error - Game::reloadInfo] Failed to reload quests." << std::endl;

            break;
        }

        case RELOAD_RAIDS:
        {
            if(!Raids::getInstance()->reload())
                std::clog << "[Error - Game::reloadInfo] Failed to reload raids." << std::endl;
            else if(!Raids::getInstance()->startup())
                std::clog << "[Error - Game::reloadInfo] Failed to startup raids when reloading." << std::endl;
            else
                done = true;

            break;
        }

        case RELOAD_SPELLS:
        {
            if(!g_spells->reload())
                std::clog << "[Error - Game::reloadInfo] Failed to reload spells." << std::endl;
            else if(!g_monsters.reload())
                std::clog << "[Error - Game::reloadInfo] Failed to reload monsters when reloading spells." << std::endl;
            else
                done = true;

            break;
        }

        case RELOAD_STAGES:
        {
            if(loadExperienceStages())
                done = true;
            else
                std::clog << "[Error - Game::reloadInfo] Failed to reload stages." << std::endl;

            break;
        }

        case RELOAD_TALKACTIONS:
        {
            if(g_talkActions->reload())
                done = true;
            else
                std::clog << "[Error - Game::reloadInfo] Failed to reload talk actions." << std::endl;

            break;
        }

        case RELOAD_VOCATIONS:
        {
            if(Vocations::getInstance()->reload())
                done = true;
            else
                std::clog << "[Notice - Game::reloadInfo] Reload type does not work." << std::endl;

            break;
        }

        case RELOAD_WEAPONS:
        {
            //TODO
            std::clog << "[Notice - Game::reloadInfo] Reload type does not work." << std::endl;
            done = true;
            break;
        }

        case RELOAD_ALL:
        {
            done = true;
            for(int32_t i = RELOAD_FIRST; i <= RELOAD_LAST; i++)
            {
                if(!reloadInfo((ReloadInfo_t)i) && done)
                    done = false;
            }

            break;
        }

        default:
        {
            std::clog << "[Warning - Game::reloadInfo] Reload type not found." << std::endl;
            break;
        }
    }

    if(reload != RELOAD_MODS && !ScriptManager::getInstance()->reloadMods())
        std::clog << "[Error - Game::reloadInfo] Failed to reload mods." << std::endl;

    if(!playerId)
        return done;

    Player* player = getPlayerByID(playerId);
    if(!player || player->isRemoved())
        return done;

    if(done)
    {
        player->sendTextMessage(MSG_STATUS_CONSOLE_BLUE, "Reloaded successfully.");
        return true;
    }

    if(reload == RELOAD_ALL)
        player->sendTextMessage(MSG_STATUS_CONSOLE_BLUE, "Failed to reload some parts.");
    else
        player->sendTextMessage(MSG_STATUS_CONSOLE_BLUE, "Failed to reload.");

    return false;
}

void Game::prepareGlobalSave()
{
    if(!globalSaveMessage[0])
    {
        setGameState(GAMESTATE_CLOSING);
        globalSaveMessage[0] = true;

        broadcastMessage("Server is going down for a global save within 5 minutes. Please logout.", MSG_STATUS_WARNING);
        Scheduler::getInstance().addEvent(createSchedulerTask(120000, boost::bind(&Game::prepareGlobalSave, this)));
    }
    else if(!globalSaveMessage[1])
    {
        globalSaveMessage[1] = true;
        broadcastMessage("Server is going down for a global save within 3 minutes. Please logout.", MSG_STATUS_WARNING);
        Scheduler::getInstance().addEvent(createSchedulerTask(120000, boost::bind(&Game::prepareGlobalSave, this)));
    }
    else if(!globalSaveMessage[2])
    {
        globalSaveMessage[2] = true;
        broadcastMessage("Server is going down for a global save in one minute, please logout!", MSG_STATUS_WARNING);
        Scheduler::getInstance().addEvent(createSchedulerTask(60000, boost::bind(&Game::prepareGlobalSave, this)));
    }
    else
        globalSave();
}

void Game::globalSave()
{
    bool close = g_config.getBool(ConfigManager::SHUTDOWN_AT_GLOBALSAVE);
    if(close) // check are we're going to close the server
        setGameState(GAMESTATE_CLOSING);

    // call the global event
    g_globalEvents->execute(GLOBALEVENT_GLOBALSAVE);
    if(close)
    {
        //shutdown server
        Dispatcher::getInstance().addTask(createTask(boost::bind(&Game::setGameState, this, GAMESTATE_SHUTDOWN)));
        return;
    }

    //close server
    Dispatcher::getInstance().addTask(createTask(boost::bind(&Game::setGameState, this, GAMESTATE_CLOSED)));
    //clean map if configured to
    if(g_config.getBool(ConfigManager::CLEAN_MAP_AT_GLOBALSAVE))
        cleanMap();

    //pay houses
    Houses::getInstance()->payHouses();
    //clear temporial and expired bans
    IOBan::getInstance()->clearTemporials();
    //remove premium days globally if configured to
    if(g_config.getBool(ConfigManager::INIT_PREMIUM_UPDATE))
        IOLoginData::getInstance()->updatePremiumDays();

    //reload everything
    reloadInfo(RELOAD_ALL);
    //reset variables
    for(int16_t i = 0; i < 3; i++)
        setGlobalSaveMessage(i, false);

    //prepare for next global save after 24 hours
    Scheduler::getInstance().addEvent(createSchedulerTask(86100000, boost::bind(&Game::prepareGlobalSave, this)));
    //open server
    Dispatcher::getInstance().addTask(createTask(boost::bind(&Game::setGameState, this, GAMESTATE_NORMAL)));
}

void Game::shutdown()
{
    std::clog << "Preparing";
    Scheduler::getInstance().shutdown();
    std::clog << " to";
    Dispatcher::getInstance().shutdown();
    std::clog << " shutdown";
    Spawns::getInstance()->clear();
    std::clog << " the";
    Raids::getInstance()->clear();
    std::clog << " server";
    cleanup();
    std::clog << "- done." << std::endl;
    if(services)
        services->stop();
#ifndef __DONT_FORCE_SHUTDOWN__

    exit(-1);
#endif
}

void Game::cleanup()
{
    //free memory
    for(std::vector<Thing*>::iterator it = releaseThings.begin(); it != releaseThings.end(); ++it)
        (*it)->unRef();

    releaseThings.clear();
    for(DecayList::iterator it = toDecayItems.begin(); it != toDecayItems.end(); ++it)
    {
        int32_t dur = (*it)->getDuration();
        if(dur >= EVENT_DECAYINTERVAL * EVENT_DECAYBUCKETS)
            decayItems[lastBucket].push_back(*it);
        else
            decayItems[(lastBucket + 1 + (*it)->getDuration() / 1000) % EVENT_DECAYBUCKETS].push_back(*it);
    }

    toDecayItems.clear();
}

void Game::freeThing(Thing* thing)
{
    releaseThings.push_back(thing);
}

void Game::showHotkeyUseMessage(Player* player, Item* item)
{
    const ItemType& it = Item::items[item->getID()];
    uint32_t count = player->__getItemTypeCount(item->getID(), -1);

    char buffer[40 + it.name.size()];
    if(count == 1)
        sprintf(buffer, "Using the last %s...", it.name.c_str());
    else
        sprintf(buffer, "Using one of %d %s...", count, it.pluralName.c_str());

    player->sendTextMessage(MSG_INFO_DESCR, buffer);
}
 


 

Acredito eu que vc vai conseguir mecher nisso pelo arquivo GAME.CPP, abraço.

Editado por Fabio Leandro (veja o histórico de edições)
Link para o post
Compartilhar em outros sites

Poderia postar um vídeo de como acontece seu sistema atual de healing?

E posta seu luascript.cpp

 

 

Testa o game.cpp, eu fiz alterações quanto ao número de healing:

game.cpp

Editado por FlavioHulk (veja o histórico de edições)
Link para o post
Compartilhar em outros sites

Olá irmão, eu lembro que nessa tentativa flustrada eu acabei mudando a cor do healing (com potions) pra azul, ou seja, toda vez que uso qualquer potion, de mana potion a ultimate sai azul. E quando healo usando magia, por exemplo, um exura ou exana mort, sai verde e mostra o total que esta healando, vou deixa imagens.

HEALANDO USANDO MAGIA:

image.thumb.png.76492db012ae84b64990b66bfdafbd4c.png

HEALANDO USANDO QUALQUER POTION (SAI EM AZUL) VOU DEMONSTRAR COM ULTIMATE HEAL:

image.thumb.png.40ae1d664f7f012dad4c8cdde87fe4b6.png

 

LUASCRIPT.CPP:
 

Spoiler

////////////////////////////////////////////////////////////////////////
// OpenTibia - an opensource roleplaying game
////////////////////////////////////////////////////////////////////////
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program.  If not, see <http://www.gnu.org/licenses/>.
////////////////////////////////////////////////////////////////////////
#include "otpch.h"
#include "luascript.h"
#include "scriptmanager.h"
#include <boost/filesystem.hpp>
#include <boost/any.hpp>
#include <iostream>
#include <iomanip>
#include "player.h"
#include "item.h"
#include "teleport.h"
#include "beds.h"
#include "town.h"
#include "house.h"
#include "housetile.h"
#include "database.h"
#include "iologindata.h"
#include "ioban.h"
#include "iomap.h"
#include "iomapserialize.h"
#include "talkaction.h"
#include "spells.h"
#include "combat.h"
#include "condition.h"
#include "baseevents.h"
#include "monsters.h"
#include "raids.h"
#include "configmanager.h"
#include "vocation.h"
#include "status.h"
#include "game.h"
#include "chat.h"
extern Game g_game;
extern Monsters g_monsters;
extern Chat g_chat;
extern ConfigManager g_config;
extern Spells* g_spells;
extern TalkActions* g_talkActions;
enum
{
EVENT_ID_LOADING = 1,
EVENT_ID_USER = 1000,
};
ScriptEnviroment::AreaMap ScriptEnviroment::m_areaMap;
uint32_t ScriptEnviroment::m_lastAreaId = 0;
ScriptEnviroment::CombatMap ScriptEnviroment::m_combatMap;
uint32_t ScriptEnviroment::m_lastCombatId = 0;
ScriptEnviroment::ConditionMap ScriptEnviroment::m_conditionMap;
uint32_t ScriptEnviroment::m_lastConditionId = 0;
ScriptEnviroment::ConditionMap ScriptEnviroment::m_tempConditionMap;
ScriptEnviroment::ThingMap ScriptEnviroment::m_globalMap;
ScriptEnviroment::StorageMap ScriptEnviroment::m_storageMap;
ScriptEnviroment::TempItemListMap ScriptEnviroment::m_tempItems;
ScriptEnviroment::ScriptEnviroment()
{
m_lastUID = 70000;
m_loaded = true;
reset();
}
ScriptEnviroment::~ScriptEnviroment()
{
for(CombatMap::iterator it = m_combatMap.begin(); it != m_combatMap.end(); ++it)
delete it->second;
m_combatMap.clear();
for(AreaMap::iterator it = m_areaMap.begin(); it != m_areaMap.end(); ++it)
delete it->second;
m_areaMap.clear();
for(ConditionMap::iterator it = m_conditionMap.begin(); it != m_conditionMap.end(); ++it)
delete it->second;
m_conditionMap.clear();
reset();
}
void ScriptEnviroment::reset()
{
m_scriptId = m_callbackId = 0;
m_timerEvent = false;
m_realPos = Position();
m_interface = NULL;
for(TempItemListMap::iterator mit = m_tempItems.begin(); mit != m_tempItems.end(); ++mit)
{
ItemList itemList = mit->second;
for(ItemList::iterator it = itemList.begin(); it != itemList.end(); ++it)
{
if((*it)->getParent() == VirtualCylinder::virtualCylinder)
g_game.freeThing(*it);
}
}
m_tempItems.clear();
for(DBResultMap::iterator it = m_tempResults.begin(); it != m_tempResults.end(); ++it)
{
if(it->second)
it->second->free();
}
m_tempResults.clear();
for(ConditionMap::iterator it = m_tempConditionMap.begin(); it != m_tempConditionMap.end(); ++it)
delete it->second;
m_tempConditionMap.clear();
m_localMap.clear();
}
bool ScriptEnviroment::saveGameState()
{
if(!g_config.getBool(ConfigManager::SAVE_GLOBAL_STORAGE))
return true;
Database* db = Database::getInstance();
DBQuery query;
query << "DELETE FROM `global_storage` WHERE `world_id` = " << g_config.getNumber(ConfigManager::WORLD_ID) << ";";
if(!db->query(query.str()))
return false;
DBInsert query_insert(db);
query_insert.setQuery("INSERT INTO `global_storage` (`key`, `world_id`, `value`) VALUES ");
for(StorageMap::const_iterator it = m_storageMap.begin(); it != m_storageMap.end(); ++it)
{
char buffer[25 + it->second.length()];
sprintf(buffer, "%u, %u, %s", it->first, g_config.getNumber(ConfigManager::WORLD_ID), db->escapeString(it->second).c_str());
if(!query_insert.addRow(buffer))
return false;
}
return query_insert.execute();
}
bool ScriptEnviroment::loadGameState()
{
Database* db = Database::getInstance();
DBResult* result;
DBQuery query;
query << "SELECT `key`, `value` FROM `global_storage` WHERE `world_id` = " << g_config.getNumber(ConfigManager::WORLD_ID) << ";";
if((result = db->storeQuery(query.str())))
{
do
m_storageMap[result->getDataInt("key")] = result->getDataString("value");
while(result->next());
result->free();
}
query.str("");
return true;
}
bool ScriptEnviroment::setCallbackId(int32_t callbackId, LuaInterface* interface)
{
if(!m_callbackId)
{
m_callbackId = callbackId;
m_interface = interface;
return true;
}
//nested callbacks are not allowed
if(m_interface)
m_interface->errorEx("Nested callbacks!");
return false;
}
void ScriptEnviroment::getInfo(int32_t& scriptId, std::string& desc, LuaInterface*& interface, int32_t& callbackId, bool& timerEvent)
{
scriptId = m_scriptId;
desc = m_event;
interface = m_interface;
callbackId = m_callbackId;
timerEvent = m_timerEvent;
}
void ScriptEnviroment::addUniqueThing(Thing* thing)
{
Item* item = thing->getItem();
if(!item || !item->getUniqueId())
return;
if(m_globalMap[item->getUniqueId()])
{
if(item->getActionId() != 2000) //scripted quest system
std::clog << "Duplicate uniqueId " << item->getUniqueId() << std::endl;
}
else
m_globalMap[item->getUniqueId()] = thing;
}
void ScriptEnviroment::removeUniqueThing(Thing* thing)
{
Item* item = thing->getItem();
if(!item || !item->getUniqueId())
return;
ThingMap::iterator it = m_globalMap.find(item->getUniqueId());
if(it != m_globalMap.end())
m_globalMap.erase(it);
}
uint32_t ScriptEnviroment::addThing(Thing* thing)
{
if(!thing || thing->isRemoved())
return 0;
for(ThingMap::iterator it = m_localMap.begin(); it != m_localMap.end(); ++it)
{
if(it->second == thing)
return it->first;
}
if(Creature* creature = thing->getCreature())
{
m_localMap[creature->getID()] = thing;
return creature->getID();
}
if(Item* item = thing->getItem())
{
uint32_t tmp = item->getUniqueId();
if(tmp)
{
m_localMap[tmp] = thing;
return tmp;
}
}
while(m_localMap.find(m_lastUID) != m_localMap.end())
++m_lastUID;
m_localMap[m_lastUID] = thing;
return m_lastUID;
}
void ScriptEnviroment::insertThing(uint32_t uid, Thing* thing)
{
if(!m_localMap[uid])
m_localMap[uid] = thing;
else
std::clog << "[Error - ScriptEnviroment::insertThing] Thing uid already taken" << std::endl;
}
Thing* ScriptEnviroment::getThingByUID(uint32_t uid)
{
Thing* tmp = m_localMap[uid];
if(tmp && !tmp->isRemoved())
return tmp;
tmp = m_globalMap[uid];
if(tmp && !tmp->isRemoved())
return tmp;
if(uid >= 0x10000000)
{
tmp = g_game.getCreatureByID(uid);
if(tmp && !tmp->isRemoved())
{
m_localMap[uid] = tmp;
return tmp;
}
}
return NULL;
}
Item* ScriptEnviroment::getItemByUID(uint32_t uid)
{
if(Thing* tmp = getThingByUID(uid))
{
if(Item* item = tmp->getItem())
return item;
}
return NULL;
}
Container* ScriptEnviroment::getContainerByUID(uint32_t uid)
{
if(Item* tmp = getItemByUID(uid))
{
if(Container* container = tmp->getContainer())
return container;
}
return NULL;
}
Creature* ScriptEnviroment::getCreatureByUID(uint32_t uid)
{
if(Thing* tmp = getThingByUID(uid))
{
if(Creature* creature = tmp->getCreature())
return creature;
}
return NULL;
}
Player* ScriptEnviroment::getPlayerByUID(uint32_t uid)
{
if(Thing* tmp = getThingByUID(uid))
{
if(Creature* creature = tmp->getCreature())
{
if(Player* player = creature->getPlayer())
return player;
}
}
return NULL;
}
void ScriptEnviroment::removeThing(uint32_t uid)
{
ThingMap::iterator it;
it = m_localMap.find(uid);
if(it != m_localMap.end())
m_localMap.erase(it);
it = m_globalMap.find(uid);
if(it != m_globalMap.end())
m_globalMap.erase(it);
}
uint32_t ScriptEnviroment::addCombatArea(CombatArea* area)
{
uint32_t newAreaId = m_lastAreaId + 1;
m_areaMap[newAreaId] = area;
m_lastAreaId++;
return newAreaId;
}
CombatArea* ScriptEnviroment::getCombatArea(uint32_t areaId)
{
AreaMap::const_iterator it = m_areaMap.find(areaId);
if(it != m_areaMap.end())
return it->second;
return NULL;
}
uint32_t ScriptEnviroment::addCombatObject(Combat* combat)
{
uint32_t newCombatId = m_lastCombatId + 1;
m_combatMap[newCombatId] = combat;
m_lastCombatId++;
return newCombatId;
}
Combat* ScriptEnviroment::getCombatObject(uint32_t combatId)
{
CombatMap::iterator it = m_combatMap.find(combatId);
if(it != m_combatMap.end())
return it->second;
return NULL;
}
uint32_t ScriptEnviroment::addConditionObject(Condition* condition)
{
m_conditionMap[++m_lastConditionId] = condition;
return m_lastConditionId;
}
uint32_t ScriptEnviroment::addTempConditionObject(Condition* condition)
{
m_conditionMap[++m_lastConditionId] = condition;
return m_lastConditionId;
}
Condition* ScriptEnviroment::getConditionObject(uint32_t conditionId)
{
ConditionMap::iterator it = m_conditionMap.find(conditionId);
if(it != m_conditionMap.end())
return it->second;
it = m_tempConditionMap.find(conditionId);
if(it != m_tempConditionMap.end())
return it->second;
return NULL;
}
void ScriptEnviroment::addTempItem(ScriptEnviroment* env, Item* item)
{
m_tempItems[env].push_back(item);
}
void ScriptEnviroment::removeTempItem(ScriptEnviroment* env, Item* item)
{
ItemList itemList = m_tempItems[env];
ItemList::iterator it = std::find(itemList.begin(), itemList.end(), item);
if(it != itemList.end())
itemList.erase(it);
}
void ScriptEnviroment::removeTempItem(Item* item)
{
for(TempItemListMap::iterator mit = m_tempItems.begin(); mit != m_tempItems.end(); ++mit)
{
ItemList itemList = mit->second;
ItemList::iterator it = std::find(itemList.begin(), itemList.end(), item);
if(it != itemList.end())
itemList.erase(it);
}
}
uint32_t ScriptEnviroment::addResult(DBResult* res)
{
uint32_t lastId = 0;
while(m_tempResults.find(lastId) != m_tempResults.end())
lastId++;
m_tempResults[lastId] = res;
return lastId;
}
bool ScriptEnviroment::removeResult(uint32_t id)
{
DBResultMap::iterator it = m_tempResults.find(id);
if(it == m_tempResults.end())
return false;
if(it->second)
it->second->free();
m_tempResults.erase(it);
return true;
}
DBResult* ScriptEnviroment::getResultByID(uint32_t id)
{
DBResultMap::iterator it = m_tempResults.find(id);
if(it != m_tempResults.end())
return it->second;
return NULL;
}
bool ScriptEnviroment::getStorage(const uint32_t key, std::string& value) const
{
StorageMap::const_iterator it = m_storageMap.find(key);
if(it != m_storageMap.end())
{
value = it->second;
return true;
}
value = "-1";
return false;
}
void ScriptEnviroment::streamVariant(std::stringstream& stream, const std::string& local, const LuaVariant& var)
{
if(!local.empty())
stream << "local " << local << " = {" << std::endl;
stream << "type = " << var.type;
switch(var.type)
{
case VARIANT_NUMBER:
stream << "," << std::endl << "number = " << var.number;
break;
case VARIANT_STRING:
stream << "," << std::endl << "string = "" << var.text << """;
break;
case VARIANT_TARGETPOSITION:
case VARIANT_POSITION:
{
stream << "," << std::endl;
streamPosition(stream, "pos", var.pos);
break;
}
case VARIANT_NONE:
default:
break;
}
if(!local.empty())
stream << std::endl << "}" << std::endl;
}
void ScriptEnviroment::streamThing(std::stringstream& stream, const std::string& local, Thing* thing, uint32_t id/* = 0*/)
{
if(!local.empty())
stream << "local " << local << " = {" << std::endl;
if(thing && thing->getItem())
{
const Item* item = thing->getItem();
if(!id)
id = addThing(thing);
stream << "uid = " << id << "," << std::endl;
stream << "itemid = " << item->getID() << "," << std::endl;
if(item->hasSubType())
stream << "type = " << item->getSubType() << "," << std::endl;
else
stream << "type = 0," << std::endl;
stream << "actionid = " << item->getActionId() << std::endl;
}
else if(thing && thing->getCreature())
{
const Creature* creature = thing->getCreature();
if(!id)
id = creature->getID();
stream << "uid = " << id << "," << std::endl;
stream << "itemid = 1," << std::endl;
if(creature->getPlayer())
stream << "type = 1," << std::endl;
else if(creature->getMonster())
stream << "type = 2," << std::endl;
else
stream << "type = 3," << std::endl;
if(const Player* player = creature->getPlayer())
stream << "actionid = " << player->getGUID() << "," << std::endl;
else
stream << "actionid = 0" << std::endl;
}
else
{
stream << "uid = 0," << std::endl;
stream << "itemid = 0," << std::endl;
stream << "type = 0," << std::endl;
stream << "actionid = 0" << std::endl;
}
if(!local.empty())
stream << "}" << std::endl;
}
void ScriptEnviroment::streamPosition(std::stringstream& stream, const std::string& local, const Position& position, uint32_t stackpos)
{
if(!local.empty())
stream << "local " << local << " = {" << std::endl;
stream << "x = " << position.x << "," << std::endl;
stream << "y = " << position.y << "," << std::endl;
stream << "z = " << position.z << "," << std::endl;
stream << "stackpos = " << stackpos << std::endl;
if(!local.empty())
stream << "}" << std::endl;
}
void ScriptEnviroment::streamOutfit(std::stringstream& stream, const std::string& local, const Outfit_t& outfit)
{
if(!local.empty())
stream << "local " << local << " = {" << std::endl;
stream << "lookType = " << outfit.lookType << "," << std::endl;
stream << "lookTypeEx = " << outfit.lookTypeEx << "," << std::endl;
stream << "lookHead = " << outfit.lookHead << "," << std::endl;
stream << "lookBody = " << outfit.lookBody << "," << std::endl;
stream << "lookLegs = " << outfit.lookLegs << "," << std::endl;
stream << "lookFeet = " << outfit.lookFeet << "," << std::endl;
stream << "lookAddons = " << outfit.lookAddons << std::endl;
if(!local.empty())
stream << "}" << std::endl;
}
std::string LuaInterface::getError(ErrorCode_t code)
{
switch(code)
{
case LUA_ERROR_PLAYER_NOT_FOUND:
return "Player not found";
case LUA_ERROR_MONSTER_NOT_FOUND:
return "Monster not found";
case LUA_ERROR_NPC_NOT_FOUND:
return "NPC not found";
case LUA_ERROR_CREATURE_NOT_FOUND:
return "Creature not found";
case LUA_ERROR_ITEM_NOT_FOUND:
return "Item not found";
case LUA_ERROR_THING_NOT_FOUND:
return "Thing not found";
case LUA_ERROR_TILE_NOT_FOUND:
return "Tile not found";
case LUA_ERROR_HOUSE_NOT_FOUND:
return "House not found";
case LUA_ERROR_COMBAT_NOT_FOUND:
return "Combat not found";
case LUA_ERROR_CONDITION_NOT_FOUND:
return "Condition not found";
case LUA_ERROR_AREA_NOT_FOUND:
return "Area not found";
case LUA_ERROR_CONTAINER_NOT_FOUND:
return "Container not found";
case LUA_ERROR_VARIANT_NOT_FOUND:
return "Variant not found";
case LUA_ERROR_VARIANT_UNKNOWN:
return "Unknown variant type";
case LUA_ERROR_SPELL_NOT_FOUND:
return "Spell not found";
default:
break;
}
return "Invalid error code!";
}
ScriptEnviroment LuaInterface::m_scriptEnv[21];
int32_t LuaInterface::m_scriptEnvIndex = -1;
LuaInterface::LuaInterface(std::string interfaceName)
{
m_luaState = NULL;
m_interfaceName = interfaceName;
m_lastTimer = 1000;
m_errors = true;
}
LuaInterface::~LuaInterface()
{
for(LuaTimerEvents::iterator it = m_timerEvents.begin(); it != m_timerEvents.end(); ++it)
Scheduler::getInstance().stopEvent(it->second.eventId);
closeState();
}
bool LuaInterface::reInitState()
{
closeState();
return initState();
}
bool LuaInterface::loadBuffer(const std::string& text, Npc* npc/* = NULL*/)
{
//loads buffer as a chunk at stack top
int32_t ret = luaL_loadbuffer(m_luaState, text.c_str(), text.length(), "LuaInterface::loadBuffer");
if(ret)
{
m_lastError = popString(m_luaState);
error(NULL, m_lastError);
return false;
}
//check that it is loaded as a function
if(!lua_isfunction(m_luaState, -1))
return false;
m_loadingFile = text;
reserveEnv();
ScriptEnviroment* env = getEnv();
env->setScriptId(EVENT_ID_LOADING, this);
env->setNpc(npc);
//execute it
ret = lua_pcall(m_luaState, 0, 0, 0);
if(ret)
{
error(NULL, popString(m_luaState));
releaseEnv();
return false;
}
releaseEnv();
return true;
}
bool LuaInterface::loadFile(const std::string& file, Npc* npc/* = NULL*/)
{
//loads file as a chunk at stack top
int32_t ret = luaL_loadfile(m_luaState, file.c_str());
if(ret)
{
m_lastError = popString(m_luaState);
std::clog << "[Error - LuaInterface::loadFile] " << m_lastError << std::endl;
return false;
}
//check that it is loaded as a function
if(!lua_isfunction(m_luaState, -1))
return false;
m_loadingFile = file;
reserveEnv();
ScriptEnviroment* env = getEnv();
env->setScriptId(EVENT_ID_LOADING, this);
env->setNpc(npc);
//execute it
ret = lua_pcall(m_luaState, 0, 0, 0);
if(ret)
{
error(NULL, popString(m_luaState));
releaseEnv();
return false;
}
releaseEnv();
return true;
}
bool LuaInterface::loadDirectory(const std::string& dir, Npc* npc/* = NULL*/)
{
StringVec files;
for(boost::filesystem::directory_iterator it(dir), end; it != end; ++it)
{
std::string s = it->leaf();
if(!boost::filesystem::is_directory(it->status()) && (s.size() > 4 ? s.substr(s.size() - 4) : "") == ".lua")
files.push_back(s);
}
std::sort(files.begin(), files.end());
for(StringVec::iterator it = files.begin(); it != files.end(); ++it)
{
if(!loadFile(dir + (*it), npc))
return false;
}
return true;
}
int32_t LuaInterface::getEvent(const std::string& eventName)
{
//get our events table
lua_getfield(m_luaState, LUA_REGISTRYINDEX, "EVENTS");
if(!lua_istable(m_luaState, -1))
{
lua_pop(m_luaState, 1);
return -1;
}
//get current event function pointer
lua_getglobal(m_luaState, eventName.c_str());
if(!lua_isfunction(m_luaState, -1))
{
lua_pop(m_luaState, 1);
return -1;
}
//save in our events table
lua_pushnumber(m_luaState, m_runningEvent);
lua_pushvalue(m_luaState, -2);
lua_rawset(m_luaState, -4);
lua_pop(m_luaState, 2);
//reset global value of this event
lua_pushnil(m_luaState);
lua_setglobal(m_luaState, eventName.c_str());
m_cacheFiles[m_runningEvent] = m_loadingFile + ":" + eventName;
++m_runningEvent;
return m_runningEvent - 1;
}
std::string LuaInterface::getScript(int32_t scriptId)
{
const static std::string tmp = "(Unknown script file)";
if(scriptId != EVENT_ID_LOADING)
{
ScriptsCache::iterator it = m_cacheFiles.find(scriptId);
if(it != m_cacheFiles.end())
return it->second;
return tmp;
}
return m_loadingFile;
}
void LuaInterface::error(const char* function, const std::string& desc)
{
int32_t script, callback;
bool timer;
std::string event;
LuaInterface* interface;
getEnv()->getInfo(script, event, interface, callback, timer);
if(interface)
{
if(!interface->m_errors)
return;
std::clog << std::endl << "[Error - " << interface->getName() << "] " << std::endl;
if(callback)
std::clog << "In a callback: " << interface->getScript(callback) << std::endl;
if(timer)
std::clog << (callback ? "from" : "In") << " a timer event called from: " << std::endl;
std::clog << interface->getScript(script) << std::endl << "Description: ";
}
else
std::clog << std::endl << "[Lua Error] ";
std::clog << event << std::endl;
if(function)
std::clog << "(" << function << ") ";
std::clog << desc << std::endl;
}
bool LuaInterface::pushFunction(int32_t function)
{
lua_getfield(m_luaState, LUA_REGISTRYINDEX, "EVENTS");
if(lua_istable(m_luaState, -1))
{
lua_pushnumber(m_luaState, function);
lua_rawget(m_luaState, -2);
lua_remove(m_luaState, -2);
if(lua_isfunction(m_luaState, -1))
return true;
}
return false;
}
bool LuaInterface::initState()
{
m_luaState = luaL_newstate();
if(!m_luaState)
return false;
luaL_openlibs(m_luaState);
#ifdef __LUAJIT__
luaJIT_setmode(m_luaState, 0, LUAJIT_MODE_ENGINE | LUAJIT_MODE_ON);
#endif
registerFunctions();
if(!loadDirectory(getFilePath(FILE_TYPE_OTHER, "lib/"), NULL))
std::clog << "[Warning - LuaInterface::initState] Cannot load " << getFilePath(FILE_TYPE_OTHER, "lib/") << std::endl;
lua_newtable(m_luaState);
lua_setfield(m_luaState, LUA_REGISTRYINDEX, "EVENTS");
m_runningEvent = EVENT_ID_USER;
return true;
}
bool LuaInterface::closeState()
{
if(!m_luaState)
return false;
m_cacheFiles.clear();
for(LuaTimerEvents::iterator it = m_timerEvents.begin(); it != m_timerEvents.end(); ++it)
{
for(std::list<int32_t>::iterator lt = it->second.parameters.begin(); lt != it->second.parameters.end(); ++lt)
luaL_unref(m_luaState, LUA_REGISTRYINDEX, *lt);
it->second.parameters.clear();
luaL_unref(m_luaState, LUA_REGISTRYINDEX, it->second.function);
}
m_timerEvents.clear();
lua_close(m_luaState);
return true;
}
void LuaInterface::executeTimer(uint32_t eventIndex)
{
LuaTimerEvents::iterator it = m_timerEvents.find(eventIndex);
if(it != m_timerEvents.end())
{
//push function
lua_rawgeti(m_luaState, LUA_REGISTRYINDEX, it->second.function);
//push parameters
for(std::list<int32_t>::reverse_iterator rt = it->second.parameters.rbegin(); rt != it->second.parameters.rend(); ++rt)
lua_rawgeti(m_luaState, LUA_REGISTRYINDEX, *rt);
//call the function
if(reserveEnv())
{
ScriptEnviroment* env = getEnv();
env->setTimerEvent();
env->setScriptId(it->second.scriptId, this);
callFunction(it->second.parameters.size());
releaseEnv();
}
else
std::clog << "[Error - LuaInterface::executeTimer] Call stack overflow." << std::endl;
//free resources
for(std::list<int32_t>::iterator lt = it->second.parameters.begin(); lt != it->second.parameters.end(); ++lt)
luaL_unref(m_luaState, LUA_REGISTRYINDEX, *lt);
it->second.parameters.clear();
luaL_unref(m_luaState, LUA_REGISTRYINDEX, it->second.function);
m_timerEvents.erase(it);
}
}
int32_t LuaInterface::handleFunction(lua_State* L)
{
lua_getfield(L, LUA_GLOBALSINDEX, "debug");
if(!lua_istable(L, -1))
{
lua_pop(L, 1);
return 1;
}
lua_getfield(L, -1, "traceback");
if(!lua_isfunction(L, -1))
{
lua_pop(L, 2);
return 1;
}
lua_pushvalue(L, 1);
lua_pushinteger(L, 2);
lua_call(L, 2, 1);
return 1;
}
bool LuaInterface::callFunction(uint32_t params)
{
int32_t size = lua_gettop(m_luaState), handler = lua_gettop(m_luaState) - params;
lua_pushcfunction(m_luaState, handleFunction);
bool result = false;
lua_insert(m_luaState, handler);
if(lua_pcall(m_luaState, params, 1, handler))
LuaInterface::error(NULL, LuaInterface::popString(m_luaState));
else
result = (int32_t)LuaInterface::popBoolean(m_luaState);
lua_remove(m_luaState, handler);
if((lua_gettop(m_luaState) + (int32_t)params + 1) != size)
LuaInterface::error(NULL, "Stack size changed!");
return result;
}
void LuaInterface::dumpStack(lua_State* L/* = NULL*/)
{
if(!L)
L = m_luaState;
int32_t stack = lua_gettop(L);
if(!stack)
return;
std::clog << "Stack size: " << stack << std::endl;
for(int32_t i = 1; i <= stack ; ++i)
std::clog << lua_typename(m_luaState, lua_type(m_luaState, -i)) << " " << lua_topointer(m_luaState, -i) << std::endl;
}
void LuaInterface::pushVariant(lua_State* L, const LuaVariant& var)
{
lua_newtable(L);
setField(L, "type", var.type);
switch(var.type)
{
case VARIANT_NUMBER:
setField(L, "number", var.number);
break;
case VARIANT_STRING:
setField(L, "string", var.text);
break;
case VARIANT_TARGETPOSITION:
case VARIANT_POSITION:
{
lua_pushstring(L, "pos");
pushPosition(L, var.pos);
pushTable(L);
break;
}
case VARIANT_NONE:
break;
}
}
void LuaInterface::pushThing(lua_State* L, Thing* thing, uint32_t id/* = 0*/)
{
lua_newtable(L);
if(thing && thing->getItem())
{
const Item* item = thing->getItem();
if(!id)
id = getEnv()->addThing(thing);
setField(L, "uid", id);
setField(L, "itemid", item->getID());
if(item->hasSubType())
setField(L, "type", item->getSubType());
else
setField(L, "type", 0);
setField(L, "actionid", item->getActionId());
}
else if(thing && thing->getCreature())
{
const Creature* creature = thing->getCreature();
if(!id)
id = creature->getID();
setField(L, "uid", id);
setField(L, "itemid", 1);
if(creature->getPlayer())
setField(L, "type", 1);
else if(creature->getMonster())
setField(L, "type", 2);
else
setField(L, "type", 3);
if(const Player* player = creature->getPlayer())
setField(L, "actionid", player->getGUID());
else
setField(L, "actionid", 0);
}
else
{
setField(L, "uid", 0);
setField(L, "itemid", 0);
setField(L, "type", 0);
setField(L, "actionid", 0);
}
}
void LuaInterface::pushPosition(lua_State* L, const Position& position, uint32_t stackpos)
{
lua_newtable(L);
setField(L, "x", position.x);
setField(L, "y", position.y);
setField(L, "z", position.z);
setField(L, "stackpos", stackpos);
}
void LuaInterface::pushOutfit(lua_State* L, const Outfit_t& outfit)
{
lua_newtable(L);
setField(L, "lookType", outfit.lookType);
setField(L, "lookTypeEx", outfit.lookTypeEx);
setField(L, "lookHead", outfit.lookHead);
setField(L, "lookBody", outfit.lookBody);
setField(L, "lookLegs", outfit.lookLegs);
setField(L, "lookFeet", outfit.lookFeet);
setField(L, "lookAddons", outfit.lookAddons);
}
void LuaInterface::pushCallback(lua_State* L, int32_t callback)
{
lua_rawgeti(L, LUA_REGISTRYINDEX, callback);
}
LuaVariant LuaInterface::popVariant(lua_State* L)
{
LuaVariant var;
var.type = (LuaVariantType_t)getField(L, "type");
switch(var.type)
{
case VARIANT_NUMBER:
var.number = getFieldUnsigned(L, "number");
break;
case VARIANT_STRING:
var.text = getField(L, "string");
break;
case VARIANT_POSITION:
case VARIANT_TARGETPOSITION:
{
lua_pushstring(L, "pos");
lua_gettable(L, -2);
popPosition(L, var.pos);
break;
}
default:
var.type = VARIANT_NONE;
break;
}
lua_pop(L, 1); //table
return var;
}
void LuaInterface::popPosition(lua_State* L, PositionEx& position)
{
if(!lua_isboolean(L, -1))
{
position.x = getField(L, "x");
position.y = getField(L, "y");
position.z = getField(L, "z");
position.stackpos = getField(L, "stackpos");
}
else
position = PositionEx();
lua_pop(L, 1); //table
}
void LuaInterface::popPosition(lua_State* L, Position& position, uint32_t& stackpos)
{
stackpos = 0;
if(!lua_isboolean(L, -1))
{
position.x = getField(L, "x");
position.y = getField(L, "y");
position.z = getField(L, "z");
stackpos = getField(L, "stackpos");
}
else
position = Position();
lua_pop(L, 1); //table
}
bool LuaInterface::popBoolean(lua_State* L)
{
lua_pop(L, 1);
return lua_toboolean(L, 0);
}
int64_t LuaInterface::popNumber(lua_State* L)
{
lua_pop(L, 1);
if(lua_isboolean(L, 0))
return (int64_t)lua_toboolean(L, 0);
return (int64_t)lua_tonumber(L, 0);
}
double LuaInterface::popFloatNumber(lua_State* L)
{
lua_pop(L, 1);
return lua_tonumber(L, 0);
}
std::string LuaInterface::popString(lua_State* L)
{
lua_pop(L, 1);
const char* str = lua_tostring(L, 0);
if(!str || !strlen(str))
return std::string();
return str;
}
int32_t LuaInterface::popCallback(lua_State* L)
{
return luaL_ref(L, LUA_REGISTRYINDEX);
}
Outfit_t LuaInterface::popOutfit(lua_State* L)
{
Outfit_t outfit;
outfit.lookAddons = getField(L, "lookAddons");
outfit.lookFeet = getField(L, "lookFeet");
outfit.lookLegs = getField(L, "lookLegs");
outfit.lookBody = getField(L, "lookBody");
outfit.lookHead = getField(L, "lookHead");
outfit.lookTypeEx = getField(L, "lookTypeEx");
outfit.lookType = getField(L, "lookType");
lua_pop(L, 1); //table
return outfit;
}
void LuaInterface::setField(lua_State* L, const char* index, int32_t val)
{
lua_pushstring(L, index);
lua_pushnumber(L, val);
pushTable(L);
}
void LuaInterface::setField(lua_State* L, const char* index, const std::string& val)
{
lua_pushstring(L, index);
lua_pushstring(L, val.c_str());
pushTable(L);
}
void LuaInterface::setFieldBool(lua_State* L, const char* index, bool val)
{
lua_pushstring(L, index);
lua_pushboolean(L, val);
pushTable(L);
}
void LuaInterface::setFieldFloat(lua_State* L, const char* index, double val)
{
lua_pushstring(L, index);
lua_pushnumber(L, val);
pushTable(L);
}
void LuaInterface::createTable(lua_State* L, const char* index)
{
lua_pushstring(L, index);
lua_newtable(L);
}
void LuaInterface::createTable(lua_State* L, const char* index, int32_t narr, int32_t nrec)
{
lua_pushstring(L, index);
lua_createtable(L, narr, nrec);
}
void LuaInterface::createTable(lua_State* L, int32_t index)
{
lua_pushnumber(L, index);
lua_newtable(L);
}
void LuaInterface::createTable(lua_State* L, int32_t index, int32_t narr, int32_t nrec)
{
lua_pushnumber(L, index);
lua_createtable(L, narr, nrec);
}
void LuaInterface::pushTable(lua_State* L)
{
lua_settable(L, -3);
}
int64_t LuaInterface::getField(lua_State* L, const char* key)
{
lua_pushstring(L, key);
lua_gettable(L, -2); // get table[key]
int64_t result = (int64_t)lua_tonumber(L, -1);
lua_pop(L, 1); // remove number and key
return result;
}
uint64_t LuaInterface::getFieldUnsigned(lua_State* L, const char* key)
{
lua_pushstring(L, key);
lua_gettable(L, -2); // get table[key]
uint64_t result = (uint64_t)lua_tonumber(L, -1);
lua_pop(L, 1); // remove number and key
return result;
}
bool LuaInterface::getFieldBool(lua_State* L, const char* key)
{
lua_pushstring(L, key);
lua_gettable(L, -2); // get table[key]
bool result = lua_toboolean(L, -1);
lua_pop(L, 1); // remove number and key
return result;
}
std::string LuaInterface::getFieldString(lua_State* L, const char* key)
{
lua_pushstring(L, key);
lua_gettable(L, -2); // get table[key]
std::string result = lua_tostring(L, -1);
lua_pop(L, 1); // remove number and key
return result;
}
std::string LuaInterface::getGlobalString(lua_State* L, const std::string& _identifier, const std::string& _default/* = ""*/)
{
lua_getglobal(L, _identifier.c_str());
if(!lua_isstring(L, -1))
{
lua_pop(L, 1);
return _default;
}
int32_t len = (int32_t)lua_strlen(L, -1);
std::string ret(lua_tostring(L, -1), len);
lua_pop(L, 1);
return ret;
}
bool LuaInterface::getGlobalBool(lua_State* L, const std::string& _identifier, bool _default/* = false*/)
{
lua_getglobal(L, _identifier.c_str());
if(!lua_isboolean(L, -1))
{
lua_pop(L, 1);
return booleanString(LuaInterface::getGlobalString(L, _identifier, _default ? "yes" : "no"));
}
bool val = lua_toboolean(L, -1);
lua_pop(L, 1);
return val;
}
int32_t LuaInterface::getGlobalNumber(lua_State* L, const std::string& _identifier, const int32_t _default/* = 0*/)
{
return (int32_t)LuaInterface::getGlobalDouble(L, _identifier, _default);
}
double LuaInterface::getGlobalDouble(lua_State* L, const std::string& _identifier, const double _default/* = 0*/)
{
lua_getglobal(L, _identifier.c_str());
if(!lua_isnumber(L, -1))
{
lua_pop(L, 1);
return _default;
}
double val = lua_tonumber(L, -1);
lua_pop(L, 1);
return val;
}
void LuaInterface::getValue(const std::string& key, lua_State* L, lua_State* _L)
{
lua_getglobal(L, key.c_str());
moveValue(L, _L);
}
void LuaInterface::moveValue(lua_State* from, lua_State* to)
{
switch(lua_type(from, -1))
{
case LUA_TNIL:
lua_pushnil(to);
break;
case LUA_TBOOLEAN:
lua_pushboolean(to, lua_toboolean(from, -1));
break;
case LUA_TNUMBER:
lua_pushnumber(to, lua_tonumber(from, -1));
break;
case LUA_TSTRING:
{
size_t len;
const char* str = lua_tolstring(from, -1, &len);
lua_pushlstring(to, str, len);
break;
}
case LUA_TTABLE:
{
lua_newtable(to);
lua_pushnil(from); // First key
while(lua_next(from, -2))
{
// Move value to the other state
moveValue(from, to); // Value is popped, key is left
// Move key to the other state
lua_pushvalue(from, -1); // Make a copy of the key to use for the next iteration
moveValue(from, to); // Key is in other state.
// We still have the key in the 'from' state ontop of the stack
lua_insert(to, -2); // Move key above value
pushTable(to); // Set the key
}
break;
}
default:
break;
}
lua_pop(from, 1); // Pop the value we just read
}
void LuaInterface::registerFunctions()
{
//example(...)
//lua_register(L, "name", C_function);
//getPlayersOnline()
lua_register(m_luaState, "getCastsOnline", LuaInterface::luaGetCastsOnline);
//doPlayerSetCastDescription(cid, desc)
lua_register(m_luaState, "doPlayerSetCastDescription", LuaInterface::luaDoPlayerSetCastDescription);
//doPlayerAddCastMute(cid, ip)
lua_register(m_luaState, "doPlayerAddCastMute", LuaInterface::luaDoPlayerAddCastMute);
//doPlayerRemoveCastMute(cidl, ip)
lua_register(m_luaState, "doPlayerRemoveCastMute", LuaInterface::luaDoPlayerRemoveCastMute);
//doPlayerGetCastMutes(cid)
lua_register(m_luaState, "getCastMutes", LuaInterface::luaGetPlayerCastMutes);
//doPlayerAddCastBan(cid, ip)
lua_register(m_luaState, "doPlayerAddCastBan", LuaInterface::luaDoPlayerAddCastBan);
//doPlayerRemoveCastBan(cidl, ip)
lua_register(m_luaState, "doPlayerRemoveCastBan", LuaInterface::luaDoPlayerRemoveCastBan);
//doPlayerGetCastBan(cid)
lua_register(m_luaState, "getCastBans", LuaInterface::luaGetPlayerCastBans);
//doPlayerAddCastBan(cid, ip)
lua_register(m_luaState, "getCastViewers", LuaInterface::luaGetPlayerCastViewers);
//doPlayerSetCastPassword(cid, password)
lua_register(m_luaState, "doPlayerSetCastPassword", LuaInterface::luaDoPlayerSetCastPassword);
//getPlayerCast(cid)
lua_register(m_luaState, "doPlayerSetCastState", LuaInterface::luaDoPlayerSetCastState);
//getPlayerCast(cid)
lua_register(m_luaState, "getPlayerCast", LuaInterface::luaGetPlayerCast);
//getCreatureHealth(cid)
lua_register(m_luaState, "getCreatureHealth", LuaInterface::luaGetCreatureHealth);
//getCreatureMaxHealth(cid)
lua_register(m_luaState, "getCreatureMaxHealth", LuaInterface::luaGetCreatureMaxHealth);
//getCreatureMana(cid)
lua_register(m_luaState, "getCreatureMana", LuaInterface::luaGetCreatureMana);
//getCreatureMaxMana(cid)
lua_register(m_luaState, "getCreatureMaxMana", LuaInterface::luaGetCreatureMaxMana);
//getCreatureHideHealth(cid)
lua_register(m_luaState, "getCreatureHideHealth", LuaInterface::luaGetCreatureHideHealth);
//doCreatureSetHideHealth(cid, hide)
lua_register(m_luaState, "doCreatureSetHideHealth", LuaInterface::luaDoCreatureSetHideHealth);
//getCreatureSpeakType(cid)
lua_register(m_luaState, "getCreatureSpeakType", LuaInterface::luaGetCreatureSpeakType);
//doCreatureSetSpeakType(cid, type)
lua_register(m_luaState, "doCreatureSetSpeakType", LuaInterface::luaDoCreatureSetSpeakType);
//getCreatureLookDirection(cid)
lua_register(m_luaState, "getCreatureLookDirection", LuaInterface::luaGetCreatureLookDirection);
//getPlayerLevel(cid)
lua_register(m_luaState, "getPlayerLevel", LuaInterface::luaGetPlayerLevel);
//getPlayerExperience(cid)
lua_register(m_luaState, "getPlayerExperience", LuaInterface::luaGetPlayerExperience);
//getPlayerMagLevel(cid[, ignoreBuffs = false])
lua_register(m_luaState, "getPlayerMagLevel", LuaInterface::luaGetPlayerMagLevel);
//getPlayerSpentMana(cid)
lua_register(m_luaState, "getPlayerSpentMana", LuaInterface::luaGetPlayerSpentMana);
//getPlayerFood(cid)
lua_register(m_luaState, "getPlayerFood", LuaInterface::luaGetPlayerFood);
//getPlayerAccess(cid)
lua_register(m_luaState, "getPlayerAccess", LuaInterface::luaGetPlayerAccess);
//getPlayerGhostAccess(cid)
lua_register(m_luaState, "getPlayerGhostAccess", LuaInterface::luaGetPlayerGhostAccess);
//getPlayerSkillLevel(cid, skillid)
lua_register(m_luaState, "getPlayerSkillLevel", LuaInterface::luaGetPlayerSkillLevel);
//getPlayerSkillTries(cid, skillid)
lua_register(m_luaState, "getPlayerSkillTries", LuaInterface::luaGetPlayerSkillTries);
//getPlayerTown(cid)
lua_register(m_luaState, "getPlayerTown", LuaInterface::luaGetPlayerTown);
//getPlayerVocation(cid)
lua_register(m_luaState, "getPlayerVocation", LuaInterface::luaGetPlayerVocation);
//getPlayerIp(cid)
lua_register(m_luaState, "getPlayerIp", LuaInterface::luaGetPlayerIp);
//getPlayerRequiredMana(cid, magicLevel)
lua_register(m_luaState, "getPlayerRequiredMana", LuaInterface::luaGetPlayerRequiredMana);
//getPlayerRequiredSkillTries(cid, skillId, skillLevel)
lua_register(m_luaState, "getPlayerRequiredSkillTries", LuaInterface::luaGetPlayerRequiredSkillTries);
//getPlayerItemCount(cid, itemid[, subType = -1])
lua_register(m_luaState, "getPlayerItemCount", LuaInterface::luaGetPlayerItemCount);
//getPlayerMoney(cid)
lua_register(m_luaState, "getPlayerMoney", LuaInterface::luaGetPlayerMoney);
//getPlayerSoul(cid)
lua_register(m_luaState, "getPlayerSoul", LuaInterface::luaGetPlayerSoul);
//getPlayerFreeCap(cid)
lua_register(m_luaState, "getPlayerFreeCap", LuaInterface::luaGetPlayerFreeCap);
//getPlayerLight(cid)
lua_register(m_luaState, "getPlayerLight", LuaInterface::luaGetPlayerLight);
//getPlayerSlotItem(cid, slot)
lua_register(m_luaState, "getPlayerSlotItem", LuaInterface::luaGetPlayerSlotItem);
//getPlayerWeapon(cid[, ignoreAmmo = false])
lua_register(m_luaState, "getPlayerWeapon", LuaInterface::luaGetPlayerWeapon);
//getPlayerItemById(cid, deepSearch, itemId[, subType = -1])
lua_register(m_luaState, "getPlayerItemById", LuaInterface::luaGetPlayerItemById);
//getPlayerDepotItems(cid, depotid)
lua_register(m_luaState, "getPlayerDepotItems", LuaInterface::luaGetPlayerDepotItems);
//getPlayerGuildId(cid)
lua_register(m_luaState, "getPlayerGuildId", LuaInterface::luaGetPlayerGuildId);
//getPlayerGuildName(cid)
lua_register(m_luaState, "getPlayerGuildName", LuaInterface::luaGetPlayerGuildName);
//getPlayerGuildRankId(cid)
lua_register(m_luaState, "getPlayerGuildRankId", LuaInterface::luaGetPlayerGuildRankId);
//getPlayerGuildRank(cid)
lua_register(m_luaState, "getPlayerGuildRank", LuaInterface::luaGetPlayerGuildRank);
//getPlayerGuildNick(cid)
lua_register(m_luaState, "getPlayerGuildNick", LuaInterface::luaGetPlayerGuildNick);
//getPlayerGuildLevel(cid)
lua_register(m_luaState, "getPlayerGuildLevel", LuaInterface::luaGetPlayerGuildLevel);
//getPlayerGUID(cid)
lua_register(m_luaState, "getPlayerGUID", LuaInterface::luaGetPlayerGUID);
//getPlayerNameDescription(cid)
lua_register(m_luaState, "getPlayerNameDescription", LuaInterface::luaGetPlayerNameDescription);
//doPlayerSetNameDescription(cid, desc)
lua_register(m_luaState, "doPlayerSetNameDescription", LuaInterface::luaDoPlayerSetNameDescription);
//getPlayerSpecialDescription(cid)
lua_register(m_luaState, "getPlayerSpecialDescription", LuaInterface::luaGetPlayerSpecialDescription);
//doPlayerSetSpecialDescription(cid, desc)
lua_register(m_luaState, "doPlayerSetSpecialDescription", LuaInterface::luaDoPlayerSetSpecialDescription);
//getPlayerAccountId(cid)
lua_register(m_luaState, "getPlayerAccountId", LuaInterface::luaGetPlayerAccountId);
//getPlayerAccount(cid)
lua_register(m_luaState, "getPlayerAccount", LuaInterface::luaGetPlayerAccount);
//getPlayerFlagValue(cid, flag)
lua_register(m_luaState, "getPlayerFlagValue", LuaInterface::luaGetPlayerFlagValue);
//getPlayerCustomFlagValue(cid, flag)
lua_register(m_luaState, "getPlayerCustomFlagValue", LuaInterface::luaGetPlayerCustomFlagValue);
//getPlayerPromotionLevel(cid)
lua_register(m_luaState, "getPlayerPromotionLevel", LuaInterface::luaGetPlayerPromotionLevel);
//doPlayerSetPromotionLevel(cid, level)
lua_register(m_luaState, "doPlayerSetPromotionLevel", LuaInterface::luaDoPlayerSetPromotionLevel);
//getPlayerGroupId(cid)
lua_register(m_luaState, "getPlayerGroupId", LuaInterface::luaGetPlayerGroupId);
//doPlayerSetGroupId(cid, newGroupId)
lua_register(m_luaState, "doPlayerSetGroupId", LuaInterface::luaDoPlayerSetGroupId);
//doPlayerSendOutfitWindow(cid)
lua_register(m_luaState, "doPlayerSendOutfitWindow", LuaInterface::luaDoPlayerSendOutfitWindow);
//doPlayerLearnInstantSpell(cid, name)
lua_register(m_luaState, "doPlayerLearnInstantSpell", LuaInterface::luaDoPlayerLearnInstantSpell);
//doPlayerUnlearnInstantSpell(cid, name)
lua_register(m_luaState, "doPlayerUnlearnInstantSpell", LuaInterface::luaDoPlayerUnlearnInstantSpell);
//getPlayerLearnedInstantSpell(cid, name)
lua_register(m_luaState, "getPlayerLearnedInstantSpell", LuaInterface::luaGetPlayerLearnedInstantSpell);
//getPlayerInstantSpellCount(cid)
lua_register(m_luaState, "getPlayerInstantSpellCount", LuaInterface::luaGetPlayerInstantSpellCount);
//getPlayerInstantSpellInfo(cid, index)
lua_register(m_luaState, "getPlayerInstantSpellInfo", LuaInterface::luaGetPlayerInstantSpellInfo);
//getInstantSpellInfo(cid, name)
lua_register(m_luaState, "getInstantSpellInfo", LuaInterface::luaGetInstantSpellInfo);
//getCreatureStorage(uid, key)
lua_register(m_luaState, "getCreatureStorage", LuaInterface::luaGetCreatureStorage);
//doCreatureSetStorage(uid, key, value)
lua_register(m_luaState, "doCreatureSetStorage", LuaInterface::luaDoCreatureSetStorage);
//getStorage(key)
lua_register(m_luaState, "getStorage", LuaInterface::luaGetStorage);
//doSetStorage(key, value)
lua_register(m_luaState, "doSetStorage", LuaInterface::luaDoSetStorage);
//getChannelUsers(channelId)
lua_register(m_luaState, "getChannelUsers", LuaInterface::luaGetChannelUsers);
//getPlayersOnline()
lua_register(m_luaState, "getPlayersOnline", LuaInterface::luaGetPlayersOnline);
//getTileInfo(pos)
lua_register(m_luaState, "getTileInfo", LuaInterface::luaGetTileInfo);
//getThingFromPos(pos[, displayError = true])
lua_register(m_luaState, "getThingFromPos", LuaInterface::luaGetThingFromPos);
//getThing(uid)
lua_register(m_luaState, "getThing", LuaInterface::luaGetThing);
//doTileQueryAdd(uid, pos[, flags[, displayError = true]])
lua_register(m_luaState, "doTileQueryAdd", LuaInterface::luaDoTileQueryAdd);
//doItemRaidUnref(uid)
lua_register(m_luaState, "doItemRaidUnref", LuaInterface::luaDoItemRaidUnref);
//getThingPosition(uid)
lua_register(m_luaState, "getThingPosition", LuaInterface::luaGetThingPosition);
//getTileItemById(pos, itemId[, subType = -1])
lua_register(m_luaState, "getTileItemById", LuaInterface::luaGetTileItemById);
//getTileItemByType(pos, type)
lua_register(m_luaState, "getTileItemByType", LuaInterface::luaGetTileItemByType);
//getTileThingByPos(pos)
lua_register(m_luaState, "getTileThingByPos", LuaInterface::luaGetTileThingByPos);
//getTopCreature(pos)
lua_register(m_luaState, "getTopCreature", LuaInterface::luaGetTopCreature);
//doRemoveItem(uid[, count = -1])
lua_register(m_luaState, "doRemoveItem", LuaInterface::luaDoRemoveItem);
//doPlayerFeed(cid, food)
lua_register(m_luaState, "doPlayerFeed", LuaInterface::luaDoPlayerFeed);
//doPlayerSendCancel(cid, text)
lua_register(m_luaState, "doPlayerSendCancel", LuaInterface::luaDoPlayerSendCancel);
//doPlayerSendDefaultCancel(cid, ReturnValue)
lua_register(m_luaState, "doPlayerSendDefaultCancel", LuaInterface::luaDoSendDefaultCancel);
//getSearchString(fromPosition, toPosition[, fromIsCreature = false[, toIsCreature = false]])
lua_register(m_luaState, "getSearchString", LuaInterface::luaGetSearchString);
//getClosestFreeTile(cid, targetpos[, extended = false[, ignoreHouse = true]])
lua_register(m_luaState, "getClosestFreeTile", LuaInterface::luaGetClosestFreeTile);
//doTeleportThing(cid, newpos[, pushmove = true[, fullTeleport = true]])
lua_register(m_luaState, "doTeleportThing", LuaInterface::luaDoTeleportThing);
//doTransformItem(uid, newId[, count/subType])
lua_register(m_luaState, "doTransformItem", LuaInterface::luaDoTransformItem);
//doCreatureSay(uid, text[, type = SPEAK_SAY[, ghost = false[, cid = 0[, pos]]]])
lua_register(m_luaState, "doCreatureSay", LuaInterface::luaDoCreatureSay);
//doSendCreatureSquare(cid, color[, player])
lua_register(m_luaState, "doSendCreatureSquare", LuaInterface::luaDoSendCreatureSquare);
//doSendMagicEffect(pos, type[, player])
lua_register(m_luaState, "doSendMagicEffect", LuaInterface::luaDoSendMagicEffect);
//doSendDistanceShoot(fromPos, toPos, type[, player])
lua_register(m_luaState, "doSendDistanceShoot", LuaInterface::luaDoSendDistanceShoot);
//doSendAnimatedText(pos, text, color[, player])
lua_register(m_luaState, "doSendAnimatedText", LuaInterface::luaDoSendAnimatedText);
//doPlayerAddSkillTry(cid, skillid, n[, useMultiplier = true])
lua_register(m_luaState, "doPlayerAddSkillTry", LuaInterface::luaDoPlayerAddSkillTry);
//doCreatureAddHealth(cid, health[, hitEffect[, hitColor[, force]]])
lua_register(m_luaState, "doCreatureAddHealth", LuaInterface::luaDoCreatureAddHealth);
//doCreatureAddMana(cid, mana)
lua_register(m_luaState, "doCreatureAddMana", LuaInterface::luaDoCreatureAddMana);
//setCreatureMaxHealth(cid, health)
lua_register(m_luaState, "setCreatureMaxHealth", LuaInterface::luaSetCreatureMaxHealth);
//setCreatureMaxMana(cid, mana)
lua_register(m_luaState, "setCreatureMaxMana", LuaInterface::luaSetCreatureMaxMana);
//doPlayerSetMaxCapacity(cid, cap)
lua_register(m_luaState, "doPlayerSetMaxCapacity", LuaInterface::luaDoPlayerSetMaxCapacity);
//doPlayerAddSpentMana(cid, amount[, useMultiplier = true])
lua_register(m_luaState, "doPlayerAddSpentMana", LuaInterface::luaDoPlayerAddSpentMana);
//doPlayerAddSoul(cid, soul)
lua_register(m_luaState, "doPlayerAddSoul", LuaInterface::luaDoPlayerAddSoul);
//doPlayerAddItem(cid, itemid[, count/subtype = 1[, canDropOnMap = true[, slot = 0]]])
//doPlayerAddItem(cid, itemid[, count = 1[, canDropOnMap = true[, subtype = 1[, slot = 0]]]])
//Returns uid of the created item
lua_register(m_luaState, "doPlayerAddItem", LuaInterface::luaDoPlayerAddItem);
//doPlayerAddItemEx(cid, uid[, canDropOnMap = false[, slot = 0]])
lua_register(m_luaState, "doPlayerAddItemEx", LuaInterface::luaDoPlayerAddItemEx);
//doPlayerSendTextMessage(cid, MessageClasses, message)
lua_register(m_luaState, "doPlayerSendTextMessage", LuaInterface::luaDoPlayerSendTextMessage);
//doPlayerSendChannelMessage(cid, author, message, SpeakClasses, channel)
lua_register(m_luaState, "doPlayerSendChannelMessage", LuaInterface::luaDoPlayerSendChannelMessage);
//doPlayerSendToChannel(cid, targetId, SpeakClasses, message, channel[, time])
lua_register(m_luaState, "doPlayerSendToChannel", LuaInterface::luaDoPlayerSendToChannel);
//doPlayerOpenChannel(cid, channelId)
lua_register(m_luaState, "doPlayerOpenChannel", LuaInterface::luaDoPlayerOpenChannel);
//doPlayerAddMoney(cid, money)
lua_register(m_luaState, "doPlayerAddMoney", LuaInterface::luaDoPlayerAddMoney);
//doPlayerRemoveMoney(cid, money)
lua_register(m_luaState, "doPlayerRemoveMoney", LuaInterface::luaDoPlayerRemoveMoney);
//doPlayerTransferMoneyTo(cid, target, money)
lua_register(m_luaState, "doPlayerTransferMoneyTo", LuaInterface::luaDoPlayerTransferMoneyTo);
//doShowTextDialog(cid, itemid, text)
lua_register(m_luaState, "doShowTextDialog", LuaInterface::luaDoShowTextDialog);
//doDecayItem(uid)
lua_register(m_luaState, "doDecayItem", LuaInterface::luaDoDecayItem);
//doCreateItem(itemid[, type/count], pos)
//Returns uid of the created item, only works on tiles.
lua_register(m_luaState, "doCreateItem", LuaInterface::luaDoCreateItem);
//doCreateItemEx(itemid[, count/subType = -1])
lua_register(m_luaState, "doCreateItemEx", LuaInterface::luaDoCreateItemEx);
//doTileAddItemEx(pos, uid)
lua_register(m_luaState, "doTileAddItemEx", LuaInterface::luaDoTileAddItemEx);
//doAddContainerItemEx(uid, virtuid)
lua_register(m_luaState, "doAddContainerItemEx", LuaInterface::luaDoAddContainerItemEx);
//doRelocate(pos, posTo[, creatures = true[, unmovable = true]])
//Moves all moveable objects from pos to posTo
lua_register(m_luaState, "doRelocate", LuaInterface::luaDoRelocate);
//doCleanTile(pos[, forceMapLoaded = false])
lua_register(m_luaState, "doCleanTile", LuaInterface::luaDoCleanTile);
//doCreateTeleport(itemid, topos, createpos)
lua_register(m_luaState, "doCreateTeleport", LuaInterface::luaDoCreateTeleport);
//doCreateMonster(name, pos[, extend = false[, force = false[, displayError = true]]])
lua_register(m_luaState, "doCreateMonster", LuaInterface::luaDoCreateMonster);
//doCreateNpc(name, pos[, displayError = true])
lua_register(m_luaState, "doCreateNpc", LuaInterface::luaDoCreateNpc);
//doSummonMonster(cid, name)
lua_register(m_luaState, "doSummonMonster", LuaInterface::luaDoSummonMonster);
//doConvinceCreature(cid, target)
lua_register(m_luaState, "doConvinceCreature", LuaInterface::luaDoConvinceCreature);
//getMonsterTargetList(cid)
lua_register(m_luaState, "getMonsterTargetList", LuaInterface::luaGetMonsterTargetList);
//getMonsterFriendList(cid)
lua_register(m_luaState, "getMonsterFriendList", LuaInterface::luaGetMonsterFriendList);
//doMonsterSetTarget(cid, target)
lua_register(m_luaState, "doMonsterSetTarget", LuaInterface::luaDoMonsterSetTarget);
//doMonsterChangeTarget(cid)
lua_register(m_luaState, "doMonsterChangeTarget", LuaInterface::luaDoMonsterChangeTarget);
//getMonsterInfo(name)
lua_register(m_luaState, "getMonsterInfo", LuaInterface::luaGetMonsterInfo);
//doAddCondition(cid, condition)
lua_register(m_luaState, "doAddCondition", LuaInterface::luaDoAddCondition);
//doRemoveCondition(cid, type[, subId])
lua_register(m_luaState, "doRemoveCondition", LuaInterface::luaDoRemoveCondition);
//doRemoveConditions(cid[, onlyPersistent])
lua_register(m_luaState, "doRemoveConditions", LuaInterface::luaDoRemoveConditions);
//doRemoveCreature(cid[, forceLogout = true])
lua_register(m_luaState, "doRemoveCreature", LuaInterface::luaDoRemoveCreature);
//doMoveCreature(cid, direction[, flag = FLAG_NOLIMIT])
lua_register(m_luaState, "doMoveCreature", LuaInterface::luaDoMoveCreature);
//doPlayerSetPzLocked(cid, locked)
lua_register(m_luaState, "doPlayerSetPzLocked", LuaInterface::luaDoPlayerSetPzLocked);
//doPlayerSetTown(cid, townid)
lua_register(m_luaState, "doPlayerSetTown", LuaInterface::luaDoPlayerSetTown);
//doPlayerSetVocation(cid,voc)
lua_register(m_luaState, "doPlayerSetVocation", LuaInterface::luaDoPlayerSetVocation);
//doPlayerRemoveItem(cid, itemid[, count[, subType = -1]])
lua_register(m_luaState, "doPlayerRemoveItem", LuaInterface::luaDoPlayerRemoveItem);
//doPlayerAddExperience(cid, amount)
lua_register(m_luaState, "doPlayerAddExperience", LuaInterface::luaDoPlayerAddExperience);
//doPlayerSetGuildId(cid, id)
lua_register(m_luaState, "doPlayerSetGuildId", LuaInterface::luaDoPlayerSetGuildId);
//doPlayerSetGuildLevel(cid, level[, rank])
lua_register(m_luaState, "doPlayerSetGuildLevel", LuaInterface::luaDoPlayerSetGuildLevel);
//doPlayerSetGuildNick(cid, nick)
lua_register(m_luaState, "doPlayerSetGuildNick", LuaInterface::luaDoPlayerSetGuildNick);
//doPlayerAddOutfit(cid, looktype, addon)
lua_register(m_luaState, "doPlayerAddOutfit", LuaInterface::luaDoPlayerAddOutfit);
//doPlayerRemoveOutfit(cid, looktype[, addon = 0])
lua_register(m_luaState, "doPlayerRemoveOutfit", LuaInterface::luaDoPlayerRemoveOutfit);
//doPlayerAddOutfitId(cid, outfitId, addon)
lua_register(m_luaState, "doPlayerAddOutfitId", LuaInterface::luaDoPlayerAddOutfitId);
//doPlayerRemoveOutfitId(cid, outfitId[, addon = 0])
lua_register(m_luaState, "doPlayerRemoveOutfitId", LuaInterface::luaDoPlayerRemoveOutfitId);
//canPlayerWearOutfit(cid, looktype[, addon = 0])
lua_register(m_luaState, "canPlayerWearOutfit", LuaInterface::luaCanPlayerWearOutfit);
//canPlayerWearOutfitId(cid, outfitId[, addon = 0])
lua_register(m_luaState, "canPlayerWearOutfitId", LuaInterface::luaCanPlayerWearOutfitId);
//getCreatureCondition(cid, condition[, subId = 0])
lua_register(m_luaState, "getCreatureCondition", LuaInterface::luaGetCreatureCondition);
//doCreatureSetDropLoot(cid, doDrop)
lua_register(m_luaState, "doCreatureSetDropLoot", LuaInterface::luaDoCreatureSetDropLoot);
//getPlayerLossPercent(cid, lossType)
lua_register(m_luaState, "getPlayerLossPercent", LuaInterface::luaGetPlayerLossPercent);
//doPlayerSetLossPercent(cid, lossType, newPercent)
lua_register(m_luaState, "doPlayerSetLossPercent", LuaInterface::luaDoPlayerSetLossPercent);
//doPlayerSetLossSkill(cid, doLose)
lua_register(m_luaState, "doPlayerSetLossSkill", LuaInterface::luaDoPlayerSetLossSkill);
//getPlayerLossSkill(cid)
lua_register(m_luaState, "getPlayerLossSkill", LuaInterface::luaGetPlayerLossSkill);
//doPlayerSwitchSaving(cid)
lua_register(m_luaState, "doPlayerSwitchSaving", LuaInterface::luaDoPlayerSwitchSaving);
//doPlayerSave(cid[, shallow = false])
lua_register(m_luaState, "doPlayerSave", LuaInterface::luaDoPlayerSave);
//isPlayerPzLocked(cid)
lua_register(m_luaState, "isPlayerPzLocked", LuaInterface::luaIsPlayerPzLocked);
//isPlayerSaving(cid)
lua_register(m_luaState, "isPlayerSaving", LuaInterface::luaIsPlayerSaving);
//isCreature(cid)
lua_register(m_luaState, "isCreature", LuaInterface::luaIsCreature);
//isContainer(uid)
lua_register(m_luaState, "isContainer", LuaInterface::luaIsContainer);
//isMovable(uid)
lua_register(m_luaState, "isMovable", LuaInterface::luaIsMovable);
//getCreatureByName(name)
lua_register(m_luaState, "getCreatureByName", LuaInterface::luaGetCreatureByName);
//getPlayerByGUID(guid)
lua_register(m_luaState, "getPlayerByGUID", LuaInterface::luaGetPlayerByGUID);
//getPlayerByNameWildcard(name~[, ret = false])
lua_register(m_luaState, "getPlayerByNameWildcard", LuaInterface::luaGetPlayerByNameWildcard);
//getPlayerGUIDByName(name[, multiworld = false])
lua_register(m_luaState, "getPlayerGUIDByName", LuaInterface::luaGetPlayerGUIDByName);
//getPlayerNameByGUID(guid[, multiworld = false[, displayError = true]])
lua_register(m_luaState, "getPlayerNameByGUID", LuaInterface::luaGetPlayerNameByGUID);
//registerCreatureEvent(uid, eventName)
lua_register(m_luaState, "registerCreatureEvent", LuaInterface::luaRegisterCreatureEvent);
//unregisterCreatureEvent(uid, eventName)
lua_register(m_luaState, "unregisterCreatureEvent", LuaInterface::luaUnregisterCreatureEvent);
//getContainerSize(uid)
lua_register(m_luaState, "getContainerSize", LuaInterface::luaGetContainerSize);
//getContainerCap(uid)
lua_register(m_luaState, "getContainerCap", LuaInterface::luaGetContainerCap);
//getContainerItem(uid, slot)
lua_register(m_luaState, "getContainerItem", LuaInterface::luaGetContainerItem);
//doAddContainerItem(uid, itemid[, count/subType = 1])
lua_register(m_luaState, "doAddContainerItem", LuaInterface::luaDoAddContainerItem);
//getHouseInfo(houseId[, displayError = true])
lua_register(m_luaState, "getHouseInfo", LuaInterface::luaGetHouseInfo);
//getHouseAccessList(houseid, listId)
lua_register(m_luaState, "getHouseAccessList", LuaInterface::luaGetHouseAccessList);
//getHouseByPlayerGUID(playerGUID)
lua_register(m_luaState, "getHouseByPlayerGUID", LuaInterface::luaGetHouseByPlayerGUID);
//getHouseFromPos(pos)
lua_register(m_luaState, "getHouseFromPos", LuaInterface::luaGetHouseFromPos);
//setHouseAccessList(houseid, listid, listtext)
lua_register(m_luaState, "setHouseAccessList", LuaInterface::luaSetHouseAccessList);
//setHouseOwner(houseId, owner[, clean])
lua_register(m_luaState, "setHouseOwner", LuaInterface::luaSetHouseOwner);
//getWorldType()
lua_register(m_luaState, "getWorldType", LuaInterface::luaGetWorldType);
//setWorldType(type)
lua_register(m_luaState, "setWorldType", LuaInterface::luaSetWorldType);
//getWorldTime()
lua_register(m_luaState, "getWorldTime", LuaInterface::luaGetWorldTime);
//getWorldLight()
lua_register(m_luaState, "getWorldLight", LuaInterface::luaGetWorldLight);
//getWorldCreatures(type)
//0 players, 1 monsters, 2 npcs, 3 all
lua_register(m_luaState, "getWorldCreatures", LuaInterface::luaGetWorldCreatures);
//getWorldUpTime()
lua_register(m_luaState, "getWorldUpTime", LuaInterface::luaGetWorldUpTime);
//getGuildId(guildName)
lua_register(m_luaState, "getGuildId", LuaInterface::luaGetGuildId);
//getGuildMotd(guildId)
lua_register(m_luaState, "getGuildMotd", LuaInterface::luaGetGuildMotd);
//getPlayerSex(cid[, full = false])
lua_register(m_luaState, "getPlayerSex", LuaInterface::luaGetPlayerSex);
//doPlayerSetSex(cid, newSex)
lua_register(m_luaState, "doPlayerSetSex", LuaInterface::luaDoPlayerSetSex);
//createCombatArea({area}[, {extArea}])
lua_register(m_luaState, "createCombatArea", LuaInterface::luaCreateCombatArea);
//createConditionObject(type[, ticks[, buff[, subId]]])
lua_register(m_luaState, "createConditionObject", LuaInterface::luaCreateConditionObject);
//setCombatArea(combat, area)
lua_register(m_luaState, "setCombatArea", LuaInterface::luaSetCombatArea);
//setCombatCondition(combat, condition)
lua_register(m_luaState, "setCombatCondition", LuaInterface::luaSetCombatCondition);
//setCombatParam(combat, key, value)
lua_register(m_luaState, "setCombatParam", LuaInterface::luaSetCombatParam);
//setConditionParam(condition, key, value)
lua_register(m_luaState, "setConditionParam", LuaInterface::luaSetConditionParam);
//addDamageCondition(condition, rounds, time, value)
lua_register(m_luaState, "addDamageCondition", LuaInterface::luaAddDamageCondition);
//addOutfitCondition(condition, outfit)
lua_register(m_luaState, "addOutfitCondition", LuaInterface::luaAddOutfitCondition);
//setCombatCallBack(combat, key, function_name)
lua_register(m_luaState, "setCombatCallback", LuaInterface::luaSetCombatCallBack);
//setCombatFormula(combat, type, mina, minb, maxa, maxb[, minl, maxl[, minm, maxm[, minc[, maxc]]]])
lua_register(m_luaState, "setCombatFormula", LuaInterface::luaSetCombatFormula);
//setConditionFormula(combat, mina, minb, maxa, maxb)
lua_register(m_luaState, "setConditionFormula", LuaInterface::luaSetConditionFormula);
//doCombat(cid, combat, param)
lua_register(m_luaState, "doCombat", LuaInterface::luaDoCombat);
//createCombatObject()
lua_register(m_luaState, "createCombatObject", LuaInterface::luaCreateCombatObject);
//doCombatAreaHealth(cid, type, pos, area, min, max, effect)
lua_register(m_luaState, "doCombatAreaHealth", LuaInterface::luaDoCombatAreaHealth);
//doTargetCombatHealth(cid, target, type, min, max, effect)
lua_register(m_luaState, "doTargetCombatHealth", LuaInterface::luaDoTargetCombatHealth);
//doCombatAreaMana(cid, pos, area, min, max, effect)
lua_register(m_luaState, "doCombatAreaMana", LuaInterface::luaDoCombatAreaMana);
//doTargetCombatMana(cid, target, min, max, effect)
lua_register(m_luaState, "doTargetCombatMana", LuaInterface::luaDoTargetCombatMana);
//doCombatAreaCondition(cid, pos, area, condition, effect)
lua_register(m_luaState, "doCombatAreaCondition", LuaInterface::luaDoCombatAreaCondition);
//doTargetCombatCondition(cid, target, condition, effect)
lua_register(m_luaState, "doTargetCombatCondition", LuaInterface::luaDoTargetCombatCondition);
//doCombatAreaDispel(cid, pos, area, type, effect)
lua_register(m_luaState, "doCombatAreaDispel", LuaInterface::luaDoCombatAreaDispel);
//doTargetCombatDispel(cid, target, type, effect)
lua_register(m_luaState, "doTargetCombatDispel", LuaInterface::luaDoTargetCombatDispel);
//doChallengeCreature(cid, target)
lua_register(m_luaState, "doChallengeCreature", LuaInterface::luaDoChallengeCreature);
//numberToVariant(number)
lua_register(m_luaState, "numberToVariant", LuaInterface::luaNumberToVariant);
//stringToVariant(string)
lua_register(m_luaState, "stringToVariant", LuaInterface::luaStringToVariant);
//positionToVariant(pos)
lua_register(m_luaState, "positionToVariant", LuaInterface::luaPositionToVariant);
//targetPositionToVariant(pos)
lua_register(m_luaState, "targetPositionToVariant", LuaInterface::luaTargetPositionToVariant);
//variantToNumber(var)
lua_register(m_luaState, "variantToNumber", LuaInterface::luaVariantToNumber);
//variantToString(var)
lua_register(m_luaState, "variantToString", LuaInterface::luaVariantToString);
//variantToPosition(var)
lua_register(m_luaState, "variantToPosition", LuaInterface::luaVariantToPosition);
//doChangeSpeed(cid, delta)
lua_register(m_luaState, "doChangeSpeed", LuaInterface::luaDoChangeSpeed);
//doCreatureChangeOutfit(cid, outfit)
lua_register(m_luaState, "doCreatureChangeOutfit", LuaInterface::luaDoCreatureChangeOutfit);
//doSetMonsterOutfit(cid, name[, time = -1])
lua_register(m_luaState, "doSetMonsterOutfit", LuaInterface::luaSetMonsterOutfit);
//doSetItemOutfit(cid, item[, time = -1])
lua_register(m_luaState, "doSetItemOutfit", LuaInterface::luaSetItemOutfit);
//doSetCreatureOutfit(cid, outfit[, time = -1])
lua_register(m_luaState, "doSetCreatureOutfit", LuaInterface::luaSetCreatureOutfit);
//getCreatureOutfit(cid)
lua_register(m_luaState, "getCreatureOutfit", LuaInterface::luaGetCreatureOutfit);
//getCreatureLastPosition(cid)
lua_register(m_luaState, "getCreatureLastPosition", LuaInterface::luaGetCreatureLastPosition);
//getCreatureName(cid)
lua_register(m_luaState, "getCreatureName", LuaInterface::luaGetCreatureName);
//getCreatureSpeed(cid)
lua_register(m_luaState, "getCreatureSpeed", LuaInterface::luaGetCreatureSpeed);
//getCreatureBaseSpeed(cid)
lua_register(m_luaState, "getCreatureBaseSpeed", LuaInterface::luaGetCreatureBaseSpeed);
//getCreatureTarget(cid)
lua_register(m_luaState, "getCreatureTarget", LuaInterface::luaGetCreatureTarget);
//isSightClear(fromPos, toPos, floorCheck)
lua_register(m_luaState, "isSightClear", LuaInterface::luaIsSightClear);
//isInArray(array, value[, caseSensitive = false])
lua_register(m_luaState, "isInArray", LuaInterface::luaIsInArray);
//addEvent(callback, delay, ...)
lua_register(m_luaState, "addEvent", LuaInterface::luaAddEvent);
//stopEvent(eventid)
lua_register(m_luaState, "stopEvent", LuaInterface::luaStopEvent);
//getPlayersByAccountId(accId)
lua_register(m_luaState, "getPlayersByAccountId", LuaInterface::luaGetPlayersByAccountId);
//getAccountIdByName(name)
lua_register(m_luaState, "getAccountIdByName", LuaInterface::luaGetAccountIdByName);
//getAccountByName(name)
lua_register(m_luaState, "getAccountByName", LuaInterface::luaGetAccountByName);
//getAccountIdByAccount(accName)
lua_register(m_luaState, "getAccountIdByAccount", LuaInterface::luaGetAccountIdByAccount);
//getAccountByAccountId(accId)
lua_register(m_luaState, "getAccountByAccountId", LuaInterface::luaGetAccountByAccountId);
//getIpByName(name)
lua_register(m_luaState, "getIpByName", LuaInterface::luaGetIpByName);
//getPlayersByIp(ip[, mask = 0xFFFFFFFF])
lua_register(m_luaState, "getPlayersByIp", LuaInterface::luaGetPlayersByIp);
//doPlayerPopupFYI(cid, message)
lua_register(m_luaState, "doPlayerPopupFYI", LuaInterface::luaDoPlayerPopupFYI);
//doPlayerSendTutorial(cid, id)
lua_register(m_luaState, "doPlayerSendTutorial", LuaInterface::luaDoPlayerSendTutorial);
//doPlayerSendMailByName(name, item[, town[, actor]])
lua_register(m_luaState, "doPlayerSendMailByName", LuaInterface::luaDoPlayerSendMailByName);
//doPlayerAddMapMark(cid, pos, type[, description])
lua_register(m_luaState, "doPlayerAddMapMark", LuaInterface::luaDoPlayerAddMapMark);
//doPlayerAddPremiumDays(cid, days)
lua_register(m_luaState, "doPlayerAddPremiumDays", LuaInterface::luaDoPlayerAddPremiumDays);
//getPlayerPremiumDays(cid)
lua_register(m_luaState, "getPlayerPremiumDays", LuaInterface::luaGetPlayerPremiumDays);
//doCreatureSetLookDirection(cid, dir)
lua_register(m_luaState, "doCreatureSetLookDirection", LuaInterface::luaDoCreatureSetLookDir);
//getCreatureGuildEmblem(cid[, target])
lua_register(m_luaState, "getCreatureGuildEmblem", LuaInterface::luaGetCreatureGuildEmblem);
//doCreatureSetGuildEmblem(cid, emblem)
lua_register(m_luaState, "doCreatureSetGuildEmblem", LuaInterface::luaDoCreatureSetGuildEmblem);
//getCreaturePartyShield(cid[, target])
lua_register(m_luaState, "getCreaturePartyShield", LuaInterface::luaGetCreaturePartyShield);
//doCreatureSetPartyShield(cid, shield)
lua_register(m_luaState, "doCreatureSetPartyShield", LuaInterface::luaDoCreatureSetPartyShield);
//getCreatureSkullType(cid[, target])
lua_register(m_luaState, "getCreatureSkullType", LuaInterface::luaGetCreatureSkullType);
//doCreatureSetSkullType(cid, skull)
lua_register(m_luaState, "doCreatureSetSkullType", LuaInterface::luaDoCreatureSetSkullType);
//getPlayerSkullEnd(cid)
lua_register(m_luaState, "getPlayerSkullEnd", LuaInterface::luaGetPlayerSkullEnd);
//doPlayerSetSkullEnd(cid, time, type)
lua_register(m_luaState, "doPlayerSetSkullEnd", LuaInterface::luaDoPlayerSetSkullEnd);
//getPlayerBlessing(cid, blessing)
lua_register(m_luaState, "getPlayerBlessing", LuaInterface::luaGetPlayerBlessing);
//doPlayerAddBlessing(cid, blessing)
lua_register(m_luaState, "doPlayerAddBlessing", LuaInterface::luaDoPlayerAddBlessing);
//getPlayerStamina(cid)
lua_register(m_luaState, "getPlayerStamina", LuaInterface::luaGetPlayerStamina);
//doPlayerSetStamina(cid, minutes)
lua_register(m_luaState, "doPlayerSetStamina", LuaInterface::luaDoPlayerSetStamina);
//getPlayerBalance(cid)
lua_register(m_luaState, "getPlayerBalance", LuaInterface::luaGetPlayerBalance);
//doPlayerSetBalance(cid, balance)
lua_register(m_luaState, "doPlayerSetBalance", LuaInterface::luaDoPlayerSetBalance);
//getCreatureNoMove(cid)
lua_register(m_luaState, "getCreatureNoMove", LuaInterface::luaGetCreatureNoMove);
//doCreatureSetNoMove(cid, block)
lua_register(m_luaState, "doCreatureSetNoMove", LuaInterface::luaDoCreatureSetNoMove);
//getPlayerIdleTime(cid)
lua_register(m_luaState, "getPlayerIdleTime", LuaInterface::luaGetPlayerIdleTime);
//doPlayerSetIdleTime(cid, amount)
lua_register(m_luaState, "doPlayerSetIdleTime", LuaInterface::luaDoPlayerSetIdleTime);
//getPlayerLastLoad(cid)
lua_register(m_luaState, "getPlayerLastLoad", LuaInterface::luaGetPlayerLastLoad);
//getPlayerLastLogin(cid)
lua_register(m_luaState, "getPlayerLastLogin", LuaInterface::luaGetPlayerLastLogin);
//getPlayerAccountManager(cid)
lua_register(m_luaState, "getPlayerAccountManager", LuaInterface::luaGetPlayerAccountManager);
//getPlayerTradeState(cid)
lua_register(m_luaState, "getPlayerTradeState", LuaInterface::luaGetPlayerTradeState);
//getPlayerModes(cid)
lua_register(m_luaState, "getPlayerModes", LuaInterface::luaGetPlayerModes);
//getPlayerRates(cid)
lua_register(m_luaState, "getPlayerRates", LuaInterface::luaGetPlayerRates);
//doPlayerSetRate(cid, type, value)
lua_register(m_luaState, "doPlayerSetRate", LuaInterface::luaDoPlayerSetRate);
//getPlayerPartner(cid)
lua_register(m_luaState, "getPlayerPartner", LuaInterface::luaGetPlayerPartner);
//doPlayerSetPartner(cid, guid)
lua_register(m_luaState, "doPlayerSetPartner", LuaInterface::luaDoPlayerSetPartner);
//doPlayerFollowCreature(cid, target)
lua_register(m_luaState, "doPlayerFollowCreature", LuaInterface::luaDoPlayerFollowCreature);
//getPlayerParty(cid)
lua_register(m_luaState, "getPlayerParty", LuaInterface::luaGetPlayerParty);
//doPlayerJoinParty(cid, lid)
lua_register(m_luaState, "doPlayerJoinParty", LuaInterface::luaDoPlayerJoinParty);
//doPlayerLeaveParty(cid[, forced = false])
lua_register(m_luaState, "doPlayerLeaveParty", LuaInterface::luaDoPlayerLeaveParty);
//getPartyMembers(lid)
lua_register(m_luaState, "getPartyMembers", LuaInterface::luaGetPartyMembers);
//getCreatureMaster(cid)
lua_register(m_luaState, "getCreatureMaster", LuaInterface::luaGetCreatureMaster);
//getCreatureSummons(cid)
lua_register(m_luaState, "getCreatureSummons", LuaInterface::luaGetCreatureSummons);
//getTownId(townName)
lua_register(m_luaState, "getTownId", LuaInterface::luaGetTownId);
//getTownName(townId)
lua_register(m_luaState, "getTownName", LuaInterface::luaGetTownName);
//getTownTemplePosition(townId[, displayError])
lua_register(m_luaState, "getTownTemplePosition", LuaInterface::luaGetTownTemplePosition);
//getTownHouses(townId)
lua_register(m_luaState, "getTownHouses", LuaInterface::luaGetTownHouses);
//getSpectators(centerPos, rangex, rangey[, multifloor = false])
lua_register(m_luaState, "getSpectators", LuaInterface::luaGetSpectators);
//getVocationInfo(id)
lua_register(m_luaState, "getVocationInfo", LuaInterface::luaGetVocationInfo);
//getGroupInfo(id[, premium = false])
lua_register(m_luaState, "getGroupInfo", LuaInterface::luaGetGroupInfo);
//getTownList()
lua_register(m_luaState, "getTownList", LuaInterface::luaGetTownList);
//getWaypointList()
lua_register(m_luaState, "getWaypointList", LuaInterface::luaGetWaypointList);
//getTalkActionList()
lua_register(m_luaState, "getTalkActionList", LuaInterface::luaGetTalkActionList);
//getExperienceStageList()
lua_register(m_luaState, "getExperienceStageList", LuaInterface::luaGetExperienceStageList);
//getItemIdByName(name[, displayError = true])
lua_register(m_luaState, "getItemIdByName", LuaInterface::luaGetItemIdByName);
//getItemInfo(itemid)
lua_register(m_luaState, "getItemInfo", LuaInterface::luaGetItemInfo);
//getItemAttribute(uid, key)
lua_register(m_luaState, "getItemAttribute", LuaInterface::luaGetItemAttribute);
//doItemSetAttribute(uid, key, value)
lua_register(m_luaState, "doItemSetAttribute", LuaInterface::luaDoItemSetAttribute);
//doItemEraseAttribute(uid, key)
lua_register(m_luaState, "doItemEraseAttribute", LuaInterface::luaDoItemEraseAttribute);
//getItemWeight(uid[, precise = true])
lua_register(m_luaState, "getItemWeight", LuaInterface::luaGetItemWeight);
//getItemParent(uid)
lua_register(m_luaState, "getItemParent", LuaInterface::luaGetItemParent);
//hasItemProperty(uid, prop)
lua_register(m_luaState, "hasItemProperty", LuaInterface::luaHasItemProperty);
//hasPlayerClient(cid)
lua_register(m_luaState, "hasPlayerClient", LuaInterface::luaHasPlayerClient);
//isIpBanished(ip[, mask])
lua_register(m_luaState, "isIpBanished", LuaInterface::luaIsIpBanished);
//isPlayerBanished(name/guid, type)
lua_register(m_luaState, "isPlayerBanished", LuaInterface::luaIsPlayerBanished);
//isAccountBanished(accountId[, playerId])
lua_register(m_luaState, "isAccountBanished", LuaInterface::luaIsAccountBanished);
//doAddIpBanishment(...)
lua_register(m_luaState, "doAddIpBanishment", LuaInterface::luaDoAddIpBanishment);
//doAddPlayerBanishment(...)
lua_register(m_luaState, "doAddPlayerBanishment", LuaInterface::luaDoAddPlayerBanishment);
//doAddAccountBanishment(...)
lua_register(m_luaState, "doAddAccountBanishment", LuaInterface::luaDoAddAccountBanishment);
//doAddNotation(...)
lua_register(m_luaState, "doAddNotation", LuaInterface::luaDoAddNotation);
//doAddStatement(...)
lua_register(m_luaState, "doAddStatement", LuaInterface::luaDoAddStatement);
//doRemoveIpBanishment(ip[, mask])
lua_register(m_luaState, "doRemoveIpBanishment", LuaInterface::luaDoRemoveIpBanishment);
//doRemovePlayerBanishment(name/guid, type)
lua_register(m_luaState, "doRemovePlayerBanishment", LuaInterface::luaDoRemovePlayerBanishment);
//doRemoveAccountBanishment(accountId[, playerId])
lua_register(m_luaState, "doRemoveAccountBanishment", LuaInterface::luaDoRemoveAccountBanishment);
//doRemoveNotations(accountId[, playerId])
lua_register(m_luaState, "doRemoveNotations", LuaInterface::luaDoRemoveNotations);
//doRemoveStatements(name/guid[, channelId])
lua_register(m_luaState, "doRemoveStatements", LuaInterface::luaDoRemoveStatements);
//getNotationsCount(accountId[, playerId])
lua_register(m_luaState, "getNotationsCount", LuaInterface::luaGetNotationsCount);
//getStatementsCount(name/guid[, channelId])
lua_register(m_luaState, "getStatementsCount", LuaInterface::luaGetStatementsCount);
//getBanData(value[, type[, param]])
lua_register(m_luaState, "getBanData", LuaInterface::luaGetBanData);
//getBanReason(id)
lua_register(m_luaState, "getBanReason", LuaInterface::luaGetBanReason);
//getBanAction(id[, ipBanishment = false])
lua_register(m_luaState, "getBanAction", LuaInterface::luaGetBanAction);
//getBanList(type[, value[, param]])
lua_register(m_luaState, "getBanList", LuaInterface::luaGetBanList);
//getExperienceStage(level)
lua_register(m_luaState, "getExperienceStage", LuaInterface::luaGetExperienceStage);
//getDataDir()
lua_register(m_luaState, "getDataDir", LuaInterface::luaGetDataDir);
//getLogsDir()
lua_register(m_luaState, "getLogsDir", LuaInterface::luaGetLogsDir);
//getConfigFile()
lua_register(m_luaState, "getConfigFile", LuaInterface::luaGetConfigFile);
//getConfigValue(key)
lua_register(m_luaState, "getConfigValue", LuaInterface::luaGetConfigValue);
//getModList()
lua_register(m_luaState, "getModList", LuaInterface::luaGetModList);
//getHighscoreString(skillId)
lua_register(m_luaState, "getHighscoreString", LuaInterface::luaGetHighscoreString);
//getWaypointPosition(name)
lua_register(m_luaState, "getWaypointPosition", LuaInterface::luaGetWaypointPosition);
//doWaypointAddTemporial(name, pos)
lua_register(m_luaState, "doWaypointAddTemporial", LuaInterface::luaDoWaypointAddTemporial);
//getGameState()
lua_register(m_luaState, "getGameState", LuaInterface::luaGetGameState);
//doSetGameState(id)
lua_register(m_luaState, "doSetGameState", LuaInterface::luaDoSetGameState);
//doExecuteRaid(name)
lua_register(m_luaState, "doExecuteRaid", LuaInterface::luaDoExecuteRaid);
//doCreatureExecuteTalkAction(cid, text[, ignoreAccess = false[, channelId = CHANNEL_DEFAULT]])
lua_register(m_luaState, "doCreatureExecuteTalkAction", LuaInterface::luaDoCreatureExecuteTalkAction);
//doReloadInfo(id[, cid])
lua_register(m_luaState, "doReloadInfo", LuaInterface::luaDoReloadInfo);
//doSaveServer([shallow = false])
lua_register(m_luaState, "doSaveServer", LuaInterface::luaDoSaveServer);
//doCleanHouse(houseId)
lua_register(m_luaState, "doCleanHouse", LuaInterface::luaDoCleanHouse);
//doCleanMap()
lua_register(m_luaState, "doCleanMap", LuaInterface::luaDoCleanMap);
//doRefreshMap()
lua_register(m_luaState, "doRefreshMap", LuaInterface::luaDoRefreshMap);
#ifdef __WAR_SYSTEM__
//doGuildAddEnemy(guild, enemy, war, type)
lua_register(m_luaState, "doGuildAddEnemy", LuaInterface::luaDoGuildAddEnemy);
//doGuildRemoveEnemy(guild, enemy)
lua_register(m_luaState, "doGuildRemoveEnemy", LuaInterface::luaDoGuildRemoveEnemy);
#endif
//doUpdateHouseAuctions()
lua_register(m_luaState, "doUpdateHouseAuctions", LuaInterface::luaDoUpdateHouseAuctions);
//loadmodlib(lib)
lua_register(m_luaState, "loadmodlib", LuaInterface::luaL_loadmodlib);
//domodlib(lib)
lua_register(m_luaState, "domodlib", LuaInterface::luaL_domodlib);
//dodirectory(dir)
lua_register(m_luaState, "dodirectory", LuaInterface::luaL_dodirectory);
//errors(var)
lua_register(m_luaState, "errors", LuaInterface::luaL_errors);
//os table
luaL_register(m_luaState, "os", LuaInterface::luaSystemTable);
//db table
luaL_register(m_luaState, "db", LuaInterface::luaDatabaseTable);
//result table
luaL_register(m_luaState, "result", LuaInterface::luaResultTable);
//bit table
luaL_register(m_luaState, "bit", LuaInterface::luaBitTable);
//std table
luaL_register(m_luaState, "std", LuaInterface::luaStdTable);
}
const luaL_Reg LuaInterface::luaSystemTable[] =
{
//os.mtime()
{"mtime", LuaInterface::luaSystemTime},
{NULL, NULL}
};
const luaL_Reg LuaInterface::luaDatabaseTable[] =
{
//db.query(query)
{"query", LuaInterface::luaDatabaseExecute},
//db.storeQuery(query)
{"storeQuery", LuaInterface::luaDatabaseStoreQuery},
//db.escapeString(str)
{"escapeString", LuaInterface::luaDatabaseEscapeString},
//db.escapeBlob(s, length)
{"escapeBlob", LuaInterface::luaDatabaseEscapeBlob},
//db.lastInsertId()
{"lastInsertId", LuaInterface::luaDatabaseLastInsertId},
//db.stringComparer()
{"stringComparer", LuaInterface::luaDatabaseStringComparer},
//db.updateLimiter()
{"updateLimiter", LuaInterface::luaDatabaseUpdateLimiter},
{NULL, NULL}
};
const luaL_Reg LuaInterface::luaResultTable[] =
{
//result.getDataInt(resId, s)
{"getDataInt", LuaInterface::luaResultGetDataInt},
//result.getDataLong(resId, s)
{"getDataLong", LuaInterface::luaResultGetDataLong},
//result.getDataString(resId, s)
{"getDataString", LuaInterface::luaResultGetDataString},
//result.getDataStream(resId, s, length)
{"getDataStream", LuaInterface::luaResultGetDataStream},
//result.next(resId)
{"next", LuaInterface::luaResultNext},
//result.free(resId)
{"free", LuaInterface::luaResultFree},
{NULL, NULL}
};
const luaL_Reg LuaInterface::luaBitTable[] =
{
//{"cast", LuaInterface::luaBitCast},
{"bnot", LuaInterface::luaBitNot},
{"band", LuaInterface::luaBitAnd},
{"bor", LuaInterface::luaBitOr},
{"bxor", LuaInterface::luaBitXor},
{"lshift", LuaInterface::luaBitLeftShift},
{"rshift", LuaInterface::luaBitRightShift},
//{"arshift", LuaInterface::luaBitArithmeticalRightShift},
//{"ucast", LuaInterface::luaBitUCast},
{"ubnot", LuaInterface::luaBitUNot},
{"uband", LuaInterface::luaBitUAnd},
{"ubor", LuaInterface::luaBitUOr},
{"ubxor", LuaInterface::luaBitUXor},
{"ulshift", LuaInterface::luaBitULeftShift},
{"urshift", LuaInterface::luaBitURightShift},
//{"uarshift", LuaInterface::luaBitUArithmeticalRightShift},
{NULL, NULL}
};
const luaL_Reg LuaInterface::luaStdTable[] =
{
{"cout", LuaInterface::luaStdCout},
{"clog", LuaInterface::luaStdClog},
{"cerr", LuaInterface::luaStdCerr},
{"md5", LuaInterface::luaStdMD5},
{"sha1", LuaInterface::luaStdSHA1},
{"sha256", LuaInterface::luaStdSHA256},
{"sha512", LuaInterface::luaStdSHA512},
{"vahash", LuaInterface::luaStdVAHash},
{NULL, NULL}
};
int32_t LuaInterface::internalGetPlayerInfo(lua_State* L, PlayerInfo_t info)
{
ScriptEnviroment* env = getEnv();
const Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
std::stringstream s;
s << getError(LUA_ERROR_PLAYER_NOT_FOUND) << " when requesting player info #" << info;
errorEx(s.str());
lua_pushboolean(L, false);
return 1;
}
int64_t value = 0;
Position pos;
switch(info)
{
case PlayerInfoNameDescription:
lua_pushstring(L, player->getNameDescription().c_str());
return 1;
case PlayerInfoSpecialDescription:
lua_pushstring(L, player->getSpecialDescription().c_str());
return 1;
case PlayerInfoAccess:
value = player->getAccess();
break;
case PlayerInfoGhostAccess:
value = player->getGhostAccess();
break;
case PlayerInfoLevel:
value = player->getLevel();
break;
case PlayerInfoExperience:
value = player->getExperience();
break;
case PlayerInfoManaSpent:
value = player->getSpentMana();
break;
case PlayerInfoTown:
value = player->getTown();
break;
case PlayerInfoPromotionLevel:
value = player->getPromotionLevel();
break;
case PlayerInfoGUID:
value = player->getGUID();
break;
case PlayerInfoAccountId:
value = player->getAccount();
break;
case PlayerInfoAccount:
lua_pushstring(L, player->getAccountName().c_str());
return 1;
case PlayerInfoPremiumDays:
value = player->getPremiumDays();
break;
case PlayerInfoFood:
{
if(Condition* condition = player->getCondition(CONDITION_REGENERATION, CONDITIONID_DEFAULT))
value = condition->getTicks() / 1000;
break;
}
case PlayerInfoVocation:
value = player->getVocationId();
break;
case PlayerInfoSoul:
value = player->getSoul();
break;
case PlayerInfoFreeCap:
value = (int64_t)player->getFreeCapacity();
break;
case PlayerInfoGuildId:
value = player->getGuildId();
break;
case PlayerInfoGuildName:
lua_pushstring(L, player->getGuildName().c_str());
return 1;
case PlayerInfoGuildRankId:
value = player->getRankId();
break;
case PlayerInfoGuildRank:
lua_pushstring(L, player->getRankName().c_str());
return 1;
case PlayerInfoGuildLevel:
value = player->getGuildLevel();
break;
case PlayerInfoGuildNick:
lua_pushstring(L, player->getGuildNick().c_str());
return 1;
case PlayerInfoGroupId:
value = player->getGroupId();
break;
case PlayerInfoBalance:
if(g_config.getBool(ConfigManager::BANK_SYSTEM))
lua_pushnumber(L, player->balance);
else
lua_pushnumber(L, 0);
return 1;
case PlayerInfoStamina:
value = player->getStaminaMinutes();
break;
case PlayerInfoLossSkill:
lua_pushboolean(L, player->getLossSkill());
return 1;
case PlayerInfoMarriage:
value = player->marriage;
break;
case PlayerInfoPzLock:
lua_pushboolean(L, player->isPzLocked());
return 1;
case PlayerInfoSaving:
lua_pushboolean(L, player->isSaving());
return 1;
case PlayerInfoIp:
value = player->getIP();
break;
case PlayerInfoSkullEnd:
value = player->getSkullEnd();
break;
case PlayerInfoOutfitWindow:
player->sendOutfitWindow();
lua_pushboolean(L, true);
return 1;
case PlayerInfoIdleTime:
value = player->getIdleTime();
break;
case PlayerInfoClient:
lua_pushboolean(L, player->hasClient());
return 1;
case PlayerInfoLastLoad:
value = player->getLastLoad();
break;
case PlayerInfoLastLogin:
value = player->getLastLogin();
break;
case PlayerInfoAccountManager:
value = player->accountManager;
break;
case PlayerInfoTradeState:
value = player->tradeState;
break;
default:
errorEx("Unknown player info #" + info);
value = 0;
break;
}
lua_pushnumber(L, value);
return 1;
}
//getPlayer[Info](uid)
int32_t LuaInterface::luaGetPlayerNameDescription(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoNameDescription);
}
int32_t LuaInterface::luaGetPlayerSpecialDescription(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoSpecialDescription);
}
int32_t LuaInterface::luaGetPlayerFood(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoFood);
}
int32_t LuaInterface::luaGetPlayerAccess(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoAccess);
}
int32_t LuaInterface::luaGetPlayerGhostAccess(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoGhostAccess);
}
int32_t LuaInterface::luaGetPlayerLevel(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoLevel);
}
int32_t LuaInterface::luaGetPlayerExperience(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoExperience);
}
int32_t LuaInterface::luaGetPlayerSpentMana(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoManaSpent);
}
int32_t LuaInterface::luaGetPlayerVocation(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoVocation);
}
int32_t LuaInterface::luaGetPlayerSoul(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoSoul);
}
int32_t LuaInterface::luaGetPlayerFreeCap(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoFreeCap);
}
int32_t LuaInterface::luaGetPlayerGuildId(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoGuildId);
}
int32_t LuaInterface::luaGetPlayerGuildName(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoGuildName);
}
int32_t LuaInterface::luaGetPlayerGuildRankId(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoGuildRankId);
}
int32_t LuaInterface::luaGetPlayerGuildRank(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoGuildRank);
}
int32_t LuaInterface::luaGetPlayerGuildLevel(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoGuildLevel);
}
int32_t LuaInterface::luaGetPlayerGuildNick(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoGuildNick);
}
int32_t LuaInterface::luaGetPlayerTown(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoTown);
}
int32_t LuaInterface::luaGetPlayerPromotionLevel(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoPromotionLevel);
}
int32_t LuaInterface::luaGetPlayerGroupId(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoGroupId);
}
int32_t LuaInterface::luaGetPlayerGUID(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoGUID);
}
int32_t LuaInterface::luaGetPlayerAccountId(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoAccountId);
}
int32_t LuaInterface::luaGetPlayerAccount(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoAccount);
}
int32_t LuaInterface::luaGetPlayerPremiumDays(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoPremiumDays);
}
int32_t LuaInterface::luaGetPlayerBalance(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoBalance);
}
int32_t LuaInterface::luaGetPlayerStamina(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoStamina);
}
int32_t LuaInterface::luaGetPlayerLossSkill(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoLossSkill);
}
int32_t LuaInterface::luaGetPlayerPartner(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoMarriage);
}
int32_t LuaInterface::luaIsPlayerPzLocked(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoPzLock);
}
int32_t LuaInterface::luaIsPlayerSaving(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoSaving);
}
int32_t LuaInterface::luaGetPlayerIp(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoIp);
}
int32_t LuaInterface::luaGetPlayerSkullEnd(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoSkullEnd);
}
int32_t LuaInterface::luaDoPlayerSendOutfitWindow(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoOutfitWindow);
}
int32_t LuaInterface::luaGetPlayerIdleTime(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoIdleTime);
}
int32_t LuaInterface::luaHasPlayerClient(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoClient);
}
int32_t LuaInterface::luaGetPlayerLastLoad(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoLastLoad);
}
int32_t LuaInterface::luaGetPlayerLastLogin(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoLastLogin);
}
int32_t LuaInterface::luaGetPlayerAccountManager(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoAccountManager);
}
int32_t LuaInterface::luaGetPlayerTradeState(lua_State* L)
{
return internalGetPlayerInfo(L, PlayerInfoTradeState);
}
//
int32_t LuaInterface::luaGetPlayerSex(lua_State* L)
{
//getPlayerSex(cid[, full = false])
bool full = false;
if(lua_gettop(L) > 1)
full = popNumber(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID((uint32_t)popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
else
lua_pushnumber(L, player->getSex(full));
return 1;
}
int32_t LuaInterface::luaDoPlayerSetNameDescription(lua_State* L)
{
//doPlayerSetNameDescription(cid, description)
std::string description = popString(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->nameDescription += description;
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetSpecialDescription(lua_State* L)
{
//doPlayerSetSpecialDescription(cid, description)
std::string description = popString(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->setSpecialDescription(description);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerMagLevel(lua_State* L)
{
//getPlayerMagLevel(cid[, ignoreBuffs = false])
bool ignoreBuffs = false;
if(lua_gettop(L) > 1)
ignoreBuffs = popNumber(L);
ScriptEnviroment* env = getEnv();
if(const Player* player = env->getPlayerByUID(popNumber(L)))
lua_pushnumber(L, (ignoreBuffs ? player->magLevel : player->getMagicLevel()));
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerRequiredMana(lua_State* L)
{
//getPlayerRequiredMana(cid, magicLevel)
uint32_t magLevel = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
lua_pushnumber(L, player->vocation->getReqMana(magLevel));
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerRequiredSkillTries(lua_State* L)
{
//getPlayerRequiredSkillTries(cid, skillId, skillLevel)
int32_t sLevel = popNumber(L), sId = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
lua_pushnumber(L, player->vocation->getReqSkillTries(sId, sLevel));
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerFlagValue(lua_State* L)
{
//getPlayerFlagValue(cid, flag)
uint32_t index = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(index < PlayerFlag_LastFlag)
lua_pushboolean(L, player->hasFlag((PlayerFlags)index));
else
{
std::stringstream ss;
ss << index;
errorEx("No valid flag index - " + ss.str());
lua_pushboolean(L, false);
}
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerCustomFlagValue(lua_State* L)
{
//getPlayerCustomFlagValue(cid, flag)
uint32_t index = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(index < PlayerCustomFlag_LastFlag)
lua_pushboolean(L, player->hasCustomFlag((PlayerCustomFlags)index));
else
{
std::stringstream ss;
ss << index;
errorEx("No valid flag index - " + ss.str());
lua_pushboolean(L, false);
}
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerLearnInstantSpell(lua_State* L)
{
//doPlayerLearnInstantSpell(cid, name)
std::string spellName = popString(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
InstantSpell* spell = g_spells->getInstantSpellByName(spellName);
if(!spell)
{
lua_pushboolean(L, false);
return 1;
}
player->learnInstantSpell(spell->getName());
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoPlayerUnlearnInstantSpell(lua_State* L)
{
//doPlayerUnlearnInstantSpell(cid, name)
std::string spellName = popString(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
InstantSpell* spell = g_spells->getInstantSpellByName(spellName);
if(!spell)
{
lua_pushboolean(L, false);
return 1;
}
player->unlearnInstantSpell(spell->getName());
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaGetPlayerLearnedInstantSpell(lua_State* L)
{
//getPlayerLearnedInstantSpell(cid, name)
std::string spellName = popString(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
InstantSpell* spell = g_spells->getInstantSpellByName(spellName);
if(!spell)
{
lua_pushboolean(L, false);
return 1;
}
lua_pushboolean(L, player->hasLearnedInstantSpell(spellName));
return 1;
}
int32_t LuaInterface::luaGetPlayerInstantSpellCount(lua_State* L)
{
//getPlayerInstantSpellCount(cid)
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
lua_pushnumber(L, g_spells->getInstantSpellCount(player));
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerInstantSpellInfo(lua_State* L)
{
//getPlayerInstantSpellInfo(cid, index)
uint32_t index = popNumber(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
InstantSpell* spell = g_spells->getInstantSpellByIndex(player, index);
if(!spell)
{
errorEx(getError(LUA_ERROR_SPELL_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
lua_newtable(L);
setField(L, "name", spell->getName());
setField(L, "words", spell->getWords());
setField(L, "level", spell->getLevel());
setField(L, "mlevel", spell->getMagicLevel());
setField(L, "mana", spell->getManaCost(player));
setField(L, "manapercent", spell->getManaPercent());
return 1;
}
int32_t LuaInterface::luaGetInstantSpellInfo(lua_State* L)
{
//getInstantSpellInfo(name)
InstantSpell* spell = g_spells->getInstantSpellByName(popString(L));
if(!spell)
{
errorEx(getError(LUA_ERROR_SPELL_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
lua_newtable(L);
setField(L, "name", spell->getName());
setField(L, "words", spell->getWords());
setField(L, "level", spell->getLevel());
setField(L, "mlevel", spell->getMagicLevel());
setField(L, "mana", spell->getManaCost(NULL));
setField(L, "manapercent", spell->getManaPercent());
return 1;
}
int32_t LuaInterface::luaDoRemoveItem(lua_State* L)
{
//doRemoveItem(uid[, count = -1])
int32_t count = -1;
if(lua_gettop(L) > 1)
count = popNumber(L);
ScriptEnviroment* env = getEnv();
Item* item = env->getItemByUID(popNumber(L));
if(!item)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
if(g_game.internalRemoveItem(NULL, item, count) != RET_NOERROR)
{
lua_pushboolean(L, false);
return 1;
}
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoPlayerRemoveItem(lua_State* L)
{
//doPlayerRemoveItem(cid, itemid, count[, subType = -1])
int32_t subType = -1;
if(lua_gettop(L) > 3)
subType = popNumber(L);
uint32_t count = popNumber(L);
uint16_t itemId = (uint16_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
lua_pushboolean(L, g_game.removeItemOfType(player, itemId, count, subType));
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerFeed(lua_State* L)
{
//doPlayerFeed(cid, food)
int32_t food = (int32_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->addDefaultRegeneration((food * 1000) * 3);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerCastBans(lua_State* L)
{
//getPlayerCastBan(cid)
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
PlayerCast pc = player->getCast();
lua_newtable(L);
for(std::list<CastBan>::iterator it = pc.bans.begin(); it != pc.bans.end(); ++it)
{
createTable(L, it->ip);
setField(L, "name", it->name);
pushTable(L);
}
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerCastMutes(lua_State* L)
{
//getPlayerCastMutes(cid)
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
PlayerCast pc = player->getCast();
lua_newtable(L); 
for(std::list<CastBan>::iterator it = pc.muted.begin(); it != pc.muted.end(); ++it)
{
createTable(L, it->ip);
setField(L, "name", it->name);
pushTable(L);
}
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerRemoveCastMute(lua_State* L)
{
//doPlayerRemoveCastMute(cid, ip)
std::string name = popString(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(player->removeCastMute(name)) 
lua_pushboolean(L, true);
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerAddCastMute(lua_State* L)
{
//doPlayerAddCastMute(cid, ip)
std::string name = popString(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(player->addCastMute(name))
lua_pushboolean(L, true);
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerCastViewers(lua_State* L)
{
//getPlayerCastBan(cid)
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
PlayerCast pc = player->getCast();
lua_newtable(L);
for(AutoList<ProtocolGame>::iterator it = Player::cSpectators.begin(); it != Player::cSpectators.end(); ++it)
{
if(it->second->getPlayer() != player)
continue;
createTable(L, it->first);
setField(L, "name", it->second->getViewerName());
setField(L, "ip", it->second->getIP());
pushTable(L);
}
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerRemoveCastBan(lua_State* L)
{
//doPlayerRemoveCastBan(cid, ip)
std::string name = popString(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(player->removeCastBan(name)) 
lua_pushboolean(L, true);
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerAddCastBan(lua_State* L)
{
//doPlayerAddCastBan(cid, ip)
std::string name = popString(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(player->addCastBan(name))
lua_pushboolean(L, true);
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetCastPassword(lua_State* L)
{
//doPlayerSetCastPassword(cid, password)
std::string str = popString(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->setCastPassword(str);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetCastDescription(lua_State* L)
{
//doPlayerSetCastPassword(cid, password)
std::string str = popString(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->setCastDescription(str);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetCastState(lua_State* L)
{
//doPlayerSetCastState(cid, bool)
bool state = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->setCasting(state);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerCast(lua_State* L)
{
//getPlayerCast(cid)
ScriptEnviroment* env = getEnv();
if(const Player* player = env->getPlayerByUID(popNumber(L)))
{
lua_newtable(L);
setFieldBool(L, "status", player->getCastingState());
setField(L, "password", player->getCastingPassword());
setField(L, "description", player->getCastDescription());
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCastsOnline(lua_State* L)
{
//
ScriptEnviroment* env = getEnv();
AutoList<Player>::iterator it = Player::castAutoList.begin();
lua_newtable(L);
for(int32_t i = 1; it != Player::castAutoList.end(); ++it, ++i)
{
lua_pushnumber(L, i);
lua_pushnumber(L, env->addThing(it->second));
pushTable(L);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSendCancel(lua_State* L)
{
//doPlayerSendCancel(cid, text)
std::string text = popString(L);
ScriptEnviroment* env = getEnv();
if(const Player* player = env->getPlayerByUID(popNumber(L)))
{
player->sendCancel(text);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoSendDefaultCancel(lua_State* L)
{
//doPlayerSendDefaultCancel(cid, ReturnValue)
ReturnValue ret = (ReturnValue)popNumber(L);
ScriptEnviroment* env = getEnv();
if(const Player* player = env->getPlayerByUID(popNumber(L)))
{
player->sendCancelMessage(ret);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetSearchString(lua_State* L)
{
//getSearchString(fromPosition, toPosition[, fromIsCreature = false[, toIsCreature = false]])
PositionEx toPos, fromPos;
bool toIsCreature = false, fromIsCreature = false;
int32_t params = lua_gettop(L);
if(params > 3)
toIsCreature = popNumber(L);
if(params > 2)
fromIsCreature = popNumber(L);
popPosition(L, toPos);
popPosition(L, fromPos);
if(!toPos.x || !toPos.y || !fromPos.x || !fromPos.y)
{
errorEx("wrong position(s) specified.");
lua_pushboolean(L, false);
}
else
lua_pushstring(L, g_game.getSearchString(fromPos, toPos, fromIsCreature, toIsCreature).c_str());
return 1;
}
int32_t LuaInterface::luaGetClosestFreeTile(lua_State* L)
{
//getClosestFreeTile(cid, targetPos[, extended = false[, ignoreHouse = true]])
uint32_t params = lua_gettop(L);
bool ignoreHouse = true, extended = false;
if(params > 3)
ignoreHouse = popNumber(L);
if(params > 2)
extended = popNumber(L);
PositionEx pos;
popPosition(L, pos);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
Position newPos = g_game.getClosestFreeTile(creature, pos, extended, ignoreHouse);
if(newPos.x != 0)
pushPosition(L, newPos, 0);
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoTeleportThing(lua_State* L)
{
//doTeleportThing(cid, newpos[, pushMove = true[, fullTeleport = true]])
bool fullTeleport = true, pushMove = true;
int32_t params = lua_gettop(L);
if(params > 3)
fullTeleport = popNumber(L);
if(params > 2)
pushMove = popNumber(L);
PositionEx pos;
popPosition(L, pos);
ScriptEnviroment* env = getEnv();
if(Thing* tmp = env->getThingByUID(popNumber(L)))
lua_pushboolean(L, g_game.internalTeleport(tmp, pos, !pushMove, FLAG_NOLIMIT, fullTeleport) == RET_NOERROR);
else
{
errorEx(getError(LUA_ERROR_THING_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoTransformItem(lua_State* L)
{
//doTransformItem(uid, newId[, count/subType])
int32_t count = -1;
if(lua_gettop(L) > 2)
count = popNumber(L);
uint16_t newId = popNumber(L);
uint32_t uid = popNumber(L);
ScriptEnviroment* env = getEnv();
Item* item = env->getItemByUID(uid);
if(!item)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
const ItemType& it = Item::items[newId];
if(it.stackable && count > 100)
count = 100;
Item* newItem = g_game.transformItem(item, newId, count);
if(item->isRemoved())
env->removeThing(uid);
if(newItem && newItem != item)
env->insertThing(uid, newItem);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoCreatureSay(lua_State* L)
{
//doCreatureSay(uid, text[, type = SPEAK_SAY[, ghost = false[, cid = 0[, pos]]]])
uint32_t params = lua_gettop(L), cid = 0, uid = 0;
PositionEx pos;
if(params > 5)
popPosition(L, pos);
if(params > 4)
cid = popNumber(L);
bool ghost = false;
if(params > 3)
ghost = popNumber(L);
SpeakClasses type = SPEAK_SAY;
if(params > 2)
type = (SpeakClasses)popNumber(L);
std::string text = popString(L);
uid = popNumber(L);
if(params > 5 && (!pos.x || !pos.y))
{
errorEx("Invalid position specified.");
lua_pushboolean(L, false);
return 1;
}
ScriptEnviroment* env = getEnv();
Creature* creature = env->getCreatureByUID(uid);
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
SpectatorVec list;
if(cid)
{
Creature* target = env->getCreatureByUID(cid);
if(!target)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
list.push_back(target);
}
if(params > 5)
lua_pushboolean(L, g_game.internalCreatureSay(creature, type, text, ghost, &list, &pos));
else
lua_pushboolean(L, g_game.internalCreatureSay(creature, type, text, ghost, &list));
return 1;
}
int32_t LuaInterface::luaDoSendMagicEffect(lua_State* L)
{
//doSendMagicEffect(pos, type[, player])
ScriptEnviroment* env = getEnv();
SpectatorVec list;
if(lua_gettop(L) > 2)
{
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
list.push_back(creature);
}
uint32_t type = popNumber(L);
PositionEx pos;
popPosition(L, pos);
if(pos.x == 0xFFFF)
pos = env->getRealPos();
if(!list.empty())
g_game.addMagicEffect(list, pos, type);
else
g_game.addMagicEffect(pos, type);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoSendDistanceShoot(lua_State* L)
{
//doSendDistanceShoot(fromPos, toPos, type[, player])
ScriptEnviroment* env = getEnv();
SpectatorVec list;
if(lua_gettop(L) > 3)
{
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
list.push_back(creature);
}
uint32_t type = popNumber(L);
PositionEx toPos, fromPos;
popPosition(L, toPos);
popPosition(L, fromPos);
if(fromPos.x == 0xFFFF)
fromPos = env->getRealPos();
if(toPos.x == 0xFFFF)
toPos = env->getRealPos();
if(!list.empty())
g_game.addDistanceEffect(list, fromPos, toPos, type);
else
g_game.addDistanceEffect(fromPos, toPos, type);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoPlayerAddSkillTry(lua_State* L)
{
//doPlayerAddSkillTry(uid, skillid, n[, useMultiplier = true])
bool multiplier = true;
if(lua_gettop(L) > 3)
multiplier = popNumber(L);
uint32_t n = popNumber(L), skillid = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->addSkillAdvance((skills_t)skillid, n, multiplier);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureSpeakType(lua_State* L)
{
//getCreatureSpeakType(uid)
ScriptEnviroment* env = getEnv();
if(const Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushnumber(L, (SpeakClasses)creature->getSpeakType());
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCreatureSetSpeakType(lua_State* L)
{
//doCreatureSetSpeakType(uid, type)
SpeakClasses type = (SpeakClasses)popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
if(type < SPEAK_CLASS_FIRST || type > SPEAK_CLASS_LAST)
{
errorEx("Invalid speak type!");
lua_pushboolean(L, false);
return 1;
}
creature->setSpeakType(type);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureHideHealth(lua_State* L)
{
//getCreatureHideHealth(cid)
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushboolean(L, creature->getHideHealth());
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCreatureSetHideHealth(lua_State* L)
{
//doCreatureSetHideHealth(cid, hide)
bool hide = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
creature->setHideHealth(hide);
g_game.addCreatureHealth(creature);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCreatureAddHealth(lua_State* L)
{
//doCreatureAddHealth(uid, health[, hitEffect[, hitColor[, force]]])
int32_t params = lua_gettop(L);
bool force = false;
if(params > 4)
force = popNumber(L);
Color_t hitColor = COLOR_UNKNOWN;
if(params > 3)
hitColor = (Color_t)popNumber(L);
MagicEffect_t hitEffect = MAGIC_EFFECT_UNKNOWN;
if(params > 2)
hitEffect = (MagicEffect_t)popNumber(L);
int32_t healthChange = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
if(healthChange) //do not post with 0 value
g_game.combatChangeHealth(healthChange < 1 ? COMBAT_UNDEFINEDDAMAGE : COMBAT_HEALING,
NULL, creature, healthChange, hitEffect, hitColor, force);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCreatureAddMana(lua_State* L)
{
//doCreatureAddMana(uid, mana[, aggressive])
bool aggressive = true;
if(lua_gettop(L) > 2)
aggressive = popNumber(L);
int32_t manaChange = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
if(aggressive)
g_game.combatChangeMana(NULL, creature, manaChange);
else
creature->changeMana(manaChange);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerAddSpentMana(lua_State* L)
{
//doPlayerAddSpentMana(cid, amount[, useMultiplier = true])
bool multiplier = true;
if(lua_gettop(L) > 2)
multiplier = popNumber(L);
uint32_t amount = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->addManaSpent(amount, multiplier);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerAddItem(lua_State* L)
{
//doPlayerAddItem(cid, itemid[, count/subtype = 1[, canDropOnMap = true[, slot = 0]]])
//doPlayerAddItem(cid, itemid[, count = 1[, canDropOnMap = true[, subtype = 1[, slot = 0]]]])
int32_t params = lua_gettop(L), subType = 1, slot = SLOT_WHEREEVER;
if(params > 5)
slot = popNumber(L);
if(params > 4)
{
if(params > 5)
subType = popNumber(L);
else
slot = popNumber(L);
}
bool canDropOnMap = true;
if(params > 3)
canDropOnMap = popNumber(L);
uint32_t count = 1;
if(params > 2)
count = popNumber(L);
uint32_t itemId = popNumber(L);
if(slot > SLOT_AMMO)
{
errorEx("Invalid slot.");
lua_pushboolean(L, false);
return 1;
}
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID((uint32_t)popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
const ItemType& it = Item::items[itemId];
int32_t itemCount = 1;
if(params > 4)
itemCount = std::max((uint32_t)1, count);
else if(it.hasSubType())
{
if(it.stackable)
itemCount = (int32_t)std::ceil((float)count / 100);
subType = count;
}
while(itemCount > 0)
{
int32_t stackCount = std::min(100, subType);
Item* newItem = Item::CreateItem(itemId, stackCount);
if(!newItem)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
if(it.stackable)
subType -= stackCount;
ReturnValue ret = g_game.internalPlayerAddItem(NULL, player, newItem, canDropOnMap, (slots_t)slot);
if(ret != RET_NOERROR)
{
delete newItem;
lua_pushboolean(L, false);
return 1;
}
--itemCount;
if(itemCount)
continue;
if(newItem->getParent())
lua_pushnumber(L, env->addThing(newItem));
else //stackable item stacked with existing object, newItem will be released
lua_pushnil(L);
return 1;
}
lua_pushnil(L);
return 1;
}
int32_t LuaInterface::luaDoPlayerAddItemEx(lua_State* L)
{
//doPlayerAddItemEx(cid, uid[, canDropOnMap = false[, slot = 0]])
int32_t params = lua_gettop(L), slot = SLOT_WHEREEVER;
if(params > 3)
slot = popNumber(L);
bool canDropOnMap = false;
if(params > 2)
canDropOnMap = popNumber(L);
uint32_t uid = (uint32_t)popNumber(L);
if(slot > SLOT_AMMO)
{
errorEx("Invalid slot.");
lua_pushboolean(L, false);
return 1;
}
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Item* item = env->getItemByUID(uid);
if(!item)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
if(item->getParent() == VirtualCylinder::virtualCylinder)
lua_pushnumber(L, g_game.internalPlayerAddItem(NULL, player, item, canDropOnMap, (slots_t)slot));
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaDoTileAddItemEx(lua_State* L)
{
//doTileAddItemEx(pos, uid)
uint32_t uid = (uint32_t)popNumber(L);
PositionEx pos;
popPosition(L, pos);
ScriptEnviroment* env = getEnv();
Tile* tile = g_game.getTile(pos.x, pos.y, pos.z);
if(!tile)
{
errorEx(getError(LUA_ERROR_TILE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Item* item = env->getItemByUID(uid);
if(!item)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
if(item->getParent() == VirtualCylinder::virtualCylinder)
lua_pushnumber(L, g_game.internalAddItem(NULL, tile, item));
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaDoRelocate(lua_State* L)
{
//doRelocate(pos, posTo[, creatures = true[, unmovable = true]])
//Moves all moveable objects from pos to posTo
bool unmovable = true, creatures = true;
int32_t params = lua_gettop(L);
if(params > 3)
unmovable = popNumber(L);
if(params > 2)
creatures = popNumber(L);
PositionEx toPos;
popPosition(L, toPos);
PositionEx fromPos;
popPosition(L, fromPos);
Tile* fromTile = g_game.getTile(fromPos.x, fromPos.y, fromPos.z);
if(!fromTile)
{
errorEx(getError(LUA_ERROR_TILE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Tile* toTile = g_game.getTile(toPos.x, toPos.y, toPos.z);
if(!toTile)
{
errorEx(getError(LUA_ERROR_TILE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
if(fromTile != toTile)
{
for(int32_t i = fromTile->getThingCount() - 1; i >= 0; --i)
{
if(Thing* thing = fromTile->__getThing(i))
{
if(Item* item = thing->getItem())
{
const ItemType& it = Item::items[item->getID()];
if(!it.isGroundTile() && !it.alwaysOnTop && !it.isMagicField())
g_game.internalTeleport(item, toPos, true, unmovable ? FLAG_IGNORENOTMOVEABLE : 0);
}
else if(creatures)
{
if(Creature* creature = thing->getCreature())
g_game.internalTeleport(creature, toPos, false);
}
}
}
}
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoCleanTile(lua_State* L)
{
//doCleanTile(pos, forceMapLoaded = false)
//Remove all items from tile, ignore creatures
bool forceMapLoaded = false;
if(lua_gettop(L) > 1)
forceMapLoaded = popNumber(L);
PositionEx pos;
popPosition(L, pos);
Tile* tile = g_game.getTile(pos);
if(!tile)
{
errorEx(getError(LUA_ERROR_TILE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
for(int32_t i = tile->getThingCount() - 1; i >= 1; --i) //ignore ground
{
if(Thing* thing = tile->__getThing(i))
{
if(Item* item = thing->getItem())
{
if(!item->isLoadedFromMap() || forceMapLoaded)
g_game.internalRemoveItem(NULL, item);
}
}
}
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoPlayerSendTextMessage(lua_State* L)
{
//doPlayerSendTextMessage(cid, MessageClasses, message)
std::string text = popString(L);
uint32_t messageClass = popNumber(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
player->sendTextMessage((MessageClasses)messageClass, text);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoPlayerSendChannelMessage(lua_State* L)
{
//doPlayerSendChannelMessage(cid, author, message, SpeakClasses, channel)
uint16_t channelId = popNumber(L);
uint32_t speakClass = popNumber(L);
std::string text = popString(L), name = popString(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
player->sendChannelMessage(name, text, (SpeakClasses)speakClass, channelId);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoPlayerSendToChannel(lua_State* L)
{
//doPlayerSendToChannel(cid, targetId, SpeakClasses, message, channel[, time])
ScriptEnviroment* env = getEnv();
uint32_t time = 0;
if(lua_gettop(L) > 5)
time = popNumber(L);
uint16_t channelId = popNumber(L);
std::string text = popString(L);
uint32_t speakClass = popNumber(L), targetId = popNumber(L);
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Creature* creature = env->getCreatureByUID(targetId);
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
player->sendToChannel(creature, (SpeakClasses)speakClass, text, channelId, time);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoPlayerOpenChannel(lua_State* L)
{
//doPlayerOpenChannel(cid, channelId)
uint16_t channelId = popNumber(L);
uint32_t cid = popNumber(L);

ScriptEnviroment* env = getEnv();
if(env->getPlayerByUID(cid))
{
lua_pushboolean(L, g_game.playerOpenChannel(cid, channelId));
return 1;
}

errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaDoSendCreatureSquare(lua_State* L)
{
//doSendCreatureSquare(cid, color[, player])
ScriptEnviroment* env = getEnv();
SpectatorVec list;
if(lua_gettop(L) > 2)
{
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
list.push_back(creature);
}
uint8_t color = popNumber(L);
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
if(!list.empty())
g_game.addCreatureSquare(list, creature, color);
else
g_game.addCreatureSquare(creature, color);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoSendAnimatedText(lua_State* L)
{
//doSendAnimatedText(pos, text, color[, player])
ScriptEnviroment* env = getEnv();
SpectatorVec list;
if(lua_gettop(L) > 3)
{
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
list.push_back(creature);
}
uint8_t color = popNumber(L);
std::string text = popString(L);
PositionEx pos;
popPosition(L, pos);
if(pos.x == 0xFFFF)
pos = env->getRealPos();
if(!list.empty())
g_game.addAnimatedText(list, pos, color, text);
else
g_game.addAnimatedText(pos, color, text);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaGetPlayerSkillLevel(lua_State* L)
{
//getPlayerSkillLevel(cid, skillid)
uint32_t skillId = popNumber(L);
ScriptEnviroment* env = getEnv();
if(const Player* player = env->getPlayerByUID(popNumber(L)))
{
if(skillId <= SKILL_LAST)
lua_pushnumber(L, player->skills[skillId][SKILL_LEVEL]);
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerSkillTries(lua_State* L)
{
//getPlayerSkillTries(cid, skillid)
uint32_t skillid = popNumber(L);
ScriptEnviroment* env = getEnv();
if(const Player* player = env->getPlayerByUID(popNumber(L)))
{
if(skillid <= SKILL_LAST)
lua_pushnumber(L, player->skills[skillid][SKILL_TRIES]);
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCreatureSetDropLoot(lua_State* L)
{
//doCreatureSetDropLoot(cid, doDrop)
bool doDrop = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
creature->setDropLoot(doDrop ? LOOT_DROP_FULL : LOOT_DROP_NONE);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerLossPercent(lua_State* L)
{
//getPlayerLossPercent(cid, lossType)
uint8_t lossType = (uint8_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(const Player* player = env->getPlayerByUID(popNumber(L)))
{
if(lossType <= LOSS_LAST)
{
uint32_t value = player->getLossPercent((lossTypes_t)lossType);
lua_pushnumber(L, value);
}
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetLossPercent(lua_State* L)
{
//doPlayerSetLossPercent(cid, lossType, newPercent)
uint32_t newPercent = popNumber(L);
uint8_t lossType = (uint8_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(lossType <= LOSS_LAST)
{
player->setLossPercent((lossTypes_t)lossType, newPercent);
lua_pushboolean(L, true);
}
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetLossSkill(lua_State* L)
{
//doPlayerSetLossSkill(cid, doLose)
bool doLose = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->setLossSkill(doLose);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoShowTextDialog(lua_State* L)
{
//doShowTextDialog(cid, itemid, text)
std::string text = popString(L);
uint32_t itemId = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->setWriteItem(NULL, 0);
player->sendTextWindow(itemId, text);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoDecayItem(lua_State* L)
{
//doDecayItem(uid)
//Note: to stop decay set decayTo = 0 in items.xml
ScriptEnviroment* env = getEnv();
if(Item* item = env->getItemByUID(popNumber(L)))
{
g_game.startDecay(item);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetThingFromPos(lua_State* L)
{
//getThingFromPos(pos[, displayError = true])
//Note:
//    stackpos = 255- top thing (movable item or creature)
//    stackpos = 254- magic field
//    stackpos = 253- top creature
bool displayError = true;
if(lua_gettop(L) > 1)
displayError = popNumber(L);
PositionEx pos;
popPosition(L, pos);
ScriptEnviroment* env = getEnv();
Thing* thing = NULL;
if(Tile* tile = g_game.getMap()->getTile(pos))
{
if(pos.stackpos == 255)
{
if(!(thing = tile->getTopCreature()))
{
Item* item = tile->getTopDownItem();
if(item && item->isMoveable())
thing = item;
}
}
else if(pos.stackpos == 254)
thing = tile->getFieldItem();
else if(pos.stackpos == 253)
thing = tile->getTopCreature();
else
thing = tile->__getThing(pos.stackpos);
if(thing)
pushThing(L, thing, env->addThing(thing));
else
pushThing(L, NULL, 0);
return 1;
}
if(displayError)
errorEx(getError(LUA_ERROR_TILE_NOT_FOUND));
pushThing(L, NULL, 0);
return 1;
}
int32_t LuaInterface::luaGetTileItemById(lua_State* L)
{
//getTileItemById(pos, itemId[, subType = -1])
ScriptEnviroment* env = getEnv();
int32_t subType = -1;
if(lua_gettop(L) > 2)
subType = (int32_t)popNumber(L);
int32_t itemId = (int32_t)popNumber(L);
PositionEx pos;
popPosition(L, pos);
Tile* tile = g_game.getTile(pos);
if(!tile)
{
pushThing(L, NULL, 0);
return 1;
}
Item* item = g_game.findItemOfType(tile, itemId, false, subType);
if(!item)
{
pushThing(L, NULL, 0);
return 1;
}
pushThing(L, item, env->addThing(item));
return 1;
}
int32_t LuaInterface::luaGetTileItemByType(lua_State* L)
{
//getTileItemByType(pos, type)
uint32_t rType = (uint32_t)popNumber(L);
if(rType >= ITEM_TYPE_LAST)
{
errorEx("Not a valid item type");
pushThing(L, NULL, 0);
return 1;
}
PositionEx pos;
popPosition(L, pos);
Tile* tile = g_game.getTile(pos);
if(!tile)
{
pushThing(L, NULL, 0);
return 1;
}
bool found = true;
switch((ItemTypes_t)rType)
{
case ITEM_TYPE_TELEPORT:
{
if(!tile->hasFlag(TILESTATE_TELEPORT))
found = false;
break;
}
case ITEM_TYPE_MAGICFIELD:
{
if(!tile->hasFlag(TILESTATE_MAGICFIELD))
found = false;
break;
}
case ITEM_TYPE_MAILBOX:
{
if(!tile->hasFlag(TILESTATE_MAILBOX))
found = false;
break;
}
case ITEM_TYPE_TRASHHOLDER:
{
if(!tile->hasFlag(TILESTATE_TRASHHOLDER))
found = false;
break;
}
case ITEM_TYPE_BED:
{
if(!tile->hasFlag(TILESTATE_BED))
found = false;
break;
}
case ITEM_TYPE_DEPOT:
{
if(!tile->hasFlag(TILESTATE_DEPOT))
found = false;
break;
}
default:
break;
}
if(!found)
{
pushThing(L, NULL, 0);
return 1;
}
ScriptEnviroment* env = getEnv();
Item* item = NULL;
for(uint32_t i = 0; i < tile->getThingCount(); ++i)
{
if(!(item = tile->__getThing(i)->getItem()))
continue;
if(Item::items[item->getID()].type != (ItemTypes_t)rType)
continue;
pushThing(L, item, env->addThing(item));
return 1;
}
pushThing(L, NULL, 0);
return 1;
}
int32_t LuaInterface::luaGetTileThingByPos(lua_State* L)
{
//getTileThingByPos(pos)
PositionEx pos;
popPosition(L, pos);
ScriptEnviroment* env = getEnv();
Tile* tile = g_game.getTile(pos.x, pos.y, pos.z);
if(!tile)
{
if(pos.stackpos == -1)
{
lua_pushnumber(L, -1);
return 1;
}
else
{
pushThing(L, NULL, 0);
return 1;
}
}
if(pos.stackpos == -1)
{
lua_pushnumber(L, tile->getThingCount());
return 1;
}
Thing* thing = tile->__getThing(pos.stackpos);
if(!thing)
{
pushThing(L, NULL, 0);
return 1;
}
pushThing(L, thing, env->addThing(thing));
return 1;
}
int32_t LuaInterface::luaGetTopCreature(lua_State* L)
{
//getTopCreature(pos)
PositionEx pos;
popPosition(L, pos);
ScriptEnviroment* env = getEnv();
Tile* tile = g_game.getTile(pos);
if(!tile)
{
pushThing(L, NULL, 0);
return 1;
}
Thing* thing = tile->getTopCreature();
if(!thing || !thing->getCreature())
{
pushThing(L, NULL, 0);
return 1;
}
pushThing(L, thing, env->addThing(thing));
return 1;
}
int32_t LuaInterface::luaDoCreateItem(lua_State* L)
{
//doCreateItem(itemid[, type/count = 1], pos)
//Returns uid of the created item, only works on tiles.
PositionEx pos;
popPosition(L, pos);
uint32_t count = 1;
if(lua_gettop(L) > 1)
count = popNumber(L);
uint32_t itemId = popNumber(L);
ScriptEnviroment* env = getEnv();
const ItemType& it = Item::items[itemId];
Tile* tile = g_game.getTile(pos);
if(!tile)
{
if(it.group == ITEM_GROUP_GROUND)
{
Item* item = Item::CreateItem(itemId);
tile = IOMap::createTile(item, NULL, pos.x, pos.y, pos.z);
g_game.setTile(tile);
lua_pushnumber(L, env->addThing(item));
return 1;
}
else
{
errorEx(getError(LUA_ERROR_TILE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
}
int32_t itemCount = 1, subType = 1;
if(it.hasSubType())
{
if(it.stackable)
itemCount = (int32_t)std::ceil((float)count / 100);
subType = count;
}
else
itemCount = std::max((uint32_t)1, count);
while(itemCount > 0)
{
int32_t stackCount = std::min(100, subType);
Item* newItem = Item::CreateItem(itemId, stackCount);
if(!newItem)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
if(it.stackable)
subType -= stackCount;
ReturnValue ret = g_game.internalAddItem(NULL, tile, newItem, INDEX_WHEREEVER, FLAG_NOLIMIT);
if(ret != RET_NOERROR)
{
delete newItem;
lua_pushboolean(L, false);
return 1;
}
--itemCount;
if(itemCount)
continue;
if(newItem->getParent())
lua_pushnumber(L, env->addThing(newItem));
else //stackable item stacked with existing object, newItem will be released
lua_pushnil(L);
return 1;
}
lua_pushnil(L);
return 1;
}
int32_t LuaInterface::luaDoCreateItemEx(lua_State* L)
{
//doCreateItemEx(itemid[, count/subType])
uint32_t count = 0;
if(lua_gettop(L) > 1)
count = popNumber(L);
ScriptEnviroment* env = getEnv();
const ItemType& it = Item::items[(uint32_t)popNumber(L)];
if(it.stackable && count > 100)
count = 100;
Item* newItem = Item::CreateItem(it.id, count);
if(!newItem)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
newItem->setParent(VirtualCylinder::virtualCylinder);
env->addTempItem(env, newItem);
lua_pushnumber(L, env->addThing(newItem));
return 1;
}
int32_t LuaInterface::luaDoCreateTeleport(lua_State* L)
{
//doCreateTeleport(itemid, toPosition, fromPosition)
PositionEx createPos;
popPosition(L, createPos);
PositionEx toPos;
popPosition(L, toPos);
uint32_t itemId = (uint32_t)popNumber(L);
ScriptEnviroment* env = getEnv();
Tile* tile = g_game.getMap()->getTile(createPos);
if(!tile)
{
errorEx(getError(LUA_ERROR_TILE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Item* newItem = Item::CreateItem(itemId);
Teleport* newTeleport = newItem->getTeleport();
if(!newTeleport)
{
delete newItem;
lua_pushboolean(L, false);
return 1;
}
newTeleport->setDestination(toPos);
if(g_game.internalAddItem(NULL, tile, newTeleport, INDEX_WHEREEVER, FLAG_NOLIMIT) != RET_NOERROR)
{
delete newItem;
lua_pushboolean(L, false);
return 1;
}
if(newItem->getParent())
lua_pushnumber(L, env->addThing(newItem));
else //stackable item stacked with existing object, newItem will be released
lua_pushnil(L);
return 1;
}
int32_t LuaInterface::luaGetCreatureStorage(lua_State* L)
{
//getCreatureStorage(cid, key)
uint32_t key = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
std::string strValue;
if(creature->getStorage(key, strValue))
{
int32_t intValue = atoi(strValue.c_str());
if(intValue || strValue == "0")
lua_pushnumber(L, intValue);
else
lua_pushstring(L, strValue.c_str());
}
else
lua_pushnumber(L, -1);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCreatureSetStorage(lua_State* L)
{
//doCreatureSetStorage(cid, key[, value])
std::string value;
bool nil = true;
if(lua_gettop(L) > 2)
{
if(!lua_isnil(L, -1))
{
value = popString(L);
nil = false;
}
else
lua_pop(L, 1);
}
uint32_t key = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
if(!nil)
nil = creature->setStorage(key, value);
else
creature->eraseStorage(key);
lua_pushboolean(L, nil);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetTileInfo(lua_State* L)
{
//getTileInfo(pos)
PositionEx pos;
popPosition(L, pos);
if(Tile* tile = g_game.getMap()->getTile(pos))
{
ScriptEnviroment* env = getEnv();
pushThing(L, tile->ground, env->addThing(tile->ground));
setFieldBool(L, "protection", tile->hasFlag(TILESTATE_PROTECTIONZONE));
setFieldBool(L, "optional", tile->hasFlag(TILESTATE_OPTIONALZONE));
setFieldBool(L, "nologout", tile->hasFlag(TILESTATE_NOLOGOUT));
setFieldBool(L, "hardcore", tile->hasFlag(TILESTATE_HARDCOREZONE));
setFieldBool(L, "refresh", tile->hasFlag(TILESTATE_REFRESH));
setFieldBool(L, "trashed", tile->hasFlag(TILESTATE_TRASHED));
setFieldBool(L, "house", tile->hasFlag(TILESTATE_HOUSE));
setFieldBool(L, "bed", tile->hasFlag(TILESTATE_BED));
setFieldBool(L, "depot", tile->hasFlag(TILESTATE_DEPOT));
setField(L, "things", tile->getThingCount());
setField(L, "creatures", tile->getCreatureCount());
setField(L, "items", tile->getItemCount());
setField(L, "topItems", tile->getTopItemCount());
setField(L, "downItems", tile->getDownItemCount());
}
else
{
errorEx(getError(LUA_ERROR_TILE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetHouseFromPos(lua_State* L)
{
//getHouseFromPos(pos)
PositionEx pos;
popPosition(L, pos);
Tile* tile = g_game.getMap()->getTile(pos);
if(!tile)
{
errorEx(getError(LUA_ERROR_TILE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
HouseTile* houseTile = tile->getHouseTile();
if(!houseTile)
{
lua_pushboolean(L, false);
return 1;
}
House* house = houseTile->getHouse();
if(!house)
{
lua_pushboolean(L, false);
return 1;
}
lua_pushnumber(L, house->getId());
return 1;
}
int32_t LuaInterface::luaDoCreateMonster(lua_State* L)
{
//doCreateMonster(name, pos[, extend = false[, force = false[, displayError = true]]])
bool displayError = true, force = false, extend = false;
int32_t params = lua_gettop(L);
if(params > 4)
displayError = popNumber(L);
if(params > 3)
force = popNumber(L);
if(params > 2)
extend = popNumber(L);
PositionEx pos;
popPosition(L, pos);
std::string name = popString(L);
Monster* monster = Monster::createMonster(name.c_str());
if(!monster)
{
if(displayError)
errorEx("Monster with name '" + name + "' not found");
lua_pushboolean(L, false);
return 1;
}
if(!g_game.placeCreature(monster, pos, extend, force))
{
delete monster;
if(displayError)
errorEx("Cannot create monster: " + name);
lua_pushboolean(L, true);
return 1;
}
ScriptEnviroment* env = getEnv();
lua_pushnumber(L, env->addThing((Thing*)monster));
return 1;
}
int32_t LuaInterface::luaDoCreateNpc(lua_State* L)
{
//doCreateNpc(name, pos[, displayError = true])
bool displayError = true;
if(lua_gettop(L) > 2)
displayError = popNumber(L);
PositionEx pos;
popPosition(L, pos);
std::string name = popString(L);
Npc* npc = Npc::createNpc(name.c_str());
if(!npc)
{
if(displayError)
errorEx("Npc with name '" + name + "' not found");
lua_pushboolean(L, false);
return 1;
}
if(!g_game.placeCreature(npc, pos))
{
delete npc;
if(displayError)
errorEx("Cannot create npc: " + name);
lua_pushboolean(L, true); //for scripting compatibility
return 1;
}
ScriptEnviroment* env = getEnv();
lua_pushnumber(L, env->addThing((Thing*)npc));
return 1;
}
int32_t LuaInterface::luaDoRemoveCreature(lua_State* L)
{
//doRemoveCreature(cid[, forceLogout = true])
bool forceLogout = true;
if(lua_gettop(L) > 1)
forceLogout = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
if(Player* player = creature->getPlayer())
player->kickPlayer(true, forceLogout); //Players will get kicked without restrictions
else
g_game.removeCreature(creature); //Monsters/NPCs will get removed
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerAddMoney(lua_State* L)
{
//doPlayerAddMoney(cid, money)
uint64_t money = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
g_game.addMoney(player, money);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerRemoveMoney(lua_State* L)
{
//doPlayerRemoveMoney(cid,money)
uint64_t money = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
lua_pushboolean(L, g_game.removeMoney(player, money));
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerTransferMoneyTo(lua_State* L)
{
//doPlayerTransferMoneyTo(cid, target, money)
uint64_t money = popNumber(L);
std::string target = popString(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
lua_pushboolean(L, player->transferMoneyTo(target, money));
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetPzLocked(lua_State* L)
{
//doPlayerSetPzLocked(cid, locked)
bool locked = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(player->isPzLocked() != locked)
{
player->setPzLocked(locked);
player->sendIcons();
}
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetTown(lua_State* L)
{
//doPlayerSetTown(cid, townid)
uint32_t townid = (uint32_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(Town* town = Towns::getInstance()->getTown(townid))
{
player->setMasterPosition(town->getPosition());
player->setTown(townid);
lua_pushboolean(L, true);
}
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetVocation(lua_State* L)
{
//doPlayerSetVocation(cid, voc)
uint32_t voc = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->setVocation(voc);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetSex(lua_State* L)
{
//doPlayerSetSex(cid, sex)
uint32_t newSex = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->setSex(newSex);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerAddSoul(lua_State* L)
{
//doPlayerAddSoul(cid, soul)
int32_t soul = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->changeSoul(soul);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerItemCount(lua_State* L)
{
//getPlayerItemCount(cid, itemid[, subType = -1])
int32_t subType = -1;
if(lua_gettop(L) > 2)
subType = popNumber(L);
uint32_t itemId = popNumber(L);
ScriptEnviroment* env = getEnv();
if(const Player* player = env->getPlayerByUID(popNumber(L)))
lua_pushnumber(L, player->__getItemTypeCount(itemId, subType));
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerMoney(lua_State* L)
{
//getPlayerMoney(cid)
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
lua_pushnumber(L, g_game.getMoney(player));
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetHouseInfo(lua_State* L)
{
//getHouseInfo(houseId[, displayError = true])
bool displayError = true;
if(lua_gettop(L) > 1)
displayError = popNumber(L);
House* house = Houses::getInstance()->getHouse(popNumber(L));
if(!house)
{
if(displayError)
errorEx(getError(LUA_ERROR_HOUSE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
lua_newtable(L);
setField(L, "id", house->getId());
setField(L, "name", house->getName().c_str());
setField(L, "owner", house->getOwner());
lua_pushstring(L, "entry");
pushPosition(L, house->getEntry(), 0);
pushTable(L);
setField(L, "rent", house->getRent());
setField(L, "price", house->getPrice());
setField(L, "town", house->getTownId());
setField(L, "paidUntil", house->getPaidUntil());
setField(L, "warnings", house->getRentWarnings());
setField(L, "lastWarning", house->getLastWarning());
setFieldBool(L, "guildHall", house->isGuild());
setField(L, "size", house->getSize());
createTable(L, "doors");
HouseDoorList::iterator dit = house->getHouseDoorBegin();
for(uint32_t i = 1; dit != house->getHouseDoorEnd(); ++dit, ++i)
{
lua_pushnumber(L, i);
pushPosition(L, (*dit)->getPosition(), 0);
pushTable(L);
}
pushTable(L);
createTable(L, "beds");
HouseBedList::iterator bit = house->getHouseBedsBegin();
for(uint32_t i = 1; bit != house->getHouseBedsEnd(); ++bit, ++i)
{
lua_pushnumber(L, i);
pushPosition(L, (*bit)->getPosition(), 0);
pushTable(L);
}
pushTable(L);
createTable(L, "tiles");
HouseTileList::iterator tit = house->getHouseTileBegin();
for(uint32_t i = 1; tit != house->getHouseTileEnd(); ++tit, ++i)
{
lua_pushnumber(L, i);
pushPosition(L, (*tit)->getPosition(), 0);
pushTable(L);
}
pushTable(L);
return 1;
}
int32_t LuaInterface::luaGetHouseAccessList(lua_State* L)
{
//getHouseAccessList(houseid, listid)
uint32_t listid = popNumber(L);
if(House* house = Houses::getInstance()->getHouse(popNumber(L)))
{
std::string list;
if(house->getAccessList(listid, list))
lua_pushstring(L, list.c_str());
else
lua_pushnil(L);
}
else
{
errorEx(getError(LUA_ERROR_HOUSE_NOT_FOUND));
lua_pushnil(L);
}
return 1;
}
int32_t LuaInterface::luaGetHouseByPlayerGUID(lua_State* L)
{
//getHouseByPlayerGUID(guid)
if(House* house = Houses::getInstance()->getHouseByPlayerId(popNumber(L)))
lua_pushnumber(L, house->getId());
else
lua_pushnil(L);
return 1;
}
int32_t LuaInterface::luaSetHouseAccessList(lua_State* L)
{
//setHouseAccessList(houseid, listid, listtext)
std::string list = popString(L);
uint32_t listid = popNumber(L);
if(House* house = Houses::getInstance()->getHouse(popNumber(L)))
{
house->setAccessList(listid, list);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_HOUSE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaSetHouseOwner(lua_State* L)
{
//setHouseOwner(houseId, owner[, clean])
bool clean = true;
if(lua_gettop(L) > 2)
clean = popNumber(L);
uint32_t owner = popNumber(L);
if(House* house = Houses::getInstance()->getHouse(popNumber(L)))
lua_pushboolean(L, house->setOwnerEx(owner, clean));
else
{
errorEx(getError(LUA_ERROR_HOUSE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetWorldType(lua_State* L)
{
lua_pushnumber(L, (uint32_t)g_game.getWorldType());
return 1;
}
int32_t LuaInterface::luaSetWorldType(lua_State* L)
{
//setWorldType(type)
WorldType_t type = (WorldType_t)popNumber(L);
if(type >= WORLDTYPE_FIRST && type <= WORLDTYPE_LAST)
{
g_game.setWorldType(type);
lua_pushboolean(L, true);
}
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaGetWorldTime(lua_State* L)
{
//getWorldTime()
lua_pushnumber(L, g_game.getLightHour());
return 1;
}
int32_t LuaInterface::luaGetWorldLight(lua_State* L)
{
//getWorldLight()
LightInfo lightInfo;
g_game.getWorldLightInfo(lightInfo);
lua_pushnumber(L, lightInfo.level);
lua_pushnumber(L, lightInfo.color);
return 2;
}
int32_t LuaInterface::luaGetWorldCreatures(lua_State* L)
{
//getWorldCreatures(type)
//0 players, 1 monsters, 2 npcs, 3 all
uint32_t type = popNumber(L), value;
switch(type)
{
case 0:
value = g_game.getPlayersOnline();
break;
case 1:
value = g_game.getMonstersOnline();
break;
case 2:
value = g_game.getNpcsOnline();
break;
case 3:
value = g_game.getCreaturesOnline();
break;
default:
lua_pushboolean(L, false);
return 1;
}
lua_pushnumber(L, value);
return 1;
}
int32_t LuaInterface::luaGetWorldUpTime(lua_State* L)
{
//getWorldUpTime()
uint32_t uptime = 0;
if(Status* status = Status::getInstance())
uptime = status->getUptime();
lua_pushnumber(L, uptime);
return 1;
}
int32_t LuaInterface::luaGetPlayerLight(lua_State* L)
{
//getPlayerLight(cid)
ScriptEnviroment* env = getEnv();
if(const Player* player = env->getPlayerByUID(popNumber(L)))
{
LightInfo lightInfo;
player->getCreatureLight(lightInfo);
lua_pushnumber(L, lightInfo.level);
lua_pushnumber(L, lightInfo.color);
return 2;
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
}
int32_t LuaInterface::luaDoPlayerAddExperience(lua_State* L)
{
//doPlayerAddExperience(cid, amount)
int64_t amount = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(amount > 0)
player->addExperience(amount);
else if(amount < 0)
player->removeExperience(std::abs(amount));
else
{
lua_pushboolean(L, false);
return 1;
}
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerSlotItem(lua_State* L)
{
//getPlayerSlotItem(cid, slot)
uint32_t slot = popNumber(L);
ScriptEnviroment* env = getEnv();
if(const Player* player = env->getPlayerByUID(popNumber(L)))
{
if(Thing* thing = player->__getThing(slot))
pushThing(L, thing, env->addThing(thing));
else
pushThing(L, NULL, 0);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
pushThing(L, NULL, 0);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerWeapon(lua_State* L)
{
//getPlayerWeapon(cid[, ignoreAmmo = false])
bool ignoreAmmo = false;
if(lua_gettop(L) > 1)
ignoreAmmo = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(Item* weapon = player->getWeapon(ignoreAmmo))
pushThing(L, weapon, env->addThing(weapon));
else
pushThing(L, NULL, 0);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushnil(L);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerItemById(lua_State* L)
{
//getPlayerItemById(cid, deepSearch, itemId[, subType = -1])
ScriptEnviroment* env = getEnv();
int32_t subType = -1;
if(lua_gettop(L) > 3)
subType = (int32_t)popNumber(L);
int32_t itemId = (int32_t)popNumber(L);
bool deepSearch = popNumber(L);
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
pushThing(L, NULL, 0);
return 1;
}
Item* item = g_game.findItemOfType(player, itemId, deepSearch, subType);
if(!item)
{
pushThing(L, NULL, 0);
return 1;
}
pushThing(L, item, env->addThing(item));
return 1;
}
int32_t LuaInterface::luaGetThing(lua_State* L)
{
//getThing(uid)
uint32_t uid = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Thing* thing = env->getThingByUID(uid))
pushThing(L, thing, uid);
else
{
errorEx(getError(LUA_ERROR_THING_NOT_FOUND));
pushThing(L, NULL, 0);
}
return 1;
}
int32_t LuaInterface::luaDoTileQueryAdd(lua_State* L)
{
//doTileQueryAdd(uid, pos[, flags[, displayError = true]])
uint32_t flags = 0, params = lua_gettop(L);
bool displayError = true;
if(params > 3)
displayError = popNumber(L);
if(params > 2)
flags = popNumber(L);
PositionEx pos;
popPosition(L, pos);
uint32_t uid = popNumber(L);
ScriptEnviroment* env = getEnv();
Tile* tile = g_game.getTile(pos);
if(!tile)
{
if(displayError)
errorEx(getError(LUA_ERROR_TILE_NOT_FOUND));
lua_pushnumber(L, (uint32_t)RET_NOTPOSSIBLE);
return 1;
}
Thing* thing = env->getThingByUID(uid);
if(!thing)
{
if(displayError)
errorEx(getError(LUA_ERROR_THING_NOT_FOUND));
lua_pushnumber(L, (uint32_t)RET_NOTPOSSIBLE);
return 1;
}
lua_pushnumber(L, (uint32_t)tile->__queryAdd(0, thing, 1, flags));
return 1;
}
int32_t LuaInterface::luaDoItemRaidUnref(lua_State* L)
{
//doItemRaidUnref(uid)
ScriptEnviroment* env = getEnv();
if(Item* item = env->getItemByUID(popNumber(L)))
{
if(Raid* raid = item->getRaid())
{
raid->unRef();
item->setRaid(NULL);
lua_pushboolean(L, true);
}
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetThingPosition(lua_State* L)
{
//getThingPosition(uid)
ScriptEnviroment* env = getEnv();
if(Thing* thing = env->getThingByUID(popNumber(L)))
{
Position pos = thing->getPosition();
uint32_t stackpos = 0;
if(Tile* tile = thing->getTile())
stackpos = tile->__getIndexOfThing(thing);
pushPosition(L, pos, stackpos);
}
else
{
errorEx(getError(LUA_ERROR_THING_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaCreateCombatObject(lua_State* L)
{
//createCombatObject()
ScriptEnviroment* env = getEnv();
if(env->getScriptId() != EVENT_ID_LOADING)
{
errorEx("This function can only be used while loading the script.");
lua_pushboolean(L, false);
return 1;
}
Combat* combat = new Combat;
if(!combat)
{
errorEx(getError(LUA_ERROR_COMBAT_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
lua_pushnumber(L, env->addCombatObject(combat));
return 1;
}
bool LuaInterface::getArea(lua_State* L, std::list<uint32_t>& list, uint32_t& rows)
{
rows = 0;
uint32_t i = 0;
lua_pushnil(L);
while(lua_next(L, -2))
{
lua_pushnil(L);
while(lua_next(L, -2))
{
list.push_back((uint32_t)lua_tonumber(L, -1));
lua_pop(L, 1); //removes value, keeps key for next iteration
++i;
}
lua_pop(L, 1); //removes value, keeps key for next iteration
++rows;
i = 0;
}
lua_pop(L, 1);
return rows;
}
int32_t LuaInterface::luaCreateCombatArea(lua_State* L)
{
//createCombatArea( {area}[, {extArea}])
ScriptEnviroment* env = getEnv();
if(env->getScriptId() != EVENT_ID_LOADING)
{
errorEx("This function can only be used while loading the script.");
lua_pushboolean(L, false);
return 1;
}
CombatArea* area = new CombatArea;
if(lua_gettop(L) > 1)
{
//has extra parameter with diagonal area information
uint32_t rowsExtArea;
std::list<uint32_t> listExtArea;
getArea(L, listExtArea, rowsExtArea);
/*setup all possible rotations*/
area->setupExtArea(listExtArea, rowsExtArea);
}
if(lua_isnoneornil(L, -1)) //prevent crash
{
lua_pop(L, 2);
lua_pushboolean(L, false);
return 1;
}
uint32_t rowsArea = 0;
std::list<uint32_t> listArea;
getArea(L, listArea, rowsArea);
area->setupArea(listArea, rowsArea);
lua_pushnumber(L, env->addCombatArea(area));
return 1;
}
int32_t LuaInterface::luaCreateConditionObject(lua_State* L)
{
//createConditionObject(type[, ticks[, buff[, subId]]])
uint32_t params = lua_gettop(L), subId = 0;
if(params > 3)
subId = popNumber(L);
bool buff = false;
if(params > 2)
buff = popNumber(L);
int32_t ticks = 0;
if(params > 1)
ticks = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Condition* condition = Condition::createCondition(CONDITIONID_COMBAT, (ConditionType_t)popNumber(L), ticks, 0, buff, subId))
{
if(env->getScriptId() != EVENT_ID_LOADING)
lua_pushnumber(L, env->addTempConditionObject(condition));
else
lua_pushnumber(L, env->addConditionObject(condition));
}
else
{
errorEx(getError(LUA_ERROR_CONDITION_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaSetCombatArea(lua_State* L)
{
//setCombatArea(combat, area)
uint32_t areaId = popNumber(L);
ScriptEnviroment* env = getEnv();
if(env->getScriptId() != EVENT_ID_LOADING)
{
errorEx("This function can only be used while loading the script.");
lua_pushboolean(L, false);
return 1;
}
Combat* combat = env->getCombatObject(popNumber(L));
if(!combat)
{
errorEx(getError(LUA_ERROR_COMBAT_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
const CombatArea* area = env->getCombatArea(areaId);
if(!area)
{
errorEx(getError(LUA_ERROR_AREA_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
combat->setArea(new CombatArea(*area));
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaSetCombatCondition(lua_State* L)
{
//setCombatCondition(combat, condition)
uint32_t conditionId = popNumber(L);
ScriptEnviroment* env = getEnv();
Combat* combat = env->getCombatObject(popNumber(L));
if(!combat)
{
errorEx(getError(LUA_ERROR_COMBAT_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
const Condition* condition = env->getConditionObject(conditionId);
if(!condition)
{
errorEx(getError(LUA_ERROR_CONDITION_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
combat->setCondition(condition->clone());
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaSetCombatParam(lua_State* L)
{
//setCombatParam(combat, key, value)
uint32_t value = popNumber(L);
CombatParam_t key = (CombatParam_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(env->getScriptId() != EVENT_ID_LOADING)
{
errorEx("This function can only be used while loading the script.");
lua_pushboolean(L, false);
return 1;
}
Combat* combat = env->getCombatObject(popNumber(L));
if(!combat)
{
errorEx(getError(LUA_ERROR_COMBAT_NOT_FOUND));
lua_pushboolean(L, false);
}
else
{
combat->setParam(key, value);
lua_pushboolean(L, true);
}
return 1;
}
int32_t LuaInterface::luaSetConditionParam(lua_State* L)
{
//setConditionParam(condition, key, value)
int32_t value = popNumber(L);
ScriptEnviroment* env = getEnv();
ConditionParam_t key = (ConditionParam_t)popNumber(L);
if(Condition* condition = env->getConditionObject(popNumber(L)))
{
condition->setParam(key, value);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CONDITION_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaAddDamageCondition(lua_State* L)
{
//addDamageCondition(condition, rounds, time, value)
int32_t value = popNumber(L), time = popNumber(L), rounds = popNumber(L);
ScriptEnviroment* env = getEnv();
if(ConditionDamage* condition = dynamic_cast<ConditionDamage*>(env->getConditionObject(popNumber(L))))
{
condition->addDamage(rounds, time, value);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CONDITION_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaAddOutfitCondition(lua_State* L)
{
//addOutfitCondition(condition, outfit)
Outfit_t outfit = popOutfit(L);
ScriptEnviroment* env = getEnv();
if(ConditionOutfit* condition = dynamic_cast<ConditionOutfit*>(env->getConditionObject(popNumber(L))))
{
condition->addOutfit(outfit);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CONDITION_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaSetCombatCallBack(lua_State* L)
{
//setCombatCallBack(combat, key, functionName)
std::string function = popString(L);
CallBackParam_t key = (CallBackParam_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(env->getScriptId() != EVENT_ID_LOADING)
{
errorEx("This function can only be used while loading the script.");
lua_pushboolean(L, false);
return 1;
}
Combat* combat = env->getCombatObject(popNumber(L));
if(!combat)
{
errorEx(getError(LUA_ERROR_COMBAT_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
LuaInterface* interface = env->getInterface();
combat->setCallback(key);
CallBack* callback = combat->getCallback(key);
if(!callback)
{
std::stringstream ss;
ss << key;
errorEx(ss.str() + " is not a valid callback key.");
lua_pushboolean(L, false);
return 1;
}
if(!callback->loadCallBack(interface, function))
{
errorEx("Cannot load callback");
lua_pushboolean(L, false);
}
else
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaSetCombatFormula(lua_State* L)
{
//setCombatFormula(combat, type, mina, minb, maxa, maxb[, minl, maxl[, minm, maxm[, minc[, maxc]]]])
ScriptEnviroment* env = getEnv();
if(env->getScriptId() != EVENT_ID_LOADING)
{
errorEx("This function can only be used while loading the script.");
lua_pushboolean(L, false);
return 1;
}
int32_t params = lua_gettop(L), minc = 0, maxc = 0;
if(params > 11)
maxc = popNumber(L);
if(params > 10)
minc = popNumber(L);
double minm = g_config.getDouble(ConfigManager::FORMULA_MAGIC), maxm = minm,
minl = g_config.getDouble(ConfigManager::FORMULA_LEVEL), maxl = minl;
if(params > ?
{
maxm = popFloatNumber(L);
minm = popFloatNumber(L);
}
if(params > 6)
{
maxl = popFloatNumber(L);
minl = popFloatNumber(L);
}
double maxb = popFloatNumber(L), maxa = popFloatNumber(L),
minb = popFloatNumber(L), mina = popFloatNumber(L);
formulaType_t type = (formulaType_t)popNumber(L);
if(Combat* combat = env->getCombatObject(popNumber(L)))
{
combat->setPlayerCombatValues(type, mina, minb, maxa, maxb, minl, maxl, minm, maxm, minc, maxc);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_COMBAT_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaSetConditionFormula(lua_State* L)
{
//setConditionFormula(condition, mina, minb, maxa, maxb)
double maxb = popFloatNumber(L), maxa = popFloatNumber(L),
minb = popFloatNumber(L), mina = popFloatNumber(L);
ScriptEnviroment* env = getEnv();
if(ConditionSpeed* condition = dynamic_cast<ConditionSpeed*>(env->getConditionObject(popNumber(L))))
{
condition->setFormulaVars(mina, minb, maxa, maxb);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CONDITION_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCombat(lua_State* L)
{
//doCombat(cid, combat, param)
ScriptEnviroment* env = getEnv();
LuaVariant var = popVariant(L);
uint32_t combatId = popNumber(L), cid = popNumber(L);
Creature* creature = NULL;
if(cid != 0)
{
creature = env->getCreatureByUID(cid);
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
}
const Combat* combat = env->getCombatObject(combatId);
if(!combat)
{
errorEx(getError(LUA_ERROR_COMBAT_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
if(var.type == VARIANT_NONE)
{
errorEx(getError(LUA_ERROR_VARIANT_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
switch(var.type)
{
case VARIANT_NUMBER:
{
Creature* target = g_game.getCreatureByID(var.number);
if(!target || !creature || !creature->canSeeCreature(target))
{
lua_pushboolean(L, false);
return 1;
}
if(combat->hasArea())
combat->doCombat(creature, target->getPosition());
else
combat->doCombat(creature, target);
break;
}
case VARIANT_POSITION:
{
combat->doCombat(creature, var.pos);
break;
}
case VARIANT_TARGETPOSITION:
{
if(!combat->hasArea())
{
combat->postCombatEffects(creature, var.pos);
g_game.addMagicEffect(var.pos, MAGIC_EFFECT_POFF);
}
else
combat->doCombat(creature, var.pos);
break;
}
case VARIANT_STRING:
{
Player* target = g_game.getPlayerByName(var.text);
if(!target || !creature || !creature->canSeeCreature(target))
{
lua_pushboolean(L, false);
return 1;
}
combat->doCombat(creature, target);
break;
}
default:
{
errorEx(getError(LUA_ERROR_VARIANT_UNKNOWN));
lua_pushboolean(L, false);
return 1;
}
}
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoCombatAreaHealth(lua_State* L)
{
//doCombatAreaHealth(cid, type, pos, area, min, max, effect)
MagicEffect_t effect = (MagicEffect_t)popNumber(L);
int32_t maxChange = (int32_t)popNumber(L), minChange = (int32_t)popNumber(L);
uint32_t areaId = popNumber(L);
PositionEx pos;
popPosition(L, pos);
CombatType_t combatType = (CombatType_t)popNumber(L);
uint32_t cid = popNumber(L);
ScriptEnviroment* env = getEnv();
Creature* creature = NULL;
if(cid)
{
if(!(creature = env->getCreatureByUID(cid)))
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
}
const CombatArea* area = env->getCombatArea(areaId);
if(area || !areaId)
{
CombatParams params;
params.combatType = combatType;
params.effects.impact = effect;
Combat::doCombatHealth(creature, pos, area, minChange, maxChange, params);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_AREA_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoTargetCombatHealth(lua_State* L)
{
//doTargetCombatHealth(cid, target, type, min, max, effect)
MagicEffect_t effect = (MagicEffect_t)popNumber(L);
int32_t maxChange = (int32_t)popNumber(L), minChange = (int32_t)popNumber(L);
CombatType_t combatType = (CombatType_t)popNumber(L);
uint32_t targetCid = popNumber(L), cid = popNumber(L);
ScriptEnviroment* env = getEnv();
Creature* creature = NULL;
if(cid)
{
if(!(creature = env->getCreatureByUID(cid)))
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
}
Creature* target = env->getCreatureByUID(targetCid);
if(target)
{
CombatParams params;
params.combatType = combatType;
params.effects.impact = effect;
Combat::doCombatHealth(creature, target, minChange, maxChange, params);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCombatAreaMana(lua_State* L)
{
//doCombatAreaMana(cid, pos, area, min, max, effect)
MagicEffect_t effect = (MagicEffect_t)popNumber(L);
int32_t maxChange = (int32_t)popNumber(L), minChange = (int32_t)popNumber(L);
uint32_t areaId = popNumber(L);
PositionEx pos;
popPosition(L, pos);
uint32_t cid = popNumber(L);
ScriptEnviroment* env = getEnv();
Creature* creature = NULL;
if(cid)
{
if(!(creature = env->getCreatureByUID(cid)))
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
}
const CombatArea* area = env->getCombatArea(areaId);
if(area || !areaId)
{
CombatParams params;
params.effects.impact = effect;
Combat::doCombatMana(creature, pos, area, minChange, maxChange, params);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_AREA_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoTargetCombatMana(lua_State* L)
{
//doTargetCombatMana(cid, target, min, max, effect)
MagicEffect_t effect = (MagicEffect_t)popNumber(L);
int32_t maxChange = (int32_t)popNumber(L), minChange = (int32_t)popNumber(L);
uint32_t targetCid = popNumber(L), cid = popNumber(L);
ScriptEnviroment* env = getEnv();
Creature* creature = NULL;
if(cid)
{
if(!(creature = env->getCreatureByUID(cid)))
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
}
if(Creature* target = env->getCreatureByUID(targetCid))
{
CombatParams params;
params.effects.impact = effect;
Combat::doCombatMana(creature, target, minChange, maxChange, params);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCombatAreaCondition(lua_State* L)
{
//doCombatAreaCondition(cid, pos, area, condition, effect)
MagicEffect_t effect = (MagicEffect_t)popNumber(L);
uint32_t conditionId = popNumber(L), areaId = popNumber(L);
PositionEx pos;
popPosition(L, pos);
uint32_t cid = popNumber(L);
ScriptEnviroment* env = getEnv();
Creature* creature = NULL;
if(cid)
{
if(!(creature = env->getCreatureByUID(cid)))
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
}
if(const Condition* condition = env->getConditionObject(conditionId))
{
const CombatArea* area = env->getCombatArea(areaId);
if(area || !areaId)
{
CombatParams params;
params.effects.impact = effect;
params.conditionList.push_back(condition);
Combat::doCombatCondition(creature, pos, area, params);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_AREA_NOT_FOUND));
lua_pushboolean(L, false);
}
}
else
{
errorEx(getError(LUA_ERROR_CONDITION_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoTargetCombatCondition(lua_State* L)
{
//doTargetCombatCondition(cid, target, condition, effect)
MagicEffect_t effect = (MagicEffect_t)popNumber(L);
uint32_t conditionId = popNumber(L), targetCid = popNumber(L), cid = popNumber(L);
ScriptEnviroment* env = getEnv();
Creature* creature = NULL;
if(cid)
{
if(!(creature = env->getCreatureByUID(cid)))
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
}
if(Creature* target = env->getCreatureByUID(targetCid))
{
if(const Condition* condition = env->getConditionObject(conditionId))
{
CombatParams params;
params.effects.impact = effect;
params.conditionList.push_back(condition);
Combat::doCombatCondition(creature, target, params);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CONDITION_NOT_FOUND));
lua_pushboolean(L, false);
}
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCombatAreaDispel(lua_State* L)
{
//doCombatAreaDispel(cid, pos, area, type, effect)
MagicEffect_t effect = (MagicEffect_t)popNumber(L);
ConditionType_t dispelType = (ConditionType_t)popNumber(L);
uint32_t areaId = popNumber(L);
PositionEx pos;
popPosition(L, pos);
uint32_t cid = popNumber(L);
ScriptEnviroment* env = getEnv();
Creature* creature = NULL;
if(cid)
{
if(!(creature = env->getCreatureByUID(cid)))
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
}
const CombatArea* area = env->getCombatArea(areaId);
if(area || !areaId)
{
CombatParams params;
params.effects.impact = effect;
params.dispelType = dispelType;
Combat::doCombatDispel(creature, pos, area, params);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_AREA_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoTargetCombatDispel(lua_State* L)
{
//doTargetCombatDispel(cid, target, type, effect)
MagicEffect_t effect = (MagicEffect_t)popNumber(L);
ConditionType_t dispelType = (ConditionType_t)popNumber(L);
uint32_t targetCid = popNumber(L), cid = popNumber(L);
ScriptEnviroment* env = getEnv();
Creature* creature = NULL;
if(cid)
{
if(!(creature = env->getCreatureByUID(cid)))
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
}
if(Creature* target = env->getCreatureByUID(targetCid))
{
CombatParams params;
params.effects.impact = effect;
params.dispelType = dispelType;
Combat::doCombatDispel(creature, target, params);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoChallengeCreature(lua_State* L)
{
//doChallengeCreature(cid, target)
ScriptEnviroment* env = getEnv();
uint32_t targetCid = popNumber(L);
Creature* creature = env->getCreatureByUID(popNumber(L));
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Creature* target = env->getCreatureByUID(targetCid);
if(!target)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
target->challengeCreature(creature);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoSummonMonster(lua_State* L)
{
//doSummonMonster(cid, name)
std::string name = popString(L);
ScriptEnviroment* env = getEnv();
Creature* creature = env->getCreatureByUID(popNumber(L));
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
lua_pushnumber(L, g_game.placeSummon(creature, name));
return 1;
}
int32_t LuaInterface::luaDoConvinceCreature(lua_State* L)
{
//doConvinceCreature(cid, target)
uint32_t cid = popNumber(L);
ScriptEnviroment* env = getEnv();
Creature* creature = env->getCreatureByUID(popNumber(L));
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Creature* target = env->getCreatureByUID(cid);
if(!target)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
target->convinceCreature(creature);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaGetMonsterTargetList(lua_State* L)
{
//getMonsterTargetList(cid)
ScriptEnviroment* env = getEnv();
Creature* creature = env->getCreatureByUID(popNumber(L));
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Monster* monster = creature->getMonster();
if(!monster)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
const CreatureList& targetList = monster->getTargetList();
CreatureList::const_iterator it = targetList.begin();
lua_newtable(L);
for(uint32_t i = 1; it != targetList.end(); ++it, ++i)
{
if(monster->isTarget(*it))
{
lua_pushnumber(L, i);
lua_pushnumber(L, env->addThing(*it));
pushTable(L);
}
}
return 1;
}
int32_t LuaInterface::luaGetMonsterFriendList(lua_State* L)
{
//getMonsterFriendList(cid)
ScriptEnviroment* env = getEnv();
Creature* creature = env->getCreatureByUID(popNumber(L));
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Monster* monster = creature->getMonster();
if(!monster)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Creature* friendCreature;
const CreatureList& friendList = monster->getFriendList();
CreatureList::const_iterator it = friendList.begin();
lua_newtable(L);
for(uint32_t i = 1; it != friendList.end(); ++it, ++i)
{
friendCreature = (*it);
if(!friendCreature->isRemoved() && friendCreature->getPosition().z == monster->getPosition().z)
{
lua_pushnumber(L, i);
lua_pushnumber(L, env->addThing(*it));
pushTable(L);
}
}
return 1;
}
int32_t LuaInterface::luaDoMonsterSetTarget(lua_State* L)
{
//doMonsterSetTarget(cid, target)
uint32_t targetId = popNumber(L);
ScriptEnviroment* env = getEnv();
Creature* creature = env->getCreatureByUID(popNumber(L));
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Monster* monster = creature->getMonster();
if(!monster)
{
errorEx(getError(LUA_ERROR_MONSTER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Creature* target = env->getCreatureByUID(targetId);
if(!target)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
if(!monster->isSummon())
lua_pushboolean(L, monster->selectTarget(target));
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaDoMonsterChangeTarget(lua_State* L)
{
//doMonsterChangeTarget(cid)
ScriptEnviroment* env = getEnv();
Creature* creature = env->getCreatureByUID(popNumber(L));
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Monster* monster = creature->getMonster();
if(!monster)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
if(!monster->isSummon())
monster->searchTarget(TARGETSEARCH_RANDOM);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaGetMonsterInfo(lua_State* L)
{
//getMonsterInfo(name)
const MonsterType* mType = g_monsters.getMonsterType(popString(L));
if(!mType)
{
errorEx(getError(LUA_ERROR_MONSTER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
lua_newtable(L);
setField(L, "name", mType->name.c_str());
setField(L, "description", mType->nameDescription.c_str());
setField(L, "experience", mType->experience);
setField(L, "health", mType->health);
setField(L, "healthMax", mType->healthMax);
setField(L, "manaCost", mType->manaCost);
setField(L, "defense", mType->defense);
setField(L, "armor", mType->armor);
setField(L, "baseSpeed", mType->baseSpeed);
setField(L, "lookCorpse", mType->lookCorpse);
setField(L, "race", mType->race);
setField(L, "skull", mType->skull);
setField(L, "partyShield", mType->partyShield);
setField(L, "guildEmblem", mType->guildEmblem);
setFieldBool(L, "summonable", mType->isSummonable);
setFieldBool(L, "illusionable", mType->isIllusionable);
setFieldBool(L, "convinceable", mType->isConvinceable);
setFieldBool(L, "attackable", mType->isAttackable);
setFieldBool(L, "hostile", mType->isHostile);
lua_pushstring(L, "outfit"); // name the table created by pushOutfit
pushOutfit(L, mType->outfit);
pushTable(L);
createTable(L, "defenses");
SpellList::const_iterator it = mType->spellDefenseList.begin();
for(uint32_t i = 1; it != mType->spellDefenseList.end(); ++it, ++i)
{
createTable(L, i);
setField(L, "speed", it->speed);
setField(L, "chance", it->chance);
setField(L, "range", it->range);
setField(L, "minCombatValue", it->minCombatValue);
setField(L, "maxCombatValue", it->maxCombatValue);
setFieldBool(L, "isMelee", it->isMelee);
pushTable(L);
}
pushTable(L);
createTable(L, "attacks");
it = mType->spellAttackList.begin();
for(uint32_t i = 1; it != mType->spellAttackList.end(); ++it, ++i)
{
createTable(L, i);
setField(L, "speed", it->speed);
setField(L, "chance", it->chance);
setField(L, "range", it->range);
setField(L, "minCombatValue", it->minCombatValue);
setField(L, "maxCombatValue", it->maxCombatValue);
setFieldBool(L, "isMelee", it->isMelee);
pushTable(L);
}
pushTable(L);
createTable(L, "loot");
LootItems::const_iterator lit = mType->lootItems.begin();
for(uint32_t i = 1; lit != mType->lootItems.end(); ++lit, ++i)
{
createTable(L, i);
if(lit->ids.size() > 1)
{
createTable(L, "ids");
std::vector<uint16_t>::const_iterator iit = lit->ids.begin();
for(uint32_t j = 1; iit != lit->ids.end(); ++iit, ++j)
{
lua_pushnumber(L, j);
lua_pushnumber(L, (*iit));
pushTable(L);
}
pushTable(L);
}
else
setField(L, "id", lit->ids[0]);
setField(L, "count", lit->count);
setField(L, "chance", lit->chance);
setField(L, "subType", lit->subType);
setField(L, "actionId", lit->actionId);
setField(L, "uniqueId", lit->uniqueId);
setField(L, "text", lit->text);
if(lit->childLoot.size() > 0)
{
createTable(L, "child");
LootItems::const_iterator cit = lit->childLoot.begin();
for(uint32_t j = 1; cit != lit->childLoot.end(); ++cit, ++j)
{
createTable(L, j);
if(cit->ids.size() > 1)
{
createTable(L, "ids");
std::vector<uint16_t>::const_iterator iit = cit->ids.begin();
for(uint32_t k = 1; iit != cit->ids.end(); ++iit, ++k)
{
lua_pushnumber(L, k);
lua_pushnumber(L, (*iit));
pushTable(L);
}
pushTable(L);
}
else
setField(L, "id", cit->ids[0]);
setField(L, "count", cit->count);
setField(L, "chance", cit->chance);
setField(L, "subType", cit->subType);
setField(L, "actionId", cit->actionId);
setField(L, "uniqueId", cit->uniqueId);
setField(L, "text", cit->text);
pushTable(L);
}
pushTable(L);
}
pushTable(L);
}
pushTable(L);
createTable(L, "summons");
SummonList::const_iterator sit = mType->summonList.begin();
for(uint32_t i = 1; sit != mType->summonList.end(); ++sit, ++i)
{
createTable(L, i);
setField(L, "name", sit->name);
setField(L, "chance", sit->chance);
setField(L, "interval", sit->interval);
setField(L, "amount", sit->amount);
pushTable(L);
}
pushTable(L);
return 1;
}
int32_t LuaInterface::luaGetTalkActionList(lua_State* L)
{
//getTalkactionList()
lua_newtable(L);
TalkActionsMap::const_iterator it = g_talkActions->getFirstTalk();
for(uint32_t i = 1; it != g_talkActions->getLastTalk(); ++it, ++i)
{
createTable(L, i);
setField(L, "words", it->first);
setField(L, "access", it->second->getAccess());
setFieldBool(L, "log", it->second->isLogged());
setFieldBool(L, "logged", it->second->isLogged());
setFieldBool(L, "hide", it->second->isHidden());
setFieldBool(L, "hidden", it->second->isHidden());
setField(L, "functionName", it->second->getFunctionName());
setField(L, "channel", it->second->getChannel());
pushTable(L);
}
return 1;
}
int32_t LuaInterface::luaGetExperienceStageList(lua_State* L)
{
//getExperienceStageList()
if(!g_config.getBool(ConfigManager::EXPERIENCE_STAGES))
{
lua_pushboolean(L, false);
return true;
}
StageList::const_iterator it = g_game.getFirstStage();
lua_newtable(L);
for(uint32_t i = 1; it != g_game.getLastStage(); ++it, ++i)
{
createTable(L, i);
setField(L, "level", it->first);
setFieldFloat(L, "multiplier", it->second);
pushTable(L);
}
return 1;
}
int32_t LuaInterface::luaDoAddCondition(lua_State* L)
{
//doAddCondition(cid, condition)
uint32_t conditionId = popNumber(L);
ScriptEnviroment* env = getEnv();
Creature* creature = env->getCreatureByUID(popNumber(L));
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Condition* condition = env->getConditionObject(conditionId);
if(!condition)
{
errorEx(getError(LUA_ERROR_CONDITION_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
creature->addCondition(condition->clone());
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoRemoveCondition(lua_State* L)
{
//doRemoveCondition(cid, type[, subId])
uint32_t subId = 0;
if(lua_gettop(L) > 2)
subId = popNumber(L);
ConditionType_t conditionType = (ConditionType_t)popNumber(L);
ScriptEnviroment* env = getEnv();
Creature* creature = env->getCreatureByUID(popNumber(L));
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Condition* condition = NULL;
while((condition = creature->getCondition(conditionType, CONDITIONID_COMBAT, subId)))
creature->removeCondition(condition);
while((condition = creature->getCondition(conditionType, CONDITIONID_DEFAULT, subId)))
creature->removeCondition(condition);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoRemoveConditions(lua_State* L)
{
//doRemoveConditions(cid[, onlyPersistent])
bool onlyPersistent = true;
if(lua_gettop(L) > 1)
onlyPersistent = popNumber(L);
ScriptEnviroment* env = getEnv();
Creature* creature = env->getCreatureByUID(popNumber(L));
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
creature->removeConditions(CONDITIONEND_ABORT, onlyPersistent);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaNumberToVariant(lua_State* L)
{
//numberToVariant(number)
LuaVariant var;
var.type = VARIANT_NUMBER;
var.number = popNumber(L);
LuaInterface::pushVariant(L, var);
return 1;
}
int32_t LuaInterface::luaStringToVariant(lua_State* L)
{
//stringToVariant(string)
LuaVariant var;
var.type = VARIANT_STRING;
var.text = popString(L);
LuaInterface::pushVariant(L, var);
return 1;
}
int32_t LuaInterface::luaPositionToVariant(lua_State* L)
{
//positionToVariant(pos)
LuaVariant var;
var.type = VARIANT_POSITION;
popPosition(L, var.pos);
LuaInterface::pushVariant(L, var);
return 1;
}
int32_t LuaInterface::luaTargetPositionToVariant(lua_State* L)
{
//targetPositionToVariant(pos)
LuaVariant var;
var.type = VARIANT_TARGETPOSITION;
popPosition(L, var.pos);
LuaInterface::pushVariant(L, var);
return 1;
}
int32_t LuaInterface::luaVariantToNumber(lua_State* L)
{
//variantToNumber(var)
LuaVariant var = popVariant(L);
uint32_t number = 0;
if(var.type == VARIANT_NUMBER)
number = var.number;
lua_pushnumber(L, number);
return 1;
}
int32_t LuaInterface::luaVariantToString(lua_State* L)
{
//variantToString(var)
LuaVariant var = popVariant(L);
std::string text = "";
if(var.type == VARIANT_STRING)
text = var.text;
lua_pushstring(L, text.c_str());
return 1;
}
int32_t LuaInterface::luaVariantToPosition(lua_State* L)
{
//luaVariantToPosition(var)
LuaVariant var = popVariant(L);
PositionEx pos(0, 0, 0, 0);
if(var.type == VARIANT_POSITION || var.type == VARIANT_TARGETPOSITION)
pos = var.pos;
pushPosition(L, pos, pos.stackpos);
return 1;
}
int32_t LuaInterface::luaDoChangeSpeed(lua_State* L)
{
//doChangeSpeed(cid, delta)
int32_t delta = (int32_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
g_game.changeSpeed(creature, delta);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaSetCreatureOutfit(lua_State* L)
{
//doSetCreatureOutfit(cid, outfit[, time = -1])
int32_t time = -1;
if(lua_gettop(L) > 2)
time = (int32_t)popNumber(L);
Outfit_t outfit = popOutfit(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushboolean(L, Spell::CreateIllusion(creature, outfit, time) == RET_NOERROR);
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureOutfit(lua_State* L)
{
//getCreatureOutfit(cid)
ScriptEnviroment* env = getEnv();
if(const Creature* creature = env->getCreatureByUID(popNumber(L)))
pushOutfit(L, creature->getCurrentOutfit());
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaSetMonsterOutfit(lua_State* L)
{
//doSetMonsterOutfit(cid, name[, time = -1])
int32_t time = -1;
if(lua_gettop(L) > 2)
time = (int32_t)popNumber(L);
std::string name = popString(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushboolean(L, Spell::CreateIllusion(creature, name, time) == RET_NOERROR);
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaSetItemOutfit(lua_State* L)
{
//doSetItemOutfit(cid, item[, time = -1])
int32_t time = -1;
if(lua_gettop(L) > 2)
time = (int32_t)popNumber(L);
uint32_t item = (uint32_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushboolean(L, Spell::CreateIllusion(creature, item, time) == RET_NOERROR);
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetStorage(lua_State* L)
{
//getStorage(key)
ScriptEnviroment* env = getEnv();
std::string strValue;
if(env->getStorage(popNumber(L), strValue))
{
int32_t intValue = atoi(strValue.c_str());
if(intValue || strValue == "0")
lua_pushnumber(L, intValue);
else
lua_pushstring(L, strValue.c_str());
}
else
lua_pushnumber(L, -1);
return 1;
}
int32_t LuaInterface::luaDoSetStorage(lua_State* L)
{
//doSetStorage(key, value)
std::string value;
bool nil = false;
if(lua_isnil(L, -1))
{
nil = true;
lua_pop(L, 1);
}
else
value = popString(L);
ScriptEnviroment* env = getEnv();
if(!nil)
env->setStorage(popNumber(L), value);
else
env->eraseStorage(popNumber(L));
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaGetPlayerDepotItems(lua_State* L)
{
//getPlayerDepotItems(cid, depotid)
uint32_t depotid = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(const Depot* depot = player->getDepot(depotid, true))
lua_pushnumber(L, depot->getItemHoldingCount());
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetGuildId(lua_State* L)
{
//doPlayerSetGuildId(cid, id)
uint32_t id = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(player->guildId)
{
player->leaveGuild();
if(!id)
lua_pushboolean(L, true);
else if(IOGuild::getInstance()->guildExists(id))
lua_pushboolean(L, IOGuild::getInstance()->joinGuild(player, id));
else
lua_pushboolean(L, false);
}
else if(id && IOGuild::getInstance()->guildExists(id))
lua_pushboolean(L, IOGuild::getInstance()->joinGuild(player, id));
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetGuildLevel(lua_State* L)
{
//doPlayerSetGuildLevel(cid, level[, rank])
uint32_t rank = 0;
if(lua_gettop(L) > 2)
rank = popNumber(L);
GuildLevel_t level = (GuildLevel_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
lua_pushboolean(L, player->setGuildLevel(level, rank));
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetGuildNick(lua_State* L)
{
//doPlayerSetGuildNick(cid, nick)
std::string nick = popString(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->setGuildNick(nick);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetGuildId(lua_State* L)
{
//getGuildId(guildName)
uint32_t guildId;
if(IOGuild::getInstance()->getGuildId(guildId, popString(L)))
lua_pushnumber(L, guildId);
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaGetGuildMotd(lua_State* L)
{
//getGuildMotd(guildId)
uint32_t guildId = popNumber(L);
if(IOGuild::getInstance()->guildExists(guildId))
lua_pushstring(L, IOGuild::getInstance()->getMotd(guildId).c_str());
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaDoMoveCreature(lua_State* L)
{
//doMoveCreature(cid, direction[, flag = FLAG_NOLIMIT])
uint32_t flags = FLAG_NOLIMIT;
if(lua_gettop(L) > 2)
flags = popNumber(L);
int32_t direction = popNumber(L);
if(direction < NORTH || direction > NORTHEAST)
{
lua_pushboolean(L, false);
return 1;
}
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushnumber(L, g_game.internalMoveCreature(creature, (Direction)direction, flags));
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaIsCreature(lua_State* L)
{
//isCreature(cid)
ScriptEnviroment* env = getEnv();
lua_pushboolean(L, env->getCreatureByUID(popNumber(L)) ? true : false);
return 1;
}
int32_t LuaInterface::luaIsContainer(lua_State* L)
{
//isContainer(uid)
ScriptEnviroment* env = getEnv();
lua_pushboolean(L, env->getContainerByUID(popNumber(L)) ? true : false);
return 1;
}
int32_t LuaInterface::luaIsMovable(lua_State* L)
{
//isMovable(uid)
ScriptEnviroment* env = getEnv();
Thing* thing = env->getThingByUID(popNumber(L));
if(thing && thing->isPushable())
lua_pushboolean(L, true);
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaGetCreatureByName(lua_State* L)
{
//getCreatureByName(name)
ScriptEnviroment* env = getEnv();
if(Creature* creature = g_game.getCreatureByName(popString(L)))
lua_pushnumber(L, env->addThing(creature));
else
lua_pushnil(L);
return 1;
}
int32_t LuaInterface::luaGetPlayerByGUID(lua_State* L)
{
//getPlayerByGUID(guid)
ScriptEnviroment* env = getEnv();
if(Player* player = g_game.getPlayerByGuid(popNumber(L)))
lua_pushnumber(L, env->addThing(player));
else
lua_pushnil(L);
return 1;
}
int32_t LuaInterface::luaGetPlayerByNameWildcard(lua_State* L)
{
//getPlayerByNameWildcard(name~[, ret = false])
Player* player = NULL;
bool pushRet = false;
if(lua_gettop(L) > 1)
pushRet = popNumber(L);
ScriptEnviroment* env = getEnv();
ReturnValue ret = g_game.getPlayerByNameWildcard(popString(L), player);
if(ret == RET_NOERROR)
lua_pushnumber(L, env->addThing(player));
else if(pushRet)
lua_pushnumber(L, ret);
else
lua_pushnil(L);
return 1;
}
int32_t LuaInterface::luaGetPlayerGUIDByName(lua_State* L)
{
//getPlayerGUIDByName(name[, multiworld = false])
bool multiworld = false;
if(lua_gettop(L) > 1)
multiworld = popNumber(L);
std::string name = popString(L);
uint32_t guid;
if(Player* player = g_game.getPlayerByName(name.c_str()))
lua_pushnumber(L, player->getGUID());
else if(IOLoginData::getInstance()->getGuidByName(guid, name, multiworld))
lua_pushnumber(L, guid);
else
lua_pushnil(L);
return 1;
}
int32_t LuaInterface::luaGetPlayerNameByGUID(lua_State* L)
{
//getPlayerNameByGUID(guid[, multiworld = false[, displayError = true]])
int32_t parameters = lua_gettop(L);
bool multiworld = false, displayError = true;
if(parameters > 2)
displayError = popNumber(L);
if(parameters > 1)
multiworld = popNumber(L);
uint32_t guid = popNumber(L);
std::string name;
if(!IOLoginData::getInstance()->getNameByGuid(guid, name, multiworld))
{
if(displayError)
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushnil(L);
return 1;
}
lua_pushstring(L, name.c_str());
return 1;
}
int32_t LuaInterface::luaGetPlayersByAccountId(lua_State* L)
{
//getPlayersByAccountId(accId)
PlayerVector players = g_game.getPlayersByAccount(popNumber(L));
ScriptEnviroment* env = getEnv();
PlayerVector::iterator it = players.begin();
lua_newtable(L);
for(uint32_t i = 1; it != players.end(); ++it, ++i)
{
lua_pushnumber(L, i);
lua_pushnumber(L, env->addThing(*it));
pushTable(L);
}
return 1;
}
int32_t LuaInterface::luaGetIpByName(lua_State* L)
{
//getIpByName(name)
std::string name = popString(L);
if(Player* player = g_game.getPlayerByName(name))
lua_pushnumber(L, player->getIP());
else
lua_pushnumber(L, IOLoginData::getInstance()->getLastIPByName(name));
return 1;
}
int32_t LuaInterface::luaGetPlayersByIp(lua_State* L)
{
//getPlayersByIp(ip[, mask])
uint32_t mask = 0xFFFFFFFF;
if(lua_gettop(L) > 1)
mask = (uint32_t)popNumber(L);
PlayerVector players = g_game.getPlayersByIP(popNumber(L), mask);
ScriptEnviroment* env = getEnv();
PlayerVector::iterator it = players.begin();
lua_newtable(L);
for(uint32_t i = 1; it != players.end(); ++it, ++i)
{
lua_pushnumber(L, i);
lua_pushnumber(L, env->addThing(*it));
pushTable(L);
}
return 1;
}
int32_t LuaInterface::luaGetAccountIdByName(lua_State* L)
{
//getAccountIdByName(name)
std::string name = popString(L);
if(Player* player = g_game.getPlayerByName(name))
lua_pushnumber(L, player->getAccount());
else
lua_pushnumber(L, IOLoginData::getInstance()->getAccountIdByName(name));
return 1;
}
int32_t LuaInterface::luaGetAccountByName(lua_State* L)
{
//getAccountByName(name)
std::string name = popString(L);
if(Player* player = g_game.getPlayerByName(name))
lua_pushstring(L, player->getAccountName().c_str());
else
{
std::string tmp;
IOLoginData::getInstance()->getAccountName(IOLoginData::getInstance()->getAccountIdByName(name), tmp);
lua_pushstring(L, tmp.c_str());
}
return 1;
}
int32_t LuaInterface::luaGetAccountIdByAccount(lua_State* L)
{
//getAccountIdByAccount(accName)
uint32_t value = 0;
IOLoginData::getInstance()->getAccountId(popString(L), value);
lua_pushnumber(L, value);
return 1;
}
int32_t LuaInterface::luaGetAccountByAccountId(lua_State* L)
{
//getAccountByAccountId(accId)
std::string value = 0;
IOLoginData::getInstance()->getAccountName(popNumber(L), value);
lua_pushstring(L, value.c_str());
return 1;
}
int32_t LuaInterface::luaRegisterCreatureEvent(lua_State* L)
{
//registerCreatureEvent(cid, name)
std::string name = popString(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushboolean(L, creature->registerCreatureEvent(name));
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaUnregisterCreatureEvent(lua_State* L)
{
//unregisterCreatureEvent(cid, name)
std::string name = popString(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushboolean(L, creature->unregisterCreatureEvent(name));
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetContainerSize(lua_State* L)
{
//getContainerSize(uid)
ScriptEnviroment* env = getEnv();
if(Container* container = env->getContainerByUID(popNumber(L)))
lua_pushnumber(L, container->size());
else
{
errorEx(getError(LUA_ERROR_CONTAINER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetContainerCap(lua_State* L)
{
//getContainerCap(uid)
ScriptEnviroment* env = getEnv();
if(Container* container = env->getContainerByUID(popNumber(L)))
lua_pushnumber(L, container->capacity());
else
{
errorEx(getError(LUA_ERROR_CONTAINER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetContainerItem(lua_State* L)
{
//getContainerItem(uid, slot)
uint32_t slot = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Container* container = env->getContainerByUID(popNumber(L)))
{
if(Item* item = container->getItem(slot))
pushThing(L, item, env->addThing(item));
else
pushThing(L, NULL, 0);
}
else
{
errorEx(getError(LUA_ERROR_CONTAINER_NOT_FOUND));
pushThing(L, NULL, 0);
}
return 1;
}
int32_t LuaInterface::luaDoAddContainerItemEx(lua_State* L)
{
//doAddContainerItemEx(uid, virtuid)
uint32_t virtuid = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Container* container = env->getContainerByUID(popNumber(L)))
{
Item* item = env->getItemByUID(virtuid);
if(!item)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
if(item->getParent() != VirtualCylinder::virtualCylinder)
{
lua_pushboolean(L, false);
return 1;
}
ReturnValue ret = g_game.internalAddItem(NULL, container, item);
if(ret == RET_NOERROR)
env->removeTempItem(item);
lua_pushnumber(L, ret);
return 1;
}
else
{
errorEx(getError(LUA_ERROR_CONTAINER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
}
int32_t LuaInterface::luaDoAddContainerItem(lua_State* L)
{
//doAddContainerItem(uid, itemid[, count/subType = 1])
uint32_t count = 1;
if(lua_gettop(L) > 2)
count = popNumber(L);
uint16_t itemId = popNumber(L);
ScriptEnviroment* env = getEnv();
Container* container = env->getContainerByUID((uint32_t)popNumber(L));
if(!container)
{
errorEx(getError(LUA_ERROR_CONTAINER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
const ItemType& it = Item::items[itemId];
int32_t itemCount = 1, subType = 1;
if(it.hasSubType())
{
if(it.stackable)
itemCount = (int32_t)std::ceil((float)count / 100);
subType = count;
}
else
itemCount = std::max((uint32_t)1, count);
while(itemCount > 0)
{
int32_t stackCount = std::min(100, subType);
Item* newItem = Item::CreateItem(itemId, stackCount);
if(!newItem)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
if(it.stackable)
subType -= stackCount;
ReturnValue ret = g_game.internalAddItem(NULL, container, newItem);
if(ret != RET_NOERROR)
{
delete newItem;
lua_pushboolean(L, false);
return 1;
}
--itemCount;
if(itemCount)
continue;
if(newItem->getParent())
lua_pushnumber(L, env->addThing(newItem));
else //stackable item stacked with existing object, newItem will be released
lua_pushnil(L);
return 1;
}
lua_pushnil(L);
return 1;
}
int32_t LuaInterface::luaDoPlayerAddOutfit(lua_State *L)
{
//Consider using doPlayerAddOutfitId instead
//doPlayerAddOutfit(cid, looktype, addon)
uint32_t addon = popNumber(L), lookType = popNumber(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID((uint32_t)popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Outfit outfit;
if(Outfits::getInstance()->getOutfit(lookType, outfit))
{
lua_pushboolean(L, player->addOutfit(outfit.outfitId, addon));
return 1;
}
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaDoPlayerRemoveOutfit(lua_State *L)
{
//Consider using doPlayerRemoveOutfitId instead
//doPlayerRemoveOutfit(cid, looktype[, addon = 0])
uint32_t addon = 0xFF;
if(lua_gettop(L) > 2)
addon = popNumber(L);
uint32_t lookType = popNumber(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID((uint32_t)popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Outfit outfit;
if(Outfits::getInstance()->getOutfit(lookType, outfit))
{
lua_pushboolean(L, player->removeOutfit(outfit.outfitId, addon));
return 1;
}
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaDoPlayerAddOutfitId(lua_State *L)
{
//doPlayerAddOutfitId(cid, outfitId, addon)
uint32_t addon = popNumber(L), outfitId = popNumber(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID((uint32_t)popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
lua_pushboolean(L, player->addOutfit(outfitId, addon));
return 1;
}
int32_t LuaInterface::luaDoPlayerRemoveOutfitId(lua_State *L)
{
//doPlayerRemoveOutfitId(cid, outfitId[, addon = 0])
uint32_t addon = 0xFF;
if(lua_gettop(L) > 2)
addon = popNumber(L);
uint32_t outfitId = popNumber(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID((uint32_t)popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
lua_pushboolean(L, player->removeOutfit(outfitId, addon));
return 1;
}
int32_t LuaInterface::luaCanPlayerWearOutfit(lua_State* L)
{
//canPlayerWearOutfit(cid, looktype[, addon = 0])
uint32_t addon = 0;
if(lua_gettop(L) > 2)
addon = popNumber(L);
uint32_t lookType = popNumber(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Outfit outfit;
if(Outfits::getInstance()->getOutfit(lookType, outfit))
{
lua_pushboolean(L, player->canWearOutfit(outfit.outfitId, addon));
return 1;
}
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaCanPlayerWearOutfitId(lua_State* L)
{
//canPlayerWearOutfitId(cid, outfitId[, addon = 0])
uint32_t addon = 0;
if(lua_gettop(L) > 2)
addon = popNumber(L);
uint32_t outfitId = popNumber(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
lua_pushboolean(L, player->canWearOutfit(outfitId, addon));
return 1;
}
int32_t LuaInterface::luaDoCreatureChangeOutfit(lua_State* L)
{
//doCreatureChangeOutfit(cid, outfit)
Outfit_t outfit = popOutfit(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
if(Player* player = creature->getPlayer())
player->changeOutfit(outfit, false);
else
creature->defaultOutfit = outfit;
if(!creature->hasCondition(CONDITION_OUTFIT, 1))
g_game.internalCreatureChangeOutfit(creature, outfit);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerPopupFYI(lua_State* L)
{
//doPlayerPopupFYI(cid, message)
std::string message = popString(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->sendFYIBox(message);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSendTutorial(lua_State* L)
{
//doPlayerSendTutorial(cid, id)
uint8_t id = (uint8_t)popNumber(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
player->sendTutorial(id);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoPlayerSendMailByName(lua_State* L)
{
//doPlayerSendMailByName(name, item[, town[, actor]])
ScriptEnviroment* env = getEnv();
int32_t params = lua_gettop(L);
Creature* actor = NULL;
if(params > 3)
actor = env->getCreatureByUID(popNumber(L));
uint32_t town = 0;
if(params > 2)
town = popNumber(L);
Item* item = env->getItemByUID(popNumber(L));
if(!item)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
if(item->getParent() != VirtualCylinder::virtualCylinder)
{
lua_pushboolean(L, false);
return 1;
}
lua_pushboolean(L, IOLoginData::getInstance()->playerMail(actor, popString(L), town, item));
return 1;
}
int32_t LuaInterface::luaDoPlayerAddMapMark(lua_State* L)
{
//doPlayerAddMapMark(cid, pos, type[, description])
std::string description;
if(lua_gettop(L) > 3)
description = popString(L);
MapMarks_t type = (MapMarks_t)popNumber(L);
PositionEx pos;
popPosition(L, pos);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
player->sendAddMarker(pos, type, description);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoPlayerAddPremiumDays(lua_State* L)
{
//doPlayerAddPremiumDays(cid, days)
int32_t days = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(player->premiumDays < 65535)
{
Account account = IOLoginData::getInstance()->loadAccount(player->getAccount());
if(days < 0)
{
account.premiumDays = std::max((uint32_t)0, uint32_t(account.premiumDays + (int32_t)days));
player->premiumDays = std::max((uint32_t)0, uint32_t(player->premiumDays + (int32_t)days));
}
else
{
account.premiumDays = std::min((uint32_t)65534, uint32_t(account.premiumDays + (uint32_t)days));
player->premiumDays = std::min((uint32_t)65534, uint32_t(player->premiumDays + (uint32_t)days));
}
IOLoginData::getInstance()->saveAccount(account);
}
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureLastPosition(lua_State* L)
{
//getCreatureLastPosition(cid)
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
pushPosition(L, creature->getLastPosition(), 0);
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureName(lua_State* L)
{
//getCreatureName(cid)
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushstring(L, creature->getName().c_str());
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureNoMove(lua_State* L)
{
//getCreatureNoMove(cid)
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushboolean(L, creature->getNoMove());
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureGuildEmblem(lua_State* L)
{
//getCreatureGuildEmblem(cid[, target])
uint32_t tid = 0;
if(lua_gettop(L) > 1)
tid = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
if(!tid)
lua_pushnumber(L, creature->getEmblem());
else if(Creature* target = env->getCreatureByUID(tid))
lua_pushnumber(L, creature->getGuildEmblem(target));
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCreatureSetGuildEmblem(lua_State* L)
{
//doCreatureSetGuildEmblem(cid, emblem)
GuildEmblems_t emblem = (GuildEmblems_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
creature->setEmblem(emblem);
g_game.updateCreatureEmblem(creature);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreaturePartyShield(lua_State* L)
{
//getCreaturePartyShield(cid[, target])
uint32_t tid = 0;
if(lua_gettop(L) > 1)
tid = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
if(!tid)
lua_pushnumber(L, creature->getShield());
else if(Creature* target = env->getCreatureByUID(tid))
lua_pushnumber(L, creature->getPartyShield(target));
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCreatureSetPartyShield(lua_State* L)
{
//doCreatureSetPartyShield(cid, shield)
PartyShields_t shield = (PartyShields_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
creature->setShield(shield);
g_game.updateCreatureShield(creature);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureSkullType(lua_State* L)
{
//getCreatureSkullType(cid[, target])
uint32_t tid = 0;
if(lua_gettop(L) > 1)
tid = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
if(!tid)
lua_pushnumber(L, creature->getSkull());
else if(Creature* target = env->getCreatureByUID(tid))
lua_pushnumber(L, creature->getSkullType(target));
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCreatureSetLookDir(lua_State* L)
{
//doCreatureSetLookDirection(cid, dir)
Direction dir = (Direction)popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
if(dir < NORTH || dir > WEST)
{
lua_pushboolean(L, false);
return 1;
}
g_game.internalCreatureTurn(creature, dir);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCreatureSetSkullType(lua_State* L)
{
//doCreatureSetSkullType(cid, skull)
Skulls_t skull = (Skulls_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
creature->setSkull(skull);
g_game.updateCreatureSkull(creature);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetSkullEnd(lua_State* L)
{
//doPlayerSetSkullEnd(cid, time, type)
Skulls_t _skull = (Skulls_t)popNumber(L);
time_t _time = (time_t)std::max((int64_t)0, popNumber(L));
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->setSkullEnd(_time, false, _skull);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureSpeed(lua_State* L)
{
//getCreatureSpeed(cid)
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushnumber(L, creature->getSpeed());
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureBaseSpeed(lua_State* L)
{
//getCreatureBaseSpeed(cid)
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushnumber(L, creature->getBaseSpeed());
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureTarget(lua_State* L)
{
//getCreatureTarget(cid)
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
Creature* target = creature->getAttackedCreature();
lua_pushnumber(L, target ? env->addThing(target) : 0);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaIsSightClear(lua_State* L)
{
//isSightClear(fromPos, toPos, floorCheck)
PositionEx fromPos, toPos;
bool floorCheck = popNumber(L);
popPosition(L, toPos);
popPosition(L, fromPos);
lua_pushboolean(L, g_game.isSightClear(fromPos, toPos, floorCheck));
return 1;
}
int32_t LuaInterface::luaIsInArray(lua_State* L)
{
//isInArray(array, value[, caseSensitive = false])
bool caseSensitive = false;
if(lua_gettop(L) > 2)
caseSensitive = popNumber(L);
boost::any value;
if(lua_isnumber(L, -1))
value = popFloatNumber(L);
else if(lua_isboolean(L, -1))
value = popBoolean(L);
else if(lua_isstring(L, -1))
value = popString(L);
else
{
lua_pop(L, 1);
lua_pushboolean(L, false);
return 1;
}
const std::type_info& type = value.type();
if(!caseSensitive && type == typeid(std::string))
value = asLowerCaseString(boost::any_cast<std::string>(value));
if(!lua_istable(L, -1))
{
boost::any data;
if(lua_isnumber(L, -1))
data = popFloatNumber(L);
else if(lua_isboolean(L, -1))
data = popBoolean(L);
else if(lua_isstring(L, -1))
data = popString(L);
else
{
lua_pop(L, 1);
lua_pushboolean(L, false);
return 1;
}
if(type != data.type()) // check is it even same data type before searching deeper
lua_pushboolean(L, false);
else if(type == typeid(bool))
lua_pushboolean(L, boost::any_cast<bool>(value) == boost::any_cast<bool>(data));
else if(type == typeid(double))
lua_pushboolean(L, boost::any_cast<double>(value) == boost::any_cast<double>(data));
else if(caseSensitive)
lua_pushboolean(L, boost::any_cast<std::string>(value) == boost::any_cast<std::string>(data));
else
lua_pushboolean(L, boost::any_cast<std::string>(value) == asLowerCaseString(boost::any_cast<std::string>(data)));
return 1;
}
lua_pushnil(L);
while(lua_next(L, -2))
{
boost::any data;
if(lua_isnumber(L, -1))
data = popFloatNumber(L);
else if(lua_isboolean(L, -1))
data = popBoolean(L);
else if(lua_isstring(L, -1))
data = popString(L);
else
{
lua_pop(L, 1);
break;
}
if(type != data.type()) // check is it same data type before searching deeper
continue;
if(type == typeid(bool))
{
if(boost::any_cast<bool>(value) != boost::any_cast<bool>(data))
continue;
lua_pushboolean(L, true);
return 1;
}
else if(type == typeid(double))
{
if(boost::any_cast<double>(value) != boost::any_cast<double>(data))
continue;
lua_pushboolean(L, true);
return 1;
}
else if(caseSensitive)
{
if(boost::any_cast<std::string>(value) != boost::any_cast<std::string>(data))
continue;
lua_pushboolean(L, true);
return 1;
}
else if(boost::any_cast<std::string>(value) == asLowerCaseString(boost::any_cast<std::string>(data)))
{
lua_pushboolean(L, true);
return 1;
}
}
lua_pop(L, 2);
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaAddEvent(lua_State* L)
{
//addEvent(callback, delay, ...)
ScriptEnviroment* env = getEnv();
LuaInterface* interface = env->getInterface();
if(!interface)
{
errorEx("No valid script interface!");
lua_pushboolean(L, false);
return 1;
}
int32_t parameters = lua_gettop(L);
if(!lua_isfunction(L, -parameters)) //-parameters means the first parameter from left to right
{
errorEx("Callback parameter should be a function.");
lua_pushboolean(L, false);
return 1;
}
std::list<int32_t> params;
for(int32_t i = 0; i < parameters - 2; ++i) //-2 because addEvent needs at least two parameters
params.push_back(luaL_ref(L, LUA_REGISTRYINDEX));
LuaTimerEvent event;
event.eventId = Scheduler::getInstance().addEvent(createSchedulerTask(std::max((int64_t)SCHEDULER_MINTICKS, popNumber(L)),
boost::bind(&LuaInterface::executeTimer, interface, ++interface->m_lastTimer)));
event.parameters = params;
event.function = luaL_ref(L, LUA_REGISTRYINDEX);
event.scriptId = env->getScriptId();
interface->m_timerEvents[interface->m_lastTimer] = event;
lua_pushnumber(L, interface->m_lastTimer);
return 1;
}
int32_t LuaInterface::luaStopEvent(lua_State* L)
{
//stopEvent(eventid)
uint32_t eventId = popNumber(L);
ScriptEnviroment* env = getEnv();
LuaInterface* interface = env->getInterface();
if(!interface)
{
errorEx("No valid script interface!");
lua_pushboolean(L, false);
return 1;
}
LuaTimerEvents::iterator it = interface->m_timerEvents.find(eventId);
if(it != interface->m_timerEvents.end())
{
Scheduler::getInstance().stopEvent(it->second.eventId);
for(std::list<int32_t>::iterator lt = it->second.parameters.begin(); lt != it->second.parameters.end(); ++lt)
luaL_unref(interface->m_luaState, LUA_REGISTRYINDEX, *lt);
it->second.parameters.clear();
luaL_unref(interface->m_luaState, LUA_REGISTRYINDEX, it->second.function);
interface->m_timerEvents.erase(it);
lua_pushboolean(L, true);
}
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaGetCreatureCondition(lua_State* L)
{
//getCreatureCondition(cid, condition[, subId = 0])
uint32_t subId = 0, condition = 0;
if(lua_gettop(L) > 2)
subId = popNumber(L);
condition = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushboolean(L, creature->hasCondition((ConditionType_t)condition, subId));
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerBlessing(lua_State* L)
{
//getPlayerBlessings(cid, blessing)
int16_t blessing = popNumber(L) - 1;
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
lua_pushboolean(L, player->hasBlessing(blessing));
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerAddBlessing(lua_State* L)
{
//doPlayerAddBlessing(cid, blessing)
int16_t blessing = popNumber(L) - 1;
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(!player->hasBlessing(blessing))
{
player->addBlessing(1 << blessing);
lua_pushboolean(L, true);
}
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetPromotionLevel(lua_State* L)
{
//doPlayerSetPromotionLevel(cid, level)
uint32_t level = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->setPromotionLevel(level);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetGroupId(lua_State* L)
{
//doPlayerSetGroupId(cid, groupId)
uint32_t groupId = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(Group* group = Groups::getInstance()->getGroup(groupId))
{
player->setGroup(group);
lua_pushboolean(L, true);
}
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureMana(lua_State* L)
{
//getCreatureMana(cid)
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushnumber(L, creature->getMana());
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureMaxMana(lua_State* L)
{
//getCreatureMaxMana(cid)
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushnumber(L, creature->getMaxMana());
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureHealth(lua_State* L)
{
//getCreatureHealth(cid)
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushnumber(L, creature->getHealth());
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureLookDirection(lua_State* L)
{
//getCreatureLookDirection(cid)
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushnumber(L, creature->getDirection());
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureMaxHealth(lua_State* L)
{
//getCreatureMaxHealth(cid)
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushnumber(L, creature->getMaxHealth());
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetStamina(lua_State* L)
{
//doPlayerSetStamina(cid, minutes)
uint32_t minutes = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->setStaminaMinutes(minutes);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetBalance(lua_State* L)
{
//doPlayerSetBalance(cid, balance)
uint64_t balance = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->balance = balance;
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetPartner(lua_State* L)
{
//doPlayerSetPartner(cid, guid)
uint32_t guid = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->marriage = guid;
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerFollowCreature(lua_State* L)
{
//doPlayerFollowCreature(cid, target)
ScriptEnviroment* env = getEnv();
Creature* creature = env->getCreatureByUID(popNumber(L));
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
lua_pushboolean(L, g_game.playerFollowCreature(player->getID(), creature->getID()));
return 1;
}
int32_t LuaInterface::luaGetPlayerParty(lua_State* L)
{
//getPlayerParty(cid)
uint32_t cid = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(cid))
{
if(Party* party = player->getParty())
lua_pushnumber(L, env->addThing(party->getLeader()));
else
lua_pushnil(L);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerJoinParty(lua_State* L)
{
//doPlayerJoinParty(cid, lid)
ScriptEnviroment* env = getEnv();
Player* leader = env->getPlayerByUID(popNumber(L));
if(!leader)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
g_game.playerJoinParty(player->getID(), leader->getID());
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoPlayerLeaveParty(lua_State* L)
{
//doPlayerLeaveParty(cid[, forced = false])
bool forced = false;
if(lua_gettop(L) > 1)
forced = popNumber(L);
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
g_game.playerLeaveParty(player->getID(), forced);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaGetPartyMembers(lua_State* L)
{
//getPartyMembers(cid)
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(Party* party = player->getParty())
{
PlayerVector list = party->getMembers();
list.push_back(party->getLeader());
PlayerVector::const_iterator it = list.begin();
lua_newtable(L);
for(uint32_t i = 1; it != list.end(); ++it, ++i)
{
lua_pushnumber(L, i);
lua_pushnumber(L, (*it)->getID());
pushTable(L);
}
return 1;
}
}
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaGetVocationInfo(lua_State* L)
{
//getVocationInfo(id)
uint32_t id = popNumber(L);
Vocation* voc = Vocations::getInstance()->getVocation(id);
if(!voc)
{
lua_pushboolean(L, false);
return 1;
}
lua_newtable(L);
setField(L, "id", voc->getId());
setField(L, "name", voc->getName().c_str());
setField(L, "description", voc->getDescription().c_str());
setField(L, "healthGain", voc->getGain(GAIN_HEALTH));
setField(L, "healthGainTicks", voc->getGainTicks(GAIN_HEALTH));
setField(L, "healthGainAmount", voc->getGainAmount(GAIN_HEALTH));
setField(L, "manaGain", voc->getGain(GAIN_MANA));
setField(L, "manaGainTicks", voc->getGainTicks(GAIN_MANA));
setField(L, "manaGainAmount", voc->getGainAmount(GAIN_MANA));
setField(L, "attackSpeed", voc->getAttackSpeed());
setField(L, "baseSpeed", voc->getBaseSpeed());
setField(L, "fromVocation", voc->getFromVocation());
setField(L, "promotedVocation", Vocations::getInstance()->getPromotedVocation(id));
setField(L, "soul", voc->getGain(GAIN_SOUL));
setField(L, "soulAmount", voc->getGainAmount(GAIN_SOUL));
setField(L, "soulTicks", voc->getGainTicks(GAIN_SOUL));
setField(L, "capacity", voc->getGainCap());
setFieldBool(L, "attackable", voc->isAttackable());
setFieldBool(L, "needPremium", voc->isPremiumNeeded());
setFieldFloat(L, "experienceMultiplier", voc->getExperienceMultiplier());
return 1;
}
int32_t LuaInterface::luaGetGroupInfo(lua_State* L)
{
//getGroupInfo(id[, premium = false])
bool premium = false;
if(lua_gettop(L) > 1)
premium = popNumber(L);
Group* group = Groups::getInstance()->getGroup(popNumber(L));
if(!group)
{
lua_pushboolean(L, false);
return 1;
}
lua_newtable(L);
setField(L, "id", group->getId());
setField(L, "name", group->getName().c_str());
setField(L, "access", group->getAccess());
setField(L, "ghostAccess", group->getGhostAccess());
setField(L, "violationReasons", group->getViolationReasons());
setField(L, "statementViolationFlags", group->getStatementViolationFlags());
setField(L, "nameViolationFlags", group->getNameViolationFlags());
setField(L, "flags", group->getFlags());
setField(L, "customFlags", group->getCustomFlags());
setField(L, "depotLimit", group->getDepotLimit(premium));
setField(L, "maxVips", group->getMaxVips(premium));
setField(L, "outfit", group->getOutfit());
return 1;
}
int32_t LuaInterface::luaGetChannelUsers(lua_State* L)
{
//getChannelUsers(channelId)
ScriptEnviroment* env = getEnv();
uint16_t channelId = popNumber(L);
if(ChatChannel* channel = g_chat.getChannelById(channelId))
{
UsersMap usersMap = channel->getUsers();
UsersMap::iterator it = usersMap.begin();
lua_newtable(L);
for(int32_t i = 1; it != usersMap.end(); ++it, ++i)
{
lua_pushnumber(L, i);
lua_pushnumber(L, env->addThing(it->second));
pushTable(L);
}
}
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaGetPlayersOnline(lua_State* L)
{
//getPlayersOnline()
ScriptEnviroment* env = getEnv();
AutoList<Player>::iterator it = Player::autoList.begin();
lua_newtable(L);
for(int32_t i = 1; it != Player::autoList.end(); ++it, ++i)
{
lua_pushnumber(L, i);
lua_pushnumber(L, env->addThing(it->second));
pushTable(L);
}
return 1;
}
int32_t LuaInterface::luaSetCreatureMaxHealth(lua_State* L)
{
//setCreatureMaxHealth(uid, health)
uint32_t maxHealth = (uint32_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
creature->changeMaxHealth(maxHealth);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaSetCreatureMaxMana(lua_State* L)
{
//setCreatureMaxMana(uid, mana)
uint32_t maxMana = (uint32_t)popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
creature->changeMaxMana(maxMana);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetMaxCapacity(lua_State* L)
{
//doPlayerSetMaxCapacity(uid, cap)
double cap = popFloatNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->setCapacity(cap);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetCreatureMaster(lua_State* L)
{
//getCreatureMaster(cid)
uint32_t cid = popNumber(L);
ScriptEnviroment* env = getEnv();
Creature* creature = env->getCreatureByUID(cid);
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
if(Creature* master = creature->getMaster())
lua_pushnumber(L, env->addThing(master));
else
lua_pushnil(L);
return 1;
}
int32_t LuaInterface::luaGetCreatureSummons(lua_State* L)
{
//getCreatureSummons(cid)
ScriptEnviroment* env = getEnv();
Creature* creature = env->getCreatureByUID(popNumber(L));
if(!creature)
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
const std::list<Creature*>& summons = creature->getSummons();
CreatureList::const_iterator it = summons.begin();
lua_newtable(L);
for(uint32_t i = 1; it != summons.end(); ++it, ++i)
{
lua_pushnumber(L, i);
lua_pushnumber(L, env->addThing(*it));
pushTable(L);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetIdleTime(lua_State* L)
{
//doPlayerSetIdleTime(cid, amount)
int64_t amount = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->setIdleTime(amount);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoCreatureSetNoMove(lua_State* L)
{
//doCreatureSetNoMove(cid, block)
bool block = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
{
creature->setNoMove(block);
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetPlayerModes(lua_State* L)
{
//getPlayerModes(cid)
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
lua_newtable(L);
setField(L, "chase", player->getChaseMode());
setField(L, "fight", player->getFightMode());
setField(L, "secure", player->getSecureMode());
return 1;
}
int32_t LuaInterface::luaGetPlayerRates(lua_State* L)
{
//getPlayerRates(cid)
ScriptEnviroment* env = getEnv();
Player* player = env->getPlayerByUID(popNumber(L));
if(!player)
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
lua_newtable(L);
for(uint32_t i = SKILL_FIRST; i <= SKILL__LAST; ++i)
{
lua_pushnumber(L, i);
lua_pushnumber(L, player->rates[(skills_t)i]);
pushTable(L);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSetRate(lua_State* L)
{
//doPlayerSetRate(cid, type, value)
float value = popFloatNumber(L);
uint32_t type = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
if(type <= SKILL__LAST)
{
player->rates[(skills_t)type] = value;
lua_pushboolean(L, true);
}
else
lua_pushboolean(L, false);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSwitchSaving(lua_State* L)
{
//doPlayerSwitchSaving(cid)
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->switchSaving();
lua_pushboolean(L, true);
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoPlayerSave(lua_State* L)
{
//doPlayerSave(cid[, shallow = false])
bool shallow = false;
if(lua_gettop(L) > 1)
shallow = popNumber(L);
ScriptEnviroment* env = getEnv();
if(Player* player = env->getPlayerByUID(popNumber(L)))
{
player->loginPosition = player->getPosition();
lua_pushboolean(L, IOLoginData::getInstance()->savePlayer(player, false, shallow));
}
else
{
errorEx(getError(LUA_ERROR_PLAYER_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaGetTownId(lua_State* L)
{
//getTownId(townName)
std::string townName = popString(L);
if(Town* town = Towns::getInstance()->getTown(townName))
lua_pushnumber(L, town->getID());
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaGetTownName(lua_State* L)
{
//getTownName(townId)
uint32_t townId = popNumber(L);
if(Town* town = Towns::getInstance()->getTown(townId))
lua_pushstring(L, town->getName().c_str());
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaGetTownTemplePosition(lua_State* L)
{
//getTownTemplePosition(townId)
bool displayError = true;
if(lua_gettop(L) >= 2)
displayError = popNumber(L);
uint32_t townId = popNumber(L);
if(Town* town = Towns::getInstance()->getTown(townId))
pushPosition(L, town->getPosition(), 255);
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaGetTownHouses(lua_State* L)
{
//getTownHouses(townId)
uint32_t townId = 0;
if(lua_gettop(L) > 0)
townId = popNumber(L);
HouseMap::iterator it = Houses::getInstance()->getHouseBegin();
lua_newtable(L);
for(uint32_t i = 1; it != Houses::getInstance()->getHouseEnd(); ++i, ++it)
{
if(townId != 0 && it->second->getTownId() != townId)
continue;
lua_pushnumber(L, i);
lua_pushnumber(L, it->second->getId());
pushTable(L);
}
return 1;
}
int32_t LuaInterface::luaGetSpectators(lua_State* L)
{
//getSpectators(centerPos, rangex, rangey[, multifloor = false])
bool multifloor = false;
if(lua_gettop(L) > 3)
multifloor = popNumber(L);
uint32_t rangey = popNumber(L), rangex = popNumber(L);
PositionEx centerPos;
popPosition(L, centerPos);
SpectatorVec list;
g_game.getSpectators(list, centerPos, false, multifloor, rangex, rangex, rangey, rangey);
if(list.empty())
{
lua_pushnil(L);
return 1;
}
ScriptEnviroment* env = getEnv();
SpectatorVec::const_iterator it = list.begin();
lua_newtable(L);
for(uint32_t i = 1; it != list.end(); ++it, ++i)
{
lua_pushnumber(L, i);
lua_pushnumber(L, env->addThing(*it));
pushTable(L);
}
return 1;
}
int32_t LuaInterface::luaGetHighscoreString(lua_State* L)
{
//getHighscoreString(skillId)
uint16_t skillId = popNumber(L);
if(skillId <= SKILL__LAST)
lua_pushstring(L, g_game.getHighscoreString(skillId).c_str());
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaGetTownList(lua_State* L)
{
//getTownList()
lua_newtable(L);
TownMap::const_iterator it = Towns::getInstance()->getFirstTown();
for(uint32_t i = 1; it != Towns::getInstance()->getLastTown(); ++it, ++i)
{
createTable(L, i);
setField(L, "id", it->first);
setField(L, "name", it->second->getName());
pushTable(L);
}
return 1;
}
int32_t LuaInterface::luaGetWaypointList(lua_State* L)
{
//getWaypointList()
WaypointMap waypointsMap = g_game.getMap()->waypoints.getWaypointsMap();
WaypointMap::iterator it = waypointsMap.begin();
lua_newtable(L);
for(uint32_t i = 1; it != waypointsMap.end(); ++it, ++i)
{
createTable(L, i);
setField(L, "name", it->first);
setField(L, "pos", it->second->pos.x);
pushTable(L);
}
return 1;
}
int32_t LuaInterface::luaGetWaypointPosition(lua_State* L)
{
//getWaypointPosition(name)
if(WaypointPtr waypoint = g_game.getMap()->waypoints.getWaypointByName(popString(L)))
pushPosition(L, waypoint->pos, 0);
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaDoWaypointAddTemporial(lua_State* L)
{
//doWaypointAddTemporial(name, pos)
PositionEx pos;
popPosition(L, pos);
g_game.getMap()->waypoints.addWaypoint(WaypointPtr(new Waypoint(popString(L), pos)));
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaGetGameState(lua_State* L)
{
//getGameState()
lua_pushnumber(L, g_game.getGameState());
return 1;
}
int32_t LuaInterface::luaDoSetGameState(lua_State* L)
{
//doSetGameState(id)
uint32_t id = popNumber(L);
if(id >= GAMESTATE_FIRST && id <= GAMESTATE_LAST)
{
Dispatcher::getInstance().addTask(createTask(
boost::bind(&Game::setGameState, &g_game, (GameState_t)id)));
lua_pushboolean(L, true);
}
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaDoCreatureExecuteTalkAction(lua_State* L)
{
//doCreatureExecuteTalkAction(cid, text[, ignoreAccess = false[, channelId = CHANNEL_DEFAULT]])
uint32_t params = lua_gettop(L), channelId = CHANNEL_DEFAULT;
if(params > 3)
channelId = popNumber(L);
bool ignoreAccess = false;
if(params > 2)
ignoreAccess = popNumber(L);
std::string text = popString(L);
ScriptEnviroment* env = getEnv();
if(Creature* creature = env->getCreatureByUID(popNumber(L)))
lua_pushboolean(L, g_talkActions->onPlayerSay(creature, channelId, text, ignoreAccess));
else
{
errorEx(getError(LUA_ERROR_CREATURE_NOT_FOUND));
lua_pushboolean(L, false);
}
return 1;
}
int32_t LuaInterface::luaDoExecuteRaid(lua_State* L)
{
//doExecuteRaid(name)
std::string raidName = popString(L);
if(Raids::getInstance()->getRunning())
{
lua_pushboolean(L, false);
return 1;
}
Raid* raid = Raids::getInstance()->getRaidByName(raidName);
if(!raid || !raid->isLoaded())
{
errorEx("Raid with name " + raidName + " does not exists.");
lua_pushboolean(L, false);
return 1;
}
lua_pushboolean(L, raid->startRaid());
return 1;
}
int32_t LuaInterface::luaDoReloadInfo(lua_State* L)
{
//doReloadInfo(id[, cid])
uint32_t cid = 0;
if(lua_gettop(L) > 1)
cid = popNumber(L);
uint32_t id = popNumber(L);
if(id >= RELOAD_FIRST && id <= RELOAD_LAST)
{
// we're passing it to scheduler since talkactions reload will
// re-init our lua state and crash due to unfinished call
Scheduler::getInstance().addEvent(createSchedulerTask(SCHEDULER_MINTICKS,
boost::bind(&Game::reloadInfo, &g_game, (ReloadInfo_t)id, cid)));
lua_pushboolean(L, true);
}
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaDoSaveServer(lua_State* L)
{
//doSaveServer([shallow = false])
bool shallow = false;
if(lua_gettop(L) > 0)
shallow = popNumber(L);
Dispatcher::getInstance().addTask(createTask(boost::bind(&Game::saveGameState, &g_game, shallow)));
lua_pushnil(L);
return 1;
}
int32_t LuaInterface::luaDoCleanHouse(lua_State* L)
{
//doCleanHouse(houseId)
uint32_t houseId = popNumber(L);
if(House* house = Houses::getInstance()->getHouse(houseId))
{
house->clean();
lua_pushboolean(L, true);
}
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaDoCleanMap(lua_State* L)
{
//doCleanMap()
uint32_t count = 0;
g_game.cleanMapEx(count);
lua_pushnumber(L, count);
return 1;
}
int32_t LuaInterface::luaDoRefreshMap(lua_State* L)
{
//doRefreshMap()
g_game.proceduralRefresh();
lua_pushnil(L);
return 1;
}
int32_t LuaInterface::luaDoUpdateHouseAuctions(lua_State* L)
{
//doUpdateHouseAuctions()
lua_pushboolean(L, IOMapSerialize::getInstance()->updateAuctions());
return 1;
}
int32_t LuaInterface::luaGetItemIdByName(lua_State* L)
{
//getItemIdByName(name[, displayError = true])
bool displayError = true;
if(lua_gettop(L) >= 2)
displayError = popNumber(L);
int32_t itemId = Item::items.getItemIdByName(popString(L));
if(itemId == -1)
{
if(displayError)
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
}
else
lua_pushnumber(L, itemId);
return 1;
}
int32_t LuaInterface::luaGetItemInfo(lua_State* L)
{
//getItemInfo(itemid)
const ItemType* item;
if(!(item = Item::items.getElement(popNumber(L))))
{
lua_pushboolean(L, false);
return 1;
}
lua_newtable(L);
setFieldBool(L, "stopTime", item->stopTime);
setFieldBool(L, "showCount", item->showCount);
setFieldBool(L, "stackable", item->stackable);
setFieldBool(L, "showDuration", item->showDuration);
setFieldBool(L, "showCharges", item->showCharges);
setFieldBool(L, "showAttributes", item->showAttributes);
setFieldBool(L, "distRead", item->allowDistRead);
setFieldBool(L, "readable", item->canReadText);
setFieldBool(L, "writable", item->canWriteText);
setFieldBool(L, "forceSerialize", item->forceSerialize);
setFieldBool(L, "vertical", item->isVertical);
setFieldBool(L, "horizontal", item->isHorizontal);
setFieldBool(L, "hangable", item->isHangable);
setFieldBool(L, "usable", item->useable);
setFieldBool(L, "movable", item->moveable);
setFieldBool(L, "pickupable", item->pickupable);
setFieldBool(L, "rotable", item->rotable);
setFieldBool(L, "replacable", item->replaceable);
setFieldBool(L, "hasHeight", item->hasHeight);
setFieldBool(L, "blockSolid", item->blockSolid);
setFieldBool(L, "blockPickupable", item->blockPickupable);
setFieldBool(L, "blockProjectile", item->blockProjectile);
setFieldBool(L, "blockPathing", item->blockPathFind);
setFieldBool(L, "allowPickupable", item->allowPickupable);
setFieldBool(L, "alwaysOnTop", item->alwaysOnTop);
createTable(L, "floorChange");
for(int32_t i = CHANGE_FIRST; i <= CHANGE_LAST; ++i)
{
lua_pushnumber(L, i);
lua_pushboolean(L, item->floorChange[i - 1]);
pushTable(L);
}
pushTable(L);
setField(L, "magicEffect", (int32_t)item->magicEffect);
setField(L, "fluidSource", (int32_t)item->fluidSource);
setField(L, "weaponType", (int32_t)item->weaponType);
setField(L, "bedPartnerDirection", (int32_t)item->bedPartnerDir);
setField(L, "ammoAction", (int32_t)item->ammoAction);
setField(L, "combatType", (int32_t)item->combatType);
setField(L, "corpseType", (int32_t)item->corpseType);
setField(L, "shootType", (int32_t)item->shootType);
setField(L, "ammoType", (int32_t)item->ammoType);
createTable(L, "transformUseTo");
setField(L, "female", item->transformUseTo[PLAYERSEX_FEMALE]);
setField(L, "male", item->transformUseTo[PLAYERSEX_MALE]);
pushTable(L);
setField(L, "transformToFree", item->transformToFree);
setField(L, "transformEquipTo", item->transformEquipTo);
setField(L, "transformDeEquipTo", item->transformDeEquipTo);
setField(L, "clientId", item->clientId);
setField(L, "maxItems", item->maxItems);
setField(L, "slotPosition", item->slotPosition);
setField(L, "wieldPosition", item->wieldPosition);
setField(L, "speed", item->speed);
setField(L, "maxTextLength", item->maxTextLen);
setField(L, "writeOnceItemId", item->writeOnceItemId);
setField(L, "attack", item->attack);
setField(L, "extraAttack", item->extraAttack);
setField(L, "defense", item->defense);
setField(L, "extraDefense", item->extraDefense);
setField(L, "armor", item->armor);
setField(L, "breakChance", item->breakChance);
setField(L, "hitChance", item->hitChance);
setField(L, "maxHitChance", item->maxHitChance);
setField(L, "runeLevel", item->runeLevel);
setField(L, "runeMagicLevel", item->runeMagLevel);
setField(L, "lightLevel", item->lightLevel);
setField(L, "lightColor", item->lightColor);
setField(L, "decayTo", item->decayTo);
setField(L, "rotateTo", item->rotateTo);
setField(L, "alwaysOnTopOrder", item->alwaysOnTopOrder);
setField(L, "shootRange", item->shootRange);
setField(L, "charges", item->charges);
setField(L, "decayTime", item->decayTime);
setField(L, "attackSpeed", item->attackSpeed);
setField(L, "wieldInfo", item->wieldInfo);
setField(L, "minRequiredLevel", item->minReqLevel);
setField(L, "minRequiredMagicLevel", item->minReqMagicLevel);
setField(L, "worth", item->worth);
setField(L, "levelDoor", item->levelDoor);
setField(L, "name", item->name.c_str());
setField(L, "plural", item->pluralName.c_str());
setField(L, "article", item->article.c_str());
setField(L, "description", item->description.c_str());
setField(L, "runeSpellName", item->runeSpellName.c_str());
setField(L, "vocationString", item->vocationString.c_str());
createTable(L, "abilities");
setFieldBool(L, "manaShield", item->abilities.manaShield);
setFieldBool(L, "invisible", item->abilities.invisible);
setFieldBool(L, "regeneration", item->abilities.regeneration);
setFieldBool(L, "preventLoss", item->abilities.preventLoss);
setFieldBool(L, "preventDrop", item->abilities.preventDrop);
setField(L, "elementType", (int32_t)item->abilities.elementType);
setField(L, "elementDamage", item->abilities.elementDamage);
setField(L, "speed", item->abilities.speed);
setField(L, "healthGain", item->abilities.healthGain);
setField(L, "healthTicks", item->abilities.healthTicks);
setField(L, "manaGain", item->abilities.manaGain);
setField(L, "manaTicks", item->abilities.manaTicks);
setField(L, "conditionSuppressions", item->abilities.conditionSuppressions);
//TODO: absorb, increment, reflect, skills, skillsPercent, stats, statsPercent
pushTable(L);
setField(L, "group", (int32_t)item->group);
setField(L, "type", (int32_t)item->type);
setFieldFloat(L, "weight", item->weight);
return 1;
}
int32_t LuaInterface::luaGetItemAttribute(lua_State* L)
{
//getItemAttribute(uid, key)
std::string key = popString(L);
ScriptEnviroment* env = getEnv();
Item* item = env->getItemByUID(popNumber(L));
if(!item)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushnil(L);
return 1;
}
boost::any value = item->getAttribute(key);
if(value.empty())
lua_pushnil(L);
else if(value.type() == typeid(std::string))
lua_pushstring(L, boost::any_cast<std::string>(value).c_str());
else if(value.type() == typeid(int32_t))
lua_pushnumber(L, boost::any_cast<int32_t>(value));
else if(value.type() == typeid(float))
lua_pushnumber(L, boost::any_cast<float>(value));
else if(value.type() == typeid(bool))
lua_pushboolean(L, boost::any_cast<bool>(value));
else
lua_pushnil(L);
return 1;
}
int32_t LuaInterface::luaDoItemSetAttribute(lua_State* L)
{
//doItemSetAttribute(uid, key, value)
boost::any value;
if(lua_isnumber(L, -1))
{
float tmp = popFloatNumber(L);
if(std::floor(tmp) < tmp)
value = tmp;
else
value = (int32_t)tmp;
}
else if(lua_isboolean(L, -1))
value = popBoolean(L);
else if(lua_isstring(L, -1))
value = popString(L);
else
{
lua_pop(L, 1);
errorEx("Invalid data type");
lua_pushboolean(L, false);
return 1;
}
std::string key = popString(L);
ScriptEnviroment* env = getEnv();
Item* item = env->getItemByUID(popNumber(L));
if(!item)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
if(value.type() == typeid(int32_t))
{
if(key == "uid")
{
int32_t tmp = boost::any_cast<int32_t>(value);
if(tmp < 1000 || tmp > 0xFFFF)
{
errorEx("Value for protected key "uid" must be in range of 1000 to 65535");
lua_pushboolean(L, false);
return 1;
}
item->setUniqueId(tmp);
}
else if(key == "aid")
item->setActionId(boost::any_cast<int32_t>(value));
else
item->setAttribute(key, boost::any_cast<int32_t>(value));
}
else
item->setAttribute(key, value);
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaDoItemEraseAttribute(lua_State* L)
{
//doItemEraseAttribute(uid, key)
std::string key = popString(L);
ScriptEnviroment* env = getEnv();
Item* item = env->getItemByUID(popNumber(L));
if(!item)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
bool ret = true;
if(key == "uid")
{
errorEx("Attempt to erase protected key "uid".");
ret = false;
}
else if(key != "aid")
item->eraseAttribute(key);
else
item->resetActionId();
lua_pushboolean(L, ret);
return 1;
}
int32_t LuaInterface::luaGetItemWeight(lua_State* L)
{
//getItemWeight(itemid[, precise = true])
bool precise = true;
if(lua_gettop(L) > 2)
precise = popNumber(L);
ScriptEnviroment* env = getEnv();
Item* item = env->getItemByUID(popNumber(L));
if(!item)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
double weight = item->getWeight();
if(precise)
{
std::stringstream ws;
ws << std::fixed << std::setprecision(2) << weight;
weight = atof(ws.str().c_str());
}
lua_pushnumber(L, weight);
return 1;
}
int32_t LuaInterface::luaGetItemParent(lua_State* L)
{
//getItemParent(uid)
ScriptEnviroment* env = getEnv();
Item* item = env->getItemByUID(popNumber(L));
if(!item)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushnil(L);
return 1;
}
Item* container = item->getParent()->getItem();
pushThing(L, container, env->addThing(container));
return 1;
}
int32_t LuaInterface::luaHasItemProperty(lua_State* L)
{
//hasItemProperty(uid, prop)
uint32_t prop = popNumber(L);
ScriptEnviroment* env = getEnv();
Item* item = env->getItemByUID(popNumber(L));
if(!item)
{
errorEx(getError(LUA_ERROR_ITEM_NOT_FOUND));
lua_pushboolean(L, false);
return 1;
}
//Check if the item is a tile, so we can get more accurate properties
bool tmp = item->hasProperty((ITEMPROPERTY)prop);
if(item->getTile() && item->getTile()->ground == item)
tmp = item->getTile()->hasProperty((ITEMPROPERTY)prop);
lua_pushboolean(L, tmp);
return 1;
}
int32_t LuaInterface::luaIsIpBanished(lua_State* L)
{
//isIpBanished(ip[, mask])
uint32_t mask = 0xFFFFFFFF;
if(lua_gettop(L) > 1)
mask = popNumber(L);
lua_pushboolean(L, IOBan::getInstance()->isIpBanished((uint32_t)popNumber(L), mask));
return 1;
}
int32_t LuaInterface::luaIsPlayerBanished(lua_State* L)
{
//isPlayerBanished(name/guid, type)
PlayerBan_t type = (PlayerBan_t)popNumber(L);
if(lua_isnumber(L, -1))
lua_pushboolean(L, IOBan::getInstance()->isPlayerBanished((uint32_t)popNumber(L), type));
else
lua_pushboolean(L, IOBan::getInstance()->isPlayerBanished(popString(L), type));
return 1;
}
int32_t LuaInterface::luaIsAccountBanished(lua_State* L)
{
//isAccountBanished(accountId[, playerId])
uint32_t playerId = 0;
if(lua_gettop(L) > 1)
playerId = popNumber(L);
lua_pushboolean(L, IOBan::getInstance()->isAccountBanished((uint32_t)popNumber(L), playerId));
return 1;
}
int32_t LuaInterface::luaDoAddIpBanishment(lua_State* L)
{
//doAddIpBanishment(ip[, mask[, length[, reason[, comment[, admin[, statement]]]]]])
uint32_t admin = 0, reason = 21, mask = 0xFFFFFFFF, params = lua_gettop(L);
int64_t length = time(NULL) + g_config.getNumber(ConfigManager::IPBANISHMENT_LENGTH);
std::string statement, comment;
if(params > 6)
statement = popString(L);
if(params > 5)
admin = popNumber(L);
if(params > 4)
comment = popString(L);
if(params > 3)
reason = popNumber(L);
if(params > 2)
length = popNumber(L);
if(params > 1)
mask = popNumber(L);
lua_pushboolean(L, IOBan::getInstance()->addIpBanishment((uint32_t)popNumber(L),
length, reason, comment, admin, mask, statement));
return 1;
}
int32_t LuaInterface::luaDoAddPlayerBanishment(lua_State* L)
{
//doAddPlayerBanishment(name/guid[, type[, length[, reason[, action[, comment[, admin[, statement]]]]]]])
uint32_t admin = 0, reason = 21, params = lua_gettop(L);
int64_t length = -1;
std::string statement, comment;
ViolationAction_t action = ACTION_NAMELOCK;
PlayerBan_t type = PLAYERBAN_LOCK;
if(params > 7)
statement = popString(L);
if(params > 6)
admin = popNumber(L);
if(params > 5)
comment = popString(L);
if(params > 4)
action = (ViolationAction_t)popNumber(L);
if(params > 3)
reason = popNumber(L);
if(params > 2)
length = popNumber(L);
if(params > 1)
type = (PlayerBan_t)popNumber(L);
if(lua_isnumber(L, -1))
lua_pushboolean(L, IOBan::getInstance()->addPlayerBanishment((uint32_t)popNumber(L),
length, reason, action, comment, admin, type, statement));
else
lua_pushboolean(L, IOBan::getInstance()->addPlayerBanishment(popString(L),
length, reason, action, comment, admin, type, statement));
return 1;
}
int32_t LuaInterface::luaDoAddAccountBanishment(lua_State* L)
{
//doAddAccountBanishment(accountId[, playerId[, length[, reason[, action[, comment[, admin[, statement]]]]]]])
uint32_t admin = 0, reason = 21, playerId = 0, params = lua_gettop(L);
int64_t length = time(NULL) + g_config.getNumber(ConfigManager::BAN_LENGTH);
std::string statement, comment;
ViolationAction_t action = ACTION_BANISHMENT;
if(params > 7)
statement = popString(L);
if(params > 6)
admin = popNumber(L);
if(params > 5)
comment = popString(L);
if(params > 4)
action = (ViolationAction_t)popNumber(L);
if(params > 3)
reason = popNumber(L);
if(params > 2)
length = popNumber(L);
if(params > 1)
playerId = popNumber(L);
lua_pushboolean(L, IOBan::getInstance()->addAccountBanishment((uint32_t)popNumber(L),
length, reason, action, comment, admin, playerId, statement));
return 1;
}
int32_t LuaInterface::luaDoAddNotation(lua_State* L)
{
//doAddNotation(accountId[, playerId[, reason[, comment[, admin[, statement]]]]]])
uint32_t admin = 0, reason = 21, playerId = 0, params = lua_gettop(L);
std::string statement, comment;
if(params > 5)
statement = popString(L);
if(params > 4)
admin = popNumber(L);
if(params > 3)
comment = popString(L);
if(params > 2)
reason = popNumber(L);
if(params > 1)
playerId = popNumber(L);
lua_pushboolean(L, IOBan::getInstance()->addNotation((uint32_t)popNumber(L),
reason, comment, admin, playerId, statement));
return 1;
}
int32_t LuaInterface::luaDoAddStatement(lua_State* L)
{
//doAddStatement(name/guid[, channelId[, reason[, comment[, admin[, statement]]]]]])
uint32_t admin = 0, reason = 21, params = lua_gettop(L);
int16_t channelId = -1;
std::string statement, comment;
if(params > 5)
statement = popString(L);
if(params > 4)
admin = popNumber(L);
if(params > 3)
comment = popString(L);
if(params > 2)
reason = popNumber(L);
if(params > 1)
channelId = popNumber(L);
if(lua_isnumber(L, -1))
lua_pushboolean(L, IOBan::getInstance()->addStatement((uint32_t)popNumber(L),
reason, comment, admin, channelId, statement));
else
lua_pushboolean(L, IOBan::getInstance()->addStatement(popString(L),
reason, comment, admin, channelId, statement));
return 1;
}
int32_t LuaInterface::luaDoRemoveIpBanishment(lua_State* L)
{
//doRemoveIpBanishment(ip[, mask])
uint32_t mask = 0xFFFFFFFF;
if(lua_gettop(L) > 1)
mask = popNumber(L);
lua_pushboolean(L, IOBan::getInstance()->removeIpBanishment(
(uint32_t)popNumber(L), mask));
return 1;
}
int32_t LuaInterface::luaDoRemovePlayerBanishment(lua_State* L)
{
//doRemovePlayerBanishment(name/guid, type)
PlayerBan_t type = (PlayerBan_t)popNumber(L);
if(lua_isnumber(L, -1))
lua_pushboolean(L, IOBan::getInstance()->removePlayerBanishment((uint32_t)popNumber(L), type));
else
lua_pushboolean(L, IOBan::getInstance()->removePlayerBanishment(popString(L), type));
return 1;
}
int32_t LuaInterface::luaDoRemoveAccountBanishment(lua_State* L)
{
//doRemoveAccountBanishment(accountId[, playerId])
uint32_t playerId = 0;
if(lua_gettop(L) > 1)
playerId = popNumber(L);
lua_pushboolean(L, IOBan::getInstance()->removeAccountBanishment((uint32_t)popNumber(L), playerId));
return 1;
}
int32_t LuaInterface::luaDoRemoveNotations(lua_State* L)
{
//doRemoveNotations(accountId[, playerId])
uint32_t playerId = 0;
if(lua_gettop(L) > 1)
playerId = popNumber(L);
lua_pushboolean(L, IOBan::getInstance()->removeNotations((uint32_t)popNumber(L), playerId));
return 1;
}
int32_t LuaInterface::luaDoRemoveStatements(lua_State* L)
{
//doRemoveStatements(name/guid[, channelId])
int16_t channelId = -1;
if(lua_gettop(L) > 1)
channelId = popNumber(L);
if(lua_isnumber(L, -1))
lua_pushboolean(L, IOBan::getInstance()->removeStatements((uint32_t)popNumber(L), channelId));
else
lua_pushboolean(L, IOBan::getInstance()->removeStatements(popString(L), channelId));
return 1;
}
int32_t LuaInterface::luaGetNotationsCount(lua_State* L)
{
//getNotationsCount(accountId[, playerId])
uint32_t playerId = 0;
if(lua_gettop(L) > 1)
playerId = popNumber(L);
lua_pushnumber(L, IOBan::getInstance()->getNotationsCount((uint32_t)popNumber(L), playerId));
return 1;
}
int32_t LuaInterface::luaGetStatementsCount(lua_State* L)
{
//getStatementsCount(name/guid[, channelId])
int16_t channelId = -1;
if(lua_gettop(L) > 1)
channelId = popNumber(L);
if(lua_isnumber(L, -1))
lua_pushnumber(L, IOBan::getInstance()->getStatementsCount((uint32_t)popNumber(L), channelId));
else
lua_pushnumber(L, IOBan::getInstance()->getStatementsCount(popString(L), channelId));
return 1;
}
int32_t LuaInterface::luaGetBanData(lua_State* L)
{
//getBanData(value[, type[, param]])
Ban tmp;
uint32_t params = lua_gettop(L);
if(params > 2)
tmp.param = popNumber(L);
if(params > 1)
tmp.type = (Ban_t)popNumber(L);
tmp.value = popNumber(L);
if(!IOBan::getInstance()->getData(tmp))
{
lua_pushboolean(L, false);
return 1;
}
lua_newtable(L);
setField(L, "id", tmp.id);
setField(L, "type", tmp.type);
setField(L, "value", tmp.value);
setField(L, "param", tmp.param);
setField(L, "added", tmp.added);
setField(L, "expires", tmp.expires);
setField(L, "adminId", tmp.adminId);
setField(L, "reason", tmp.reason);
setField(L, "action", tmp.action);
setField(L, "comment", tmp.comment);
setField(L, "statement", tmp.statement);
return 1;
}
int32_t LuaInterface::luaGetBanReason(lua_State* L)
{
//getBanReason(id)
lua_pushstring(L, getReason((ViolationAction_t)popNumber(L)).c_str());
return 1;
}
int32_t LuaInterface::luaGetBanAction(lua_State* L)
{
//getBanAction(id[, ipBanishment = false])
bool ipBanishment = false;
if(lua_gettop(L) > 1)
ipBanishment = popNumber(L);
lua_pushstring(L, getAction((ViolationAction_t)popNumber(L), ipBanishment).c_str());
return 1;
}
int32_t LuaInterface::luaGetBanList(lua_State* L)
{
//getBanList(type[, value[, param]])
int32_t param = 0, params = lua_gettop(L);
if(params > 2)
param = popNumber(L);
uint32_t value = 0;
if(params > 1)
value = popNumber(L);
BansVec bans = IOBan::getInstance()->getList((Ban_t)popNumber(L), value, param);
BansVec::const_iterator it = bans.begin();
lua_newtable(L);
for(uint32_t i = 1; it != bans.end(); ++it, ++i)
{
createTable(L, i);
setField(L, "id", it->id);
setField(L, "type", it->type);
setField(L, "value", it->value);
setField(L, "param", it->param);
setField(L, "added", it->added);
setField(L, "expires", it->expires);
setField(L, "adminId", it->adminId);
setField(L, "reason", it->reason);
setField(L, "action", it->action);
setField(L, "comment", it->comment);
setField(L, "statement", it->statement);
pushTable(L);
}
return 1;
}
int32_t LuaInterface::luaGetExperienceStage(lua_State* L)
{
//getExperienceStage(level[, divider])
double divider = 1.0f;
if(lua_gettop(L) > 1)
divider = popFloatNumber(L);
lua_pushnumber(L, g_game.getExperienceStage(popNumber(L), divider));
return 1;
}
int32_t LuaInterface::luaGetDataDir(lua_State* L)
{
//getDataDir()
lua_pushstring(L, getFilePath(FILE_TYPE_OTHER, "").c_str());
return 1;
}
int32_t LuaInterface::luaGetLogsDir(lua_State* L)
{
//getLogsDir()
lua_pushstring(L, getFilePath(FILE_TYPE_LOG, "").c_str());
return 1;
}
int32_t LuaInterface::luaGetConfigFile(lua_State* L)
{
//getConfigFile()
lua_pushstring(L, g_config.getString(ConfigManager::CONFIG_FILE).c_str());
return 1;
}
#ifdef __WAR_SYSTEM__
int32_t LuaInterface::luaDoGuildAddEnemy(lua_State* L)
{
//doGuildAddEnemy(guild, enemy, war, type)
War_t war;
war.type = (WarType_t)popNumber(L);
war.war = popNumber(L);
uint32_t enemy = popNumber(L), guild = popNumber(L), count = 0;
for(AutoList<Player>::iterator it = Player::autoList.begin(); it != Player::autoList.end(); ++it)
{
if(it->second->isRemoved() || it->second->getGuildId() != guild)
continue;
++count;
it->second->addEnemy(enemy, war);
g_game.updateCreatureEmblem(it->second);
}
lua_pushnumber(L, count);
return 1;
}
int32_t LuaInterface::luaDoGuildRemoveEnemy(lua_State* L)
{
//doGuildRemoveEnemy(guild, enemy)
uint32_t enemy = popNumber(L), guild = popNumber(L), count = 0;
for(AutoList<Player>::iterator it = Player::autoList.begin(); it != Player::autoList.end(); ++it)
{
if(it->second->isRemoved() || it->second->getGuildId() != guild)
continue;
++count;
it->second->removeEnemy(enemy);
g_game.updateCreatureEmblem(it->second);
}
lua_pushnumber(L, count);
return 1;
}
#endif
int32_t LuaInterface::luaGetConfigValue(lua_State* L)
{
//getConfigValue(key)
g_config.getValue(popString(L), L);
return 1;
}
int32_t LuaInterface::luaGetModList(lua_State* L)
{
//getModList()
ModMap::iterator it = ScriptManager::getInstance()->getFirstMod();
lua_newtable(L);
for(uint32_t i = 1; it != ScriptManager::getInstance()->getLastMod(); ++it, ++i)
{
createTable(L, i);
setField(L, "name", it->first);
setField(L, "description", it->second.description);
setField(L, "file", it->second.file);
setField(L, "version", it->second.version);
setField(L, "author", it->second.author);
setField(L, "contact", it->second.contact);
setFieldBool(L, "enabled", it->second.enabled);
pushTable(L);
}
return 1;
}
int32_t LuaInterface::luaL_loadmodlib(lua_State* L)
{
//loadmodlib(lib)
std::string name = asLowerCaseString(popString(L));
for(LibMap::iterator it = ScriptManager::getInstance()->getFirstLib();
it != ScriptManager::getInstance()->getLastLib(); ++it)
{
if(asLowerCaseString(it->first) != name)
continue;
luaL_loadstring(L, it->second.second.c_str());
lua_pushvalue(L, -1);
break;
}
return 1;
}
int32_t LuaInterface::luaL_domodlib(lua_State* L)
{
//domodlib(lib)
std::string name = asLowerCaseString(popString(L));
for(LibMap::iterator it = ScriptManager::getInstance()->getFirstLib();
it != ScriptManager::getInstance()->getLastLib(); ++it)
{
if(asLowerCaseString(it->first) != name)
continue;
bool ret = luaL_dostring(L, it->second.second.c_str());
if(ret)
error(NULL, popString(L));
lua_pushboolean(L, !ret);
break;
}
return 1;
}
int32_t LuaInterface::luaL_dodirectory(lua_State* L)
{
//dodirectory(dir)
std::string dir = popString(L);
if(!getEnv()->getInterface()->loadDirectory(dir, NULL))
{
errorEx("Failed to load directory " + dir + ".");
lua_pushboolean(L, false);
}
else
lua_pushboolean(L, true);
return 1;
}
int32_t LuaInterface::luaL_errors(lua_State* L)
{
//errors(var)
lua_pushboolean(L, getEnv()->getInterface()->m_errors);
getEnv()->getInterface()->m_errors = popNumber(L);
return 1;
}
#define EXPOSE_LOG(Name, Stream)
int32_t LuaInterface::luaStd##Name(lua_State* L)
{
StringVec data;
for(int32_t i = 0, params = lua_gettop(L); i < params; ++i)
data.push_back(popString(L));
for(StringVec::reverse_iterator it = data.rbegin(); it != data.rend(); ++it)
Stream << (*it) << std::endl;
lua_pushnumber(L, data.size());
return 1;
}
EXPOSE_LOG(Cout, std::cout)
EXPOSE_LOG(Clog, std::clog)
EXPOSE_LOG(Cerr, std::cerr)
#undef EXPOSE_LOG
int32_t LuaInterface::luaStdMD5(lua_State* L)
{
//std.md5(string[, upperCase = false])
bool upperCase = false;
if(lua_gettop(L) > 1)
upperCase = popNumber(L);
lua_pushstring(L, transformToMD5(popString(L), upperCase).c_str());
return 1;
}
int32_t LuaInterface::luaStdSHA1(lua_State* L)
{
//std.sha1(string[, upperCase = false])
bool upperCase = false;
if(lua_gettop(L) > 1)
upperCase = popNumber(L);
lua_pushstring(L, transformToSHA1(popString(L), upperCase).c_str());
return 1;
}
int32_t LuaInterface::luaStdSHA256(lua_State* L)
{
//std.sha256(string[, upperCase = false])
bool upperCase = false;
if(lua_gettop(L) > 1)
upperCase = popNumber(L);
lua_pushstring(L, transformToSHA256(popString(L), upperCase).c_str());
return 1;
}
int32_t LuaInterface::luaStdSHA512(lua_State* L)
{
//std.sha512(string[, upperCase = false])
bool upperCase = false;
if(lua_gettop(L) > 1)
upperCase = popNumber(L);
lua_pushstring(L, transformToSHA512(popString(L), upperCase).c_str());
return 1;
}
int32_t LuaInterface::luaStdVAHash(lua_State* L)
{
//std.vahash(string[, upperCase = false])
bool upperCase = false;
if(lua_gettop(L) > 1)
upperCase = popNumber(L);
lua_pushstring(L, transformToVAHash(popString(L), upperCase).c_str());
return 1;
}
int32_t LuaInterface::luaSystemTime(lua_State* L)
{
//os.mtime()
lua_pushnumber(L, OTSYS_TIME());
return 1;
}
int32_t LuaInterface::luaDatabaseExecute(lua_State* L)
{
//db.query(query)
DBQuery query; //lock mutex
lua_pushboolean(L, Database::getInstance()->query(popString(L)));
return 1;
}
int32_t LuaInterface::luaDatabaseStoreQuery(lua_State* L)
{
//db.storeQuery(query)
ScriptEnviroment* env = getEnv();
DBQuery query; //lock mutex
if(DBResult* res = Database::getInstance()->storeQuery(popString(L)))
lua_pushnumber(L, env->addResult(res));
else
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaDatabaseEscapeString(lua_State* L)
{
//db.escapeString(str)
DBQuery query; //lock mutex
lua_pushstring(L, Database::getInstance()->escapeString(popString(L)).c_str());
return 1;
}
int32_t LuaInterface::luaDatabaseEscapeBlob(lua_State* L)
{
//db.escapeBlob(s, length)
uint32_t length = popNumber(L);
DBQuery query; //lock mutex
lua_pushstring(L, Database::getInstance()->escapeBlob(popString(L).c_str(), length).c_str());
return 1;
}
int32_t LuaInterface::luaDatabaseLastInsertId(lua_State* L)
{
//db.lastInsertId()
DBQuery query; //lock mutex
lua_pushnumber(L, Database::getInstance()->getLastInsertId());
return 1;
}
int32_t LuaInterface::luaDatabaseStringComparer(lua_State* L)
{
//db.stringComparer()
lua_pushstring(L, Database::getInstance()->getStringComparer().c_str());
return 1;
}
int32_t LuaInterface::luaDatabaseUpdateLimiter(lua_State* L)
{
//db.updateLimiter()
lua_pushstring(L, Database::getInstance()->getUpdateLimiter().c_str());
return 1;
}
#define CHECK_RESULT()
if(!res)
{
lua_pushboolean(L, false);
return 1;
}
int32_t LuaInterface::luaResultGetDataInt(lua_State* L)
{
//result.getDataInt(res, s)
const std::string& s = popString(L);
ScriptEnviroment* env = getEnv();
DBResult* res = env->getResultByID(popNumber(L));
CHECK_RESULT()
lua_pushnumber(L, res->getDataInt(s));
return 1;
}
int32_t LuaInterface::luaResultGetDataLong(lua_State* L)
{
//result.getDataLong(res, s)
const std::string& s = popString(L);
ScriptEnviroment* env = getEnv();
DBResult* res = env->getResultByID(popNumber(L));
CHECK_RESULT()
lua_pushnumber(L, res->getDataLong(s));
return 1;
}
int32_t LuaInterface::luaResultGetDataString(lua_State* L)
{
//result.getDataString(res, s)
const std::string& s = popString(L);
ScriptEnviroment* env = getEnv();
DBResult* res = env->getResultByID(popNumber(L));
CHECK_RESULT()
lua_pushstring(L, res->getDataString(s).c_str());
return 1;
}
int32_t LuaInterface::luaResultGetDataStream(lua_State* L)
{
//result.getDataStream(res, s)
const std::string s = popString(L);
ScriptEnviroment* env = getEnv();
DBResult* res = env->getResultByID(popNumber(L));
CHECK_RESULT()
uint64_t length = 0;
lua_pushstring(L, res->getDataStream(s, length));
lua_pushnumber(L, length);
return 2;
}
int32_t LuaInterface::luaResultNext(lua_State* L)
{
//result.next(res)
ScriptEnviroment* env = getEnv();
DBResult* res = env->getResultByID(popNumber(L));
CHECK_RESULT()
lua_pushboolean(L, res->next());
return 1;
}
int32_t LuaInterface::luaResultFree(lua_State* L)
{
//result.free(res)
uint32_t rid = popNumber(L);
ScriptEnviroment* env = getEnv();
DBResult* res = env->getResultByID(rid);
CHECK_RESULT()
lua_pushboolean(L, env->removeResult(rid));
return 1;
}
#undef CHECK_RESULT
int32_t LuaInterface::luaBitNot(lua_State* L)
{
int32_t number = (int32_t)popNumber(L);
lua_pushnumber(L, ~number);
return 1;
}
int32_t LuaInterface::luaBitUNot(lua_State* L)
{
uint32_t number = (uint32_t)popNumber(L);
lua_pushnumber(L, ~number);
return 1;
}
#define MULTI_OPERATOR(type, name, op)
int32_t LuaInterface::luaBit##name(lua_State* L)
{
int32_t params = lua_gettop(L);
type value = (type)popNumber(L);
for(int32_t i = 2; i <= params; ++i)
value op popNumber(L);
lua_pushnumber(L, value);
return 1;
}
MULTI_OPERATOR(int32_t, And, &=)
MULTI_OPERATOR(int32_t, Or, |=)
MULTI_OPERATOR(int32_t, Xor, ^=)
MULTI_OPERATOR(uint32_t, UAnd, &=)
MULTI_OPERATOR(uint32_t, UOr, |=)
MULTI_OPERATOR(uint32_t, UXor, ^=)
#undef MULTI_OPERATOR
#define SHIFT_OPERATOR(type, name, op)
int32_t LuaInterface::luaBit##name(lua_State* L)
{
type v2 = (type)popNumber(L), v1 = (type)popNumber(L);
lua_pushnumber(L, (v1 op v2));
return 1;
}
SHIFT_OPERATOR(int32_t, LeftShift, <<)
SHIFT_OPERATOR(int32_t, RightShift, >>)
SHIFT_OPERATOR(uint32_t, ULeftShift, <<)
SHIFT_OPERATOR(uint32_t, URightShift, >>)
#undef SHIFT_OPERATOR
 

 

A propósito, coloquei o seu game.cpp mas não mudou extremamente nada.

A propósito, coloquei o seu game.cpp mas não mudou extremamente nada.

Link para o post
Compartilhar em outros sites

@Fabio Leandro

Poste o script da ultimate. 

VlVKQKC.png&key=d5c17620ae9567a1f898dd7a

 

 

 

  • 532144234_Logo_NTO_BLOOD_Finish_HIM_By_Antonio_Luckas(3).png.fd58d1af125a7e82ccdd751637e9ca93.png
Link para o post
Compartilhar em outros sites
local MIN = 1250
local MAX = 1600
local EMPTY_POTION = 7635

local exhaust = createConditionObject(CONDITION_EXHAUST)
setConditionParam(exhaust, CONDITION_PARAM_TICKS, (getConfigInfo('timeBetweenExActions') - 100))

function onUse(cid, item, fromPosition, itemEx, toPosition)
	if isPlayer(itemEx.uid) == FALSE then
		return FALSE
	end

	if hasCondition(cid, CONDITION_EXHAUST_HEAL) == TRUE then
		doPlayerSendDefaultCancel(cid, RETURNVALUE_YOUAREEXHAUSTED)
		return TRUE
	end

	if((not(isKnight(itemEx.uid)) or getPlayerLevel(itemEx.uid) < 130) and getPlayerCustomFlagValue(itemEx.uid, PlayerCustomFlag_GamemasterPrivileges) == FALSE) then
		doCreatureSay(itemEx.uid, "Only knights of level 130 or above may drink this fluid.", TALKTYPE_ORANGE_1)
		return TRUE
	end

	if doCreatureAddHealth(itemEx.uid, math.random(MIN, MAX)) == LUA_ERROR then
		return FALSE
	end

	doAddCondition(cid, exhaust)
	doSendMagicEffect(getThingPos(itemEx.uid), CONST_ME_MAGIC_BLUE)
	doCreatureSay(itemEx.uid, "Aaaah...", TALKTYPE_ORANGE_1)
	return TRUE
end

 

Mas eu queria que ficasse assim tanto nas potions quanto nas magias de healing, igual o global, n sei se vc ja viu

por exemplo, eu tenho 1000 de life, dai perco 300 de life fico com 700 né, daí uso uma exura gran que enxeria 500 igual ta ai, mas no jogo so mostra enchendo 300 que é o que falta pra completar 1000, entendeu?

Link para o post
Compartilhar em outros sites

@Fabio Leandro Desculpe, mas preciso da magia também.

VlVKQKC.png&key=d5c17620ae9567a1f898dd7a

 

 

 

  • 532144234_Logo_NTO_BLOOD_Finish_HIM_By_Antonio_Luckas(3).png.fd58d1af125a7e82ccdd751637e9ca93.png
Link para o post
Compartilhar em outros sites

@Fabio Leandro da spell que você quer que fique azul que nem o potion

VlVKQKC.png&key=d5c17620ae9567a1f898dd7a

 

 

 

  • 532144234_Logo_NTO_BLOOD_Finish_HIM_By_Antonio_Luckas(3).png.fd58d1af125a7e82ccdd751637e9ca93.png
Link para o post
Compartilhar em outros sites

ue eu qero de todas kk, quer que eu faça upload das pastas actions, e spells?

Link para o post
Compartilhar em outros sites

@Fabio Leandro só uma spell para usar e testar

VlVKQKC.png&key=d5c17620ae9567a1f898dd7a

 

 

 

  • 532144234_Logo_NTO_BLOOD_Finish_HIM_By_Antonio_Luckas(3).png.fd58d1af125a7e82ccdd751637e9ca93.png
Link para o post
Compartilhar em outros sites

vou mandar a INTENSE HEALING mais conhecida como EXURA GRAN rs

local combat = createCombatObject()
setCombatParam(combat, COMBAT_PARAM_TYPE, COMBAT_HEALING)
setCombatParam(combat, COMBAT_PARAM_EFFECT, CONST_ME_MAGIC_BLUE)
setCombatParam(combat, COMBAT_PARAM_AGGRESSIVE, false)
setCombatParam(combat, COMBAT_PARAM_DISPEL, CONDITION_PARALYZE)
setHealingFormula(combat, COMBAT_FORMULA_LEVELMAGIC, 5, 5, 5, 6)

function onCastSpell(cid, var)
	return doCombat(cid, combat, var)
end

 

Link para o post
Compartilhar em outros sites
Link para o post
Compartilhar em outros sites

Eu ja tinha visto esse topico obrigado, entenda que o impotante pra mim é o sistema de mostrar a numeração enxendo so o que falta do life, a cor azul é só um detalhe. 

Link para o post
Compartilhar em outros sites

Participe da conversa

Você pode postar agora e se cadastrar mais tarde. Se você tem uma conta, faça o login para postar com sua conta.

Visitante
Responder

×   Você colou conteúdo com formatação.   Remover formatação

  Apenas 75 emojis são permitidos.

×   Seu link foi automaticamente incorporado.   Mostrar como link

×   Seu conteúdo anterior foi restaurado.   Limpar o editor

×   Não é possível colar imagens diretamente. Carregar ou inserir imagens do URL.

  • Quem Está Navegando   0 membros estão online

    Nenhum usuário registrado visualizando esta página.

  • Conteúdo Similar

    • Por cloudrun2023
      CloudRun - Sua Melhor Escolha para Hospedagem de OTServer!
      Você está procurando a solução definitiva para hospedar seu OTServer com desempenho imbatível e segurança inigualável? Não procure mais! Apresentamos a CloudRun, sua parceira confiável em serviços de hospedagem na nuvem.
       
      Recursos Exclusivos - Proteção DDoS Avançada:
      Mantenha seu OTServer online e seguro com nossa robusta proteção DDoS, garantindo uma experiência de jogo ininterrupta para seus jogadores.
       
      Servidores Ryzen 7 Poderosos: Desfrute do poder de processamento superior dos servidores Ryzen 7 para garantir um desempenho excepcional do seu OTServer. Velocidade e estabilidade garantidas!
       
      Armazenamento NVMe de Alta Velocidade:
      Reduza o tempo de carregamento do jogo com nosso armazenamento NVMe ultrarrápido. Seus jogadores vão adorar a rapidez com que podem explorar o mundo do seu OTServer.
       
      Uplink de até 1GB:
      Oferecemos uma conexão de alta velocidade com até 1GB de largura de banda, garantindo uma experiência de jogo suave e livre de lag para todos os seus jogadores, mesmo nos momentos de pico.
       
      Suporte 24 Horas:
      Estamos sempre aqui para você! Nossa equipe de suporte está disponível 24 horas por dia, 7 dias por semana, para resolver qualquer problema ou responder a qualquer pergunta que você possa ter. Sua satisfação é a nossa prioridade.
       
      Fácil e Rápido de Começar:
      Configurar seu OTServer na CloudRun é simples e rápido. Concentre-se no desenvolvimento do seu jogo enquanto cuidamos da hospedagem.
       
      Entre em Contato Agora!
      Website: https://central.cloudrun.com.br/index.php?rp=/store/cloud-ryzen-brasil
      Email: [email protected]
      Telefone: (47) 99902-5147

      Não comprometa a qualidade da hospedagem do seu OTServer. Escolha a CloudRun e ofereça aos seus jogadores a melhor experiência de jogo possível. Visite nosso site hoje mesmo para conhecer nossos planos e começar!
       
      https://central.cloudrun.com.br/index.php?rp=/store/cloud-ryzen-brasil
       
      CloudRun - Onde a Velocidade Encontra a Confiabilidade!
       

    • Por FeeTads
      SALVE rapaziada do TK, esses dias vim pensando em novos scripts pro meu OT, e em um deles eu precisava que determinada area não contasse frag pro player que matasse outros, PORÉM eu precisava que os players que morressem nessa area ainda assim tivessem as penalidades da sua morte, procurei por ai, achei alguns scripts que apenas tiravam o SKULL e não realmente o FRAG do player.

      **script atualizado 22/10/2023** - melhorado e otimizado, levei o script pra puxar as infos por .lua / creatureScripts

      vou disponibilizar o code aqui, e o que fazer pra determinada area não contar frag.

      SOURCE OTX 2 / TFS 0.x, Funciona em TFS 1.x mudando as tags e ajeitando as sintaxes.

      vá em creatureevent.cpp

      procure por:
      else if(type == "preparedeath") _type = CREATURE_EVENT_PREPAREDEATH;
      Adiciona abaixo:
      else if(type == "nocountfrag") _type = CREATURE_EVENT_NOCOUNTFRAG;

      procure por:
      case CREATURE_EVENT_PREPAREDEATH: return "onPrepareDeath";  
      Adicione abaixo: 
      case CREATURE_EVENT_NOCOUNTFRAG: return "noCountFragArea";

      procure por:
      case CREATURE_EVENT_PREPAREDEATH: return "cid, deathList";
      Adicione abaixo:
      case CREATURE_EVENT_NOCOUNTFRAG: return "cid, target";

      agora no mesmo arquivo, vá até o final do arquivo e adicione essa função:
      uint32_t CreatureEvent::executeNoCountFragArea(Creature* creature, Creature* target) { //noCountFragArea(cid, target) if(m_interface->reserveEnv()) { ScriptEnviroment* env = m_interface->getEnv(); if(m_scripted == EVENT_SCRIPT_BUFFER) { env->setRealPos(creature->getPosition()); std::ostringstream scriptstream; scriptstream << "local cid = " << env->addThing(creature) << std::endl; scriptstream << "local target = " << env->addThing(target) << std::endl; if(m_scriptData) scriptstream << *m_scriptData; bool result = true; if(m_interface->loadBuffer(scriptstream.str())) { lua_State* L = m_interface->getState(); result = m_interface->getGlobalBool(L, "_result", true); } m_interface->releaseEnv(); return result; } else { #ifdef __DEBUG_LUASCRIPTS__ std::ostringstream desc; desc << creature->getName(); env->setEvent(desc.str()); #endif env->setScriptId(m_scriptId, m_interface); env->setRealPos(creature->getPosition()); lua_State* L = m_interface->getState(); m_interface->pushFunction(m_scriptId); lua_pushnumber(L, env->addThing(creature)); lua_pushnumber(L, env->addThing(target)); bool result = m_interface->callFunction(2); m_interface->releaseEnv(); return result; } } else { std::clog << "[Error - CreatureEvent::noCountFragArea] Call stack overflow." << std::endl; return 0; } }

      agora vá em creatureevent.h

      procure por:
      CREATURE_EVENT_PREPAREDEATH
      adicione abaixo:
      CREATURE_EVENT_NOCOUNTFRAG

      procure por:
      uint32_t executePrepareDeath(Creature* creature, DeathList deathList);
      Adicione abaixo:
      uint32_t executeNoCountFragArea(Creature* creature, Creature* target);

      agora vá em player.cpp

      procure por:
      bool Player::onKilledCreature(Creature* target, DeathEntry& entry)
      abaixo de:
      War_t enemy; if(targetPlayer->getEnemy(this, enemy)) { if(entry.isLast()) IOGuild::getInstance()->updateWar(enemy); entry.setWar(enemy); }
      Adicione o seguinte código:
      if (targetPlayer){ CreatureEventList killEvents = getCreatureEvents(CREATURE_EVENT_NOCOUNTFRAG); for (const auto &event : killEvents) { if (!event->executeNoCountFragArea(this, target)) { return true; } } }

      //

      Feito isso, tudo completo na sua source, agora é necessário adicionar o creaturescript dentro do servidor

      vá até creaturescripts/scripts
      crie um arquivo chamado, "noCountFragInArea.lua"
      e dentro dele cole o código:
       
      --[[ script feito por feetads / TibiaKing ]]-- --[[ discord: feetads / FeeTads#0246 ]]-- -- Add positions here for which you do not want to count frags local areas = { [1] = {from = {x = 91, y = 122, z = 7}, to = {x = 98, y = 127, z = 7}}, -- from = area superior esquerda / to = area inferior direita (formando um quadrado) } local onlyKillerInArea = false -- only killer need to be in area? function noCountFragArea(cid, target) if not isCreature(cid) or not isCreature(target) then return true end local posKiller = getPlayerPosition(cid) local posTarget = getPlayerPosition(target) for i = 1, #areas do local area = areas[i] if isInArea(posKiller, area.from, area.to) then if onlyKillerInArea then return false elseif isInArea(posTarget, area.from, area.to) then return false end end end return true end
      agora em creaturescripts.xml
      <event type="nocountfrag" name="fragarea" event="script" value="noCountFragInArea.lua"/>
      agora em creaturescripts/scripts/login.lua
       procure por OU semelhante a esse:
      registerCreatureEvent(cid, "AdvanceSave")
      e abaixo adicione:
      registerCreatureEvent(cid, "fragarea")

      //


      Agora tudo certo, quando quiser adiciona uma area que não pega frag, vá até o script e apenas coloque a area, igual o demonstrado no script

      Exemplo:
      local areas = { [1] = {from = {x = 91, y = 122, z = 7}, to = {x = 98, y = 127, z = 7}}, [2] = {from = {x = 1000, y = 1000, z = 7}, to = {x = 1100, y = 1100, z = 7}}, }
      assim somente colocando a area no script e abrindo o server ou dando /reload, já funcionará a area como não pegar frag.
      Esse sistema pode ser bom pra areas de pvp ativo, onde você ainda quer que o player que morrer perca os atributos, como se fosse uma morte normal, porém não conta frag pra quem matar.
      Bom pra sistemas tipo castle 48h (guild war), onde há diversas mortes e risco de pegar red, atrapalhando a war.

      Façam bom proveito dos scripts, e deixem os créditos no script rsrs

      **Eu fiz as alterações e o simples código por isso vim disponibilizar, créditos meus**
    • Por Muvuka
      Abri canal a força creaturescript acho que funcione no creaturescript cria script creaturescript
       
      <channel id="9" name="HELP" logged="yes"/>
      <channel id="12" name="Report Bugs" logged="yes"/>
      <channel id="13" name="Loot" logged="yes"/>
      <channel id="14" name="Report Character Rules Tibia Rules" logged="yes"/>
      <channel id="15" name="Death Channel"/>
      <channel id="6548" name="DexSoft" level="1"/>
      <channel id="7" name="Reports" logged="yes"/>
       
      antes de 
              if(lastLogin > 0) then adicione isso:
                      doPlayerOpenChannel(cid, CHANNEL_HELP) doPlayerOpenChannel(cid, 1,  2, 3) = 1,2 ,3 Channels, entendeu? NÃO FUNCIONA EU QUERO UM MEIO DE ABRI SEM USA A SOURCE
       
      EU NÃO CONSEGUI ABRI EU NÃO TENHO SOURCE
       
       
    • Por bolachapancao
      Rapaziada seguinte preciso de um script que ao utilizar uma alavanca para até 4 jogadores.
      Os jogadores serão teleportados para hunt durante uma hora e depois de uma hora os jogadores serão teleportados de volta para o templo.
       
      Observação: caso o jogador morra ou saia da hunt o evento hunt é cancelado.

      Estou a base canary
      GitHub - opentibiabr/canary: Canary Server 13.x for OpenTibia community.
       
    • Por RAJADAO
      .Qual servidor ou website você utiliza como base? 
      Sabrehaven 8.0
      Qual o motivo deste tópico? 
      Ajuda com novos efeitos
       
      Olá amigos, gostaria de ajuda para introduzir os seguintes efeitos no meu servidor (usando o Sabrehaven 8.0 como base), adicionei algumas runas novas (avalanche, icicle, míssil sagrado, stoneshower & Thunderstorm) e alguns novos feitiços (exevo mas san, exori san, exori tera, exori frigo, exevo gran mas frigo, exevo gran mas tera, exevo tera hur, exevo frigo hur) mas nenhum dos efeitos dessas magias parece existir no servidor, alguém tem um link para um tutorial ou algo assim para que eu possa fazer isso funcionar?
      Desculpe pelo mau inglês, sou brasileiro.

      Obrigado!


      AVALANCHE RUNE id:3161 \/
      (COMBAT_PARAM_TYPE, COMBAT_ICEDAMAGE)
      (COMBAT_PARAM_EFFECT, CONST_ME_ICEAREA)
      (COMBAT_PARAM_DISTANCEEFFECT, CONST_ANI_ICE)

      STONESHOWER RUNE id:3175 \/
      (COMBAT_PARAM_TYPE, COMBAT_EARTHDAMAGE)
      (COMBAT_PARAM_EFFECT, CONST_ME_STONES)
      (COMBAT_PARAM_DISTANCEEFFECT, CONST_ANI_EARTH)

      THUNDERSTORM RUNE id:3202 \/
      (COMBAT_PARAM_TYPE, COMBAT_ENERGYDAMAGE)
      (COMBAT_PARAM_EFFECT, CONST_ME_E NERGYHIT)
      (COMBAT_PARAM_DISTANCEEFFECT, CONST_ANI_ENERGYBALL)

      ICICLE RUNE id:3158 \/
      COMBAT_ICEDAMAGE
      CONST_ME_ICEAREA
      CONST_ANI_ICE

      SANTO MÍSSIL RUNA id:3182 \/
      (COMBAT_PARAM_TYPE, COMBAT_HOLYDAMAGE)
      (COMBAT_PARAM_EFFECT, CONST_ME_HOLYDAMAGE)
      (COMBAT_PARAM_EFFECT, CONST_ME_HOLYAREA)
      (COMBAT_PARAM_DISTANCEEFFECT, CONST_ANI_HOLY)

      CONST_ME_PLANTATTACK (exevo gran mas tera)
      CONST_ME_ICETORNADO (exevo gran mas frigo)
      CONST_ME_SMALLPLANTS (exevo tera hur)
      CONST_ME_ICEAREA (exevo frigo hur)
      CONST_ME_ICEATTACK (exori frigo)
      CONST_ME_CARNIPHILA (exori tera)

      EXORI SAN \/
      (COMBAT_PARAM_DISTANCEEFFECT, CONST_ANI_SMALLHOLY)
      CONST_ME_HOLYDAM IDADE

      EXEVO MAS SAN \/
      CONST_ME_HOLYAREA
×
×
  • Criar Novo...

Informação Importante

Confirmação de Termo