为ARAnchor实现Codable:“无法在扩展程序中自动合成...”

时间:2019-04-14 16:41:29

标签: swift augmented-reality arkit codable

代码extension ARAnchor: Codable {}产生错误:

“不能在与该类型不同的文件的扩展名中自动合成'Decodable'的实现。”

这是什么意思?我能够以类似的方式为另一本机类型实现Codable,而不会出现任何错误。

3 个答案:

答案 0 :(得分:1)

Swift编译器AFAICT当前不支持在不同的源文件中对CodableEquatableHashable进行整合。有关跟踪此问题的信息,请参见https://bugs.swift.org/browse/SR-6101

答案 1 :(得分:1)

您可以创建一个实现Codable的容器对象,然后使用该对象来编码和解码锚点。我在操场上尝试了此代码,它对我有用。您将需要对它进行调整,以适应来自锚点的数据。例如,我对name进行了编码,但是这对您可能没有用,并且如果您的锚点初始化时没有名称,它甚至可能会中断。您也可以使用simd_float4x4做同样的事情。

import Foundation
import ARKit

class AnchorContainer: Codable {

    let anchor: ARAnchor

    init(anchor: ARAnchor) {
        self.anchor = anchor
    }

    required init(from decoder: Decoder) throws {
        let container = try decoder.container(keyedBy: CodingKeys.self)
        let name = try container.decode(String.self, forKey: .name)
        let transform0 = try container.decode(simd_float4.self, forKey: .transform0)
        let transform1 = try container.decode(simd_float4.self, forKey: .transform1)
        let transform2 = try container.decode(simd_float4.self, forKey: .transform2)
        let transform3 = try container.decode(simd_float4.self, forKey: .transform3)
        let matrix = simd_float4x4(columns: (transform0, transform1, transform2, transform3))
        anchor = ARAnchor(name: name, transform: matrix)
    }

    func encode(to encoder: Encoder) throws {
        var container = encoder.container(keyedBy: CodingKeys.self)
        try container.encode(anchor.name, forKey: .name) // Might want to make sure that the name is not nil here
        try container.encode(anchor.transform.columns.0, forKey: .transform0)
        try container.encode(anchor.transform.columns.1, forKey: .transform1)
        try container.encode(anchor.transform.columns.2, forKey: .transform2)
        try container.encode(anchor.transform.columns.3, forKey: .transform3)
    }

    enum CodingKeys: String, CodingKey {
        case name
        case transform0
        case transform1
        case transform2
        case transform3
    }

}

// EXAMPLE:

let anchor = ARAnchor(name: "Bill", transform: simd_float4x4(float4(repeating: 4), float4(repeating: 5), float4(repeating: 6), float4(repeating: 7))) // Make a arbitrary anchor
print(anchor) // Figure out what it's value is


do {
    let data = try JSONEncoder().encode(AnchorContainer(anchor: anchor))
    let anchorDecode = try JSONDecoder().decode(AnchorContainer.self, from: data)
    print(anchorDecode.anchor) // Print the value after decoding to make sure that the result is the same
} catch {
    print(error.localizedDescription)
}

答案 2 :(得分:0)

由于CodableEncodableDecodable协议的类型别名,因此在将其用作 type 通用约束 ,它匹配符合两种协议的任何类型:

public typealias Codable = Decodable & Encodable
  

当前(Xcode 10.2.1 / Swift 5.0.1)Codable目前不受支持,如果一个文件中的扩展名增加了另一个文件中的一致性。在https://bugs.swift.org/处进行检查。

希望这会有所帮助。