Want to center MKMapView on a point N-pixels below a given pin (which may or may not be visible in the current MapRect).
I've been trying to solve this using various plays with -(CLLocationCoordinate2D)convertPoint:(CGPoint)point toCoordinateFromView:(UIView *)view
to no success.
Anyone been down this road (no pun intended)?
after reading this thread ad playing around, especially with the zooming on annotation, I ended up with following procedures:
** Centering on annotation:**
** Zooming on annotation:**
-(CGFloat) sizeOfBottom
and-(CGFloat) sizeOfTop
both return height of panels covering the mapview from the layout guidesThe only way to reliably do this is to use the following:
In order to do that given a map region you want to center on, you have to convert the map region to a MKMapRect. Use the edge padding for the pixel offset, obviously.
See here for that: Convert MKCoordinateRegion to MKMapRect
Comment: I find it rather strange that that's the only way to do it, given that MKMapRect is not something one normally uses with an MKMapView - all the conversion methods are for MKMapRegion. But, ok, at least it works. Tested in my own project.
The easiest technique is to just shift the map down, say 40% from where the
coordinate
would be, taking advantage of thespan
of theregion
of theMKMapView
. If you don't need actual pixels, but just need it to move down so that theCLLocationCoordinate2D
in question is near the top of the map (say 10% away from the top):If you want to account for rotation and pitch of the camera, the above technique may not be adequate. In that case, you could:
Identify the position in the view to which you want to shift the user location;
Convert that to a
CLLocation
;Calculate the distance of the current user location from that new desired location;
Move the camera by that distance in the direction 180° from the current heading of the map's camera.
E.g. in Swift 3, something like:
Where
CLLocationCoordinate2D
has the followingextension
:So, even with the camera pitched and at a heading other than due north, this moves the user's location (which is centered, where the lower crosshair is) up 150 pixels (where the upper crosshair is), yielding something like:
Obviously, you should be conscious about degenerate situations (e.g. you're 1 km from the south pole and you try to shift the map up 2 km meters; you're using a camera angle pitched so far that the desired screen location is past the horizon; etc.), but for practical, real-world scenarios, something like the above might be sufficient. Obviously, if you don't let the user change the pitch of the camera, the answer is even easier.
Original answer: for moving the annotation
n
pixelsIf you have a
CLLocationCoordinate2D
, you can convert it to aCGPoint
, move it x pixels, and then convert it back to aCLLocationCoordinate2D
:You can call this by:
Unfortunately, this only works if the
coordinate
is visible before you start, so you might have to go to the original coordinate first, and then adjust the center.One easy solution is that you make the frame of your map view larger than the visible area. Then position your pin in the center of the map view and hide all the unwanted areas behind another view or outside of the screen bounds.
Let me elaborate. If I look at your screen shot, do the following:
The distance between you pin and the bottom is 353 pixel. So make your map views frame twice the height: 706 pixel. You screenshot has a height of 411 pixel. Position your frame at an origin of 706px - 411px = -293 pixel. Now center your map view at the coordinate of the pin and you are done.
Update 4-March-2014:
I created a small sample application with Xcode 5.0.2 to demo this: http://cl.ly/0e2v0u3G2q1d
Look at this method on
MKMapView
:As an alternative to the accepted answer, I suggest your original instincts were correct. You can work strictly within the map views pixel coordinate space to get offsets and final positioning. Then using the conversion call from location to screen view, you can get the final location and set the maps center.
This will work with the camera rotated and is in respect to the screen space. In my case I needed to center the map on a pin, with an offset to account for a map drawer.
Here are the conversion calls
And here is a swift 4 example