1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-26 18:35:20 +00:00
OpenMW/components/lua_ui/element.cpp
2022-01-18 08:12:56 +00:00

183 lines
5.7 KiB
C++

#include "element.hpp"
#include <MyGUI_Gui.h>
#include "content.hpp"
#include "util.hpp"
namespace LuaUi
{
std::string widgetType(const sol::table& layout)
{
return layout.get_or("type", std::string("LuaWidget"));
}
Content content(const sol::table& layout)
{
auto optional = layout.get<sol::optional<Content>>("content");
if (optional.has_value())
return optional.value();
else
return Content();
}
void setProperties(LuaUi::WidgetExtension* ext, const sol::table& layout)
{
ext->setProperties(layout.get<sol::object>("props"));
}
void setEventCallbacks(LuaUi::WidgetExtension* ext, const sol::table& layout)
{
ext->clearCallbacks();
auto events = layout.get<sol::optional<sol::table>>("events");
if (events.has_value())
{
events.value().for_each([ext](const sol::object& name, const sol::object& callback)
{
if (name.is<std::string>() && callback.is<LuaUtil::Callback>())
ext->setCallback(name.as<std::string>(), callback.as<LuaUtil::Callback>());
else if (!name.is<std::string>())
Log(Debug::Warning) << "UI event key must be a string";
else if (!callback.is<LuaUtil::Callback>())
Log(Debug::Warning) << "UI event handler for key \"" << name.as<std::string>()
<< "\" must be an openmw.async.callback";
});
}
}
void setLayout(LuaUi::WidgetExtension* ext, const sol::table& layout)
{
ext->setLayout(layout);
}
LuaUi::WidgetExtension* createWidget(const sol::table& layout, LuaUi::WidgetExtension* parent)
{
std::string type = widgetType(layout);
std::string skin = layout.get_or("skin", std::string());
std::string name = layout.get_or("name", std::string());
static auto widgetTypeMap = widgetTypeToName();
if (widgetTypeMap.find(type) == widgetTypeMap.end())
throw std::logic_error(std::string("Invalid widget type ") += type);
MyGUI::Widget* widget = MyGUI::Gui::getInstancePtr()->createWidgetT(
type, skin,
MyGUI::IntCoord(), MyGUI::Align::Default,
std::string(), name);
LuaUi::WidgetExtension* ext = dynamic_cast<LuaUi::WidgetExtension*>(widget);
if (!ext)
throw std::runtime_error("Invalid widget!");
ext->initialize(layout.lua_state(), widget);
if (parent != nullptr)
widget->attachToWidget(parent->widget());
setEventCallbacks(ext, layout);
setProperties(ext, layout);
setLayout(ext, layout);
Content cont = content(layout);
for (size_t i = 0; i < cont.size(); i++)
ext->addChild(createWidget(cont.at(i), ext));
return ext;
}
void destroyWidget(LuaUi::WidgetExtension* ext)
{
ext->deinitialize();
MyGUI::Gui::getInstancePtr()->destroyWidget(ext->widget());
}
void updateWidget(const sol::table& layout, LuaUi::WidgetExtension* ext)
{
setEventCallbacks(ext, layout);
setProperties(ext, layout);
setLayout(ext, layout);
Content newContent = content(layout);
size_t oldSize = ext->childCount();
size_t newSize = newContent.size();
size_t minSize = std::min(oldSize, newSize);
for (size_t i = 0; i < minSize; i++)
{
LuaUi::WidgetExtension* oldWidget = ext->childAt(i);
sol::table newChild = newContent.at(i);
if (oldWidget->widget()->getTypeName() != widgetType(newChild))
{
destroyWidget(oldWidget);
ext->assignChild(i, createWidget(newChild, ext));
}
else
updateWidget(newChild, oldWidget);
}
for (size_t i = minSize; i < oldSize; i++)
destroyWidget(ext->eraseChild(i));
for (size_t i = minSize; i < newSize; i++)
ext->addChild(createWidget(newContent.at(i), ext));
}
void setLayer(const sol::table& layout, LuaUi::WidgetExtension* ext)
{
MyGUI::ILayer* layerNode = ext->widget()->getLayer();
std::string currentLayer = layerNode ? layerNode->getName() : std::string();
std::string newLayer = layout.get_or("layer", std::string());
if (!newLayer.empty() && !MyGUI::LayerManager::getInstance().isExist(newLayer))
throw std::logic_error(std::string("Layer ") += newLayer += " doesn't exist");
else if (newLayer != currentLayer)
{
MyGUI::LayerManager::getInstance().attachToLayerNode(newLayer, ext->widget());
}
}
std::map<Element*, std::shared_ptr<Element>> Element::sAllElements;
Element::Element(sol::table layout)
: mRoot{ nullptr }
, mLayout{ std::move(layout) }
, mUpdate{ false }
, mDestroy{ false }
{}
std::shared_ptr<Element> Element::make(sol::table layout)
{
std::shared_ptr<Element> ptr(new Element(std::move(layout)));
sAllElements[ptr.get()] = ptr;
return ptr;
}
void Element::create()
{
assert(!mRoot);
if (!mRoot)
{
mRoot = createWidget(mLayout, nullptr);
setLayer(mLayout, mRoot);
}
}
void Element::update()
{
if (mRoot && mUpdate)
{
updateWidget(mLayout, mRoot);
setLayer(mLayout, mRoot);
}
mUpdate = false;
}
void Element::destroy()
{
if (mRoot)
destroyWidget(mRoot);
mRoot = nullptr;
sAllElements.erase(this);
}
}