mirror of
https://github.com/OpenMW/openmw.git
synced 2025-01-16 19:59:55 +00:00
114 lines
3.6 KiB
C++
114 lines
3.6 KiB
C++
#ifndef COMPONENTS_LUAUI_CONTENT
|
|
#define COMPONENTS_LUAUI_CONTENT
|
|
|
|
#include <map>
|
|
#include <string>
|
|
|
|
#include <sol/sol.hpp>
|
|
|
|
#include <components/lua/luastate.hpp>
|
|
|
|
namespace LuaUi::Content
|
|
{
|
|
sol::protected_function loadConstructor(LuaUtil::LuaState* state);
|
|
|
|
class View
|
|
{
|
|
public:
|
|
// accepts only Lua tables returned by ui.content
|
|
explicit View(sol::table table)
|
|
: mTable(std::move(table))
|
|
{
|
|
if (!isValid(mTable))
|
|
throw std::domain_error("Expected a Content table");
|
|
}
|
|
|
|
static bool isValid(const sol::object& object)
|
|
{
|
|
if (object.get_type() != sol::type::table)
|
|
return false;
|
|
sol::table table = object;
|
|
return table.traverse_get<sol::optional<bool>>(sol::metatable_key, "__Content").value_or(false);
|
|
}
|
|
|
|
size_t size() const { return mTable.size(); }
|
|
|
|
void assign(std::string_view name, const sol::table& table)
|
|
{
|
|
if (indexOf(name).has_value())
|
|
mTable[name] = table;
|
|
else
|
|
throw std::domain_error("Invalid Content key");
|
|
}
|
|
void assign(size_t index, const sol::table& table)
|
|
{
|
|
if (index <= size())
|
|
mTable[toLua(index)] = table;
|
|
else
|
|
throw std::range_error("Invalid Content index");
|
|
}
|
|
void insert(size_t index, const sol::table& table) { callMethod("insert", toLua(index), table); }
|
|
|
|
sol::table at(size_t index) const
|
|
{
|
|
if (index < size())
|
|
return mTable.get<sol::table>(toLua(index));
|
|
else
|
|
throw std::range_error("Invalid Content index");
|
|
}
|
|
sol::table at(std::string_view name) const
|
|
{
|
|
if (indexOf(name).has_value())
|
|
return mTable.get<sol::table>(name);
|
|
else
|
|
throw std::range_error("Invalid Content key");
|
|
}
|
|
void remove(size_t index)
|
|
{
|
|
if (index < size())
|
|
// for some reason mTable[key] = value doesn't call __newindex
|
|
mTable[sol::metatable_key][sol::meta_function::new_index].get<sol::protected_function>()(
|
|
mTable, toLua(index), sol::nil);
|
|
else
|
|
throw std::range_error("Invalid Content index");
|
|
}
|
|
void remove(std::string_view name)
|
|
{
|
|
auto index = indexOf(name);
|
|
if (index.has_value())
|
|
remove(index.value());
|
|
else
|
|
throw std::domain_error("Invalid Content key");
|
|
}
|
|
std::optional<size_t> indexOf(std::string_view name) const
|
|
{
|
|
sol::object result = callMethod("indexOf", name);
|
|
if (result.is<size_t>())
|
|
return fromLua(result.as<size_t>());
|
|
else
|
|
return std::nullopt;
|
|
}
|
|
std::optional<size_t> indexOf(const sol::table& table) const
|
|
{
|
|
sol::object result = callMethod("indexOf", table);
|
|
if (result.is<size_t>())
|
|
return fromLua(result.as<size_t>());
|
|
else
|
|
return std::nullopt;
|
|
}
|
|
|
|
private:
|
|
sol::table mTable;
|
|
|
|
template <typename... Arg>
|
|
sol::object callMethod(std::string_view name, Arg&&... arg) const
|
|
{
|
|
return mTable.get<sol::protected_function>(name)(mTable, arg...);
|
|
}
|
|
|
|
static inline size_t toLua(size_t index) { return index + 1; }
|
|
static inline size_t fromLua(size_t index) { return index - 1; }
|
|
};
|
|
}
|
|
|
|
#endif // COMPONENTS_LUAUI_CONTENT
|