expo-gaode-map-navigation 2.0.9-next.0 → 2.0.9

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (127) hide show
  1. package/README.md +258 -3
  2. package/android/src/main/java/expo/modules/gaodemap/map/ExpoGaodeMapModule.kt +4 -2
  3. package/android/src/main/java/expo/modules/gaodemap/navigation/ExpoGaodeMapNaviView.kt +931 -391
  4. package/android/src/main/java/expo/modules/gaodemap/navigation/ExpoGaodeMapNaviViewModule.kt +86 -1
  5. package/android/src/main/java/expo/modules/gaodemap/navigation/ExpoGaodeMapNavigationModule.kt +4 -5
  6. package/android/src/main/java/expo/modules/gaodemap/navigation/listeners/IndependentRouteListener.kt +4 -3
  7. package/android/src/main/java/expo/modules/gaodemap/navigation/listeners/RouteCalculateListener.kt +2 -2
  8. package/android/src/main/java/expo/modules/gaodemap/navigation/managers/IndependentRouteManager.kt +96 -14
  9. package/android/src/main/java/expo/modules/gaodemap/navigation/routes/drive/DriveTruckRouteCalculator.kt +2 -0
  10. package/android/src/main/java/expo/modules/gaodemap/navigation/utils/Converters.kt +19 -10
  11. package/android/src/main/res/drawable/landback_0.png +0 -0
  12. package/android/src/main/res/drawable/landback_1.png +0 -0
  13. package/android/src/main/res/drawable/landback_2.png +0 -0
  14. package/android/src/main/res/drawable/landback_3.png +0 -0
  15. package/android/src/main/res/drawable/landback_4.png +0 -0
  16. package/android/src/main/res/drawable/landback_5.png +0 -0
  17. package/android/src/main/res/drawable/landback_6.png +0 -0
  18. package/android/src/main/res/drawable/landback_7.png +0 -0
  19. package/android/src/main/res/drawable/landback_8.png +0 -0
  20. package/android/src/main/res/drawable/landback_9.png +0 -0
  21. package/android/src/main/res/drawable/landback_a.png +0 -0
  22. package/android/src/main/res/drawable/landback_b.png +0 -0
  23. package/android/src/main/res/drawable/landback_c.png +0 -0
  24. package/android/src/main/res/drawable/landback_d.png +0 -0
  25. package/android/src/main/res/drawable/landback_e.png +0 -0
  26. package/android/src/main/res/drawable/landback_f.png +0 -0
  27. package/android/src/main/res/drawable/landback_g.png +0 -0
  28. package/android/src/main/res/drawable/landback_h.png +0 -0
  29. package/android/src/main/res/drawable/landback_i.png +0 -0
  30. package/android/src/main/res/drawable/landback_j.png +0 -0
  31. package/android/src/main/res/drawable/landback_k.png +0 -0
  32. package/android/src/main/res/drawable/landback_l.png +0 -0
  33. package/android/src/main/res/drawable/landfront_0.png +0 -0
  34. package/android/src/main/res/drawable/landfront_00.png +0 -0
  35. package/android/src/main/res/drawable/landfront_1.png +0 -0
  36. package/android/src/main/res/drawable/landfront_11.png +0 -0
  37. package/android/src/main/res/drawable/landfront_20.png +0 -0
  38. package/android/src/main/res/drawable/landfront_21.png +0 -0
  39. package/android/src/main/res/drawable/landfront_22.png +0 -0
  40. package/android/src/main/res/drawable/landfront_3.png +0 -0
  41. package/android/src/main/res/drawable/landfront_33.png +0 -0
  42. package/android/src/main/res/drawable/landfront_40.png +0 -0
  43. package/android/src/main/res/drawable/landfront_43.png +0 -0
  44. package/android/src/main/res/drawable/landfront_44.png +0 -0
  45. package/android/src/main/res/drawable/landfront_5.png +0 -0
  46. package/android/src/main/res/drawable/landfront_55.png +0 -0
  47. package/android/src/main/res/drawable/landfront_61.png +0 -0
  48. package/android/src/main/res/drawable/landfront_63.png +0 -0
  49. package/android/src/main/res/drawable/landfront_66.png +0 -0
  50. package/android/src/main/res/drawable/landfront_70.png +0 -0
  51. package/android/src/main/res/drawable/landfront_71.png +0 -0
  52. package/android/src/main/res/drawable/landfront_73.png +0 -0
  53. package/android/src/main/res/drawable/landfront_77.png +0 -0
  54. package/android/src/main/res/drawable/landfront_8.png +0 -0
  55. package/android/src/main/res/drawable/landfront_88.png +0 -0
  56. package/android/src/main/res/drawable/landfront_90.png +0 -0
  57. package/android/src/main/res/drawable/landfront_95.png +0 -0
  58. package/android/src/main/res/drawable/landfront_99.png +0 -0
  59. package/android/src/main/res/drawable/landfront_a0.png +0 -0
  60. package/android/src/main/res/drawable/landfront_a8.png +0 -0
  61. package/android/src/main/res/drawable/landfront_aa.png +0 -0
  62. package/android/src/main/res/drawable/landfront_b1.png +0 -0
  63. package/android/src/main/res/drawable/landfront_b5.png +0 -0
  64. package/android/src/main/res/drawable/landfront_bb.png +0 -0
  65. package/android/src/main/res/drawable/landfront_c3.png +0 -0
  66. package/android/src/main/res/drawable/landfront_c8.png +0 -0
  67. package/android/src/main/res/drawable/landfront_cc.png +0 -0
  68. package/android/src/main/res/drawable/landfront_d.png +0 -0
  69. package/android/src/main/res/drawable/landfront_dd.png +0 -0
  70. package/android/src/main/res/drawable/landfront_e1.png +0 -0
  71. package/android/src/main/res/drawable/landfront_e5.png +0 -0
  72. package/android/src/main/res/drawable/landfront_ee.png +0 -0
  73. package/android/src/main/res/drawable/landfront_f0.png +0 -0
  74. package/android/src/main/res/drawable/landfront_f1.png +0 -0
  75. package/android/src/main/res/drawable/landfront_f5.png +0 -0
  76. package/android/src/main/res/drawable/landfront_ff.png +0 -0
  77. package/android/src/main/res/drawable/landfront_g3.png +0 -0
  78. package/android/src/main/res/drawable/landfront_g5.png +0 -0
  79. package/android/src/main/res/drawable/landfront_gg.png +0 -0
  80. package/android/src/main/res/drawable/landfront_h1.png +0 -0
  81. package/android/src/main/res/drawable/landfront_h3.png +0 -0
  82. package/android/src/main/res/drawable/landfront_h5.png +0 -0
  83. package/android/src/main/res/drawable/landfront_hh.png +0 -0
  84. package/android/src/main/res/drawable/landfront_i0.png +0 -0
  85. package/android/src/main/res/drawable/landfront_i3.png +0 -0
  86. package/android/src/main/res/drawable/landfront_i5.png +0 -0
  87. package/android/src/main/res/drawable/landfront_ii.png +0 -0
  88. package/android/src/main/res/drawable/landfront_j1.png +0 -0
  89. package/android/src/main/res/drawable/landfront_j8.png +0 -0
  90. package/android/src/main/res/drawable/landfront_jj.png +0 -0
  91. package/android/src/main/res/drawable/landfront_kk.png +0 -0
  92. package/android/src/main/res/drawable/landfront_ll.png +0 -0
  93. package/android/src/main/res/drawable/navi_arrow_leftline.png +0 -0
  94. package/android/src/main/res/drawable/navi_lane_shape_bg_center.xml +5 -0
  95. package/android/src/main/res/drawable/navi_lane_shape_bg_left.xml +8 -0
  96. package/android/src/main/res/drawable/navi_lane_shape_bg_over.xml +6 -0
  97. package/android/src/main/res/drawable/navi_lane_shape_bg_right.xml +8 -0
  98. package/build/ExpoGaodeMapNaviView.d.ts +8 -0
  99. package/build/ExpoGaodeMapNaviView.d.ts.map +1 -1
  100. package/build/ExpoGaodeMapNaviView.js +38 -1
  101. package/build/ExpoGaodeMapNaviView.js.map +1 -1
  102. package/build/index.d.ts +8 -4
  103. package/build/index.d.ts.map +1 -1
  104. package/build/index.js +408 -4
  105. package/build/index.js.map +1 -1
  106. package/build/types/independent.types.d.ts +91 -0
  107. package/build/types/independent.types.d.ts.map +1 -1
  108. package/build/types/independent.types.js.map +1 -1
  109. package/build/types/naviview.types.d.ts +256 -12
  110. package/build/types/naviview.types.d.ts.map +1 -1
  111. package/build/types/naviview.types.js.map +1 -1
  112. package/build/types/route.types.d.ts +7 -1
  113. package/build/types/route.types.d.ts.map +1 -1
  114. package/build/types/route.types.js.map +1 -1
  115. package/ios/ExpoGaodeMapNaviView.swift +920 -67
  116. package/ios/ExpoGaodeMapNaviViewModule.swift +87 -1
  117. package/ios/ExpoGaodeMapNavigationModule.swift +90 -9
  118. package/ios/managers/IndependentRouteManager.swift +1 -0
  119. package/ios/map/ExpoGaodeMapModule.swift +9 -4
  120. package/ios/map/ExpoGaodeMapView.swift +13 -2
  121. package/ios/map/GaodeMapPrivacyManager.swift +23 -3
  122. package/ios/map/modules/LocationManager.swift +17 -0
  123. package/ios/map/utils/PermissionManager.swift +11 -6
  124. package/ios/routes/drive/DriveTruckRouteCalculator.swift +9 -0
  125. package/ios/routes/walkride/WalkRideRouteCalculator.swift +30 -0
  126. package/ios/services/IndependentRouteService.swift +211 -44
  127. package/package.json +16 -4
package/README.md CHANGED
@@ -6,7 +6,7 @@
6
6
 
7
7
  - 🗺️ **地图渲染**:内置完整地图能力,支持 Marker、Polyline、Polygon、Circle、Cluster、HeatMap 等覆盖物。
8
8
  - 🚗 **多模式路径规划**:支持驾车、步行、骑行、电动车、货车、摩托车等多种出行方式。
9
- - 🧭 **实时导航 UI**:提供 `NaviView` 组件,内置完整的导航界面、语音播报、转向指引、路况显示等。
9
+ - 🧭 **实时导航 UI**:提供 `NaviView` 官方嵌入视图,并暴露完整事件与原生参数,方便你自行定制导航界面。
10
10
  - 🛣️ **独立路径规划**:支持“先算路、再导航”的高级模式,可实现多路线对比与选择。
11
11
  - ⚙️ **策略丰富**:支持速度优先、避让拥堵、少收费、不走高速等多种算路策略。
12
12
  - ✅ **开箱即用**:封装了 Android/iOS 原生导航 SDK,统一 JS 接口。
@@ -57,6 +57,35 @@ npx expo run:android
57
57
  npx expo run:ios
58
58
  ```
59
59
 
60
+ ## 示例工程
61
+
62
+ 仓库内提供了可直接运行的 [`example-navigation`](/Volumes/xinxin/expo-gaode-map/example-navigation/README.md) 示例工程,专门用于验证导航能力。
63
+
64
+ 推荐场景:
65
+
66
+ - 调试 `NaviView` 与示例工程里的自定义 HUD / 车道 HUD / 路况光柱
67
+ - 对比官方黑盒页、官方嵌入式页、自绘嵌入式页
68
+ - 验证独立算路、多路线选择、近似跟线导航
69
+
70
+ 快速运行:
71
+
72
+ ```bash
73
+ cd example-navigation
74
+ cp .env.example .env
75
+ npm install
76
+ npx expo run:android
77
+ ```
78
+
79
+ 如需 iOS:
80
+
81
+ ```bash
82
+ cd example-navigation
83
+ cp .env.example .env
84
+ npm install
85
+ npx pod-install ios
86
+ npx expo run:ios
87
+ ```
88
+
60
89
 
61
90
  ## 快速开始
62
91
 
@@ -128,6 +157,22 @@ export default function NavigationScreen() {
128
157
  }
129
158
  ```
130
159
 
160
+ ### 3. 自定义嵌入式导航 UI
161
+
162
+ 如果你要做“嵌入在自己页面里的导航页”,库本身提供的是底层 `NaviView`、导航事件和原生参数;完整的自定义 HUD / 车道 HUD / 路况光柱参考实现,已经迁移到仓库内的 [`example-navigation`](/Volumes/xinxin/expo-gaode-map/example-navigation/README.md)。
163
+
164
+ 建议做法:
165
+
166
+ - 用 `NaviView` 负责底层导航地图、语音、车道事件、路况事件、路口大图事件
167
+ - 用 `onNaviInfoUpdate`、`onLaneInfoUpdate`、`onTrafficStatusesUpdate`、`onNaviVisualStateChange` 在业务侧自绘 HUD
168
+ - 直接参考 `example-navigation/lib/navigation-ui/EmbeddedNaviView.tsx` 及配套 UI 文件,按你的产品需求裁剪
169
+
170
+ 注意:
171
+
172
+ - Android 官方嵌入式 `NaviView` 在部分 React Native / Expo 宿主中,顶部信息区、车道条、路口大图联动效果可能与高德官方 Demo 不完全一致
173
+ - 如果你要验证官方嵌入式 UI 本身,请直接跑 `example-navigation` 里的 `official-embedded` 示例页
174
+ - 如果你要交付稳定的嵌入式导航页,建议以示例工程里的“自定义 UI 导航界面”作为起点
175
+
131
176
  ## 详细用法
132
177
 
133
178
  ### 路径规划 (API)
@@ -145,12 +190,26 @@ const result = await calculateRoute({
145
190
  to: { latitude: 39.91, longitude: 116.41 },
146
191
  strategy: DriveStrategy.FASTEST, // 速度优先
147
192
  avoidRoad: '京通快速路', // 避让道路名称
193
+ avoidPolygons: [
194
+ [
195
+ { latitude: 39.905, longitude: 116.395 },
196
+ { latitude: 39.905, longitude: 116.405 },
197
+ { latitude: 39.915, longitude: 116.405 },
198
+ { latitude: 39.915, longitude: 116.395 },
199
+ ],
200
+ ],
148
201
  });
149
202
 
150
203
  console.log(`总距离: ${result.routes[0].distance}米`);
151
204
  console.log(`预计耗时: ${result.routes[0].duration}秒`);
152
205
  ```
153
206
 
207
+ 说明:
208
+
209
+ - 当传入 `avoidRoad` 或 `avoidPolygons` 时,`calculateRoute` / `calculateDriveRoute` 会优先尝试通过 `expo-gaode-map-web-api` 获取官方“规避后路线预览”结果。
210
+ - 该回退仅用于路线预览与地图绘制;Web API 返回的是 polyline,不是导航 SDK 可直接启动的 `routeGroup/path`。
211
+ - 如果未安装 `expo-gaode-map-web-api`,则保持原有原生驾车算路逻辑不变。Android 仍可能命中底层 SDK 的避让重载;iOS 则没有官方导航 SDK 接口可直接消费任意规避道路/区域。
212
+
154
213
  #### 步行/骑行路径规划
155
214
 
156
215
  ```typescript
@@ -192,6 +251,8 @@ const truckResult = await calculateRoute({
192
251
 
193
252
  “独立路径规划”允许你先计算路线,并在地图上展示多条方案,用户选择其中一条后再开始导航。这通常比直接开始导航体验更好。
194
253
 
254
+ 注意:`independentDriveRoute` 仍然依赖导航 SDK 自身的独立算路能力,因此这里不接 Web API 的规避预览结果。若你需要“规避道路/区域后再开始导航”,建议先用 `calculateRoute` 做预览与确认,再按终点重新发起原生导航。
255
+
195
256
  ```typescript
196
257
  import {
197
258
  independentDriveRoute,
@@ -219,6 +280,87 @@ await startNaviWithIndependentPath({
219
280
  });
220
281
  ```
221
282
 
283
+ ### 近似跟线导航(第一版)
284
+
285
+ 当你已经通过 Web API 拿到一条想要的路线,但导航 SDK 不能直接吃这条 `polyline` 时,可以使用 `followWebPlannedRoute`。
286
+
287
+ 它会:
288
+
289
+ - 从 Web 路线提炼一组途经锚点
290
+ - 用这些锚点重新发起原生独立算路
291
+ - 选择最接近 Web 路线的一条原生路线
292
+ - 仅在匹配足够接近时才启动导航
293
+
294
+ ```typescript
295
+ import { followWebPlannedRoute } from 'expo-gaode-map-navigation';
296
+
297
+ const result = await followWebPlannedRoute({
298
+ from: { latitude: 39.9, longitude: 116.4 },
299
+ to: { latitude: 39.91, longitude: 116.41 },
300
+ webRoute: {
301
+ polyline: webResult.routes[0].polyline ?? [],
302
+ },
303
+ maxViaPoints: 8,
304
+ maxDeviationMeters: 120,
305
+ startNavigation: true,
306
+ naviType: 1, // 1 = 模拟导航
307
+ });
308
+
309
+ console.log(result.mode); // matched | approximate | preview_only
310
+ console.log(result.anchorWaypoints);
311
+ console.log(result.candidateMatches);
312
+ ```
313
+
314
+ 说明:
315
+
316
+ - 这不是“强制按 Web 线导航”,而是“尽量贴近 Web 线”。
317
+ - 若返回 `preview_only`,说明原生导航 SDK 算出的路线与 Web 线路偏差过大,建议仅做预览,不要直接开导航。
318
+ - 若你只想拿锚点,不立即导航,可以单独使用 `buildAnchorWaypointsFromWebRoute`。
319
+
320
+ 如果你希望继续使用嵌入式官方导航 UI,可以先完成近似跟线选路,再通过 `ExpoGaodeMapNaviView` 的 ref 使用独立路径启动:
321
+
322
+ ```typescript
323
+ const matchResult = await followWebPlannedRoute({
324
+ from,
325
+ to,
326
+ webRoute,
327
+ startNavigation: false,
328
+ });
329
+
330
+ if (matchResult.mode !== 'preview_only') {
331
+ await naviRef.current?.startNavigationWithIndependentPath(matchResult.token, {
332
+ routeId: matchResult.selectedRouteId,
333
+ routeIndex: matchResult.selectedRouteIndex,
334
+ naviType: 1,
335
+ });
336
+ }
337
+ ```
338
+
339
+ ### 官方导航页(openOfficialNaviPage)
340
+
341
+ 新增支持直接调起高德官方导航组件(Android: `AmapNaviPage`,iOS: `AMapNaviCompositeManager`)。
342
+
343
+ ```typescript
344
+ import { openOfficialNaviPage } from 'expo-gaode-map-navigation';
345
+
346
+ await openOfficialNaviPage({
347
+ to: { latitude: 39.908823, longitude: 116.39747, name: '终点' }, // 必填
348
+ pageType: 'NAVI', // ROUTE | NAVI
349
+ startNaviDirectly: true,
350
+ naviMode: 2, // 1=实时导航, 2=模拟导航(iOS 官方组件不支持模拟)
351
+ theme: 'BLUE', // BLUE | WHITE | BLACK
352
+ trafficEnabled: true,
353
+ showCrossImage: true,
354
+ });
355
+ ```
356
+
357
+ 说明:
358
+
359
+ - 支持 Android / iOS 平台差异参数(如 `dayAndNightMode`、`broadcastMode`、`mapViewModeType`、`trackingMode` 等)。
360
+ - iOS 直接进导航页时需开启后台定位 `UIBackgroundModes: location`。
361
+ - iOS 官方导航组件模式不支持模拟导航;若传 `naviMode: 2` 会直接返回错误提示。
362
+ - Android 依赖 `AmapRouteActivity`,Config Plugin 会自动注入 Manifest。
363
+
222
364
  ### 地图组件 (Map)
223
365
 
224
366
  模块导出了完整的地图组件,与 `expo-gaode-map` API 保持一致。
@@ -311,23 +453,136 @@ const result = await calculateTransitRoute({
311
453
  | `AVOID_CONGESTION` (4) | 躲避拥堵 |
312
454
  | ... | 更多策略请参考类型定义 |
313
455
 
456
+ ### 自定义嵌入式 UI 参考实现
457
+
458
+ 库不再直接导出 `EmbeddedNaviView` 这类成品 UI 组件;这部分实现现在放在示例工程里,便于你直接查看和复制。
459
+
460
+ 参考文件:
461
+
462
+ - `example-navigation/lib/navigation-ui/EmbeddedNaviView.tsx`
463
+ - `example-navigation/lib/navigation-ui/EmbeddedNaviHud.tsx`
464
+ - `example-navigation/lib/navigation-ui/EmbeddedNaviLaneView.tsx`
465
+ - `example-navigation/lib/navigation-ui/EmbeddedNaviTrafficBar.tsx`
466
+
467
+ 这套示例实现演示了:
468
+
469
+ - 默认 `showUIElements={false}` 的完整自定义 UI 模式
470
+ - 基于 `driveViewEdgePadding` / `screenAnchor` 的嵌入式地图可视区域管理
471
+ - 基于 `onNaviInfoUpdate` 的顶部 HUD
472
+ - 基于 `onLaneInfoUpdate` 的自绘车道 HUD
473
+ - 基于 `onTrafficStatusesUpdate` 的自绘路况光柱
474
+ - “全览 / 锁车”与路况开关等浮层控制按钮
475
+
314
476
  ### NaviView Props
315
477
 
316
478
  | 属性 | 类型 | 说明 |
317
479
  |---|---|---|
318
480
  | `naviType` | number | 导航类型(0: GPS, 1: 模拟) |
319
- | `showCrossImage` | boolean | 是否显示路口放大图 |
481
+ | `realCrossDisplay` | boolean | 是否显示路口放大图 |
320
482
  | `showCamera` | boolean | 是否显示摄像头 |
321
- | `showTrafficButton` | boolean | 是否显示路况按钮 |
483
+ | `carImage` | string \| ImageSourcePropType | 自定义导航车标;iOS 映射 `setCarImage`,Android 映射 `setCarBitmap` |
484
+ | `startPointImage` | string \| ImageSourcePropType | 自定义起点标注图 |
485
+ | `wayPointImage` | string \| ImageSourcePropType | 自定义途经点标注图 |
486
+ | `endPointImage` | string \| ImageSourcePropType | 自定义终点标注图 |
487
+ | `trafficLayerEnabled` | boolean | 是否显示实时交通路况线 |
488
+ | `showTrafficButton` | boolean | 是否显示交通按钮/交通图层开关 |
489
+ | `showDriveCongestion` | boolean | 是否显示拥堵气泡 |
490
+ | `showTrafficLightView` | boolean | 是否显示红绿灯倒计时气泡 |
491
+ | `showUIElements` | boolean | Android / iOS 均支持整体 UI 显隐 |
492
+ | `laneInfoVisible` | boolean | Android 是否显示官方车道信息 |
493
+ | `hideNativeLaneInfoLayout` | boolean | iOS 是否隐藏官方车道信息条,交给 RN 自绘 |
494
+ | `modeCrossDisplay` | boolean | Android 是否显示 3D 路口模型;iOS 当前不支持,会忽略 |
495
+ | `eyrieCrossDisplay` | boolean | Android 是否显示鹰眼路口图 |
496
+ | `secondActionVisible` | boolean | Android 是否显示辅助操作区域 |
497
+ | `backupOverlayVisible` | boolean | Android 是否显示备用路线覆盖物 |
498
+ | `androidStatusBarPaddingTop` | number | Android 顶部额外间距;若显示官方原生顶部信息区且未显式传值,封装会自动补系统状态栏高度 |
499
+ | `naviStatusBarEnabled` | boolean | Android 是否启用高德官方导航状态栏;若当前 AMap SDK 不支持该接口,则自动降级为 no-op |
500
+ | `lockZoom` | number | Android 锁车态缩放级别 |
501
+ | `lockTilt` | number | Android 锁车态倾斜角度 |
502
+ | `eagleMapVisible` | boolean | Android 是否显示鹰眼小地图 |
503
+ | `pointToCenter` | object | Android 锁车态自车锚点位置 |
504
+ | `driveViewEdgePadding` | object | iOS 导航内容边距 |
505
+ | `screenAnchor` | object | iOS 地图视图锚点 |
506
+ | `showBackupRoute` | boolean | iOS 是否显示备选路线 |
507
+ | `showEagleMap` | boolean | iOS 是否显示鹰眼小地图 |
322
508
  | `enableVoice` | boolean | 是否开启语音播报 |
323
509
  | `onArrive` | function | 到达目的地回调 |
324
510
  | `onNaviInfoUpdate` | function | 导航信息更新(剩余距离、时间等) |
511
+ | `onLaneInfoUpdate` | function | Android / iOS 车道信息更新,用于自绘车道 HUD |
512
+
513
+ ### NaviView UI 能力清单
514
+
515
+ 已开放且两端都有实现:
516
+
517
+ - `showCamera`
518
+ - `autoLockCar`
519
+ - `autoChangeZoom`
520
+ - `trafficLayerEnabled`
521
+ - `realCrossDisplay`
522
+ - `naviMode`
523
+ - `showMode`
524
+ - `isNightMode`
525
+ - `showTrafficBar`
526
+ - `showTrafficButton`
527
+ - `showUIElements`
528
+ - `showGreyAfterPass`
529
+ - `showVectorline`
530
+ - `showCompassEnabled`
531
+ - `showDriveCongestion`
532
+ - `showTrafficLightView`
533
+
534
+ 仅 Android 已开放:
535
+
536
+ - `carOverlayVisible`
537
+ - `fourCornersImage`
538
+ - `routeMarkerVisible`
539
+ - `naviArrowVisible`
540
+ - `laneInfoVisible`
541
+ - `modeCrossDisplay`
542
+ - `eyrieCrossDisplay`
543
+ - `secondActionVisible`
544
+ - `backupOverlayVisible`
545
+ - `androidStatusBarPaddingTop`
546
+ - `naviStatusBarEnabled`
547
+ - `lockZoom`
548
+ - `lockTilt`
549
+ - `eagleMapVisible`
550
+ - `pointToCenter`
551
+ - `isNaviTravelView`
552
+
553
+ 仅 iOS 已开放:
554
+
555
+ - `hideNativeLaneInfoLayout`
556
+ - `showRoute`
557
+ - `carCompassImage`
558
+ - `cameraImage`
559
+ - `trafficBarFrame`
560
+ - `trafficBarColors`
561
+ - `showMoreButton`
562
+ - `mapViewModeType`
563
+ - `lineWidth`
564
+ - `driveViewEdgePadding`
565
+ - `screenAnchor`
566
+
567
+ 关于 iOS 路口放大图能力:
568
+
569
+ - iOS 官方公开的是 `showCrossImage / hideCrossImage`
570
+ - 因此库里的 `realCrossDisplay` 对应 iOS 实景路口放大图显示控制
571
+ - Android 的 `modeCrossDisplay` 没有 iOS 对等公开接口,传入 iOS 时会被忽略
572
+ - `showBackupRoute`
573
+ - `showEagleMap`
574
+
575
+ 当前这份清单里此前列出的“剩余代表项”已全部开放。
576
+
577
+ 如果后续还要继续往下包,更适合继续补的是更底层的样式类配置,而不是核心导航 UI 能力。
325
578
 
326
579
  ## 注意事项
327
580
 
328
581
  1. **二进制冲突**:严禁与 `expo-gaode-map` 共存。本模块已包含 `3dmap` SDK。
329
582
  2. **Web API**:如果需要更灵活的 HTTP 算路(如公交跨城规划、Web端展示),推荐配合 `expo-gaode-map-web-api` 使用。
330
583
  3. **权限**:使用导航功能前,请确保应用已获取定位权限(`ACCESS_FINE_LOCATION`)。
584
+ 4. **Android 状态栏兼容性**:`naviStatusBarEnabled` 依赖高德 Android 导航 SDK 某些版本才提供的 `AMapNaviViewOptions.setNaviStatusBarEnabled(...)`。当前封装已做兼容处理:若宿主工程解析到的 SDK 不包含该方法,则不会再编译失败,而是在运行时跳过该设置并输出 warning。此时该 prop 在 Android 上等价于 no-op。
585
+ 5. **嵌入式 UI 边界**:库导出的是底层 `NaviView` 能力;完整自定义导航界面请参考 `example-navigation` 里的示例实现,它也不是高德官方黑盒导航页的 UI 替代品。
331
586
 
332
587
 
333
588
  ## 📚 文档与资源
@@ -771,7 +771,7 @@ class ExpoGaodeMapModule : Module() {
771
771
  // 使用 WeakReference 避免内存泄露
772
772
  val contextRef = java.lang.ref.WeakReference(appContext.reactContext)
773
773
  val handler = android.os.Handler(android.os.Looper.getMainLooper())
774
- val attempts = 0
774
+ var attempts = 0
775
775
  val maxAttempts = 50 // 增加到 5 秒 / 100ms,给用户足够时间操作
776
776
 
777
777
  val checkPermission = object : Runnable {
@@ -783,6 +783,7 @@ class ExpoGaodeMapModule : Module() {
783
783
  }
784
784
 
785
785
  val status = PermissionHelper.checkForegroundLocationPermission(context)
786
+ attempts += 1
786
787
 
787
788
  // 如果权限已授予或达到最大尝试次数,返回结果并清理 Handler
788
789
  if (status.granted || attempts >= maxAttempts) {
@@ -842,7 +843,7 @@ class ExpoGaodeMapModule : Module() {
842
843
  // 轮询检查权限状态
843
844
  val contextRef = java.lang.ref.WeakReference(appContext.reactContext)
844
845
  val handler = android.os.Handler(android.os.Looper.getMainLooper())
845
- val attempts = 0
846
+ var attempts = 0
846
847
  val maxAttempts = 30
847
848
 
848
849
  val checkPermission = object : Runnable {
@@ -854,6 +855,7 @@ class ExpoGaodeMapModule : Module() {
854
855
  }
855
856
 
856
857
  val status = PermissionHelper.checkBackgroundLocationPermission(context)
858
+ attempts += 1
857
859
 
858
860
  if (status.granted || attempts >= maxAttempts) {
859
861
  handler.removeCallbacks(this)