简体   繁体   English

如何在 SwiftUI 中使 TextAlignment 可编码?

[英]How can I make TextAlignment Codable in SwiftUI?

I know how to use rawValue to make custom enums adhere to the Codable protocol, but is there a way to make built-in native enums work?我知道如何使用rawValue使自定义枚举遵守Codable协议,但是有没有办法使内置本机枚举工作?

I want to be able to store and retrieve TextAlignment values without having to re-invent the wheel and create some custom solution.我希望能够存储和检索TextAlignment值,而不必重新发明轮子并创建一些自定义解决方案。

I can't find a single example anywhere!我在任何地方都找不到一个例子!

Unfortunately not for now.不幸的是,暂时没有。

You can possibly convert them into strings like \\(alignment) and then restore by iterating through allCases and pick the one, but I do not recommend this approach as there is no guarantee that the names won't change in future.您可以将它们转换为像\\(alignment)这样的字符串,然后通过遍历allCases并选择一个来恢复,但我不推荐这种方法,因为不能保证名称将来不会改变。

What I would recommend - is to implement a custom Codable conformance using switch...case like:我建议 - 是使用switch...case来实现自定义Codable一致性:

extension TextAlignment: Codable {

    /// Adding constants to make it less error prone when refactoring
    private static var leadingIntRepresentation = -1
    private static var centerIntRepresentation = 0
    private static var trailingIntRepresentation = 1

    /// Adding a way to represent TextAlignment as Int value
    /// You may choose a different type if more appropriate
    /// for your coding practice
    private var intRepresentation: Int {
        switch self {
        case .leading: return TextAlignment.leadingIntRepresentation
        case .center: return TextAlignment.centerIntRepresentation
        case .trailing: return TextAlignment.trailingIntRepresentation
        }
    }

    /// Initializing TextAlignment using Int
    /// Making the method private as our intention is to only use it for coding
    private init(_ intRepresentation: Int) {
        switch intRepresentation {
        case TextAlignment.leadingIntRepresentation: self = .leading
        case TextAlignment.trailingIntRepresentation: self = .trailing
        default: self = .center
        }
    }

    /// Conforming to Encodable
    public func encode(to encoder: Encoder) throws {
        var container = encoder.singleValueContainer()
        try container.encode(intRepresentation)
    }

    /// Conforming to Decodable
    public init(from decoder: Decoder) throws {
        let container = try decoder.singleValueContainer()
        self.init(try container.decode(Int.self))
    }

}

This method is pretty safe.这种方法非常安全。 The one disadvantage is that we could possibly receive a value other than -1 , 0 and 1 .一个缺点是我们可能会收到-101以外的值。 We would treat such a case as a center .我们会将这种情况视为一个center You might consider throwing an error instead.您可能会考虑抛出错误。

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM