2016-01-12 03:41:44 +00:00
|
|
|
//
|
|
|
|
// Created by koncord on 19.03.16.
|
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef PLUGINSYSTEM3_SCRIPT_HPP
|
|
|
|
#define PLUGINSYSTEM3_SCRIPT_HPP
|
2018-12-01 01:03:39 +00:00
|
|
|
|
|
|
|
#include <boost/any.hpp>
|
|
|
|
#include <unordered_map>
|
|
|
|
#include <memory>
|
|
|
|
|
2016-01-12 03:41:44 +00:00
|
|
|
#include "Types.hpp"
|
|
|
|
#include "SystemInterface.hpp"
|
|
|
|
#include "ScriptFunction.hpp"
|
|
|
|
#include "ScriptFunctions.hpp"
|
|
|
|
#include "Language.hpp"
|
|
|
|
|
2018-12-01 01:03:39 +00:00
|
|
|
#include "Networking.hpp"
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
|
|
class Script : private ScriptFunctions
|
|
|
|
{
|
|
|
|
// http://imgur.com/hU0N4EH
|
|
|
|
private:
|
|
|
|
|
|
|
|
Language *lang;
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
SCRIPT_CPP,
|
2018-12-08 22:40:02 +00:00
|
|
|
SCRIPT_LUA,
|
|
|
|
SCRIPT_MONO
|
2016-01-12 03:41:44 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
template<typename R>
|
|
|
|
R GetScript(const char *name)
|
|
|
|
{
|
2016-08-17 15:20:36 +00:00
|
|
|
if (script_type == SCRIPT_CPP)
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
|
|
|
return SystemInterface<R>(lang->GetInterface(), name).result;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return reinterpret_cast<R>(lang->IsCallbackPresent(name));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int script_type;
|
|
|
|
std::unordered_map<unsigned int, FunctionEllipsis<void>> callbacks_;
|
|
|
|
|
|
|
|
typedef std::vector<std::unique_ptr<Script>> ScriptList;
|
|
|
|
static ScriptList scripts;
|
|
|
|
|
|
|
|
Script(const char *path);
|
|
|
|
|
|
|
|
Script(const Script&) = delete;
|
|
|
|
Script& operator=(const Script&) = delete;
|
|
|
|
|
2018-12-29 03:40:31 +00:00
|
|
|
protected:
|
|
|
|
static std::string moddir;
|
2016-01-12 03:41:44 +00:00
|
|
|
public:
|
|
|
|
~Script();
|
|
|
|
|
|
|
|
static void LoadScript(const char *script, const char* base);
|
|
|
|
static void LoadScripts(char* scripts, const char* base);
|
|
|
|
static void UnloadScripts();
|
2018-12-29 03:40:31 +00:00
|
|
|
static void SetModDir(const std::string &moddir);
|
|
|
|
static const char* GetModDir();
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
|
|
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<size_t N>
|
|
|
|
static constexpr unsigned int CallbackIdentity(const char(&str)[N])
|
|
|
|
{
|
|
|
|
return Utils::hash(str);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<unsigned int I, bool B = false, typename... Args>
|
|
|
|
static unsigned int Call(Args&&... args) {
|
|
|
|
constexpr ScriptCallbackData const& data = CallBackData(I);
|
2017-02-27 09:13:07 +00:00
|
|
|
static_assert(data.callback.matches(TypeString<typename std::remove_reference<Args>::type...>::value),
|
|
|
|
"Wrong number or types of arguments");
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
|
|
unsigned int count = 0;
|
|
|
|
|
|
|
|
for (auto& script : scripts)
|
|
|
|
{
|
|
|
|
if (!script->callbacks_.count(I))
|
|
|
|
script->callbacks_.emplace(I, script->GetScript<FunctionEllipsis<void>>(data.name));
|
|
|
|
|
|
|
|
auto callback = script->callbacks_[I];
|
|
|
|
|
|
|
|
if (!callback)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (script->script_type == SCRIPT_CPP)
|
2018-10-30 07:32:52 +00:00
|
|
|
(callback)(std::forward<Args>(args)...);
|
2018-12-08 22:40:02 +00:00
|
|
|
else
|
2018-11-30 20:01:02 +00:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
script->lang->Call(data.name, data.callback.types, B, std::forward<Args>(args)...);
|
|
|
|
}
|
|
|
|
catch (std::exception &e)
|
|
|
|
{
|
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_ERROR, e.what());
|
2018-12-01 01:03:39 +00:00
|
|
|
|
|
|
|
if (!mwmp::Networking::getPtr()->getScriptErrorIgnoringState())
|
|
|
|
throw;
|
2018-11-30 20:01:02 +00:00
|
|
|
}
|
|
|
|
}
|
2016-01-12 03:41:44 +00:00
|
|
|
++count;
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif //PLUGINSYSTEM3_SCRIPT_HPP
|