Как использовать связанные объекты с перечислениями?
У меня есть ViewController, к которому я добавил два новых свойства, используя связанные объекты: перечисление и строку (версия строки взята отсюда)
Вот мой пример кода:
extension UIViewController {
private struct AssociatedKeys {
static var handle = "handle"
}
enum CustomStringEnum: String {
case One = "One"
case Two = "Two"
case Three = "Three"
}
var customEnum: CustomStringEnum {
get {
return objc_getAssociatedObject(self, &AssociatedKeys.handle) as? CustomStringEnum ?? .One
}
set {
objc_setAssociatedObject(self, &AssociatedKeys.handle, newValue.rawValue, objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
}
}
var descriptiveName: String {
get {
return objc_getAssociatedObject(self, &AssociatedKeys.handle) as! String
}
set {
objc_setAssociatedObject(
self,
&AssociatedKeys.handle,
newValue as NSString?,
.OBJC_ASSOCIATION_RETAIN_NONATOMIC
)
}
}
}
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
let vc = UIViewController()
vc.customEnum = .Three
vc.descriptiveName = "Three"
print(vc.customEnum.rawValue) // -> This prints "One"
print(vc.descriptiveName) // -> This prints "Three"
}
}
Строковая версия работает правильно, а enum - нет. И я не уверен, в чем проблема.
Это проблема с objc_getAssociatedObject
или objc_setAssociatedObject
, Версия get все время равна nil, поэтому возвращается значение по умолчанию One.
2 ответа
Решение
Измените свой код на этот
extension UIViewController {
private struct AssociatedKeys {
static var handle = "handle"
static var enumContext = "enumContext"
}
enum CustomStringEnum: String {
case One = "One"
case Two = "Two"
case Three = "Three"
}
var customEnum: CustomStringEnum {
get {
let rawvalue = objc_getAssociatedObject(self, &AssociatedKeys.enumContext)
if rawvalue == nil{
return .One
}else{
return CustomStringEnum(rawValue: rawvalue as! String)!;
}
}
set {
objc_setAssociatedObject(self, &AssociatedKeys.enumContext, newValue.rawValue, objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
}
}
var descriptiveName: String {
get {
return objc_getAssociatedObject(self, &AssociatedKeys.handle) as! String
}
set {
objc_setAssociatedObject(
self,
&AssociatedKeys.handle,
newValue as NSString?,
.OBJC_ASSOCIATION_RETAIN_NONATOMIC
)
}
}
}
Тогда будет работать
Вы жестко закодированы .One
в вашем get
функция.
Поскольку, согласно вашему комментарию, ваши сложности со связанными значениями не имеют смысла, вам следует упростить:
enum Numbers: String {
case One = "One"
case Two = "Two"
case Three = "Three"
// default
init() { self = .One }
static let germanNumbers = [One: "Eins", Two: "Zwei", Three: "Drei"]
var germanString: String { return Numbers.germanNumbers[self]! }
}
let num = Numbers.Three
print(num) // "Three"
print(num.rawValue) // "Three"
let defaultNum = Numbers()
print(defaultNum) // "One"
print(num.germanString) // "Drei"
print(defaultNum.germanString) // "Eins"