uts-for-ios.md 27.9 KB
Newer Older
D
DCloud_LXH 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
## uts for iOS

本文旨在帮助 iOS 开发者,快速上手 UTS。

需要阅读者具备 iOS 原生应用开发经验。

## 1 了解 UTS 插件是什么

UTS 插件是 uni-app 新型插件形式 [详情](/plugin/uts-plugin)

对于 iOS 开发者来说,我们需要了解的是:

1. 编译时:当我们在保存 UTS 源码文件时,IDE 会同步将其编译为对应的 swift 代码,并且会生成一个对应的插件 Framework 工程在编译出对应的`framework`依赖库
2. 运行时:在真机运行/云打包时,会将`framework`依赖库添加到打包工程生成最终的 ipa 包

## 2 掌握UTS语法

### 2.1  对于掌握 swift 语言者

因为 UTS 语法与 swift 较类似,建议快速阅读后,在实践中掌握 UTS 语法。[uts语法介绍](/uts/)

### 2.2  对于仅掌握`objective-c`语言者

尽管开发 UTS 插件,并不要求一定掌握 swift,但是鉴于 UTS 目前在 iOS 平台上,会编译为 swift 源码,掌握 swift 语言,方便排查问题和复杂功能实现。

因此建议学习一下 swift 语法,推荐阅读

+ [swift 官方文档](https://docs.swift.org/swift-book/)
+ [swift 中文版](https://swiftgg.gitbook.io/swift/)

### 2.3 数据类型差异

UTS 和 swift 在数据类型上基本保持了一致,但是在部分场景下,还是会有差异,在此特别说明

D
DCloud_LXH 已提交
35
原则上:
D
DCloud_LXH 已提交
36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52

**数据类型以 UTS 内置的类型为准, 各原生平台都会对其自动适配。**

**当具体平台的 api 参数无法使用 UTS 类型兼容时,允许以对方明确要求的数据类型为准。**

-------------------------


#### 举例一: Int、Float、Double 和 Number

UTS 中不存在 Int、Float、Double 类型开发者在开发过程中应该使用  Number 类型覆盖 iOS 平台上使用 Int、Float、Double 的场景

但是当开发中需要重写系统方法或实现第三方依赖库的协议方法(delegate 方法)时,比如 API 明确要求参数为 Int 时,则需要写原生的类型 Int

下面以一个协议方法为例,需要实现一个三方依赖库中定义的协议方法

```swift
D
DCloud_LXH 已提交
53
// swift
D
DCloud_LXH 已提交
54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142
// 此协议定义在其他三方 SDK 中
protocol TestProtocol {
   func addTwoInts(_ a: Int, _ b: Int) -> Int
}
```

我们在 UTS 中需要实现上面三方库中的协议方法时,由于参数和返回值类型都要求是 Int 类型,为了适应这种情况,UTS 允许开发者使用原生平台的数据类型 Int,来满足原生 API 对数据类型的要求:

```ts
// UTS 中实现协议方法
class TestClass implements TestProtocol {
	addTwoInts(a: Int, b: Int): Int {
		return a + b
	}
}
```

注意:UTS 中使用 `implements` 关键字表示遵循某个协议,下文会有详细说明

## 3 iOS 原生环境配置

对于 iOS 项目来说,除了源码之外,还会涉及依赖,资源,工程配置等常见问题

本章节将会介绍,UTS插件开发环境中如何配置这些属性

注意:

+ 1 本章节内的实例代码均取自Hello UTS [项目地址](https://gitcode.net/dcloud/hello-uts)
+ 2 本章节设计的配置,均需自定义基座后才能生效

### 3.1 配置 Info.plist

当插件需要在原生工程 Info.plist 中添加配置项时,需要在插件的 app-ios 目录中创建 Info.plist 文件

以 hello uts 中的 uts-tencentgeolocation 腾讯定位插件中的配置文件为例:

示例文件在 hello uts 中的位置:

`~/uni_modules/uts-tencentgeolocation/utssdk/app-ios/Info.plist`

此示例表示需要在 Info.plist 中配置 APIKey 及开启后台定位权限

Info.plist 示例:

```xml
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
  <dict>
	<key>TencentLBSAPIKey</key>
	<string>您申请的APIKey</string>
	<key>UIBackgroundModes</key>
	<array>
		<string>location</string>
	</array>
  </dict>
</plist>
```

Info.plist 格式及配置规则与 iOS 工程中是一致的,云端打包时会将配置信息合并到原生工程的 Info.plist 中

### 3.2 配置 entitlements
> HBuilder X 3.6.11+ 版本支持

当插件需要开启 capabilities 中的相关服务时,需要在插件的 app-ios 目录中创建 UTS.entitlements 文件

比如需要在 capabilities 中勾选 Access WiFi Information 项,对应的 UTS.entitlements 的配置示例:

```xml
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
	<key>com.apple.developer.networking.wifi-info</key>
	<true/>
</dict>
</plist>
```

UTS.entitlements 格式及配置规则与 iOS 工程中是一致的,云端打包时会将配置信息合并到原生工程的 entitlements 配置文件中

### 3.3 依赖资源文件

如果您的插件需要依赖资源文件比如图片,音频等,可将资源文件放到插件目录下  `~/utssdk/app-ios/Resources/`路径中

云端打包时会将此目录下的所有文件添加到应用 main bundle 中,建议只保存 uts 插件内置的资源文件。

### 3.4 依赖三方库

lizhongyi_'s avatar
lizhongyi_ 已提交
143
uts 插件支持依赖三方库,目前支持 framework、xcframework、.a库
D
DCloud_LXH 已提交
144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225

#### 3.4.1 framework依赖库说明

需要将依赖的framework或者xcframework文件存放到插件目录下  `~/utssdk/app-ios/Frameworks/`路径中

云端打包时会将此目录中所有的依赖库添加到工程中,建议只存放与插件相关的依赖库

以 hello uts 中的 uts-tencentgeolocation 腾讯定位插件为例,本插件需要依赖腾讯定位库 `TencentLBS.framework`,则将依赖库存放到 `~/uni_modules/uts-tencentgeolocation/utssdk/app-ios/Framework/TencentLBS.framework` 位置即可

#### 3.4.2 .a依赖库相关说明
##### 3.4.2.1 .a库存放的目录结构

<pre v-pre="" data-lang="">
	<code class="lang-" style="padding:0">
└─Libs                         				// .a库存放目录
	├─MyStaticLibA                 			//A静态库(该库所有文件放在此文件夹内,OC库)
	│	├─libMyStaticLib.a                  //.a文件,必须
	│	├─MyStaticLib.h                    	//A.a库对应的头文件,必须
	│	├─MyClassA.h                     	//需要暴露的头文件A,可选
	│	└─MyClassB.h              		 	//需要暴露的头文件B,可选
	└─TestSwiftLibrary                      //B静态库(该库所有文件放在此文件夹内,Swift库)
		├─libTestSwiftLibrary.a             //.a文件,必须
		└─TestSwiftLibrary.swiftmodule      //.swiftmodule文件夹,必须
</code>
</pre>

注意:
- 将.a库的所有文件存放在一个文件夹内,多个.a库就创建多个文件夹;
- 未对某个.a库文件夹下的文件做递归查找,请不要将.a或.h文件嵌套在多层文件夹内,以免发生错误;

##### 3.4.2.2 .a库的使用说明

- OC语言创建的.a库在使用时无需import,可直接使用;
- Swift语言创建的.a库在使用前需要在uts文件中import;
- HBuilder X目前暂不支持.a库相关代码的语法提示;

##### 3.4.2.3 .a库的使用示例

- OC语言创建的.a库使用示例:

```ts
// uts
const aResult = ToolA.toolAMethod();
const bResult = ToolB.toolBMethod();
const libResult = TestLib.testLib();

const res = {
	aResult: aResult,
	bResult: bResult,
	libResult: libResult
};
options.success?.(res);
```

- Swift语言创建的.a库使用示例:

```ts
// uts
import { Tool, Manager, TestLibraryExa } from 'TestLibraryExa';

Manager.testManager();
Tool.testTool()
let lib = TestLibraryExa();
lib.test()
console.log(lib.version);
```

#### 3.4.3 不包含 Modules 的 framework 使用说明@nomodules

有些第三方的 SDK 使用 OC 语言开发,且产物 .framework 文件里不包含 Moudules 文件夹。这就造成该 SDK 不支持 use module 模式,不能直接在 Swift 文件中导入。
这样的 SDK 不能直接被 uts 插件引用,需要做以下处理:

##### 有源码的情况

如果你有 SDK 的源码,那么有以下几种方法可以生成 Modules。
以名称为 `TestSDK` 的 framework 为例:

+ 通过创建和 SDK 的 target 同名的头文件方式:
	- 打开 XCode, 创建一个和 SDK target 同名的 `.h` 文件,如 `TestSDK.h`
	- 将需要暴露的 public 文件 导入到 `TestSDK.h` 中,如 `#import <TestSDK/TestA.h>`;
	-`target` -> `Build Phases` -> `Headers` 中将刚创建的 `TestSDK.h` 设置为 `public`;
	- 重新编译 SDK, 编译后可以看到 Modules 已经生成。
D
DCloud_LXH 已提交
226

D
DCloud_LXH 已提交
227 228 229 230 231
+ 通过自定义 Module Map 的方式:
	- 打开 XCode, 在 `TestSDK` SDK 源码目录下创建 ` module.map.modulemap ` 文件;
	- 在上述文件中键入下面代码中类似的内容,下述代码是以 `TestSDK` 为例,实践时需要改为自己的 SDK 和文件名;
	-`target` -> `Build Settings` -> `Module Map File` 设置 `Module Map File``$(PROJECT_DIR)/TestSDK/module.map.modulemap`;
	- 重新编译 SDK, 编译后可以看到 Modules 已经生成。
D
DCloud_LXH 已提交
232

D
DCloud_LXH 已提交
233 234 235
```ts
framework module TestSDK {
    header "TestA.h"   //需要对外暴露的头文件,需要为 plubic 的文件
D
DCloud_LXH 已提交
236

D
DCloud_LXH 已提交
237 238 239 240 241 242 243
	header "TestB.h"  //需要对外暴露的头文件,需要为 plubic 的文件

    export *
}
```

##### 无源码的情况
D
DCloud_LXH 已提交
244

D
DCloud_LXH 已提交
245 246 247 248 249 250 251 252
如果使用的是第三方非开源的 SDK, 那么可以使用下面的方式来生成 Modules:
`TestSDK` 为例:

+`TestSDK.framework` 文件夹下创建 `Modules` 文件夹;
+`Modules` 文件夹下创建 `module.modulemap` 文件;
+ 在上述文件中键入下述的代码(其中的 .h 文件都要是 TestSDK.framework -> Headers 文件夹里的头文件)。
+ 至此 TestSDK 就可以直接放在 uts 插件中使用了。

D
DCloud_LXH 已提交
253
> 注意:
D
DCloud_LXH 已提交
254 255 256 257 258
> 实践时要将 `TestSDK` 改成你要操作的 SDK 名称,.h 文件也要改成你要暴露的头文件名字。

```ts
framework module TestSDK {
	// 下面的.h 文件都要是 TestSDK.framework -> Headers 文件夹下的头文件
D
DCloud_LXH 已提交
259 260
    header "AClass.h"

D
DCloud_LXH 已提交
261
    header "BClass.h"
D
DCloud_LXH 已提交
262

D
DCloud_LXH 已提交
263
    header "CClass.h"
D
DCloud_LXH 已提交
264

D
DCloud_LXH 已提交
265
    header "DClass.h"
D
DCloud_LXH 已提交
266

D
DCloud_LXH 已提交
267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302
    export *
}

```



## 4 iOS 平台内置库 DCloudUTSFoundation
> HBuilder X 3.6.11+ 版本支持

DCloudUTSFoundation 为框架内置库,所有 uts 插件都会依赖此基础库

DCloudUTSFoundation 会封装一些常用方法便于开发者直接调用

使用时需要在 uts 文件中先导入 UTSiOS 类,所有方法都通过 UTSiOS 类调用

```ts
// 从 DCloudUTSFoundation 依赖库中导入 UTSiOS 类
import { UTSiOS } from "DCloudUTSFoundation"
```

### 4.1 getCurrentViewController(): UIViewController
> HBuilder X 3.6.11+ 版本支持

获取当前 app 显示的 UIViewController 实例

以 hello uts 中的 uts-alert 为例:

示例文件在 hello uts 中的位置:

`~/uni_modules/uts-alert/utssdk/app-ios/index.uts`

```ts
export function showAlert(title: string|null, message: string|null, result: (index: Number) => void) {
	// uts方法默认会在子线程中执行,涉及 UI 操作必须在主线程中运行,通过 DispatchQueue.main.async 方法可将代码在主线程中运行
	DispatchQueue.main.async(execute=():void => {
D
DCloud_LXH 已提交
303

D
DCloud_LXH 已提交
304 305
		// 初始化 UIAlertController 实例对象 alert
		let alert = new UIAlertController(title=title,message=message,preferredStyle=UIAlertController.Style.alert)
D
DCloud_LXH 已提交
306

D
DCloud_LXH 已提交
307 308 309 310 311
		// 创建 UIAlertAction 按钮
		let okAction = new UIAlertAction(title="确认", style=UIAlertAction.Style.default, handler=(action: UIAlertAction):void => {
			// 点击按钮的回调方法
			result(0)
		})
D
DCloud_LXH 已提交
312

D
DCloud_LXH 已提交
313 314 315 316 317
		// 创建 UIAlertAction 按钮
		let cancelAction = new UIAlertAction(title="取消", style=UIAlertAction.Style.cancel, handler=(action: UIAlertAction):void => {
			// 点击按钮的回调方法
			result(1)
		})
D
DCloud_LXH 已提交
318

D
DCloud_LXH 已提交
319 320 321
		// 将 UIAlertAction 添加到 alert 上
		alert.addAction(okAction)
		alert.addAction(cancelAction)
D
DCloud_LXH 已提交
322

D
DCloud_LXH 已提交
323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341
		// 打开 alert 弹窗
		UTSiOS.getCurrentViewController().present(alert, animated= true)
	})
}
```


#### 4.2 colorWithString(value: string): UIColor
> HBuilder X 3.6.11+ 版本支持

将字符串色值转换为 UIColor

格式支持

- 精简写法的十六进制 如:#f00
- 十六进制 如:#ff0000
- RGB 如:rgb(255, 0, 0)
- RGBA 如:rgba(255, 0, 0, 0.5)
- 色值关键字,如: red
D
DCloud_LXH 已提交
342

D
DCloud_LXH 已提交
343
示例
D
DCloud_LXH 已提交
344

D
DCloud_LXH 已提交
345 346 347 348 349 350 351 352 353 354 355
```ts
let bgColor = UTSiOS.colorWithString("#000000")
view.backgroundColor = bgColor
```

#### 4.3 getResourcePath(resourceName: string): string
> HBuilder X 3.6.11+ 版本支持

获取指定插件资源的运行期绝对路径

插架资源路径请传该资源在工程目录下的绝对路径
D
DCloud_LXH 已提交
356

D
DCloud_LXH 已提交
357
示例
D
DCloud_LXH 已提交
358

D
DCloud_LXH 已提交
359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441
```ts
const imagePath = UTSiOS.getResourcePath("/static/logo.png")
console.log(imagePath)
const image = new UIImage(contentsOfFile = imagePath)
/* imagePath: "/var/mobile/Containers/Data/Application/FA7080BA-3EF7-4C4E-B7C5-0332539B2964/Documents/Pandora/apps/__UNI__FB95CAB/www/static/logo.png" */

```

持续更新中

## 5 swift 与 UTS 差异重点介绍 (持续更新)

通过上面的章节的阅读。

至此我们认为你已经掌握了UTS语法,掌握了基本的 swift 语法,掌握了 UTS 对于 iOS 资源的支持。

但是对于一个熟悉 iOS 开发的 swift 语言者来说,有很多常用的习惯发生了改变,我们会在这个章节特别指出,便于开发者加深认识。


### 5.1 语法差异

-------------------------------

#### 5.1.1  常量和变量

swift 中用 `let` 来声明常量,用 `var` 来声明变量

```swift
// swift
var str = "abc" // 声明一个字符串变量
let str1 = "abc" // 声明一个字符串常量
```

`uts`中用 `const` 来声明常量,用 `let` 来声明变量

```ts
// swift
let str = "abc" // 声明一个字符串变量
const str1 = "abc" // 声明一个字符串常量
```

#### 5.1.2 可选类型

swift 中的可选类型定义为 `类型?`

```swift
// swift
var user: String? = nil
```

 uts 中可选类型的定义为 `类型 | null`

```ts
// uts
let user: string | null = null
```

#### 5.1.3 调用构造方法

swift 中调用构造方法创建实例对象时不需要使用 `new` 关键字

```swift
var alert = UIAlertController()
```

uts 中调用构造方法实例化对象时需要在构造方法前加上 `new` 关键字

```ts
var alert = new UIAlertController()
```


#### 5.1.4 函数参数

在 swift 中参数名称使用 `:` 连接参数值,在 uts 中需要使用 `=` 连接

示例

```swift
// swift
var alert = UIAlertController(title: "提示", message: "提示内容", preferredStyle: .alert);
```

D
DCloud_LXH 已提交
442

D
DCloud_LXH 已提交
443 444 445 446 447 448 449 450 451 452 453 454 455 456
```ts
// uts 中写法
let alert = new UIAlertController(title="提示", message="提示内容", preferredStyle=UIAlertController.Style.alert)
```

#### 5.1.5 枚举值

枚举在 swift 中可以忽略枚举类型直接简写 `.枚举值` ,在 uts 中不支持简写,需要完整的写出 `枚举类型.枚举值`
上面的示例中 swift 中最后一个参数 preferredStyle 的值可以简写为

```swift
.alert
```

D
DCloud_LXH 已提交
457
在 uts 中需要完整的写出
D
DCloud_LXH 已提交
458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497

```ts
UIAlertController.Style.alert
```

枚举在 swift 中可以定义关联类型,可在关联值中传递信息,但是在 ts 中没有这种语法,所以目前 uts 还暂不支持此种用法。

```swift

// 定义带关联值的枚举
enum Barcode {
    case upc(Int, Int, Int, Int)
    case qrCode(String)
}

// 定义枚举值
var productBarcode = Barcode.upc(8, 85909, 51226, 3)
productBarcode = .qrCode("ABCDEFGHIJKLMNOP")

// 匹配
switch productBarcode {
case let .upc(numberSystem, manufacturer, product, check):
    print("UPC : \(numberSystem), \(manufacturer), \(product), \(check).")
case let .qrCode(productCode):
    print("QR code: \(productCode).")
}

```

如果遇到上述类型的枚举,且在三方库中无法改动的,可以在 Swift 文件中进行调用,然后把 该 Swift 文件打包在 framework 中供 uts 插件使用;
如果上述类型的枚举定义在有源码的 swift 中时,可以将其定义成不包含关联值的枚举,然后使用合适的数据结构来表示关联值携带的信息。


#### 5.1.6 类继承

swift 中定义子类继承父类时需要在子类名称后加上父类名称,中间以冒号`:`分隔

```swift
// swift
class Son: Father {
D
DCloud_LXH 已提交
498

D
DCloud_LXH 已提交
499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524
}
```

uts 中需要使用`extends`关键字代替冒号`:`

```ts
// uts
class Son extends Father {

}
```

#### 5.1.7 遵循协议方法

swift 中要让自定义类型遵循某个协议,在定义类型时,需要在类型名称后加上协议名称,中间以冒号`:`分隔。遵循多个协议时,各协议之间用逗号`,`分隔:

```swift
class SomeClass: FirstProtocol, AnotherProtocol {

}
```

uts 中需要使用`implements`关键字代替冒号 `:`

```ts
class SomeClass implements FirstProtocol, AnotherProtocol {
D
DCloud_LXH 已提交
525

D
DCloud_LXH 已提交
526 527 528 529 530 531 532 533 534 535
}
```

#### 5.1.8 系统版本判断

swift 中系统版本判断的方法

```swift
// swift
if #available(iOS 10.0, *) {
D
DCloud_LXH 已提交
536

D
DCloud_LXH 已提交
537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554
}
```

在 uts 中不支持这种语法可使用下面方式代替

```ts
if (UTSiOS.available("iOS 10.0, *")) {

}
```

#### 5.1.9 闭包

swift 中闭包可以简写

```swift
// swift 中最后一个参数如果是闭包称作为尾随闭包,可以忽略参数标签类型等简写为下面的方式
let action = UIAlertAction(title: "确认", style: .default) { action in
D
DCloud_LXH 已提交
555

D
DCloud_LXH 已提交
556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628
}
```

uts 中不支持简写语法,需要完整的写出闭包函数

```ts
// uts 中 handler 对应的闭包函数必须写完整
let action = new UIAlertAction(title="确认", style=UIAlertAction.Style.default, handler=(action: UIAlertAction):void => {

})
```

原生中有些方法的闭包参数是逃逸闭包,此时就要在闭包前面添加 `@escaping` 标记:

```ts
// 在闭包参数前添加@escaping
function requestLocationPromise(@escaping completion: (res: boolean)=>void) {

}
```


#### 5.1.10 target-action 方法

uts 中调用原生中涉及 target-action 的方法时,比如给`UIButton`添加点击事件方法、注册通知中心事件方法时注意事项,

1. 接口要求的 selector 通过 Selector("方法名字符串") 的方法构建
2. 定义的回调方法需要添加 @objc 前缀

下面以监听截屏事件为例:

示例文件在 hello uts 中的位置:

`~/uni_modules/uts-screenshot-listener/utssdk/app-ios/index.uts`

```ts
// 注册监听截屏事件及回调方法
// target-action 回调方法需要通过 Selector("方法名") 构建
const method = Selector("userDidTakeScreenshot")
NotificationCenter.default.addObserver(this, selector = method, name = UIApplication.userDidTakeScreenshotNotification, object = null)

// 捕获截屏回调的方法
// target-action 的方法前需要添加 @objc 前缀
@objc static userDidTakeScreenshot() {
    const obj = new UTSJSONObject()
    // 回调
    this.listener?.(obj)
}
```

#### 5.1.11 字典类型

swift 中的 Dictionary 类型,在 uts 中使用 Map 类型代替

```swift
// swift
var value: Dictionary<String,Any> = Dictionary()
value["name"] = "uts"
```

```ts
// uts
let map: Map<string,any> = new Map()
map.set("name","uts")
```
#### 5.1.12 参数标签的兼容问题
> HBuilder X 3.6.11+ 版本支持

当覆写系统方法,或实现三方SDK的协议方法时,一些方法可能会存在参数标签的情况

以 hello uts 中腾讯定位为例,监听位置变化时需要实现协议方法:

 `tencentLBSLocationManager(_ manager: TencentLBSLocationManager, didUpdate location: TencentLBSLocation)`
D
DCloud_LXH 已提交
629

D
DCloud_LXH 已提交
630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666
 此方法第二个参数存在 `didUpdate` 参数标签

原生 swift 中的实现为

```swift
// swift
func tencentLBSLocationManager(_ manager: TencentLBSLocationManager, didUpdate location: TencentLBSLocation) {
        var response = LocationResponse();
        response.name = location.name;
        response.address = location.address;
        response.latitude = NSNumber(location.location.coordinate.latitude);
        response.longitude = NSNumber(location.location.coordinate.longitude);
        self.locationOptions?.success(response);
}
```

uts 中需要用注解语法 @argumentLabel("didUpdate") 来表示参数标签

```ts
// uts
// 实现位置更新的 delegate 方法
tencentLBSLocationManager(manager: TencentLBSLocationManager, @argumentLabel("didUpdate") location: TencentLBSLocation) {
		let response = new LocationResponse();
		response.name = location.name;
		response.address = location.address;
		response.latitude = Number(location.location.coordinate.latitude);
		response.longitude = Number(location.location.coordinate.longitude);
		this.locationOptions?.success(response)
}
```

示例文件在 hello uts 中的位置:

`~/uni_modules/uts-tencentgeolocation/utssdk/app-ios/index.uts`

#### 5.1.12.1 无参数标签

D
DCloud_LXH 已提交
667
只写参数名称的参数,编译后会在参数前默认增加 `_` 来忽略参数标签(如上面的示例,第一个参数 manager,这种方式能兼容绝大多数方法,尤其是Swift 调用 OC 方法),但是有些参数没有参数标签,默认添加 `_` 的行为会和原生方法定义不一致,这种情况需要定义一个空的参数标签来解决 `@argumentLabel("didUpdate")`
D
DCloud_LXH 已提交
668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684

以高德定位 SDK 的代理方法为例:第三个参数 reGeocode 只有参数名称,没有参数标签

```swift
// swift
func amapLocationManager(_ manager: AMapLocationManager!, didUpdate location: CLLocation!, reGeocode: AMapLocationReGeocode!)
```

uts 实现此方法时,需要给 reGeocode 参数添加一个空的参数标签

```ts
// uts
amapLocationManager(manager : AMapLocationManager, @argumentLabel("didUpdate") location : CLLocation, @argumentLabel("") reGeocode : AMapLocationReGeocode) {

}
```

D
DCloud_LXH 已提交
685
#### 5.1.13 异步方法 @async-method
D
DCloud_LXH 已提交
686 687 688 689

swift 标记某个函数或者方法是异步的,你可以在它的声明中的参数列表后边加上 `async` 关键字

```swift
D
DCloud_LXH 已提交
690
// swift
D
DCloud_LXH 已提交
691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732
@available(iOS 13.0.0, *)
func testAsync(_ opts: AsyncOptions) async -> UTSJSONObject {
    if (opts.type == "success") {
        opts.success("success");
    }
     else {
        opts.fail("fail");
    }
    opts.complete("complete");
    return UTSJSONObject([
        "name": "testAsync"
    ]);
}
```

uts 中定义异步方法是在方法最前面加上 `async` 关键字

```ts
// uts
async function testAsync(opts: AsyncOptions) {
  if (opts.type == "success") {
    opts.success("success");
  } else {
    opts.fail("fail");
  }
  opts.complete("complete");
  return { name: "testAsync" };
}
```

**需要注意:使用 async 定义异步方法只有 iOS 13+ 版本才支持,低版本调用会报错**


#### 5.1.14 try catch @try

swift中try有以下三种方式:

以JSON反序列化为例

1. 使用try (注意:要和do {} catch {} 一起使用,捕获可能的异常)

```swift
D
DCloud_LXH 已提交
733
// swift
D
DCloud_LXH 已提交
734 735 736 737 738 739 740 741 742 743 744 745 746
	do{
		let dict = try JSONSerialization.jsonObject(with: d, options: [])
		print(dict)
	}catch{
	   // catch 中默认提供error信息, 当序列化不成功是, 返回error
		print(error)
	}

```

2. 使用try? 如果能发序列化成功,就返回成功的值,不能成功就返回nil

```swift
D
DCloud_LXH 已提交
747
// swift
D
DCloud_LXH 已提交
748 749 750 751 752 753 754 755
// 注意:dict是个可选值
	let dict = try? JSONSerialization.jsonObject(with: data, options: [])

```

3. 使用try! 强行try,如果不能反序列化成功,会造成应用闪退, 如果能序列化成功,就返回成功的值,注意该值是个可选值。

```swift
D
DCloud_LXH 已提交
756
// swift
D
DCloud_LXH 已提交
757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788
// 注意:dict是个可选值
	let dict = try! JSONSerialization.jsonObject(with: data, options: [])

```

为了满足Swift上述语法,UTS使用特殊语法来支持,以上三种写法分别对应为:

1. try

```ts
// uts
try {
	let dict = UTSiOS.try(JSONSerialization.jsonObject(with = data, options = []))
}catch (e) {
	console.log(e)
}
```
2. try?

```ts
// uts
UTSiOS.try(JSONSerialization.jsonObject(with = data, options = []), "?" )

```
3. try!

```ts
// uts
UTSiOS.try(JSONSerialization.jsonObject(with = data, options = []), "!" )

```

lizhongyi_'s avatar
lizhongyi_ 已提交
789 790
#### 5.1.15 swift 特有修饰符 @keyword

lizhongyi_'s avatar
lizhongyi_ 已提交
791 792
> HBuilder X 4.06+ 版本支持

lizhongyi_'s avatar
lizhongyi_ 已提交
793
`swift` 中有一些特有的属性、方法、类的修饰符如 `open`, `fileprivate`, `internal`, `weak`, `optional` 等,这些修饰符在 `ts` 中没有对应的替代者,但是在原生语法场景下又是必须的。
lizhongyi_'s avatar
lizhongyi_ 已提交
794
为了支持这些修饰符,我们提供了 `UTSiOS.keyword("xxx")` 的语法糖。你可以在符合 `swift` 语法要求的场景下,使用诸如 `UTSiOS.keyword("private")` 这样的语法来修饰对应的属性、方法、类等。
lizhongyi_'s avatar
lizhongyi_ 已提交
795 796 797 798 799 800
我们对具体的修饰符没有做特别的限制,`swfit` 中的合法修饰符都能通过这个语法糖来使用,但是请记住一个重要的前提:你所使用的修饰符,需要满足 `swift` 语法所要求的场景。

下面是一些示例:

```ts
// ts 中 private 修饰符不可出现在模块或命名空间元素上, 如果想将一个类 private,需要使用 @UTSiOS.keyword("private")
D
DCloud_LXH 已提交
801
@UTSiOS.keyword("private")
lizhongyi_'s avatar
lizhongyi_ 已提交
802 803
class TestA  {
	// 如果需要使用 weak 修饰属性,来避免循环引用,需要使用 @UTSiOS.keyword("weak")
D
DCloud_LXH 已提交
804
	@UTSiOS.keyword("weak")
lizhongyi_'s avatar
lizhongyi_ 已提交
805 806 807 808 809 810 811
	private delegate: TestProtocol | null = null
}

```

```ts

D
DCloud_LXH 已提交
812
@UTSiOS.keyword("fileprivate")
lizhongyi_'s avatar
lizhongyi_ 已提交
813 814
class TestB  {
	// 一个属性可以同时有多个修饰符,前提是所写的修饰符符合Swift语法
D
DCloud_LXH 已提交
815 816
	@UTSiOS.keyword("weak")
	@UTSiOS.keyword("fileprivate")
lizhongyi_'s avatar
lizhongyi_ 已提交
817 818 819 820 821
	delegate: TestProtocol | null = null
}

```

taohebin@dcloud.io's avatar
taohebin@dcloud.io 已提交
822
#### 5.1.16 指针操作
taohebin@dcloud.io's avatar
taohebin@dcloud.io 已提交
823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845

在Swift操作指针,需要用到`&`操作符隐式转换得到`UnsafePointer`类型,UTS中提供了`UTSiOS.getPointer()`来表示`&`符号。

以获取字符串MD5为例,如下
```ts
private convertToMD5(param : string) : string {
		const strData = param.data(using = String.Encoding.utf8)!
		let digest = new Array<UInt8>(repeating = 0, count = new Int(CC_MD5_DIGEST_LENGTH))
		strData.withUnsafeBytes((body : UnsafeRawBufferPointer) => {
			CC_MD5(body.baseAddress, new UInt32(strData.count), UTSiOS.getPointer(digest))
		})
		let md5String = ""
		digest.forEach((byte : UInt8) => {
			md5String += new String(format = "%02x", new UInt8(byte))
		})

		return md5String
	}
```


其中`UTSiOS.getPointer(digest)`编译后会变成`&digest`

lizhongyi_'s avatar
lizhongyi_ 已提交
846

D
DCloud_LXH 已提交
847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869
## 6  常见问题(持续更新)

### 6.1 如何在UTS环境中,获取当前 UIViewController 实例

参考 Hello UTS 项目中的 uts-alert 插件

路径:
> ~/uni_modules/uts-alert/utssdk/app-ios/index.uts


### 6.2 如何在UTS环境中,操作 UI 线程

```
DispatchQueue.main.async(execute=():void => {
	// 在主线程中可操作 UI
})
```

参考Hello UTS项目中的 uts-toast 插件

路径:
> ~/uni_modules/uts-toast/utssdk/app-ios/index.uts

taohebin@dcloud.io's avatar
taohebin@dcloud.io 已提交
870 871
### 6.3 插件中需要显示使用类型,不能省略

taohebin@dcloud.io's avatar
taohebin@dcloud.io 已提交
872
由于在uts插件环境,无法默认推断出类型,所以需要显示类型,以`uni.request()`为例:
taohebin@dcloud.io's avatar
taohebin@dcloud.io 已提交
873 874 875 876 877 878 879 880 881 882 883 884 885 886 887

```
		uni.request<any>({
			url: "http://xxx",
			method: "GET",
			success: (e : RequestSuccess<any>) => {
				
			},
			fail(e : RequestFail) {
				
			},
		} as RequestOptions<any>)
```


D
DCloud_LXH 已提交
888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903


## 7  已知待解决问题(持续更新)

### 7.1 语法提示问题

HBuilderX 目前写iOS uts 插件时部分语法提示会有缺失、参数类型不准确的问题,例如:

- 类的构造方法目前只会提示一个,实际上可能会存在多个;
- 缺失可选类型标识;
- 参数标签没有标记无法知道是否需要忽略参数标签;
- 不支持导入包含有子模块的原生模块;
- 暂不支持.a依赖库的代码提示;

这些问题会在后续版本中优化

lizhongyi_'s avatar
lizhongyi_ 已提交
904
### 7.2 类型兼容问题
D
DCloud_LXH 已提交
905 906 907 908 909 910

- 元组类型目前不支持

## 8 有关Swift语言创建的Framework和.a的Swift版本兼容性问题

- 由于高版本XCode编译的Swift语言Framework动态库、静态库、.a库在低版本XCode上无法编译通过,因此存在Swift版本兼容性问题;
lizhongyi_'s avatar
lizhongyi_ 已提交
911 912 913 914 915 916 917
- 目前打包机使用的XCode版本号是15.2,对应的Swift版本是5.9.2;
- uts 插件开发者在编译Swift相关Framework和.a库时请选择和打包机相同或者更低版本的XCode;
- uts 插件开发者在选择比打包机更低版本XCode编译Swift库时请在Target->buildSettings设置Buid Libraries for Distribution 为Yes;
- uts 插件使用者所依赖的 xcode 版本号应大于或者等于打包机的 XCode 版本号。

> 特别注意
> 如果在使用真机运行编译uts插件时报 swift 版本不兼容的错误,请先检查自己的 XCode 版本,确保安装XCode 版本应大于或者等于打包机的 XCode版本;
D
DCloud_LXH 已提交
918
> 如果在使用真机运行编译uts插件时报 XCode 版本应大于 13.2.1的错误,这是说明本地安装的 XCode版本过低,请忽略 13.2.1这个版本限制,直接将本地 XCode升级到大于或者等于打包机的版本,后续我们会优化提示。