reloadData() 不更新单元格标签
Posted
技术标签:
【中文标题】reloadData() 不更新单元格标签【英文标题】:reloadData() doesn't update a cell label 【发布时间】:2017-02-08 17:28:44 【问题描述】:在我的应用程序中,我有一个标签通知用户缺少一些设置:
设置在:
override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell
let service = (fetchedResultsController?.object(at: indexPath))! as Service
let cell = tableView.dequeueReusableCell(withIdentifier: cellId, for: indexPath) as! OfferedServiceCell
cell.setupDataCell(service: service)
if service.employee?.count == 0 || service.price?.price == nil || service.price?.slotDuration == nil
cell.infoLabel.text = "attention: you need to complete this service set up!"
let lpGestureRecognizer: UILongPressGestureRecognizer = UILongPressGestureRecognizer(target: self, action: #selector(didLongPressCell(gr:)))
cell.addGestureRecognizer(lpGestureRecognizer)
return cell
用户可以在另一个 tableViewController 中添加价格和员工证明该服务,当所有需要的信息都保存在核心数据中时,标签应该消失。问题是标签红色警告只有在我重新加载视图控制器时才会消失。
这是我迄今为止尝试过的:
由于我使用的是 fetchedResultsController,所以我实现了这个方法:
func controllerDidChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>)
DispatchQueue.main.async
self.tableView.reloadData()
一旦我更改核心数据,该方法就会被调用,但警告仍然存在(除非重新加载 TableViewController
)。我还尝试在OfferedServiceCell
中添加self.tableView.layoutIfNeeded()
和infoLabel.layoutIfNeeded()
。
当我添加价格和员工的TableViewController
被解雇时发布通知:
NotificationCenter.default.addObserver(self, selector: #selector(reloadAllData), name: NSNotification.Name(rawValue: "priceListModified"), object: nil)
在reloadAllData
,我什至尝试再次致电performFetch
func reloadAllData()
DispatchQueue.main.async
do
try self.fetchedResultsController?.performFetch()
catch
fatalError("Failed to initialize FetchedResultsController: \(error)")
self.tableView.reloadData()
self.tableView.layoutIfNeeded()
我不知道我还能做什么。谢谢你的建议。
-----UPDATE---问题解决了:
正如 Chris Trahey 的回答正确指出的那样,我最终将设置红色标签的代码移到了我的自定义单元格中,并在不需要时明确隐藏它:
func setupDataCell(service:Service)
self.service = service
if service.service_description != nil
serviceName.text = service.service_description
if service.employee?.count == 0 || service.price?.price == nil || service.price?.slotDuration == nil
if service.selected == 0
infoLabel.text = "Drag service to Service Offered to start using it"
else
infoLabel.text = "attention: you need to complete this service set up!"
else
infoLabel.isHidden = true
我使用 prepareForReuse 将标签设置为 nil。 (仍然不确定这一点,但没有它就无法工作)
override func prepareForReuse()
super.prepareForReuse()
infoLabel.text = nil
【问题讨论】:
【参考方案1】:我怀疑单元设置例程在不需要时不会清除标签。因此,标签在重复使用周期中幸存下来。我会做这两件事:
setupDataCell
不应只在需要时设置红色标签属性,还应在不需要时显式隐藏它。
在自定义单元格类上实现prepareForReuse
并清除标签
【讨论】:
非常感谢。我只需将设置红色标签的代码移动到setupDataCell
并在一切设置正确的情况下隐藏标签。我还要使用prepareForReuse
吗?
prepareForReuse
是这样的逻辑的好地方,但如果你已经有一种强大的方法来设置单元格,那么它并不是绝对必要的。如果您有一个复杂的单元格,它可以帮助清理您的代码。让您的 prepareForReuse
方法专注于进入“默认状态”,让您的设置方法专注于根据当前模型项更新的内容。
我最终使用了 prepareForReuse,我将文本设置为 nil。原因还不是很清楚,但是如果没有那段代码,标签就不会更新。因为OfferedServiceCell
是ServiceCell
的子类,所以我也必须覆盖setupDataCell
。感谢您的帮助以上是关于reloadData() 不更新单元格标签的主要内容,如果未能解决你的问题,请参考以下文章
UITableView, reloadData 重新加载单元格但不调整表格大小
Swift UICollectionView 在单击时更新单元格标签而不是实时更新
UICollectionView reloadData() 不会更新集合视图中的单元格