【移动端】AMap Flutter与Android AMap SDK交互

背景

本文的背景,是因为我在开发高德地图时,需要自定义高德比例尺位置和样式;但结果查看了AMap Flutter插件和AMap SDK源码后,发现AMap无法添加自定义MyMethodCallHandler的实现类!

why?

源码

在Flutter中,高德地图的每个地图视图都是通过AMapPlatformView类生成且管理的,源码如下

package com.amap.flutter.map;import android.content.Context;
import android.os.Bundle;
import android.view.View;import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.lifecycle.DefaultLifecycleObserver;
import androidx.lifecycle.LifecycleOwner;import com.amap.api.maps.AMap;
import com.amap.api.maps.AMapOptions;
import com.amap.api.maps.TextureMapView;
import com.amap.flutter.map.core.MapController;
import com.amap.flutter.map.overlays.marker.MarkersController;
import com.amap.flutter.map.overlays.polygon.PolygonsController;
import com.amap.flutter.map.overlays.polyline.PolylinesController;
import com.amap.flutter.map.utils.LogUtil;import java.util.HashMap;
import java.util.Map;import io.flutter.embedding.engine.plugins.activity.ActivityPluginBinding;
import io.flutter.plugin.common.BinaryMessenger;
import io.flutter.plugin.common.MethodCall;
import io.flutter.plugin.common.MethodChannel;
import io.flutter.plugin.platform.PlatformView;/*** @author whm* @date 2020/10/27 5:49 PM* @mail hongming.whm@alibaba-inc.com* @since*/
public class AMapPlatformViewimplementsDefaultLifecycleObserver,ActivityPluginBinding.OnSaveInstanceStateListener,MethodChannel.MethodCallHandler,PlatformView {private static final String CLASS_NAME = "AMapPlatformView";private final MethodChannel methodChannel;private MapController mapController;private MarkersController markersController;private PolylinesController polylinesController;private PolygonsController polygonsController;private TextureMapView mapView;private boolean disposed = false;private final Map<String, MyMethodCallHandler> myMethodCallHandlerMap;AMapPlatformView(int id,Context context,BinaryMessenger binaryMessenger,LifecycleProvider lifecycleProvider,AMapOptions options) {methodChannel = new MethodChannel(binaryMessenger, "amap_flutter_map_" + id);methodChannel.setMethodCallHandler(this);myMethodCallHandlerMap = new HashMap<String, MyMethodCallHandler>(8);try {mapView = new TextureMapView(context, options);AMap amap = mapView.getMap();mapController = new MapController(methodChannel, mapView);markersController = new MarkersController(methodChannel, amap);polylinesController = new PolylinesController(methodChannel, amap);polygonsController = new PolygonsController(methodChannel, amap);initMyMethodCallHandlerMap();lifecycleProvider.getLifecycle().addObserver(this);} catch (Throwable e) {LogUtil.e(CLASS_NAME, "<init>", e);}}private void initMyMethodCallHandlerMap() {String[] methodIdArray = mapController.getRegisterMethodIdArray();if (null != methodIdArray && methodIdArray.length > 0) {for (String methodId : methodIdArray) {myMethodCallHandlerMap.put(methodId, mapController);}}methodIdArray = markersController.getRegisterMethodIdArray();if (null != methodIdArray && methodIdArray.length > 0) {for (String methodId : methodIdArray) {myMethodCallHandlerMap.put(methodId, markersController);}}methodIdArray = polylinesController.getRegisterMethodIdArray();if (null != methodIdArray && methodIdArray.length > 0) {for (String methodId : methodIdArray) {myMethodCallHandlerMap.put(methodId, polylinesController);}}methodIdArray = polygonsController.getRegisterMethodIdArray();if (null != methodIdArray && methodIdArray.length > 0) {for (String methodId : methodIdArray) {myMethodCallHandlerMap.put(methodId, polygonsController);}}}public MapController getMapController() {return mapController;}public MarkersController getMarkersController() {return markersController;}public PolylinesController getPolylinesController() {return polylinesController;}public PolygonsController getPolygonsController() {return polygonsController;}@Overridepublic void onMethodCall(@NonNull MethodCall call, @NonNull MethodChannel.Result result) {LogUtil.i(CLASS_NAME, "onMethodCall==>" + call.method + ", arguments==> " + call.arguments);String methodId = call.method;if (myMethodCallHandlerMap.containsKey(methodId)) {myMethodCallHandlerMap.get(methodId).doMethodCall(call, result);} else {LogUtil.w(CLASS_NAME, "onMethodCall, the methodId: " + call.method + ", not implemented");result.notImplemented();}}@Overridepublic void onCreate(@NonNull LifecycleOwner owner) {LogUtil.i(CLASS_NAME, "onCreate==>");try {if (disposed) {return;}if (null != mapView) {mapView.onCreate(null);}} catch (Throwable e) {LogUtil.e(CLASS_NAME, "onCreate", e);}}@Overridepublic void onStart(@NonNull LifecycleOwner owner) {LogUtil.i(CLASS_NAME, "onStart==>");}@Overridepublic void onResume(@NonNull LifecycleOwner owner) {LogUtil.i(CLASS_NAME, "onResume==>");try {if (disposed) {return;}if (null != mapView) {mapView.onResume();}} catch (Throwable e) {LogUtil.e(CLASS_NAME, "onResume", e);}}@Overridepublic void onPause(@NonNull LifecycleOwner owner) {LogUtil.i(CLASS_NAME, "onPause==>");try {if (disposed) {return;}mapView.onPause();} catch (Throwable e) {LogUtil.e(CLASS_NAME, "onPause", e);}}@Overridepublic void onStop(@NonNull LifecycleOwner owner) {LogUtil.i(CLASS_NAME, "onStop==>");}@Overridepublic void onDestroy(@NonNull LifecycleOwner owner) {LogUtil.i(CLASS_NAME, "onDestroy==>");try {if (disposed) {return;}destroyMapViewIfNecessary();} catch (Throwable e) {LogUtil.e(CLASS_NAME, "onDestroy", e);}}@Overridepublic void onSaveInstanceState(@NonNull Bundle bundle) {LogUtil.i(CLASS_NAME, "onDestroy==>");try {if (disposed) {return;}mapView.onSaveInstanceState(bundle);} catch (Throwable e) {LogUtil.e(CLASS_NAME, "onSaveInstanceState", e);}}@Overridepublic void onRestoreInstanceState(@Nullable Bundle bundle) {LogUtil.i(CLASS_NAME, "onDestroy==>");try {if (disposed) {return;}mapView.onCreate(bundle);} catch (Throwable e) {LogUtil.e(CLASS_NAME, "onRestoreInstanceState", e);}}@Overridepublic View getView() {LogUtil.i(CLASS_NAME, "getView==>");return mapView;}@Overridepublic void dispose() {LogUtil.i(CLASS_NAME, "dispose==>");try {if (disposed) {return;}methodChannel.setMethodCallHandler(null);destroyMapViewIfNecessary();disposed = true;} catch (Throwable e) {LogUtil.e(CLASS_NAME, "dispose", e);}}private void destroyMapViewIfNecessary() {if (mapView == null) {return;}mapView.onDestroy();}}

分析

通过上面的源码可以发现如下几点:

  1. 在它onMethodCall回调方法中,判断了方法名是否在myMethodCallHandlerMap中存在
@Overridepublic void onMethodCall(@NonNull MethodCall call, @NonNull MethodChannel.Result result) {LogUtil.i(CLASS_NAME, "onMethodCall==>" + call.method + ", arguments==> " + call.arguments);String methodId = call.method;if (myMethodCallHandlerMap.containsKey(methodId)) {myMethodCallHandlerMap.get(methodId).doMethodCall(call, result);} else {LogUtil.w(CLASS_NAME, "onMethodCall, the methodId: " + call.method + ", not implemented");result.notImplemented();}}
  1. myMethodCallHandlerMap 是一个final属性,这也就意味着,它在构造函数中已经实例化了,且不能更改值,只能修改内容,它通过initMyMethodCallHandlerMap注册所有的Hnadler
private final Map<String, MyMethodCallHandler> myMethodCallHandlerMap;AMapPlatformView(int id,Context context,BinaryMessenger binaryMessenger,LifecycleProvider lifecycleProvider,AMapOptions options) {methodChannel = new MethodChannel(binaryMessenger, "amap_flutter_map_" + id);methodChannel.setMethodCallHandler(this);myMethodCallHandlerMap = new HashMap<String, MyMethodCallHandler>(8);try {mapView = new TextureMapView(context, options);AMap amap = mapView.getMap();mapController = new MapController(methodChannel, mapView);markersController = new MarkersController(methodChannel, amap);polylinesController = new PolylinesController(methodChannel, amap);polygonsController = new PolygonsController(methodChannel, amap);initMyMethodCallHandlerMap();lifecycleProvider.getLifecycle().addObserver(this);} catch (Throwable e) {LogUtil.e(CLASS_NAME, "<init>", e);}
}
  1. 通过initMyMethodCallHandlerMap方法可以看出,它只注册了mapControllermarkersControllerpolylinesControllerpolygonsController四类Handler,且没有提供可以自定义注册的方法,所以我也无能为力了
private void initMyMethodCallHandlerMap() {String[] methodIdArray = mapController.getRegisterMethodIdArray();if (null != methodIdArray && methodIdArray.length > 0) {for (String methodId : methodIdArray) {myMethodCallHandlerMap.put(methodId, mapController);}}methodIdArray = markersController.getRegisterMethodIdArray();if (null != methodIdArray && methodIdArray.length > 0) {for (String methodId : methodIdArray) {myMethodCallHandlerMap.put(methodId, markersController);}}methodIdArray = polylinesController.getRegisterMethodIdArray();if (null != methodIdArray && methodIdArray.length > 0) {for (String methodId : methodIdArray) {myMethodCallHandlerMap.put(methodId, polylinesController);}}methodIdArray = polygonsController.getRegisterMethodIdArray();if (null != methodIdArray && methodIdArray.length > 0) {for (String methodId : methodIdArray) {myMethodCallHandlerMap.put(methodId, polygonsController);}}}

解决思路

1. 通过修改initMyMethodCallHandlerMap属性值(不推荐

通过上面分析可以想到,通过修改initMyMethodCallHandlerMap属性值,自定义handler,但前提时需要拿到对应的AMapPlatformView实例,通常的方法可以通过ServiceLoader来获取,但在Android中ServiceLoader是不用生效的,Android环境中必须通过PathClassLoaderDexClassLoader来加载类,具体加载方法,可以参考其他大神的博客,我这里不做过多追述,因为我不推荐

2. 通过PlatformViewsController来获取视图(推荐

推荐这个方法是因为它可以i通过mapId直接获取,且不用过多介入到内存和进程中的交互中

  1. 创建一个自定义MethodChannel
var methodChannel = MethodChannel(flutterEngine.dartExecutor.binaryMessenger, "method_channel_id")methodChannel.setMethodCallHandler { call, result ->run {//....}}
  1. AMapPlatformView继承PlatformView,所以可以通过FlutterEngine.getPlatformViewsController().getPlatformViewById(viewId)方式获取,获取到的是TextureMapView视图,TextureMapView.getMap()可以获取AMap类,从而实现对当前地图视图进行任何操作,
class MainActivity : FlutterActivity() {var handerControllerMap = HashMap<String, IMyMethodCallHander>();override fun configureFlutterEngine(flutterEngine: FlutterEngine) {super.configureFlutterEngine(flutterEngine)IFlutterFactory.engine=flutterEngine;}
}

/*** 当前缩放级别下,地图上1像素点对应的长度,单位米。* @param call* @param result*/public void getScalePerPixel(MethodCall call, MethodChannel.Result result) {Object mapId = call.argument("mapId");if (null == call || null == mapId) {return;}try {View view = IFlutterFactory.engine.getPlatformViewsController().getPlatformViewById(Integer.parseInt(mapId.toString()));if (view != null) {result.success(((TextureMapView) view).getMap().getScalePerPixel());} else {result.error("ScaleControllerExecption", "获取比例尺数据失败!", "");}} catch (Exception e) {result.error("ScaleControllerExecption", e.getMessage(), e.getLocalizedMessage());}}
  1. viewId 其实就是mapId,可以通过AMapController.mapId获取,而AMapController可以通过AMapWidgetonMapCreated的回调方法获取
void onMapCreated(AMapController controller) {//连接自定义的MethodChannelMethodChannel _channel=MethodChannel("method_channel_id");//将mapId传给android端double? scale=await _channel.invokeMethod<double>("scale#get",<String,dynamic>{"mapId":_mapController.mapId});print('-----scale---$scale');
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/758294.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

AI算力池化赋能企业大模型价值探索

1. 大语言模型企业落地中的算力痛点 随着人工智能技术的飞速发展&#xff0c;自然语言处理&#xff08;NLP&#xff09;成为了热门的研究领域之一。在这一领域中&#xff0c;大语言模型&#xff08;Large Language Models&#xff09;凭借其强大的语言理解和生成能力&#xff…

学习笔记-华为IPD转型2020:3,IPD的实施

3. IPD的实施 1999 年开始的 IPD 转型是计划中的多个转型项目中的第一个&#xff08;Liu&#xff0c;2015&#xff09;。华为为此次转型成立了一个专门的团队&#xff0c;从大约20人开始&#xff0c;他们是华为第一产业的高层领导。董事会主席孙雅芳是这个团队的负责人。该团…

【Maven】使用maven-jar、maven-assembly、maven-shade优雅的实现第三方依赖一同打Jar包

文章目录 一.前言二.常规Jar 打包&#xff1a;maven-jar-plugin三.Shade 打包&#xff1a;maven-shade-plugin1.如何使用2.将部分jar包添加或排除3.将依赖jar包内部资源添加或排除4.自动将所有不使用的类排除5.将依赖的类重命名并打包进来 &#xff08;隔离方案&#xff09;6.修…

软件测试面试接口测试常见问题

接口测试中,依赖登录状态的接口如何测试? 答:依赖登录状态的接口的本质上是在每次发送请求时需要带上 session或者 cookie才能发送成功,在构建POST请求时添加必要的 session或者cookie 按你的理解,软件接口是什么? 就是指程序中具体负责在不同模块之间传输或接受数据的并做…

【笔记】本地笔记本ubuntu 远程传输服务器(ubuntu系统) 文件

命令格式&#xff1a; &#xff08;1&#xff09;将本地文件拷贝到远程&#xff1a;scp 文件名 用户名计算机IP或者计算机名称:远程路径 &#xff08;2&#xff09;从远程将文件拷回本地&#xff1a;scp 用户名计算机IP或者计算机名称:文件名本地路径 &#xff08;3&#xff0…

【Linux】项目部署CPU彪高如何定位

1.查看所有CPU占比 使用top指令获取彪高进程的PID 2.输出进程的信息 ps H -eo pid,tid,%cpu | grep 1313 3.查看线程的信息 jstack tid nid都是十六进制的 4.进制转换 将 tid的十进制转为十六进制 找到nid 可以定位到具体位置 5.关闭程序 ps -ef | grep java kill -9 jav…

机器人路径规划:基于迪杰斯特拉算法(Dijkstra)的机器人路径规划(提供Python代码)

迪杰斯特拉算法(Dijkstra)是由荷兰计算机科学家狄克斯特拉于1959年提出的&#xff0c;因此又叫狄克斯特拉算法。是从一个顶点到其余各顶点的最短路径算法&#xff0c;解决的是有权图中最短路径问题。迪杰斯特拉算法主要特点是从起始点开始&#xff0c;采用贪心算法的策略&#…

你真的了解可穿戴技术吗?

可穿戴技术(wearable technology)&#xff0c;最早是20世纪60年代由麻省理工学院媒体实验室提出的创新技术。利用该技术&#xff0c;可以把多媒体、传感器和无线通信等技术嵌入人们的衣物中&#xff0c;可支持手势和眼动操作等多种交互方式&#xff0c;主要探索和创造可直接穿戴…

通过更新路书当前坐标下marker的icon来展示沿途的风景

通过更新路书当前坐标下marker的icon来展示沿途的风景 1.效果图2.[工程链接](https://download.csdn.net/download/m0_61864577/88978866)3.需修改地方: 本文演示了如何通过百度地图的路书功能,展示途经的风景。定时缩放,既有全局路径,又有当前位置和运动轨迹;可以显示当前坐标…

万界星空科技铜拉丝行业生产管理MES系统

铜拉丝加工作为金属加工行业中的一个重要分支&#xff0c;在国内乃至全球工业领域都扮演着不可或缺的角色。特别是在高导材料领域&#xff0c;铜拉丝加工更是发挥着重要作用。 铜作为一种传统的金属材料&#xff0c;其拉丝加工技术早在古代就有应用。随着工业化进程的加快&…

Python:unable to import ‘smart_open.gcs‘, disabling that module

程序出现报错 unable to import smart_open.gcs, disabling that module查看smart_open的版本 $ pip show smart_open Name: smart-open Version: 1.11.1 Requires: requests, boto, boto3 Required-by: gensim解决办法是&#xff0c;安装其他版本 pip install smart_open1.…

去除项目git的控制 端口号的关闭

以下操作都是在windows下。只是记录一下。 find . -name “.git” | xargs rm -rf 查看所有分支 git branch -a 查看当前分支 git branch -a 切换分支 git chenkout develop docker 查看容器的ip docker inspect -f ‘{{.Name}} - {{range .NetworkSettings.Networks}}{{.IP…

web高可用集群(lvs负载均衡+keepalved高可用)

web高可用集群(lvs负载均衡keepalved高可用) 主机IP地址VIP192.168.88.88DS1(MASTER)192.168.88.38DS2(BACKUP)192.168.88.66web1192.168.88.10web2192.168.88.20 |---------------------------------| | 192.168.88.38|---- VIP:192.16…

综合练习(python)

前言 有了前面的知识积累&#xff0c;我们这里做两个小练习&#xff0c;都要灵活运用前面的知识。 First 需求 根据美国/英国各自YouTube的数据&#xff0c;绘制出各自的评论数量的直方图 第一版 import numpy as np from matplotlib import pyplot as plt import matplo…

Flutter Widget:State 状态管理

响应式的编程框架永恒的主题——“状态(State)管理” 无论是在 React/Vue/Flutter 中讨论的问题和解决的思想都是一致的。 StatefulWidget的状态应该被谁管理&#xff1f;Widget本身&#xff1f;父 Widget &#xff1f;都会&#xff1f;还是另一个对象&#xff1f; 下面是官…

webpack的热更新是如何做到的?

Webpack的热更新&#xff08;Hot Module Replacement&#xff0c;HMR&#xff09;技术是现代前端开发中极为重要的一部分&#xff0c;它极大地提高了开发效率&#xff0c;并让开发者能够实时地查看代码更改后的效果&#xff0c;而无需手动刷新页面。在本文中&#xff0c;我们将…

什么是晶振精度?对电路有什么影响?

晶振的精度是衡量其频率稳定性的重要指标&#xff0c;指在温度为25℃条件下晶振正常工作时输出的频率工差范围&#xff0c;通常以ppm(parts per million)表示即“PPM”。这意味着晶振的频率每百万分之一的偏差&#xff0c;如果以10ppm的精度为例&#xff0c;意味着每秒钟的工作…

VMware虚拟机硬盘容量扩容方法

扩容后不会影响原文件。亲测有效&#xff0c;高效便捷 - 在关机状态下&#xff0c;先在VM上直接扩容硬盘容量&#xff0c;输入扩容后的硬盘最大容量 注意&#xff0c;如果想在原硬盘上增加容量&#xff0c;需要将原来的快照都删除 - 输入最大磁盘大小 运行虚拟机进入系统&…

CMake实现target_link_libraries_with_definitions

target_link_libraries( … … …) https://cmake.org/cmake/help/latest/command/target_link_libraries.html#target-link-libraries target_link_libraries为target指定链接的库&#xff0c;有时候想要携带库本身的编译参数。保证看到的头文件是一致的。 可以使用方法&…

VUE3实现一键复制功能

1.使用vueuse/core (安全策略的问题&#xff0c;非https网页下面不让你拷贝) vueuse/core是一个为 Vue.js 提供常用功能和实用工具函数的库。它是基于 Composition API&#xff0c;为开发者提供了一组易于使用和可复用的函数 http网页使用解决方法:解决方法! 2.使用插件 n…