隐藏在模型未表示的可见单元格后面的 TableViews 单元格

Posted

技术标签:

【中文标题】隐藏在模型未表示的可见单元格后面的 TableViews 单元格【英文标题】:Hidden TableViews cells behind visible cells that are not represented by models 【发布时间】:2019-05-17 08:30:32 【问题描述】:

我目前正在为客户构建一个聊天机器人应用程序。 聊天是使用带有自定义单元格的 UITableView 实现的。 来自 bot 的文本消息构建为自定义 UITableViewCell,带有 UITextView(因为它需要可点击链接,否则我会使用 UILabel)包裹在 UIView 容器中以在其周围绘制气泡。这个容器视图也有一个阴影来取悦设计师:P

每次收到来自 bot 的新消息时,此消息都会延迟一小段时间,以便用户之前可以阅读该消息。在那个时候,一个所谓的 WaitingMessageTableViewCell 显示在索引处,新单元格将到达。时间到期后,WaitingMessageTableViewCell 将从模型中删除,然后从 tableView 中删除,并插入新单元格。

            tableView.beginUpdates()
            if let insertRows = insertRows 
                tableView.insertRows(at: insertRows, with: .none)
            
            if let deleteRows = deleteRows 
                tableView.deleteRows(at: deleteRows, with: .none)
            
            if let reloadRows = reloadRows 
                tableView.reloadRows(at: reloadRows, with: .none)
            
            tableView.endUpdates()

设计者发现问题是,当WaitingMessageTableViewCell即将出现时,之前的单元格的阴影会在一小段时间内变强。我调试了一下,发现在插入WaitingMessageTableViewCell之前它出现在之前的单元格后面,这对我来说没有意义。(是重用机制吗?!) 现在到了非常奇怪的部分:我使用了 Xcode 中的 UI 分析器,发现在可见单元格后面有隐藏的单元格,这些单元格没有被模型表示。

(lldb) po tableView.subviews
▿ 5 elements
  ▿ 0 : <CHAT.WaitingMessageTableViewCell: 0x7fdef286a600; baseClass = UITableViewCell; frame = (0 395.5; 375 40); hidden = YES; opaque = NO; autoresize = W; layer = <CALayer: 0x600003952d80>>
  ▿ 1 : <CHAT.AgentMessageTableViewCell: 0x7fdef2861e00; baseClass = UITableViewCell; frame = (0 294.5; 375 101); hidden = YES; opaque = NO; autoresize = W; layer = <CALayer: 0x6000039bf160>>
  ▿ 2 : <CHAT.AgentMessageTableViewCell: 0x7fdef3897600; baseClass = UITableViewCell; frame = (0 395.5; 375 101.5); opaque = NO; autoresize = W; layer = <CALayer: 0x6000039aba20>>
  ▿ 3 : <CHAT.AgentMessageTableViewCell: 0x7fdef382e200; baseClass = UITableViewCell; frame = (0 294.5; 375 101); opaque = NO; autoresize = W; layer = <CALayer: 0x600003955a00>>
  ▿ 4 : <CHAT.WelcomeTableViewCell: 0x7fdef2882c00; baseClass = UITableViewCell; frame = (0 0; 375 294.5); clipsToBounds = YES; autoresize = W; layer = <CALayer: 0x600003951160>>
(lldb) po tableView.visibleCells
▿ 3 elements
  ▿ 0 : <CHAT.WelcomeTableViewCell: 0x7fdef2882c00; baseClass = UITableViewCell; frame = (0 0; 375 294.5); clipsToBounds = YES; autoresize = W; layer = <CALayer: 0x600003951160>>
  ▿ 1 : <CHAT.AgentMessageTableViewCell: 0x7fdef382e200; baseClass = UITableViewCell; frame = (0 294.5; 375 101); opaque = NO; autoresize = W; layer = <CALayer: 0x600003955a00>>
  ▿ 2 : <CHAT.AgentMessageTableViewCell: 0x7fdef3897600; baseClass = UITableViewCell; frame = (0 395.5; 375 101.5); opaque = NO; autoresize = W; layer = <CALayer: 0x6000039aba20>>
(lldb) po AssistantDataManager.shared.cellModels
▿ 3 elements
  ▿ 0 : <WelcomeCellModel: 0x600002c80740>
  ▿ 1 : <SimpleAgentTextCellModel: 0x600001af0050>
  ▿ 2 : <AskQuestionCellModel: 0x600002c80400>

这也是我的 cellForRows 方法:

    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell 

        guard indexPath.row < AssistantDataManager.shared.cellModels.count else 
            return UITableViewCell()
        
        let cellModel = AssistantDataManager.shared.cellModels[indexPath.row]

        if let cell = tableView.dequeueReusableCell(withIdentifier: cellModel.cellReuseIdentifier) as? AssistantActionBaseTableViewCell 
            //Quickfix for stronger shadow
            cell.backgroundColor = Colors.tableViewBackgroundColor

            cell.configure(with: cellModel)
            if let topContraint = cell.topContraint 
                topContraint.constant = calculateTopContraintSpacing(for: indexPath)
            

            if let bottomContraint = cell.bottomContraint 
                bottomContraint.constant = 0
            

            cell.selectionStyle = .none
            return cell
        
        return UITableViewCell()
    

谁能告诉我为什么新插入的 WaitingMessageTableViewCell 首先在上面的单元格后面然后向下移动,为什么我的 tableView 填充了比它需要的更多的单元格?

非常感谢您。

【问题讨论】:

能否说明calculateTopContraintSpacing(for: indexPath)的用途 此方法计算使用顶部约束实现的单元格之间的间距。单元格的间距取决于它之前的单元格的类型。 【参考方案1】:

我发现了为什么新单元格首先显示在前一个单元格的后面,然后向下移动。它是隐式动画和生命周期的结合。

以前我在一次批量更新中插入删除并重新加载了所有单元格。由于隐式动画,tableView 决定为该更改设置动画。但是当我第一次插入和删除,然后重新加载单元格时,一切都按预期工作。

tableView.performBatchUpdates(
    if let insertRows = insertRows 
        tableView.insertRows(at: insertRows, with: .none)
    
    if let deleteRows = deleteRows 
        tableView.deleteRows(at: deleteRows, with: .none)
    

)  (finished) in
    self.tableView.performBatchUpdates(
        if let reloadRows = reloadRows 
            self.tableView.reloadRows(at: reloadRows, with: .none)
        
    , completion: nil)

【讨论】:

以上是关于隐藏在模型未表示的可见单元格后面的 TableViews 单元格的主要内容,如果未能解决你的问题,请参考以下文章

调用 reloadSections:withRowAnimation 后,使用 IB 创建的自定义单元格被隐藏:

保持某些单元格高度不变,同时更改其他单元格高度

为啥设置“可见性:隐藏”时表格单元格边框在谷歌浏览器中消失和“边界崩溃:崩溃;”?

UITableView 中的顶部单元格隐藏在 UINavigationBar 后面

UICollectionView 隐藏/防止单元格在 Sticky Header 后面滚动

调整 TableView 大小时,Swift 3.0 表格单元格未填充