gpt4 book ai didi

swift - 在 Swift 中使用核心数据在 TableView 中显示部分数据

转载 作者:搜寻专家 更新时间:2023-11-01 05:38:41 25 4
gpt4 key购买 nike

我在使用 Swift 中的 Core Data 显示表格 View 中按部分分组的数据时遇到了一些麻烦。

我会用一个例子来说明这个问题。让我们假设三个类:订单、产品和服务。类 Order 具有属性“日期”以及与产品和服务的对多关系。

对于每个订单,应创建一个部分来显示在特定日期订购的所有产品和服务。

使用 NSFetchedResultsController 所有的订单都是根据属性日期获取的。这提供了正确数量的部分。但是,我遇到了在每个部分单独的行中显示所有产品和服务的问题,因为“numberOfRowsInSection”是基于订单数量而不是产品和服务的数量。

下面是说明我的问题的代码。希望有人能帮助我找到正确的方向。

提前致谢

杰拉德

import UIKit
import CoreData

class ExampleViewController: UIViewController, UITableViewDataSource, UITableViewDelegate, NSFetchedResultsControllerDelegate {

@IBOutlet weak var tableView: UITableView!

let appDelegate = UIApplication.sharedApplication().delegate as! AppDelegate
let managedObjectContext = (UIApplication.sharedApplication().delegate as! AppDelegate).managedObjectContext
var fetchedResultController: NSFetchedResultsController = NSFetchedResultsController()

var orders = [Order]()

override func viewDidLoad() {
super.viewDidLoad()
fetchedResultController.delegate = self
tableView.dataSource = self
tableView.delegate = self

fetchData()
}

func numberOfSectionsInTableView(tableView: UITableView) -> Int {
return fetchedResultController.sections?.count ?? 0
}

func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
let sectionInfo = fetchedResultController.sections![section] as NSFetchedResultsSectionInfo
return sectionInfo.numberOfObjects
// Unfortunatelly, this provides not the total number of products and services per section
}

func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
let textCellIdentifier = "ExampleTableViewCell"
let cell = tableView.dequeueReusableCellWithIdentifier(textCellIdentifier, forIndexPath: indexPath) as! ExampleTableViewCell

let order = orders[indexPath.row] // Data fetched using NSFetchedResultsController

let products = order.products!.allObjects as! [Product] // Swift Array
let services = order.services!.allObjects as! [Service] // Swift Array

// Each product and each service should be displayed in separate rows.
// Instead, all products and services are displayed in a single row as below
var displaytext = ""
for product in products {
displaytext += product.name! + "\n"
}

for service in services {
displaytext += service.name! + "\n"
}

cell.orderLabel.text = displaytext // display text in ExampleTableViewCell

return cell
}

func tableView(tableView: UITableView, titleForHeaderInSection section: Int) -> String? {
if let sections = fetchedResultController.sections {
let currentSection = sections[section]
return currentSection.name
}
return nil
}


func orderFetchRequest() -> NSFetchRequest {
let fetchRequest = NSFetchRequest(entityName: "Order")
let sortDescriptor = NSSortDescriptor(key: "date", ascending: true)
let predicate = NSPredicate(format: "date >= %@ AND date <= %@", startDate, endDate) // startDate and endData are defined elsewhere

fetchRequest.sortDescriptors = [sortDescriptor]
fetchRequest.predicate = predicate

return fetchRequest
}

func fetchData() {
let fetchRequest = orderFetchRequest()
fetchedResultController = NSFetchedResultsController(fetchRequest: fetchRequest, managedObjectContext: managedObjectContext, sectionNameKeyPath: "date", cacheName: nil)

do {
orders = try managedObjectContext.executeFetchRequest(fetchRequest) as! [Order]
}
catch let error as NSError {
print("Could not fetch \(error), \(error.userInfo)")
}

do {
try fetchedResultController.performFetch()
}
catch _ {
}
}

最佳答案

正如 Wain 所说,FRC 帮不上什么忙。 FRC 有两个主要优点,超越普通的获取:它可以为您管理这些部分,并且它可以通过其委托(delegate)方法自动处理插入/删除/更新。第一个对你的情况没有用,因为你想要每个订单一个部分。第二个帮助不大,因为 FRC 只会监视一个实体的更改,而您的 TableView 是由三个实体构建的。

不过,我认为您可以使用 FRC 来达到某种效果。首先,不要理会 sectionNameKeyPath:您在 Orders 和 TableView 部分之间有一个一对一的映射,因此您可以使用 TableView 的 section作为 FRC 的 fetchedObjects 的索引来标识每个部分的顺序。然后可以通过对相关 Orderproductsservices 的计数求和来找到 numberOrRowsInSection。困惑(并且可能很慢)的一点是将 TableView 的行映射到 productsservices 的正确元素。

import UIKit
import CoreData

class ExampleViewController: UIViewController, UITableViewDataSource, UITableViewDelegate, NSFetchedResultsControllerDelegate {

@IBOutlet weak var tableView: UITableView!

let appDelegate = UIApplication.sharedApplication().delegate as! AppDelegate
let managedObjectContext = (UIApplication.sharedApplication().delegate as! AppDelegate).managedObjectContext!
var fetchedResultController: NSFetchedResultsController = NSFetchedResultsController()

var orders = [Order]()
var startDate : NSDate = NSDate()
var endDate : NSDate = NSDate()

override func viewDidLoad() {
super.viewDidLoad()
fetchedResultController.delegate = self
tableView.dataSource = self
tableView.delegate = self

fetchData()
}

func numberOfSectionsInTableView(tableView: UITableView) -> Int {
return fetchedResultController.fetchedObjects!.count
}

func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
let order = fetchedResultController.fetchedObjects![section] as! Order
return ((order.products?.count ?? 0) + (order.services?.count ?? 0))
}

func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
let textCellIdentifier = "ExampleTableViewCell"
let row = indexPath.row

let cell = tableView.dequeueReusableCellWithIdentifier(textCellIdentifier, forIndexPath: indexPath) as! ExampleTableViewCell

let order = fetchedResultController.fetchedObjects![indexPath.section] as! Order // Data fetched using NSFetchedResultsController
let products = (order.products?.allObjects ?? [Product]()) as! [Product] // Swift Array
let services = (order.services?.allObjects ?? [Service]()) as! [Service] // Swift Array

if (row < products.count) { // this is a Product row
cell.orderLabel.text = products[row].name!
} else { // this is a Service row
cell.orderLabel.text = services[row-products.count].name!
}
return cell
}

func tableView(tableView: UITableView, titleForHeaderInSection section: Int) -> String? {
let order = fetchedResultController.fetchedObjects![section] as! Order
return "\(order.date)"
}


func orderFetchRequest() -> NSFetchRequest {
let fetchRequest = NSFetchRequest(entityName: "Order")
let sortDescriptor = NSSortDescriptor(key: "date", ascending: true)
let predicate = NSPredicate(format: "date >= %@ AND date <= %@", startDate, endDate) // startDate and endData are defined elsewhere

fetchRequest.sortDescriptors = [sortDescriptor]
fetchRequest.predicate = predicate

return fetchRequest
}

func fetchData() {
let fetchRequest = orderFetchRequest()
fetchedResultController = NSFetchedResultsController(fetchRequest: fetchRequest, managedObjectContext: managedObjectContext, sectionNameKeyPath:nil, cacheName: nil)

do {
try fetchedResultController.performFetch()
}
catch let error as NSError {
print("Could not fetch \(error), \(error.userInfo)")
}
}
}

可能有更简洁的编码方式;我不是 Swift 的狂热爱好者。

如果您使用 FRC 委托(delegate)方法,您至少可以观察新的/已删除的订单并根据需要向 TV 添加/删除部分,并且您可以使用更新来重新加载相关部分。

关于swift - 在 Swift 中使用核心数据在 TableView 中显示部分数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33653096/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com