¿Cómo codifico enum usando NSCoder en swift?
enums persistence (3)
Actualización para Xcode 6.3, Swift 1.2:
self.stage = Stage(rawValue: aDecoder.decodeObjectForKey("stage") as! String) ?? .DisplayAll
nota el as!
Fondo
Intento codificar una enumeración de estilo String utilizando el protocolo NSCoding, pero me estoy encontrando con errores al convertir y regresar de String.
Obtengo los siguientes errores al decodificar y codificar:
La cadena no es convertible a Stage
Argumento extra ForKey: en llamada
Código
enum Stage : String
{
case DisplayAll = "Display All"
case HideQuarter = "Hide Quarter"
case HideHalf = "Hide Half"
case HideTwoThirds = "Hide Two Thirds"
case HideAll = "Hide All"
}
class AppState : NSCoding, NSObject
{
var idx = 0
var stage = Stage.DisplayAll
override init() {}
required init(coder aDecoder: NSCoder) {
self.idx = aDecoder.decodeIntegerForKey( "idx" )
self.stage = aDecoder.decodeObjectForKey( "stage" ) as String // ERROR
}
func encodeWithCoder(aCoder: NSCoder) {
aCoder.encodeInteger( self.idx, forKey:"idx" )
aCoder.encodeObject( self.stage as String, forKey:"stage" ) // ERROR
}
// ...
}
Aquí hay una solución para Swift 4.2 . Como se indica en las otras respuestas, el problema es que intenta asignar directamente la variable de stage
con una cadena decodificada e intenta convertir la variable de stage
en una cadena en el método encodeWithCoder
. Necesita usar valores sin formato en su lugar.
enum Stage: String {
case DisplayAll = "Display All"
case HideQuarter = "Hide Quarter"
case HideHalf = "Hide Half"
case HideTwoThirds = "Hide Two Thirds"
case HideAll = "Hide All"
}
class AppState: NSCoding, NSObject {
var idx = 0
var stage = Stage.DisplayAll
override init() {}
required init(coder aDecoder: NSCoder) {
self.idx = aDecoder.decodeInteger(forKey: "idx")
self.stage = Stage(rawValue: aDecoder.decodeObject(forKey: "stage") as String)
}
func encodeWithCoder(aCoder: NSCoder) {
aCoder.encode(self.idx, forKey:"idx")
aCoder.encode(self.stage.rawValue, forKey:"stage")
}
// ...
}
Necesita convertir la enumeración hacia y desde el valor en bruto. En Swift 1.2 (Xcode 6.3), esto se vería así:
class AppState : NSObject, NSCoding
{
var idx = 0
var stage = Stage.DisplayAll
override init() {}
required init(coder aDecoder: NSCoder) {
self.idx = aDecoder.decodeIntegerForKey( "idx" )
self.stage = Stage(rawValue: (aDecoder.decodeObjectForKey( "stage" ) as! String)) ?? .DisplayAll
}
func encodeWithCoder(aCoder: NSCoder) {
aCoder.encodeInteger( self.idx, forKey:"idx" )
aCoder.encodeObject( self.stage.rawValue, forKey:"stage" )
}
// ...
}
Swift 1.1 (Xcode 6.1), utiliza as
lugar de as!
:
self.stage = Stage(rawValue: (aDecoder.decodeObjectForKey( "stage" ) as String)) ?? .DisplayAll
Swift 1.0 (Xcode 6.0) usa toRaw()
y fromRaw()
esta manera:
self.stage = Stage.fromRaw(aDecoder.decodeObjectForKey( "stage" ) as String) ?? .DisplayAll
aCoder.encodeObject( self.stage.toRaw(), forKey:"stage" )