7

I have the following struct…

struct Photo: Codable {

    let hasShadow: Bool
    let image: UIImage?

    enum CodingKeys: String, CodingKey {
        case `self`, hasShadow, image
    }

    init(hasShadow: Bool, image: UIImage?) {
        self.hasShadow = hasShadow
        self.image = image
    }

    init(from decoder: Decoder) throws {
        let container = try decoder.container(keyedBy: CodingKeys.self)
        hasShadow = try container.decode(Bool.self, forKey: .hasShadow)

        // This fails
        image = try container.decode(UIImage?.self, forKey: .image) 
    }

    func encode(to encoder: Encoder) throws {
        var container = encoder.container(keyedBy: CodingKeys.self)
        try container.encode(hasShadow, forKey: .hasShadow)

        // This also fails
        try container.encode(image, forKey: .image)
    }
}

Encoding a Photo fails with …

Optional does not conform to Encodable because UIImage does not conform to Encodable

Decoding fails with…

Key not found when expecting non-optional type Optional for coding key \"image\""))

Is there a way to encode Swift objects that include NSObject subclass properties that conform to NSCoding (UIImage, UIColor, etc)?

1
  • 4
    You have to write custom encode / decode code to archive / unarchive the objects to and from Data. Please read Encoding and Decoding Custom Types Commented Jul 26, 2017 at 9:16

1 Answer 1

9

Thanks to @vadian pointing me in the direction of encoding/decoding Data

class Photo: Codable {

    let hasShadow: Bool
    let image: UIImage?

    enum CodingKeys: String, CodingKey {
        case hasShadow, imageData
    }

    init(from decoder: Decoder) throws {
        let container = try decoder.container(keyedBy: CodingKeys.self)
        hasShadow = try container.decode(Bool.self, forKey: .hasShadow)

        if let imageData = try container.decodeIfPresent(Data.self, forKey: .imageData) {
            image = NSKeyedUnarchiver.unarchiveObject(with: imageData) as? UIImage
        } else {
            image = nil
        }
    }

    func encode(to encoder: Encoder) throws {
        var container = encoder.container(keyedBy: CodingKeys.self)
        try container.encode(hasShadow, forKey: .hasShadow)

        if let image = image {
            let imageData = NSKeyedArchiver.archivedData(withRootObject: image)
            try container.encode(imageData, forKey: .imageData)
        }
    }
}
Sign up to request clarification or add additional context in comments.

4 Comments

So in the end Codable doesn't really make anything easier, when using "custom types"? :-|
Well - it allows you to encode / decode non-NSObject subclasses (enums & structs)
@AshleyMills, I am getting this error "Type 'Photo' does not conform to protocol 'Decodable'" while copying this code in my file.
@GopalDevra Please ask this as another question.

Your Answer

By clicking “Post Your Answer”, you agree to our terms of service and acknowledge you have read our privacy policy.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.