summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMichael Weghorn <m.weghorn@posteo.de>2024-12-18 20:46:32 +0100
committerMichael Weghorn <m.weghorn@posteo.de>2024-12-19 08:28:27 +0100
commit111edc82743369fd1557784566a46c8e03dc4bbd (patch)
tree448b9b907d2165fe01d9a3454fd3212ee1d3cabd
parentfb3f9d72d102dcfde93d32660ebb59a4bb75a5b3 (diff)
tdf#130857 qt weld: Map ID to to parent QWidget for layouts
QBoxLayout (created for "GtkBox" in .ui files) and QGridLayout (created for "GtkGrid" in .ui files) are QLayout subclasses, not QWidgets. So far, QtBuilder::m_aChildren was containing QObjects that could be either QWidgets or QLayouts. QtInstanceBuilder::weld_container and QtInstanceBuilder::weld_grid were retrieving the corresponding QLayout using QtBuilder::get, all other QtInstanceBuilder::weld_* methods the corresponding QWidget. However, for a "GtkBox" or "GtkGrid", it's not only possible to call QtInstanceBuilder::weld_container or QtInstanceBuilder::weld_grid, but it's also possible to just get a pointer to the weld::Widget subclass by calling QtInstanceBuilder::weld_widget with the ID of the grid/box. So far, this would have resulted in QtInstanceBuilder::weld_widget calling QtBuilder::get<QWidget>, in which case the latter would incorrectly `static_cast` the QLayout to a QWidget. Since commit bf42162fc50d0c6f8e567d8765f8b14b96d7cc50 Author: Michael Weghorn <m.weghorn@posteo.de> Date: Mon Nov 4 19:06:15 2024 +0100 tdf#130857 qt weld: Add extra QWidget parents for layouts , extra parent widgets for the layouts (except for the top-level ones in dialogs) are created. Insert these into QtBuilder::m_aChildren instead and retrieve them by the ID, which allows to uniformly use QWidget. Adjust QtInstanceBuilder::weld_{container,grid} accordingly. (Special handling for top-level layouts in containers can still be added as needed later.) This fixes an assert seen in a WIP branch where suport for the FontFeaturesDialog (open context menu for paragraph in Writer, select "Character" -> "Character" to open the "Character" dialog, in the "Font" tab, click on the "Features" button) is declared. Backtrace, where FontFeaturesDialog::m_xContainer is a weld::Widget referring to the "fontentry" GtkGrid in cui/uiconfig/ui/fontfragment.ui: 1 __pthread_kill_implementation pthread_kill.c 44 0x7ffff789dc8c 2 __pthread_kill_internal pthread_kill.c 78 0x7ffff789dcef 3 __GI_raise raise.c 26 0x7ffff7849c42 4 __GI_abort abort.c 79 0x7ffff78324f0 5 __assert_fail_base assert.c 94 0x7ffff7832418 6 __assert_fail assert.c 103 0x7ffff7842552 7 QtInstanceGrid::set_child_left_attach(weld::Widget&, int)::$_0::operator()() const QtInstanceGrid.cxx 25 0x7fffe438b5e1 8 std::__invoke_impl<void, QtInstanceGrid::set_child_left_attach(weld::Widget&, int)::$_0&>(std::__invoke_other, QtInstanceGrid::set_child_left_attach(weld::Widget&, int)::$_0&) invoke.h 61 0x7fffe438b4f5 9 std::__invoke_r<void, QtInstanceGrid::set_child_left_attach(weld::Widget&, int)::$_0&>(QtInstanceGrid::set_child_left_attach(weld::Widget&, int)::$_0&) invoke.h 111 0x7fffe438b4b5 10 std::_Function_handler<void(), QtInstanceGrid::set_child_left_attach(weld::Widget&, int)::$_0>::_M_invoke std_function.h 290 0x7fffe438b39d 11 std::function<void()>::operator() std_function.h 591 0x7fffe4345aee 12 QtInstance::RunInMainThread QtInstance.cxx 226 0x7fffe433d67a 13 QtInstanceGrid::set_child_left_attach QtInstanceGrid.cxx 23 0x7fffe438aa63 14 cui::FontFeaturesDialog::fillGrid FontFeaturesDialog.cxx 142 0x7fff6db25fb1 15 cui::FontFeaturesDialog::initialize FontFeaturesDialog.cxx 73 0x7fff6db2541d 16 cui::FontFeaturesDialog::FontFeaturesDialog FontFeaturesDialog.cxx 35 0x7fff6db24f61 17 SvxCharNamePage::FontFeatureButtonClicked chardlg.cxx 1154 0x7fff6de651d1 18 SvxCharNamePage::LinkStubFontFeatureButtonClicked chardlg.cxx 1131 0x7fff6de5fa7d 19 Link<weld::Button&, void>::Call link.hxx 111 0x7fffeec54551 20 weld::Button::signal_clicked weld.hxx 1558 0x7fffeec3078c 21 SalInstanceButton::ClickHdl salvtables.cxx 2955 0x7fffeec018ee 22 SalInstanceButton::LinkStubClickHdl salvtables.cxx 2943 0x7fffeec00e2d 23 Link<Button *, void>::Call link.hxx 111 0x7fffee508da1 24 Button::Click()::$_0::operator()() const button.cxx 130 0x7fffee5005f2 ... Change-Id: I7901b9085f887acef7b5505eef0baa792816cd0d Reviewed-on: https://gerrit.libreoffice.org/c/core/+/178752 Tested-by: Jenkins Reviewed-by: Michael Weghorn <m.weghorn@posteo.de>
-rw-r--r--vcl/inc/qt5/QtBuilder.hxx8
-rw-r--r--vcl/qt5/QtBuilder.cxx13
-rw-r--r--vcl/qt5/QtInstanceBuilder.cxx24
3 files changed, 19 insertions, 26 deletions
diff --git a/vcl/inc/qt5/QtBuilder.hxx b/vcl/inc/qt5/QtBuilder.hxx
index aaed05ef434e..db4505902380 100644
--- a/vcl/inc/qt5/QtBuilder.hxx
+++ b/vcl/inc/qt5/QtBuilder.hxx
@@ -27,14 +27,14 @@
class QtBuilder : public WidgetBuilder<QObject, QObject*, QMenu, QMenu*>
{
private:
- QObject* get_by_name(std::u16string_view sID);
+ QWidget* get_by_name(std::u16string_view sID);
struct WinAndId
{
OUString m_sID;
- QObject* m_pWindow;
- WinAndId(OUString sId, QObject* pWindow)
+ QWidget* m_pWidget;
+ WinAndId(OUString sId, QWidget* pWidget)
: m_sID(std::move(sId))
- , m_pWindow(pWindow)
+ , m_pWidget(pWidget)
{
}
};
diff --git a/vcl/qt5/QtBuilder.cxx b/vcl/qt5/QtBuilder.cxx
index 85ad02384437..24accdb4203e 100644
--- a/vcl/qt5/QtBuilder.cxx
+++ b/vcl/qt5/QtBuilder.cxx
@@ -59,12 +59,12 @@ QtBuilder::QtBuilder(QObject* pParent, std::u16string_view sUIRoot, const OUStri
QtBuilder::~QtBuilder() {}
-QObject* QtBuilder::get_by_name(std::u16string_view sID)
+QWidget* QtBuilder::get_by_name(std::u16string_view sID)
{
for (auto const& child : m_aChildren)
{
if (child.m_sID == sID)
- return child.m_pWindow;
+ return child.m_pWidget;
}
return nullptr;
@@ -397,7 +397,8 @@ QObject* QtBuilder::makeObject(QObject* pParent, std::u16string_view sName, std:
pParentGridLayout->addLayout(pLayout, pParentGridLayout->rowCount(), 0);
}
- m_aChildren.emplace_back(sID, pObject);
+ if (pWidget)
+ m_aChildren.emplace_back(sID, pWidget);
return pObject;
}
@@ -492,12 +493,12 @@ void QtBuilder::tweakInsertedChild(QObject* pParent, QObject* pCurrentChild, std
void QtBuilder::setMnemonicWidget(const OUString& rLabelId, const OUString& rMnemonicWidgetId)
{
QLabel* pLabel = get<QLabel>(rLabelId);
- QObject* pBuddy = get_by_name(rMnemonicWidgetId);
+ QWidget* pBuddy = get_by_name(rMnemonicWidgetId);
- if (!pLabel || !pBuddy || !pBuddy->isWidgetType())
+ if (!pLabel || !pBuddy)
return;
- pLabel->setBuddy(static_cast<QWidget*>(pBuddy));
+ pLabel->setBuddy(pBuddy);
}
void QtBuilder::setRadioButtonGroup(const OUString& rRadioButtonId, const OUString& rRadioGroupId)
diff --git a/vcl/qt5/QtInstanceBuilder.cxx b/vcl/qt5/QtInstanceBuilder.cxx
index b76a0ad344d6..b253f53fd6a9 100644
--- a/vcl/qt5/QtInstanceBuilder.cxx
+++ b/vcl/qt5/QtInstanceBuilder.cxx
@@ -136,16 +136,12 @@ std::unique_ptr<weld::Widget> QtInstanceBuilder::weld_widget(const OUString& rId
std::unique_ptr<weld::Container> QtInstanceBuilder::weld_container(const OUString& rId)
{
- QLayout* pLayout = m_xBuilder->get<QLayout>(rId);
- if (!pLayout)
+ QWidget* pWidget = m_xBuilder->get<QWidget>(rId);
+ if (!pWidget)
return nullptr;
- QWidget* pParentWidget = pLayout->parentWidget();
- assert(pParentWidget && pParentWidget == pLayout->parent()
- && "layout has no direct widget parent");
- std::unique_ptr<weld::Container> xRet(
- pParentWidget ? std::make_unique<QtInstanceContainer>(pParentWidget) : nullptr);
- return xRet;
+ assert(pWidget->layout() && "no layout");
+ return std::make_unique<QtInstanceContainer>(pWidget);
}
std::unique_ptr<weld::Box> QtInstanceBuilder::weld_box(const OUString&)
@@ -156,16 +152,12 @@ std::unique_ptr<weld::Box> QtInstanceBuilder::weld_box(const OUString&)
std::unique_ptr<weld::Grid> QtInstanceBuilder::weld_grid(const OUString& rId)
{
- QGridLayout* pLayout = m_xBuilder->get<QGridLayout>(rId);
- if (!pLayout)
+ QWidget* pWidget = m_xBuilder->get<QWidget>(rId);
+ if (!pWidget)
return nullptr;
- QWidget* pParentWidget = pLayout->parentWidget();
- assert(pParentWidget && pParentWidget == pLayout->parent()
- && "layout has no direct widget parent");
- std::unique_ptr<weld::Grid> xRet(pParentWidget ? std::make_unique<QtInstanceGrid>(pParentWidget)
- : nullptr);
- return xRet;
+ assert(qobject_cast<QGridLayout*>(pWidget->layout()) && "no grid layout");
+ return std::make_unique<QtInstanceGrid>(pWidget);
}
std::unique_ptr<weld::Paned> QtInstanceBuilder::weld_paned(const OUString&)