WPF简易局部地图 实现标记 绘制轨迹

控件可以实现:
在知道显示的地图区域时:

  1. 设定地图的左上角和右下角经纬度;
  2. 设置使用的地图图片路径;
  3. 添加地图图钉或者轨迹点;
  4. 绘制点或轨迹。

地图控件所需的数据结构:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Media;namespace SmartScreen.Controls.Models
{public class MapItem{public Brush Fill {  get; protected set; }}public class GeoPint{public GeoPint(double lat, double lon){Latitude = lat;Longitude = lon;}public double Latitude { get; set; }public double Longitude { get; set; }}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace SmartScreen.Controls.Models
{public class MapPolyLineItem : MapItem{public double Longitude { get; set; }public double Latitude { get; set; }}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Media;namespace SmartScreen.Controls.Models
{public class MapPushPinItem : MapItem{public string Title { get; set; }public double Longitude { get; set; }public double Latitude { get; set; }private int warnningStatus;/// <summary>/// 警示状态,0:安全 1:预警 -1:警告/// </summary>public int WarnningStatus{get { return warnningStatus; }set{warnningStatus = value;if (warnningStatus == 0){Fill = (SolidColorBrush)Application.Current.FindResource("SafeColor");}else if (warnningStatus == 1){Fill = (SolidColorBrush)Application.Current.FindResource("WarnningColor");}else if (warnningStatus == -1){Fill = (SolidColorBrush)Application.Current.FindResource("DangerColor");}}}}
}

地图控件XAML(可以去掉zoomPanel:ZoomAndPanControl):

<UserControl x:Class="SmartScreen.Controls.MapControl"xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" xmlns:d="http://schemas.microsoft.com/expression/blend/2008" xmlns:local="clr-namespace:SmartScreen.Controls" xmlns:zoomPanel="clr-namespace:ZoomAndPan;assembly=ZoomAndPan"mc:Ignorable="d" x:Name="root"d:DesignHeight="450" d:DesignWidth="800"><UserControl.Resources><Style TargetType="Line"><Setter Property="Stroke" Value="{StaticResource DangerColor}"/><Setter Property="StrokeThickness" Value="2"/><Setter Property="Effect"><Setter.Value><DropShadowEffect Color="Gray" BlurRadius="8" ShadowDepth="0"/></Setter.Value></Setter></Style></UserControl.Resources><Grid><zoomPanel:ZoomAndPanControl x:Name="zoomPanel"  MouseWheel="map_MouseWheel"MaxContentScale="2.5"MinContentScale="1"MouseLeftButtonDown="Grid_MouseLeftButtonDown" MouseLeftButtonUp="Grid_MouseLeftButtonUp" MouseMove="Grid_MouseMove"><zoomPanel:ZoomAndPanControl.RenderTransform><TransformGroup><TranslateTransform x:Name="trans"/><ScaleTransform x:Name="scale"/></TransformGroup></zoomPanel:ZoomAndPanControl.RenderTransform><Grid><Image x:Name="bottomImage" Width="{Binding ElementName=root,Path=ActualWidth}" Height="{Binding ElementName=root,Path=ActualHeight}" Stretch="Fill" RenderOptions.BitmapScalingMode="Fant"/><!--标记层 ../Images/maptest.png--><Canvas x:Name="PushPincanvas"></Canvas><Canvas x:Name="PolyLinecanvas"></Canvas></Grid></zoomPanel:ZoomAndPanControl></Grid>
</UserControl>

地图控件后台代码:

using SmartScreen.Controls.Models;
using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;namespace SmartScreen.Controls
{/// <summary>/// MapControl.xaml 的交互逻辑/// </summary>public partial class MapControl : UserControl{public MapControl(){InitializeComponent();SizeChanged += (s, e) => { RefreshPushPin(); RefreshPolyline(); };}/// <summary>/// 地图图片的路径/// </summary>public Uri MapPath{get { return (Uri)GetValue(MapPathProperty); }set { SetValue(MapPathProperty, value); }}public static readonly DependencyProperty MapPathProperty =DependencyProperty.Register("MapPath", typeof(Uri), typeof(MapControl), new PropertyMetadata(default(Uri), OnMapPathChanged));private static void OnMapPathChanged(DependencyObject d, DependencyPropertyChangedEventArgs e){MapControl ctrl = (MapControl)d;ctrl.bottomImage.Source = new BitmapImage(ctrl.MapPath);}/// <summary>/// 左上角经纬度/// </summary>public Point LeftTop{get { return (Point)GetValue(LeftTopProperty); }set { SetValue(LeftTopProperty, value); }}public static readonly DependencyProperty LeftTopProperty =DependencyProperty.Register("LeftTop", typeof(Point), typeof(MapControl), new PropertyMetadata(default(Point), OnMapReDrawChanged));/// <summary>/// 右下角经纬度/// </summary>public Point RightBottom{get { return (Point)GetValue(RightBottomProperty); }set { SetValue(RightBottomProperty, value); }}public static readonly DependencyProperty RightBottomProperty =DependencyProperty.Register("RightBottom", typeof(Point), typeof(MapControl), new PropertyMetadata(default(Point), OnMapReDrawChanged));/// <summary>/// 地图显示的图钉部件/// </summary>public ObservableCollection<MapPushPinItem> PushPins{get { return (ObservableCollection<MapPushPinItem>)GetValue(PushPinsProperty); }set { SetValue(PushPinsProperty, value); }}public static readonly DependencyProperty PushPinsProperty =DependencyProperty.Register("PushPins", typeof(ObservableCollection<MapPushPinItem>), typeof(MapControl), new PropertyMetadata(null, OnPushpinChanged));private static void OnPushpinChanged(DependencyObject d, DependencyPropertyChangedEventArgs e){if (d is MapControl map){if (map.PushPins != null){map.PushPins.CollectionChanged += (s, er) => map.RefreshPushPin();}map.RefreshPushPin();}}/// <summary>/// 地图显示的轨迹线段/// </summary>public ObservableCollection<MapPolyLineItem> PolyLines{get { return (ObservableCollection<MapPolyLineItem>)GetValue(PolyLinesProperty); }set { SetValue(PolyLinesProperty, value); }}public static readonly DependencyProperty PolyLinesProperty =DependencyProperty.Register("PolyLines", typeof(ObservableCollection<MapPolyLineItem>), typeof(MapControl), new PropertyMetadata(null, OnPolyLineChaned));private static void OnPolyLineChaned(DependencyObject d, DependencyPropertyChangedEventArgs e){if (d is MapControl map){if (map.PolyLines != null){map.PolyLines.CollectionChanged += (s, er) => map.RefreshPolyline();}map.RefreshPolyline();}}private static void OnMapReDrawChanged(DependencyObject d, DependencyPropertyChangedEventArgs e){MapControl control = (MapControl)d;control.RefreshPushPin();control.RefreshPolyline();}/// <summary>/// 刷新标记点的显示/// </summary>private void RefreshPushPin(){if (RightBottom == default(Point) || LeftTop == default(Point)){return;}var mapWidth = ActualWidth;var mapHeight = ActualHeight;var xRange = RightBottom.X - LeftTop.X;var yRange = LeftTop.Y - RightBottom.Y;if (PushPins != null){PushPincanvas.Children.Clear();foreach (var pushpin in PushPins){// 先计算在X Y 轴上的位置var canvasLeft = (pushpin.Longitude - LeftTop.X) / xRange * mapWidth;var canvasTop = (LeftTop.Y - pushpin.Latitude) / yRange * mapHeight;var pushpinCtrl = new MapPushPin();pushpinCtrl.DataContext = pushpin;Canvas.SetLeft(pushpinCtrl, canvasLeft);Canvas.SetTop(pushpinCtrl, canvasTop);PushPincanvas.Children.Add(pushpinCtrl);}}}/// <summary>/// 刷新轨迹的显示/// </summary>private void RefreshPolyline(){if (RightBottom == default(Point) || LeftTop == default(Point)){return;}var mapWidth = ActualWidth;var mapHeight = ActualHeight;var xRange = RightBottom.X - LeftTop.X;var yRange = LeftTop.Y - RightBottom.Y;if (PolyLines != null){PolyLinecanvas.Children.Clear();MapPolyLineItem lastPoint = null;foreach (var polyline in PolyLines){if (lastPoint == null){lastPoint = polyline;continue;}Line line = new Line();line.DataContext = polyline;line.X1 = 0;line.Y1 = 0;var canvasLeft = (lastPoint.Longitude - LeftTop.X) / xRange * mapWidth;var canvasTop = (LeftTop.Y - lastPoint.Latitude) / yRange * mapHeight;var point2x = (polyline.Longitude - LeftTop.X) / xRange * mapWidth - canvasLeft;var point2y = (LeftTop.Y - polyline.Latitude) / yRange * mapHeight - canvasTop;line.X2 = point2x;line.Y2 = point2y;Canvas.SetLeft(line, canvasLeft);Canvas.SetTop(line, canvasTop);PolyLinecanvas.Children.Add(line);lastPoint = polyline;}}}#region 缩放移动public FrameworkElement Container{get { return (FrameworkElement)GetValue(ContainerProperty); }set { SetValue(ContainerProperty, value); }}public static readonly DependencyProperty ContainerProperty =DependencyProperty.Register("Container", typeof(FrameworkElement), typeof(MapControl), new PropertyMetadata(null));public bool EnableZoomPan{get { return (bool)GetValue(EnableZoomPanProperty); }set { SetValue(EnableZoomPanProperty, value); }}public static readonly DependencyProperty EnableZoomPanProperty =DependencyProperty.Register("EnableZoomPan", typeof(bool), typeof(MapControl), new PropertyMetadata(false));Point lastPoint = default;private void Grid_MouseLeftButtonDown(object sender, MouseButtonEventArgs e){if (!EnableZoomPan){return;}lastPoint = e.GetPosition(Container);zoomPanel.CaptureMouse();}private void Grid_MouseLeftButtonUp(object sender, MouseButtonEventArgs e){if (!EnableZoomPan){return;}lastPoint = default;zoomPanel.ReleaseMouseCapture();}private void Grid_MouseMove(object sender, MouseEventArgs e){if (!EnableZoomPan){return;}if (e.LeftButton == MouseButtonState.Pressed){var nowPoint = e.GetPosition(Container);if (lastPoint == default){lastPoint = nowPoint;return;}var offsetX = (nowPoint.X - lastPoint.X) / scale.ScaleX;var offsetY = (nowPoint.Y - lastPoint.Y) / scale.ScaleX;lastPoint = nowPoint;trans.X += offsetX;trans.Y += offsetY;}}private void map_MouseWheel(object sender, MouseWheelEventArgs e){if (!EnableZoomPan){return;}var nowMouse = e.GetPosition(this);if (e.Delta > 0){zoomPanel.ZoomIn(nowMouse);}else{zoomPanel.ZoomOut(nowMouse);}if (zoomPanel.ContentScale == 1){trans.X = 0;trans.Y = 0;}}#endregion}
}

WPF简易局部地图 实现标记 绘制轨迹相关推荐

  1. 百度地图根据经纬度绘制轨迹

    首先上效果图 Html源码 <!DOCTYPE html> <html> <head> <meta http-equiv="Content-Type ...

  2. 百度地图API绘制轨迹

    百度地图Javascript API绘制轨迹 (首先贴两个网站: 1.百度地图API官方示例. 2.百度地图Javascript API v3.0参考类. 一.新建Html文档 新建文档可以直接从官方 ...

  3. 高德地图单点、多点标记,轨迹绘制,自定义标记覆盖物,给标记添加点击事件,移除所有标记和轨迹,轨迹回放功能

    文档地址 https://lbs.amap.com/api/javascript-api/guide/abc/prepare 注册 到官网注册一下就可以了https://lbs.amap.com/并创 ...

  4. android百度地图轨迹实现,android 获取GPS经纬度在百度地图上绘制轨迹

    实现将一组GPS模块获取的经纬度数据在百度地图上绘制轨迹 1.将经纬度转换成百度地图坐标 /** * 标准的GPS经纬度坐标直接在地图上绘制会有偏移,这是测绘局和地图商设置的加密,要转换成百度地图坐标 ...

  5. java地图上标记_绘制点标记-在地图上绘制-开发指南-Android 地图SDK | 高德地图API...

    点标记用来在地图上标记任何位置,例如用户位置.车辆位置.店铺位置等一切带有位置属性的事物. 地图 SDK 提供的点标记功能包含两大部分,一部分是点(俗称 Marker).另一部分是浮于点上方的信息窗体 ...

  6. tablueau地图标记圆形_高德地图实现自定义小蓝点 自定义点标记 绘制多边形/圆形区域 根据地图的移动显示或者隐藏自定义点标记的相关实现...

    最近项目中有需要应用到高德地图的模块,在参考别的app地图相关模块实现过程中,自己产生了一些想法.首先说明要实现的功能需求.类似支付宝app内的跑腿功能,在全市的所有商铺,电梯广告等任意地点发布任务, ...

  7. 高德地图自定义点标记大小_高德地图实现自定义小蓝点 自定义点标记 绘制多边形/圆形区域 根据地图的移动显示或者隐藏自定义点标记的相关实现...

    最近项目中有需要应用到高德地图的模块,在参考别的app地图相关模块实现过程中,自己产生了一些想法.首先说明要实现的功能需求.类似支付宝app内的跑腿功能,在全市的所有商铺,电梯广告等任意地点发布任务, ...

  8. 高德地图JavaScript根据后台坐标点绘制轨迹

    项目中用到所以看了一下,可以完善之处请指正 <script type="text/javascript">              //这里可以传入后台的json数据 ...

  9. iOS百度地图SDK之实时绘制轨迹(后台仍执行)

    首先,对于百度地图SDK的配置和环境搭建就不做说明,需要的人可以博客中另一篇文章看 <iOS百度地图SDK基本使用> ,本文的重点在于实现实时绘制轨迹的功能,并且对细节进行处理和优化 1. ...

最新文章

  1. 高手都这么给 Spring MVC 做单元测试!
  2. 动态代理的两种方式_动态代理是基于什么原理?
  3. IE6.0、IE7.0 、FireFox 在样式中的不同写法.doc
  4. 全文索引 - Pomelo.EFCore.MySql
  5. martin fowler_用Java和Java 8创建内部DSL,采用Martin Fowler的方法
  6. Java-分析类初始化
  7. iphone黑屏转圈_iphone8一直在黑屏转圈
  8. 微信封杀lol手游活动小程序?《英雄联盟手游》回应了
  9. mysql 命令大小写_MySQL连接方式及大小写问题
  10. 函数的参数对象$event的使用和利用他找到事件对象
  11. ubuntu 18.04安装微信
  12. 社交网络的数据挖掘与分析,什么是社交网络分析
  13. 一个简单的apk破解
  14. NanDigits GOF ECO 9介绍
  15. 引用 八卦象数疗法--配方1
  16. 摩拜6亿美元融资之后:终极目标是打造全球移动物联网
  17. linux marvell 网卡驱动,如何在linux下安装MarvellYukon网卡驱动?
  18. Excel中换行的问题
  19. 1.国民技术N32G45X例程之-串口打印
  20. 备份android分区,备份分区

热门文章

  1. 8热水器用户行为分析与事件识别
  2. 密钥交换协议之IKEv2
  3. mysql cte 表不存在_MySQL8的新特性CTE
  4. 程序员必备!5款小众高效的开发神器
  5. 《西南联大行思录》之感想20180911
  6. 2023新华为OD机试题 - 数组排序(JavaScript) | 刷完来对接OD招聘渠道
  7. MQSim模拟器论文阅读
  8. 2020年下一个创业风口是什么?
  9. 数据产品经理日常工作
  10. 关于虚拟机无法创建目录的问题