Мне нужно сохранить созданные пользователем контроллеры ViewController в файле json, и позже мне нужно будет воссоздать их либо на том же устройстве, либо на другом устройстве.
Контроллеры View доступны в раскадровке, но Я не могу должным образом создать их экземпляр в функции init (from decoder: Decoder).
Это ViewController для сохранения и восстановления.
class SecondViewController: UIViewController, Encodable, Decodable {
@IBOutlet weak var textField: UITextField!
private var restoreText: String?
enum CodingKeys: CodingKey {
case textField
}
func encode(to encoder: Encoder) throws {
var container = encoder.container(keyedBy: CodingKeys.self)
try container.encode(textField.text, forKey: .textField)
}
required init(from decoder: Decoder) throws {
let container = try decoder.container(keyedBy: CodingKeys.self)
restoreText = try? container.decode(String.self, forKey: .textField)
super.init(nibName: "", bundle: nil)
}
required init?(coder: NSCoder) {
super.init(coder: coder)
}
override func viewDidLoad() {
super.viewDidLoad()
if let text = restoreText {
textField.text = text
}
}
@IBAction func save(_ sender: Any) {
do {
let jsonData = try JSONEncoder().encode(self)
let s = String(data: jsonData, encoding: .ascii)
print("Widgets \(s!)")
let filename = getDocumentsDirectory().appendingPathComponent("output.txt")
try jsonData.write(to: filename)
} catch {
print(error.localizedDescription)
}
}
}
Здесь я загружаю контроллер из файла json, и я пытаюсь воссоздать его.
...
do {
let filename = getDocumentsDirectory().appendingPathComponent("output.txt")
let jsonData = try Data(contentsOf: filename)
if let s = String(data: jsonData, encoding: .ascii) {
let controller = try JSONDecoder().decode(SecondViewController.self, from: jsonData)
let frame = CGRect(x: 0, y: 0, width: view.frame.width * 0.8, height: view.frame.height * 0.8)
controller.view.frame = frame
controller.view.center = view.center
view.addSubview(controller.view)
addChild(controller)
}
} catch {
print(error.localizedDescription)
}
...
Запустив этот код, я получаю следующую ошибку:
*** Assertion failure in -[UINib initWithNibName:directory:bundle:], /AppleInternal/BuildRoot/Library/Caches/com.apple.xbs/Sources/UIKitCore_Sim/UIKit-3920.26.113/UINib.m:97
2020-04-18 09:44:03.543891+0200 ViewControllerSerialization[7764:504964] *** Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: 'Invalid parameter not satisfying: (name != nil) && ([name length] > 0)'
Насколько я понимаю, эта ошибка потому что контроллер не был загружен из раскадровки.
Вопрос в том, как правильно инициализировать ViewController из раскадровки в функции init (from decoder: Decoder)?