Как позвонить с помощью Qt прямо из приложения?
Я хочу реализовать функцию дозвона в моем приложении. На самом деле, это сделано, но работает так, как я не хочу. Когда кнопка нажата, родная звонилка открывается и ждет нажатия кнопки. Можно ли звонить напрямую без двойного нажатия? Вот мой код:
Button {
id: callButton
anchors.centerIn: parent
text: 'Make a call'
onClicked: Qt.openUrlExternally('tel:+77051085322')
}
2 ответа
Принимая во внимание, что в iOS вызов может быть сделан непосредственно, то же самое не относится к Android. Чтобы преодолеть проблему, вы можете определить класс C++ Wrapper
который обрабатывает вызов, в зависимости от текущей ОС. Экземпляр этого класса зарегистрирован как свойство контекста и непосредственно используется в QML.
Внутри класса вы можете использовать собственные API-интерфейсы Android, которые предоставляют функцию автоматического набора через Intent
действие ACTION_CALL
(но помните, что есть некоторые ограничения в его использовании). Обычно в 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 во время выполнения, чтобы установить правильный.
Чтобы вызывать эти API в C++, вам нужны дополнительные функции Qt Android и, в частности, 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
создать собственный манифест.
Заголовок нашего класса выглядит следующим образом - конструктор / деконструктор отсутствует:
#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
Соответствующий исходный файл выглядит следующим образом - опять не хватает конструктора / деконструктора:
#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
}
Как обсуждалось в начале, вы можете включить экземпляр этого класса в качестве свойства контекста. 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 (возможно, путем включения отдельной пары заголовок / источник). Наконец, использование условных включений гарантирует, что код корректно компилируется, даже если используемый набор изменяется на лету.
В заключение хочу добавить, что политики Google Play не так строги, как политики Apple App Store. Следовательно, отклонение приложения из-за использования ACTION_CALL
вряд ли произойдет.
Вам понадобится разрешение
<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, который открывает номеронабиратель вместо прямого вызова.