QKeyEvent nativeVirtualKey tanımsız
QML uygulamamda bir anahtarın yerel anahtar koduna ihtiyacım var. Bir QML öğesinde aşağıdaki anahtar işleyiciye sahibim.
Keys.onPressed: {
console.log("Key: ", event.key)
console.log("Native: ", event.nativeVirtualKey);
event.accepted = true
}
event.keyTuşlarına basarak, ama ne zaman cezası çalışır event.nativeVirtualKeytanımsız edildi. Örneğin.
qml: Key: 70
qml: Native: undefined
Kodumda bir sorun mu var? NativeVirtualKey'i nasıl edinebilirim?
Artık dokümantasyonda "Not: Anahtar olayı genişletilmiş bilgi içerse bile yerel sanal anahtar 0 olabilir" ifadesini görüyorum. https://doc.qt.io/qt-5/qkeyevent.html#nativeVirtualKey Maalesef, sanal yerel anahtarın ne zaman veya hangi koşulların kaybolmasına neden olduğu konusunda hiçbir açıklama yoktur.
Yanıtlar
Zaten içinde belirttiği gibi bu cevap : KeyEventdeğil bir QKeyEventama bazı özelliklerini ancak tümü gösterir bir QObject. Bir geçici çözüm, öğeye bir olay filtresi yükleyen ve bu özelliği ortaya çıkaran bir QObject oluşturmaktır:
#include <QGuiApplication>
#include <QQmlApplicationEngine>
#include <QQuickItem>
class KeyHelper: public QObject{
Q_OBJECT
Q_PROPERTY(QQuickItem* target READ target WRITE setTarget NOTIFY targetChanged)
public:
using QObject::QObject;
QQuickItem* target() const {
return m_target;
}
void setTarget(QQuickItem* item){
if(m_target)
m_target->removeEventFilter(this);
m_target = item;
if(m_target)
m_target->installEventFilter(this);
Q_EMIT targetChanged(m_target);
}
bool eventFilter(QObject *watched, QEvent *event){
if(watched == m_target && event->type() == QEvent::KeyPress){
if(QKeyEvent *ke = static_cast<QKeyEvent *>(event))
Q_EMIT nativeVirtualKeyChanged(ke->nativeVirtualKey());
}
return QObject::eventFilter(watched, event);
}
signals:
void nativeVirtualKeyChanged(quint32 nativeVirtualKey);
void targetChanged(QQuickItem* item);
private:
QPointer<QQuickItem> m_target;
};
int main(int argc, char *argv[])
{
QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
QGuiApplication app(argc, argv);
qmlRegisterType<KeyHelper>("qt.keyhelper", 1, 0, "KeyHelper");
QQmlApplicationEngine engine;
const QUrl url(QStringLiteral("qrc:/main.qml"));
QObject::connect(&engine, &QQmlApplicationEngine::objectCreated,
&app, [url](QObject *obj, const QUrl &objUrl) {
if (!obj && url == objUrl)
QCoreApplication::exit(-1);
}, Qt::QueuedConnection);
engine.load(url);
return app.exec();
}
#include "main.moc"
import QtQuick 2.12
import QtQuick.Window 2.12
import qt.keyhelper 1.0
Window {
width: 640
height: 480
visible: true
title: qsTr("Hello World")
Item {
id: item
focus: true
anchors.fill: parent
}
KeyHelper{
target: item
onNativeVirtualKeyChanged: console.log(nativeVirtualKey)
}
}
Burada, @ eyllanesc'in yanıtında küçük bir ayarlama var eventFilterve QEvent'i QVariantMap olarak yeniden paketlemek için değiştiriliyor , böylece KeyEventAPI'yi bekleyen işlevlerle doğrudan çağrılabiliyor .
class KeyHelper: public QObject{
... // same as @ellyanesc's answer
bool eventFilter(QObject *watched, QEvent *event){
if(watched == m_target && event->type() == QEvent::KeyPress){
if(QKeyEvent *ke = static_cast<QKeyEvent *>(event)) {
// it seems we cannot send the event in a signal since it doesnt inherit from QObject.
// copy the relevant values to an event object
QVariantMap eventObject;
eventObject["key"] = ke->key();
eventObject["modifiers"] = int(ke->modifiers());
eventObject["nativeVirtualKey"] = ke->nativeVirtualKey();
eventObject["nativeModifiers"] = ke->nativeModifiers();
Q_EMIT nativeKeyPress(eventObject);
}
}
return QObject::eventFilter(watched, event);
}
signals:
void nativeKeyPress(QVariantMap event);
... // same as @ellyanesc's answer
};
Ve sonra qml dosyasında
KeyHelper{
target: ...
onNativeKeyPress: {
console.log("native key press ", JSON.stringify(event))
}
}