iOS - MKMapView place annotation by using address instead of lat / long
Asked Answered
T

7

40

I am able to place annotations on my MKMapView by using latitude and longitude, however, my feed coming in that I need to use location for is using street addresses instead of Lat and Long. e.g 1234 west 1234 east, San Francisco, CA ...

Would this have something to do with the CLLocationManager?

Has anyone attempted this before?

Tomtom answered 7/3, 2012 at 17:7 Comment(2)
Do you mean street address? I'm not sure what you mean by "1234 west 1234 east".Landlocked
Yes, by street address. I can get the State, City, and address. If possible, I'd like to convert that street address into Lat and Long.Tomtom
T
76

Based on psoft's excellent information, I was able to achieve what I was looking for with this code.

NSString *location = @"some address, state, and zip";
CLGeocoder *geocoder = [[CLGeocoder alloc] init];
            [geocoder geocodeAddressString:location 
                 completionHandler:^(NSArray* placemarks, NSError* error){
                     if (placemarks && placemarks.count > 0) {
                         CLPlacemark *topResult = [placemarks objectAtIndex:0];
                         MKPlacemark *placemark = [[MKPlacemark alloc] initWithPlacemark:topResult];

                         MKCoordinateRegion region = self.mapView.region;
                         region.center = placemark.region.center;
                         region.span.longitudeDelta /= 8.0;
                         region.span.latitudeDelta /= 8.0;

                         [self.mapView setRegion:region animated:YES];
                         [self.mapView addAnnotation:placemark];
                     }
                 }
             ];
Tomtom answered 7/3, 2012 at 20:46 Comment(4)
Setting the region.center as shown is deprecated and setting the span to 1/8 of the current span won't always make sense. This would be simpler: MKCoordinateRegion region = MKCoordinateRegionMakeWithDistance(placemark.coordinate, 5000, 5000); (the 5000 is in meters).Malaria
Always use firstObject instead of objectAtIndex:0Hyphenated
Just as a side note, if (placemarks && placemarks.count > 0) is redundant as placemarks.count will return 0 if placemarks is nil. It can be replaced with if (placemarks.count > 0).Lubin
In case anyone was wondering the completionHandler gets called on the main threadCoachwhip
U
16

Refactored Swift version:

let location = "some address, state, and zip"
let geocoder = CLGeocoder()
geocoder.geocodeAddressString(location) { [weak self] placemarks, error in
    if let placemark = placemarks?.first, let location = placemark.location {
        let mark = MKPlacemark(placemark: placemark)

        if var region = self?.mapView.region {
            region.center = location.coordinate
            region.span.longitudeDelta /= 8.0
            region.span.latitudeDelta /= 8.0
            self?.mapView.setRegion(region, animated: true)
            self?.mapView.addAnnotation(mark)
        }
    }
}
Unholy answered 13/8, 2016 at 23:28 Comment(0)
L
11

What you're after is called geocoding or forward-geocoding. Reverse-geocoding is the process of converting a lat/long pair to street address.

iOS5 provides the CLGeocoder class for geocoding. MKPlacemark supports reverse-goecoding in iOS >= 3.0. The data involved of course is very large, so your app will generally need network access to the functionality.

A good place to start is Apple's Location Awareness Programming Guide. Also, there are LOTS of questions about this here on SO. https://stackoverflow.com/search?q=geocoding

Good luck!

Landlocked answered 7/3, 2012 at 17:43 Comment(0)
C
10

Swift version

        let location = self.txtLocation.text;
        let geocoder:CLGeocoder = CLGeocoder();
        geocoder.geocodeAddressString(location!) { (placemarks: [CLPlacemark]?, error: NSError?) -> Void in
            if placemarks?.count > 0 {
                let topResult:CLPlacemark = placemarks![0];
                let placemark: MKPlacemark = MKPlacemark(placemark: topResult);

                var region: MKCoordinateRegion = self.mkMapView.region;
                region.center = (placemark.location?.coordinate)!;
                region.span.longitudeDelta /= 8.0;
                region.span.latitudeDelta /= 8.0;
                self.mkMapView.setRegion(region, animated: true);
                self.mkMapView.addAnnotation(placemark);

            }
        }
Casaubon answered 18/3, 2016 at 9:49 Comment(0)
C
7

Since iOS 7, placemark.region.center is deprecated. Now needs to use:

region.center = [(CLCircularRegion *)placemark.region center];

You can also read Apple Documentation about this and here too.

Condyloma answered 17/2, 2014 at 10:1 Comment(0)
F
6

Here's another version ...

- Swift 2017 syntax

- Shows a placename for the point, down the bottom

- Choose any size, say 5 km, as the area shown

func map() {
    let 5km:CLLocationDistance = 5000
    
    let a= "100 smith avenue some town 90210 SD"
    let g = CLGeocoder()
    
    g.geocodeAddressString(a) { [weak self] placemarks, error in
        if let p = placemarks?.first, let l = placemark.location {
            
            let p = MKPlacemark(coordinate: l.coordinate, addressDictionary: nil)
            
            let cr = MKCoordinateRegionMakeWithDistance(l.coordinate, 5km, 5km)
            let options = [
                MKLaunchOptionsMapCenterKey: NSValue(mkCoordinate: cr.center),
                MKLaunchOptionsMapSpanKey: NSValue(mkCoordinateSpan: cr.span)
            ]
            
            let m = MKMapItem(placemark: p)
            m.name = "Your House"
            m.openInMaps(launchOptions: options)
        }
    }


}
Fanti answered 27/3, 2017 at 16:19 Comment(0)
S
1

That is not possible. But you can obtain the long/lat from an Address by using the CLGeocoder class.

Salman answered 7/3, 2012 at 17:42 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.