关于开发一个应用,要有自己的应用名(显示用),和包名(真正唯一的应用名),简单说一台 Android 手机中所有应用的包名是唯一的,如果新安装的应用包名和已安装的应用重复则只能替换安装(不可共存)。这就需要修改 AndroidManifest.xml。
先看一下之前的例子里的 AndroidManifest.xml
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="gooid.mainnative" ... android:label="WallPaperTwinkle" android:hasCode="false" >
...
<meta-data android:name="android.app.lib_name" android:value="WallPaperTwinkle" />
...
application>
manifest>
中 “WallPaperTwinkle” 是表示引用的so名,此处表示 lib 名为 libWallPaperTwinkle.so 因此如果改成 “UIDemo” 的话,前面讲的编译命令要作相应修改: go build -buildmode c-shared -o basic\lib\%ABI%\libUIDemo.so
通过定制包名,应用名,和库名就可以区分每个不同的应用了(其实还有应用图标,这个以后再讲)。
接下来就该编码了。
从NDK的角度来说,一个 Android 应用实现以下的 Callbacks 就可以了,其中的回调函数可以为空,实现必要的回调就行了,所以下面忽略了不常用的回调。
type Callbacks struct {
Create func(*Activity, []byte)
...
Destroy func(*Activity)
...
FocusChanged func(*Activity, bool)
....
// Window
WindowCreated func(*Activity, *Window)
WindowDestroyed func(*Activity, *Window)
WindowDraw func(*Activity, *Window)
WindowResized func(*Activity, *Window)
WindowRedrawNeeded func(*Activity, *Window)
// Touch is called by the app when a touch event occurs.
Event func(*Activity, *InputEvent)
// Sensor
Sensor func(*Activity, []SensorEvent)
}
如果想了解这些回调的详细信息,当然是参考管方文档了。在这里就简单说一下常用的几个。
在 Android NDK 下要实现图形界面(GUI)需要用OpenGLES来实现,如果熟练OpenGL,那就事半工倍了。这里并不打算细说 OpenGLES 的用法。直接参考 UIDemo, 它简单显示了一个经典的三角形,点击会出现变化。
有人会说了,我不会 OpenGL, 况且就算会,要用它来实现一套GUI也是很困难的事性。
稍安勿躁,之所以先给这个例子就是让大家清楚它的起源 ^_^ 。
接下来就是一个直正的GUI的例子。
这个应用用 Tree 来显示 Android property (shell getprop),它是通过使用 Dear ImGui 封装的 Go 版本 来显示 Android 所有属性。
因为 Dear Imgui 源码是 C++,因此这次编译要用到 g++ ,还是先要配置一下环境,这次比较简单,先按上一篇所说的配置好,在上一篇的基础上:
set CXX=%GCC_H%-g++
set CGO_CXXFLAGS=-D__ANDROID_API__=%SDK_API%
这样就可以编译过了。
可能有老鸟会感觉出来总觉得缺点什么?恭喜你,你的感觉是对的,还没有说明关于 C++ 头文件相关的操作,幸运的是 Dear Imgui 并没有 include C++ 的头文件,所以上面没有提到它。如果要用到,类似的考考考(拷贝)就好了。
通过这一节,同时也了解了在 golang 中集成 C++ 代码的环境配置方法。
这里就不详细讲 Dear Imgui 的用法了,看看它的例子就差不多能会。
如果上面的实例在你的手机上运行出现问题,首先确认你的手机内存和分辨率,会不会分辨率高但内存少,特别是用仿真或低端机,仿真可能是配置的时候没配置合适的内存大小。
可用 adb shell free
查看一下内存大小。
也可以通过把增加 WINDOWSCALE 来减少内存的使用。
在之后的例子中看到有这样的代码
item := cam.previewIndex
if imgui.Combo("pixels", &item, cam.comboText) {
cam.previewIndex = item
}
这里为什么不直接写imgui.Combo("pixels", &cam.previewIndex, cam.comboText)
? 请参考cgocheck的说明。如果不这样会出现panic,这个需要特别注意。
这里有 Go 中文文档 。
如果想尝试其它的 GUI 还有其它选择。这里列举其中一些。
关于 Golang 版的 Dear Imgui 有另外两个:
nuklear C 版本的单文件 GUI
gxui 纯 Go 实现的 GUI (实验性的,已不再维护)
NanoGUI.go 也是纯 Go