提交 79bc2145 authored 作者: 史晓晨's avatar 史晓晨

feat:集成location实现web定位功能

上级 25203569
...@@ -17,9 +17,9 @@ class MyApp extends StatefulWidget { ...@@ -17,9 +17,9 @@ class MyApp extends StatefulWidget {
} }
class _MyAppState extends State<MyApp> { class _MyAppState extends State<MyApp> {
Map<String, Object>? _locationResult; Map<String, Object?>? _locationResult;
StreamSubscription<Map<String, Object>>? _locationListener; StreamSubscription<Map<String, Object?>>? _locationListener;
AMapFlutterLocation _locationPlugin = new AMapFlutterLocation(); AMapFlutterLocation _locationPlugin = new AMapFlutterLocation();
...@@ -49,7 +49,7 @@ class _MyAppState extends State<MyApp> { ...@@ -49,7 +49,7 @@ class _MyAppState extends State<MyApp> {
AMapFlutterLocation.updatePrivacyAgree(true); AMapFlutterLocation.updatePrivacyAgree(true);
/// 动态申请定位权限 /// 动态申请定位权限
requestPermission(); // requestPermission();
///设置Android和iOS的apiKey<br> ///设置Android和iOS的apiKey<br>
///key的申请请参考高德开放平台官网说明<br> ///key的申请请参考高德开放平台官网说明<br>
...@@ -71,7 +71,7 @@ class _MyAppState extends State<MyApp> { ...@@ -71,7 +71,7 @@ class _MyAppState extends State<MyApp> {
///注册定位结果监听 ///注册定位结果监听
_locationListener = _locationPlugin _locationListener = _locationPlugin
.onLocationChanged() .onLocationChanged()
.listen((Map<String, Object> result) { .listen((Map<String, Object?> result) {
setState(() { setState(() {
_locationResult = result; _locationResult = result;
}); });
......
...@@ -59,7 +59,7 @@ class AMapFlutterLocation { ...@@ -59,7 +59,7 @@ class AMapFlutterLocation {
} }
/// 定位结果回调 /// 定位结果回调
Stream<Map<String, Object>> onLocationChanged() { Stream<Map<String, Object?>> onLocationChanged() {
final result = AmapLocationPluginPlatform.instance.onLocationChanged(); final result = AmapLocationPluginPlatform.instance.onLocationChanged();
return result; return result;
} }
......
...@@ -27,6 +27,7 @@ class AmapLocationPluginMethodChannel extends AmapLocationPluginPlatform { ...@@ -27,6 +27,7 @@ class AmapLocationPluginMethodChannel extends AmapLocationPluginPlatform {
String? _pluginKey; String? _pluginKey;
/// 初始化key /// 初始化key
@override
void initKey() { void initKey() {
_pluginKey = DateTime.now().millisecondsSinceEpoch.toString(); _pluginKey = DateTime.now().millisecondsSinceEpoch.toString();
} }
...@@ -48,17 +49,20 @@ class AmapLocationPluginMethodChannel extends AmapLocationPluginPlatform { ...@@ -48,17 +49,20 @@ class AmapLocationPluginMethodChannel extends AmapLocationPluginPlatform {
} }
/// 开始定位 /// 开始定位
@override
void startLocation() { void startLocation() {
_methodChannel.invokeMethod('startLocation', {'pluginKey': _pluginKey}); _methodChannel.invokeMethod('startLocation', {'pluginKey': _pluginKey});
} }
/// 停止定位 /// 停止定位
@override
void stopLocation() { void stopLocation() {
_methodChannel.invokeMethod('stopLocation', {'pluginKey': _pluginKey}); _methodChannel.invokeMethod('stopLocation', {'pluginKey': _pluginKey});
return; return;
} }
///计算2点之间的距离 ///计算2点之间的距离
@override
Future<double> calculateDistance( Future<double> calculateDistance(
double lat1, double lon1, double lat2, double lon2) async { double lat1, double lon1, double lat2, double lon2) async {
var result = await _methodChannel.invokeMethod<double>('calculateDistance', var result = await _methodChannel.invokeMethod<double>('calculateDistance',
...@@ -72,12 +76,14 @@ class AmapLocationPluginMethodChannel extends AmapLocationPluginPlatform { ...@@ -72,12 +76,14 @@ class AmapLocationPluginMethodChannel extends AmapLocationPluginPlatform {
///iOS端: https://lbs.amap.com/api/ios-location-sdk/guide/create-project/get-key<br> ///iOS端: https://lbs.amap.com/api/ios-location-sdk/guide/create-project/get-key<br>
///[androidKey] Android平台的key<br> ///[androidKey] Android平台的key<br>
///[iosKey] ios平台的key<br> ///[iosKey] ios平台的key<br>
@override
void setApiKey(String androidKey, String iosKey, String webKey) { void setApiKey(String androidKey, String iosKey, String webKey) {
_methodChannel _methodChannel
.invokeMethod('setApiKey', {'android': androidKey, 'ios': iosKey}); .invokeMethod('setApiKey', {'android': androidKey, 'ios': iosKey});
} }
/// 设置定位参数 /// 设置定位参数
@override
void setLocationOption(AMapLocationOption locationOption) { void setLocationOption(AMapLocationOption locationOption) {
Map option = locationOption.getOptionsMap(); Map option = locationOption.getOptionsMap();
option['pluginKey'] = _pluginKey; option['pluginKey'] = _pluginKey;
...@@ -85,6 +91,7 @@ class AmapLocationPluginMethodChannel extends AmapLocationPluginPlatform { ...@@ -85,6 +91,7 @@ class AmapLocationPluginMethodChannel extends AmapLocationPluginPlatform {
} }
/// 销毁定位 /// 销毁定位
@override
void destroy() { void destroy() {
_methodChannel.invokeListMethod('destroy', {'pluginKey': _pluginKey}); _methodChannel.invokeListMethod('destroy', {'pluginKey': _pluginKey});
if (_subscription != null) { if (_subscription != null) {
...@@ -140,6 +147,7 @@ class AmapLocationPluginMethodChannel extends AmapLocationPluginPlatform { ...@@ -140,6 +147,7 @@ class AmapLocationPluginMethodChannel extends AmapLocationPluginPlatform {
/// `errorCode`: 错误码,当定位失败时才会返回对应的错误码, 具体错误请参考:https://lbs.amap.com/api/android-location-sdk/guide/utilities/errorcode /// `errorCode`: 错误码,当定位失败时才会返回对应的错误码, 具体错误请参考:https://lbs.amap.com/api/android-location-sdk/guide/utilities/errorcode
/// ///
/// `errorInfo`: 错误信息, 当定位失败时才会返回 /// `errorInfo`: 错误信息, 当定位失败时才会返回
@override
Stream<Map<String, Object>> onLocationChanged() { Stream<Map<String, Object>> onLocationChanged() {
if (_receiveStream == null) { if (_receiveStream == null) {
_receiveStream = StreamController(); _receiveStream = StreamController();
...@@ -166,6 +174,7 @@ class AmapLocationPluginMethodChannel extends AmapLocationPluginPlatform { ...@@ -166,6 +174,7 @@ class AmapLocationPluginMethodChannel extends AmapLocationPluginPlatform {
/// 高德SDK合规使用方案请参考官网地址:https://lbs.amap.com/news/sdkhgsy /// 高德SDK合规使用方案请参考官网地址:https://lbs.amap.com/news/sdkhgsy
/// [hasContains] 隐私声明中是否包含高德隐私政策说明<br> /// [hasContains] 隐私声明中是否包含高德隐私政策说明<br>
/// [hasShow] 隐私权政策是否弹窗展示告知用户<br> /// [hasShow] 隐私权政策是否弹窗展示告知用户<br>
@override
void updatePrivacyShow(bool hasContains, bool hasShow) { void updatePrivacyShow(bool hasContains, bool hasShow) {
_methodChannel.invokeMethod('updatePrivacyStatement', _methodChannel.invokeMethod('updatePrivacyStatement',
{'hasContains': hasContains, 'hasShow': hasShow}); {'hasContains': hasContains, 'hasShow': hasShow});
...@@ -175,6 +184,7 @@ class AmapLocationPluginMethodChannel extends AmapLocationPluginPlatform { ...@@ -175,6 +184,7 @@ class AmapLocationPluginMethodChannel extends AmapLocationPluginPlatform {
/// 高德SDK合规使用方案请参考官网地址:https://lbs.amap.com/news/sdkhgsy<br> /// 高德SDK合规使用方案请参考官网地址:https://lbs.amap.com/news/sdkhgsy<br>
/// <b>必须保证在调用定位功能之前调用, 建议首次启动App时弹出《隐私政策》并取得用户同意</b><br> /// <b>必须保证在调用定位功能之前调用, 建议首次启动App时弹出《隐私政策》并取得用户同意</b><br>
/// [hasAgree] 隐私权政策是否已经取得用户同意<br> /// [hasAgree] 隐私权政策是否已经取得用户同意<br>
@override
void updatePrivacyAgree(bool hasAgree) { void updatePrivacyAgree(bool hasAgree) {
_methodChannel _methodChannel
.invokeMethod('updatePrivacyStatement', {'hasAgree': hasAgree}); .invokeMethod('updatePrivacyStatement', {'hasAgree': hasAgree});
......
...@@ -111,7 +111,7 @@ abstract class AmapLocationPluginPlatform extends PlatformInterface { ...@@ -111,7 +111,7 @@ abstract class AmapLocationPluginPlatform extends PlatformInterface {
/// `errorCode`: 错误码,当定位失败时才会返回对应的错误码, 具体错误请参考:https://lbs.amap.com/api/android-location-sdk/guide/utilities/errorcode /// `errorCode`: 错误码,当定位失败时才会返回对应的错误码, 具体错误请参考:https://lbs.amap.com/api/android-location-sdk/guide/utilities/errorcode
/// ///
/// `errorInfo`: 错误信息, 当定位失败时才会返回 /// `errorInfo`: 错误信息, 当定位失败时才会返回
Stream<Map<String, Object>> onLocationChanged() { Stream<Map<String, Object?>> onLocationChanged() {
throw UnimplementedError('onLocationChanged() has not been implemented.'); throw UnimplementedError('onLocationChanged() has not been implemented.');
} }
......
...@@ -3,96 +3,121 @@ ...@@ -3,96 +3,121 @@
// package as the core of your plugin. // package as the core of your plugin.
// ignore: avoid_web_libraries_in_flutter // ignore: avoid_web_libraries_in_flutter
import 'dart:convert'; import 'dart:async';
import 'dart:developer';
import 'dart:ui';
import 'package:amap_flutter_location/amap_location_plugin_platform_interface.dart'; import 'package:amap_flutter_location/amap_location_plugin_platform_interface.dart';
import 'package:flutter/material.dart'; import 'package:amap_flutter_location/web/web_location_imp.dart';
import 'package:flutter_web_plugins/flutter_web_plugins.dart'; import 'package:flutter_web_plugins/flutter_web_plugins.dart';
import 'package:fluttertoast/fluttertoast.dart';
import 'package:js/js_util.dart';
import 'amap_location_option.dart'; import 'amap_location_option.dart';
import 'web/amap.dart'; import 'package:location/location.dart';
import 'web/loader.dart';
import 'web/utils.dart';
/// A web implementation of the TestWebPluginPlatform of the TestWebPlugin plugin. /// A web implementation of the TestWebPluginPlatform of the TestWebPlugin plugin.
class AmapLocationPluginWeb extends AmapLocationPluginPlatform { class AmapLocationPluginWeb extends AmapLocationPluginPlatform {
/// Constructs a AmapLocationPluginWeb /// Constructs a AmapLocationPluginWeb
AmapLocationPluginWeb(); AmapLocationPluginWeb();
late AMap _aMap;
String? _pluginKey; String? _pluginKey;
Map<String, WebLocationImp> _webLocationImpMap = {};
StreamController<Map<String, Object?>>? _receiveStream;
StreamSubscription<Map<String, Object?>>? _subscription;
static void registerWith(Registrar registrar) { static void registerWith(Registrar registrar) {
AmapLocationPluginPlatform.instance = AmapLocationPluginWeb(); AmapLocationPluginPlatform.instance = AmapLocationPluginWeb();
} }
/// 初始化key
@override @override
void initKey() { void initKey() {
_pluginKey = DateTime.now().millisecondsSinceEpoch.toString(); _pluginKey = DateTime.now().millisecondsSinceEpoch.toString();
} }
/// 隐私权政策弹窗展示告知用户
@override @override
void updatePrivacyShow(bool hasContains, bool hasShow) { void updatePrivacyShow(bool hasContains, bool hasShow) {
// 暂无要求 // 暂不需要
} }
/// 隐私权政策取得用户同意
@override @override
void updatePrivacyAgree(bool hasAgree) { void updatePrivacyAgree(bool hasAgree) {
// 暂无要求 // 暂不需要
} }
/// 设置apiKey
@override @override
void setApiKey(String androidKey, String iosKey, String webKey) { void setApiKey(String androidKey, String iosKey, String webKey) {
var promise = load(LoaderOptions( // 暂不需要
key: webKey,
version: '1.4.15',
plugins: ['AMap.Geolocation'],
));
promiseToFuture(promise).then((value) {
print('初始化成功');
MapOptions _mapOptions = MapOptions(
zoom: 10,
);
_aMap = AMap('container', _mapOptions);
}, onError: (e) {
print('初始化错误:$e');
});
} }
/// 设置定位参数
@override @override
Stream<Map<String, Object>> onLocationChanged() { void setLocationOption(AMapLocationOption locationOption) {
return Stream.value({}); Map option = locationOption.getOptionsMap();
option['pluginKey'] = _pluginKey;
_getWebLocationImp(option)?.setLocationOption(option);
} }
/// 开始定位
@override @override
void setLocationOption(AMapLocationOption locationOption) {} void startLocation() {
_getWebLocationImp({"pluginKey": _pluginKey})?.startLocation();
}
/// 停止定位
@override @override
void startLocation() { void stopLocation() {
_aMap.plugin(['AMap.Geolocation'], allowInterop(() { _getWebLocationImp({"pluginKey": _pluginKey})?.stopLocation();
GeolocationOptions _geolocationOptions = GeolocationOptions( }
enableHighAccuracy: false,
timeout: 10000, /// 定位结果回调
convert: true, @override
); Stream<Map<String, Object?>> onLocationChanged() {
var geolocation = Geolocation(_geolocationOptions); if (_receiveStream == null) {
geolocation.getCurrentPosition(allowInterop((status, result) { _receiveStream = StreamController();
Fluttertoast.showToast( WebLocationImp? webLocationImp =
msg: "${status} ${result.message}", _getWebLocationImp({"pluginKey": _pluginKey});
toastLength: Toast.LENGTH_LONG, _subscription = webLocationImp?.onLocationChanged()?.listen(
gravity: ToastGravity.CENTER, (Map<String, Object?> event) {
// timeInSecForIos: 1, if (event['pluginKey'] == _pluginKey) {
backgroundColor: const Color(0xB3000000), Map<String, Object?> newEvent = Map<String, Object?>.of(event);
textColor: Colors.white, newEvent.remove('pluginKey');
fontSize: 17.0); _receiveStream?.add(newEvent);
consoleLog("=====11111 ${status} ${stringify(result)}"); }
})); }, onError: (Object error) {
})); print('onLocationChanged error: $error');
_receiveStream?.addError(error);
}, onDone: () {
print('onLocationChanged done');
_receiveStream?.close();
});
}
return _receiveStream!.stream;
}
/// 销毁定位
@override
void destroy() {
WebLocationImp? webLocationImp =
_getWebLocationImp({"pluginKey": _pluginKey});
webLocationImp?.destroy();
_webLocationImpMap.remove(webLocationImp);
if (_subscription != null) {
_receiveStream?.close();
_subscription?.cancel();
_receiveStream = null;
_subscription = null;
}
}
/// 获取web定位实现
WebLocationImp? _getWebLocationImp(Map? argsMap) {
String? pluginKey = argsMap?["pluginKey"];
if (pluginKey == null || pluginKey.isEmpty) return null;
if (!_webLocationImpMap.containsKey(pluginKey)) {
WebLocationImp webLocationImp = WebLocationImp(pluginKey: pluginKey);
_webLocationImpMap[pluginKey] = webLocationImp;
}
return _webLocationImpMap[pluginKey];
} }
} }
@JS('AMap') // @JS('AMap')
library amap; // library amap;
//
import 'package:js/js.dart'; // import 'package:js/js.dart';
import 'dart:html' as html show window; // import 'dart:html' as html show window;
//
// 这里`new Map(id)` 调用js的`new AMap.Map(id)` // // 这里`new Map(id)` 调用js的`new AMap.Map(id)`
@JS('Map') // @JS('Map')
class AMap { // class AMap {
external AMap(dynamic /*String|HTMLDivElement*/ div, MapOptions opts); // external AMap(dynamic /*String|HTMLDivElement*/ div, MapOptions opts);
//
/// 加载插件 // /// 加载插件
external plugin(dynamic /*String|List*/ name, void Function() callback); // external plugin(dynamic /*String|List*/ name, void Function() callback);
} // }
//
@JS() // @JS()
@anonymous // @anonymous
class MapOptions { // class MapOptions {
external LngLat get center; // external LngLat get center;
//
external set center(LngLat v); // external set center(LngLat v);
//
external factory MapOptions({ // external factory MapOptions({
LngLat center, // LngLat center,
num zoom, // num zoom,
String /*‘2D’|‘3D’*/ viewMode, // String /*‘2D’|‘3D’*/ viewMode,
}); // });
} // }
//
@JS() // @JS()
class LngLat { // class LngLat {
external num getLng(); // external num getLng();
//
external num getLat(); // external num getLat();
//
external LngLat(num lng, num lat); // external LngLat(num lng, num lat);
} // }
//
/// 定位 // /// 定位
@JS() // @JS()
class Geolocation { // class Geolocation {
external Geolocation(GeolocationOptions opts); // external Geolocation(GeolocationOptions opts);
//
external getCurrentPosition( // external getCurrentPosition(
void Function(String status, GeolocationResult result) callback); // void Function(String status, GeolocationResult result) callback);
} // }
//
@JS() // @JS()
@anonymous // @anonymous
class GeolocationOptions { // class GeolocationOptions {
external factory GeolocationOptions({ // external factory GeolocationOptions({
bool enableHighAccuracy, // 进行浏览器原生定位的时候是否尝试获取较高精度,可能影响定位效率,默认为false // bool enableHighAccuracy, // 进行浏览器原生定位的时候是否尝试获取较高精度,可能影响定位效率,默认为false
num timeout, // 定位的超时时间,毫秒 // num timeout, // 定位的超时时间,毫秒
bool convert, // 是否将定位结果转换为高德坐标 // bool convert, // 是否将定位结果转换为高德坐标
}); // });
} // }
//
@JS() // @JS()
@anonymous // @anonymous
class GeolocationResult { // class GeolocationResult {
external LngLat get position; // external LngLat get position;
//
external String get message; // external String get message;
} // }
@JS('AMapLoader') // @JS('AMapLoader')
library loader; // library loader;
//
import 'package:js/js.dart'; // import 'package:js/js.dart';
//
/// 高德地图 Loader js // /// 高德地图 Loader js
external load(LoaderOptions options); // external load(LoaderOptions options);
//
@JS() // @JS()
@anonymous // @anonymous
class LoaderOptions { // class LoaderOptions {
//
external factory LoaderOptions({ // external factory LoaderOptions({
///您申请的key值 // ///您申请的key值
String key, // String key,
/// JSAPI 版本号 // /// JSAPI 版本号
String version, // String version,
//同步加载的插件列表 // //同步加载的插件列表
List<String> plugins, // List<String> plugins,
}); // });
} // }
@JS() // @JS()
library stringify; // library stringify;
//
import 'package:js/js.dart'; // import 'package:js/js.dart';
//
// Calls invoke JavaScript `JSON.stringify(obj)`. // // Calls invoke JavaScript `JSON.stringify(obj)`.
@JS('JSON.stringify') // @JS('JSON.stringify')
external String stringify(Object obj); // external String stringify(Object obj);
//
@JS('console.log') // @JS('console.log')
external String consoleLog(Object obj); // external String consoleLog(Object obj);
\ No newline at end of file //
// @JS()
// external void alert(Object msg);
\ No newline at end of file
import 'dart:async';
import 'package:location/location.dart';
/// web端定位实现
class WebLocationImp {
Location? _location;
String? pluginKey;
/// StreamController
StreamController<Map<String, Object?>>? streamController =
StreamController<Map<String, Object?>>();
WebLocationImp({required this.pluginKey}) {
_location ??= Location();
}
/// 开启定位
void startLocation() async {
try {
// 检测是否开启定位服务
bool? _serviceEnabled = await _location?.serviceEnabled();
if (_serviceEnabled == false) {
_serviceEnabled = await _location?.requestService();
if (_serviceEnabled == false) {
assert(false, "定位服务未开启");
return;
}
}
// 检测是否开启定位权限
PermissionStatus? _permissionGranted = await _location?.hasPermission();
if (_permissionGranted == PermissionStatus.denied) {
_permissionGranted = await _location?.requestPermission();
if (_permissionGranted != PermissionStatus.granted) {
assert(false, "定位权限未开启");
return;
}
}
// 获取定位信息
LocationData? locationData = await _location?.getLocation();
var map = {
"latitude": locationData?.latitude,
"longitude": locationData?.longitude,
"pluginKey": pluginKey,
};
streamController?.add(map);
} catch (e) {
print("定位获取异常:$e");
assert(false, "定位获取异常:$e");
}
}
/// 定位回调
Stream<Map<String, Object?>>? onLocationChanged() {
return streamController?.stream;
}
/// 设置定位参数
void setLocationOption(Map optionMap) {}
/// 停止定位
void stopLocation() {}
/// 销毁定位
void destroy() {
_location = null;
streamController?.close();
streamController = null;
}
}
...@@ -13,9 +13,8 @@ dependencies: ...@@ -13,9 +13,8 @@ dependencies:
flutter_web_plugins: flutter_web_plugins:
sdk: flutter sdk: flutter
plugin_platform_interface: ^2.0.2 plugin_platform_interface: ^2.0.2
js: ^0.7.1 # https://pub.dev/packages/location
# toast https://pub.dev/packages/fluttertoast location: ^6.0.2
fluttertoast: ^8.2.1
dev_dependencies: dev_dependencies:
flutter_test: flutter_test:
......
Markdown 格式
0%
您添加了 0 到此讨论。请谨慎行事。
请先完成此评论的编辑!
注册 或者 后发表评论