c# wpf使用GMap.NET类库,实现地图轨迹回放

前言

实现轨迹回放,GMap.NET有对应的类GMapRoute。这个类函数很少,功能有限,只能实现简单的轨迹回放。要实现更复杂的轨迹回放,就需要自己动手了。

本文介绍一种方法,可以实现复杂的轨迹回放。有句话“功夫在诗外”,GMap.NET给你提供了基本地图处理功能;但是不要让CMap.NET束缚了手脚。你需要有深刻理解地图实现原理,深入理解WPF动画的原理,才能到达随心所欲。最终的效果如下:

GMap.NET 显示原理

地图就是由许多方格“瓦片”组合而来。当你移动或缩放时,GMap.NET会根据当前位置、显示窗口、缩放级别,到地图服务器获取图片。所以地图控件本质上就是显示图片的控件,只是这些图片包含了坐标信息。

地图上加轨迹,就是在图片上画线。这些线要与gps坐标点吻合。通过GMapMarker不仅可以加标注,也可以实现轨迹。需要将gps坐标点转换成控件的坐标点,再连成线就可以了。本文就是通过GMapMarker实现了轨迹回放。

1  实现轨迹显示

通过自定义控件UserControlMapRoute实现了轨迹显示功能。需要将此控件加入到GMapMarker。

GMapMarker _routeMaker = new GMapMarker(point);
 UserControlMapRoute routeCtrl = new UserControlMapRoute() { Marker = _routeMaker, MapCtrl = MainMap };
 routeCtrl.Init();
 _routeMaker.Shape = routeCtrl;
 //将图层添加到地图
 this.MainMap.Markers.Add(_routeMaker);

UserControlMapRoute有两个功能:显示轨迹起始点,显示轨迹。将轨迹显示功能放在类MapRoutePath中实现。该类实现的功能就是根据gps坐标显示轨迹。该类包含的变量有:

class MapRoutePath
 {
 public GMapControl MapCtrl { get; private set; } //地图控件
 public Panel ParentPanel { get; private set; } //父面板,将PathRouteLine加入面板。

 public Path PathRouteLine { get; private set; } //显示轨迹

 List<PointLatLng> _listGpsPoint = new List<PointLatLng>();
 List<Point> _listCtrlPt = new List<Point>();

 public List<PointLatLng> ListGpsPoint => _listGpsPoint; //包含的gps坐标
 public List<Point> ListPathPoint => _listCtrlPt; //转换成立控件坐标
}

实现轨迹功能是变量PathRouteLine,该变量的父控件是ParentPanel(就是控件UserControlMapRoute 中的根Grid控件)。MapCtrl 控件主要作用就是提供了将gps坐标转换成控件坐标的函数。

实现将gps做标注转换成控件坐标的方法:

private void ToLocalPoint()
 {
  //_listGpsPoint存储所gps坐标 _listCtrlPt存储转换后控件坐标
  _listCtrlPt.Clear();
  foreach (PointLatLng pt in _listGpsPoint)
  {
  Point ptGrid = ToCtrlPoint(pt);
  _listCtrlPt.Add(ptGrid);
  }
 }

 Point ToCtrlPoint(PointLatLng gpsPoint)
 {
  //转换成GMap.NET控件坐标
  GPoint ptOfMapCtrl = MapCtrl.FromLatLngToLocal(gpsPoint);

  //GMap.NET控件坐标要转换成 控件相对于直接父面板的坐标
  Point ptToMapCtrl2 = new Point(ptOfMapCtrl.X, ptOfMapCtrl.Y);
  Point ptOfScreen = MapCtrl.PointToScreen(ptToMapCtrl2);
  Point ptOfParentPanel = ParentPanel.PointFromScreen(ptOfScreen);

  return ptOfParentPanel;
 }

坐标转换过程就是: GPS坐标 --》 GMap.NET控件坐标 --》 屏幕坐标 --》 控件相对于直接父面板的坐标。获取了控件坐标,就根据这些坐标画直线就行了。

private static void CreatPath(Path path, List<Point> listPt)
 {
  if (listPt.Count <= 1)
  {
  path.Data = null;
  return;
  }

  PathFigure pathFigure = new PathFigure();
  pathFigure.StartPoint = listPt[0]; //起始点

  for (int i = 1; i < listPt.Count; i++)
  {
  //加入线段
  LineSegment line = new LineSegment() { Point = listPt[i] };
  pathFigure.Segments.Add(line);
  }

  PathGeometry geometry = new PathGeometry();
  geometry.Figures.Add(pathFigure);
  path.Data = geometry;
 }

2 实现轨迹回放.

要实现两个功能:通过不同的线颜色来指示当前行动轨迹;提示当前所在的位置、用时等信息的tip框。

轨迹移动 假如显示轨迹的线颜色为红色,通过绿色来显示当前经过的位置。再增加一个变量_pathMoveRouteLine(类型也为Path) 就可以了。_pathMoveRouteLine的颜色为绿色,所显示的路径要和PathRouteLine 路径完全相同。_pathMoveRouteLine路径长度要实时计算出来,随着时间推移,路径不断变长。需要增加一个定时器,不停的计算当前所在的位置。在定时器中,调用函数ShowRouteMove();

private void ShowRouteMove()
 {
  //线路总长度
  double totalDistance = GetDistance();
  if (totalDistance == 0)
  return;

  //更加时间,计算当前走过的长度
  TimeSpan span = DateTime.Now - _startMoveTime;
  double curDistance = _moveSpeed * span.TotalHours;
  if (curDistance > totalDistance)
  {
  StopMove();
  curDistance = totalDistance;
  }

  Path path = CreateMovePath();
  //根据已走过的距离,获取需要显示的点
  List<Point> listCtrlPoint = GetListByDistance(curDistance);
  CreatPath(path, listCtrlPoint);

  ShowMoveTip(listCtrlPoint.Last(), curDistance);
 }
//根据当前移动的距离,获取相应的控件坐标
 private List<Point> GetListByDistance(double distance)
 {
  List<Point> result = new List<Point>();

  double start = 0;
  int i = 0;
  PointLatLng lastPt = new PointLatLng();
  foreach (PointLatLng pt in ListGpsPoint)
  {
  i++;
  if (i == 1) //第一个点
  {
   result.Add(ToCtrlPoint(pt));
   lastPt = pt;
   continue;
  }
  else
  {
   double lineDistance = MapHelper.GetDistance(lastPt, pt);
   lastPt = pt;
   if (lineDistance == 0)
   continue;

   if ((start + lineDistance) == distance) //gps坐标恰好符合当前的距离
   {
   result.Add(ToCtrlPoint(pt));
   break;
   }
   else if ((start + lineDistance) < distance) //当前的点小于需要的距离
   {
   result.Add(ToCtrlPoint(pt));
   start += lineDistance;
   }
   else
   {
   //最终的点 落在两个gps点之间,需要进一步计算
   double midDistance = distance - start;
   double rate = midDistance / lineDistance;

   Point endPoint = ToCtrlPoint(pt);
   Point midPoint = MapHelper.GetMidPoint(result.Last(), endPoint, rate);
   result.Add(midPoint);
   break;
   }
  }
  }
  return result;
 }
class MapHelper
 {
 //根据两点坐标,和在这两点之间的比例,获取计算后的坐标
 internal static Point GetMidPoint(Point start, Point end, double rate)
 {
  Point result = new Point();
  result.X = start.X + rate * (end.X - start.X);
  result.Y = start.Y + rate * (end.Y - start.Y);
  return result;
 }
 }

提示框显示

提示框所在的位置就是移动轨迹的最后一个点的位置。为了更好的显示效果,对这个坐标做一定的偏移:

UserControlMoveTip _userControlMoveTip;
 private void ShowMoveTip(Point startPoint,double moveDistance)
 {
  if(_userControlMoveTip == null)
  {
  _userControlMoveTip = new UserControlMoveTip();
  _userControlMoveTip.HorizontalAlignment = HorizontalAlignment.Left;
  _userControlMoveTip.VerticalAlignment = VerticalAlignment.Top;
  ParentPanel.Children.Add(_userControlMoveTip);
  }

  if (_userControlMoveTip.ActualHeight == double.NaN)
  {
  _userControlMoveTip.Visibility = Visibility.Hidden;
  return;
  }

  _userControlMoveTip.Visibility = Visibility.Visible;
  _userControlMoveTip.TotalDistance = GetDistance();
  _userControlMoveTip.TotalTimeSpan = TimeSpan.FromHours(_userControlMoveTip.TotalDistance / _moveSpeed);
  _userControlMoveTip.MoveSpeed = _moveSpeed;
  _userControlMoveTip.MoveDistance = moveDistance;
  _userControlMoveTip.TimeElapse = (DateTime.Now - _startMoveTime);

  _userControlMoveTip.Margin = new Thickness(startPoint.X+5, startPoint.Y - _userControlMoveTip.ActualHeight-2, 0, 0);
 }

_userControlMoveTip是用户控件,用来显示总距离、已移动距离、移动时间等信息。

后记:

有些开发者反映GMap.NET控件的WPF版提供的功能不够完善,有些功能不能采用拿来主义的方式。诚然,软件开发越来越复杂,借鉴别人的代码是必须的,但是不能丢弃软件开发的一些“基本功”。WPF确实不太好学,好多新的概念难以理解。好多开发者学习WPF浅尝辄止,所以在使用一些控件时,感到茫然。

以上就是c# wpf使用GMap.NET类库,实现地图轨迹回放的详细内容,更多关于c# wpf GMap.NET类库的资料请关注我们其它相关文章!

(0)

相关推荐

  • C# WPF使用AForge类库操作USB摄像头拍照并保存

    项目中用到 USB 摄像头,需要根据情况进行图像抓拍,查了半天资料,比较多的是使用 WPFMediaKit 和 AForge . 但是由于项目要求不显示 USB 摄像头拍摄的画面,最终确定使用 AForge 解决. 下面用一个测试程序记录一下. 一.无预览拍照 首先建立一个 WPF 项目,我的就叫 AForgeTest,你们随意就好: 然后在 NuGet 包管理器中安装 AForge 库: 我只安装了图中打勾的几个库,这个根据自己项目需要安装就好. 不过用 USB 摄像头拍照必须安装: AFor

  • c# WPF实现Windows资源管理器(附源码)

      今天我来写一篇关于利用WPF来实现Windows的资源管理器功能,当然只是局部实现这个功能,因为在很多时候我们需要来实现对本机资源的管理,当然我们可以使用OpenFileDialog dialog = new OpenFileDialog()这个Microsoft.Win32命名空间下的这个类来实现一些资源查找和导入的功能,但是在很多时候我们可能需要更多的功能,并且希望能够集成到我们自己的项目中,但是我们这个时候就不得不自己来写一套来集成到我们的软件中去了,因为OpenFileDialog这

  • c# WPF设置软件界面背景为MediaElement并播放视频

    在我们的常见的软件界面设计中我们经常会设置软件的背景为SolidColorBrush或者LinerColorBrush.RadialGradientBrush 等一系列的颜色画刷为背景,有时我们也会使用ImageBrush添加图片来作为界面的背景,另外常用的还有DrawingBrush以及今天需要进行总结的VisualBrush,这些我们都是比较容易实现的,那么我们如果想将软件的界面设计成一个动画或者干脆播放一段视频作为背景,这个对于整个软件的效果又是一个巨大的提升. 首先我们来看看backgr

  • C# WPF 自定义按钮的方法

    本文介绍WPF一种自定义按钮的方法. 实现效果 使用图片做按钮背景: 自定义鼠标进入时效果: 自定义按压效果: 自定义禁用效果 实现效果如下图所示: 实现步骤 创建CustomButton.cs,继承自Button: 创建一个资源文件ButtonStyles.xaml: 在资源文件中设计按钮的Style: 在CustomButton.cs中添加Style中需要的依赖属性: 在程序中添加资源并引用(为了方便在不同的程序中引用自定义按钮,自定义按钮放在独立的类库中,应用程序中进行资源合并即可). 示

  • C# WPF Image控件的绑定方法

    在我们平时的开发中会经常用到Image控件,通过设置Image控件的Source属性,我们可以加载图片,设置Image的source属性时可以使用相对路径也可以使用绝对路径,一般情况下建议使用绝对路径,类似于下面的形式Source="/Demo;Component/Images/Test.jpg"其中Demo表示工程的名称,后面表示具体哪个文件夹下面的哪个图片资源,在程序中,我们甚至可以为Image控件设置X:Name属性,在后台代码中动态去改变Image的Source,但我个人认为这

  • C#中WPF依赖属性的正确学习方法

    前言 我在学习WPF的早期,对依赖属性理解一直都非常的不到位,其恶果就是,我每次在写依赖属性的时候,需要翻过去的代码来复制黏贴. 相信很多朋友有着和我相同的经历,所以这篇文章希望能帮助到那些刚刚开始学依赖属性的朋友. 那些[讨厌]的依赖属性的讲解文章 初学者肯定会面临一件事,就是百度,谷歌,或者MSDN来查看依赖属性的定义和使用,而这些文章虽然都写的很好,但,那是相对于已经学会使用依赖属性的朋友而言. 而对于初学者而言,说是误导都不过分. 比如,官网的这篇文章https://docs.micro

  • c# WPF中自定义加载时实现带动画效果的Form和FormItem

    背景 今天我们来谈一下我们自定义的一组WPF控件Form和FormItem,然后看一下如何自定义一组完整地组合WPF控件,在我们很多界面显示的时候我们需要同时显示文本.图片并且我们需要将这些按照特定的顺序整齐的排列在一起,这样的操作当然通过定义Grid和StackPanel然后组合在一起当然也是可以的,我们的这一组控件就是将这个过程组合到一个Form和FormItem中间去,从而达到这样的效果,我们首先来看看这组控件实现的效果. 一 动画效果 看了这个效果之后我们来看看怎么来使用Form和For

  • c# WPF中通过双击编辑DataGrid中Cell的示例(附源码)

    背景 在很多的时候我们需要编辑DataGrid中每一个Cell,编辑后保存数据,原生的WPF中的DataGrid并没有提供这样的功能,今天通过一个具体的例子来实现这一个功能,在这个例子中DataGrid中的数据类型可能是多种多样的,有枚举.浮点类型.布尔类型.DateTime类型,每一种不同的类型需要双击以后呈现不同的效果,本文通过使用Xceed.Wpf.DataGrid这个动态控件库来实现这个功能,当前使用的Dll版本是2.5.0.0,不同的版本可能实现上面有差别,这个在使用的时候需要特别注意

  • c# 基于wpf,开发OFD电子文档阅读器

    前言 OFD是国家标准版式文档格式,于2016年生效.OFD文档国家标准参见<电子文件存储与交换格式版式文档>.既然是国家标准,OFD随后肯定会首先在政务系统使用,并逐步推向社会各个方面.OFD是在研究当下各类文件格式后,推出的标准,有如下优点: 1 产权属于自主产权 2 具有便携性:文件小,可压缩比率大.测试显示生成的文件体量比PDF还要小. 3 具有开放性:易于入门,对于使用者来说更具开放性. 4 具有扩展性:预留了可扩展入口和自定义标引,设置了非接触式引用机制,为特性化提供支持. 5 呈

  • 在C# WPF下自定义滚动条ScrollViewer样式的操作

    一.实现对ScrollViewer样式的自定义主要包括: 1.滚动条宽度设置 2.滚动条颜色 3.滚动条圆角 4.滚动条拉动时的效果mouseover 二.实现效果: 三.实现方法 1.创建资源字典( ResourceDictionary)文件 由于style代码比较多,之间在控件文件中加载style比较混乱,也不利于其它窗口复用,这里单独创建了ScrollViewDictionary.xaml文件代码如下: <ResourceDictionary xmlns="http://schema

  • c# WPF中CheckBox样式的使用总结

    背景 很多时候我们使用WPF开发界面的时候经常会用到各种空间,很多时候我们需要去自定义控件的样式来替换默认的样式,今天通过两个方法来替换WPF中的CheckBox样式,透过这两个例子我们可以掌握基本的WPF样式的开发如何定义ControlTemplate以及使用附加属性来为我们的控件增加新的样式. 常规使用 我们在使用CheckBox的时候,原始的样式有时不能满足我们的需求,这是我们就需要更改其模板,比如我们常用的一种,在播放器中"播放"."暂停"按钮,其实这也是一

  • C# WPF上位机实现和下位机TCP通讯的方法

    下位机使用北京大华程控电源DH1766-1,上位机使用WPF.实现了电压电流实时采集,曲线显示.上午在公司调试成功,手头没有程控电源,使用TCP服务端模拟.昨天写的TCP服务端正好排上用场. 界面如下: 服务端 服务端实在上篇基础上实现的.需要做如下更改: while (true) { try { byte[] bufferDate = new byte[1024]; int realLen = pSocket.Receive(bufferDate); if (realLen <= 0) { t

随机推荐