expo-gaode-map-navigation 2.0.11-next.0 → 2.0.11
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.
- package/README.md +291 -6
- package/android/build.gradle +4 -0
- package/android/src/main/AndroidManifest.xml +2 -1
- package/android/src/main/java/expo/modules/gaodemap/map/ExpoGaodeMapModule.kt +4 -2
- package/android/src/main/java/expo/modules/gaodemap/navigation/ExpoGaodeMapNaviView.kt +1407 -393
- package/android/src/main/java/expo/modules/gaodemap/navigation/ExpoGaodeMapNaviViewModule.kt +121 -1
- package/android/src/main/java/expo/modules/gaodemap/navigation/ExpoGaodeMapNavigationModule.kt +14 -28
- package/android/src/main/java/expo/modules/gaodemap/navigation/listeners/IndependentRouteListener.kt +28 -3
- package/android/src/main/java/expo/modules/gaodemap/navigation/listeners/RouteCalculateListener.kt +2 -2
- package/android/src/main/java/expo/modules/gaodemap/navigation/managers/IndependentRouteManager.kt +114 -15
- package/android/src/main/java/expo/modules/gaodemap/navigation/routes/drive/DriveTruckRouteCalculator.kt +24 -35
- package/android/src/main/java/expo/modules/gaodemap/navigation/services/IndependentRouteService.kt +50 -36
- package/android/src/main/java/expo/modules/gaodemap/navigation/services/NavigationForegroundService.kt +661 -0
- package/android/src/main/java/expo/modules/gaodemap/navigation/utils/Converters.kt +21 -12
- package/android/src/main/res/drawable/ic_nav_notification_small.xml +10 -0
- package/android/src/main/res/drawable/landback_0.png +0 -0
- package/android/src/main/res/drawable/landback_1.png +0 -0
- package/android/src/main/res/drawable/landback_2.png +0 -0
- package/android/src/main/res/drawable/landback_3.png +0 -0
- package/android/src/main/res/drawable/landback_4.png +0 -0
- package/android/src/main/res/drawable/landback_5.png +0 -0
- package/android/src/main/res/drawable/landback_6.png +0 -0
- package/android/src/main/res/drawable/landback_7.png +0 -0
- package/android/src/main/res/drawable/landback_8.png +0 -0
- package/android/src/main/res/drawable/landback_9.png +0 -0
- package/android/src/main/res/drawable/landback_a.png +0 -0
- package/android/src/main/res/drawable/landback_b.png +0 -0
- package/android/src/main/res/drawable/landback_c.png +0 -0
- package/android/src/main/res/drawable/landback_d.png +0 -0
- package/android/src/main/res/drawable/landback_e.png +0 -0
- package/android/src/main/res/drawable/landback_f.png +0 -0
- package/android/src/main/res/drawable/landback_g.png +0 -0
- package/android/src/main/res/drawable/landback_h.png +0 -0
- package/android/src/main/res/drawable/landback_i.png +0 -0
- package/android/src/main/res/drawable/landback_j.png +0 -0
- package/android/src/main/res/drawable/landback_k.png +0 -0
- package/android/src/main/res/drawable/landback_l.png +0 -0
- package/android/src/main/res/drawable/landfront_0.png +0 -0
- package/android/src/main/res/drawable/landfront_00.png +0 -0
- package/android/src/main/res/drawable/landfront_1.png +0 -0
- package/android/src/main/res/drawable/landfront_11.png +0 -0
- package/android/src/main/res/drawable/landfront_20.png +0 -0
- package/android/src/main/res/drawable/landfront_21.png +0 -0
- package/android/src/main/res/drawable/landfront_22.png +0 -0
- package/android/src/main/res/drawable/landfront_3.png +0 -0
- package/android/src/main/res/drawable/landfront_33.png +0 -0
- package/android/src/main/res/drawable/landfront_40.png +0 -0
- package/android/src/main/res/drawable/landfront_43.png +0 -0
- package/android/src/main/res/drawable/landfront_44.png +0 -0
- package/android/src/main/res/drawable/landfront_5.png +0 -0
- package/android/src/main/res/drawable/landfront_55.png +0 -0
- package/android/src/main/res/drawable/landfront_61.png +0 -0
- package/android/src/main/res/drawable/landfront_63.png +0 -0
- package/android/src/main/res/drawable/landfront_66.png +0 -0
- package/android/src/main/res/drawable/landfront_70.png +0 -0
- package/android/src/main/res/drawable/landfront_71.png +0 -0
- package/android/src/main/res/drawable/landfront_73.png +0 -0
- package/android/src/main/res/drawable/landfront_77.png +0 -0
- package/android/src/main/res/drawable/landfront_8.png +0 -0
- package/android/src/main/res/drawable/landfront_88.png +0 -0
- package/android/src/main/res/drawable/landfront_90.png +0 -0
- package/android/src/main/res/drawable/landfront_95.png +0 -0
- package/android/src/main/res/drawable/landfront_99.png +0 -0
- package/android/src/main/res/drawable/landfront_a0.png +0 -0
- package/android/src/main/res/drawable/landfront_a8.png +0 -0
- package/android/src/main/res/drawable/landfront_aa.png +0 -0
- package/android/src/main/res/drawable/landfront_b1.png +0 -0
- package/android/src/main/res/drawable/landfront_b5.png +0 -0
- package/android/src/main/res/drawable/landfront_bb.png +0 -0
- package/android/src/main/res/drawable/landfront_c3.png +0 -0
- package/android/src/main/res/drawable/landfront_c8.png +0 -0
- package/android/src/main/res/drawable/landfront_cc.png +0 -0
- package/android/src/main/res/drawable/landfront_d.png +0 -0
- package/android/src/main/res/drawable/landfront_dd.png +0 -0
- package/android/src/main/res/drawable/landfront_e1.png +0 -0
- package/android/src/main/res/drawable/landfront_e5.png +0 -0
- package/android/src/main/res/drawable/landfront_ee.png +0 -0
- package/android/src/main/res/drawable/landfront_f0.png +0 -0
- package/android/src/main/res/drawable/landfront_f1.png +0 -0
- package/android/src/main/res/drawable/landfront_f5.png +0 -0
- package/android/src/main/res/drawable/landfront_ff.png +0 -0
- package/android/src/main/res/drawable/landfront_g3.png +0 -0
- package/android/src/main/res/drawable/landfront_g5.png +0 -0
- package/android/src/main/res/drawable/landfront_gg.png +0 -0
- package/android/src/main/res/drawable/landfront_h1.png +0 -0
- package/android/src/main/res/drawable/landfront_h3.png +0 -0
- package/android/src/main/res/drawable/landfront_h5.png +0 -0
- package/android/src/main/res/drawable/landfront_hh.png +0 -0
- package/android/src/main/res/drawable/landfront_i0.png +0 -0
- package/android/src/main/res/drawable/landfront_i3.png +0 -0
- package/android/src/main/res/drawable/landfront_i5.png +0 -0
- package/android/src/main/res/drawable/landfront_ii.png +0 -0
- package/android/src/main/res/drawable/landfront_j1.png +0 -0
- package/android/src/main/res/drawable/landfront_j8.png +0 -0
- package/android/src/main/res/drawable/landfront_jj.png +0 -0
- package/android/src/main/res/drawable/landfront_kk.png +0 -0
- package/android/src/main/res/drawable/landfront_ll.png +0 -0
- package/android/src/main/res/drawable/nav_notification_brand_icon.xml +16 -0
- package/android/src/main/res/drawable/navi_arrow_leftline.png +0 -0
- package/android/src/main/res/drawable/navi_lane_shape_bg_center.xml +5 -0
- package/android/src/main/res/drawable/navi_lane_shape_bg_left.xml +8 -0
- package/android/src/main/res/drawable/navi_lane_shape_bg_over.xml +6 -0
- package/android/src/main/res/drawable/navi_lane_shape_bg_right.xml +8 -0
- package/android/src/main/res/drawable-nodpi/nav_tracker_car.png +0 -0
- package/build/ExpoGaodeMapNaviView.d.ts +16 -0
- package/build/ExpoGaodeMapNaviView.d.ts.map +1 -1
- package/build/ExpoGaodeMapNaviView.js +74 -1
- package/build/ExpoGaodeMapNaviView.js.map +1 -1
- package/build/index.d.ts +46 -7
- package/build/index.d.ts.map +1 -1
- package/build/index.js +446 -7
- package/build/index.js.map +1 -1
- package/build/types/independent.types.d.ts +111 -12
- package/build/types/independent.types.d.ts.map +1 -1
- package/build/types/independent.types.js.map +1 -1
- package/build/types/native-module.types.d.ts +1 -1
- package/build/types/native-module.types.js.map +1 -1
- package/build/types/naviview.types.d.ts +304 -14
- package/build/types/naviview.types.d.ts.map +1 -1
- package/build/types/naviview.types.js.map +1 -1
- package/build/types/route.types.d.ts +12 -4
- package/build/types/route.types.d.ts.map +1 -1
- package/build/types/route.types.js.map +1 -1
- package/ios/ExpoGaodeMapNaviView.swift +2305 -203
- package/ios/ExpoGaodeMapNaviViewModule.swift +109 -1
- package/ios/ExpoGaodeMapNavigationModule.swift +22 -24
- package/ios/managers/IndependentRouteManager.swift +90 -26
- package/ios/map/ExpoGaodeMapModule.swift +34 -15
- package/ios/map/ExpoGaodeMapView.swift +13 -2
- package/ios/map/modules/LocationManager.swift +27 -1
- package/ios/map/utils/PermissionManager.swift +115 -6
- package/ios/routes/drive/DriveTruckRouteCalculator.swift +165 -77
- package/ios/routes/walkride/WalkRideRouteCalculator.swift +127 -1
- package/ios/services/IndependentRouteService.swift +198 -39
- package/ios/services/NavigationLiveActivityAttributes.swift +48 -0
- package/ios/services/NavigationLiveActivityManager.swift +359 -0
- package/package.json +17 -4
- package/plugin/build/withGaodeMap.d.ts +8 -0
- package/plugin/build/withGaodeMap.js +48 -4
- package/widget-template/README.md +46 -0
- package/widget-template/ios/NavigationLiveActivityWidget.swift +367 -0
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 接口。
|
|
@@ -29,6 +29,13 @@ npm install expo-gaode-map-navigation
|
|
|
29
29
|
**⚠️ 重要提示:**
|
|
30
30
|
如果项目中已安装 `expo-gaode-map`,请务必先卸载,否则会导致 Android 端二进制冲突(`3dmap` vs `navi-3dmap`)。`expo-gaode-map` 和 `expo-gaode-map-navigation` 由于 SDK 冲突不能同时安装,二选一使用。
|
|
31
31
|
|
|
32
|
+
> ⚠️ **版本兼容性说明**:
|
|
33
|
+
> - 如果你的项目使用 **Expo SDK 54 及以上**,请安装 默认的 版本。
|
|
34
|
+
> - 如果你的项目使用 **Expo SDK 53 及以下**(如 50, 51, 52, 53),请使用 **V1** 版本(Tag: `v1`)。
|
|
35
|
+
> ```bash
|
|
36
|
+
> npm install expo-gaode-map-navigation@1.1.8
|
|
37
|
+
> ```
|
|
38
|
+
|
|
32
39
|
|
|
33
40
|
### Config Plugin 配置(推荐)
|
|
34
41
|
|
|
@@ -42,7 +49,12 @@ npm install expo-gaode-map-navigation
|
|
|
42
49
|
"expo-gaode-map-navigation",
|
|
43
50
|
{
|
|
44
51
|
"androidKey": "your-android-key",
|
|
45
|
-
"iosKey": "your-ios-key"
|
|
52
|
+
"iosKey": "your-ios-key",
|
|
53
|
+
"enableBackgroundLocation": true,
|
|
54
|
+
"enableBackgroundAudio": true,
|
|
55
|
+
"enableNavigationNotification": true,
|
|
56
|
+
"enableIOSLiveActivity": true,
|
|
57
|
+
"enableIOSLiveActivityFrequentUpdates": true
|
|
46
58
|
}
|
|
47
59
|
]
|
|
48
60
|
]
|
|
@@ -57,6 +69,68 @@ npx expo run:android
|
|
|
57
69
|
npx expo run:ios
|
|
58
70
|
```
|
|
59
71
|
|
|
72
|
+
说明:
|
|
73
|
+
|
|
74
|
+
- `enableNavigationNotification` 仅 Android 生效,用于注入导航前台通知所需权限与 `NavigationForegroundService` 声明。
|
|
75
|
+
- `enableBackgroundAudio` 仅 iOS 生效(默认随 `enableBackgroundLocation` 自动开启),用于注入 `UIBackgroundModes: audio`,保障后台导航语音持续播报。
|
|
76
|
+
- `enableIOSLiveActivity` 仅 iOS 生效,用于注入 `NSSupportsLiveActivities`。
|
|
77
|
+
- `enableIOSLiveActivityFrequentUpdates` 仅 iOS 生效,用于注入 `NSSupportsLiveActivitiesFrequentUpdates`。
|
|
78
|
+
- 运行时还需要在 `NaviView` 里显式传 `androidBackgroundNavigationNotificationEnabled={true}` 才会在应用退到后台后显示导航常驻通知。
|
|
79
|
+
- iOS 运行时还需要在 `NaviView` 里显式传 `iosLiveActivityEnabled={true}` 才会持续更新 Live Activity。
|
|
80
|
+
|
|
81
|
+
## 示例工程
|
|
82
|
+
|
|
83
|
+
仓库内提供了可直接运行的 [`example-navigation`](/Volumes/xinxin/expo-gaode-map/example-navigation/README.md) 示例工程,专门用于验证导航能力。
|
|
84
|
+
|
|
85
|
+
推荐场景:
|
|
86
|
+
|
|
87
|
+
- 调试 `NaviView` 与示例工程里的自定义 HUD / 车道 HUD / 路况光柱
|
|
88
|
+
- 对比官方黑盒页、官方嵌入式页、自绘嵌入式页
|
|
89
|
+
- 验证独立算路、多路线选择、近似跟线导航
|
|
90
|
+
|
|
91
|
+
快速运行:
|
|
92
|
+
|
|
93
|
+
```bash
|
|
94
|
+
cd example-navigation
|
|
95
|
+
cp .env.example .env
|
|
96
|
+
npm install
|
|
97
|
+
npx expo run:android
|
|
98
|
+
```
|
|
99
|
+
|
|
100
|
+
如需 iOS:
|
|
101
|
+
|
|
102
|
+
```bash
|
|
103
|
+
cd example-navigation
|
|
104
|
+
cp .env.example .env
|
|
105
|
+
npm install
|
|
106
|
+
npx pod-install ios
|
|
107
|
+
npx expo run:ios
|
|
108
|
+
```
|
|
109
|
+
|
|
110
|
+
### iOS Live Activity Widget Extension
|
|
111
|
+
|
|
112
|
+
`iosLiveActivityEnabled` 只负责从导航回调里请求/更新 ActivityKit 状态。
|
|
113
|
+
真正显示在锁屏/灵动岛的 UI,需要你在 App 里提供 Widget Extension 的 `ActivityConfiguration`。
|
|
114
|
+
|
|
115
|
+
仓库已提供模板:
|
|
116
|
+
|
|
117
|
+
- [`packages/navigation/widget-template/README.md`](/Volumes/xinxin/expo-gaode-map/packages/navigation/widget-template/README.md)
|
|
118
|
+
- [`packages/navigation/widget-template/ios/NavigationLiveActivityWidget.swift`](/Volumes/xinxin/expo-gaode-map/packages/navigation/widget-template/ios/NavigationLiveActivityWidget.swift)
|
|
119
|
+
|
|
120
|
+
模板里直接使用库导出的 `NavigationLiveActivityAttributes`(`import ExpoGaodeMapNavigation`)。
|
|
121
|
+
|
|
122
|
+
#### iOS Live Activity 行为说明(补充)
|
|
123
|
+
|
|
124
|
+
- 导航进行中会持续推送实时状态到锁屏/灵动岛(前提是已启用 `iosLiveActivityEnabled` 且 Widget Extension 配置正确)。
|
|
125
|
+
- 到达目的地后,模块会先更新卡片为“到达目的地”,再延时约 6 秒自动结束 Live Activity。
|
|
126
|
+
- 为避免 ActivityKit `Payload maximum size exceeded`:
|
|
127
|
+
- 模块会优先保留转向图标,先裁剪文案字段;
|
|
128
|
+
- 如仍超限,才会降级去掉图标,确保状态更新不中断。
|
|
129
|
+
- 调试日志关键字:
|
|
130
|
+
- `payload ... keeping turn icon`
|
|
131
|
+
- `payload still too large ... dropped turn icon`
|
|
132
|
+
- `arrived destination card displayed for ... stopping activity`
|
|
133
|
+
|
|
60
134
|
|
|
61
135
|
## 快速开始
|
|
62
136
|
|
|
@@ -122,12 +196,30 @@ export default function NavigationScreen() {
|
|
|
122
196
|
style={{ flex: 1 }}
|
|
123
197
|
showCamera={true} // 显示摄像头
|
|
124
198
|
enableVoice={true} // 开启语音
|
|
199
|
+
androidBackgroundNavigationNotificationEnabled={true} // Android 后台导航常驻通知(需配合插件 enableNavigationNotification)
|
|
200
|
+
iosLiveActivityEnabled={true} // iOS Live Activity 状态更新(需配合插件 enableIOSLiveActivity + Widget Extension)
|
|
125
201
|
/>
|
|
126
202
|
</View>
|
|
127
203
|
);
|
|
128
204
|
}
|
|
129
205
|
```
|
|
130
206
|
|
|
207
|
+
### 3. 自定义嵌入式导航 UI
|
|
208
|
+
|
|
209
|
+
如果你要做“嵌入在自己页面里的导航页”,库本身提供的是底层 `NaviView`、导航事件和原生参数;完整的自定义 HUD / 车道 HUD / 路况光柱参考实现,已经迁移到仓库内的 [`example-navigation`](/Volumes/xinxin/expo-gaode-map/example-navigation/README.md)。
|
|
210
|
+
|
|
211
|
+
建议做法:
|
|
212
|
+
|
|
213
|
+
- 用 `NaviView` 负责底层导航地图、语音、车道事件、路况事件、路口大图事件
|
|
214
|
+
- 用 `onNaviInfoUpdate`、`onLaneInfoUpdate`、`onTrafficStatusesUpdate`、`onNaviVisualStateChange` 在业务侧自绘 HUD
|
|
215
|
+
- 直接参考 `example-navigation/lib/navigation-ui/EmbeddedNaviView.tsx` 及配套 UI 文件,按你的产品需求裁剪
|
|
216
|
+
|
|
217
|
+
注意:
|
|
218
|
+
|
|
219
|
+
- Android 官方嵌入式 `NaviView` 在部分 React Native / Expo 宿主中,顶部信息区、车道条、路口大图联动效果可能与高德官方 Demo 不完全一致
|
|
220
|
+
- 如果你要验证官方嵌入式 UI 本身,请直接跑 `example-navigation` 里的 `official-embedded` 示例页
|
|
221
|
+
- 如果你要交付稳定的嵌入式导航页,建议以示例工程里的“自定义 UI 导航界面”作为起点
|
|
222
|
+
|
|
131
223
|
## 详细用法
|
|
132
224
|
|
|
133
225
|
### 路径规划 (API)
|
|
@@ -145,12 +237,26 @@ const result = await calculateRoute({
|
|
|
145
237
|
to: { latitude: 39.91, longitude: 116.41 },
|
|
146
238
|
strategy: DriveStrategy.FASTEST, // 速度优先
|
|
147
239
|
avoidRoad: '京通快速路', // 避让道路名称
|
|
240
|
+
avoidPolygons: [
|
|
241
|
+
[
|
|
242
|
+
{ latitude: 39.905, longitude: 116.395 },
|
|
243
|
+
{ latitude: 39.905, longitude: 116.405 },
|
|
244
|
+
{ latitude: 39.915, longitude: 116.405 },
|
|
245
|
+
{ latitude: 39.915, longitude: 116.395 },
|
|
246
|
+
],
|
|
247
|
+
],
|
|
148
248
|
});
|
|
149
249
|
|
|
150
250
|
console.log(`总距离: ${result.routes[0].distance}米`);
|
|
151
251
|
console.log(`预计耗时: ${result.routes[0].duration}秒`);
|
|
152
252
|
```
|
|
153
253
|
|
|
254
|
+
说明:
|
|
255
|
+
|
|
256
|
+
- 当传入 `avoidRoad` 或 `avoidPolygons` 时,`calculateRoute` / `calculateDriveRoute` 会优先尝试通过 `expo-gaode-map-web-api` 获取官方“规避后路线预览”结果。
|
|
257
|
+
- 该回退仅用于路线预览与地图绘制;Web API 返回的是 polyline,不是导航 SDK 可直接启动的 `routeGroup/path`。
|
|
258
|
+
- 如果未安装 `expo-gaode-map-web-api`,则保持原有原生驾车算路逻辑不变。Android 仍可能命中底层 SDK 的避让重载;iOS 则没有官方导航 SDK 接口可直接消费任意规避道路/区域。
|
|
259
|
+
|
|
154
260
|
#### 步行/骑行路径规划
|
|
155
261
|
|
|
156
262
|
```typescript
|
|
@@ -192,6 +298,8 @@ const truckResult = await calculateRoute({
|
|
|
192
298
|
|
|
193
299
|
“独立路径规划”允许你先计算路线,并在地图上展示多条方案,用户选择其中一条后再开始导航。这通常比直接开始导航体验更好。
|
|
194
300
|
|
|
301
|
+
注意:`independentDriveRoute` 仍然依赖导航 SDK 自身的独立算路能力,因此这里不接 Web API 的规避预览结果。`avoidRoad` / `avoidPolygons` 不再作为 `IndependentDriveRouteOptions` 的标准公开参数暴露。若你需要“规避道路/区域后再开始导航”,建议先用 `calculateRoute` 做预览与确认,再按终点重新发起原生导航。
|
|
302
|
+
|
|
195
303
|
```typescript
|
|
196
304
|
import {
|
|
197
305
|
independentDriveRoute,
|
|
@@ -210,13 +318,71 @@ const result = await independentDriveRoute({
|
|
|
210
318
|
// 2. 选择某一条路线(例如 index=1 的路线)
|
|
211
319
|
// 这通常配合地图上的点击事件,高亮显示某条路线
|
|
212
320
|
await selectIndependentRoute({
|
|
321
|
+
token: result.token,
|
|
213
322
|
routeId: result.routes[1].id
|
|
214
323
|
});
|
|
215
324
|
|
|
216
325
|
// 3. 使用当前选中的路线开始导航
|
|
217
326
|
await startNaviWithIndependentPath({
|
|
218
|
-
|
|
327
|
+
token: result.token,
|
|
328
|
+
naviType: 1, // 1 = 模拟导航
|
|
329
|
+
});
|
|
330
|
+
```
|
|
331
|
+
|
|
332
|
+
### 近似跟线导航(第一版)
|
|
333
|
+
|
|
334
|
+
当你已经通过 Web API 拿到一条想要的路线,但导航 SDK 不能直接吃这条 `polyline` 时,可以使用 `followWebPlannedRoute`。
|
|
335
|
+
|
|
336
|
+
它会:
|
|
337
|
+
|
|
338
|
+
- 从 Web 路线提炼一组途经锚点
|
|
339
|
+
- 用这些锚点重新发起原生独立算路
|
|
340
|
+
- 选择最接近 Web 路线的一条原生路线
|
|
341
|
+
- 仅在匹配足够接近时才启动导航
|
|
342
|
+
|
|
343
|
+
```typescript
|
|
344
|
+
import { followWebPlannedRoute } from 'expo-gaode-map-navigation';
|
|
345
|
+
|
|
346
|
+
const result = await followWebPlannedRoute({
|
|
347
|
+
from: { latitude: 39.9, longitude: 116.4 },
|
|
348
|
+
to: { latitude: 39.91, longitude: 116.41 },
|
|
349
|
+
webRoute: {
|
|
350
|
+
polyline: webResult.routes[0].polyline ?? [],
|
|
351
|
+
},
|
|
352
|
+
maxViaPoints: 8,
|
|
353
|
+
maxDeviationMeters: 120,
|
|
354
|
+
startNavigation: true,
|
|
355
|
+
naviType: 1, // 1 = 模拟导航
|
|
356
|
+
});
|
|
357
|
+
|
|
358
|
+
console.log(result.mode); // matched | approximate | preview_only
|
|
359
|
+
console.log(result.anchorWaypoints);
|
|
360
|
+
console.log(result.candidateMatches);
|
|
361
|
+
```
|
|
362
|
+
|
|
363
|
+
说明:
|
|
364
|
+
|
|
365
|
+
- 这不是“强制按 Web 线导航”,而是“尽量贴近 Web 线”。
|
|
366
|
+
- 若返回 `preview_only`,说明原生导航 SDK 算出的路线与 Web 线路偏差过大,建议仅做预览,不要直接开导航。
|
|
367
|
+
- 若你只想拿锚点,不立即导航,可以单独使用 `buildAnchorWaypointsFromWebRoute`。
|
|
368
|
+
|
|
369
|
+
如果你希望继续使用嵌入式官方导航 UI,可以先完成近似跟线选路,再通过 `ExpoGaodeMapNaviView` 的 ref 使用独立路径启动:
|
|
370
|
+
|
|
371
|
+
```typescript
|
|
372
|
+
const matchResult = await followWebPlannedRoute({
|
|
373
|
+
from,
|
|
374
|
+
to,
|
|
375
|
+
webRoute,
|
|
376
|
+
startNavigation: false,
|
|
219
377
|
});
|
|
378
|
+
|
|
379
|
+
if (matchResult.mode !== 'preview_only') {
|
|
380
|
+
await naviRef.current?.startNavigationWithIndependentPath(matchResult.token, {
|
|
381
|
+
routeId: matchResult.selectedRouteId,
|
|
382
|
+
routeIndex: matchResult.selectedRouteIndex,
|
|
383
|
+
naviType: 1,
|
|
384
|
+
});
|
|
385
|
+
}
|
|
220
386
|
```
|
|
221
387
|
|
|
222
388
|
### 官方导航页(openOfficialNaviPage)
|
|
@@ -230,7 +396,7 @@ await openOfficialNaviPage({
|
|
|
230
396
|
to: { latitude: 39.908823, longitude: 116.39747, name: '终点' }, // 必填
|
|
231
397
|
pageType: 'NAVI', // ROUTE | NAVI
|
|
232
398
|
startNaviDirectly: true,
|
|
233
|
-
naviMode: 2, // 1=实时导航, 2
|
|
399
|
+
naviMode: 2, // 1=实时导航, 2=模拟导航(iOS 官方组件不支持模拟)
|
|
234
400
|
theme: 'BLUE', // BLUE | WHITE | BLACK
|
|
235
401
|
trafficEnabled: true,
|
|
236
402
|
showCrossImage: true,
|
|
@@ -241,6 +407,7 @@ await openOfficialNaviPage({
|
|
|
241
407
|
|
|
242
408
|
- 支持 Android / iOS 平台差异参数(如 `dayAndNightMode`、`broadcastMode`、`mapViewModeType`、`trackingMode` 等)。
|
|
243
409
|
- iOS 直接进导航页时需开启后台定位 `UIBackgroundModes: location`。
|
|
410
|
+
- iOS 官方导航组件模式不支持模拟导航;若传 `naviMode: 2` 会直接返回错误提示。
|
|
244
411
|
- Android 依赖 `AmapRouteActivity`,Config Plugin 会自动注入 Manifest。
|
|
245
412
|
|
|
246
413
|
### 地图组件 (Map)
|
|
@@ -335,23 +502,141 @@ const result = await calculateTransitRoute({
|
|
|
335
502
|
| `AVOID_CONGESTION` (4) | 躲避拥堵 |
|
|
336
503
|
| ... | 更多策略请参考类型定义 |
|
|
337
504
|
|
|
505
|
+
### 自定义嵌入式 UI 参考实现
|
|
506
|
+
|
|
507
|
+
库不再直接导出 `EmbeddedNaviView` 这类成品 UI 组件;这部分实现现在放在示例工程里,便于你直接查看和复制。
|
|
508
|
+
|
|
509
|
+
参考文件:
|
|
510
|
+
|
|
511
|
+
- `example-navigation/lib/navigation-ui/EmbeddedNaviView.tsx`
|
|
512
|
+
- `example-navigation/lib/navigation-ui/EmbeddedNaviHud.tsx`
|
|
513
|
+
- `example-navigation/lib/navigation-ui/EmbeddedNaviLaneView.tsx`
|
|
514
|
+
- `example-navigation/lib/navigation-ui/EmbeddedNaviTrafficBar.tsx`
|
|
515
|
+
|
|
516
|
+
这套示例实现演示了:
|
|
517
|
+
|
|
518
|
+
- 默认 `showUIElements={false}` 的完整自定义 UI 模式
|
|
519
|
+
- 基于 `driveViewEdgePadding` / `screenAnchor` 的嵌入式地图可视区域管理
|
|
520
|
+
- 基于 `onNaviInfoUpdate` 的顶部 HUD
|
|
521
|
+
- 基于 `onLaneInfoUpdate` 的自绘车道 HUD
|
|
522
|
+
- 基于 `onTrafficStatusesUpdate` 的自绘路况光柱
|
|
523
|
+
- “全览 / 锁车”与路况开关等浮层控制按钮
|
|
524
|
+
|
|
338
525
|
### NaviView Props
|
|
339
526
|
|
|
340
527
|
| 属性 | 类型 | 说明 |
|
|
341
528
|
|---|---|---|
|
|
342
529
|
| `naviType` | number | 导航类型(0: GPS, 1: 模拟) |
|
|
343
|
-
| `
|
|
530
|
+
| `realCrossDisplay` | boolean | 是否显示路口放大图 |
|
|
344
531
|
| `showCamera` | boolean | 是否显示摄像头 |
|
|
345
|
-
| `
|
|
532
|
+
| `carImage` | string \| ImageSourcePropType | 自定义导航车标;iOS 映射 `setCarImage`,Android 映射 `setCarBitmap` |
|
|
533
|
+
| `carImageSize` | object | 自定义导航车标尺寸,格式 `{ width, height }`(单位 dp/pt,需同时传宽高) |
|
|
534
|
+
| `startPointImage` | string \| ImageSourcePropType | 自定义起点标注图 |
|
|
535
|
+
| `wayPointImage` | string \| ImageSourcePropType | 自定义途经点标注图 |
|
|
536
|
+
| `endPointImage` | string \| ImageSourcePropType | 自定义终点标注图 |
|
|
537
|
+
| `trafficLayerEnabled` | boolean | 是否显示实时交通路况线 |
|
|
538
|
+
| `showTrafficButton` | boolean | 是否显示交通按钮/交通图层开关 |
|
|
539
|
+
| `showDriveCongestion` | boolean | 是否显示拥堵气泡 |
|
|
540
|
+
| `showTrafficLightView` | boolean | 是否显示红绿灯倒计时气泡 |
|
|
541
|
+
| `mapViewModeType` | number | 地图样式模式:`0` 白天、`1` 黑夜、`2` 自动、`3` 自定义(Android 当前未开放样式路径时会降级为白天) |
|
|
542
|
+
| `isNightMode` | boolean | 兼容属性,等价于 `mapViewModeType` 的 `1/0`;若同时传 `mapViewModeType`,以后者为准 |
|
|
543
|
+
| `showUIElements` | boolean | Android / iOS 均支持整体 UI 显隐 |
|
|
544
|
+
| `laneInfoVisible` | boolean | Android 是否显示官方车道信息 |
|
|
545
|
+
| `hideNativeLaneInfoLayout` | boolean | iOS 是否隐藏官方车道信息条,交给 RN 自绘 |
|
|
546
|
+
| `iosLiveActivityEnabled` | boolean | iOS 是否启用导航 Live Activity 状态更新(需已配置 Widget Extension) |
|
|
547
|
+
| `modeCrossDisplay` | boolean | Android 是否显示 3D 路口模型;iOS 当前不支持,会忽略 |
|
|
548
|
+
| `eyrieCrossDisplay` | boolean | Android 是否显示鹰眼路口图 |
|
|
549
|
+
| `secondActionVisible` | boolean | Android 是否显示辅助操作区域 |
|
|
550
|
+
| `backupOverlayVisible` | boolean | Android 是否显示备用路线覆盖物 |
|
|
551
|
+
| `androidStatusBarPaddingTop` | number | Android 顶部额外间距;若显示官方原生顶部信息区且未显式传值,封装会自动补系统状态栏高度 |
|
|
552
|
+
| `naviStatusBarEnabled` | boolean | Android 是否启用高德官方导航状态栏;若当前 AMap SDK 不支持该接口,则自动降级为 no-op |
|
|
553
|
+
| `lockZoom` | number | Android 锁车态缩放级别 |
|
|
554
|
+
| `lockTilt` | number | Android 锁车态倾斜角度 |
|
|
555
|
+
| `eagleMapVisible` | boolean | Android 是否显示鹰眼小地图 |
|
|
556
|
+
| `pointToCenter` | object | Android 锁车态自车锚点位置 |
|
|
557
|
+
| `androidBackgroundNavigationNotificationEnabled` | boolean | Android 退到后台时是否启用前台服务导航常驻通知(默认 false) |
|
|
558
|
+
| `driveViewEdgePadding` | object | iOS 导航内容边距 |
|
|
559
|
+
| `screenAnchor` | object | iOS 地图视图锚点 |
|
|
560
|
+
| `showBackupRoute` | boolean | iOS 是否显示备选路线 |
|
|
561
|
+
| `showEagleMap` | boolean | iOS 是否显示鹰眼小地图 |
|
|
346
562
|
| `enableVoice` | boolean | 是否开启语音播报 |
|
|
347
563
|
| `onArrive` | function | 到达目的地回调 |
|
|
348
564
|
| `onNaviInfoUpdate` | function | 导航信息更新(剩余距离、时间等) |
|
|
565
|
+
| `onLaneInfoUpdate` | function | Android / iOS 车道信息更新,用于自绘车道 HUD |
|
|
566
|
+
|
|
567
|
+
### NaviView UI 能力清单
|
|
568
|
+
|
|
569
|
+
已开放且两端都有实现:
|
|
570
|
+
|
|
571
|
+
- `showCamera`
|
|
572
|
+
- `autoLockCar`
|
|
573
|
+
- `autoChangeZoom`
|
|
574
|
+
- `trafficLayerEnabled`
|
|
575
|
+
- `realCrossDisplay`
|
|
576
|
+
- `naviMode`
|
|
577
|
+
- `showMode`
|
|
578
|
+
- `mapViewModeType`
|
|
579
|
+
- `isNightMode`
|
|
580
|
+
- `showTrafficBar`
|
|
581
|
+
- `showTrafficButton`
|
|
582
|
+
- `showUIElements`
|
|
583
|
+
- `showGreyAfterPass`
|
|
584
|
+
- `showVectorline`
|
|
585
|
+
- `showCompassEnabled`
|
|
586
|
+
- `showDriveCongestion`
|
|
587
|
+
- `showTrafficLightView`
|
|
588
|
+
|
|
589
|
+
仅 Android 已开放:
|
|
590
|
+
|
|
591
|
+
- `carOverlayVisible`
|
|
592
|
+
- `fourCornersImage`
|
|
593
|
+
- `routeMarkerVisible`
|
|
594
|
+
- `naviArrowVisible`
|
|
595
|
+
- `laneInfoVisible`
|
|
596
|
+
- `modeCrossDisplay`
|
|
597
|
+
- `eyrieCrossDisplay`
|
|
598
|
+
- `secondActionVisible`
|
|
599
|
+
- `backupOverlayVisible`
|
|
600
|
+
- `androidStatusBarPaddingTop`
|
|
601
|
+
- `naviStatusBarEnabled`
|
|
602
|
+
- `lockZoom`
|
|
603
|
+
- `lockTilt`
|
|
604
|
+
- `eagleMapVisible`
|
|
605
|
+
- `pointToCenter`
|
|
606
|
+
- `isNaviTravelView`
|
|
607
|
+
|
|
608
|
+
仅 iOS 已开放:
|
|
609
|
+
|
|
610
|
+
- `hideNativeLaneInfoLayout`
|
|
611
|
+
- `showRoute`
|
|
612
|
+
- `carCompassImage`
|
|
613
|
+
- `cameraImage`
|
|
614
|
+
- `trafficBarFrame`
|
|
615
|
+
- `trafficBarColors`
|
|
616
|
+
- `showMoreButton`
|
|
617
|
+
- `lineWidth`
|
|
618
|
+
- `driveViewEdgePadding`
|
|
619
|
+
- `screenAnchor`
|
|
620
|
+
|
|
621
|
+
关于 iOS 路口放大图能力:
|
|
622
|
+
|
|
623
|
+
- iOS 官方公开的是 `showCrossImage / hideCrossImage`
|
|
624
|
+
- 因此库里的 `realCrossDisplay` 对应 iOS 实景路口放大图显示控制
|
|
625
|
+
- Android 的 `modeCrossDisplay` 没有 iOS 对等公开接口,传入 iOS 时会被忽略
|
|
626
|
+
- `showBackupRoute`
|
|
627
|
+
- `showEagleMap`
|
|
628
|
+
|
|
629
|
+
当前这份清单里此前列出的“剩余代表项”已全部开放。
|
|
630
|
+
|
|
631
|
+
如果后续还要继续往下包,更适合继续补的是更底层的样式类配置,而不是核心导航 UI 能力。
|
|
349
632
|
|
|
350
633
|
## 注意事项
|
|
351
634
|
|
|
352
635
|
1. **二进制冲突**:严禁与 `expo-gaode-map` 共存。本模块已包含 `3dmap` SDK。
|
|
353
636
|
2. **Web API**:如果需要更灵活的 HTTP 算路(如公交跨城规划、Web端展示),推荐配合 `expo-gaode-map-web-api` 使用。
|
|
354
637
|
3. **权限**:使用导航功能前,请确保应用已获取定位权限(`ACCESS_FINE_LOCATION`)。
|
|
638
|
+
4. **Android 状态栏兼容性**:`naviStatusBarEnabled` 依赖高德 Android 导航 SDK 某些版本才提供的 `AMapNaviViewOptions.setNaviStatusBarEnabled(...)`。当前封装已做兼容处理:若宿主工程解析到的 SDK 不包含该方法,则不会再编译失败,而是在运行时跳过该设置并输出 warning。此时该 prop 在 Android 上等价于 no-op。
|
|
639
|
+
5. **嵌入式 UI 边界**:库导出的是底层 `NaviView` 能力;完整自定义导航界面请参考 `example-navigation` 里的示例实现,它也不是高德官方黑盒导航页的 UI 替代品。
|
|
355
640
|
|
|
356
641
|
|
|
357
642
|
## 📚 文档与资源
|
package/android/build.gradle
CHANGED
|
@@ -65,4 +65,8 @@ dependencies {
|
|
|
65
65
|
} else {
|
|
66
66
|
implementation 'com.amap.api:navi-3dmap:latest.integration'
|
|
67
67
|
}
|
|
68
|
+
|
|
69
|
+
// Live Update APIs on NotificationCompat.Builder
|
|
70
|
+
// setRequestPromotedOngoing / setShortCriticalText were added in androidx.core 1.17.0.
|
|
71
|
+
implementation 'androidx.core:core-ktx:1.17.0'
|
|
68
72
|
}
|
|
@@ -16,8 +16,9 @@
|
|
|
16
16
|
|
|
17
17
|
<!-- 设备信息权限 - SDK 初始化必需 -->
|
|
18
18
|
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
|
|
19
|
+
<uses-permission android:name="android.permission.WAKE_LOCK" />
|
|
19
20
|
|
|
20
21
|
<!-- 蓝牙权限 - 用于室内定位 (可选) -->
|
|
21
22
|
<uses-permission android:name="android.permission.BLUETOOTH" />
|
|
22
23
|
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
|
|
23
|
-
</manifest>
|
|
24
|
+
</manifest>
|
|
@@ -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
|
-
|
|
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
|
-
|
|
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)
|