How do I get a value as a Swift type from an object of type Unmanaged<AnyObject>
. My example is using an ABRecordRef
I'm creating a contact object to manage once I get the ABRecordRef
, but I'm having trouble translating from ObjC. Here's what I have:
init(recordRef: ABRecordRef) { if let firstName = ABRecordCopyValue(recordRef, kABPersonFirstNameProperty) { self.firstName = firstName } }
If it were ObjC, I would do:
CFTypeRef firstNameRef = ABRecordCopyValue(recordRef, kABPersonFirstNameProperty); if (firstNameRef) { self.firstName = (__bridge NSString *)firstNameRef; }
I can't seem to find the right combination of downcasting / conversion, so any help is appreciated.
Since NoOne answered before I solved it, I'll add the answer here:
firstName.takeRetainedValue() as? String
If you look at the header of the Unmanaged
struct type, you'll find this:
/// Get the value of this unmanaged reference as a managed /// reference and consume an unbalanced retain of it. /// /// This is useful when a function returns an unmanaged reference /// and you know that you're responsible for releasing the result. func takeRetainedValue() -> T
So, because the CFTypeRef
is converted to Unmanaged<AnyObject>
in Swift.
Unmanaged
uses generics to define a return type, and it is declared like so:
Unmanaged<T>
Our object is of type Unmanaged<AnyObject>
which means that takeRetainedValue()
will return type T
, or in our case, type AnyObject
. I use optional downcasting since my property firstName
is of type String?
.
You can use the takeRetainedValue
method to get your value out of your Unmanaged
object. In Foundation API's, I'm guessing that they will mostly be of type Unmanaged<AnyObject>!
and a downcast will be required. The generic formula appears to be:
var unmanagedObject: Unmanaged<AnyObject> = someFunctionThatReturnsAnUnmanagedObject() var newValue: Type = unmanagedObject.takeRetainedValue as Type
import UIKit import AddressBook class ViewController: UIViewController { lazy var addressBook: ABAddressBookRef = { var error: Unmanaged<CFError>? return ABAddressBookCreateWithOptions(nil, &error).takeRetainedValue() as ABAddressBookRef }() @IBAction func GetPeople(sender: UIButton) { var ab: ABAddressBookRef = addressBook switch ABAddressBookGetAuthorizationStatus(){ case .Authorized: println("Authorized") readFromAddressBook(addressBook) case .Denied: println("Denied") case .Restricted: println("Restricted") case .NotDetermined: println("Not determined") ABAddressBookRequestAccessWithCompletion(addressBook, {[weak self] (granted: Bool, error: CFError!) in if granted { let strongSelf = self! println("Access is granted") strongSelf.readFromAddressBook(strongSelf.addressBook) } else{ println("Access is not granted") } }) default: break } } func readFromAddressBook(addressBook: ABAddressBookRef){ let allPeople = ABAddressBookCopyArrayOfAllPeople(addressBook).takeRetainedValue() as NSArray for person:ABRecordRef in allPeople{ if let firstName = ABRecordCopyValue(person, kABPersonFirstNameProperty) { if let lastName = ABRecordCopyValue(person, kABPersonLastNameProperty) { let ln:String = (lastName.takeRetainedValue() as? String) ?? "" let fn:String = (firstName.takeRetainedValue() as? String) ?? "" println("\(ln) - \(fn)") } } } }
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With