在 UICollectionView 或 UITableView 中实现粘性单元格

问题描述

我要实现一个包含项目列表的表格,其中包含一个应始终显示在屏幕上的项目。因此,例如:

  • 列表中有 50 个项目
  • 您的“粘性”列表项是第 25 个
  • 您可以同时在屏幕上显示 10 个项目
  • 尽管您位于列表中,“粘性”列表应始终保持可见
  • 如果您的项目低于您在列表中的位置,它会显示在列表底部
  • 如果您的项目在之前的项目之间,它应该显示在列表的顶部
  • 一旦您到达列表中项目的实际位置,它应该与列表的滚动一起移动

以下插图可以更好地理解实施要求:

3 screens. On 1st screen,25th cell is stuck at bottom,overlapping other cells underneath. On 2nd screen,25th screen is inline with the other cells. On 3rd screen,25th cell is stuck at the top.

很高兴就如何实现这一点提出任何可能的想法、建议或建议。不幸的是,我没有找到任何有用的库或解决方案来解决这个问题。 UICollectionView 和 UITableView 在这种情况下都是可以接受的。

根据我的理解,粘性页眉或页脚在这种情况下不起作用,因为它们仅涵盖了我需要的一半功能

预先感谢您的评论和回答!!!

解决方法

我很确定您实际上不能让相同的实际单元格像那样粘。不过,您可以通过自动布局技巧来创造粘性的错觉。基本上,我的建议是,您可以拥有与您想要“粘性”的单元格相同的视图,并在您的粘性单元格可见时将它们限制在您的粘性单元格顶部。如果你慢慢滚动,我能做到的最好的看起来并不完美。 (粘性单元格在捕捉到顶部或底部位置之前大部分时间会离开屏幕。在我看来,在相当正常的滚动速度下并不明显。您的里程可能会有所不同。)

关键是设置一个表格视图委托,以便您可以在单元格何时出现或不出现在屏幕上时收到通知。

我已经包含了一个示例视图控制器。我确信我的示例代码在某些地方不起作用。 (例如,我没有处理堆叠多个“粘性”单元格或动态行高。此外,我将粘性单元格设为蓝色,以便更容易看到粘性。)

为了运行示例代码,如果您创建一个新的 UIKit 应用程序,您应该能够将其粘贴到 Xcode 生成的默认项目中。只需用这个替换他们给你的视图控制器,看看它的效果。

import UIKit

struct StickyView {
    let view: UIView
    let constraint: NSLayoutConstraint
}

class ViewController: UIViewController,UITableViewDataSource,UITableViewDelegate {
    lazy var stickyViewConstraints = [Int: StickyView]()
    
    lazy var tableView: UITableView = {
        let table = UITableView()
        table.translatesAutoresizingMaskIntoConstraints = false
        table.register(UITableViewCell.self,forCellReuseIdentifier: "cell")
        table.rowHeight = 40
        table.dataSource = self
        table.delegate = self
        return table
    }()

    override func viewDidLoad() {
        super.viewDidLoad()
        // Do any additional setup after loading the view.
        addTable()
        setupStickyViews()
    }
    
    private func addTable() {
        view.addSubview(tableView)
        tableView.topAnchor.constraint(equalTo: view.topAnchor).isActive = true
        tableView.leadingAnchor.constraint(equalTo: view.leadingAnchor).isActive = true
        tableView.trailingAnchor.constraint(equalTo: view.trailingAnchor).isActive = true
        tableView.bottomAnchor.constraint(equalTo: view.bottomAnchor).isActive = true
    }
    
    private func setupStickyViews() {
        let cell25 = UITableViewCell()
        cell25.translatesAutoresizingMaskIntoConstraints = false
        cell25.backgroundColor = .blue
        cell25.textLabel?.text = "25"
        view.addSubview(cell25)
        cell25.leadingAnchor.constraint(equalTo: view.leadingAnchor).isActive = true
        cell25.widthAnchor.constraint(equalTo: view.widthAnchor).isActive = true
        cell25.heightAnchor.constraint(equalToConstant: tableView.rowHeight).isActive = true
        
        let bottom = cell25.bottomAnchor.constraint(equalTo: view.safeAreaLayoutGuide.bottomAnchor)
        bottom.isActive = true
        stickyViewConstraints[25] = StickyView(view: cell25,constraint: bottom)
    }

    // MARK: - Data Source
    func tableView(_ tableView: UITableView,numberOfRowsInSection section: Int) -> Int {
        return section == 0 ? 50 : 0
    }
    
    func tableView(_ tableView: UITableView,cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "cell",for: indexPath)
        cell.textLabel?.text = "\(indexPath.row)"
        return cell
    }
    
    // MARK: - Delegate
    func tableView(_ tableView: UITableView,didEndDisplaying cell: UITableViewCell,forRowAt indexPath: IndexPath) {
        guard let stickyView = stickyViewConstraints[indexPath.row] else { return }
        stickyView.constraint.isActive = false
        var verticalConstraint: NSLayoutConstraint
        if shouldPlaceStickyViewAtTop(stickyRow: indexPath.row) {
            verticalConstraint = stickyView.view.topAnchor.constraint(equalTo: view.topAnchor)
        } else {
            verticalConstraint = stickyView.view.bottomAnchor.constraint(equalTo: view.bottomAnchor)
        }
        verticalConstraint.isActive = true
        stickyViewConstraints[indexPath.row] = StickyView(view: stickyView.view,constraint: verticalConstraint)
    }
    
    private func shouldPlaceStickyViewAtTop(stickyRow: Int) -> Bool {
        let visibleRows = tableView.indexPathsForVisibleRows?.map(\.row)
        guard let min = visibleRows?.min() else { return false }
        return min > stickyRow
    }
    
    func tableView(_ tableView: UITableView,willDisplay cell: UITableViewCell,forRowAt indexPath: IndexPath) {
        if let stickyView = stickyViewConstraints[indexPath.row] {
            stickyView.constraint.isActive = false
            let bottom = stickyView.view.bottomAnchor.constraint(equalTo: cell.bottomAnchor)
            bottom.isActive = true
            stickyViewConstraints[indexPath.row] = StickyView(view: stickyView.view,constraint: bottom)
        }
    }
}