mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-26 21:26:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			332 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			332 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "element.hpp"
 | |
| 
 | |
| #include <MyGUI_Gui.h>
 | |
| 
 | |
| #include "content.hpp"
 | |
| #include "util.hpp"
 | |
| #include "widget.hpp"
 | |
| 
 | |
| namespace LuaUi
 | |
| {
 | |
|     namespace
 | |
|     {
 | |
|         namespace LayoutKeys
 | |
|         {
 | |
|             constexpr std::string_view type = "type";
 | |
|             constexpr std::string_view name = "name";
 | |
|             constexpr std::string_view layer = "layer";
 | |
|             constexpr std::string_view templateLayout = "template";
 | |
|             constexpr std::string_view props = "props";
 | |
|             constexpr std::string_view events = "events";
 | |
|             constexpr std::string_view content = "content";
 | |
|             constexpr std::string_view external = "external";
 | |
|         }
 | |
| 
 | |
|         const std::string defaultWidgetType = "LuaWidget";
 | |
| 
 | |
|         constexpr uint64_t maxDepth = 250;
 | |
| 
 | |
|         std::string widgetType(const sol::table& layout)
 | |
|         {
 | |
|             sol::object typeField = LuaUtil::getFieldOrNil(layout, LayoutKeys::type);
 | |
|             std::string type = LuaUtil::getValueOrDefault(typeField, defaultWidgetType);
 | |
|             sol::object templateTypeField
 | |
|                 = LuaUtil::getFieldOrNil(layout, LayoutKeys::templateLayout, LayoutKeys::type);
 | |
|             if (templateTypeField != sol::nil)
 | |
|             {
 | |
|                 std::string templateType = LuaUtil::getValueOrDefault(templateTypeField, defaultWidgetType);
 | |
|                 if (typeField != sol::nil && templateType != type)
 | |
|                     throw std::logic_error(std::string("Template layout type ") + type
 | |
|                         + std::string(" doesn't match template type ") + templateType);
 | |
|                 type = std::move(templateType);
 | |
|             }
 | |
|             return type;
 | |
|         }
 | |
| 
 | |
|         void destroyWidget(WidgetExtension* ext)
 | |
|         {
 | |
|             ext->deinitialize();
 | |
|             MyGUI::Gui::getInstancePtr()->destroyWidget(ext->widget());
 | |
|         }
 | |
| 
 | |
|         void destroyChild(WidgetExtension* ext)
 | |
|         {
 | |
|             if (!ext->isRoot())
 | |
|                 destroyWidget(ext);
 | |
|             else
 | |
|                 ext->detachFromParent();
 | |
|         }
 | |
| 
 | |
|         void detachElements(WidgetExtension* ext)
 | |
|         {
 | |
|             auto predicate = [](WidgetExtension* child) {
 | |
|                 if (child->isRoot())
 | |
|                     return true;
 | |
|                 detachElements(child);
 | |
|                 return false;
 | |
|             };
 | |
|             ext->detachChildrenIf(predicate);
 | |
|             ext->detachTemplateChildrenIf(predicate);
 | |
|         }
 | |
| 
 | |
|         void destroyRoot(WidgetExtension* ext)
 | |
|         {
 | |
|             detachElements(ext);
 | |
|             destroyWidget(ext);
 | |
|         }
 | |
| 
 | |
|         void updateRootCoord(WidgetExtension* ext)
 | |
|         {
 | |
|             WidgetExtension* root = ext;
 | |
|             while (root->getParent())
 | |
|                 root = root->getParent();
 | |
|             root->updateCoord();
 | |
|         }
 | |
| 
 | |
|         WidgetExtension* pluckElementRoot(const sol::object& child, uint64_t depth)
 | |
|         {
 | |
|             std::shared_ptr<Element> element = child.as<std::shared_ptr<Element>>();
 | |
|             if (element->mState == Element::Destroyed || element->mState == Element::Destroy)
 | |
|                 throw std::logic_error("Using a destroyed element as a layout child");
 | |
|             // child Element was created in the same frame and its action hasn't been processed yet
 | |
|             if (element->mState == Element::New)
 | |
|                 element->create(depth + 1);
 | |
|             WidgetExtension* root = element->mRoot;
 | |
|             assert(root);
 | |
|             return root;
 | |
|         }
 | |
| 
 | |
|         WidgetExtension* createWidget(const sol::table& layout, bool isRoot, uint64_t depth);
 | |
|         void updateWidget(WidgetExtension* ext, const sol::table& layout, uint64_t depth);
 | |
| 
 | |
|         std::vector<WidgetExtension*> updateContent(
 | |
|             const std::vector<WidgetExtension*>& children, const sol::object& contentObj, uint64_t depth)
 | |
|         {
 | |
|             ++depth;
 | |
|             std::vector<WidgetExtension*> result;
 | |
|             if (contentObj == sol::nil)
 | |
|             {
 | |
|                 for (WidgetExtension* w : children)
 | |
|                     destroyChild(w);
 | |
|                 return result;
 | |
|             }
 | |
|             ContentView content(LuaUtil::cast<sol::table>(contentObj));
 | |
|             result.resize(content.size());
 | |
|             size_t minSize = std::min(children.size(), content.size());
 | |
|             std::vector<WidgetExtension*> toDestroy;
 | |
|             for (size_t i = 0; i < minSize; i++)
 | |
|             {
 | |
|                 WidgetExtension* ext = children[i];
 | |
|                 sol::object child = content.at(i);
 | |
|                 if (child.is<Element>())
 | |
|                 {
 | |
|                     WidgetExtension* root = pluckElementRoot(child, depth);
 | |
|                     if (ext != root)
 | |
|                         toDestroy.emplace_back(ext);
 | |
|                     result[i] = root;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     sol::table newLayout = child.as<sol::table>();
 | |
|                     if (ext->widget()->getTypeName() == widgetType(newLayout))
 | |
|                     {
 | |
|                         updateWidget(ext, newLayout, depth);
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         toDestroy.emplace_back(ext);
 | |
|                         ext = createWidget(newLayout, false, depth);
 | |
|                     }
 | |
|                     result[i] = ext;
 | |
|                 }
 | |
|             }
 | |
|             for (size_t i = minSize; i < content.size(); i++)
 | |
|             {
 | |
|                 sol::object child = content.at(i);
 | |
|                 if (child.is<Element>())
 | |
|                     result[i] = pluckElementRoot(child, depth);
 | |
|                 else
 | |
|                     result[i] = createWidget(child.as<sol::table>(), false, depth);
 | |
|             }
 | |
|             // Don't destroy anything until element creation has had a chance to throw
 | |
|             for (size_t i = minSize; i < children.size(); i++)
 | |
|                 destroyChild(children[i]);
 | |
|             for (WidgetExtension* ext : toDestroy)
 | |
|                 destroyChild(ext);
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         void setTemplate(WidgetExtension* ext, const sol::object& templateLayout, uint64_t depth)
 | |
|         {
 | |
|             ++depth;
 | |
|             sol::object props = LuaUtil::getFieldOrNil(templateLayout, LayoutKeys::props);
 | |
|             ext->setTemplateProperties(props);
 | |
|             sol::object content = LuaUtil::getFieldOrNil(templateLayout, LayoutKeys::content);
 | |
|             ext->setTemplateChildren(updateContent(ext->templateChildren(), content, depth));
 | |
|         }
 | |
| 
 | |
|         void setEventCallbacks(WidgetExtension* ext, const sol::object& eventsObj)
 | |
|         {
 | |
|             ext->clearCallbacks();
 | |
|             if (eventsObj == sol::nil)
 | |
|                 return;
 | |
|             if (!eventsObj.is<sol::table>())
 | |
|                 throw std::logic_error("The \"events\" layout field must be a table of callbacks");
 | |
|             auto events = eventsObj.as<sol::table>();
 | |
|             events.for_each([ext](const sol::object& name, const sol::object& callback) {
 | |
|                 if (name.is<std::string>() && LuaUtil::Callback::isLuaCallback(callback))
 | |
|                     ext->setCallback(name.as<std::string>(), LuaUtil::Callback::fromLua(callback));
 | |
|                 else if (!name.is<std::string>())
 | |
|                     Log(Debug::Warning) << "UI event key must be a string";
 | |
|                 else
 | |
|                     Log(Debug::Warning) << "UI event handler for key \"" << name.as<std::string>()
 | |
|                                         << "\" must be an openmw.async.callback";
 | |
|             });
 | |
|         }
 | |
| 
 | |
|         WidgetExtension* createWidget(const sol::table& layout, bool isRoot, uint64_t depth)
 | |
|         {
 | |
|             static auto widgetTypeMap = widgetTypeToName();
 | |
|             std::string type = widgetType(layout);
 | |
|             if (widgetTypeMap.find(type) == widgetTypeMap.end())
 | |
|                 throw std::logic_error(std::string("Invalid widget type ") += type);
 | |
| 
 | |
|             std::string name = layout.get_or(LayoutKeys::name, std::string());
 | |
|             MyGUI::Widget* widget
 | |
|                 = MyGUI::Gui::getInstancePtr()->createWidgetT(type, {}, {}, MyGUI::Align::Default, {}, name);
 | |
| 
 | |
|             WidgetExtension* ext = dynamic_cast<WidgetExtension*>(widget);
 | |
|             if (!ext)
 | |
|                 throw std::runtime_error("Invalid widget!");
 | |
|             ext->initialize(layout.lua_state(), widget, isRoot);
 | |
| 
 | |
|             updateWidget(ext, layout, depth);
 | |
|             return ext;
 | |
|         }
 | |
| 
 | |
|         void updateWidget(WidgetExtension* ext, const sol::table& layout, uint64_t depth)
 | |
|         {
 | |
|             if (depth >= maxDepth)
 | |
|                 throw std::runtime_error("Maximum layout depth exceeded, probably caused by a circular reference");
 | |
|             ext->reset();
 | |
|             ext->setLayout(layout);
 | |
|             ext->setExternal(layout.get<sol::object>(LayoutKeys::external));
 | |
|             setTemplate(ext, layout.get<sol::object>(LayoutKeys::templateLayout), depth);
 | |
|             ext->setProperties(layout.get<sol::object>(LayoutKeys::props));
 | |
|             setEventCallbacks(ext, layout.get<sol::object>(LayoutKeys::events));
 | |
|             ext->setChildren(updateContent(ext->children(), layout.get<sol::object>(LayoutKeys::content), depth));
 | |
|             ext->updateCoord();
 | |
|         }
 | |
| 
 | |
|         std::string setLayer(WidgetExtension* ext, const sol::table& layout)
 | |
|         {
 | |
|             MyGUI::ILayer* layerNode = ext->widget()->getLayer();
 | |
|             std::string_view currentLayer;
 | |
|             if (layerNode)
 | |
|                 currentLayer = layerNode->getName();
 | |
|             std::string newLayer = layout.get_or(LayoutKeys::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());
 | |
|             }
 | |
|             return newLayer;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     std::map<Element*, std::shared_ptr<Element>> Element::sMenuElements;
 | |
|     std::map<Element*, std::shared_ptr<Element>> Element::sGameElements;
 | |
| 
 | |
|     Element::Element(sol::table layout)
 | |
|         : mRoot(nullptr)
 | |
|         , mLayout(std::move(layout))
 | |
|         , mLayer()
 | |
|         , mState(Element::New)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     std::shared_ptr<Element> Element::make(sol::table layout, bool menu)
 | |
|     {
 | |
|         std::shared_ptr<Element> ptr(new Element(std::move(layout)));
 | |
|         auto& container = menu ? sMenuElements : sGameElements;
 | |
|         container[ptr.get()] = ptr;
 | |
|         return ptr;
 | |
|     }
 | |
| 
 | |
|     void Element::erase(Element* element)
 | |
|     {
 | |
|         element->destroy();
 | |
|         sMenuElements.erase(element);
 | |
|         sGameElements.erase(element);
 | |
|     }
 | |
| 
 | |
|     void Element::create(uint64_t depth)
 | |
|     {
 | |
|         if (mState == New)
 | |
|         {
 | |
|             assert(!mRoot);
 | |
|             mRoot = createWidget(layout(), true, depth);
 | |
|             mLayer = setLayer(mRoot, layout());
 | |
|             updateRootCoord(mRoot);
 | |
|             mState = Created;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void Element::update()
 | |
|     {
 | |
|         if (mState == Update)
 | |
|         {
 | |
|             assert(mRoot);
 | |
|             if (mRoot->widget()->getTypeName() != widgetType(layout()))
 | |
|             {
 | |
|                 destroyRoot(mRoot);
 | |
|                 WidgetExtension* parent = mRoot->getParent();
 | |
|                 auto children = parent->children();
 | |
|                 auto it = std::find(children.begin(), children.end(), mRoot);
 | |
|                 assert(it != children.end());
 | |
|                 try
 | |
|                 {
 | |
|                     mRoot = createWidget(layout(), true, 0);
 | |
|                     *it = mRoot;
 | |
|                 }
 | |
|                 catch (...)
 | |
|                 {
 | |
|                     // Remove mRoot from its parent's children even if we couldn't replace it
 | |
|                     children.erase(it);
 | |
|                     parent->setChildren(children);
 | |
|                     mRoot = nullptr;
 | |
|                     mState = New;
 | |
|                     throw;
 | |
|                 }
 | |
|                 parent->setChildren(children);
 | |
|                 mRoot->updateCoord();
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 updateWidget(mRoot, layout(), 0);
 | |
|             }
 | |
|             mLayer = setLayer(mRoot, layout());
 | |
|             updateRootCoord(mRoot);
 | |
|             mState = Created;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void Element::destroy()
 | |
|     {
 | |
|         if (mState != Destroyed)
 | |
|         {
 | |
|             if (mRoot != nullptr)
 | |
|             {
 | |
|                 // If someone decided to destroy an element used as another element's content, we need to detach it
 | |
|                 // first so the parent doesn't end up holding a stale pointer
 | |
|                 if (WidgetExtension* parent = mRoot->getParent())
 | |
|                     parent->detachChildrenIf([&](WidgetExtension* child) { return child == mRoot; });
 | |
|                 destroyRoot(mRoot);
 | |
|                 mRoot = nullptr;
 | |
|             }
 | |
|             mLayout.reset();
 | |
|         }
 | |
|         mState = Destroyed;
 | |
|     }
 | |
| }
 |