如何在 QWidgetAction 的子类中设置小部件的可见性

How to set visibility on widgets within a subclass of QWidgetAction

本文关键字:设置 小部 可见性 子类 QWidgetAction      更新时间:2023-10-16

Qt 5.6.3, eglfs Linux platform.

我有一系列从QWidgetAction派生的类。QWidgetActions 都是从一个菜单父级的,它们包含的小部件是从同一个菜单父级的。包含的小部件都设置为QWidgetAction的默认小部件。QWidgetAction没有重新实现任何内容。

我以为设置 QWidgetAction 的可见性会自动设置其中包含的自定义小部件集的可见性?这不是真的吗,因为这样做肯定不会根据需要显示和隐藏小部件!?我必须执行其他操作才能将可见性更改传递给包含的小部件吗?我必须直接从QWidgetAction请求小部件,然后直接对其应用可见性(这似乎是一种黑客)吗?

我对QWidgetActions应该如何实现感兴趣。文档几乎不存在,所以我追求人们与他们的经验一样多。我遇到了间歇性的问题,看起来像是自定义小部件的双重删除,并且可见性无法正常运行。

class Base : public QWidgetAction
{
Q_OBJECT
public:
explicit Base(QWidget* parent, QString labelText = "", QString iconPath = "", Qt::AlignmentFlag alignment = Qt::AlignHCenter) :
QWidgetAction(parent),
mCustomWidget(nullptr),
mParentWidget(nullptr),
mTextLabel(nullptr),
mAlignment(alignment),
mLabelText(labelText),
mIconPath(iconPath) {}
virtual ~Base() {}
protected:
QWidget *mCustomWidget;
QWidget *createTheWidgetSet(QWidget *parent)
{
if (mParentWidget == nullptr) {
mParentWidget = new QWidget(parent);
mCustomWidget = createCustomWidget(mParentWidget);
if (mCustomWidget != nullptr) {
if (!mLabelText.isEmpty()) {
mCustomWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Ignored);
}
}
int rightMargin = QApplication::style()->pixelMetric(QStyle::PM_SmallIconSize);
QBoxLayout* layout = new QBoxLayout(QBoxLayout::LeftToRight, mParentWidget);
layout->setContentsMargins(1, 2, rightMargin, 2);
if (!mLabelText.isEmpty()) {
QString some_calced_text{};
mTextLabel = new QLabel(some_calced_text, mParentWidget);
layout->addWidget(mTextLabel);
} else {
if(mAlignment == Qt::AlignLeft){
int some_calced_val{20};
layout->addSpacing(some_calced_val);
}
}
if(mAlignment == Qt::AlignRight){
layout->addStretch();
}
layout->addWidget(mCustomWidget);
if(mAlignment == Qt::AlignLeft){
layout->addStretch();
}
}
setDefaultWidget(mParentWidget);
return mCustomWidget;
}
virtual QWidget *createCustomWidget(QWidget *parent) = 0;
private:
Q_DISABLE_COPY(Base)
QWidget *mParentWidget;
QLabel  *mTextLabel;
Qt::AlignmentFlag mAlignment;
QString mLabelText;
QString mIconPath;
};
class SpinBoxActionWidget : public Base
{
Q_OBJECT
public:
explicit SpinBoxActionWidget(QWidget* parent, QString labelText = "", QString iconPath = "") :
Base(parent, labelText, iconPath),
mSpinBox(nullptr)
{
createTheWidgetSet(parent);
}
virtual ~SpinBoxActionWidget() {}
QSpinBox* getSpinBox() const
{
return mSpinBox;
}
protected:
QWidget *createCustomWidget(QWidget *parent) override
{
if (mSpinBox == nullptr) {
mSpinBox = new QSpinBox(parent);
mSpinBox->setFixedHeight(22);
}
return mSpinBox;
}
private:
Q_DISABLE_COPY(SpinBoxActionWidget)
QSpinBox *mSpinBox;
};
/* Elsewhere in code.... */
{
QMenu theMenu = new QMenu(parentWindow);
SpinBoxActionWidget theAct = new SpinBoxActionWidget(theMenu);
SpinBoxActionWidget theSecondAct = new SpinBoxActionWidget(theMenu);
theMenu->addAction(theAct);
theMenu->addAction(theSecondAct);
/* I now assume that I can do this, and the entire entry in the menu
* represented by "theAct" can be made visible and invisible.
* This doesn't work however, either the widget remains visible,
* or is partially hidden.
theAct->setVisible(true);
theAct->setVisible(false);
*/
}

您没有重新实现接口,这就是它不起作用的原因。

首先,请注意,QWidgetAction派生自不是QWidgetQAction;但是,它确实有一个setVisible()函数,它实际上只是将调用转发到操作创建的所有小部件。

你必须重新实现QWidgetAction::createWidget(parent)来添加新的小部件;你的createCustomWidget没有做任何有用的事情。这是一个非常简单的例子:

class SpinAction : public QWidgetAction
{
Q_OBJECT
public:
SpinAction(QObject* parent) : QWidgetAction(parent) {}
virtual ~SpinAction() {}
QWidget* createWidget(QWidget* parent) { return new QSpinBox(parent); }
// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
// reimplement this function
};

您可以将操作添加到所需的任何容器,菜单,工具栏等...此示例将为每个容器创建一个新小组件,并且这些创建的小组件将不会同步(例如在微调框值上)。

我刚刚在主窗口中对其进行了测试,将小部件操作添加到菜单和工具栏中,并且调用setVisible()完美无缺。