## uts for Android 本文旨在帮助Android开发者,快速上手UTS。 需要阅读者具备Android原生应用开发经验。 ## 1 了解UTS插件是什么 `UTS插件`是`uni-app`新型插件形式,拥有跨平台,高效率,易调试等优点。[详情](https://uniapp.dcloud.net.cn/plugin/uts-plugin.html#) 对于Android开发者来说,我们需要了解的是: 1. 编译时:当我们在保存`UTS`源码文件时,IDE会同步将其编译为对应的Kotlin代码。 2. 运行时:在真机运行/云打包时,这些编译后的kotlin源码也会成为apk的一部分 ## 2 掌握UTS语法 ### 2.1 对于掌握kotlin语言者 因为UTS语法与kotlin很类似,建议快速阅读后,在实践中掌握这UTS语法。[uts语法介绍](https://uniapp.dcloud.net.cn/tutorial/syntax-uts)。 ### 2.2 对于仅掌握java语言者 与js相比,uts的语法和java更加类似。但是依然存在较大的差异,需要详细阅读2.3语法部分。 尽管开发UTS插件,并不要求一定掌握kotlin,但是鉴于`UTS`目前在android平台上,会编译为kotlin源码。学会kotlin语言,方便排查问题和复杂功能实现。 因此建议学习一下kotlin语法。 + kotlin [https://kotlinlang.org/](https://kotlinlang.org/) + kotlin for android [https://developer.android.com/kotlin](https://developer.android.com/kotlin) ### 2.3 数据类型差异 虽然 UTS 和 koltin 在数据类型上基本保持了一致,但是在部分场景下,还是会有差异,在此特别说明 原则上: **数据类型以UTS 内置的类型为准, 各原生平台都会对其自动适配。** **但是 UTS本身是跨平台语言,当具体平台的api 有明确要求时,需要以对方明确要求的数据类型为准。** ------------------------- #### 举例一: Int 和Number 默认情况下`UTS` 开发者可以使用 `Number` 覆盖`android` 平台上使用 `Int`的场景。 但是当开发者重写 `Service` 组件`onStartCommand` 方法时,`Android` API要求 明确要求后两个参数 必须为Int 原生开发环境中,应该这样写: ```kotlin override fun onStartCommand(intent: Intent, flags: Int, startId: Int): Int { return super.onStartCommand(intent, flags, startId); } ``` 标准的TS环境中,只有`Number`类型而没有`Int`类型 为了适应这种情况,UTS 允许开发者使用原生平台的数据类型Int,来满足原生API对数据类型的要求: ```ts override onStartCommand(intent:Intent ,flags:Int ,startId:Int):Int { return super.onStartCommand(intent, flags, startId); } ``` #### 举例二:`MutableList` `MutableList`是`android`平台 特有的数据类型,一般场景下,可以使用UTS中内置类型 `Array` 替代 但是在 调用`onAppActivityRequestPermissionsResult` 函数监听权限申请结果时,明确要求使用此类型的参数 在原生环境中,应该这样写: ```kotlin onAppActivityRequestPermissionsResult(fun(requestCode: Number, permissions: MutableList, grantResults: MutableList){ }); ``` 标准的TS环境中,没有`MutableList`类型,与之相近的数据类型是 `Array` 为了适应这种情况,UTS 允许开发者使用原生平台的数据类型`MutableList`,来满足原生平台API对数据类型的要求: ```ts onAppActivityRequestPermissionsResult((requestCode: number,permissions: MutableList,grantResults: MutableList) => { }); ``` ## 3 Android原生环境配置 对于Android项目来说,除了源码之外,还会涉及依赖,资源,配置等常见问题 本章节将会介绍,UTS插件开发环境中如何配置这些属性 注意: + 1 本章节内的实例代码均取自Hello UTS [项目地址](https://gitcode.net/dcloud/hello-uts) + 2 本章节设计的配置,均需自定义基座后才能生效 + 3 R文件的自动生成,已经在HBuilder X 3.6.9 版本支持,请使用最新版本开发 ### 3.1 配置AndroidManifest.xml 以hello UTS中的native-page插件中的配置文件为例: 示例文件在hello uts中的位置: ~\uni_modules\uts-nativepage\utssdk\app-android\AndroidManifest.xml AndroidManifest.xml示例: ```xml // 配置权限 // 配置service / activity ``` AndroidManifest.xml配置规则与android中的规则是一致的。 特别提示: 每一个UTS插件对应android项目中的一个 lib module. 与你在android studio中手动输入包名不同的是,如果你没有手动包名,HX会按照下面的规则默认生成一个: ``` uts插件默认包名规则: 如果是根目录utssdk下的uts插件 包名:uts.sdk.(插件ID转驼峰) 如果是uni_modules目录下的uts插件 包名:uts.sdk.modules.(插件ID转驼峰) 举例: uni-getbatteryinfo -> uts.sdk.modules.uniGetbatteryinfo; uts-nativepage -> uts.sdk.modules.utsNativepage ``` ### 3.2 配置res资源 ![](https://native-res.dcloud.net.cn/images/uts/forAndroid/uts_android_res_folder.jpg) 示例文件在hello uts中的位置: ~\uni_modules\uts-nativepage\utssdk\app-android\res 除了这里列出的layout、values目录外,还支持anim等所有android标准资源目录 ### 3.3 配置asset资源 以hello UTS中的uts-advance插件为例。 ![目录结构](https://native-res.dcloud.net.cn/images/uts/forAndroid/uts_android_assets_folder.jpg) 关键代码: ```ts // 获取asset管理器 let assetManager = getAppContext()!.getAssets(); // 加载free.mp3 资源 let afd = assetManager.openFd("free.mp3"); // 使用android 自带的媒体组件进行播放 let mediaPlayer = new MediaPlayer(); mediaPlayer.setDataSource(afd.getFileDescriptor(),afd.getStartOffset(), afd.getLength()); mediaPlayer.prepare(); mediaPlayer.start(); ``` 完整的代码在hello uts中的位置: ~\uni_modules\uts-advance\utssdk\app-android\assets ### 3.4 增加libs依赖资源 以Hello UTS项目下的uts-tencentgeolocation 插件为例 ![](https://native-res.dcloud.net.cn/images/uts/forAndroid/uts_android_libs_folder.jpg) 示例文件在hello uts中的位置: ~\uni_modules\uts-tencentgeolocation\utssdk\app-android\libs ------ HX3.6.7 版本内置了以下依赖 开发者在使用列表中的依赖时,需要注意两点: + 真机运行时,不需要添加列表中的依赖,即可直接引用相关类 + 请勿通过 手动添加jar/aar 等方式引入相同的依赖,否则会因依赖冲突导致云打包失败。 ```gradle +--- my-imageloader.jar +--- my-nineoldandroids-2.4.0.jar +--- zip4j-2.8.0.jar +--- uts-runtime-jvm-1.0.jar +--- com.google.code.gson:gson@2.8.9.jar +--- android-gif-drawable-release@1.2.23.aar +--- msa_mdid_1.0.13.aar +--- breakpad-build-release.aar +--- androidx.multidex:multidex:2.0.0@aar +--- androidx.recyclerview:recyclerview:1.0.0@aar +--- androidx.legacy:legacy-support-v4:1.0.0@aar +--- androidx.appcompat:appcompat:1.0.0@aar +--- com.github.bumptech.glide:glide:4.9.0@aar +--- com.alibaba:fastjson:1.1.46.android@jar +--- androidx.fragment:fragment:1.0.0@aar +--- androidx.vectordrawable:vectordrawable-animated:1.0.0@aar +--- androidx.legacy:legacy-support-core-ui:1.0.0@aar +--- androidx.media:media:1.0.0@aar +--- androidx.legacy:legacy-support-core-utils:1.0.0@aar +--- androidx.vectordrawable:vectordrawable:1.0.0@aar +--- androidx.viewpager:viewpager:1.0.0@aar +--- androidx.coordinatorlayout:coordinatorlayout:1.0.0@aar +--- androidx.drawerlayout:drawerlayout:1.0.0@aar +--- androidx.slidingpanelayout:slidingpanelayout:1.0.0@aar +--- androidx.customview:customview:1.0.0@aar +--- androidx.swiperefreshlayout:swiperefreshlayout:1.0.0@aar +--- androidx.asynclayoutinflater:asynclayoutinflater:1.0.0@aar +--- androidx.loader:loader:1.0.0@aar +--- androidx.core:core:1.0.0@aar +--- androidx.versionedparcelable:versionedparcelable:1.0.0@aar +--- androidx.collection:collection:1.0.0@jar +--- androidx.cursoradapter:cursoradapter:1.0.0@aar +--- com.github.bumptech.glide:gifdecoder:4.9.0@aar +--- androidx.lifecycle:lifecycle-runtime:2.0.0@aar +--- androidx.interpolator:interpolator:1.0.0@aar +--- androidx.documentfile:documentfile:1.0.0@aar +--- androidx.localbroadcastmanager:localbroadcastmanager:1.0.0@aar +--- androidx.print:print:1.0.0@aar +--- androidx.lifecycle:lifecycle-viewmodel:2.0.0@aar +--- androidx.lifecycle:lifecycle-livedata:2.0.0@aar +--- androidx.lifecycle:lifecycle-livedata-core:2.0.0@aar +--- androidx.lifecycle:lifecycle-common:2.0.0@jar +--- androidx.arch.core:core-runtime:2.0.0@aar +--- androidx.arch.core:core-common:2.0.0@jar +--- androidx.annotation:annotation:1.0.0@jar +--- com.github.bumptech.glide:disklrucache:4.9.0@jar \--- com.github.bumptech.glide:annotations:4.9.0@jar ``` ## 4 Android内置库@iodcloudutsandroid 在uts里,Android的所有api都可以访问。 但Android开发中经常要复写application和activity,uni-app主引擎已经复写了相关类。所以想要操作application和activity,需要调用uni-app引擎封装的API。 这些api在`io.dcloud.uts.android`库中,具体见下。 ### 4.1 application 上下文相关 #### 4.1.1 getAppContext > HBuilderX 3.6.3+ ```ts import { getAppContext } from "io.dcloud.uts.android"; ``` 用法说明:获取当前应用Application上下文,对应android平台 Context.getApplicationContext 函数实现 Android开发场景中,调用应用级别的资源/能力,需要使用此上下文。更多用法,参考[Android官方文档](https://developer.android.google.cn/docs) ```ts // [示例]获取asset下的音频,并且播放 let assetManager = getAppContext()!.getAssets(); let afd = assetManager.openFd("free.mp3"); let mediaPlayer = new MediaPlayer(); mediaPlayer.setDataSource(afd.getFileDescriptor(),afd.getStartOffset(), afd.getLength()); mediaPlayer.prepare(); mediaPlayer.start(); ``` #### 4.1.2 getResourcePath(resourceName:String) > HBuilderX 3.6.3+ ```ts import { getResourcePath } from "io.dcloud.uts.android"; ``` 获取指定插件资源的运行期绝对路径 ```ts // [示例]获取指定资源路径 // 得到文件运行时路径: `/storage/emulated/0/Android/data/io.dcloud.HBuilder/apps/__UNI__3732623/www/uni_modules/test-uts-static/static/logo.png` getResourcePath("uni_modules/test-uts-static/static/logo.png") ``` #### 4.1.3 onAppTrimMemory / offAppTrimMemory ##### onAppTrimMemory > HBuilderX 3.6.8+ ```ts import { onAppTrimMemory } from "io.dcloud.uts.android"; ``` App 内存不足时,系统回调函数 对应原生的API: onTrimMemory ```ts onAppTrimMemory((level:Number) => { let eventName = "onAppTrimMemory - " + level; console.log(eventName); }); ``` ##### offAppTrimMemory > HBuilderX 3.6.9+ ```ts import { offAppTrimMemory } from "io.dcloud.uts.android"; ``` onAppTrimMemory 对应的反注册函数 如果传入的函数可为空,如果为空,则视为移除所有监听 ```ts // 移除所有监听 offAppTrimMemory() // 移除指定监听 offAppTrimMemory((level:Number) => { }); ``` #### 4.1.4 onAppConfigChange / offAppConfigChange ##### onAppConfigChange > HBuilderX 3.6.8+ ```ts import { onAppConfigChange } from "io.dcloud.uts.android"; ``` App 配置发生变化时触发,比如横竖屏切换 对应原生的API: onConfigurationChanged ```ts onAppConfigChange((ret:UTSJSONObject) => { let eventName = "onAppConfigChange - " + JSON.stringify(ret); console.log(eventName); }); ``` ##### offAppConfigChange > HBuilderX 3.6.9+ ```ts import { offAppConfigChange } from "io.dcloud.uts.android"; ``` 与onAppConfigChange 对应的反注册函数 如果传入的函数可为空,如果为空,则视为移除所有监听 ```ts // 移除所有监听 offAppConfigChange(); // 移除指定监听 offAppConfigChange(function(ret){ }); ``` -------------------------------- 特别说明:除了本章节列出的函数外,android环境下 application 其他上下文方法都可以通过 getAppContext()!.xxx()的方式实现 比如获取app缓存目录: ``` getAppContext()!.getExternalCacheDir()!.getPath() ``` ### 4.2 Activity 上下文 #### 4.2.1 getUniActivity > HBuilderX 3.6.3+ ```ts import { getUniActivity } from "io.dcloud.uts.android"; ``` 获取当前插件所属的activity实例,对应android平台 getActivity 函数实现 Android开发场景中,调用活动的级别的资源/能力,需要使用此上下文。更多用法,参考[Android官方文档](https://developer.android.google.cn/docs) ```ts // [示例]获取当前activity顶层容器 let frameContent = decorView.findViewById(android.R.id.content) ``` #### 4.2.2 onAppActivityPause / offAppActivityPause ##### onAppActivityPause > HBuilderX 3.6.3+ ```ts import { onAppActivityPause } from "io.dcloud.uts.android"; ``` App的activity onPause时触发 ```ts onAppActivityPause(() => { let eventName = "onAppActivityPause - " + Date.now(); console.log(eventName); }); ``` ##### offAppActivityPause > HBuilderX 3.6.9+ onAppActivityPause 对应的反注册函数 如果传入的函数可为空,如果为空,则视为移除所有监听 ```ts import { offAppActivityPause } from "io.dcloud.uts.android"; ``` ```ts // 移除全部监听 offAppActivityPause(); // 移除指定监听 offAppActivityPause(() => { }); ``` #### 4.2.3 onAppActivityResume / offAppActivityResume ##### onAppActivityResume > HBuilderX 3.6.3+ ```ts import { onAppActivityResume } from "io.dcloud.uts.android"; ``` App的activity onResume时触发 ```ts onAppActivityResume(() => { let eventName = "onAppActivityResume - " + Date.now(); console.log(eventName); }); ``` ##### offAppActivityResume > HBuilderX 3.6.9+ onAppActivityResume 对应的反注册函数 如果传入的函数可为空,如果为空,则视为移除所有监听 ```ts import { onAppActivityResume } from "io.dcloud.uts.android"; ``` ```ts // 移除全部监听 onAppActivityResume(); // 移除指定监听 onAppActivityResume(() => { }); ``` #### 4.2.4 onAppActivityDestroy / offAppActivityDestroy ##### onAppActivityDestroy > HBuilderX 3.6.3+ ```ts import { onAppActivityDestroy } from "io.dcloud.uts.android"; ``` App 的 activity onDestroy时触发 ```ts onAppActivityDestroy(() => { let eventName = "onAppActivityDestroy- " + Date.now(); console.log(eventName); }); ``` ##### offAppActivityDestroy > HBuilderX 3.6.9+ onAppActivityDestroy 对应的反注册函数 如果传入的函数可为空,如果为空,则视为移除所有监听 ```ts import { offAppActivityDestroy } from "io.dcloud.uts.android"; ``` ```ts // 移除全部监听 offAppActivityDestroy(); // 移除指定监听 offAppActivityDestroy(() => { }); ``` #### 4.2.5 onAppActivityBack / offAppActivityBack ##### onAppActivityBack > HBuilderX 3.6.3+ ```ts import { onAppActivityBack } from "io.dcloud.uts.android"; ``` App 的 activity 回退物理按键点击时触发 ```ts onAppActivityBack(() => { let eventName = "onAppActivityBack- " + Date.now(); console.log(eventName); }); ``` ##### offAppActivityBack > HBuilderX 3.6.9+ onAppActivityBack 对应的反注册函数 如果传入的函数可为空,如果为空,则视为移除所有监听 ```ts import { offAppActivityBack } from "io.dcloud.uts.android"; ``` ```ts // 移除全部监听 offAppActivityBack(); // 移除指定监听 offAppActivityBack(() => { }); ``` #### 4.2.6 onAppActivityResult / offAppActivityResult ##### onAppActivityResult > HBuilderX 3.6.8+ ```ts import { onAppActivityResult } from "io.dcloud.uts.android"; ``` App 的 activity 启动其他activity的回调结果监听 对应原生的 onActivityResult ```ts onAppActivityResult((requestCode: Int, resultCode: Int, data?: Intent) => { let eventName = "onAppActivityResult - requestCode:" + requestCode + " -resultCode:"+resultCode + " -data:"+JSON.stringify(data); console.log(eventName); }); ``` ##### offAppActivityResult > HBuilderX 3.6.9+ onAppActivityResult 对应的反注册函数 如果传入的函数可为空,如果为空,则视为移除所有监听 ```ts import { offAppActivityResult } from "io.dcloud.uts.android"; ``` ```ts // 移除全部监听 offAppActivityResult(); // 移除指定监听 offAppActivityResult(() => { }); ``` #### 4.2.7 onAppActivityRequestPermissionsResult / offAppActivityRequestPermissionsResult ##### onAppActivityRequestPermissionsResult > HBuilderX 3.6.3+ ```ts import { onAppActivityRequestPermissionsResult } from "io.dcloud.uts.android"; ``` App 的 activity 获得权限请求结果的回调 ```ts onAppActivityRequestPermissionsResult((requestCode: number, permissions: MutableList, grantResults: MutableList) => { console.log(grantResults); console.log(permissions); console.log(requestCode); }); //发起定位权限申请 ActivityCompat.requestPermissions(getUniActivity()!, arrayOf(Manifest.permission.ACCESS_COARSE_LOCATION), 1001); ``` ##### offAppActivityRequestPermissionsResult > HBuilderX 3.6.9+ onAppActivityRequestPermissionsResult 对应的反注册函数 如果传入的函数可为空,如果为空,则视为移除所有监听 ```ts import { offAppActivityRequestPermissionsResult } from "io.dcloud.uts.android"; ``` ```ts // 移除全部监听 offAppActivityRequestPermissionsResult(); // 移除指定监听 offAppActivityRequestPermissionsResult(() => { }); ----------------------------- 特别说明:除了本章节列出的函数外,android环境下 activity 其他上下文方法都可以通过 getUniActivity()!.xxx()的方式实现 比如获取当前activity的顶层View容器 ```ts getUniActivity()!.getWindow().getDecorView(); ``` ## 5 Kotlin与UTS差异重点介绍 (持续更新) 通过上面的章节的阅读。 至此我们认为你已经掌握了UTS语法,掌握了基本的Kotlin语法,掌握了UTS对于android资源的支持。 但是对于一个熟悉android开发的kotlin语言者来说,有很多常用的习惯发生了改变,我们会在这个章节特别指出,便于开发者加深认识。 ### 5.1 语法差异 ------------------------------- #### 5.1.1 可为空的语法标识 kotlin中可为空的语法统一为类型后加`?`,以下面的代码为例 ```ts // 一个可为空的字符串变量,变量名为user var user:String? = null ``` 但是ts中分两种情况,如果是全局变量,可为空,需要这样写 ```ts let user:string | null ``` 如果是成员变量,与kotlin类似,但是区别在于?写在变量后,而非类型后 ```ts let user?:string ``` #### 5.1.2 let和var `kotlin`中 可变变量修饰为 `var`、`val`。 区别在于 val 不可变,var可变。 `uts`中对应`var`的变量类型为 `var/let` 推荐使用`let` 因为只会在作用域内生效,需要慎用`var`,因为它具备有更大的作用范围 #### 5.1.3 方法定义 方法定义 `kotlin`里的方法只有一种定义方式 ```kotlin fun startListener():void{ } ``` uts中,需要区分全局方法、成员方法 ```ts // 成员方法 startListener():void{ } ``` ```ts // 全局方法 function startListener():void{ } ``` #### 5.1.4 extends `kotlin`中的: 继承操作符,需要用`extends`取代 |语法|kotlin|uts| |---|-------|---| |继承类|:|extends| |实现接口|:|extends| ```kotlin class MediaContentObserver : ContentObserver { } ``` ```ts class MediaContentObserver extends ContentObserver { } ``` #### 5.1.5 非空断言 kotlin中的非空断言是`!!`,ts中是一个`!` ```ts user!.sayHello(); ``` ```kotlin user!!.sayHello(); ``` #### 5.1.6 快速调用父类实现 ```ts //ts 中快速实现super constructor() : super() { } ``` ```kotlin //kotlin 中快速实现super constructor (){ super(); } ``` #### 5.1.7 匿名内部类 `kotlin`中可以使用匿名内部类 ```kotlin // kotlin 新建事件监听 user.setListener(Listener(){ //todo }); ``` 目前版本UTS还不支持匿名内部类,需要显性的声明再新建 ```ts // 声明一个新的类,实现Listener class MyListener extends Listener{ // todo } // 新建实例 let myListener = new MyListener(); user.setListener(myListener); ``` #### 5.1.8 可为空函数调用 有一种特殊场景,我们需要定义一些可为空的函数变量,比如下面的 success,fail: ```ts type Option = { success?: (res: object) => void; fail?: (res: object) => void; }; ``` 这个时候我们需要这样调用 ```ts options.success?.(res) ``` 这样的调用方式在kotlin中是非法的,属于TS中的特有语法,需要特别注意。 #### 5.1.9 一个类只能有一个构造函数 在`Kotlin`/`java`中允许一个函数有多个构造器,但是UTS中是不被允许的 #### 5.1.10 界面跳转写法 android开发中场景的 intent跳转需要传入 目标界面的class对象,目前UTS中仅支持一种写法 ```uts let intent = new Intent(getUniActivity(),DemoActivity().javaClass); getUniActivity()!.startActivity(intent); ``` #### 5.1.11 指定double数据类型 某些场景下开发者需要获得 指定double数据类型的数据 前端常用的写法: ``` let a:Int =3 let b:Int =4 let c:Double = a/b ``` 但是Android原生环境中,数据类型的精度是向下兼容的,如果想要获得一个double类型,必须要有一个double类型参与运算: ``` let a:Int =3 let b:Int =4 let c:Double = a * 1.0 / b ``` --------------------------------- ### 5.2 警告优化 下面的内容不会影响功能使用,但是在UTS环境中,有合适的解决办法 #### 5.2.1 java lang包的引入问题 `kotlin` 或者`java` 中java.lang.*是被特殊处理的,可以直接使用而不需要引入。 ```kotlin // 获取当前时间戳 System.currentTimeMillis() ``` UTS环境中,lang包没有被特殊对待,需要手动引入。 ```ts // 手动引入lang包下的类 import System from 'java.lang.System'; // 获取当前时间戳 System.currentTimeMillis() ``` #### 5.2.2 `UTS` 不建议使用 快捷构造 `kotlin` 中 支持通过()的方式,快速实现无参构造器的声明 ```kotlin // 获取当前时间戳 class ScreenReceiver extends BroadcastReceiver(){ } ``` UTS环境中,不建议这样做(虽然目前这样做不会影响编译),建议使用手动声明无参构造 ```ts class ScreenReceiver extends BroadcastReceiver{ constructor (){ super(); } } ``` #### 5.2.3 `UTS` 中下划线前缀的变量,有屏蔽未使用警告的含义 ```ts // IDE会提示 name,status,desc 变量未使用 onStatusUpdate(name:string, status:Int, desc:string){ } // 不会警告变量未使用 onStatusUpdate(_name:string, _status:Int, _desc:string){ } ``` ## 6 常见问题(持续更新) ### 6.1 如何在UTS环境中,新建一个`activity`? 参考Hello UTS项目中的uts-nativepage插件 路径: > ~\uni_modules\uts-nativepage ### 6.2 如何在UTS环境中,新建一个`service`? 参考Hello UTS项目中的uts-nativepage插件 路径: > ~\uni_modules\uts-nativepage ### 6.3 如何在UTS环境中,新建一个`Thread`? 简单示例 ```uts class CustomThread extends Thread{ constructor(){ super(); } override run(){ Thread.sleep(1000) console.log("CustomThread = " + Thread.currentThread().getName()) } } ``` 完整示例参考Hello UTS项目中的uts-nativepage插件 路径: > ~\uni_modules\uts-nativepage ## 7 已知待解决问题(持续更新) ### 7.1 结构入参 boolean 参数默认为true 当以type 结构体为参数时,其内部boolean字段 默认值为false,不支持指定。