Zon*_*ame 2 enums swift property-wrapper
下面的结构适用于基本数据类型,例如Int、String、Double等,有没有办法使其适用于枚举,这样我就不必使用 rawValues 进行手动解析?
@propertyWrapper
struct Storage<T> {
let objectName: String
let defaultValue: T
let defaults: UserDefaults
init(_ objectName: String, defaultValue: T, defaults: UserDefaults = .standard) {
self.objectName = objectName
self.defaultValue = defaultValue
self.defaults = defaults
}
var wrappedValue: T {
get { return self.defaults.object(forKey: self.objectName) as? T ?? self.defaultValue }
set { self.defaults.set(newValue, forKey: self.objectName) }
}
}
Run Code Online (Sandbox Code Playgroud)
目前我的解决方法是将对象包装在枚举的另一个 getter 和 setter 中,如下所示。
enum SomeType: String {
case foo, bar, baz
}
class Defaults {
@Storage("object", default: "")
private var: objectContainer: String
var object: SomeType? {
get {
return SomeType(rawValue: self.objectContainer)
}
set {
self.objectContainer = newValue.rawValue ?? ""
}
}
}
Run Code Online (Sandbox Code Playgroud)
我尝试创建一个特定于类型的不同属性包装器,但 Swift 无法从中RawRepresentable推断出类型。T
@propertyWrapper
struct RawRepresentableStorage<T: RawRepresentable> {
let objectName: String
let defaultValue: T
let defaults: UserDefaults
init(_ objectName: String, defaultValue: T, defaults: UserDefaults = .standard) {
self.objectName = objectName
self.defaultValue = defaultValue
self.defaults = defaults
}
var wrappedValue: T {
get {
guard let object = self.defaults.object(forKey: self.objectName) as? T else {
return self.defaultValue
}
// Error on this line:
// `Cannot convert value of type 'T' to expected argument type 'T.RawValue'`
return T(rawValue: object) ?? self.defaultValue
}
set {
self.defaults.set(newValue.rawValue, forKey: self.objectName)
}
}
}
Run Code Online (Sandbox Code Playgroud)
即使只有少数可存储枚举,为代码库中的每个可存储枚举创建包装类型也不是理想的选择
我是否遗漏了什么,或者这是 Swift 当前的限制?
在你的wrappedValuegetter 中,你正在转换为错误的类型。你正在传递object到T(rawValue:),所以它的类型需要是T.RawValue,而不是T。
@propertyWrapper
struct RawRepresentableStorage<T: RawRepresentable> {
let objectName: String
let defaultValue: T
let defaults: UserDefaults
init(_ objectName: String, defaultValue: T, defaults: UserDefaults = .standard) {
self.objectName = objectName
self.defaultValue = defaultValue
self.defaults = defaults
}
var wrappedValue: T {
get {
guard let object = self.defaults.object(forKey: self.objectName) as? T.RawValue else {
return self.defaultValue
}
return T(rawValue: object) ?? self.defaultValue
}
set {
self.defaults.set(newValue.rawValue, forKey: self.objectName)
}
}
}
Run Code Online (Sandbox Code Playgroud)