2

我在嵌入式设备上使用 qt 小部件并且虚拟键盘有问题。键盘显示为全屏并覆盖所有应用程序。

在Yocto 中的虚拟键盘顶部黑屏文章中描述了如何解决此问题。

总之,你需要用键盘找到QQuickWindow,并在这个窗口上调用setMask。那么键盘上方的区域就会是透明的

我有问题如何使用虚拟键盘找到 QQuickWindow。我试着用

QApplication::allWidgets()

但窗户不在这里。

4

3 回答 3

4

要获得所有可以使用的窗口,QGuiApplication::allWindows()但这还不够,因为 QtVirtualKeyboard 窗口不一定在开始时创建,因此必须使用 QInputMethod 的 visibleChanged 信号。我没有使用来自 QQuickWindow 的信息进行过滤,因为通常应用程序可能有其他信息,而是使用窗口所属类的名称。

#include <QApplication>
#include <QWindow>
#include <cstring>

static void handleVisibleChanged(){
    if (!QGuiApplication::inputMethod()->isVisible())
        return;
    for(QWindow * w: QGuiApplication::allWindows()){
        if(std::strcmp(w->metaObject()->className(), "QtVirtualKeyboard::InputView") == 0){
            if(QObject *keyboard = w->findChild<QObject *>("keyboard")){
                QRect r = w->geometry();
                r.moveTop(keyboard->property("y").toDouble());
                w->setMask(r);
                return;
            }
        }
    }
}

int main(int argc, char *argv[])
{
    qputenv("QT_IM_MODULE", QByteArray("qtvirtualkeyboard"));
    QApplication a(argc, argv);
    QObject::connect(QGuiApplication::inputMethod(), &QInputMethod::visibleChanged, &handleVisibleChanged);
    // ...

蟒蛇版本:

import os
import sys

from PySide2 import QtCore, QtGui, QtWidgets
# from PyQt5 import QtCore, QtGui, QtWidgets


def handleVisibleChanged():
    if not QtGui.QGuiApplication.inputMethod().isVisible():
        return
    for w in QtGui.QGuiApplication.allWindows():
        if w.metaObject().className() == "QtVirtualKeyboard::InputView":
            keyboard = w.findChild(QtCore.QObject, "keyboard")
            if keyboard is not None:
                r = w.geometry()
                r.moveTop(keyboard.property("y"))
                w.setMask(QtGui.QRegion(r))
                return


def main():
    os.environ["QT_IM_MODULE"] = "qtvirtualkeyboard"
    app = QtWidgets.QApplication(sys.argv)

    QtGui.QGuiApplication.inputMethod().visibleChanged.connect(handleVisibleChanged)

    w = QtWidgets.QLineEdit()
    w.show()
    sys.exit(app.exec_())


if __name__ == "__main__":
    main()
于 2020-09-18T22:56:36.143 回答
0

您可以findChildren与任何继承的类一起使用,QObject例如QApplication. 例如在main.cpp中:

QApplication a(argc, argv);
QList<QQuickWindow *> wind = a.findChildren<QQuickWindow *>();

QQuickWindow这将为您提供指向应用程序中所有内容的指针列表。

于 2020-09-18T14:36:09.877 回答
0

我为使用 Raspberry 和 PyQt5 的人提供了一个解决方案。完成eyllanesc的回答是因为Python版本不支持PyQt5,其实我们有这个问题:

TypeError: setMask(self, QRegion): 参数 1 有意外类型 'QRect'

要解决它:

def handleVisibleChanged():
    if not QtGui.QGuiApplication.inputMethod().isVisible():
        return
    for w in QtGui.QGuiApplication.allWindows():
        if w.metaObject().className() == "QtVirtualKeyboard::InputView":
            keyboard = w.findChild(QtCore.QObject, "keyboard")
            if keyboard is not None:
                region = w.mask()
                rect = [w.geometry()]
                rect[0].moveTop(keyboard.property("y"))
                region.setRects(rect)
                w.setMask(region)
                return
于 2021-04-19T16:41:44.493 回答