How can I keep a pin centered on a map whilst I move (via Pan Gesture) another view vertically over the map such that the pin remains above the overlay (not an actual MapKit overlay).
See attached screenshots for the first and final states.
I've got the CGRect of the space between the overlay and the top of the screen as the user pan's up / down. However, how I use that to move the map and pin whilst zooming into the map as the user pans upward..and zoom back out again when the user pans downward, has eluded me so far.
I've tried different approaches, from attempting to adjust the visible rect to adjusting the map view's frame. The answer may lie in some MKMapRect / Region trickery..
(Hand icon by Freepik CC BY 3.0)
Actually, keithbhunter's code is slow because besides updating the region faster than it can load it, the map is also changing height which causes extra overhead!
I updated the code so that it runs smooth.
With this code what i do is keep the map view the same size but instead i move the point of center to compensate for the height of the sliding view.
For this code to work, you have to modify keithbhunter's setup so that the mapView's bottom constraint is pinned completely to the superview's bottom (and not to the slidingView (so that the mapView is always the same size as the super view). For the rest the setup is the same.
Also it is possible to customize the amount of zoom with the variable
maxMetersDistance
Here i'm, always centering on the Eifel tower
To setup the views, place a map view and a
UIView
in your view controller. Use autolayout to pin the map view's left, top and right sides to the superview. Then pin the bottom of the map view to the top of theUIView
. Next, pin the left, bottom and right sides of theUIView
to the superview. Finally, set a height constraint on theUIView
to whatever you want it to initialize to. This height value will be changed as the user drags the view. This allows theUIView
to grow as we please and appease autolayout at the same time.Add an
@IBOutlet
to your view controller for the map view, theUIView
and theUIView
's height constraint, like in the code above. TheregionDistance
property is what is doing the zooming magic here. It is an exponential equation (that I made up randomly) which will calculate the region for either larger or smaller based on the map view's height.reloadMap()
uses this to update the map's zoom. Tying it all together is theUIPanGestureRecognizer
on theUIView
, which is what controls theUIView
's height, causing the zooming action on the map.Pitfall: This forces the map to update the region faster than it can load the region, making it look very jumpy. There is probably ways around this. Get creative.
The coordinate I used in the example is Apple's headquarters.