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。原因还不是很清楚,但是如果没有那段代码,标签就不会更新。因为OfferedServiceCellServiceCell 的子类,所以我也必须覆盖setupDataCell。感谢您的帮助

以上是关于reloadData() 不更新单元格标签的主要内容,如果未能解决你的问题,请参考以下文章

UITableView, reloadData 重新加载单元格但不调整表格大小

Swift UICollectionView 在单击时更新单元格标签而不是实时更新

ios reloadData 仅重新加载不可见的单元格

UICollectionView reloadData() 不会更新集合视图中的单元格

.reloadData() 没有重新创建可见单元格或内存中的单元格?

点击/ reloadData不更改单元格时TableView单元格消失?