首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >使浮动QDockWidget不可聚焦

使浮动QDockWidget不可聚焦
EN

Stack Overflow用户
提问于 2013-09-01 12:59:59
回答 4查看 4.2K关注 0票数 2

我在QDockWidget中创建一个简单的虚拟键盘.

当小部件停靠到QMainWindow中时,选中的小部件(例如,qdoublespinbox)将被高亮显示,如果我单击虚拟键盘clearFocus(),则工作.

当QDockWidget在窗口上方浮动时,我单击一个按钮,clearFocus无法工作,并且在QMainWindow中看不到焦点小部件.

我怎么能强迫QDockWidget一点注意力都没有呢?

谢谢:-)

这是代码:

代码语言:javascript
复制
// class MyVirtualKeyboard : public QDockWidget

void MyVirtualKeyboard::sendKey(Qt::Key key, Qt::KeyboardModifier mod)
{
    this->clearFocus();

    QMainWindow *w = dynamic_cast<QMainWindow *>(this->parent());
    if(w == NULL) return;

    QWidget *widget = w->focusWidget();

    QString repr = QKeySequence(key).toString();

    QKeyEvent *pressEvent = new QKeyEvent(QEvent::KeyPress, key, mod, repr);
    QKeyEvent *releaseEvent = new QKeyEvent(QEvent::KeyRelease, key, mod, repr);

    qDebug("%s", pressEvent->text().toAscii().data());

    MyApplication *app = MyApplication::myInstance();

    app->postEvent(widget, pressEvent);
    app->postEvent(widget, releaseEvent);
}

void MyVirtualKeyboard::on_BTN_1_clicked()
{
    sendKey(Qt::Key_1);
}

...
EN

回答 4

Stack Overflow用户

回答已采纳

发布于 2013-09-05 10:06:36

我想我找到更好的方法了!只需使用this->setAttribute(Qt::WA_X11DoNotAcceptFocus);和瞧!

示例:

代码语言:javascript
复制
MyVirtualKeyboard::MyVirtualKeyboard(QWidget *parent) :
    QDockWidget(parent),
    ui(new Ui::MyVirtualKeyboard)
{
    ui->setupUi(this);
    this->setAttribute(Qt::WA_X11DoNotAcceptFocus);
}
票数 0
EN

Stack Overflow用户

发布于 2013-09-02 16:29:36

clearFocus()调用应该是不必要的。您的坞小部件及其所有部件都必须有Qt::NoFocus策略。

下面的代码显示了您可能如何做到这一点。

代码语言:javascript
复制
// https://github.com/KubaO/stackoverflown/tree/master/questions/vkb-focus-18558664
#include <QtGui>
#if QT_VERSION >= QT_VERSION_CHECK(5,0,0)
#include <QtWidgets>
#endif

class Keyboard : public QDockWidget {
   Q_OBJECT
   QWidget m_widget;
   QGridLayout m_layout{&m_widget};
   QToolButton m_buttons[10];
   void sendKey(Qt::Key key, Qt::KeyboardModifier mod)
   {
      if (! parentWidget()) return;
      auto target = parentWidget()->focusWidget();
      if (! target) return;

      auto repr = QKeySequence(key).toString();
      auto pressEvent = new QKeyEvent(QEvent::KeyPress, key, mod, repr);
      auto releaseEvent = new QKeyEvent(QEvent::KeyRelease, key, mod, repr);
      qApp->postEvent(target, pressEvent);
      qApp->postEvent(target, releaseEvent);
      qDebug() << repr;
   }
   Q_SLOT void clicked() {
      auto key = sender()->property("key");
      if (key.isValid()) sendKey((Qt::Key)key.toInt(), Qt::NoModifier);
   }
public:
   explicit Keyboard(const QString & title, QWidget *parent = nullptr) : Keyboard(parent) {
      setWindowTitle(title);
   }
   explicit Keyboard(QWidget *parent = nullptr) : QDockWidget(parent) {
      int i{};
      for (auto & btn : m_buttons) {
         btn.setText(QString::number(i));
         btn.setProperty("key", Qt::Key_0 + i);
         m_layout.addWidget(&btn, 0, i, 1, 1);
         connect(&btn, SIGNAL(clicked()), SLOT(clicked()));
         btn.setFocusPolicy(Qt::NoFocus);
         ++i;
      }
      setWidget(&m_widget);
      setFeatures(QDockWidget::DockWidgetFloatable | QDockWidget::DockWidgetMovable);
   }
};

int main(int argc, char ** argv)
{
   QApplication a(argc, argv);
   QMainWindow w;
   w.setCentralWidget(new QLineEdit);
   w.addDockWidget(Qt::TopDockWidgetArea, new Keyboard("Keyboard", &w));
   w.show();
   return a.exec();
}

#include "main.moc"
票数 2
EN

Stack Overflow用户

发布于 2013-09-02 16:37:20

您可以通过设置QWidget::focusPolicy = Qt::NoFocus来防止小部件集中注意力。

但是,这里有两个你正在混合的概念--聚焦控件(每个窗口)和活动窗口(每个桌面)。我认为在您所描述的场景中(一个撕开的弹出式窗口),即使Qt没有设置焦点控件,OS窗口管理器仍然可能会更改活动的顶层窗口。这将导致没有键盘焦点(这是一个有效的状态!)。

因此,我认为,你的问题的完整答案将涉及一些不可携带的比特。我不知道您在哪个GUI环境中工作,但我知道Win32的一些答案,所以我将继续下去,希望这是有用的:

Win32

在Win32的文章中,对MSDN上的状态跟踪进行了很好的讨论。我不知道Qt做了什么来包装这个级别,所以您必须使用QWidget::nativeEventQCoreApplication::installNativeEventFilter来获取低级别的事件。如果您可以对窗口进行子类化,我更喜欢前者,因为它更独立。

代码语言:javascript
复制
bool FooWidget::nativeEvent(const QByteArray & eventType, void * message, long * result)
{
#ifdef Q_OS_WIN
    if(eventType == "windows_generic_MSG") {
        const MSG *msg = reinterpret_cast<MSG *>(message);
        if(msg->message == WM_MOUSEACTIVATE) {
            *result = MA_NOACTIVATE;
            return true;
        }
    }
#else
    #error Need platform-specific code to suppress click-activation
#endif
    return false;
}

这将阻止单击激活窗口(MA_NOACTIVATE),阻止Qt进一步处理它(return true),同时将其他所有事件(包括单击,因为我们也没有使用MA_NOACTIVATEANDEAT来阻止它)正常处理为QEvents和Qt信号(最后是return false)。

如果您需要更低级别的访问(尽管我不认为您会),请参见QWidget::effectiveWinId()QWidget::windowHandle

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/18558664

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档