【问题标题】:reloadData() doesn't update a cell labelreloadData() 不更新单元格标签
【发布时间】: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 中添加价格和员工证明该服务,当所有需要的信息都保存在核心数据中时,标签应该消失。问题是标签红色警告只有在我重新加载视图控制器时才会消失。

这是我迄今为止尝试过的:

  1. 由于我使用的是 fetchedResultsController,所以我实现了这个方法:

    func controllerDidChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>) {
        DispatchQueue.main.async {
            self.tableView.reloadData()
        }
    }
    

    一旦我更改核心数据,该方法就会被调用,但警告仍然存在(除非重新加载 TableViewController)。我还尝试在OfferedServiceCell 中添加self.tableView.layoutIfNeeded()infoLabel.layoutIfNeeded()

  2. 当我添加价格和员工的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

}

【问题讨论】:

    标签: ios swift uitableview core-data


    【解决方案1】:

    我怀疑单元设置例程在不需要时不会清除标签。因此,标签在重复使用周期中幸存下来。我会做这两件事:

    • setupDataCell 不应只在需要时设置红色标签属性,还应在不需要时显式隐藏它。
    • 在自定义单元格类上实现prepareForReuse并清除标签

    【讨论】:

    • 非常感谢。我只需将设置红色标签的代码移动到setupDataCell 并在一切设置正确的情况下隐藏标签。我还要使用prepareForReuse 吗?
    • prepareForReuse 是这样的逻辑的好地方,但如果你已经有一种强大的方法来设置单元格,那么它并不是绝对必要的。如果您有一个复杂的单元格,它可以帮助清理您的代码。让您的 prepareForReuse 方法专注于进入“默认状态”,让您的设置方法专注于根据当前模型项更新的内容。
    • 我最终使用了 prepareForReuse,我将文本设置为 nil。原因还不是很清楚,但是如果没有那段代码,标签就不会更新。因为OfferedServiceCellServiceCell 的子类,所以我也必须覆盖setupDataCell。感谢您的帮助
    猜你喜欢
    • 2020-03-14
    • 2018-11-22
    • 2014-11-15
    • 1970-01-01
    • 1970-01-01
    • 2018-12-12
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多