MKGeodesicPolyline
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 flatMKMap
was a necessary conceit until recently.
What changed? The discovery ofMKGeodesic
, which is the subject of this week’s article.
MKGeodesic
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.
AnMKGeodesic
is created with an array of 2MKMap
s orCLLocation
s:
MKGeodesic Polyline
Creating anletLAX=CLLocation(latitude:33.9424955,longitude:-118.4080684)letJFK=CLLocation(latitude:40.6397511,longitude:-73.7789256)varcoordinates=[LAX.coordinate,JFK.coordinate]letgeodesic Polyline=MKGeodesic Polyline(coordinates:&coordinates,count:2)map View.add Overlay(geodesic Polyline)
CLLocation*LAX=[[CLLocationalloc]init With Latitude:33.9424955longitude:-118.4080684];CLLocation*JFK=[[CLLocationalloc]init With Latitude:40.6397511longitude:-73.7789256];CLLocation Coordinate2Dcoordinates[2]={LAX.coordinate,JFK.coordinate};MKGeodesic Polyline*geodesic Polyline=[MKGeodesic Polylinepolyline With Coordinates:coordinatescount:2];[map Viewadd Overlay:geodesic Polyline];
Although the overlay looks like a smooth curve, it is actually comprised of thousands of tiny line segments (true to itsMKPolyline
lineage):
print(geodesic Polyline.point Count)// 3984
NSLog(@"%d",geodesic Polyline.point Count)// 3984
Like any object conforming to theMKOverlay
protocol, anMKGeodesic
instance is displayed by adding it to anMKMap
withadd
and implementingmap
:
MKGeodesic Polyline
on anMKMap View
Rendering// MARK: MKMap View Delegatefuncmap View(map View:MKMap View,renderer For Overlayoverlay:MKOverlay)->MKOverlay Renderer{guardletpolyline=overlayas?MKPolylineelse{returnMKOverlay Renderer()}letrenderer=MKPolyline Renderer(polyline:polyline)renderer.line Width=3.0renderer.alpha=0.5renderer.stroke Color=UIColor.blue Color()returnrenderer}
#pragma mark - MKMap View Delegate-(MKOverlay Renderer*)map View:(MKMap View*)map Viewrenderer For Overlay:(id<MKOverlay>)overlay{if(![overlayis Kind Of Class:[MKPolylineclass]]){returnnil;}MKPolyline Renderer*renderer=[[MKPolyline Rendereralloc]init With Polyline:(MKPolyline*)overlay];renderer.line Width=3.0f;renderer.stroke Color=[UIColorblue Color];renderer.alpha=0.5;returnrenderer;}
For comparison, here’s the same geodesic overlaid with a route created from
MKDirections
:
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.
MKAnnotation View
on aMKGeodesic Polyline
Animating anSince 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 theplane
andplane
(the index of the current map point for the polyline):
// MARK: Flight Path Propertiesvarmap View:MKMap View!varflightpath Polyline:MKGeodesic Polyline!varplane Annotation:MKPoint Annotation!varplane Annotation Position=0
@interfaceMap View Controller()<MKMap View Delegate>@propertyMKMap View*map View;@propertyMKGeodesic Polyline*flightpath Polyline;@propertyMKPoint Annotation*plane Annotation;@propertyNSUIntegerplane Annotation Position;@end
Next, right below the initialization of our map view and polyline, we create anMKPoint
for our plane:
letannotation=MKPoint Annotation()annotation.title=NSLocalized String("Plane",comment:"Plane marker")map View.add Annotation(annotation)self.plane Annotation=annotationself.update Plane Position()
self.plane Annotation=[[MKPoint Annotationalloc]init];self.plane Annotation.title=NSLocalized String(@"Plane",nil);[self.map Viewadd Annotation:self.plane Annotation];[selfupdate Plane Position];
That call toupdate
in the last line ticks the animation and updates the position of the plane:
funcupdate Plane Position(){letstep=5guardplane Annotation Position+step<flightpath Polyline.point Countelse{return}letpoints=flightpath Polyline.points()self.plane Annotation Position+=stepletnext Map Point=points[plane Annotation Position]self.plane Annotation.coordinate=MKCoordinate For Map Point(next Map Point)perform Selector("update Plane Position",with Object:nil,after Delay:0.03)}
-(void)update Plane Position{staticNSUIntegerconststep=5;if(self.plane Annotation Position+step>=self.flightpath Polyline.point Count){return;}self.plane Annotation Position+=step;MKMap Pointnext Map Point=self.flightpath Polyline.points[self.plane Annotation Position];self.plane Annotation.coordinate=MKCoordinate For Map Point(next Map Point);[selfperform Selector:@selector(update Plane Position)with Object:nilafter Delay:0.03];}
We’ll perform this method roughly 30 times a second, until the plane has arrived at its final destination.
Finally, we implementmap
to have the annotation render on the map view:
funcmap View(map View:MKMap View,view For Annotationannotation:MKAnnotation)->MKAnnotation View?{letplane Identifier="Plane"letannotation View=map View.dequeue Reusable Annotation View With Identifier(plane Identifier)??MKAnnotation View(annotation:annotation,reuse Identifier:plane Identifier)annotation View.image=UIImage(named:"airplane")returnannotation View}
-(MKAnnotation View*)map View:(MKMap View*)map Viewview For Annotation:(id<MKAnnotation>)annotation{staticNSString*Pin Identifier=@"Pin";MKAnnotation View*annotation View=[map Viewdequeue Reusable Annotation View With Identifier:Pin Identifier];if(!annotation View){annotation View=[[MKAnnotation Viewalloc]init With Annotation:annotationreuse Identifier:Pin Identifier];}annotation View.image=[UIImageimage Named:@"plane"];returnannotation View;}
Hmm… close but noSkyMall Personalized Cigar Case Flask.
Let’s update the rotation of the plane as it moves across its flightpath.
MKAnnotation View
along a Path
Rotating anTo calculate the plane’s direction, we’ll take the slope from the previous and next points:
letprevious Map Point=points[plane Annotation Position]plane Annotation Position+=stepletnext Map Point=points[plane Annotation Position]self.plane Direction=direction Between Points(previous Map Point,next Map Point)self.plane Annotation.coordinate=MKCoordinate For Map Point(next Map Point)
MKMap Pointprevious Map Point=self.flightpath Polyline.points[self.plane Annotation Position];self.plane Annotation Position+=step;MKMap Pointnext Map Point=self.flightpath Polyline.points[self.plane Annotation Position];self.plane Direction=XXDirection Between Points(previous Map Point,next Map Point);self.plane Annotation.coordinate=MKCoordinate For Map Point(next Map Point);
direction
is a function that returns aCLLocation
(0 – 360 degrees, where North = 0) given twoMKMap
s.
We calculate from
MKMap
s rather than converted coordinates, because we’re interested in the slope of the line on the flat projection.Point
privatefuncdirection Between Points(source Point:MKMap Point,_destination Point:MKMap Point)->CLLocation Direction{letx=destination Point.x-source Point.xlety=destination Point.y-source Point.yreturnradians To Degrees(atan2(y,x))%360+90}
staticCLLocation DirectionXXDirection Between Points(MKMap Pointsource Point,MKMap Pointdestination Point){doublex=destination Point.x-source Point.x;doubley=destination Point.y-source Point.y;returnfmod(XXRadians To Degrees(atan2(y,x)),360.0f)+90.0f;}
That convenience functionradians
(and its partner,degrees
) are simply:
privatefuncradians To Degrees(radians:Double)->Double{returnradians*180/M_PI}privatefuncdegrees To Radians(degrees:Double)->Double{returndegrees*M_PI/180}
staticinlinedoubleXXRadians To Degrees(doubleradians){returnradians*180.0f/M_PI;}staticinlinedoubleXXDegrees To Radians(doubledegrees){returndegrees*M_PI/180.0f;}
That direction is stored in a new property,var plane
, calculated fromself.plane
inupdate
(ideally renamed toupdate
with this addition). To make the annotation rotate, we apply atransform
onannotation
:
annotation View.transform=CGAffine Transform Rotate(map View.transform,degrees To Radians(plane Direction))
self.annotation View.transform=CGAffine Transform Rotate(self.map View.transform,XXDegrees To Radians(self.plane Direction));
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.