我在QDockWidget中创建一个简单的虚拟键盘.
当小部件停靠到QMainWindow中时,选中的小部件(例如,qdoublespinbox)将被高亮显示,如果我单击虚拟键盘clearFocus(),则工作.
当QDockWidget在窗口上方浮动时,我单击一个按钮,clearFocus无法工作,并且在QMainWindow中看不到焦点小部件.
我怎么能强迫QDockWidget一点注意力都没有呢?
谢谢:-)
这是代码:
// 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);
}
...发布于 2013-09-05 10:06:36
我想我找到更好的方法了!只需使用this->setAttribute(Qt::WA_X11DoNotAcceptFocus);和瞧!
示例:
MyVirtualKeyboard::MyVirtualKeyboard(QWidget *parent) :
QDockWidget(parent),
ui(new Ui::MyVirtualKeyboard)
{
ui->setupUi(this);
this->setAttribute(Qt::WA_X11DoNotAcceptFocus);
}发布于 2013-09-02 16:29:36
clearFocus()调用应该是不必要的。您的坞小部件及其所有部件都必须有Qt::NoFocus策略。
下面的代码显示了您可能如何做到这一点。

// 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"发布于 2013-09-02 16:37:20
您可以通过设置QWidget::focusPolicy = Qt::NoFocus来防止小部件集中注意力。
但是,这里有两个你正在混合的概念--聚焦控件(每个窗口)和活动窗口(每个桌面)。我认为在您所描述的场景中(一个撕开的弹出式窗口),即使Qt没有设置焦点控件,OS窗口管理器仍然可能会更改活动的顶层窗口。这将导致没有键盘焦点(这是一个有效的状态!)。
因此,我认为,你的问题的完整答案将涉及一些不可携带的比特。我不知道您在哪个GUI环境中工作,但我知道Win32的一些答案,所以我将继续下去,希望这是有用的:
Win32
在Win32的文章中,对MSDN上的状态跟踪进行了很好的讨论。我不知道Qt做了什么来包装这个级别,所以您必须使用QWidget::nativeEvent或QCoreApplication::installNativeEventFilter来获取低级别的事件。如果您可以对窗口进行子类化,我更喜欢前者,因为它更独立。
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。
https://stackoverflow.com/questions/18558664
复制相似问题