ios - UITableViewCell 内的可滚动 StackView - ScrollView ContentSize 未更新

标签 ios swift uitableview uiscrollview uistackview

我最近开始学习 swift 和 iOS 应用程序开发。到目前为止,我一直在做 php 后端和底层 iOS/macOS 编程,使用 UI 对我来说有点困难,所以请容忍我的愚蠢。

如果我理解正确的话,stackviews会自动在其框架中间隔并包含其 subview 。所有的数学和布局都是由它自动完成的。我在自定义 UITableViewCell 内有一个水平堆栈 View 。 UIStackView 位于 UIScrollView 内,因为我希望内容可以滚动。我已经以编程方式设置了 anchor (我只是不知道如何使用 Storyboard)。 This is what the cells look like

当我加载 View 时,堆栈 View 不滚动。但如果我至少选择该单元格一次,它就会滚动。 ScrollView 的 contentSize 是在我的自定义单元格的layoutsubviews 方法内设置的。

我的自定义单元

class TableViewCell: UITableViewCell
{
    let stackViewLabelContainer  = UIStackView()
    let scrollViewContainer = UIScrollView()

    override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?)
    {
        super.init(style: style, reuseIdentifier: reuseIdentifier)
        backgroundColor = .black
        stackViewLabelContainer.axis = .horizontal
        stackViewLabelContainer.distribution = .equalSpacing
        stackViewLabelContainer.alignment = .leading
        stackViewLabelContainer.spacing = 5
        for _ in 1...10
        {
            let labelView = UILabel();
            labelView.backgroundColor = tintColor
            labelView.textColor = .white
            labelView.text = "ABCD 123"

            stackViewLabelContainer.addArrangedSubview(labelView)
        }
        scrollViewContainer.addSubview(stackViewLabelContainer)
        stackViewLabelContainer.translatesAutoresizingMaskIntoConstraints = false
        stackViewLabelContainer.leadingAnchor.constraint(equalTo: scrollViewContainer.leadingAnchor).isActive = true
        stackViewLabelContainer.topAnchor.constraint(equalTo: scrollViewContainer.topAnchor).isActive = true
        addSubview(scrollViewContainer)
        scrollViewContainer.translatesAutoresizingMaskIntoConstraints = false
        scrollViewContainer.leadingAnchor.constraint(equalTo: leadingAnchor, constant: 10).isActive = true
        scrollViewContainer.topAnchor.constraint(equalTo: topAnchor, constant: 5).isActive = true
        scrollViewContainer.heightAnchor.constraint(equalTo:stackViewLabelContainer.heightAnchor).isActive = true
        scrollViewContainer.trailingAnchor.constraint(equalTo: trailingAnchor).isActive = true
        scrollViewContainer.showsHorizontalScrollIndicator = false



    }

    required init?(coder: NSCoder)
    {
        fatalError("init(coder:) has not been implemented")
    }

    override func layoutSubviews()
    {
        super.layoutSubviews()
        scrollViewContainer.contentSize = CGSize(width: stackViewLabelContainer.frame.width, height: stackViewLabelContainer.frame.height)

    }





}

这是 TableViewController

class TableViewController: UITableViewController {

    override func viewDidLoad()
    {
        super.viewDidLoad()
        tableView.register(TableViewCell.self, forCellReuseIdentifier: "reuse_cell")

    }

    override func numberOfSections(in tableView: UITableView) -> Int
    {
        return 1
    }

    override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int
    {
        return 5
    }

    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell
    {
        let cell = tableView.dequeueReusableCell(withIdentifier: "reuse_cell") as! TableViewCell
        return cell
    }

    override func viewDidLayoutSubviews()
    {
        print("called")
        super.viewDidLayoutSubviews()
//        let cells  = tableView.visibleCells as! Array<TableViewCell>
//        cells.forEach
//        {
//            cell in
//            cell.scrollViewContainer.contentSize = CGSize(width: cell.stackViewLabelContainer.frame.width, height: cell.stackViewLabelContainer.frame.height)
//
//        }
    }

}

我找到了一种方法来完成这项工作,但它会影响抽象,感觉就像一个奇怪的黑客。您可以从 UITableViewController 中获取可见单元格,访问每个 ScrollView 并更新其 contentSize。我通过反转 dyld_shared_cache 发现了另一个修复,我在其中重写 draw 方法并停止重用单元格。这两种解决方案都感觉离“正确”还很远。

最佳答案

您应该将 ScrollView 限制为单元格的 contentView。

contentView.addSubview(scrollView)
scrollView.translatesAutoresizingMaskIntoConstraints = false
scrollView.topAnchor.constraint(equalTo: contentView.safeAreaLayoutGuide.topAnchor).isActive = true
contentView.bottomAnchor.constraint(equalTo: scrollView.bottomAnchor).isActive = true
scrollView.leadingAnchor.constraint(equalTo: contentView.safeAreaLayoutGuide.leadingAnchor).isActive = true
scrollView.trailingAnchor.constraint(equalTo: contentView.safeAreaLayoutGuide.trailingAnchor).isActive = true
scrollView.addSubview(stackView)

stackView.translatesAutoresizingMaskIntoConstraints = false
stackView.topAnchor.constraint(equalTo: scrollView.topAnchor).isActive = true
stackView.leadingAnchor.constraint(equalTo: scrollView.leadingAnchor).isActive = true
stackView.trailingAnchor.constraint(equalTo: scrollView.trailingAnchor).isActive = true
stackView.bottomAnchor.constraint(equalTo: scrollView.bottomAnchor).isActive = true
stackView.widthAnchor.constraint(equalTo: scrollView.widthAnchor).isActive = true

现在您可以循环标签并将它们添加为排列的 subview

for _ in 1...10
{
                let labelView = UILabel();
                labelView.backgroundColor = tintColor
                labelView.textColor = .white
                labelView.text = "ABCD 123"

                stackView.addArrangedSubview(labelView)
 }

关于ios - UITableViewCell 内的可滚动 StackView - ScrollView ContentSize 未更新,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59426804/

相关文章:

ios - 如何在iOS中清除文本字段数据后隐藏 View

ios - UITextView,编辑时滚动?

ios - 使用 viewDidAppear 自动登录不利用导航 Controller

ios - 如何从 Storyboard创建的 UITableView 中拉出 UIView?

ios - 检查负值 iOS

ios - 如何在按下完成时关闭 SwiftUI 键盘中的键盘?

swift - 在 Swift 类中使用指向对象的指针

将 JSON 响应数据转换为 swift 对象 [ObjectMapper]

ios - 动态单元格高度 iOS

iphone - 将 UILabel 转换为具有相同大小和对齐方式的 NSString