Я искал ответ в Стэке все выше и ниже, но я не могу найти ничего, что работает. Проблема проста - у меня есть View Controller, который имеет дочерний Table View Controller.
class ViewController: UIView Controller {
var child: UIViewController!
var managedObjectContext: NSManagedObjectContext!
init(context: NSManagedObjectContext) {
self.managedObjectContext = context
}
viewDidLoad() {
self.child = CustomTVC(context: self.managedObjectContext)
self.child?.view.frame = SomeFrameThatFits
self.addChild(child!)
self.addSubview(child!.view)
//Targeting
self.someButton.addTarget(self, action: #selector(addHello()), for: .touchUpInside)
}
@objc func addHello() {
_ = CoreDataWriter(context: self.managedObjectContext).save()
}
}
Отлично, классно, аккуратно, хорошо сделано. Я создаю класс ViewController
в SceneDelegate, используя предоставленный context
.
SceneDelegate
...
let context = (UIApplication.shared.delegate as! AppDelegate).persistentContainer.viewContext
if let windowScene = scene as? UIWindowScene {
let window = UIWindow(windowScene: windowScene)
window.rootViewController = ViewController(context: context)
self.window = window
window.makeKeyAndVisible()
}
Теперь в моем классе ViewController
у меня есть кнопка sendButton
. Вся эта кнопка позволяет добавить еще один элемент в мои CoreData, используя отдельный класс CoreDataWriter, написанный ниже.
class CoreDataWriter {
init(context: NSManagedObjectContext) {
self.managedObjectContext = context
}
func save() {
let message = Message(context: managedObjectContext)
message.text = "Hello World!"
do {
try self.managedObjectContext.save()
return nil
} catch let error {
print(error.localizedDescription)
}
}
}
Теперь у меня есть буквально копия и вставленный пример кода Apple в мой класс Table View Controller:
class CustomTVC: UITableViewController, NSFetchedResultsControllerDelegate {
var managedObjectContext: NSManagedObjectContext!
var fetchedResultsController: NSFetchedResultsController<Message>!
init(context: NSManagedObjectContext) {
self.managedObjectContext = context
super.init(nibName: nil, bundle: nil)
}
override init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: Bundle?) {
super.init(nibName: nil, bundle: nil)
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
override func viewDidLoad() {
super.viewDidLoad()
self.tableView.register(UITableViewCell.self, forCellReuseIdentifier: "default")
initializeFetchedResultsController()
}
//MARK: CoreData connections
func initializeFetchedResultsController() {
let request = NSFetchRequest<Message>(entityName: "Message")
let sort = NSSortDescriptor(key: "sent", ascending: true)
let predicate = NSPredicate(format: "chat.id == %@", "123454")
fetchedResultsController = NSFetchedResultsController(fetchRequest: request, managedObjectContext: self.managedObjectContext, sectionNameKeyPath: "chat.id", cacheName: "rootCache")
fetchedResultsController.delegate = self
do {
try fetchedResultsController.performFetch()
} catch {
fatalError("Failed to initialize FetchedResultsController: \(error)")
}
}
// MARK: - Table view data source
override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(withIdentifier: "default", for: indexPath)
// Set up the cell
guard let object = self.fetchedResultsController?.object(at: indexPath) else {
fatalError("Attempt to configure cell without a managed object")
}
//Populate the cell from the object
cell.textLabel?.text = object.text
return cell
}
override func numberOfSections(in tableView: UITableView) -> Int {
return 1
}
override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
guard let sections = fetchedResultsController.sections else {
fatalError("No sections in fetchedResultsController")
}
let sectionInfo = sections[section]
return sectionInfo.numberOfObjects
}
//MARK: More stuff for NSData
func controllerWillChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>) {
print("controller will change content")
tableView.beginUpdates()
}
func controller(_ controller: NSFetchedResultsController<NSFetchRequestResult>, didChange sectionInfo: NSFetchedResultsSectionInfo, atSectionIndex sectionIndex: Int, for type: NSFetchedResultsChangeType) {
print("in ctronoller")
switch type {
case .insert:
tableView.insertSections(IndexSet(integer: sectionIndex), with: .fade)
case .delete:
tableView.deleteSections(IndexSet(integer: sectionIndex), with: .fade)
case .move:
break
case .update:
break
@unknown default:
fatalError()
}
}
func controller(_ controller: NSFetchedResultsController<NSFetchRequestResult>, didChange anObject: Any, at indexPath: IndexPath?, for type: NSFetchedResultsChangeType, newIndexPath: IndexPath?) {
print("in didChange controller")
switch type {
case .insert:
tableView.insertRows(at: [newIndexPath!], with: .fade)
case .delete:
tableView.deleteRows(at: [indexPath!], with: .fade)
case .update:
tableView.reloadRows(at: [indexPath!], with: .fade)
case .move:
tableView.moveRow(at: indexPath!, to: newIndexPath!)
@unknown default:
fatalError()
}
}
func controllerDidChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>) {
tableView.endUpdates()
}
На мой взгляд, если я нажму sh удобную кнопку денди, табличное представление должно поменять sh с новым «Hello World» - но Я ничего не получаю. Нет отпечатков, нет ошибок. Ничего.
Я проверил ссылку моего контекста из контекста AppDelegate вплоть до кнопки отправки - того же адреса. Я проверил fetchedResultsController
и его делегата - не ноль.
Я в полной, полной и полной растерянности относительно того, что происходит. Есть идеи?