3

我对 QGroupBox 类进行了子类化,并启用了 checkable 属性。我正在尝试覆盖切换/检查事件的行为。

这是代码:

class SideWidgetGroupBox: public QGroupBox
{
    Q_OBJECT
public:
    SideWidgetGroupBox(QWidget* parent = 0): QGroupBox(parent)
    {
        this->setCheckable(true);
        connect(this, SIGNAL(toggled(bool)), this, SLOT(my_toggled(bool)));
    }

private slots:
    void my_toggled (bool on)
    {
        std::cout << "my toggled method" <<std::endl;
    }
};

到目前为止一切顺利,我的插槽被执行。然而,组框的内容也被启用/禁用。有没有办法防止这种情况?还是我必须手动重置原始启用/禁用状态?

4

1 回答 1

6

有没有办法防止启用/禁用内容?

是的,但这种方式并不容易,因为那里没有QCheckBox。看起来像复选框的东西是QGroupBox. 并且所有事件都通过以下方式处理QGroupBox
1.覆盖event方法并防止基类处理QEvent::KeyReleaseQEvent::MouseRelease事件。

bool SideWidgetGroupBox::event(QEvent *e)
{
  switch (e->type()) {
   case QEvent::KeyRelease:
   case QEvent::MouseButtonRelease:
     myHandler(e);
     return true;
  }
 return QGroupBox::event(e);
}

2.myHandler检查是否space按下或鼠标单击复选框。存储复选框值并执行您需要的操作。使用此代码检查光标下的内容:

QStyleOptionGroupBox box;
initStyleOption(&box);
QStyle::SubControl released = style()->hitTestComplexControl(QStyle::CC_GroupBox, &box,
                                                             event->pos(), this);
bool toggle = released == QStyle::SC_GroupBoxLabel || released == QStyle::SC_GroupBoxCheckBox;
if (toggle)
{
    m_state = !m_state;
    update();
}

3.添加方法initStyleOption并设置state为checkBox的状态(你应该自己存储):

void SideWidgetGroupBox::initStyleOption(QStyleOptionGroupBox *option) const
{
    QGroupBox::initStyleOption(option);
    QStyle::State flagToSet = m_state ? QStyle::State_On : QStyle::State_Off;
    QStyle::State flagToRemove = m_state ? QStyle::State_Off : QStyle::State_On;

    option->state |= flagToSet;     
    option->state &= ~flagToRemove;
    option->state &= ~QStyle::State_Sunken;
}

4.方法initStyleOptioninQGroupBox不是虚拟的,这就是为什么您还需要重新实现paintEvent

void paintEvent(QPaintEvent *)
{
    QStylePainter paint(this);
    QStyleOptionGroupBox option;
    initStyleOption(&option);
    paint.drawComplexControl(QStyle::CC_GroupBox, option);
}

我是否必须手动重置原始启用/禁用状态?
您不能这样做,setEnabled因为它会检查当前的检查状态并阻止启用子级。虽然您可以setEnabled直接使用this->findChildren<QWidget*>

建议
您可以使用上述方法或删除标准复选框和(或)标签并将自己的QCheckBox放在组上(当然没有布局)并根据需要使用它。如果您的组可以移动,您还需要移动该复选框。

于 2014-08-29T07:47:12.690 回答