#include "asyncpackage.hpp" namespace sol { template <> struct is_automagical : std::false_type { }; template <> struct is_automagical : std::false_type { }; } namespace LuaUtil { struct TimerCallback { AsyncPackageId mAsyncId; std::string mName; }; Callback Callback::fromLua(const sol::table& t) { const sol::object& function = t.get_or(1, sol::nil); const sol::object& asyncPackageId = t.get_or(2, sol::nil); if (!function.is() || !asyncPackageId.is()) throw std::domain_error("Expected an async:callback, received a table"); return Callback{ function.as(), asyncPackageId.as().mHiddenData }; } bool Callback::isLuaCallback(const sol::object& t) { if (!t.is()) return false; sol::object meta = sol::table(t)[sol::metatable_key]; if (!meta.is()) return false; return sol::table(meta).raw_get_or("isCallback", false); } sol::function getAsyncPackageInitializer( lua_State* L, std::function simulationTimeFn, std::function gameTimeFn) { sol::state_view lua(L); using TimerType = ScriptsContainer::TimerType; sol::usertype api = lua.new_usertype("AsyncPackage"); api["registerTimerCallback"] = [](const AsyncPackageId& asyncId, std::string_view name, sol::main_protected_function callback) { asyncId.mContainer->registerTimerCallback(asyncId.mScriptId, name, std::move(callback)); return TimerCallback{ asyncId, std::string(name) }; }; api["newSimulationTimer"] = [simulationTimeFn](const AsyncPackageId&, double delay, const TimerCallback& callback, sol::main_object callbackArg) { callback.mAsyncId.mContainer->setupSerializableTimer(TimerType::SIMULATION_TIME, simulationTimeFn() + delay, callback.mAsyncId.mScriptId, callback.mName, std::move(callbackArg)); }; api["newGameTimer"] = [gameTimeFn](const AsyncPackageId&, double delay, const TimerCallback& callback, sol::main_object callbackArg) { callback.mAsyncId.mContainer->setupSerializableTimer(TimerType::GAME_TIME, gameTimeFn() + delay, callback.mAsyncId.mScriptId, callback.mName, std::move(callbackArg)); }; api["newUnsavableSimulationTimer"] = [simulationTimeFn](const AsyncPackageId& asyncId, double delay, sol::main_protected_function callback) { asyncId.mContainer->setupUnsavableTimer( TimerType::SIMULATION_TIME, simulationTimeFn() + delay, asyncId.mScriptId, std::move(callback)); }; api["newUnsavableGameTimer"] = [gameTimeFn](const AsyncPackageId& asyncId, double delay, sol::main_protected_function callback) { asyncId.mContainer->setupUnsavableTimer( TimerType::GAME_TIME, gameTimeFn() + delay, asyncId.mScriptId, std::move(callback)); }; sol::table callbackMeta = sol::table::create(L); callbackMeta[sol::meta_function::call] = [](const sol::table& callback, sol::variadic_args va) { return Callback::fromLua(callback).call(sol::as_args(va)); }; callbackMeta[sol::meta_function::to_string] = [] { return "Callback"; }; callbackMeta[sol::meta_function::metatable] = false; callbackMeta["isCallback"] = true; api["callback"] = [callbackMeta](const AsyncPackageId& asyncId, sol::main_protected_function fn) -> sol::table { sol::table c = sol::table::create(fn.lua_state(), 2); c.raw_set(1, std::move(fn), 2, asyncId); c[sol::metatable_key] = callbackMeta; return c; }; auto initializer = [](sol::table hiddenData) { ScriptId id = hiddenData[ScriptsContainer::sScriptIdKey]; return AsyncPackageId{ id.mContainer, id.mIndex, std::move(hiddenData) }; }; return sol::make_object(lua, initializer); } }