如何直接从应用程序调用 Qt?
How to make a call with Qt directly from the application?
我想在我的应用程序中实现拨号器功能。实际上,它已经完成了,但是它以我不希望的方式工作。按下按钮时,native dialer opens and waiting for pressing a button。不双击可以直接调用吗? 这是我的代码:
Button {
id: callButton
anchors.centerIn: parent
text: 'Make a call'
onClicked: Qt.openUrlExternally('tel:+77051085322')
}
您将需要权限
<uses-permission android:name="android.permission.CALL_PHONE" />
在你的 AndroidManifest.xml
在 java 你会做:
Intent dialIntent = new Intent(Intent.ACTION_CALL, Uri.parse("tel:+1123123"));
startActivity(dialIntent);
等效的 Qt 代码类似于:
QAndroidJniObject action = QAndroidJniObject::fromString("android.intent.action.CALL");
QAndroidJniObject uriString = QAndroidJniObject::fromString("tel:+1123123");
QAndroidJniObject uri = QAndroidJniObject::callStaticObjectMethod("android/net/Uri", "parse", "(Ljava/lang/String)V", uriString);
QAndroidJniObject intent("android/content/Intent","(Ljava/lang/String, Landroid/net/Uri)V", action, uri);
QAndroidJniObject activity = QAndroidJniObject::callStaticObjectMethod("org/qtproject/qt5/android/QtNative", "activity", "()Landroid/app/Activity;");
activity.callObjectMethod("startActivity","(Landroid/content/Intent;)V",intent.object<jobject>());
但是,请注意,使用 ACTION_CALL 可能会导致您被应用商店拒绝,并且 google 建议使用 ACTION_DIAL,它会打开拨号器而不是直接调用。
而在 iOS 中可以发出调用 directly, the same does not apply to Android. To overcome the problem you can define a C++ class Wrapper
which handles the call, depending on the current OS. An instance of this class is registered as a context property 并直接在 QML 中使用。
在 class 中,您可以利用 Android 本机 API,它通过 Intent
操作 ACTION_CALL
提供自动拨号功能(但请记住 there are some restrictions in using it ).通常在 Android 你写:
Intent callIntent = new callIntent(Intent.ACTION_CALL);
callIntent.setPackage("com.android.phone"); // force native dialer (Android < 5)
callIntent.setPackage("com.android.server.telecom"); // force native dialer (Android >= 5)
callIntent.setData(Uri.parse("tel:" + number));
callIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(callIntent);
通过设置包我们可以强制使用本地拨号器。如果没有它,用户将被提示在可用的拨号器(即 Skype、Viber 等)中进行选择如果该设备上安装了其他。 Lollipop 和以前版本之间的系统拨号程序包发生了变化,因此有必要在运行时检查 SDK 以设置正确的。
要在 C++ 中调用这些 API,您需要 Qt Android Extras and in particular QAndroidJniObject
以及自定义 Android 清单中的相关权限。只需添加到您的 .pro
文件:
android: QT += androidextras #included only in Android builds
以及清单的下一行:
<uses-permission android:name="android.permission.CALL_PHONE"/>
如果您没有定义自定义清单,只需添加一个。从 Qt Creator 3.3 开始,只需转至 Projects > Build > Build Android APK > Create Templates
即可生成自定义清单。
我们 class 的 header 如下所示 - constructor/deconstructor 缺失:
#ifndef WRAPPER_H
#define WRAPPER_H
#include <QObject>
#include <QString>
#include <QDebug>
#if defined(Q_OS_IOS)
#include <QUrl>
#include <QDesktopServices>
#elif defined(Q_OS_ANDROID)
#include <QtAndroid>
#include <QAndroidJniObject>
#endif
#include <QDesktopServices>
#include <QUrl>
class Wrapper: public QObject
{
Q_OBJECT
public:
Q_INVOKABLE void directCall(QString number);
};
#endif // WRAPPER_H
相应的源文件如下所示 - 再次 constructor/deconstructor 缺失:
#include "wrapper.h"
void Wrapper::directCall(QString number)
{
#if defined(Q_OS_IOS)
QDesktopServices::openUrl(QUrl(QString("tel://%1").arg(number)));
#elif defined(Q_OS_ANDROID)
// get the Qt android activity
QAndroidJniObject activity = QAndroidJniObject::callStaticObjectMethod("org/qtproject/qt5/android/QtNative", "activity", "()Landroid/app/Activity;");
//
if (activity.isValid()){
// real Java code to C++ code
// Intent callIntent = new callIntent(Intent.ACTION_CALL);
QAndroidJniObject callConstant = QAndroidJniObject::getStaticObjectField<jstring>("android/content/Intent", "ACTION_CALL");
QAndroidJniObject callIntent("android/content/Intent", "(Ljava/lang/String;)V", callConstant.object());
// callIntent.setPackage("com.android.phone"); (<= 4.4w) intent.setPackage("com.android.server.telecom"); (>= 5)
QAndroidJniObject package;
if(QtAndroid::androidSdkVersion() >= 21)
package = QAndroidJniObject::fromString("com.android.server.telecom");
else
package = QAndroidJniObject::fromString("com.android.phone");
callIntent.callObjectMethod("setPackage", "(Ljava/lang/String;)Landroid/content/Intent;", package.object<jstring>());
// callIntent.setData(Uri.parse("tel:" + number));
QAndroidJniObject jNumber = QAndroidJniObject::fromString(QString("tel:%1").arg(number));
QAndroidJniObject uri = QAndroidJniObject::callStaticObjectMethod("android/net/Uri","parse","(Ljava/lang/String;)Landroid/net/Uri;", jNumber.object());
callIntent.callObjectMethod("setData", "(Landroid/net/Uri;)Landroid/content/Intent;", uri.object<jobject>());
// callIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
jint flag = QAndroidJniObject::getStaticField<jint>("android/content/Intent", "FLAG_ACTIVITY_NEW_TASK");
callIntent.callObjectMethod("setFlags", "(I)Landroid/content/Intent;", flag);
//startActivity(callIntent);
activity.callMethod<void>("startActivity","(Landroid/content/Intent;)V", callIntent.object<jobject>());
}
else
qDebug() << "Something wrong with Qt activity...";
#else
qDebug() << "Does nothing here...";
#endif
}
如开头所述,您可以将此 class 的一个实例作为上下文 属性。用于此目的的 main
如下所示:
#include <QApplication>
#include <QQmlContext>
#include <QQmlApplicationEngine>
#include "wrapper.h"
int main(int argc, char *argv[])
{
QApplication app(argc, argv);
QQmlApplicationEngine engine;
Wrapper jw;
engine.rootContext()->setContextProperty("caller", &jw);
engine.load(QUrl(QStringLiteral("qrc:/main.qml")));
return app.exec();
}
最后在 QML 中你可以简单地写:
Button {
id: callButton
anchors.centerIn: parent
text: 'Make a call'
onClicked: caller.directCall("+0123456789")
}
代码可以轻松扩展以支持 WinPhone,同时保持相同的 QML 接口(可能通过包含专用的 header/source 对)。最后,条件包含的使用保证了代码正确编译,即使使用的工具包在运行中被更改。
最后一点,我要补充一点,Google Play 政策不像 Apple App Store 政策那么严格。因此,由于使用 ACTION_CALL
而被拒绝的应用程序不太可能发生。
我想在我的应用程序中实现拨号器功能。实际上,它已经完成了,但是它以我不希望的方式工作。按下按钮时,native dialer opens and waiting for pressing a button。不双击可以直接调用吗? 这是我的代码:
Button {
id: callButton
anchors.centerIn: parent
text: 'Make a call'
onClicked: Qt.openUrlExternally('tel:+77051085322')
}
您将需要权限
<uses-permission android:name="android.permission.CALL_PHONE" />
在你的 AndroidManifest.xml
在 java 你会做:
Intent dialIntent = new Intent(Intent.ACTION_CALL, Uri.parse("tel:+1123123"));
startActivity(dialIntent);
等效的 Qt 代码类似于:
QAndroidJniObject action = QAndroidJniObject::fromString("android.intent.action.CALL");
QAndroidJniObject uriString = QAndroidJniObject::fromString("tel:+1123123");
QAndroidJniObject uri = QAndroidJniObject::callStaticObjectMethod("android/net/Uri", "parse", "(Ljava/lang/String)V", uriString);
QAndroidJniObject intent("android/content/Intent","(Ljava/lang/String, Landroid/net/Uri)V", action, uri);
QAndroidJniObject activity = QAndroidJniObject::callStaticObjectMethod("org/qtproject/qt5/android/QtNative", "activity", "()Landroid/app/Activity;");
activity.callObjectMethod("startActivity","(Landroid/content/Intent;)V",intent.object<jobject>());
但是,请注意,使用 ACTION_CALL 可能会导致您被应用商店拒绝,并且 google 建议使用 ACTION_DIAL,它会打开拨号器而不是直接调用。
而在 iOS 中可以发出调用 directly, the same does not apply to Android. To overcome the problem you can define a C++ class Wrapper
which handles the call, depending on the current OS. An instance of this class is registered as a context property 并直接在 QML 中使用。
在 class 中,您可以利用 Android 本机 API,它通过 Intent
操作 ACTION_CALL
提供自动拨号功能(但请记住 there are some restrictions in using it ).通常在 Android 你写:
Intent callIntent = new callIntent(Intent.ACTION_CALL);
callIntent.setPackage("com.android.phone"); // force native dialer (Android < 5)
callIntent.setPackage("com.android.server.telecom"); // force native dialer (Android >= 5)
callIntent.setData(Uri.parse("tel:" + number));
callIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(callIntent);
通过设置包我们可以强制使用本地拨号器。如果没有它,用户将被提示在可用的拨号器(即 Skype、Viber 等)中进行选择如果该设备上安装了其他。 Lollipop 和以前版本之间的系统拨号程序包发生了变化,因此有必要在运行时检查 SDK 以设置正确的。
要在 C++ 中调用这些 API,您需要 Qt Android Extras and in particular QAndroidJniObject
以及自定义 Android 清单中的相关权限。只需添加到您的 .pro
文件:
android: QT += androidextras #included only in Android builds
以及清单的下一行:
<uses-permission android:name="android.permission.CALL_PHONE"/>
如果您没有定义自定义清单,只需添加一个。从 Qt Creator 3.3 开始,只需转至 Projects > Build > Build Android APK > Create Templates
即可生成自定义清单。
我们 class 的 header 如下所示 - constructor/deconstructor 缺失:
#ifndef WRAPPER_H
#define WRAPPER_H
#include <QObject>
#include <QString>
#include <QDebug>
#if defined(Q_OS_IOS)
#include <QUrl>
#include <QDesktopServices>
#elif defined(Q_OS_ANDROID)
#include <QtAndroid>
#include <QAndroidJniObject>
#endif
#include <QDesktopServices>
#include <QUrl>
class Wrapper: public QObject
{
Q_OBJECT
public:
Q_INVOKABLE void directCall(QString number);
};
#endif // WRAPPER_H
相应的源文件如下所示 - 再次 constructor/deconstructor 缺失:
#include "wrapper.h"
void Wrapper::directCall(QString number)
{
#if defined(Q_OS_IOS)
QDesktopServices::openUrl(QUrl(QString("tel://%1").arg(number)));
#elif defined(Q_OS_ANDROID)
// get the Qt android activity
QAndroidJniObject activity = QAndroidJniObject::callStaticObjectMethod("org/qtproject/qt5/android/QtNative", "activity", "()Landroid/app/Activity;");
//
if (activity.isValid()){
// real Java code to C++ code
// Intent callIntent = new callIntent(Intent.ACTION_CALL);
QAndroidJniObject callConstant = QAndroidJniObject::getStaticObjectField<jstring>("android/content/Intent", "ACTION_CALL");
QAndroidJniObject callIntent("android/content/Intent", "(Ljava/lang/String;)V", callConstant.object());
// callIntent.setPackage("com.android.phone"); (<= 4.4w) intent.setPackage("com.android.server.telecom"); (>= 5)
QAndroidJniObject package;
if(QtAndroid::androidSdkVersion() >= 21)
package = QAndroidJniObject::fromString("com.android.server.telecom");
else
package = QAndroidJniObject::fromString("com.android.phone");
callIntent.callObjectMethod("setPackage", "(Ljava/lang/String;)Landroid/content/Intent;", package.object<jstring>());
// callIntent.setData(Uri.parse("tel:" + number));
QAndroidJniObject jNumber = QAndroidJniObject::fromString(QString("tel:%1").arg(number));
QAndroidJniObject uri = QAndroidJniObject::callStaticObjectMethod("android/net/Uri","parse","(Ljava/lang/String;)Landroid/net/Uri;", jNumber.object());
callIntent.callObjectMethod("setData", "(Landroid/net/Uri;)Landroid/content/Intent;", uri.object<jobject>());
// callIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
jint flag = QAndroidJniObject::getStaticField<jint>("android/content/Intent", "FLAG_ACTIVITY_NEW_TASK");
callIntent.callObjectMethod("setFlags", "(I)Landroid/content/Intent;", flag);
//startActivity(callIntent);
activity.callMethod<void>("startActivity","(Landroid/content/Intent;)V", callIntent.object<jobject>());
}
else
qDebug() << "Something wrong with Qt activity...";
#else
qDebug() << "Does nothing here...";
#endif
}
如开头所述,您可以将此 class 的一个实例作为上下文 属性。用于此目的的 main
如下所示:
#include <QApplication>
#include <QQmlContext>
#include <QQmlApplicationEngine>
#include "wrapper.h"
int main(int argc, char *argv[])
{
QApplication app(argc, argv);
QQmlApplicationEngine engine;
Wrapper jw;
engine.rootContext()->setContextProperty("caller", &jw);
engine.load(QUrl(QStringLiteral("qrc:/main.qml")));
return app.exec();
}
最后在 QML 中你可以简单地写:
Button {
id: callButton
anchors.centerIn: parent
text: 'Make a call'
onClicked: caller.directCall("+0123456789")
}
代码可以轻松扩展以支持 WinPhone,同时保持相同的 QML 接口(可能通过包含专用的 header/source 对)。最后,条件包含的使用保证了代码正确编译,即使使用的工具包在运行中被更改。
最后一点,我要补充一点,Google Play 政策不像 Apple App Store 政策那么严格。因此,由于使用 ACTION_CALL
而被拒绝的应用程序不太可能发生。