diff options
author | Thorbjørn Lindeijer <thorbjorn@lindeijer.nl> | 2012-03-05 20:53:18 +0100 |
---|---|---|
committer | Thorbjørn Lindeijer <thorbjorn@lindeijer.nl> | 2012-03-07 22:10:00 +0100 |
commit | 99227026607ecfaca1ff07705930a83d41c16042 (patch) | |
tree | 49153afd9f05d65c2102ff974f763342eb59684e | |
parent | b0225ccd7997e900d4276eacc1541ec80fadbb48 (diff) | |
download | manaserv-99227026607ecfaca1ff07705930a83d41c16042.tar.gz manaserv-99227026607ecfaca1ff07705930a83d41c16042.tar.bz2 manaserv-99227026607ecfaca1ff07705930a83d41c16042.tar.xz manaserv-99227026607ecfaca1ff07705930a83d41c16042.zip |
Added a function that returns the current map or raises an error
The new function 'checkCurrentMap' will raise an error when no current map has
been set, eliminating the need to do custom error handling all over the place.
This also fixes several functions that would otherwise have simply crashed
when there was no current map.
Also cleaned up some "empty string parameter" checks.
Reviewed-by: Erik Schilling
-rw-r--r-- | src/scripting/lua.cpp | 180 | ||||
-rw-r--r-- | src/scripting/luascript.cpp | 2 | ||||
-rw-r--r-- | src/scripting/luascript.h | 4 | ||||
-rw-r--r-- | src/scripting/luautil.cpp | 25 | ||||
-rw-r--r-- | src/scripting/luautil.h | 5 |
5 files changed, 80 insertions, 136 deletions
diff --git a/src/scripting/lua.cpp b/src/scripting/lua.cpp index 1c32c4fd..48404d55 100644 --- a/src/scripting/lua.cpp +++ b/src/scripting/lua.cpp @@ -66,15 +66,6 @@ extern "C" { * http://doc.manasource.org/scripting */ -static Script *getScript(lua_State *s) -{ - lua_pushlightuserdata(s, (void *)®istryKey); - lua_gettable(s, LUA_REGISTRYINDEX); - Script *script = static_cast<Script *>(lua_touserdata(s, -1)); - lua_pop(s, 1); - return script; -} - /** * mana.on_character_death( function(Character*) ): void @@ -345,15 +336,10 @@ static int npc_create(lua_State *s) const int y = luaL_checkint(s, 5); Script *t = getScript(s); + MapComposite *m = checkCurrentMap(s, t); + NPC *q = new NPC(name, id, t); q->setGender(getGender(gender)); - - MapComposite *m = t->getMap(); - if (!m) - { - raiseScriptError(s, "npc_create called outside a map."); - return 0; - } q->setMap(m); q->setPosition(Point(x, y)); GameState::enqueueInsert(q); @@ -435,20 +421,17 @@ static int chr_warp(lua_State *s) MapComposite *m; if (b) { - m = getScript(s)->getMap(); + m = checkCurrentMap(s); } else if (lua_isnumber(s, 2)) { m = MapManager::getMap(lua_tointeger(s, 2)); + luaL_argcheck(s, m, 2, "invalid map id"); } else { m = MapManager::getMap(lua_tostring(s, 2)); - } - if (!m) - { - raiseScriptError(s, "chr_warp called with a non-existing map."); - return 0; + luaL_argcheck(s, m, 2, "invalid map name"); } Map *map = m->getMap(); @@ -1008,11 +991,6 @@ static int being_damage(lua_State *s) { Being *being = checkBeing(s, 1); - if (!being) - { - raiseScriptError(s, "being_damage called with invalid victim"); - return 0; - } if (!being->canFight()) { raiseScriptError(s, "being_damage called with " @@ -1030,10 +1008,7 @@ static int being_damage(lua_State *s) if (lua_gettop(s) >= 7) { source = checkBeing(s, 7); - if (!source) - { - raiseScriptError(s, "being_damage called withd invalid source"); - } + if (!source->canFight()) { raiseScriptError(s, "being_damage called with " @@ -1250,13 +1225,7 @@ static int monster_create(lua_State *s) MonsterClass *monsterClass = checkMonsterClass(s, 1); const int x = luaL_checkint(s, 2); const int y = luaL_checkint(s, 3); - - MapComposite *m = getScript(s)->getMap(); - if (!m) - { - raiseScriptError(s, "monster_create called outside a map."); - return 0; - } + MapComposite *m = checkCurrentMap(s); Monster *q = new Monster(monsterClass); q->setMap(m); @@ -1323,13 +1292,10 @@ static int monster_remove(lua_State *s) static int chr_get_quest(lua_State *s) { Character *q = checkCharacter(s, 1); - const char *m = luaL_checkstring(s, 2); - if (m[0] == 0) - { - raiseScriptError(s, "chr_get_quest called with empty string."); - return 0; - } - std::string value, name = m; + const char *name = luaL_checkstring(s, 2); + luaL_argcheck(s, name[0] != 0, 2, "empty variable name"); + + std::string value; bool res = getQuestVar(q, name, value); if (res) { @@ -1347,14 +1313,11 @@ static int chr_get_quest(lua_State *s) */ static int getvar_map(lua_State *s) { - const char *m = luaL_checkstring(s, 1); - if (m[0] == 0) - { - raiseScriptError(s, "getvar_map called for unnamed variable."); - return 0; - } + const char *name = luaL_checkstring(s, 1); + luaL_argcheck(s, name[0] != 0, 1, "empty variable name"); - std::string value = getScript(s)->getMap()->getVariable(m); + MapComposite *map = checkCurrentMap(s); + std::string value = map->getVariable(name); lua_pushstring(s, value.c_str()); return 1; @@ -1366,16 +1329,12 @@ static int getvar_map(lua_State *s) */ static int setvar_map(lua_State *s) { - const char *m = luaL_checkstring(s, 1); - if (m[0] == 0) - { - raiseScriptError(s, "setvar_map called for unnamed variable."); - return 0; - } + const char *name = luaL_checkstring(s, 1); + const char *value = luaL_checkstring(s, 2); + luaL_argcheck(s, name[0] != 0, 1, "empty variable name"); - std::string key = m; - std::string value = lua_tostring(s, 2); - getScript(s)->getMap()->setVariable(key, value); + MapComposite *map = checkCurrentMap(s); + map->setVariable(name, value); return 0; } @@ -1386,15 +1345,10 @@ static int setvar_map(lua_State *s) */ static int getvar_world(lua_State *s) { - const char *m = luaL_checkstring(s, 1); - if (m[0] == 0) - { - raiseScriptError(s, "getvar_world called for unnamed variable."); - return 0; - } - - std::string value = GameState::getVariable(m); + const char *name = luaL_checkstring(s, 1); + luaL_argcheck(s, name[0] != 0, 1, "empty variable name"); + std::string value = GameState::getVariable(name); lua_pushstring(s, value.c_str()); return 1; } @@ -1405,17 +1359,11 @@ static int getvar_world(lua_State *s) */ static int setvar_world(lua_State *s) { - const char *m = luaL_checkstring(s, 1); - if (m[0] == 0) - { - raiseScriptError(s, "setvar_world called with unnamed variable."); - return 0; - } - - std::string key = lua_tostring(s, 1); - std::string value = lua_tostring(s, 2); - GameState::setVariable(key, value); + const char *name = luaL_checkstring(s, 1); + const char *value = luaL_checkstring(s, 2); + luaL_argcheck(s, name[0] != 0, 1, "empty variable name"); + GameState::setVariable(name, value); return 0; } @@ -1426,14 +1374,11 @@ static int setvar_world(lua_State *s) static int chr_set_quest(lua_State *s) { Character *q = checkCharacter(s, 1); - const char *m = luaL_checkstring(s, 2); - const char *n = luaL_checkstring(s, 3); - if (m[0] == 0 || strlen(m) == 0) - { - raiseScriptError(s, "chr_set_quest called with incorrect parameters."); - return 0; - } - setQuestVar(q, m, n); + const char *name = luaL_checkstring(s, 2); + const char *value = luaL_checkstring(s, 3); + luaL_argcheck(s, name[0] != 0, 2, "empty variable name"); + + setQuestVar(q, name, value); return 0; } @@ -1459,13 +1404,7 @@ static int trigger_create(lua_State *s) } Script *script = getScript(s); - MapComposite *m = script->getMap(); - - if (!m) - { - raiseScriptError(s, "trigger_create called for nonexistent a map."); - return 0; - } + MapComposite *m = checkCurrentMap(s, script); const bool once = lua_toboolean(s, 7); @@ -1523,7 +1462,7 @@ static int get_beings_in_circle(lua_State *s) r = luaL_checkint(s, 3); } - MapComposite *m = getScript(s)->getMap(); + MapComposite *m = checkCurrentMap(s); //create a lua table with the beings in the given area. lua_newtable(s); @@ -1561,7 +1500,7 @@ static int get_beings_in_rectangle(lua_State *s) const int w = luaL_checkint(s, 3); const int h = luaL_checkint(s, 4); - MapComposite *m = getScript(s)->getMap(); + MapComposite *m = checkCurrentMap(s); //create a lua table with the beings in the given area. lua_newtable(s); @@ -1618,19 +1557,18 @@ static int effect_create(lua_State *s) { const int id = luaL_checkint(s, 1); - MapComposite *m = getScript(s)->getMap(); - if (lua_isuserdata(s, 2)) { // being mode Being *b = checkBeing(s, 2); - Effects::show(id, m, b); + Effects::show(id, m, b->getMap()); } else { // positional mode int x = luaL_checkint(s, 2); int y = luaL_checkint(s, 3); + MapComposite *m = checkCurrentMap(s); Effects::show(id, m, Point(x, y)); } @@ -1912,14 +1850,8 @@ static int test_tableget(lua_State *s) */ static int get_map_id(lua_State *s) { - MapComposite *m = getScript(s)->getMap(); - if (!m) - { - raiseScriptError(s, "get_map_id called outside a map."); - return 0; - } - int id = m->getID(); - lua_pushinteger(s, id); + MapComposite *m = checkCurrentMap(s); + lua_pushinteger(s, m->getID()); return 1; } @@ -1930,17 +1862,10 @@ static int get_map_id(lua_State *s) static int get_map_property(lua_State *s) { const char *property = luaL_checkstring(s, 1); - MapComposite *m = getScript(s)->getMap(); - if (!m) - { - raiseScriptError(s, "get_map_property called outside a map."); - return 0; - } - Map *map = m->getMap(); - std::string value = map->getProperty(property); - const char *v = &value[0]; + Map *map = checkCurrentMap(s)->getMap(); - lua_pushstring(s, v); + std::string value = map->getProperty(property); + lua_pushstring(s, value.c_str()); return 1; } @@ -1952,14 +1877,7 @@ static int is_walkable(lua_State *s) { const int x = luaL_checkint(s, 1); const int y = luaL_checkint(s, 2); - - MapComposite *m = getScript(s)->getMap(); - if (!m) - { - raiseScriptError(s, "is_walkable called outside a map."); - return 0; - } - Map *map = m->getMap(); + Map *map = checkCurrentMap(s)->getMap(); // If the wanted warp place is unwalkable if (map->getWalk(x / map->getTileWidth(), y / map->getTileHeight())) @@ -1972,12 +1890,7 @@ static int is_walkable(lua_State *s) static int map_get_pvp(lua_State *s) { - MapComposite *m = getScript(s)->getMap(); - if (!m) - { - raiseScriptError(s, "map_get_pvp called outside a map."); - return 0; - } + MapComposite *m = checkCurrentMap(s); lua_pushinteger(s, m->getPvP()); return 1; } @@ -2002,11 +1915,10 @@ static int item_drop(lua_State *s) const int y = luaL_checkint(s, 2); ItemClass *ic = checkItemClass(s, 3); const int number = luaL_optint(s, 4, 1); + MapComposite *map = checkCurrentMap(s); Item *i = new Item(ic, number); - MapComposite *map = getScript(s)->getMap(); - i->setMap(map); Point pos(x, y); i->setPosition(pos); @@ -2098,7 +2010,7 @@ static int map_get_objects(lua_State *s) filter = luaL_checkstring(s, 1); } - MapComposite *m = getScript(s)->getMap(); + MapComposite *m = checkCurrentMap(s); const std::vector<MapObject*> &objects = m->getMap()->getObjects(); if (!filtered) diff --git a/src/scripting/luascript.cpp b/src/scripting/luascript.cpp index 231dece1..c9337d59 100644 --- a/src/scripting/luascript.cpp +++ b/src/scripting/luascript.cpp @@ -35,6 +35,8 @@ Script::Ref LuaScript::mPostReplyCallback; Script::Ref LuaScript::mDeathNotificationCallback; Script::Ref LuaScript::mRemoveNotificationCallback; +const char LuaScript::registryKey = 0; + LuaScript::~LuaScript() { lua_close(mState); diff --git a/src/scripting/luascript.h b/src/scripting/luascript.h index 6ea44bac..e26fd9ef 100644 --- a/src/scripting/luascript.h +++ b/src/scripting/luascript.h @@ -81,6 +81,8 @@ class LuaScript : public Script static void setRemoveNotificationCallback(Script *script) { script->assignCallback(mRemoveNotificationCallback); } + static const char registryKey; + private: lua_State *mState; int nbArgs; @@ -91,8 +93,6 @@ class LuaScript : public Script static Ref mRemoveNotificationCallback; }; -static char const registryKey = 0; - static Script *LuaFactory() { return new LuaScript(); diff --git a/src/scripting/luautil.cpp b/src/scripting/luautil.cpp index 217fc60c..0a49e834 100644 --- a/src/scripting/luautil.cpp +++ b/src/scripting/luautil.cpp @@ -29,6 +29,8 @@ #include "utils/logger.h" +#include "scripting/luascript.h" + void raiseScriptError(lua_State *s, const char *format, ...) { @@ -112,6 +114,16 @@ void UserDataCache::insert(lua_State *s, void *object) } +Script *getScript(lua_State *s) +{ + lua_pushlightuserdata(s, (void *)&LuaScript::registryKey); + lua_gettable(s, LUA_REGISTRYINDEX); + Script *script = static_cast<Script *>(lua_touserdata(s, -1)); + lua_pop(s, 1); + return script; +} + + /* Functions below are unsafe, as they assume the script has passed pointers to objects which have not yet been destroyed. If the script never keeps pointers around, there will be no problem. In order to be safe, the engine @@ -241,6 +253,19 @@ NPC *checkNPC(lua_State *s, int p) } +MapComposite *checkCurrentMap(lua_State *s, Script *script /* = 0 */) +{ + if (!script) + script = getScript(s); + + MapComposite *mapComposite = script->getMap(); + if (!mapComposite) + luaL_error(s, "no current map"); + + return mapComposite; +} + + void push(lua_State *s, int val) { lua_pushinteger(s, val); diff --git a/src/scripting/luautil.h b/src/scripting/luautil.h index 78930a66..2a255280 100644 --- a/src/scripting/luautil.h +++ b/src/scripting/luautil.h @@ -34,10 +34,12 @@ extern "C" { class Being; class Character; class ItemClass; +class MapComposite; class MapObject; class Monster; class MonsterClass; class NPC; +class Script; class StatusEffect; class Thing; @@ -149,6 +151,7 @@ typedef LuaUserData<MapObject> LuaMapObject; typedef LuaUserData<MonsterClass> LuaMonsterClass; typedef LuaUserData<StatusEffect> LuaStatusEffect; +Script * getScript(lua_State *s); Being * getBeing(lua_State *s, int p); Character * getCharacter(lua_State *s, int p); @@ -164,6 +167,8 @@ Monster * checkMonster(lua_State *s, int p); MonsterClass * checkMonsterClass(lua_State *s, int p); NPC * checkNPC(lua_State *s, int p); +MapComposite * checkCurrentMap(lua_State *s, Script *script = 0); + /* Polymorphic wrapper for pushing variables. Useful for templates.*/ |