#ifndef PLUGINSYSTEM3_SCRIPT_HPP #define PLUGINSYSTEM3_SCRIPT_HPP #include #include #include #include "Types.hpp" #include "SystemInterface.hpp" #include "ScriptFunction.hpp" #include "ScriptFunctions.hpp" #include "Language.hpp" #include "Networking.hpp" class Script : private ScriptFunctions { // http://imgur.com/hU0N4EH private: Language *lang; enum { SCRIPT_CPP, SCRIPT_LUA }; template R GetScript(const char *name) { if (script_type == SCRIPT_CPP) { return SystemInterface(lang->GetInterface(), name).result; } else { return reinterpret_cast(lang->IsCallbackPresent(name)); } } int script_type; std::unordered_map> callbacks_; typedef std::vector> ScriptList; static ScriptList scripts; Script(const char *path); Script(const Script&) = delete; Script& operator=(const Script&) = delete; protected: static std::string moddir; public: ~Script(); static void LoadScript(const char *script, const char* base); static void LoadScripts(char* scripts, const char* base); static void UnloadScripts(); static void SetModDir(const std::string &moddir); static const char* GetModDir(); static constexpr ScriptCallbackData const& CallBackData(const unsigned int I, const unsigned int N = 0) { return callbacks[N].index == I ? callbacks[N] : CallBackData(I, N + 1); } template static constexpr unsigned int CallbackIdentity(const char(&str)[N]) { return Utils::hash(str); } template static unsigned int Call(Args&&... args) { constexpr ScriptCallbackData const& data = CallBackData(I); static_assert(data.callback.matches(TypeString::type...>::value), "Wrong number or types of arguments"); unsigned int count = 0; for (auto& script : scripts) { if (!script->callbacks_.count(I)) script->callbacks_.emplace(I, script->GetScript>(data.name)); auto callback = script->callbacks_[I]; if (!callback) continue; if (script->script_type == SCRIPT_CPP) (callback)(std::forward(args)...); #if defined (ENABLE_LUA) else if (script->script_type == SCRIPT_LUA) { try { script->lang->Call(data.name, data.callback.types, B, std::forward(args)...); } catch (std::exception &e) { LOG_MESSAGE_SIMPLE(TimedLog::LOG_ERROR, e.what()); Script::Call(e.what()); if (!mwmp::Networking::getPtr()->getScriptErrorIgnoringState()) throw; } } #endif ++count; } return count; } }; #endif //PLUGINSYSTEM3_SCRIPT_HPP