Как получить отформатированный адрес NSString из AddressDictionary? - PullRequest
15 голосов
/ 21 октября 2011

Попытка получить отформатированный адрес из AddressDictionary, который я получил от CLGeocoder. Использовал следующий код без результата:

subtitle = [NSString stringWithString:[[addressDict objectForKey:@"FormattedAddressLines"]objectAtIndex:0]];

Также пробовал:

subtitle = [[[ABAddressBook sharedAddressBook] formattedAddressFromDictionary:placemark.addressDictionary] string];

но этот код работает только на Mac OS X.

Компилятор спрашивает об ABAdressBook, но у меня импортированы оба заголовочных файла.

#import <AddressBook/ABAddressBook.h>
#import <AddressBook/AddressBook.h>

Ответы [ 7 ]

40 голосов
/ 21 октября 2011

Документация для свойства addressDictionary гласит:

Вы можете отформатировать содержимое этого словаря, чтобы получить полную строку адреса, а не создавать адрес самостоятельно.Чтобы отформатировать словарь, используйте функцию ABCreateStringWithAddressDictionary, как описано в Справочнике по функциям интерфейса адресной книги.

Поэтому добавьте и импортируйте структуру AddressBookUI и попробуйте:

subtitle = 
    ABCreateStringWithAddressDictionary(placemark.addressDictionary, NO);
25 голосов
/ 12 августа 2013

После некоторых копаний под iOS 6.1 я обнаружил, что словарь адресов CLPlacemark содержит предварительно отформатированный адрес:

CLLocation *location = [[CLLocation alloc]initWithLatitude:37.3175 longitude:-122.041944];
[[[CLGeocoder alloc]init] reverseGeocodeLocation:location completionHandler:^(NSArray *placemarks, NSError *error) {
    CLPlacemark *placemark = placemarks[0];
    NSArray *lines = placemark.addressDictionary[ @"FormattedAddressLines"];
    NSString *addressString = [lines componentsJoinedByString:@"\n"];
    NSLog(@"Address: %@", addressString);
}];

Я пока не могу найти документацию по этому поводу, но она работает для всех адресов, которые я тестировал.

19 голосов
/ 18 декабря 2015

Как подчеркнул Мартен Дэвис, ABCreateStringWithAddressDictionary устарела в iOS 9.

Вы можете использовать приведенные ниже функции для преобразования addressDictionary в более новую CNMutablePostalAddress, затем используйте CNPostalAddressFormatter для генерации локализованной строки, пока вы импортируете структуру Contacts.

Swift 3.x

// Convert to the newer CNPostalAddress
func postalAddressFromAddressDictionary(_ addressdictionary: Dictionary<NSObject,AnyObject>) -> CNMutablePostalAddress {
   let address = CNMutablePostalAddress()

   address.street = addressdictionary["Street" as NSObject] as? String ?? ""
   address.state = addressdictionary["State" as NSObject] as? String ?? ""
   address.city = addressdictionary["City" as NSObject] as? String ?? ""
   address.country = addressdictionary["Country" as NSObject] as? String ?? ""
   address.postalCode = addressdictionary["ZIP" as NSObject] as? String ?? ""

   return address
}

// Create a localized address string from an Address Dictionary
func localizedStringForAddressDictionary(addressDictionary: Dictionary<NSObject,AnyObject>) -> String {
    return CNPostalAddressFormatter.string(from: postalAddressFromAddressDictionary(addressDictionary), style: .mailingAddress)
}

Swift 2.x

import Contacts

// Convert to the newer CNPostalAddress
func postalAddressFromAddressDictionary(addressdictionary: Dictionary<NSObject,AnyObject>) -> CNMutablePostalAddress {

    let address = CNMutablePostalAddress()

    address.street = addressdictionary["Street"] as? String ?? ""
    address.state = addressdictionary["State"] as? String ?? ""
    address.city = addressdictionary["City"] as? String ?? ""
    address.country = addressdictionary["Country"] as? String ?? ""
    address.postalCode = addressdictionary["ZIP"] as? String ?? ""

    return address
}

// Create a localized address string from an Address Dictionary
func localizedStringForAddressDictionary(addressDictionary: Dictionary<NSObject,AnyObject>) -> String {

    return CNPostalAddressFormatter.stringFromPostalAddress(postalAddressFromAddressDictionary(addressDictionary), style: .MailingAddress)
}
13 голосов
/ 29 марта 2015
func locationManager(manager: CLLocationManager, didUpdateLocations locations: [CLLocation]) {
    // get the address
    if let location = locations.last {
        CLGeocoder().reverseGeocodeLocation(location, completionHandler: { (result: [CLPlacemark]?, err: NSError?) -> Void in
            if let placemark = result?.last
                , addrList = placemark.addressDictionary?["FormattedAddressLines"] as? [String]
            {
                let address =  addrList.joinWithSeparator(", ")
                print(address)
            }
        })
    }
}

Над быстрой версией.

4 голосов
/ 05 октября 2016

Я использую Swift 3 / XCode 8

Ответ ZYiOS был приятным и коротким, но не компилировался для меня.

Вопрос состоит в том, как перейти от существующего словаря адресов к строковому адресу. Вот что я сделал:

import CoreLocation

func getAddressString(placemark: CLPlacemark) -> String? {
    var originAddress : String?

    if let addrList = placemark.addressDictionary?["FormattedAddressLines"] as? [String]
    {
        originAddress =  addrList.joined(separator: ", ")
    }

    return originAddress
}
2 голосов
/ 28 октября 2016

Swift 3 / Xcode 8 Помощник Mehtod для получения адреса из CLPlaceMark

class func formattedAddress(fromPlacemark placemark: CLPlacemark) -> String{
    var address = ""

    if let name = placemark.addressDictionary?["Name"] as? String {
        address = constructAddressString(address, newString: name)
    }

    if let city = placemark.addressDictionary?["City"] as? String {
        address = constructAddressString(address, newString: city)
    }

    if let state = placemark.addressDictionary?["State"] as? String {
        address = constructAddressString(address, newString: state)
    }

    if let country = placemark.country{
      address = constructAddressString(address, newString: country)
    }

    return address
  }
0 голосов
/ 10 мая 2017

Просто создайте расширение для CLLocation:

typealias AddressDictionaryHandler = ([String: Any]?) -> Void

extension CLLocation {

    func addressDictionary(completion: @escaping AddressDictionaryHandler) {

        CLGeocoder().reverseGeocodeLocation(self) { placemarks, _ in
            completion(placemarks?.first?.addressDictionary as? [String: AnyObject])
        }
    }
}

Пример:

let location = CLLocation()

location.addressDictionary { dictionary in

    let city = dictionary?["City"] as? String
    let street = dictionary?["Street"] as? String
}
...