Android高德地图填坑

项目中用到地图的地方越来越多,从O2O商城、出行、交通、单车等无处不在使用地图,以下是在多个项目中集成高德地图常用的几个功能点,及填坑。

定位功能

使用了最新的SdkAndroid_Map3D_SDK_V5.1.0_20170518.jar,与之前的sdk的定位回调有部分差异。

  1. 小蓝点策略
myLocationStyle.myLocationType(MyLocationStyle.LOCATION_TYPE_SHOW);//只定位一次。
myLocationStyle.myLocationType(MyLocationStyle.LOCATION_TYPE_LOCATE) ;//定位一次,且将视角移动到地图中心点。
myLocationStyle.myLocationType(MyLocationStyle.LOCATION_TYPE_FOLLOW) ;//连续定位、且将视角移动到地图中心点,定位蓝点跟随设备移动。(1秒1次定位)
myLocationStyle.myLocationType(MyLocationStyle.LOCATION_TYPE_MAP_ROTATE);//连续定位、且将视角移动到地图中心点,地图依照设备方向旋转,定位点会跟随设备移动。(1秒1次定位)
myLocationStyle.myLocationType(MyLocationStyle.LOCATION_TYPE_LOCATION_ROTATE);//连续定位、且将视角移动到地图中心点,定位点依照设备方向旋转,并且会跟随设备移动。(1秒1次定位)默认执行此种模式。
//以下三种模式从5.1.0版本开始提供
myLocationStyle.myLocationType(MyLocationStyle.LOCATION_TYPE_LOCATION_ROTATE_NO_CENTER);//连续定位、蓝点不会移动到地图中心点,定位点依照设备方向旋转,并且蓝点会跟随设备移动。
myLocationStyle.myLocationType(MyLocationStyle.LOCATION_TYPE_FOLLOW_NO_CENTER);//连续定位、蓝点不会移动到地图中心点,并且蓝点会跟随设备移动。
myLocationStyle.myLocationType(MyLocationStyle.LOCATION_TYPE_MAP_ROTATE_NO_CENTER);//连续定位、蓝点不会移动到地图中心点,地图依照设备方向旋转,并且蓝点会跟随设备移动。

增加的比较有用的是LOCATION_TYPE_FOLLOW_NO_CENTER,小蓝点的移动位置由我们来控制。

  1. 定位的实现
aMap.setOnMyLocationChangeListener(this);
@Override
public void onMyLocationChange(Location location) {
    if (location != null) {
        Bundle bundle = location.getExtras();
        if (bundle != null) {
            mLocationLatitude = location.getLatitude();
            mLocationLongitude = location.getLongitude();
            mLocationLatitude = Double.valueOf(df.format(mLocationLatitude));
            mLocationLongitude = Double.valueOf(df.format(mLocationLongitude));
            if (isFirst) {
                if (mLocationLatitude > 0 && mLocationLongitude > 0) {
                    CameraUpdate cu = CameraUpdateFactory.newLatLngZoom(new LatLng(mLocationLatitude, mLocationLongitude), 17);
                    aMap.moveCamera(cu);
                } else {
                    CameraUpdate cu = CameraUpdateFactory.newLatLngZoom(new LatLng(Constant.DEFAULT_LATITUDE, Constant.DEFAULT_LONGITUDE), 17);
                    aMap.moveCamera(cu);
                }
                isFirst = false;
            }
        } else {
            ToastUtil.showShort(mContext, "定位失败,请检查您的定位权限");
        }
    } 
}

Marker

  1. 始终固定在屏幕中心位置的点
    private void addMarkerInScreenCenter() {
        if (screenMarker == null) {
            screenMarker = aMap.addMarker(new MarkerOptions().zIndex(2)
                    .icon(BitmapDescriptorFactory.fromResource(R.drawable.ic_screen_location)));
        }
        screenMarker.setAnchor(0.5f, 1.0f);
        LatLng latLng = aMap.getCameraPosition().target;
        Point screenPosition = aMap.getProjection().toScreenLocation(latLng);
        screenMarker.setPositionByPixels(screenPosition.x, screenPosition.y);
        screenMarker.setClickable(false);
    }
  1. 给中心点添加跳跃动画
public void screenMarkerJump(AMap aMap, Marker screenMarker) {
    if (screenMarker != null) {
        final LatLng latLng = screenMarker.getPosition();
        Point point = aMap.getProjection().toScreenLocation(latLng);
        point.y -= Utils.dip2px(mContext, 20);
        LatLng target = aMap.getProjection()
                .fromScreenLocation(point);
        //使用TranslateAnimation,填写一个需要移动的目标点
        Animation animation = new TranslateAnimation(target);
        animation.setInterpolator(new Interpolator() {
            @Override
            public float getInterpolation(float input) {
                // 模拟重加速度的interpolator
                if (input <= 0.5) {
                    return (float) (0.5f - 2 * (0.5 - input) * (0.5 - input));
                } else {
                    return (float) (0.5f - Math.sqrt((input - 0.5f) * (1.5f - input)));
                }
            }
        });
        //整个移动所需要的时间
        animation.setDuration(600);
        //设置动画
        screenMarker.setAnimation(animation);
        //开始动画
        screenMarker.startAnimation();
    }   
}
  1. 添加多个Marker标记
    获取到需要添加的Marker列表,循环添加到地图上。
        for (int i = 0; i < mParkingListBeen.size(); i++) {
            LatLng latLng = new LatLng(mParkingListBeen.get(i).getLatitude(), mParkingListBeen.get(i).getLongitude());
            Marker marker = aMap.addMarker(new MarkerOptions()
                    .title("i")
                    .anchor(0.5f, 0.92f)
                    .position(latLng).zIndex(1)
                    .icon(BitmapDescriptorFactory
                            .fromBitmap(mParkingBitmap))
                    .draggable(false));
            marker.setObject(mParkingListBeen.get(i));
            mBikeMarkers.add(marker);
            aMap.setOnMarkerClickListener(this);
        }

需要注意的以下几点:

1.Marker的bitmap要提前初始化,添加Marker的时候使用。
2.Marker需要点击弹窗,必须设置title属性
3.anchor的默认值是0.5f和1.0f,是相对x和y方向的偏移比例
4.zIndex设置z轴的重叠顺序
5.Marker的信息可以setObject(),获取的时候getObject()并进行强制转换
6.添加Marker的点击事件
  1. Marker的生长动画
public void startGrowAnimation(AMap aMap, Marker growMarker) {
        try {
            if (growMarker != null) {
                Animation animation = new ScaleAnimation(0, 1, 0, 1);
                animation.setInterpolator(new LinearInterpolator());
                //整个移动所需要的时间
                animation.setDuration(150);
                //设置动画
                growMarker.setAnimation(animation);
                //开始动画
                growMarker.startAnimation();
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
  1. 移除Marker
    移除marker有两种方式,一个是remove,一个是destory,对于在地图上需要大量移除并添加新的marker来说,慎用destory,否则会出现marker在地图上的各种问题。
if (marker != null) {
        marker.remove;
        marker = null;
}

导航路线的绘制和清除

使用了Sdk里的路线导航确实可以绘制,但是如果我想在当前页面再次绘制某个点,除了使用aMap.clear()方法,其他的均无效果。这样就会导致一个问题,已经绘制到地图上的信息均被清除,导致体验性很差。于是有了第二种绘制路线的方法:使用Web服务API

  1. 根据返回的数据封装成model
public class WalkingPathBean {

    private String status;
    private String info;
    private String infocode;
    private String count;
    private RouteBean route;

    public String getStatus() {
        return status;
    }

    public void setStatus(String status) {
        this.status = status;
    }

    public String getInfo() {
        return info;
    }

    public void setInfo(String info) {
        this.info = info;
    }

    public String getInfocode() {
        return infocode;
    }

    public void setInfocode(String infocode) {
        this.infocode = infocode;
    }

    public String getCount() {
        return count;
    }

    public void setCount(String count) {
        this.count = count;
    }

    public RouteBean getRoute() {
        return route;
    }

    public void setRoute(RouteBean route) {
        this.route = route;
    }

    public static class RouteBean {

        private String origin;
        private String destination;
        private List<PathsBean> paths;

        public String getOrigin() {
            return origin;
        }

        public void setOrigin(String origin) {
            this.origin = origin;
        }

        public String getDestination() {
            return destination;
        }

        public void setDestination(String destination) {
            this.destination = destination;
        }

        public List<PathsBean> getPaths() {
            return paths;
        }

        public void setPaths(List<PathsBean> paths) {
            this.paths = paths;
        }

        public static class PathsBean {

            private String distance;
            private String duration;
            private List<StepsBean> steps;

            public String getDistance() {
                return distance;
            }

            public void setDistance(String distance) {
                this.distance = distance;
            }

            public String getDuration() {
                return duration;
            }

            public void setDuration(String duration) {
                this.duration = duration;
            }

            public List<StepsBean> getSteps() {
                return steps;
            }

            public void setSteps(List<StepsBean> steps) {
                this.steps = steps;
            }

            public static class StepsBean {

                private String polyline;

                public String getPolyline() {
                    return polyline;
                }

                public void setPolyline(String polyline) {
                    this.polyline = polyline;
                }
            }
        }
    }
}
  1. 循环组装成一个所有路线的list
if (size > 0) {
        String distance = walkingPathBean.getContent().getRoute().getPaths().get(0).getDistance();
        String duration = walkingPathBean.getContent().getRoute().getPaths().get(0).getDuration();
        List<LatLng> latLngs = new ArrayList<>();
        int steps = walkingPathBean.getContent().getRoute().getPaths().get(0).getSteps().size();
        latLngs.add(orignLat);
        for (int i = 0; i < steps; i++) {
            String polylines = walkingPathBean.getContent().getRoute().getPaths().get(0).getSteps().get(i).getPolyline();
            String[] latlngArr = polylines.split(";");
            if (latlngArr.length > 0) {
                for (String lanLon : latlngArr) {
                    String[] latlng_str = lanLon.split(",");
                    if (latlng_str.length > 1) {
                        LatLng mLatLng = new LatLng(Double.valueOf(latlng_str[1]), Double.valueOf(latlng_str[0]));
                        latLngs.add(mLatLng);
                    }
                }
            }
        }
        latLngs.add(destLat);
}
  1. 然后就可以绘制了
 polyline = aMap.addPolyline(new PolylineOptions().
                        addAll(latLngs).width(14f).color(Color.parseColor("#CEE00E")));
zoomToSpan(latLngs);
//根据屏幕的位置对路线进行适应屏幕的缩放
    private void zoomToSpan(List<LatLng> latLngs) {
        LatLngBounds.Builder b = LatLngBounds.builder();
        for (LatLng latLng : latLngs) {
            b.include(latLng);
        }
        LatLngBounds bounds = b.build();
        int top_padding = Utils.dip2px(mContext, 96 + 70);
        int bottom_padding = Utils.dip2px(mContext, 40 + 20 + 10);
        int left_right_padding = Utils.dip2px(mContext, 15);
        aMap.moveCamera(CameraUpdateFactory
                .newLatLngBoundsRect(bounds, left_right_padding, left_right_padding, top_padding, bottom_padding));
    }
  1. 清除再绘制
 if (polyline != null) {
       polyline.remove();
       polyline = null;
 }

这样我们就轻松的完成了路线规划的绘制和清除,而不用考虑地图上其他的元素。
---------------------------------------2017.12.18更新------------------------------------

fragment显示地图 没有完全释放

fragment中第一次显示地图没问题,但是在退出程序重新打开地图会发现地图定位失败,并且定位图层和缩放图层都不见了。一开始想到的是地图没有释放干净,然后就把所有的地图图层全部remove,并且setXXX=null,然后并没有什么效果。后来把fragment的初始化方式更改为new Instance()的方式就解决了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • 背景 一年多以前我在知乎上答了有关LeetCode的问题, 分享了一些自己做题目的经验。 张土汪:刷leetcod...
    土汪阅读 12,741评论 0 33
  • 【 申请密钥AndroidSDK:应用程序包名+数字签名 As查看数字签名:cmd-----cd .android...
    征程_Journey阅读 2,272评论 0 5
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,605评论 18 399
  • 不知不觉,已经工作两年多的时间了,有太多的成长,有太多变化,认识了很多人,也见识了很多人性的本质…… 依稀记得20...
    宇智波圣烨阅读 318评论 0 0