如何阻止回车键关闭我的QDialog(Qt 4.8.1)

时间:2013-04-06 00:05:42

标签: c++ qt qdialog qpushbutton

我有QDialog QDialogButtonBox确定取消按钮处于活动状态。有时我会根据对话框的状态禁用或隐藏“确定”按钮。看来,无论我做什么,Enter键总是激活OK按钮。我真的不希望这种情况发生。我试过了:

  • 每次显示/隐藏/启用/禁用/无论按钮
  • 时,将default和autoDefault属性设置为false
  • 在OK按钮上安装事件过滤器以拦截返回,输入和空格的按键事件(按下和释放)
  • 将按钮上的焦点政策设置为 NoFocus

使用上述所有组合,Enter键仍然接受对话框。有没有人知道如何阻止这个?看起来我应该能够阻止这么简单的事情吗?

9 个答案:

答案 0 :(得分:11)

按键事件过滤应该在对话框本身上完成,因为处理ReturnEnter键转发到默认按钮的代码在QDialog::keyPressEvent

void Dialog::keyPressEvent(QKeyEvent *evt)
{
    if(evt->key() == Qt::Key_Enter || evt->key() == Qt::Key_Return)
        return;
    QDialog::keyPressEvent(evt);
}

theDialog−>installEventFilter(anotherClassObject);

bool AnotherClass::eventFilter(QObject *obj, QEvent *evt)
{
    if(evt->type() == QEvent::KeyPress) {
        QKeyEvent *keyEvent = static_cast<QKeyEvent*>(evt);
        if(keyEvent->key() == Qt::Key_Enter || keyEvent->key() == Qt::Key_Return )
            return true; // mark the event as handled
    }
    return false;
}

答案 1 :(得分:3)

如果对话框中有正常的QPushButtons,那么如果按钮上设置了autoDefault和/或默认属性,那么您将获得一个默认按钮 - 这是enter键触发的按钮。在这种情况下,摆脱按钮上的autoDefault并按下另一个小部件中的enter键不再关闭对话框。

在QDialogBu​​ttonBox的情况下,您可以迭代按钮以在对话框的ctor中关闭这些内容。没有在这里测试,但应该工作。如果没有,那么您还需要查看是否在QDialog本身上设置了默认按钮。

答案 2 :(得分:1)

QDialog有一个名为accept()的私人广告位。每当QDialogBu​​ttonBox发出accepted()时(按返回键或单击“确定”),就会调用该专用槽。所以尝试断开它们。

disconnect(ui->buttonBox, SIGNAL(accepted()), this, SLOT(accept()));

这对我有用。

答案 3 :(得分:1)

问题是不应在OK按钮上安装事件过滤器。

如果您的确定按钮被禁用,那么它将不会收到输入事件。无论哪个小部件都有焦点。如果他们不接受输入事件,则QDialog将转到accept()本身。

解决问题的两种方式

1)覆盖QDialog::accept(),只有在启用确定后才会在新QDialog功能中调用accept的接受方法

void MyDialog::accept() {
    if (okEnabled) {
        QDialog::accept();
    }
}

2)对话框中的每个小部件上安装一个不接受回车键的事件过滤器(行编辑,...)。

事件过滤器就像这样:

class KeyPressEater : public QObject
{
    Q_OBJECT

protected:
    bool eventFilter(QObject *obj, QEvent *event);
};

bool KeyPressEater::eventFilter(QObject *obj, QEvent *event)
{
    if (event->type() == QEvent::KeyPress) {
        QKeyEvent *keyEvent = static_cast<QKeyEvent *>(event);
        bool res = QObject::eventFilter(obj, event);

        if (keyEvent->key() == Qt::Key_Return) {
            return true; /* Always accept return */
        } else {
            return res;
        }
    } else {
        // standard event processing
        return QObject::eventFilter(obj, event);
    }
}

在您的代码中,对话框中的每个小部件:

myWidget->installEventFilter(myKeyPressEater);

答案 4 :(得分:0)

避免&#34; OK&#34;按钮或&#34;输入&#34;关闭对话框中的键: 在ui xml文件中,删除接受/拒绝的连接/插槽。然后,在您的代码中,根据需要发出accept();

连接accept()插槽的ui文件示例:

 <connections>  
  <connection>  

   <sender>products_ButtonBox</sender>

    <signal>accepted()</signal>  

     <receiver>Products_Dialog</receiver>
      <slot>accept()</slot>
       <hints>
        <hint type="sourcelabel">
         <x>248</x>
         <y>254</y>
        </hint>
        <hint type="destinationlabel">
         <x>157</x>
         <y>274</y>
        </hint>e
       </hints>
      </connection>

答案 5 :(得分:0)

在对话框的accept()方法中,检查“确定”按钮以获得焦点:

void accept() override
{ if (!dialogButtonBox->button(QDialogButtonBox::Ok)->hasFocus())
  return;

...

  QDialog::accept();
}

答案 6 :(得分:0)

关键是设置自己的按钮,全部使用NoRole,并且既不接受也不拒绝来自按钮箱的信号。这将允许您为默认按钮设置自己的行为。

class Dialog(QDialog):
    def __init__():
        super(Dialog, self).__init__()
        self.buttonBox = QDialogButtonBox()

        self.btn_save = self.buttonBox.addButton('Save', QDialogButtonBox.NoRole)
        self.btn_cancel = self.buttonBox.addButton('Cancel', QDialogButtonBox.NoRole)
        self.btn_save.clicked.connect(self.onAccept)
        self.btn_save.setMouseTracking(True)
        self.btn_cancel.clicked.connect(self.onReject)

        #  STATUS BAR
        self.status = QStatusBar()
        self.status.addPermanentWidget(self.buttonBox)

    def onAccept(self):
        if not self.btn_save.underMouse():
            return
        self.submitChanges(self.model)
        self.accept()

    def onReject(self):
        self.reject()

答案 7 :(得分:0)

一个选项是覆盖对话框的显示事件以允许显示 QDialogBu​​ttonBox,之后它将设置一个带有 AcceptRole 的默认按钮,然后将所有按钮设置为非默认按钮。

void MyDialog::showEvent(QShowEvent* event)
{
    // When a QDialogButtonBox is shown, it will set a default button if none are found so we need to disable the
    // default buttons after the button box has been shown.
    QDialog::showEvent(event);

    // For example, with a dialog which has two buttons, Save and Cancel, we remove all defaults
    // It might be good enough to remove the default on just the buttons with have the AcceptRole, but
    // I didn't test extensively enough to see what would happen if any buttons had "autoDefault" set or
    // verify this behavior on all platforms.
    ui->buttonBox->button(QDialogButtonBox::Save)->setDefault(false);
    ui->buttonBox->button(QDialogButtonBox::Cancel)->setDefault(false);
}

在显示 QDialogBu​​ttonBox 之前尝试删除默认值,例如在对话框的构造函数中,只会被 QDialogBu​​ttonBox::showEvent() 中的代码覆盖。

答案 8 :(得分:-1)

在PySide中(我想象PyQt)我能够重新定义QDialog的接受和拒绝功能。

def custom_accept ():
    # perform custom actions when you hit open
    pass

def custom_reject ():
    # perform custom actions when you hit cancel
    pass

file_dialog = QtGui.QFileDialog(directory=".")
file_dialog.accept = custom_accept
file_dialog.reject = custom_reject

这使文件对话框无法关闭,并在触发'ok'(接受)或'取消'(拒绝)功能时(通过输入或单击按钮)让我访问数据