1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-19 19:53:53 +00:00
openmw/components/lua_ui/content.hpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

115 lines
3.6 KiB
C++
Raw Normal View History

2021-11-18 15:19:54 +00:00
#ifndef COMPONENTS_LUAUI_CONTENT
#define COMPONENTS_LUAUI_CONTENT
#include <map>
#include <string>
#include <sol/sol.hpp>
2023-01-31 18:50:33 +00:00
#include <components/lua/luastate.hpp>
namespace LuaUi::Content
2021-11-18 15:19:54 +00:00
{
2023-01-31 18:50:33 +00:00
sol::protected_function loadConstructor(LuaUtil::LuaState* state);
class View
2021-11-18 15:19:54 +00:00
{
public:
// accepts only Lua tables returned by ui.content
explicit View(sol::table table)
: mTable(std::move(table))
2023-01-13 20:14:23 +00:00
{
if (!isValid(mTable))
throw std::domain_error("Expected a Content table");
2023-01-13 20:14:23 +00:00
}
2021-11-18 15:19:54 +00:00
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);
}
2021-11-18 15:19:54 +00:00
size_t size() const { return mTable.size(); }
2021-11-18 15:19:54 +00:00
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
2023-02-01 15:24:45 +00:00
throw std::range_error("Invalid Content index");
}
void insert(size_t index, const sol::table& table) { callMethod("insert", toLua(index), table); }
2023-01-13 20:14:23 +00:00
sol::table at(size_t index) const
{
if (index < size())
return mTable.get<sol::table>(toLua(index));
else
2023-02-01 15:24:45 +00:00
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
2023-02-01 15:24:45 +00:00
throw std::range_error("Invalid Content key");
}
void remove(size_t index)
{
if (index < size())
2023-01-29 17:52:18 +00:00
// for some reason mTable[key] = value doesn't call __newindex
2023-01-29 18:36:21 +00:00
mTable[sol::metatable_key][sol::meta_function::new_index].get<sol::protected_function>()(
mTable, toLua(index), sol::nil);
else
2023-02-01 15:24:45 +00:00
throw std::range_error("Invalid Content index");
}
void remove(std::string_view name)
{
2023-01-29 17:52:18 +00:00
auto index = indexOf(name);
if (index.has_value())
remove(index.value());
else
2023-01-29 17:52:18 +00:00
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;
}
2021-11-18 15:19:54 +00:00
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; }
2021-11-18 15:19:54 +00:00
};
}
#endif // COMPONENTS_LUAUI_CONTENT