前面几篇我们讲解了 QtAndroid 名字空间的基本用法,这次我们使用前面讲过的方法和类库,展示一些简单的小示例。我在《Qt on Android核心编程》一书中主要通过“继承 QtActivity ,实现自己的 Activity 并添加 static 方法”这种形式来调用 Android 系统的一些功能。这一系列的文章,我们主要使用 Qt 5.3 里引入的 QtAndroid 名字空间内的方法和 QAndroidJniObject 类来展示 Qt 中如何进行 JNI 调用,只在必要时才重写 QtActivity 。
Qt on Android 应用,根据你的需求,经常会调用到 Android 系统提供的一些功能,比如判断网络连接、获取外部存储路径,或者缓存文件目录等等。这些经常被朋友问到,我会在这一系列文章中慢慢把 Qt on Android 开发中经常用到的功能点都演示一下。希望对大家有所帮助。
示例很简单,使用 Qt Widgets 来展示。下图是效果:
如上图所示,界面非常简陋,点下 Refresh 按钮,就获取一些 Android 系统信息和当前应用的一些信息,放在 QListWidget 中。包括下面的内容:
代码没什么逻辑可讲……都在下面了:
#include "widget.h" #include <QVBoxLayout> #include <QListWidgetItem> #include <QtAndroid> #include <QAndroidJniEnvironment> #include <QAndroidJniObject> #include <QDebug> using namespace QtAndroid; #define CHECK_EXCEPTION() \ if(env->ExceptionCheck())\ {\ qDebug() << "exception occured";\ env->ExceptionClear();\ } Widget::Widget(QWidget *parent) : QWidget(parent) { QVBoxLayout *layout = new QVBoxLayout(this); m_refresh = new QPushButton("Refresh"); connect(m_refresh, SIGNAL(clicked()), this, SLOT(onRefresh())); layout->addWidget(m_refresh); m_list = new QListWidget(); layout->addWidget(m_list, 1); } Widget::~Widget() { } void Widget::onRefresh() { m_list->clear(); QAndroidJniEnvironment env; //get Android SDK version m_list->addItem(QString("SDK版本:%1").arg(androidSdkVersion())); QAndroidJniObject activity = androidActivity(); //get network state QAndroidJniObject connectivity = QAndroidJniObject::getStaticObjectField( "android/content/Context", "CONNECTIVITY_SERVICE", "Ljava/lang/String;"); if(connectivity.isValid()){ qDebug() << "connectivity id - " << connectivity.toString(); CHECK_EXCEPTION() QAndroidJniObject connectivityService = activity.callObjectMethod( "getSystemService", "(Ljava/lang/String;)Ljava/lang/Object;", connectivity.object<jstring>()); CHECK_EXCEPTION() qDebug() << "got connectivity service - " << connectivityService.isValid(); if(connectivityService.isValid()) { QAndroidJniObject networkInfo = connectivityService.callObjectMethod( "getActiveNetworkInfo", "()Landroid/net/NetworkInfo;"); CHECK_EXCEPTION() qDebug() << "got NetworkInfo - " << networkInfo.isValid(); if(networkInfo.isValid()) { m_list->addItem(QString("网络状态:已连接(%1)").arg(networkInfo.toString())); } else { m_list->addItem("网络状态:未连接"); } } } //get variable directories of Android System QAndroidJniObject externalStorageDir = QAndroidJniObject::callStaticObjectMethod( "android/os/Environment", "getExternalStorageDirectory", "()Ljava/io/File;" ); CHECK_EXCEPTION() m_list->addItem(QString("外部存储目录:%1").arg(externalStorageDir.toString())); QAndroidJniObject dataDir = QAndroidJniObject::callStaticObjectMethod( "android/os/Environment", "getDataDirectory", "()Ljava/io/File;" ); CHECK_EXCEPTION() m_list->addItem(QString("数据目录:%1").arg(dataDir.toString())); QAndroidJniObject dcim = QAndroidJniObject::getStaticObjectField( "android/os/Environment", "DIRECTORY_DCIM", "Ljava/lang/String;" ); CHECK_EXCEPTION() QAndroidJniObject dcimDir = QAndroidJniObject::callStaticObjectMethod( "android/os/Environment", "getExternalStoragePublicDirectory", "(Ljava/lang/String;)Ljava/io/File;", dcim.object<jstring>() ); CHECK_EXCEPTION() m_list->addItem(QString("照片目录:%1").arg(dcimDir.toString())); QAndroidJniObject music = QAndroidJniObject::getStaticObjectField( "android/os/Environment", "DIRECTORY_MUSIC", "Ljava/lang/String;" ); CHECK_EXCEPTION() QAndroidJniObject musicDir = QAndroidJniObject::callStaticObjectMethod( "android/os/Environment", "getExternalStoragePublicDirectory", "(Ljava/lang/String;)Ljava/io/File;", music.object<jstring>() ); CHECK_EXCEPTION() m_list->addItem(QString("音乐目录:%1").arg(musicDir.toString())); QAndroidJniObject movie = QAndroidJniObject::getStaticObjectField( "android/os/Environment", "DIRECTORY_MOVIES", "Ljava/lang/String;" ); CHECK_EXCEPTION() QAndroidJniObject movieDir = QAndroidJniObject::callStaticObjectMethod( "android/os/Environment", "getExternalStoragePublicDirectory", "(Ljava/lang/String;)Ljava/io/File;", movie.object<jstring>() ); CHECK_EXCEPTION() m_list->addItem(QString("视频目录:%1").arg(movieDir.toString())); QAndroidJniObject ringtones = QAndroidJniObject::getStaticObjectField( "android/os/Environment", "DIRECTORY_RINGTONES", "Ljava/lang/String;" ); CHECK_EXCEPTION() QAndroidJniObject ringtonesDir = QAndroidJniObject::callStaticObjectMethod( "android/os/Environment", "getExternalStoragePublicDirectory", "(Ljava/lang/String;)Ljava/io/File;", ringtones.object<jstring>() ); CHECK_EXCEPTION() m_list->addItem(QString("铃声目录:%1").arg(ringtonesDir.toString())); //app's infomation QAndroidJniObject filesDir = activity.callObjectMethod( "getFilesDir", "()Ljava/io/File;"); CHECK_EXCEPTION() m_list->addItem(QString("应用文件目录:%1").arg(filesDir.toString())); QAndroidJniObject packageName = activity.callObjectMethod<jstring>("getPackageName"); CHECK_EXCEPTION() m_list->addItem(QString("应用包名:%1").arg(packageName.toString())); QAndroidJniObject appCacheDir = activity.callObjectMethod( "getCacheDir", "()Ljava/io/File;"); CHECK_EXCEPTION() m_list->addItem(QString("应用缓存目录:%1").arg(appCacheDir.toString())); QAndroidJniObject appInfo = activity.callObjectMethod( "getApplicationInfo", "()Landroid/content/pm/ApplicationInfo;"); CHECK_EXCEPTION() QAndroidJniObject appClassName = appInfo.getObjectField<jstring>("className"); CHECK_EXCEPTION() m_list->addItem(QString("应用类名:%1").arg(appClassName.toString())); QAndroidJniObject appLocation = appInfo.getObjectField( "sourceDir", "Ljava/lang/String;"); CHECK_EXCEPTION() m_list->addItem(QString("APK位置:%1").arg(appLocation.toString())); }
其实在 Qt 中通过 JNI 调用 Android 功能,关键的就是两点:
Qt 提供的 API ,在“QtAndroid详解(1):QAndroidJniObject”、"QtAndroid详解(2):startActivity和它的小伙伴们"、"QtAndroid详解(3):startActivity实战Android拍照功能"这三篇文章中已有详细讲解,不再赘述了。
Android 类库这方面,我们搞 C++ 开发的朋友,可能不熟悉。不过没关系,可以通过 Android 在线 SDK 来学习,另外我这里提供的 Qt JNI 代码,都是实测可用的,里面演示一些功能的代码,如果需要,可以直接在项目中使用。
好了,我们开始慢慢介绍吧。
这个很贴心,QtAndroid名字空间直接提供了一个方法: androidSdkVersion() 。它返回一个整数值,表示 Android SDK 版本号。
在 Android 中,有一个 ConnectivityManager 类,可以查询系统的网络状态。
ConnectivityManager 类的 getActiveNetworkInfo() 方法可以获取到当前活跃的网络连接信息,它返回一个 NetworkInfo 类的实例。如果未联网,这个方法返回 null 。
ConnectivityManager 在 Android 系统里以一个服务存在,需要通过 Context 的 getSystemService() 方法来获取到这个服务。 getSystemService() 接受一个代表服务名字的字符串作为参数。对于网络连接管理服务,名字是 CONNECTIVITY_SERVICE ,它是 Context 类的静态成员变量。
获取网络连接管理服务的 Java 代码如下:
Context.getSystemService(Context.CONNECTIVITY_SERVICE);
QAndroidJniObject connectivity = QAndroidJniObject::getStaticObjectField( "android/content/Context", "CONNECTIVITY_SERVICE", "Ljava/lang/String;");
QAndroidJniObject connectivityService = activity.callObjectMethod( "getSystemService", "(Ljava/lang/String;)Ljava/lang/Object;", connectivity.object<jstring>());
拿到了 ConnectivityManager 实例,就该调用它的 getActiveNetworkInfo() 方法来获取当前的活动连接了。下面是代码:
QAndroidJniObject networkInfo = connectivityService.callObjectMethod( "getActiveNetworkInfo", "()Landroid/net/NetworkInfo;");
其它的都是辅助性代码,看最前面的源码好了。
示例里的各种系统级的目录,都是通过 android.os.Enviroment 这个类获取的。
我们先说图片、视频、铃声这些吧,他们通过 getExternalStoragePublicDirectory(String) 方法获取。Android给每个公共存储目录提供了一个字符串类型的名字,定义为 Enviroment 类的静态成员变量。所以,我们使用 Qt JNI 获取这些目录的步骤是:
按照这个逻辑来看获取图片目录的代码,关键的就下面两行:
QAndroidJniObject dcim = QAndroidJniObject::getStaticObjectField( "android/os/Environment", "DIRECTORY_DCIM", "Ljava/lang/String;" ); QAndroidJniObject dcimDir = QAndroidJniObject::callStaticObjectMethod( "android/os/Environment", "getExternalStoragePublicDirectory", "(Ljava/lang/String;)Ljava/io/File;", dcim.object<jstring>() );
当前应用的一些信息,可以通过 Android 里的 Activity 类获取。
我们需要一个 Activity 对象,在 Qt on Android 应用里,对应的类是 QtActivity ,之前在“QtAndroid详解(3):startActivity实战Android拍照功能”中我们已经介绍过了。不多说了。
获取当前应用 files 目录的代码如下:
QAndroidJniObject filesDir = activity.callObjectMethod( "getFilesDir", "()Ljava/io/File;");
--------
好啦,这次就到这里吧。下一次我们会展示更有意思的一些实用功能,如让手机震动、让屏幕常亮、动态切换横屏竖屏等。再再往后可能还会介绍调节音量、调整屏幕亮度、使用推送、通知栏、选取联系人等等,不过要看我的时间哈。
回顾一下: