ArcGIS轨迹回放

2017-8-20

前言

又到了周六,打完球,吃完饭,闲来无事,便把这周干的事情总结一下,顺便写个博客,分享给大家。

本来说好这周研究三维GIS的,但是看完官方文档发现只有最新的100版本推出了三维GIS,在加上小组长临时让我在2DGIS上研究一下轨迹回放,经过多次改动,最终在周四完成了这个功能,然后周五摸了一天的鱼(心里绞痛)。

整个功能通过你给的点的集合(小组长要求每个点相隔十分钟,当然,你们可以随便间隔),在地图上一步一步通过动画绘制出路线,每次到达一个点时,会显示出在这个点的时间,如果前后2个点没有太大的变化(也就是原地不动),每次会暂停一秒,然后继续绘制。

一些ARCGIS基础的开发,我就不说了,源码我也上传到GItHub上了,有兴趣的童鞋可以去看看。

好了,闲话就扯到这,开始吧。

效果实现

源码在这里:
https://github.com/Hyyzt/DrawTrack/tree/master

让我们先看看效果图

从动图中我们可以看到3个效果,下面我们依次分析下怎么实现。

  • 首先轨迹回放的动画,是通过handler发送消息绘制Polyline来实现的,将所有的点分段绘制,正在走的为一种颜色,走完的为另一种颜色,没2个点为一段动画,通过你要求的所有的动画时间,计算每一段动画的绘制速度(100毫秒),每当发送一个消息(每100毫秒发送一次),便增加polyline.lineto()的坐标,以实现线移动的效果。

  • 每一段动画中一共有2个polyline,第一个是正在走的polyline(临时线段),没100毫秒画一次(根据计算出的速度),当每一段动画结束时,移除第一个polyline,绘制第二个动画结束的polyline(最终线段),这样就实现了我们轨迹回访时得绘制动画。还有一个就是移动的图标,我们通过PictureMarkerSymbol来构建自己的移动图标,同样通过handler发送消息,根据速度增加坐标,每发送一次,移除上一个图片,重新绘制,实现移动的效果。

  • 显示这个点的时间,无非就是判断一下这个点的经纬度时候是否和增加后的经纬度一致,之后再弹出一个显示框就可以了。

以上就是我们实现的思路了,下面我们来看看代码和难点。

代码

在代码之前,要说几个MapView的绘制时的特性。

  • 我们绘制时,无论是线路还是图标,我们都是通过底图的投影坐标绘制的,所以我们需要将经纬度坐标转换为底图的投影坐标,否则无论你的点在哪里,都只会绘制在一个地方。
    要将WGS-84转换为投影坐标,我们可以通过
    Point mapPoint = (Point) GeometryEngine.project(wgsPoint ,SpatialReference.create(4326),map.getSpatialReference());

但有时候(没错,我就是那个但是),mapview是拿不到SpatialReference的,所以我们需要通过查阅自己的SpatialReference来转换投影坐标。

Point mapPoint = (Point) GeometryEngine.project(wgsPoint ,SpatialReference.create(4326),SpatialReference.create(你的投影坐标系参数));

每次的绘制时的点都必须转换,否则会出现上面的错误。

我们传入的数据格式是一个Point(ArcGIS)和一个表示时间的字符串的集合,通过下列方式构造,构造完成后加入集合。

new Data("2017-08-17 13:56:00", new Point(116.37489, 40.06644));
  • 线和图片的移动时的动画

    之前说过无论是线还是图片的动画,都是通过handler每100毫秒发送一个消息,然后动过重绘来实现的动画,而这2个动画是同时完成的,所以使用一个handler来实现就可以了。

     Handler handler = new Handler() {
    @Override
    public void handleMessage(Message msg) {
        time++;//每一段动画绘制了多少秒
        draw();
    }
    };      

    private void draw() {
    if (!isPause) {//是否暂停
        if (index >= pointList.size() - 1)
            return;
        if (Math.abs(pointList.get(index).getX() - pointList.get(index + 1).getX()) < precision
                && Math.abs(pointList.get(index).getY() - pointList.get(index + 1).getY()) < precision) {
            //前一个点和后一个点没有变化,原地跳动
            index++;
            myCallOut.show(wgs2(pointList.get(index)), list.get(index).data);
            time = 0;
            ClickList.add(pointList.get(index));
            drawLines(index);
            handler.sendEmptyMessageDelayed(0, 1000);
        } else {
            if (Math.abs(point.getX() - pointList.get(index + 1).getX()) < 0.000001
                    && Math.abs(point.getY() - pointList.get(index + 1).getY()) < 0.000001) {
                //上一段动画完成
                showResult(pointList.get(index), pointList.get(index + 1));
                index++;
                myCallOut.show(wgs2(pointList.get(index)), list.get(index).data);
                time = 0;
                ClickList.add(pointList.get(index));
                setOnClickListener();//添加点击事件的监听
                drawPoint(index);//画点
                if (index == pointList.size() - 1) {
                    //路程结束
                    Toast.makeText(mainActivity, "路程结束", Toast.LENGTH_SHORT).show();
                    onDraw.onFinish();//提供给外部的接口
                    return;
                }
            }
            drawLines(index);
            handler.sendEmptyMessageDelayed(0, 100);
        }
    }
    }
  • 在每一段动画结束后,移除之前的线段

      //画动画的线,在每一段完成后,移除
      public void drawLines(int index) {
      carLayer.removeAll();
      //计算每100毫秒的速度
      speedX = (pointList.get(index + 1).getX() - pointList.get(index).getX()) / a;
      speedY = (pointList.get(index + 1).getY() - pointList.get(index).getY()) / a;
      Log.e("TAG", speedX + "," + speedY);
      SimpleLineSymbol lineSymbol = new SimpleLineSymbol(color1, 5, SimpleLineSymbol.STYLE.SOLID);
      point = new Point(pointList.get(index).getX(), pointList.get(index).getY());
      Log.e("TAG", "drawLines: " + point.toString());
      point.setX(point.getX() + speedX * time);
      point.setY(point.getY() + speedY * time);
      if (isFellow) {//是否开启跟随模式
          mMapView.centerAt(wgs2(point), true);
      }
      carGraphic = new Graphic(wgs2(point), car);
      Polyline polyline = new Polyline();
      polyline.startPath(wgs2(pointList.get(index)));
      Log.e("TAG1", pointList.get(index).toString());
      polyline.lineTo(wgs2(point));
      Log.e("Po", "drawLines: " + point.toString());
      Graphic graphic = new Graphic(polyline, lineSymbol);
      drawLayer.addGraphic(graphic);
      carLayer.addGraphic(carGraphic);
    

    }

  • 展示最终的结果

      private void showResult(Point start, Point end) {
      drawLayer.removeAll();//移除之前的临时线段
      SimpleLineSymbol lineSymbol = new SimpleLineSymbol(color2, 5, SimpleLineSymbol.STYLE.SOLID);
      Polyline polyline = new Polyline();
      polyline.startPath(wgs2(start));
      polyline.lineTo(wgs2(end));
      Graphic graphic = new Graphic(polyline, lineSymbol);
      resultLayer.addGraphic(graphic);
      }
    
  • 设置点击事件

      public void setOnClickListener() {
      mMapView.setOnSingleTapListener(new OnSingleTapListener() {
          @Override
          public void onSingleTap(float v, float v1) {
              Point clickPoint = mMapView.toMapPoint(v, v1);
              Point wgsPoint = (Point) GeometryEngine.project(clickPoint, spatialReference, SpatialReference.create(4326));
              for (int i = 0; i < ClickList.size(); i++) {
                  Point p = ClickList.get(i);
                  if (Math.abs(p.getX() - wgsPoint.getX()) < 0.001 && Math.abs(p.getY() - wgsPoint.getY()) < 0.001) {
                      String timeInterval = getTimeInterval(i);//计算前后2个点不动时的时间间隔
                      myCallOut.show(wgs2(ClickList.get(i)), timeInterval);
                      break;
                  }
              }
          }
      });
      }
    
      //计算时间间隔
       private String getTimeInterval(int index) {
      //通过栈来获取时间间隔,每次判断时,当前点先入栈,若后一个点不相同,则清空栈,若相同,则将后一个点入栈,随后继续判断,当有一个点与前一个点不同时,拿到栈顶,清空整个栈,这样就拿到了整个时间间隔。
      String startTime = list.get(index).data;
      String endTime;
      //开始遍历
      Stack<Data> stack = new Stack<>();
      stack.add(list.get(index));
      for (int i = index; i < list.size() - 1; i++) {
          if (Math.abs(pointList.get(i).getX() - pointList.get(i + 1).getX()) < 0.0001
                  && Math.abs(pointList.get(i).getY() - pointList.get(i + 1).getY()) < 0.0001) {
              //前后两个点相同
              stack.add(list.get(i + 1));
          } else {
              //前后2个点不同
              break;
          }
      }
      if (stack.size() > 1) {
          endTime = stack.pop().data;
          stack.removeAllElements();
          return startTime + "---" + endTime;
      } else {
          return startTime;
      }
       }
    

这样和逻辑相关的东西就结束了,至于弹出时间的文本框,过于简单也就不多说了,大家可以去源码中查看。

使用

根据小组长的要求,要我将整个过程集成,只暴露出几个方法供他使用(!!!),于是变成了最终版本,只需要拷贝2个类到你的项目,在初始化一下然后start就可以使用了。2个类是:

  1. MoveHelper,处理整个逻辑和动画的绘制
  2. MyCallOut,弹出的文本框

当你将2个类拷贝后,你直接进行初始化

    //mapView一定要初始化结束在传入进去
    //初始化MoveHelper时的参数 Context,MapView,List<Bean>
    moveHelper = new MoveHelper(this, mMapView, list);
    //移动时的图标(***必须设置),正在移动时轨迹的颜色,移动结束后轨迹的颜色
    moveHelper.setIconAndColor(R.drawable.point, Color.YELLOW, Color.GRAY);
    //判断是否移动的精度,默认0.0001
    moveHelper.setPrecision(0.0001);
    //是否跟随,默认false
    moveHelper.setFellow(false);
    //设置整个动画的时间,***必须设置
    moveHelper.setTime(40);
    //设置底图的空间参考,默认为3857,若参数和底图不一致,绘制会出现偏差
    moveHelper.setSpatialReference(SpatialReference.create(3857));
    //在整个动画结束后的逻辑,在OnFinish()里完成
    moveHelper.setOnDraw(new MoveHelper.onDraw() {
        @Override
        public void onFinish() {
            
        }
    });
    //注意:开始动画时,你必须要在之前初始化所有的必要参数,如:动画时间,移动时的图标等等。
    moveHelper.start();//开始动画
    moveHelper.pause();//暂停动画

这样就可以开始自己的动画了。

后记

可能讲的不是很清楚,望大家多见谅。(滑稽.jpg)但是功能和源码都已经实现了,可以直接去看源码,注释的也很清楚,也可以直接拿去使用。

到这也就基本结束了,下次见。

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

推荐阅读更多精彩内容

  • 贪心算法 贪心算法总是作出在当前看来最好的选择。也就是说贪心算法并不从整体最优考虑,它所作出的选择只是在某种意义上...
    fredal阅读 9,223评论 3 52
  • 背景 一年多以前我在知乎上答了有关LeetCode的问题, 分享了一些自己做题目的经验。 张土汪:刷leetcod...
    土汪阅读 12,740评论 0 33
  • 我的唯一的宝贝儿子是我的骄傲!不是因为他学习好,而是因为他做人好! 无论去哪里他都记得告诉我,不让我为他担心。 出...
    真诚永恒阅读 373评论 3 2
  • 很久没有一个人胡思乱想成这样了,因为一个人不敢说让人误会的话,因为别人感到伤心,内心那么混沌,为什么不能过的乐观点呢?
    另一个自己WYH阅读 243评论 0 0
  • 1.他看中一台电脑,需要9000元。他每月的收入只有2000。老婆对他说,你疯了,你买了就离婚。他问我怎么办。 我...
    笔行天下阅读 1,078评论 0 0