Movatterモバイル変換


[0]ホーム

URL:


MKGeodesic​Polyline

Written byMattt

We knew that the Earth was not flat long before 1492. Early navigators observed the way ships would dip out of view over the horizon many centuries before the Age of Discovery.

For many iOS developers, though, a flatMKMapView was a necessary conceit until recently.

What changed? The discovery ofMKGeodesicPolyline, which is the subject of this week’s article.


MKGeodesicPolyline was introduced to the Map Kit framework in iOS 7. As its name implies, it creates ageodesic—essentially a straight line over a curved surface.

On the surface of asphereoblate spheroidgeoid, the shortest distance between two points appears as an arc on a flat projection. Over large distances, this takes apronounced, circular shape.

AnMKGeodesicPolyline is created with an array of 2MKMapPoints orCLLocationCoordinate2Ds:

Creating anMKGeodesicPolyline

letLAX=CLLocation(latitude:33.9424955,longitude:-118.4080684)letJFK=CLLocation(latitude:40.6397511,longitude:-73.7789256)varcoordinates=[LAX.coordinate,JFK.coordinate]letgeodesicPolyline=MKGeodesicPolyline(coordinates:&coordinates,count:2)mapView.addOverlay(geodesicPolyline)

Although the overlay looks like a smooth curve, it is actually comprised of thousands of tiny line segments (true to itsMKPolyline lineage):

print(geodesicPolyline.pointCount)// 3984

Like any object conforming to theMKOverlay protocol, anMKGeodesicPolyline instance is displayed by adding it to anMKMapView withaddOverlay() and implementingmapView(_:rendererForOverlay:):

RenderingMKGeodesicPolyline on anMKMapView

// MARK: MKMapViewDelegatefuncmapView(mapView:MKMapView,rendererForOverlayoverlay:MKOverlay)->MKOverlayRenderer{guardletpolyline=overlayas?MKPolylineelse{returnMKOverlayRenderer()}letrenderer=MKPolylineRenderer(polyline:polyline)renderer.lineWidth=3.0renderer.alpha=0.5renderer.strokeColor=UIColor.blueColor()returnrenderer}

MKGeodesicPolyline on an MKMapView

For comparison, here’s the same geodesic overlaid with a route created fromMKDirections:

MKGeodesicPolyline on an MKMapView compared to MKDirections Polyline

As the crow flies, it’s 3,983 km.
As the wolf runs, it’s 4,559 km—nearly 15% longer.
…and that’s just distance; taking into account average travel speed, the total time is ~5 hours by air and 40+ hours by land.

Animating anMKAnnotationView on aMKGeodesicPolyline

Since geodesics make reasonable approximations for flight paths, a common use case would be to animate the trajectory of a flight over time.

To do this, we’ll make properties for our map view and geodesic polyline between LAX and JFK, and add new properties for theplaneAnnotation andplaneAnnotationPosition (the index of the current map point for the polyline):

// MARK: Flight Path PropertiesvarmapView:MKMapView!varflightpathPolyline:MKGeodesicPolyline!varplaneAnnotation:MKPointAnnotation!varplaneAnnotationPosition=0

Next, right below the initialization of our map view and polyline, we create anMKPointAnnotation for our plane:

letannotation=MKPointAnnotation()annotation.title=NSLocalizedString("Plane",comment:"Plane marker")mapView.addAnnotation(annotation)self.planeAnnotation=annotationself.updatePlanePosition()

That call toupdatePlanePosition in the last line ticks the animation and updates the position of the plane:

funcupdatePlanePosition(){letstep=5guardplaneAnnotationPosition+step<flightpathPolyline.pointCountelse{return}letpoints=flightpathPolyline.points()self.planeAnnotationPosition+=stepletnextMapPoint=points[planeAnnotationPosition]self.planeAnnotation.coordinate=MKCoordinateForMapPoint(nextMapPoint)performSelector("updatePlanePosition",withObject:nil,afterDelay:0.03)}

We’ll perform this method roughly 30 times a second, until the plane has arrived at its final destination.

Finally, we implementmapView(_:viewForAnnotation:) to have the annotation render on the map view:

funcmapView(mapView:MKMapView,viewForAnnotationannotation:MKAnnotation)->MKAnnotationView?{letplaneIdentifier="Plane"letannotationView=mapView.dequeueReusableAnnotationViewWithIdentifier(planeIdentifier)??MKAnnotationView(annotation:annotation,reuseIdentifier:planeIdentifier)annotationView.image=UIImage(named:"airplane")returnannotationView}

MKAnnotationView without Rotation

Hmm… close but noSkyMall Personalized Cigar Case Flask.

Let’s update the rotation of the plane as it moves across its flightpath.

Rotating anMKAnnotationView along a Path

To calculate the plane’s direction, we’ll take the slope from the previous and next points:

letpreviousMapPoint=points[planeAnnotationPosition]planeAnnotationPosition+=stepletnextMapPoint=points[planeAnnotationPosition]self.planeDirection=directionBetweenPoints(previousMapPoint,nextMapPoint)self.planeAnnotation.coordinate=MKCoordinateForMapPoint(nextMapPoint)

directionBetweenPoints is a function that returns aCLLocationDirection (0 – 360 degrees, where North = 0) given twoMKMapPoints.

We calculate fromMKMapPoints rather than converted coordinates, because we’re interested in the slope of the line on the flat projection.

privatefuncdirectionBetweenPoints(sourcePoint:MKMapPoint,_destinationPoint:MKMapPoint)->CLLocationDirection{letx=destinationPoint.x-sourcePoint.xlety=destinationPoint.y-sourcePoint.yreturnradiansToDegrees(atan2(y,x))%360+90}

That convenience functionradiansToDegrees (and its partner,degreesToRadians) are simply:

privatefuncradiansToDegrees(radians:Double)->Double{returnradians*180/M_PI}privatefuncdegreesToRadians(degrees:Double)->Double{returndegrees*M_PI/180}

That direction is stored in a new property,var planeDirection: CLLocationDirection, calculated fromself.planeDirection = directionBetweenPoints(currentMapPoint, nextMapPoint) inupdatePlanePosition (ideally renamed toupdatePlanePositionAndDirection with this addition). To make the annotation rotate, we apply atransform onannotationView:

annotationView.transform=CGAffineTransformRotate(mapView.transform,degreesToRadians(planeDirection))

MKAnnotationView with Rotation

Ah much better! At last, we have mastered the skies with a fancy visualization, worthy of any travel-related app.


Perhaps more than any other system framework, MapKit has managed to get incrementally better, little by little with every iOS release[1][2]. For anyone with a touch-and-go relationship to the framework, returning after a few releases is a delightful experience of discovery and rediscovery.

I look forward to seeing what lies on the horizon with iOS 8 and beyond.

NSMutableHipster

Questions? Corrections?Issues andpull requests are always welcome.

This article uses Swift version 2.0 and was last reviewed on November 12, 2015. Find status information for all articles on thestatus page.

Written byMattt
Mattt

Mattt (@mattt) is a writer and developer in Portland, Oregon.

Next Article

In programming, what often begins as a necessary instruction eventually becomes a vestigial cue for humans. For developers just starting with Cocoa & Cocoa Touch, the IBAction, IBOutlet, and IBOutletCollection macros are particularly bewildering examples of this phenomenon


[8]ページ先頭

©2009-2025 Movatter.jp