QShortCut与QSpinBox发生冲突

时间:2011-04-12 18:18:49

标签: qt qt4

我正在编写一个应用程序,我使用自己的快捷方式。它看起来像这样:

myShortcut= new QShortcut(Qt::SHIFT + Qt::Key_B,this);
connect(myShortcut, SIGNAL(activated()), this, SLOT(setCameraBack()));

我在主窗口小部件的构造函数中定义了它,它工作正常,直到我点击其中一个也位于主窗口小部件上的旋转框按钮。之后我的快捷方式停止工作,直到我点击按钮或复选框后它才能工作。当我这样做时,一切都很好。我想补充说,点击旋转框后,它似乎是“活动的”(因为光标仍然在其上“闪烁”),直到我点击其他按钮之一。你知道什么是错的吗?是某种过程或事件问题吗?谢谢你的所有答案 〜Marwroc

3 个答案:

答案 0 :(得分:4)

  

捷径是由Qt's“听取”的   事件循环时快捷方式的父级   小部件正在接收事件。

当QSpinBox有keyboard focus时,QShortcut对象的父对象不再接收事件。因此,在从QSpinBox中删除keyboard focus之前,快捷方式不起作用。您可以通过将Qt::WidgetWithChildrenShortcut or Qt::ApplicationShortcut传递给QShortcut的QShortcut::setContext方法来更改此行为。

答案 1 :(得分:2)

在激活快捷方式之前,焦点窗口小部件会被赋予ShortcutOverride事件。如果事件被接受,则键事件将传递给窗口小部件,并且不会激活快捷方式。

来源:https://wiki.qt.io/ShortcutOverride

关注Qt源

QAbstractSpinBox::event(QEvent *event)
{
    Q_D(QAbstractSpinBox);
    switch (event->type()) {
    ...
    case QEvent::ShortcutOverride:
        if (d->edit->event(event))
            return true;
        break;
    ...
    }
    return QWidget::event(event);
}

QAbstractSpinBox允许内部编辑接受该事件。 QLineEdit遵循QLineControl。来自qt / src / gui / widgets / qlinecontrol.cpp

    case QEvent::ShortcutOverride:{
        if (isReadOnly())
            return false;
        QKeyEvent* ke = static_cast<QKeyEvent*>(ev);
        if (ke == QKeySequence::Copy
            || ke == QKeySequence::Paste
            || ke == QKeySequence::Cut
            || ke == QKeySequence::Redo
            || ke == QKeySequence::Undo
            || ke == QKeySequence::MoveToNextWord
            || ke == QKeySequence::MoveToPreviousWord
            || ke == QKeySequence::MoveToStartOfDocument
            || ke == QKeySequence::MoveToEndOfDocument
            || ke == QKeySequence::SelectNextWord
            || ke == QKeySequence::SelectPreviousWord
            || ke == QKeySequence::SelectStartOfLine
            || ke == QKeySequence::SelectEndOfLine
            || ke == QKeySequence::SelectStartOfBlock
            || ke == QKeySequence::SelectEndOfBlock
            || ke == QKeySequence::SelectStartOfDocument
            || ke == QKeySequence::SelectAll
            || ke == QKeySequence::SelectEndOfDocument) {
            ke->accept();
        } else if (ke->modifiers() == Qt::NoModifier || ke->modifiers() == Qt::ShiftModifier
                   || ke->modifiers() == Qt::KeypadModifier) {
            if (ke->key() < Qt::Key_Escape) {
                ke->accept();
            } else {
                switch (ke->key()) {
                case Qt::Key_Delete:
                case Qt::Key_Home:
                case Qt::Key_End:
                case Qt::Key_Backspace:
                case Qt::Key_Left:
                case Qt::Key_Right:
                    ke->accept();
                default:
                    break;
                }
            }
        }
    }

如果未按下控制键,此代码接受大多数键。

所以最简单的解决方案是更改快捷方式以包含控件修饰符。

或者,您可以子类化旋转框并覆盖事件函数

bool MySpinBox::event(QEvent *event)
{
    if( event->type() == QEvent::ShortcutOverride && !isReadOnly() )
    {
        QKeyEvent *keyEvent = static_cast<QKeyEvent*>(event);
        // Ignore 'B' shortcuts
        if( keyEvent->key() == Qt::Key_B )
        {
            Q_ASSERT( !event->isAccepted() );
            return true;
    }
    return QSpinBox::event(event);
}

答案 2 :(得分:1)

您是否尝试过MySpinBox -> setFocusPolicy (Qt::NoFocus)