带有核心数据的TableViewController中的搜索栏
问题描述:
我正在制作带有核心数据的TableViewController。实际上,用户可以将新项目添加到表格视图中,这些项目保存在核心数据中。视图将显示联系人列表。 一切工作正常,但我无法让搜索栏工作。我尝试了很多,但我对swift很陌生,这是我的第一个应用程序。请告诉我应该将哪些内容添加到我的代码中?带有核心数据的TableViewController中的搜索栏
class ContactsViewController: UITableViewController {
@IBOutlet var searchBar: UISearchBar!
var contacts: [NSManagedObject] = []
override func viewDidLoad() {
super.viewDidLoad()
fetch()
tableView.reloadData()
}
override func didReceiveMemoryWarning() {
super.didReceiveMemoryWarning()
// Dispose of any resources that can be recreated.
}
//MARK: - Data Source
func fetch() {
guard let appDelegate = UIApplication.shared.delegate as? AppDelegate else { return }
let managedObjectContext = appDelegate.persistentContainer.viewContext
let fetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName:"Contact")
do {
contacts = try managedObjectContext.fetch(fetchRequest) as! [NSManagedObject]
} catch let error as NSError {
print("Could not fetch. \(error)")
}
}
func save(name: String, phoneNumber: String, dataUltimei: String) {
guard let appDelegate = UIApplication.shared.delegate as? AppDelegate else { return }
let managedObjectContext = appDelegate.persistentContainer.viewContext
guard let entity = NSEntityDescription.entity(forEntityName:"Contact", in: managedObjectContext) else { return }
let contact = NSManagedObject(entity: entity, insertInto: managedObjectContext)
contact.setValue(name, forKey: "name")
contact.setValue(phoneNumber, forKey: "phoneNumber")
contact.setValue(dataUltimei, forKey: "dataUltimei")
do {
try managedObjectContext.save()
self.contacts.append(contact)
} catch let error as NSError {
print("Couldn't save. \(error)")
}
}
func update(indexPath: IndexPath, name:String, phoneNumber: String, dataUltimei: String) {
guard let appDelegate = UIApplication.shared.delegate as? AppDelegate else { return }
let managedObjectContext = appDelegate.persistentContainer.viewContext
let contact = contacts[indexPath.row]
contact.setValue(name, forKey:"name")
contact.setValue(phoneNumber, forKey: "phoneNumber")
contact.setValue(dataUltimei, forKey: "dataUltimei")
do {
try managedObjectContext.save()
contacts[indexPath.row] = contact
} catch let error as NSError {
print("Couldn't update. \(error)")
}
}
func delete(_ contact: NSManagedObject, at indexPath: IndexPath) {
guard let appDelegate = UIApplication.shared.delegate as? AppDelegate else { return }
let managedObjectContext = appDelegate.persistentContainer.viewContext
managedObjectContext.delete(contact)
contacts.remove(at: indexPath.row)
}
override func numberOfSections(in tableView: UITableView) -> Int {
return 1
}
override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return contacts.count
}
override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(withIdentifier: "ContactCell", for: indexPath)
var contact = contacts[indexPath.row]
if isSearching {
contact = contacts[indexPath.row]
}
else {
contact = contacts[indexPath.row]
}
cell.textLabel?.text = contact.value(forKey:"name") as? String
}
return cell
}
override func tableView(_ tableView: UITableView, commit editingStyle: UITableViewCellEditingStyle, forRowAt indexPath: IndexPath) {
if editingStyle == .delete {
do {
let contact = contacts[indexPath.row]
delete(contact, at: indexPath)
fetch()
tableView.reloadData()
} catch let error as NSError {
print("Could not save. \(error), \(error.userInfo)")
}
}
}
override func tableView(_ tableView: UITableView, canEditRowAt indexPath: IndexPath) -> Bool {
// Return false if you do not want the specified item to be editable.
return true
}
@IBAction func unwindToContactList(segue: UIStoryboardSegue) {
if let viewController = segue.source as? AddContactViewController {
guard let name: String = viewController.nameTextField.text, let phoneNumber: String = viewController.phoneNumberTextField.text, let dataUltimei: String = viewController.tabel.text else { return }
if name != "" && phoneNumber != "" {
if let indexPath = viewController.indexPathForContact {
update(indexPath: indexPath, name: name, phoneNumber: phoneNumber, dataUltimei: dataUltimei)
} else {
save(name:name, phoneNumber:phoneNumber, dataUltimei:dataUltimei)
}
}
tableView.reloadData()
} else if let viewController = segue.source as? ContactDetailViewController {
if viewController.isDeleted {
guard let indexPath: IndexPath = viewController.indexPath else { return }
let contact = contacts[indexPath.row]
delete(contact, at: indexPath)
tableView.reloadData()
}
}
}
override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
if segue.identifier == "contactDetailSegue" {
guard let navViewController = segue.destination as? UINavigationController else { return }
guard let viewController = navViewController.topViewController as? ContactDetailViewController else { return }
guard let indexPath = tableView.indexPathForSelectedRow else { return }
let contact = contacts[indexPath.row]
viewController.contact = contact
viewController.indexPath = indexPath
}
}
}
答
添加以下功能:
func searchBar(_ searchBar: UISearchBar, textDidChange searchText: String) {
if !searchText.isEmpty {
var predicate: NSPredicate = NSPredicate()
predicate = NSPredicate(format: "name contains[c] '\(searchText)'")
guard let appDelegate = UIApplication.shared.delegate as? AppDelegate else { return }
let managedObjectContext = appDelegate.persistentContainer.viewContext
let fetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName:"Contact")
fetchRequest.predicate = predicate
do {
contacts = try managedObjectContext.fetch(fetchRequest) as! [NSManagedObject]
} catch let error as NSError {
print("Could not fetch. \(error)")
}
}
tableView.reloadData()
}
添加UISearchBarDelegate和UISearchDisplayDelegate象下面这样:
class ViewController: UIViewController, UISearchBarDelegate, UISearchDisplayDelegate
并添加搜索栏委托在viewDidLoad中:
searchBar.delegate=self
嗨,欢迎来到SO。请添加有关不工作的含义。您也可以参考[如何创建一个最小的,可验证的示例](https://stackoverflow.com/help/mcve),以获得更好的使用本站点的结果。祝你好运! – theFunkyEngineer