#include "Base.h" #include "FileSystem.h" #include "ScriptController.h" #include "lua/lua_all_bindings.h" #define GENERATE_LUA_GET_POINTER(type, checkFunc) \ ScriptController* sc = Game::getInstance()->getScriptController(); \ /* Check that the parameter is the correct type. */ \ if (!lua_istable(sc->_lua, index)) \ { \ if (lua_islightuserdata(sc->_lua, index)) \ return (type*)lua_touserdata(sc->_lua, index); \ lua_pushfstring(sc->_lua, "Expected a " #type " pointer (an array represented as a Lua table), got '%s' instead.", \ luaL_typename(sc->_lua, index)); \ lua_error(sc->_lua); \ return NULL; \ } \ \ /* Get the size of the array. */ \ lua_len(sc->_lua, index); \ int size = luaL_checkint(sc->_lua, -1); \ if (size <= 0) \ return NULL; \ \ /* Create an array to store the values. */ \ type* values = (type*)malloc(sizeof(type)*size); \ \ /* Push the first key. */ \ lua_pushnil(sc->_lua); \ int i = 0; \ for (; lua_next(sc->_lua, index) != 0 && i < size; i++) \ { \ values[i] = (checkFunc(sc->_lua, -1)); \ \ /* Remove the value we just retrieved, but leave the key for the next iteration. */ \ lua_pop(sc->_lua, 1); \ } \ \ return values namespace gameplay { extern void splitURL(const std::string& url, std::string* file, std::string* id); void ScriptUtil::registerLibrary(const char* name, const luaL_Reg* functions) { ScriptController* sc = Game::getInstance()->getScriptController(); lua_newtable(sc->_lua); // Go through the list of functions and add them to the table. const luaL_Reg* iter = functions; for (; iter && iter->name; iter++) { lua_pushcfunction(sc->_lua, iter->func); lua_setfield(sc->_lua, -2, iter->name); } lua_setglobal(sc->_lua, name); } void ScriptUtil::registerConstantBool(std::string name, bool value, std::vector scopePath) { ScriptController* sc = Game::getInstance()->getScriptController(); // If the constant is within a scope, get the correct parent // table on the stack before setting its value. if (scopePath.size() > 0) { lua_getglobal(sc->_lua, scopePath[0].c_str()); for (unsigned int i = 1; i < scopePath.size(); i++) { lua_pushstring(sc->_lua, scopePath[i].c_str()); lua_gettable(sc->_lua, -2); } // Add the constant to the parent table. lua_pushboolean(sc->_lua, value); lua_setfield(sc->_lua, -2, name.c_str()); // Pop all the parent tables off the stack. int size = scopePath.size(); lua_pop(sc->_lua, size); } else { // TODO: Currently unsupported (we don't parse for this yet). // If the constant is global, add it to the global table. lua_pushboolean(sc->_lua, value); lua_pushvalue(sc->_lua, -1); lua_setglobal(sc->_lua, name.c_str()); } } void ScriptUtil::registerConstantNumber(std::string name, double value, std::vector scopePath) { ScriptController* sc = Game::getInstance()->getScriptController(); // If the constant is within a scope, get the correct parent // table on the stack before setting its value. if (scopePath.size() > 0) { lua_getglobal(sc->_lua, scopePath[0].c_str()); for (unsigned int i = 1; i < scopePath.size(); i++) { lua_pushstring(sc->_lua, scopePath[i].c_str()); lua_gettable(sc->_lua, -2); } // Add the constant to the parent table. lua_pushnumber(sc->_lua, value); lua_setfield(sc->_lua, -2, name.c_str()); // Pop all the parent tables off the stack. int size = scopePath.size(); lua_pop(sc->_lua, size); } else { // TODO: Currently unsupported (we don't parse for this yet). // If the constant is global, add it to the global table. lua_pushnumber(sc->_lua, value); lua_pushvalue(sc->_lua, -1); lua_setglobal(sc->_lua, name.c_str()); } } void ScriptUtil::registerConstantString(std::string name, std::string value, std::vector scopePath) { ScriptController* sc = Game::getInstance()->getScriptController(); // If the constant is within a scope, get the correct parent // table on the stack before setting its value. if (scopePath.size() > 0) { lua_getglobal(sc->_lua, scopePath[0].c_str()); for (unsigned int i = 1; i < scopePath.size(); i++) { lua_pushstring(sc->_lua, scopePath[i].c_str()); lua_gettable(sc->_lua, -2); } // Add the constant to the parent table. lua_pushstring(sc->_lua, value.c_str()); lua_setfield(sc->_lua, -2, name.c_str()); // Pop all the parent tables off the stack. int size = scopePath.size(); lua_pop(sc->_lua, size); } else { // TODO: Currently unsupported (we don't parse for this yet). // If the constant is global, add it to the global table. lua_pushstring(sc->_lua, value.c_str()); lua_pushvalue(sc->_lua, -1); lua_setglobal(sc->_lua, name.c_str()); } } void ScriptUtil::registerClass(const char* name, const luaL_Reg* members, lua_CFunction newFunction, lua_CFunction deleteFunction, const luaL_Reg* statics, std::vector scopePath) { ScriptController* sc = Game::getInstance()->getScriptController(); // If the type is an inner type, get the correct parent // table on the stack before creating the table for the class. if (scopePath.size() > 0) { std::string tablename = name; // Strip off the scope path part of the name. lua_getglobal(sc->_lua, scopePath[0].c_str()); std::size_t index = tablename.find(scopePath[0]); if (index != tablename.npos) tablename = tablename.substr(index + scopePath[0].size()); for (unsigned int i = 1; i < scopePath.size(); i++) { lua_pushstring(sc->_lua, scopePath[i].c_str()); lua_gettable(sc->_lua, -2); index = tablename.find(scopePath[i]); if (index != tablename.npos) tablename = tablename.substr(index + scopePath[i].size()); } lua_pushstring(sc->_lua, tablename.c_str()); lua_newtable(sc->_lua); } else { // If the type is not an inner type, set it as a global table. lua_newtable(sc->_lua); lua_pushvalue(sc->_lua, -1); lua_setglobal(sc->_lua, name); } // Create the metatable and populate it with the member functions. lua_pushliteral(sc->_lua, "__metatable"); luaL_newmetatable(sc->_lua, name); if (members) luaL_setfuncs(sc->_lua, members, 0); lua_pushstring(sc->_lua, "__index"); lua_pushvalue(sc->_lua, -2); lua_settable(sc->_lua, -3); // Add the delete function if it was specified. if (deleteFunction) { lua_pushstring(sc->_lua, "__gc"); lua_pushcfunction(sc->_lua, deleteFunction); lua_settable(sc->_lua, -3); } // Set the metatable on the main table. lua_settable(sc->_lua, -3); // Populate the main table with the static functions. if (statics) luaL_setfuncs(sc->_lua, statics, 0); // Set the new function(s) for the class. if (newFunction) { lua_pushliteral(sc->_lua, "new"); lua_pushcfunction(sc->_lua, newFunction); lua_settable(sc->_lua, -3); } // Set the table we just created within the correct parent table. if (scopePath.size() > 0) { lua_settable(sc->_lua, -3); // Pop all the parent tables off the stack. int size = scopePath.size(); lua_pop(sc->_lua, size); } else { // Pop the main table off the stack. lua_pop(sc->_lua, 1); } } void ScriptUtil::registerFunction(const char* luaFunction, lua_CFunction cppFunction) { lua_pushcfunction(Game::getInstance()->getScriptController()->_lua, cppFunction); lua_setglobal(Game::getInstance()->getScriptController()->_lua, luaFunction); } void ScriptUtil::setGlobalHierarchyPair(std::string base, std::string derived) { Game::getInstance()->getScriptController()->_hierarchy[base].push_back(derived); } void ScriptUtil::addStringFromEnumConversionFunction(luaStringEnumConversionFunction stringFromEnum) { Game::getInstance()->getScriptController()->_stringFromEnum.push_back(stringFromEnum); } bool* ScriptUtil::getBoolPointer(int index) { GENERATE_LUA_GET_POINTER(bool, luaCheckBool); } short* ScriptUtil::getShortPointer(int index) { GENERATE_LUA_GET_POINTER(short, (short)luaL_checkint); } int* ScriptUtil::getIntPointer(int index) { GENERATE_LUA_GET_POINTER(int, (int)luaL_checkint); } long* ScriptUtil::getLongPointer(int index) { GENERATE_LUA_GET_POINTER(long, (long)luaL_checkint); } unsigned char* ScriptUtil::getUnsignedCharPointer(int index) { GENERATE_LUA_GET_POINTER(unsigned char, (unsigned char)luaL_checkunsigned); } unsigned short* ScriptUtil::getUnsignedShortPointer(int index) { GENERATE_LUA_GET_POINTER(unsigned short, (unsigned short)luaL_checkunsigned); } unsigned int* ScriptUtil::getUnsignedIntPointer(int index) { GENERATE_LUA_GET_POINTER(unsigned int, (unsigned int)luaL_checkunsigned); } unsigned long* ScriptUtil::getUnsignedLongPointer(int index) { GENERATE_LUA_GET_POINTER(unsigned long, (unsigned long)luaL_checkunsigned); } float* ScriptUtil::getFloatPointer(int index) { GENERATE_LUA_GET_POINTER(float, (float)luaL_checknumber); } double* ScriptUtil::getDoublePointer(int index) { GENERATE_LUA_GET_POINTER(double, (double)luaL_checknumber); } const char* ScriptUtil::getString(int index, bool isStdString) { if (lua_type(Game::getInstance()->getScriptController()->_lua, index) == LUA_TSTRING) return luaL_checkstring(Game::getInstance()->getScriptController()->_lua, index); else if (lua_type(Game::getInstance()->getScriptController()->_lua, index) == LUA_TNIL && !isStdString) return NULL; else { GP_ERROR("Invalid string parameter (index = %d).", index); return NULL; } } bool ScriptUtil::luaCheckBool(lua_State* state, int n) { if (!lua_isboolean(state, n)) { const char* msg = lua_pushfstring(state, "%s expected, got %s", lua_typename(state, LUA_TBOOLEAN), luaL_typename(state, n)); luaL_argerror(state, n, msg); return false; } return (lua_toboolean(state, n) != 0); } void ScriptController::loadScript(const char* path, bool forceReload) { std::set::iterator iter = _loadedScripts.find(path); if (iter == _loadedScripts.end() || forceReload) { const char* scriptContents = FileSystem::readAll(path); if (luaL_dostring(_lua, scriptContents)) GP_WARN("Failed to run Lua script with error: '%s'.", lua_tostring(_lua, -1)); SAFE_DELETE_ARRAY(scriptContents); if (iter == _loadedScripts.end()) _loadedScripts.insert(path); } } std::string ScriptController::loadUrl(const char* url) { std::string file; std::string id; splitURL(url, &file, &id); // Make sure the function isn't empty. if (id.size() <= 0) { GP_ERROR("Got an empty function name when parsing function url '%s'.", url); return std::string(); } // Ensure the script is loaded. if (file.size() > 0) Game::getInstance()->getScriptController()->loadScript(file.c_str()); // Return the function name. return id; } bool ScriptController::getBool(const char* name) { lua_getglobal(_lua, name); return ScriptUtil::luaCheckBool(_lua, -1); } char ScriptController::getChar(const char* name) { lua_getglobal(_lua, name); return (char)luaL_checkint(_lua, -1); } short ScriptController::getShort(const char* name) { lua_getglobal(_lua, name); return (short)luaL_checkint(_lua, -1); } int ScriptController::getInt(const char* name) { lua_getglobal(_lua, name); return luaL_checkint(_lua, -1); } long ScriptController::getLong(const char* name) { lua_getglobal(_lua, name); return luaL_checklong(_lua, -1); } unsigned char ScriptController::getUnsignedChar(const char* name) { lua_getglobal(_lua, name); return (unsigned char)luaL_checkunsigned(_lua, -1); } unsigned short ScriptController::getUnsignedShort(const char* name) { lua_getglobal(_lua, name); return (unsigned short)luaL_checkunsigned(_lua, -1); } unsigned int ScriptController::getUnsignedInt(const char* name) { lua_getglobal(_lua, name); return (unsigned int)luaL_checkunsigned(_lua, -1); } unsigned long ScriptController::getUnsignedLong(const char* name) { lua_getglobal(_lua, name); return (unsigned long)luaL_checkunsigned(_lua, -1); } float ScriptController::getFloat(const char* name) { lua_getglobal(_lua, name); return (float)luaL_checknumber(_lua, -1); } double ScriptController::getDouble(const char* name) { lua_getglobal(_lua, name); return (double)luaL_checknumber(_lua, -1); } const char* ScriptController::getString(const char* name) { lua_getglobal(_lua, name); return luaL_checkstring(_lua, -1); } void ScriptController::setBool(const char* name, bool v) { lua_pushboolean(_lua, v); lua_setglobal(_lua, name); } void ScriptController::setChar(const char* name, char v) { lua_pushinteger(_lua, v); lua_setglobal(_lua, name); } void ScriptController::setShort(const char* name, short v) { lua_pushinteger(_lua, v); lua_setglobal(_lua, name); } void ScriptController::setInt(const char* name, int v) { lua_pushinteger(_lua, v); lua_setglobal(_lua, name); } void ScriptController::setLong(const char* name, long v) { lua_pushinteger(_lua, v); lua_setglobal(_lua, name); } void ScriptController::setUnsignedChar(const char* name, unsigned char v) { lua_pushunsigned(_lua, v); lua_setglobal(_lua, name); } void ScriptController::setUnsignedShort(const char* name, unsigned short v) { lua_pushunsigned(_lua, v); lua_setglobal(_lua, name); } void ScriptController::setUnsignedInt(const char* name, unsigned int v) { lua_pushunsigned(_lua, v); lua_setglobal(_lua, name); } void ScriptController::setUnsignedLong(const char* name, unsigned long v) { lua_pushunsigned(_lua, v); lua_setglobal(_lua, name); } void ScriptController::setFloat(const char* name, float v) { lua_pushnumber(_lua, v); lua_setglobal(_lua, name); } void ScriptController::setDouble(const char* name, double v) { lua_pushnumber(_lua, v); lua_setglobal(_lua, name); } void ScriptController::setString(const char* name, const char* v) { lua_pushstring(_lua, v); lua_setglobal(_lua, name); } void ScriptController::print(const char* str) { gameplay::print("%s", str); } void ScriptController::print(const char* str1, const char* str2) { gameplay::print("%s%s", str1, str2); } ScriptController::ScriptController() : _lua(NULL) { memset(_callbacks, 0, sizeof(std::string*) * CALLBACK_COUNT); } ScriptController::~ScriptController() { for (unsigned int i = 0; i < CALLBACK_COUNT; i++) { SAFE_DELETE(_callbacks[i]); } } static const char* lua_print_function = "function print(...)\n" " ScriptController.print(table.concat({...},\"\\t\"), \"\\n\")\n" "end\n"; #ifndef WIN32 static const char* lua_loadfile_function = "do\n" " local oldLoadfile = loadfile\n" " loadfile = function(filename)\n" " if filename ~= nil and not FileSystem.isAbsolutePath(filename) then\n" " filename = FileSystem.getResourcePath() .. filename\n" " end\n" " return oldLoadfile(filename)\n" " end\n" "end\n"; static const char* lua_dofile_function = "do\n" " local oldDofile = dofile\n" " dofile = function(filename)\n" " if filename ~= nil and not FileSystem.isAbsolutePath(filename) then\n" " filename = FileSystem.getResourcePath() .. filename\n" " end\n" " return oldDofile(filename)\n" " end\n" "end\n"; #endif void ScriptController::initialize() { _lua = luaL_newstate(); if (!_lua) GP_ERROR("Failed to initialize Lua scripting engine."); luaL_openlibs(_lua); lua_RegisterAllBindings(); // Create our own print() function that uses gameplay::print. if (luaL_dostring(_lua, lua_print_function)) GP_ERROR("Failed to load custom print() function with error: '%s'.", lua_tostring(_lua, -1)); #ifndef WIN32 // Change the functions that read a file to use FileSystem.getResourcePath as their base path. if (luaL_dostring(_lua, lua_loadfile_function)) GP_ERROR("Failed to load custom loadfile() function with error: '%s'.", lua_tostring(_lua, -1)); if (luaL_dostring(_lua, lua_dofile_function)) GP_ERROR("Failed to load custom dofile() function with error: '%s'.", lua_tostring(_lua, -1)); #endif } void ScriptController::initializeGame() { if (_callbacks[INITIALIZE]) { executeFunction(_callbacks[INITIALIZE]->c_str()); } } void ScriptController::finalize() { if (_lua) lua_close(_lua); } void ScriptController::finalizeGame() { if (_callbacks[FINALIZE]) { executeFunction(_callbacks[FINALIZE]->c_str()); } // Perform a full garbage collection cycle. lua_gc(_lua, LUA_GCCOLLECT, 0); } void ScriptController::update(float elapsedTime) { if (_callbacks[UPDATE]) { executeFunction(_callbacks[UPDATE]->c_str(), "f", elapsedTime); } } void ScriptController::render(float elapsedTime) { if (_callbacks[RENDER]) { executeFunction(_callbacks[RENDER]->c_str(), "f", elapsedTime); } } void ScriptController::keyEvent(Keyboard::KeyEvent evt, int key) { if (_callbacks[KEY_EVENT]) { executeFunction(_callbacks[KEY_EVENT]->c_str(), "[Keyboard::KeyEvent][Keyboard::Key]", evt, key); } } void ScriptController::touchEvent(Touch::TouchEvent evt, int x, int y, unsigned int contactIndex) { if (_callbacks[TOUCH_EVENT]) { executeFunction(_callbacks[TOUCH_EVENT]->c_str(), "[Touch::TouchEvent]iiui", evt, x, y, contactIndex); } } bool ScriptController::mouseEvent(Mouse::MouseEvent evt, int x, int y, int wheelDelta) { if (_callbacks[MOUSE_EVENT]) { return executeFunction(_callbacks[MOUSE_EVENT]->c_str(), "[Mouse::MouseEvent]iiii", evt, x, y, wheelDelta); } return false; } void ScriptController::gamepadEvent(Gamepad::GamepadEvent evt, Gamepad* gamepad) { if (_callbacks[GAMEPAD_EVENT]) { executeFunction(_callbacks[GAMEPAD_EVENT]->c_str(), "[Gamepad::GamepadEvent]", evt, gamepad); } } void ScriptController::executeFunctionHelper(int resultCount, const char* func, const char* args, va_list* list) { if (func == NULL) { GP_ERROR("Lua function name must be non-null."); return; } const char* sig = args; int argumentCount = 0; lua_getglobal(_lua, func); // Push the arguments to the Lua stack if there are any. if (sig) { while (true) { if (!(*sig)) break; switch(*sig++) { // Signed integers. case 'c': case 'h': case 'i': case 'l': lua_pushinteger(_lua, va_arg(*list, int)); break; // Unsigned integers. case 'u': // Skip past the actual type (long, int, short, char). sig++; lua_pushunsigned(_lua, va_arg(*list, int)); break; // Booleans. case 'b': lua_pushboolean(_lua, va_arg(*list, int)); break; // Floating point numbers. case 'f': case 'd': lua_pushnumber(_lua, va_arg(*list, double)); break; // Strings. case 's': lua_pushstring(_lua, va_arg(*list, char*)); break; // Pointers. case 'p': lua_pushlightuserdata(_lua, va_arg(*list, void*)); break; // Enums. case '[': { std::string type = sig; type = type.substr(0, type.find("]")); // Skip past the closing ']' (the semi-colon here is intentional-do not remove). while (*sig++ != ']'); unsigned int value = va_arg(*list, int); std::string enumStr = ""; for (unsigned int i = 0; enumStr.size() == 0 && i < _stringFromEnum.size(); i++) { enumStr = (*_stringFromEnum[i])(type, value); } lua_pushstring(_lua, enumStr.c_str()); break; } // Object references/pointers (Lua userdata). case '<': { std::string type = sig; type = type.substr(0, type.find(">")); // Skip past the closing '>' (the semi-colon here is intentional-do not remove). while (*sig++ != '>'); // Calculate the unique Lua type name. size_t i = type.find("::"); while (i != type.npos) { // We use "" as the replacement here-this must match the preprocessor // define SCOPE_REPLACEMENT from the gameplay-luagen project. type.replace(i, 2, ""); i = type.find("::"); } void* ptr = va_arg(*list, void*); if (ptr == NULL) { lua_pushnil(_lua); } else { ScriptUtil::LuaObject* object = (ScriptUtil::LuaObject*)lua_newuserdata(_lua, sizeof(ScriptUtil::LuaObject)); object->instance = ptr; object->owns = false; luaL_getmetatable(_lua, type.c_str()); lua_setmetatable(_lua, -2); } break; } default: GP_ERROR("Invalid argument type '%d'.", *(sig - 1)); } argumentCount++; luaL_checkstack(_lua, 1, "Too many arguments."); } } // Perform the function call. if (lua_pcall(_lua, argumentCount, resultCount, 0) != 0) GP_WARN("Failed to call function '%s' with error '%s'.", func, lua_tostring(_lua, -1)); } void ScriptController::registerCallback(ScriptCallback callback, std::string function) { SAFE_DELETE(_callbacks[callback]); _callbacks[callback] = new std::string(function); } ScriptController::ScriptCallback ScriptController::toCallback(const char* name) { if (strcmp(name, "initialize") == 0) return ScriptController::INITIALIZE; else if (strcmp(name, "update") == 0) return ScriptController::UPDATE; else if (strcmp(name, "render") == 0) return ScriptController::RENDER; else if (strcmp(name, "finalize") == 0) return ScriptController::FINALIZE; else if (strcmp(name, "keyEvent") == 0) return ScriptController::KEY_EVENT; else if (strcmp(name, "touchEvent") == 0) return ScriptController::TOUCH_EVENT; else if (strcmp(name, "mouseEvent") == 0) return ScriptController::MOUSE_EVENT; else if (strcmp(name, "gamepadEvent") == 0) return ScriptController::GAMEPAD_EVENT; else return ScriptController::INVALID_CALLBACK; } // Helper macros. #define SCRIPT_EXECUTE_FUNCTION_NO_PARAM(type, checkfunc) \ executeFunctionHelper(1, func, NULL, NULL); \ type value = (type)checkfunc(_lua, -1); \ lua_pop(_lua, -1); \ return value; #define SCRIPT_EXECUTE_FUNCTION_PARAM(type, checkfunc) \ va_list list; \ va_start(list, args); \ executeFunctionHelper(1, func, args, &list); \ type value = (type)checkfunc(_lua, -1); \ lua_pop(_lua, -1); \ va_end(list); \ return value; #define SCRIPT_EXECUTE_FUNCTION_PARAM_LIST(type, checkfunc) \ executeFunctionHelper(1, func, args, list); \ type value = (type)checkfunc(_lua, -1); \ lua_pop(_lua, -1); \ return value; template<> void ScriptController::executeFunction(const char* func) { executeFunctionHelper(0, func, NULL, NULL); } template<> bool ScriptController::executeFunction(const char* func) { SCRIPT_EXECUTE_FUNCTION_NO_PARAM(bool, ScriptUtil::luaCheckBool); } template<> char ScriptController::executeFunction(const char* func) { SCRIPT_EXECUTE_FUNCTION_NO_PARAM(char, luaL_checkint); } template<> short ScriptController::executeFunction(const char* func) { SCRIPT_EXECUTE_FUNCTION_NO_PARAM(short, luaL_checkint); } template<> int ScriptController::executeFunction(const char* func) { SCRIPT_EXECUTE_FUNCTION_NO_PARAM(int, luaL_checkint); } template<> long ScriptController::executeFunction(const char* func) { SCRIPT_EXECUTE_FUNCTION_NO_PARAM(long, luaL_checklong); } template<> unsigned char ScriptController::executeFunction(const char* func) { SCRIPT_EXECUTE_FUNCTION_NO_PARAM(unsigned char, luaL_checkunsigned); } template<> unsigned short ScriptController::executeFunction(const char* func) { SCRIPT_EXECUTE_FUNCTION_NO_PARAM(unsigned short, luaL_checkunsigned); } template<> unsigned int ScriptController::executeFunction(const char* func) { SCRIPT_EXECUTE_FUNCTION_NO_PARAM(unsigned int, luaL_checkunsigned); } template<> unsigned long ScriptController::executeFunction(const char* func) { SCRIPT_EXECUTE_FUNCTION_NO_PARAM(unsigned long, luaL_checkunsigned); } template<> float ScriptController::executeFunction(const char* func) { SCRIPT_EXECUTE_FUNCTION_NO_PARAM(float, luaL_checknumber); } template<> double ScriptController::executeFunction(const char* func) { SCRIPT_EXECUTE_FUNCTION_NO_PARAM(double, luaL_checknumber); } template<> std::string ScriptController::executeFunction(const char* func) { SCRIPT_EXECUTE_FUNCTION_NO_PARAM(std::string, luaL_checkstring); } /** Template specialization. */ template<> void ScriptController::executeFunction(const char* func, const char* args, ...) { va_list list; va_start(list, args); executeFunctionHelper(0, func, args, &list); va_end(list); } /** Template specialization. */ template<> bool ScriptController::executeFunction(const char* func, const char* args, ...) { SCRIPT_EXECUTE_FUNCTION_PARAM(bool, ScriptUtil::luaCheckBool); } /** Template specialization. */ template<> char ScriptController::executeFunction(const char* func, const char* args, ...) { SCRIPT_EXECUTE_FUNCTION_PARAM(char, luaL_checkint); } /** Template specialization. */ template<> short ScriptController::executeFunction(const char* func, const char* args, ...) { SCRIPT_EXECUTE_FUNCTION_PARAM(short, luaL_checkint); } /** Template specialization. */ template<> int ScriptController::executeFunction(const char* func, const char* args, ...) { SCRIPT_EXECUTE_FUNCTION_PARAM(int, luaL_checkint); } /** Template specialization. */ template<> long ScriptController::executeFunction(const char* func, const char* args, ...) { SCRIPT_EXECUTE_FUNCTION_PARAM(long, luaL_checklong); } /** Template specialization. */ template<> unsigned char ScriptController::executeFunction(const char* func, const char* args, ...) { SCRIPT_EXECUTE_FUNCTION_PARAM(unsigned char, luaL_checkunsigned); } /** Template specialization. */ template<> unsigned short ScriptController::executeFunction(const char* func, const char* args, ...) { SCRIPT_EXECUTE_FUNCTION_PARAM(unsigned short, luaL_checkunsigned); } /** Template specialization. */ template<> unsigned int ScriptController::executeFunction(const char* func, const char* args, ...) { SCRIPT_EXECUTE_FUNCTION_PARAM(unsigned int, luaL_checkunsigned); } /** Template specialization. */ template<> unsigned long ScriptController::executeFunction(const char* func, const char* args, ...) { SCRIPT_EXECUTE_FUNCTION_PARAM(unsigned long, luaL_checkunsigned); } /** Template specialization. */ template<> float ScriptController::executeFunction(const char* func, const char* args, ...) { SCRIPT_EXECUTE_FUNCTION_PARAM(float, luaL_checknumber); } /** Template specialization. */ template<> double ScriptController::executeFunction(const char* func, const char* args, ...) { SCRIPT_EXECUTE_FUNCTION_PARAM(double, luaL_checknumber); } /** Template specialization. */ template<> std::string ScriptController::executeFunction(const char* func, const char* args, ...) { SCRIPT_EXECUTE_FUNCTION_PARAM(std::string, luaL_checkstring); } /** Template specialization. */ template<> void ScriptController::executeFunction(const char* func, const char* args, va_list* list) { executeFunctionHelper(0, func, args, list); } /** Template specialization. */ template<> bool ScriptController::executeFunction(const char* func, const char* args, va_list* list) { SCRIPT_EXECUTE_FUNCTION_PARAM_LIST(bool, ScriptUtil::luaCheckBool); } /** Template specialization. */ template<> char ScriptController::executeFunction(const char* func, const char* args, va_list* list) { SCRIPT_EXECUTE_FUNCTION_PARAM_LIST(char, luaL_checkint); } /** Template specialization. */ template<> short ScriptController::executeFunction(const char* func, const char* args, va_list* list) { SCRIPT_EXECUTE_FUNCTION_PARAM_LIST(short, luaL_checkint); } /** Template specialization. */ template<> int ScriptController::executeFunction(const char* func, const char* args, va_list* list) { SCRIPT_EXECUTE_FUNCTION_PARAM_LIST(int, luaL_checkint); } /** Template specialization. */ template<> long ScriptController::executeFunction(const char* func, const char* args, va_list* list) { SCRIPT_EXECUTE_FUNCTION_PARAM_LIST(long, luaL_checklong); } /** Template specialization. */ template<> unsigned char ScriptController::executeFunction(const char* func, const char* args, va_list* list) { SCRIPT_EXECUTE_FUNCTION_PARAM_LIST(unsigned char, luaL_checkunsigned); } /** Template specialization. */ template<> unsigned short ScriptController::executeFunction(const char* func, const char* args, va_list* list) { SCRIPT_EXECUTE_FUNCTION_PARAM_LIST(unsigned short, luaL_checkunsigned); } /** Template specialization. */ template<> unsigned int ScriptController::executeFunction(const char* func, const char* args, va_list* list) { SCRIPT_EXECUTE_FUNCTION_PARAM_LIST(unsigned int, luaL_checkunsigned); } /** Template specialization. */ template<> unsigned long ScriptController::executeFunction(const char* func, const char* args, va_list* list) { SCRIPT_EXECUTE_FUNCTION_PARAM_LIST(unsigned long, luaL_checkunsigned); } /** Template specialization. */ template<> float ScriptController::executeFunction(const char* func, const char* args, va_list* list) { SCRIPT_EXECUTE_FUNCTION_PARAM_LIST(float, luaL_checknumber); } /** Template specialization. */ template<> double ScriptController::executeFunction(const char* func, const char* args, va_list* list) { SCRIPT_EXECUTE_FUNCTION_PARAM_LIST(double, luaL_checknumber); } /** Template specialization. */ template<> std::string ScriptController::executeFunction(const char* func, const char* args, va_list* list) { SCRIPT_EXECUTE_FUNCTION_PARAM_LIST(std::string, luaL_checkstring); } }