隐藏在模型未表示的可见单元格后面的 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 后面