Windows Phone BingMap控件计算路径
一.实现过程
首先需要一个Key,可上官网免费注册
二.添加服务引用
这个是地图服务:
http://dev.virtualearth.net/webservices/v1/geocodeservice/geocodeservice.svc
这个是计算路径服务:
http://dev.virtualearth.net/webservices/v1/routeservice/routeservice.svc
一般只要这两个就可以了。如果需要实现搜索服务,则需要添加SearchService服务。
http://dev.virtualearth.net/webservices/v1/searchservice/searchservice.svc
此时的项目视图如下所示:
三.添加服务之后需要修改配置文件:
ServiceReferences.ClientConfig.将配置文件修改如下所示:
<configuration> <system.serviceModel> <bindings> <basicHttpBinding> <binding name="BasicHttpBinding_IGeocodeService" maxBufferSize="2147483647" maxReceivedMessageSize="2147483647"> <security mode="None"> <transport> <extendedProtectionPolicy policyEnforcement="Never" /> </transport> </security> </binding> <binding name="BasicHttpBinding_IRouteService" maxBufferSize="2147483647" maxReceivedMessageSize="2147483647"> <security mode="None" /> </binding> </basicHttpBinding> <customBinding> <binding name="CustomBinding_IGeocodeService"> <binaryMessageEncoding /> <httpTransport maxReceivedMessageSize="2147483647" maxBufferSize="2147483647" /> </binding> <binding name="CustomBinding_IRouteService"> <binaryMessageEncoding /> <httpTransport maxReceivedMessageSize="2147483647" maxBufferSize="2147483647" /> </binding> </customBinding> </bindings> <client> <endpoint address="http://dev.virtualearth.net/webservices/v1/geocodeservice/GeocodeService.svc" binding="basicHttpBinding" bindingConfiguration="BasicHttpBinding_IGeocodeService" contract="GeocodeService.IGeocodeService" name="BasicHttpBinding_IGeocodeService" /> <endpoint address="http://dev.virtualearth.net/webservices/v1/geocodeservice/GeocodeService.svc" binding="basicHttpBinding" bindingConfiguration="BasicHttpBinding_IGeocodeService1" contract="GeocodeService.IGeocodeService" name="BasicHttpBinding_IGeocodeService1" /> <endpoint address="http://dev.virtualearth.net/webservices/v1/geocodeservice/GeocodeService.svc/binaryHttp" binding="customBinding" bindingConfiguration="CustomBinding_IGeocodeService" contract="GeocodeService.IGeocodeService" name="CustomBinding_IGeocodeService" /> <endpoint address="http://dev.virtualearth.net/webservices/v1/routeservice/routeservice.svc" binding="basicHttpBinding" bindingConfiguration="BasicHttpBinding_IRouteService" contract="RouteService.IRouteService" name="BasicHttpBinding_IRouteService" /> <endpoint address="http://dev.virtualearth.net/webservices/v1/routeservice/routeservice.svc/binaryHttp" binding="customBinding" bindingConfiguration="CustomBinding_IRouteService" contract="RouteService.IRouteService" name="CustomBinding_IRouteService" /> </client> </system.serviceModel> </configuration>
以上的内容配置了三个服务的引用。
四.添加BingMap服务引用请求代码(方法)
该函数包含一个string类型和一个int类型变量,string类型变量用来存储地址,int类型变量用来存储位置坐标点的索引
示例代码如下:
1 private void Geocode(string strAddress, int waypointIndex) 2 { 3 //声明一个GeocodeService的客户端变量,并通过其GeocodeCompleted属性设置其回调方法 4 GeocodeService.GeocodeServiceClient geocodeService = new GeocodeService.GeocodeServiceClient("BasicHttpBinding_IGeocodeService"); 5 geocodeService.GeocodeCompleted += new EventHandler<GeocodeService.GeocodeCompletedEventArgs>(geocodeService_GeocodeCompleted); 6 //声明一个获取服务请求的变量,由于必须要一个Key值作为提供服务的凭据,微软的服务器才会提供BingMap的服务,所以利用ApplicationId这个属性绑定到我们的Key 7 GeocodeService.GeocodeRequest geocodeRequest = new GeocodeService.GeocodeRequest(); 8 geocodeRequest.Credentials = new Credentials(); 9 geocodeRequest.Credentials.ApplicationId = ((ApplicationIdCredentialsProvider)bingmap.CredentialsProvider).ApplicationId; 10 //设定请求查询的地址 11 geocodeRequest.Query = strAddress; 12 //发送异步调用的请求,并通过waypointIndex来跟踪不同的请求并识别返回的结果 13 geocodeService.GeocodeAsync(geocodeRequest, waypointIndex); 14 }
下面是GeocodeCompleted的回调函数。
由于异步发送Geocde请求,所以需要处理回调函数返回的请求结果,这里回调函数返回的结果包含用户的状态,以及查询地点的经纬度等。在回调函数中我们需要保存这些信息,因为在后面计算路径的方法中需要用到这些信息。同时这里我们还需判断geocodeResaults中的元素是否为空,如果为空则表示还有请求未被处理,若都不为空,则开始计算路径。
1 private void geocodeService_GeocodeCompleted(object sender, GeocodeService.GeocodeCompletedEventArgs e) 2 { 3 //检索反馈(Response)回来的用户的状态来鉴别用请求符合那种情况,这里的e.UserState其实就等于之前的waypointIndex 4 int waypointIndex = System.Convert.ToInt32(e.UserState); 5 //检索反馈回来的GeocodeResault,并将它存储在全局变量geocodeResaults数组中 6 geocodeResults[waypointIndex] = e.Result.Results[0]; 7 //这个变量用来判断geocodeResault数组是否被填满,如果未填满则表明还有请求未处理 8 bool doneGeocoding = true; 9 foreach (GeocodeService.GeocodeResult gr in geocodeResults) 10 { 11 if (gr == null) 12 { 13 doneGeocoding = false; 14 } 15 } 16 //如果所有的请求都已得到处理,则进行路径的计算 17 if (doneGeocoding) 18 { 19 // CaculateRoute是用于计算路径的函数,将在先一步定于这个函数。 20 CaculateRoute(geocodeResults); 21 } 22 }
接着我们定义一个全局变量,这个变量用来存储请求返回来的结果,并应用于之后的路径计算。
internal GeocodeService.GeocodeResult[] geocodeResults;
五.在我们的请求得到应答,并对返回的数据进行处理之后,如果满足条件,我们将进行下一步,那就是计算两个地点的路径,因此我们需要将之前请求的结果 geocodeResults 传入进来,以供我们的计算。
下面是具体实现的代码:
1 private void CaculateRoute(GeocodeService.GeocodeResult[] results) 2 { 3 //定义路径计算服务的客户端变量,并声明其回调方法 4 RouteService.RouteServiceClient routeService = new RouteService.RouteServiceClient("BasicHttpBinding_IRouteService"); 5 routeService.CalculateRouteCompleted += new EventHandler<RouteService.CalculateRouteCompletedEventArgs>(routeService_CalculateRouteCompleted); 6 //设置RouteService的一些属性以及获取服务的Key 7 RouteService.RouteRequest routeRequest = new RouteService.RouteRequest(); 8 routeRequest.Credentials = new Credentials(); 9 routeRequest.Credentials.ApplicationId = ((ApplicationIdCredentialsProvider)bingmap.CredentialsProvider).ApplicationId; 10 //返回计算路径的点,以便在地图上画出 11 routeRequest.Options = new RouteService.RouteOptions(); 12 routeRequest.Options.RoutePathType = RouteService.RoutePathType.Points; 13 //通过使用GeocodeService服务所得到的geocodeResults,来设定用于计算路径的点。 14 routeRequest.Waypoints = new System.Collections.ObjectModel.ObservableCollection<RouteService.Waypoint>(); 15 foreach (GeocodeService.GeocodeResult result in results) 16 { 17 //将存储在GeocodeService.GeocodeResult中的点(包含经纬度等信息)添加到Waypoints集合中 18 // GeocodeResultToWaypoint为自定义的一个函数(下一步声明),该函数的作用是将geocodeResults中的点的信息的结果转化为具体的点,即Waypoint,这样才可以进行路径的计算,在转化之后我们就可发送计算路径的请求,计算两点之间的路劲 19 routeRequest.Waypoints.Add(GeocodeResultToWaypoint(result)); 20 } 21 //异步调用计算路径的服务请求 22 routeService.CalculateRouteAsync(routeRequest); 23 } 24 这是将GeocodeResult转化为Waypoint类型的自定义函数: 25 26 //将GeocodeResult中的坐标信息转化为具体的点的信息(Waypoint) 27 Private RouteService.Waypoint GeocodeResultToWaypoint 28 (GeocodeService.GeocodeResult result) 29 { 30 RouteService.Waypoint waypoint = new RouteService.Waypoint(); 31 waypoint.Description = result.DisplayName; 32 waypoint.Location = new Location(); 33 waypoint.Location.Latitude = result.Locations[0].Latitude; 34 waypoint.Location.Longitude = result.Locations[0].Longitude; 35 return waypoint; 36 }
接着我们需要处理计算路径的的回调函数,在回调函数中需要处理的是我们请求服务的结果,当计算路径的服务请求成功,且路径数不为0,即大于等于1条时,我们则将这条路径画出来,并在地图上显示。代码如下:
1 //CaculateRoute的回调函数 2 private void routeService_CalculateRouteCompleted(object sender, RouteService.CalculateRouteCompletedEventArgs e) 3 { 4 //如果计算路径成功,且路线条数不为0,则绘出路线 5 if ((e.Result.ResponseSummary.StatusCode == RouteService.ResponseStatusCode.Success) & (e.Result.Result.Legs.Count != 0)) 6 { 7 //设置绘制路线画笔的属性 8 Color routeColor = Colors.Blue; 9 SolidColorBrush routeBrush = new SolidColorBrush(routeColor); 10 MapPolyline routeLine = new MapPolyline(); 11 routeLine.Locations = new LocationCollection(); 12 routeLine.Stroke = routeBrush; 13 routeLine.Opacity = 0.65; 14 routeLine.StrokeThickness = 5.0; 15 //检索坐标点,用于绘制路线 16 foreach (Location p in e.Result.Result.RoutePath.Points) 17 { 18 //MSDN上是这么写的 19 //routeLine.Locations.Add(new Location(p.Latitude, p.Longitude)); 20 //以上代码不可用,原因是由于在WP7中Location没有包含两个参数的方法,WP7中将经纬度封装成了属性 21 Location location = new Location() 22 { 23 Longitude=p.Longitude, 24 Latitude=p.Latitude 25 }; 26 routeLine.Locations.Add(location); 27 } 28 //将绘制的路线的图层添加到当前地图中 29 MapLayer myRouteLayer = new MapLayer(); 30 bingmap.Children.Add(myRouteLayer); 31 myRouteLayer.Children.Add(routeLine); 32 //用点来标示每一种路线的Waypoint 33 foreach (GeocodeService.GeocodeResult gr in geocodeResults) 34 { 35 Ellipse point = new Ellipse(); 36 point.Width = 10; 37 point.Height = 10; 38 point.Fill = new SolidColorBrush(Colors.Red); 39 point.Opacity = 0.65; 40 Location location = new Location() 41 { 42 Latitude=gr.Locations[0].Latitude, 43 Longitude=gr.Locations[0].Longitude 44 }; 45 myRouteLayer.AddChild(point, location); 46 } 47 //计算出包含全部路径的矩形大小,这个将用于后面设置地图的视图(使查询得到的路线居中) 48 //RouteModel是自己定义的一个类,该类的作用是计算路径所在的矩形范围,并将路径视图居中显示在屏幕上 49 var routeModel = new RouteModel(e.Result.Result.RoutePath.Points); 50 LocationRect rect = LocationRect.CreateLocationRect(routeModel.Locations); 51 //设置地图的视图为路线所在的矩形 52 bingmap.SetView(rect); 53 } 54 }
其实在这里我们已经实现了路径的查询计算功能,只是当我们查询的路线不在当前屏幕范围时,我们需要调整一下屏幕显示的地图,将查询的结果显示很出来(例如当前的地图是沈阳市,但是我查询的长沙至武汉的路径,虽然以上的工作可以完成查询工作,但是并不会将我们的查询结果显示在屏幕上,所以我们将结果居中一下,在屏幕上显示出来),下面是RouteModel的代码:
1 /// <summary> 2 /// 用于设置地图路线的视图位置 3 /// </summary> 4 public class RouteModel 5 { 6 private readonly LocationCollection _locations; 7 8 /// <summary> 9 /// 得到路线的地理位置的集合 10 /// </summary> 11 public ICollection<GeoCoordinate> Locations 12 { 13 get { return _locations; } 14 } 15 /// <summary> 16 /// 初始化一个新的LocationCollection对象 17 /// </summary> 18 /// <param name="locations">位置信息的集合</param> 19 public RouteModel(ICollection<Location> locations) 20 { 21 _locations = new LocationCollection(); 22 foreach (Location location in locations) 23 { 24 _locations.Add(location); 25 } 26 } 27 } 28 //最后添加一个计算路劲的Button响应函数,即开始发送服务请求,计算路径: 29 private void CaculateButton_Click(object sender, EventArgs e) 30 { 31 //初始化geocodeResault数组变量,这里由于我们只查询两个地点的路径,所以数组长度设为2 32 geocodeResults = new GeocodeService.GeocodeResult[2]; 33 //调用Geocde方法发送服务请求 34 Geocode("Changsha", 0); 35 Geocode("Shenyang", 1); 36 }
在这里我直接的输入了两个地名,一个是沈阳,一个是长沙,其实理想的做法是添加两个TextBox控件,根据TextBox的内容实现不同地点的路径查询。实现的过程没有什么变化,不多说。
这里注意的一点是:在Windows Phone 7中的BingMap暂不支持中文,所以地名都需要用拼音。
到此为止,所有的工作都已完成,我们来看看最后的结果吧:
以上的内容参考了MSDN上的一篇文章,以及Jake Lin的视频,这里非常推荐Jake Lin的视频,初学者值得一看。
MSDN文章地址:
http://msdn.microsoft.com/en-us/library/ee681887.aspx
(版权所有,转载请标明出处)