Fix GUI element click-through by changing visibility (#9534)

This adds a vector that holds pointers to elements that should only be visible while being drawn.

In the guifsmenu's draw func, all elements in this vector are made visible and invisible again. Apart from there, they are always invisible. (Well they are still visible before the first drawn, does this matter? If yes, it could be fixed easily with some lines of code everywhere.)
This commit is contained in:
DS 2020-03-31 19:34:42 +02:00 committed by GitHub
parent 9953145a4e
commit d7825bca1b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 30 additions and 1 deletions

@ -141,6 +141,8 @@ GUIFormSpecMenu::~GUIFormSpecMenu()
background_it->drop(); background_it->drop();
for (auto &tooltip_rect_it : m_tooltip_rects) for (auto &tooltip_rect_it : m_tooltip_rects)
tooltip_rect_it.first->drop(); tooltip_rect_it.first->drop();
for (auto &clickthrough_it : m_clickthrough_elements)
clickthrough_it->drop();
delete m_selected_item; delete m_selected_item;
delete m_form_src; delete m_form_src;
@ -742,6 +744,9 @@ void GUIFormSpecMenu::parseImage(parserData* data, const std::string &element)
e->setNotClipped(style.getBool(StyleSpec::NOCLIP, m_formspec_version < 3)); e->setNotClipped(style.getBool(StyleSpec::NOCLIP, m_formspec_version < 3));
m_fields.push_back(spec); m_fields.push_back(spec);
// images should let events through
e->grab();
m_clickthrough_elements.push_back(e);
return; return;
} }
@ -775,6 +780,9 @@ void GUIFormSpecMenu::parseImage(parserData* data, const std::string &element)
e->setNotClipped(style.getBool(StyleSpec::NOCLIP, m_formspec_version < 3)); e->setNotClipped(style.getBool(StyleSpec::NOCLIP, m_formspec_version < 3));
m_fields.push_back(spec); m_fields.push_back(spec);
// images should let events through
e->grab();
m_clickthrough_elements.push_back(e);
return; return;
} }
errorstream<< "Invalid image element(" << parts.size() << "): '" << element << "'" << std::endl; errorstream<< "Invalid image element(" << parts.size() << "): '" << element << "'" << std::endl;
@ -882,7 +890,9 @@ void GUIFormSpecMenu::parseItemImage(parserData* data, const std::string &elemen
core::rect<s32>(pos, pos + geom), name, m_font, m_client); core::rect<s32>(pos, pos + geom), name, m_font, m_client);
auto style = getStyleForElement("item_image", spec.fname); auto style = getStyleForElement("item_image", spec.fname);
e->setNotClipped(style.getBool(StyleSpec::NOCLIP, false)); e->setNotClipped(style.getBool(StyleSpec::NOCLIP, false));
e->drop();
// item images should let events through
m_clickthrough_elements.push_back(e);
m_fields.push_back(spec); m_fields.push_back(spec);
return; return;
@ -1747,6 +1757,10 @@ void GUIFormSpecMenu::parseLabel(parserData* data, const std::string &element)
e->setOverrideColor(style.getColor(StyleSpec::TEXTCOLOR, video::SColor(0xFFFFFFFF))); e->setOverrideColor(style.getColor(StyleSpec::TEXTCOLOR, video::SColor(0xFFFFFFFF)));
m_fields.push_back(spec); m_fields.push_back(spec);
// labels should let events through
e->grab();
m_clickthrough_elements.push_back(e);
} }
return; return;
@ -1823,6 +1837,10 @@ void GUIFormSpecMenu::parseVertLabel(parserData* data, const std::string &elemen
e->setOverrideColor(style.getColor(StyleSpec::TEXTCOLOR, video::SColor(0xFFFFFFFF))); e->setOverrideColor(style.getColor(StyleSpec::TEXTCOLOR, video::SColor(0xFFFFFFFF)));
m_fields.push_back(spec); m_fields.push_back(spec);
// vertlabels should let events through
e->grab();
m_clickthrough_elements.push_back(e);
return; return;
} }
errorstream<< "Invalid vertlabel element(" << parts.size() << "): '" << element << "'" << std::endl; errorstream<< "Invalid vertlabel element(" << parts.size() << "): '" << element << "'" << std::endl;
@ -2745,6 +2763,8 @@ void GUIFormSpecMenu::regenerateGui(v2u32 screensize)
background_it->drop(); background_it->drop();
for (auto &tooltip_rect_it : m_tooltip_rects) for (auto &tooltip_rect_it : m_tooltip_rects)
tooltip_rect_it.first->drop(); tooltip_rect_it.first->drop();
for (auto &clickthrough_it : m_clickthrough_elements)
clickthrough_it->drop();
mydata.size= v2s32(100,100); mydata.size= v2s32(100,100);
mydata.screensize = screensize; mydata.screensize = screensize;
@ -2767,6 +2787,7 @@ void GUIFormSpecMenu::regenerateGui(v2u32 screensize)
m_dropdowns.clear(); m_dropdowns.clear();
theme_by_name.clear(); theme_by_name.clear();
theme_by_type.clear(); theme_by_type.clear();
m_clickthrough_elements.clear();
m_bgnonfullscreen = true; m_bgnonfullscreen = true;
m_bgfullscreen = false; m_bgfullscreen = false;
@ -3248,12 +3269,19 @@ void GUIFormSpecMenu::drawMenu()
e->setVisible(false); e->setVisible(false);
} }
// Some elements are only visible while being drawn
for (gui::IGUIElement *e : m_clickthrough_elements)
e->setVisible(true);
/* /*
Call base class Call base class
(This is where all the drawing happens.) (This is where all the drawing happens.)
*/ */
gui::IGUIElement::draw(); gui::IGUIElement::draw();
for (gui::IGUIElement *e : m_clickthrough_elements)
e->setVisible(false);
// Draw hovered item tooltips // Draw hovered item tooltips
for (const std::string &tooltip : m_hovered_item_tooltips) { for (const std::string &tooltip : m_hovered_item_tooltips) {
showTooltip(utf8_to_wide(tooltip), m_default_tooltip_color, showTooltip(utf8_to_wide(tooltip), m_default_tooltip_color,

@ -307,6 +307,7 @@ protected:
std::vector<std::pair<gui::IGUIElement *, TooltipSpec>> m_tooltip_rects; std::vector<std::pair<gui::IGUIElement *, TooltipSpec>> m_tooltip_rects;
std::vector<std::pair<FieldSpec, GUIScrollBar *>> m_scrollbars; std::vector<std::pair<FieldSpec, GUIScrollBar *>> m_scrollbars;
std::vector<std::pair<FieldSpec, std::vector<std::string>>> m_dropdowns; std::vector<std::pair<FieldSpec, std::vector<std::string>>> m_dropdowns;
std::vector<gui::IGUIElement *> m_clickthrough_elements;
GUIInventoryList::ItemSpec *m_selected_item = nullptr; GUIInventoryList::ItemSpec *m_selected_item = nullptr;
u16 m_selected_amount = 0; u16 m_selected_amount = 0;