Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qandroidviewsignalmanager.cpp
Go to the documentation of this file.
1// Copyright (C) 2024 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3// Qt-Security score:significant reason:default
4
5#include "platform/android/qandroidviewsignalmanager_p.h"
6#include <QtCore/qcoreapplication.h>
7#include <QtCore/qmetaobject.h>
8#include <QtQuick/qquickitem.h>
9#include <QtQuick/qquickview.h>
10
11#include <QtCore/private/qandroidtypeconverter_p.h>
12#include <QtQuick/private/qandroidviewsignalmanager_p.h>
13
15
16QAndroidViewSignalManager::QAndroidViewSignalManager(QQuickView *view, QObject *parent)
17 : QObject(parent), m_view(view)
18{
19 connect(m_view, &QQuickView::statusChanged, this,
20 &QAndroidViewSignalManager::onViewStatusChanged);
21}
22
23void QAndroidViewSignalManager::onViewStatusChanged(QQuickView::Status status)
24{
25 if (status == QQuickView::Ready) {
26 QObject::disconnect(m_view, &QQuickView::statusChanged, this,
27 &QAndroidViewSignalManager::onViewStatusChanged);
28 QMutexLocker lock(&m_queueMutex);
29 for (const auto &info : m_queuedConnections)
30 addConnection(info.signalName, info.argTypes, info.listener, info.id);
31 m_queuedConnections.clear();
32 } else if (status == QQuickView::Error) {
33 m_queuedConnections.clear();
34 }
35}
36
37QByteArray qmlTypeArgStringFromJavaTypeArgs(const QJniArray<jclass> &javaArgClasses)
38{
39 static const QHash<QByteArray, QMetaType::Type> javaToQMetaTypeMap = {
40 { "java/lang/Void", QMetaType::Type::Void },
41 { "java/lang/String", QMetaType::Type::QString },
42 { "java/lang/Integer", QMetaType::Type::Int },
43 { "java/lang/Double", QMetaType::Type::Double },
44 { "java/lang/Float", QMetaType::Type::Float },
45 { "java/lang/Boolean", QMetaType::Type::Bool }
46 };
47
48 QList<QByteArray> qmlArgTypes;
49 for (const auto javaArgClass : javaArgClasses) {
50 const auto javaArgClassName = QJniObject(javaArgClass).className();
51 const auto javaArgMetaType =
52 javaToQMetaTypeMap.value(javaArgClassName, QMetaType::Type::UnknownType);
53
54 if (javaArgMetaType == QMetaType::Type::UnknownType) {
55 qWarning() << "Unknown type for signal argument" << javaArgClassName;
56 return "";
57 }
58
59 qmlArgTypes.emplace_back(QMetaType(javaArgMetaType).name());
60 }
61 return qmlArgTypes.join(',');
62}
63
64QByteArray qmlSignalSignature(const QString &signalName, const QJniArray<jclass> &javaArgClasses)
65{
66 const auto qmlArgTypes = qmlTypeArgStringFromJavaTypeArgs(javaArgClasses);
67 return QMetaObject::normalizedSignature(
68 qPrintable(QStringLiteral("%1(%2)").arg(signalName).arg(qmlArgTypes)));
69}
70
71QList<QMetaType::Type> metaMethodArgumentTypes(const QMetaMethod &method)
72{
73 QList<QMetaType::Type> types;
74 for (auto i = 0; i < method.parameterCount(); ++i)
75 types.push_back(static_cast<QMetaType::Type>(method.parameterType(i)));
76 return types;
77}
78
79int indexOfSignal(const QMetaObject &metaObject, const QString &signalName,
80 const QJniArray<jclass> &javaArgClasses)
81{
82 int signalIndex = metaObject.indexOfSignal(qmlSignalSignature(signalName, javaArgClasses));
83 if (signalIndex == -1) // Signal is maybe a parameterless property signal
84 signalIndex =
85 metaObject.indexOfSignal(QMetaObject::normalizedSignature(qPrintable(signalName)));
86 return signalIndex;
87}
88
89std::optional<int> propertyIndexForSignal(const QMetaMethod &signal, const QMetaObject &object)
90{
91 for (int i = 0; i < object.propertyCount(); ++i) {
92 const auto metaProperty = object.property(i);
93 if (signal.methodSignature() == metaProperty.notifySignal().methodSignature())
94 return metaProperty.propertyIndex();
95 }
96 return std::nullopt;
97}
98
99bool QAndroidViewSignalManager::addConnection(const QString &signalName,
100 const QJniArray<jclass> &argTypes,
101 const QJniObject &listener,
102 int id)
103{
104 if (m_view->status() == QQuickView::Error) {
105 qWarning("Can not connect to signals due to errors while loading the view");
106 return false;
107 }
108
109 if (m_view->status() != QQuickView::Ready) {
110 return queueConnection(signalName, argTypes, listener, id);
111 }
112
113 const auto *rootMetaObject = m_view->rootObject()->metaObject();
114 int signalIndex = indexOfSignal(*rootMetaObject, signalName, argTypes);
115 if (signalIndex == -1) {
116 qWarning("Failed to find matching signal from root object for signal: %s",
117 qPrintable(signalName));
118 return false;
119 }
120
121 if (m_connections.contains(signalIndex))
122 return true;
123
124 // Connect the signal to this class' qt_metacall
125 const auto connection =
126 QMetaObject::connect(m_view->rootObject(), signalIndex, this,
127 QObject::metaObject()->methodCount(), Qt::QueuedConnection);
128
129 const auto signal = rootMetaObject->method(signalIndex);
130 const auto propertyIndex = propertyIndexForSignal(signal, *rootMetaObject);
131 const auto argumentTypes = metaMethodArgumentTypes(signal);
132
133 m_connections.insert(signalIndex,
134 { .connection = connection,
135 .listenerObject = listener,
136 .qmlSignalName = signalName,
137 .qmlArgumentTypes = argumentTypes,
138 .isPropertySignal = propertyIndex.has_value(),
139 .qmlPropertyIndex = propertyIndex,
140 .connectionId = id });
141 return true;
142}
143
144bool QAndroidViewSignalManager::queueConnection(const QString &signalName,
145 const QJniArray<jclass> &argTypes,
146 const QJniObject &listener,
147 int id)
148{
149 QMutexLocker lock(&m_queueMutex);
150 m_queuedConnections.push_back({
151 .id = id,
152 .signalName = signalName,
153 .argTypes = argTypes,
154 .listener = listener
155 });
156 return true;
157}
158
159QJniObject voidStarToQJniObject(const QMetaType::Type type, const void *data)
160{
161 switch (type) {
162 case QMetaType::Type::Bool:
163 return QtJniTypes::Boolean::construct(*reinterpret_cast<const bool *>(data));
164 case QMetaType::Type::Int:
165 return QtJniTypes::Integer::construct(*reinterpret_cast<const int *>(data));
166 case QMetaType::Type::Double:
167 return QtJniTypes::Double::construct(*reinterpret_cast<const double *>(data));
168 case QMetaType::Type::Float:
169 return QtJniTypes::Float::construct(*reinterpret_cast<const float *>(data));
170 case QMetaType::Type::QString:
171 return QtJniTypes::String::construct(*reinterpret_cast<const QString *>(data));
172 default:
173 qWarning() << "Unknown type for signal argument" << type;
174 break;
175 }
176 return {};
177}
178
179QJniObject qVariantToQJniObject(const QVariant &data)
180{
181 const auto type = data.metaType().id();
182 switch (type) {
183 case QMetaType::Type::Bool:
184 return QtJniTypes::Boolean::construct(data.toBool());
185 case QMetaType::Type::Int:
186 return QtJniTypes::Integer::construct(data.toInt());
187 case QMetaType::Type::Double:
188 return QtJniTypes::Double::construct(data.toDouble());
189 case QMetaType::Type::Float:
190 return QtJniTypes::Float::construct(data.toFloat());
191 case QMetaType::Type::QString:
192 return QtJniTypes::String::construct(data.toString());
193 default:
194 qWarning() << "Unknown type for signal argument" << type;
195 break;
196 }
197 return {};
198}
199
200int QAndroidViewSignalManager::qt_metacall(QMetaObject::Call call, int methodId, void **args)
201{
202 Q_UNUSED(call);
203 if (!m_connections.contains(senderSignalIndex())) {
204 qWarning() << "QAndroidViewSignalManager::qt_metacall received unexpected signal";
205 return methodId;
206 }
207
208 const auto connection = m_connections.value(senderSignalIndex());
209
210 if (connection.isPropertySignal) {
211 const auto senderMetaObject = sender()->metaObject();
212 const auto property = senderMetaObject->property(connection.qmlPropertyIndex.value());
213 const auto data = property.read(sender());
214
215 connection.listenerObject.callMethod<void>("onSignalEmitted", connection.qmlSignalName,
216 qVariantToQJniObject(data));
217 } else {
218 QJniArray<QJniObject> data(connection.qmlArgumentTypes.size());
219 for (auto i = 0; i < connection.qmlArgumentTypes.size(); ++i) {
220 const auto argType = connection.qmlArgumentTypes.at(i);
221 const auto receivedRawData = args[i + 1];
222
223 if (receivedRawData == nullptr) {
224 qWarning() << "Received null data for signal" << connection.qmlSignalName;
225 return methodId;
226 }
227
228 const auto receivedData = voidStarToQJniObject(argType, receivedRawData);
229 if (!receivedData.isValid()) {
230 qWarning() << "Received invalid data for signal" << connection.qmlSignalName;
231 return methodId;
232 }
233
234 data.setValue(i, receivedData);
235 }
236
237 if (data.size() == 0) { // Signal with no params, uses QtSignalListener<Void>
238 QNativeInterface::QAndroidApplication::runOnAndroidMainThread([connection] {
239 connection.listenerObject.callMethod<void>("onSignalEmitted",
240 connection.qmlSignalName,
241 QtJniTypes::Void::construct().object());
242 });
243 } else if (data.size() == 1) { // Signal with a single param, uses QtSignalListener<T>
244 QNativeInterface::QAndroidApplication::runOnAndroidMainThread(
245 [connection, signalArg = data.at(0)] {
246 connection.listenerObject.callMethod<void>("onSignalEmitted",
247 connection.qmlSignalName,
248 signalArg);
249 });
250 } else { // Signal with multiple params, uses custom listener interface with Object[] API
251 QNativeInterface::QAndroidApplication::runOnAndroidMainThread([connection, data] {
252 connection.listenerObject.callMethod<void>("onSignalEmitted", data);
253 });
254 }
255 }
256 return methodId;
257}
258
259bool QAndroidViewSignalManager::hasConnection(connection_key_t key) const
260{
261 for (const auto &info : m_connections) {
262 if (info.connectionId == key)
263 return true;
264 }
265 return false;
266}
267
268void QAndroidViewSignalManager::removeConnection(connection_key_t key)
269{
270 if (hasConnection(key)) {
271 m_connections.removeIf([key](const auto &iter) {
272 if (iter->connectionId == key)
273 return QObject::disconnect(iter->connection);
274 return false;
275 });
276 } else {
277 QMutexLocker lock(&m_queueMutex);
278 m_queuedConnections.removeIf([key](const auto &info) { return info.id == key; });
279 }
280}
281
282QT_END_NAMESPACE
QObject * parent
Definition qobject.h:74
The QQuickView class provides a window for displaying a Qt Quick user interface.
Definition qquickview.h:21
Combined button and popup list for selecting options.
QJniObject qVariantToQJniObject(const QVariant &data)
std::optional< int > propertyIndexForSignal(const QMetaMethod &signal, const QMetaObject &object)
QByteArray qmlSignalSignature(const QString &signalName, const QJniArray< jclass > &javaArgClasses)
QList< QMetaType::Type > metaMethodArgumentTypes(const QMetaMethod &method)
int indexOfSignal(const QMetaObject &metaObject, const QString &signalName, const QJniArray< jclass > &javaArgClasses)
QByteArray qmlTypeArgStringFromJavaTypeArgs(const QJniArray< jclass > &javaArgClasses)
QJniObject voidStarToQJniObject(const QMetaType::Type type, const void *data)