#include "widget.hpp" #include #include namespace LuaUi { WidgetExtension::WidgetExtension() : mForcePosition(false) , mForceSize(false) , mPropagateEvents(true) , mVisible(true) , mLua(nullptr) , mWidget(nullptr) , mSlot(this) , mLayout(sol::nil) , mProperties(sol::nil) , mTemplateProperties(sol::nil) , mExternal(sol::nil) , mParent(nullptr) , mTemplateChild(false) , mElementRoot(false) { } void WidgetExtension::initialize(lua_State* lua, MyGUI::Widget* self, bool isRoot) { mLua = lua; mWidget = self; mElementRoot = isRoot; initialize(); updateTemplate(); } void WidgetExtension::initialize() { // \todo might be more efficient to only register these if there are Lua callbacks registerEvents(mWidget); } void WidgetExtension::deinitialize() { clearCallbacks(); clearEvents(mWidget); for (WidgetExtension* w : mChildren) w->deinitialize(); for (WidgetExtension* w : mTemplateChildren) w->deinitialize(); } void WidgetExtension::registerEvents(MyGUI::Widget* w) { w->eventKeyButtonPressed += MyGUI::newDelegate(this, &WidgetExtension::keyPress); w->eventKeyButtonReleased += MyGUI::newDelegate(this, &WidgetExtension::keyRelease); w->eventMouseButtonClick += MyGUI::newDelegate(this, &WidgetExtension::mouseClick); w->eventMouseButtonDoubleClick += MyGUI::newDelegate(this, &WidgetExtension::mouseDoubleClick); w->eventMouseButtonPressed += MyGUI::newDelegate(this, &WidgetExtension::mousePress); w->eventMouseButtonReleased += MyGUI::newDelegate(this, &WidgetExtension::mouseRelease); w->eventMouseMove += MyGUI::newDelegate(this, &WidgetExtension::mouseMove); w->eventMouseDrag += MyGUI::newDelegate(this, &WidgetExtension::mouseDrag); w->eventMouseSetFocus += MyGUI::newDelegate(this, &WidgetExtension::focusGain); w->eventMouseLostFocus += MyGUI::newDelegate(this, &WidgetExtension::focusLoss); w->eventKeySetFocus += MyGUI::newDelegate(this, &WidgetExtension::focusGain); w->eventKeyLostFocus += MyGUI::newDelegate(this, &WidgetExtension::focusLoss); } void WidgetExtension::clearEvents(MyGUI::Widget* w) { w->eventKeyButtonPressed.clear(); w->eventKeyButtonReleased.clear(); w->eventMouseButtonClick.clear(); w->eventMouseButtonDoubleClick.clear(); w->eventMouseButtonPressed.clear(); w->eventMouseButtonReleased.clear(); w->eventMouseMove.clear(); w->eventMouseDrag.clear(); w->eventMouseSetFocus.clear(); w->eventMouseLostFocus.clear(); w->eventKeySetFocus.clear(); w->eventKeyLostFocus.clear(); } void WidgetExtension::reset() { // detach all children from the slot widget, in case it gets destroyed for (auto& w : mChildren) w->widget()->detachFromWidget(); } void WidgetExtension::updateVisible() { // workaround for MyGUI bug // parent visibility doesn't affect added children MyGUI::Widget* parent = widget()->getParent(); bool inheritedVisible = mVisible && (parent == nullptr || parent->getInheritedVisible()); widget()->setVisible(inheritedVisible); } void WidgetExtension::attach(WidgetExtension* ext) { if (ext->mParent != this) { if (ext->mParent) { auto children = ext->mParent->children(); std::erase(children, this); ext->mParent->setChildren(children); } ext->detachFromParent(); } ext->mParent = this; ext->mTemplateChild = false; ext->widget()->attachToWidget(mSlot->widget()); } void WidgetExtension::attachTemplate(WidgetExtension* ext) { ext->mParent = this; ext->mTemplateChild = true; ext->widget()->attachToWidget(widget()); } void WidgetExtension::detachFromParent() { mParent = nullptr; widget()->detachFromWidget(); widget()->detachFromLayer(); } WidgetExtension* WidgetExtension::findDeep(std::string_view flagName) { for (WidgetExtension* w : mChildren) { WidgetExtension* result = w->findDeep(flagName); if (result != nullptr) return result; } if (externalValue(flagName, false)) return this; return nullptr; } void WidgetExtension::findAll(std::string_view flagName, std::vector& result) { if (externalValue(flagName, false)) result.push_back(this); for (WidgetExtension* w : mChildren) w->findAll(flagName, result); } WidgetExtension* WidgetExtension::findDeepInTemplates(std::string_view flagName) { for (WidgetExtension* w : mTemplateChildren) { WidgetExtension* result = w->findDeep(flagName); if (result != nullptr) return result; } return nullptr; } std::vector WidgetExtension::findAllInTemplates(std::string_view flagName) { std::vector result; for (WidgetExtension* w : mTemplateChildren) w->findAll(flagName, result); return result; } sol::table WidgetExtension::makeTable() const { return sol::table(lua(), sol::create); } sol::object WidgetExtension::keyEvent(MyGUI::KeyCode code) const { auto keySym = SDL_Keysym(); keySym.sym = SDLUtil::myGuiKeyToSdl(code); keySym.scancode = SDL_GetScancodeFromKey(keySym.sym); keySym.mod = SDL_GetModState(); return sol::make_object(lua(), keySym); } sol::object WidgetExtension::mouseEvent( int left, int top, MyGUI::MouseButton button = MyGUI::MouseButton::None) const { osg::Vec2f position(left, top); MyGUI::IntPoint absolutePosition = mWidget->getAbsolutePosition(); osg::Vec2f offset = position - osg::Vec2f(absolutePosition.left, absolutePosition.top); sol::table table = makeTable(); int sdlButton = SDLUtil::myGuiMouseButtonToSdl(button); table["position"] = position; table["offset"] = offset; if (sdlButton != 0) // nil if no button was pressed table["button"] = sdlButton; return table; } void WidgetExtension::setChildren(const std::vector& children) { mChildren.resize(children.size()); for (size_t i = 0; i < children.size(); ++i) { mChildren[i] = children[i]; attach(mChildren[i]); } updateChildren(); } void WidgetExtension::setTemplateChildren(const std::vector& children) { mTemplateChildren.resize(children.size()); for (size_t i = 0; i < children.size(); ++i) { mTemplateChildren[i] = children[i]; attachTemplate(mTemplateChildren[i]); } updateTemplate(); } void WidgetExtension::updateTemplate() { WidgetExtension* slot = findDeepInTemplates("slot"); if (slot == nullptr) mSlot = this; else mSlot = slot->mSlot; } void WidgetExtension::setCallback(const std::string& name, const LuaUtil::Callback& callback) { mCallbacks[name] = callback; } void WidgetExtension::clearCallbacks() { mCallbacks.clear(); } MyGUI::IntCoord WidgetExtension::forcedCoord() { return mForcedCoord; } void WidgetExtension::forceCoord(const MyGUI::IntCoord& offset) { mForcePosition = true; mForceSize = true; mForcedCoord = offset; } void WidgetExtension::forcePosition(const MyGUI::IntPoint& pos) { mForcePosition = true; mForcedCoord = pos; } void WidgetExtension::forceSize(const MyGUI::IntSize& size) { mForceSize = true; mForcedCoord = size; } void WidgetExtension::clearForced() { mForcePosition = false; mForceSize = false; } void WidgetExtension::updateCoord() { updateVisible(); MyGUI::IntCoord oldCoord = mWidget->getCoord(); MyGUI::IntCoord newCoord = calculateCoord(); if (oldCoord != newCoord) mWidget->setCoord(newCoord); updateChildrenCoord(); } void WidgetExtension::setProperties(const sol::object& props) { mProperties = props; updateProperties(); } void WidgetExtension::updateProperties() { mPropagateEvents = propertyValue("propagateEvents", true); mAbsoluteCoord = propertyValue("position", MyGUI::IntPoint()); mAbsoluteCoord = propertyValue("size", MyGUI::IntSize()); mRelativeCoord = propertyValue("relativePosition", MyGUI::FloatPoint()); mRelativeCoord = propertyValue("relativeSize", MyGUI::FloatSize()); mAnchor = propertyValue("anchor", MyGUI::FloatSize()); mVisible = propertyValue("visible", true); mWidget->setVisible(mVisible); mWidget->setPointer(propertyValue("pointer", std::string("arrow"))); mWidget->setAlpha(propertyValue("alpha", 1.f)); mWidget->setInheritsAlpha(propertyValue("inheritAlpha", true)); } void WidgetExtension::updateChildrenCoord() { for (WidgetExtension* w : mTemplateChildren) w->updateCoord(); for (WidgetExtension* w : mChildren) w->updateCoord(); } MyGUI::IntSize WidgetExtension::parentSize() const { if (!mParent) return widget()->getParentSize(); // size of the layer if (mTemplateChild) return mParent->templateScalingSize(); else return mParent->childScalingSize(); } MyGUI::IntSize WidgetExtension::calculateSize() const { if (mForceSize) return mForcedCoord.size(); MyGUI::IntSize pSize = parentSize(); MyGUI::IntSize newSize; newSize = mAbsoluteCoord.size(); newSize.width += mRelativeCoord.width * pSize.width; newSize.height += mRelativeCoord.height * pSize.height; return newSize; } MyGUI::IntPoint WidgetExtension::calculatePosition(const MyGUI::IntSize& size) const { if (mForcePosition) return mForcedCoord.point(); MyGUI::IntSize pSize = parentSize(); MyGUI::IntPoint newPosition; newPosition = mAbsoluteCoord.point(); newPosition.left += mRelativeCoord.left * pSize.width - mAnchor.width * size.width; newPosition.top += mRelativeCoord.top * pSize.height - mAnchor.height * size.height; return newPosition; } MyGUI::IntCoord WidgetExtension::calculateCoord() const { MyGUI::IntCoord newCoord; newCoord = calculateSize(); newCoord = calculatePosition(newCoord.size()); return newCoord; } MyGUI::IntSize WidgetExtension::childScalingSize() const { return mSlot->widget()->getSize(); } MyGUI::IntSize WidgetExtension::templateScalingSize() const { return widget()->getSize(); } void WidgetExtension::triggerEvent(std::string_view name, sol::object argument) const { auto it = mCallbacks.find(name); if (it != mCallbacks.end()) it->second.call(argument, mLayout); } void WidgetExtension::keyPress(MyGUI::Widget*, MyGUI::KeyCode code, MyGUI::Char ch) { if (code == MyGUI::KeyCode::None) { propagateEvent("textInput", [ch](auto w) { MyGUI::UString uString; uString.push_back(static_cast(ch)); return sol::make_object(w->lua(), uString.asUTF8()); }); } else propagateEvent("keyPress", [code](auto w) { return w->keyEvent(code); }); } void WidgetExtension::keyRelease(MyGUI::Widget*, MyGUI::KeyCode code) { propagateEvent("keyRelease", [code](auto w) { return w->keyEvent(code); }); } void WidgetExtension::mouseMove(MyGUI::Widget*, int left, int top) { propagateEvent("mouseMove", [left, top](auto w) { return w->mouseEvent(left, top); }); } void WidgetExtension::mouseDrag(MyGUI::Widget*, int left, int top, MyGUI::MouseButton button) { propagateEvent("mouseMove", [left, top, button](auto w) { return w->mouseEvent(left, top, button); }); } void WidgetExtension::mouseClick(MyGUI::Widget* _widget) { propagateEvent("mouseClick", [](auto) { return sol::nil; }); } void WidgetExtension::mouseDoubleClick(MyGUI::Widget* _widget) { propagateEvent("mouseDoubleClick", [](auto) { return sol::nil; }); } void WidgetExtension::mousePress(MyGUI::Widget*, int left, int top, MyGUI::MouseButton button) { propagateEvent("mousePress", [left, top, button](auto w) { return w->mouseEvent(left, top, button); }); } void WidgetExtension::mouseRelease(MyGUI::Widget*, int left, int top, MyGUI::MouseButton button) { propagateEvent("mouseRelease", [left, top, button](auto w) { return w->mouseEvent(left, top, button); }); } void WidgetExtension::focusGain(MyGUI::Widget*, MyGUI::Widget*) { propagateEvent("focusGain", [](auto) { return sol::nil; }); } void WidgetExtension::focusLoss(MyGUI::Widget*, MyGUI::Widget*) { propagateEvent("focusLoss", [](auto) { return sol::nil; }); } }