UITableView 自定义部分页脚

Posted

技术标签:

【中文标题】UITableView 自定义部分页脚【英文标题】:UITableView custom section footer 【发布时间】:2016-11-14 14:20:48 【问题描述】:

我们确实有一个TableViewModel,它正在实现UITableViewDelegateUITableViewDataSource。它保存部分并返回我们的UITableViews 的正确行和元素。

现在我们想要有一个自定义的表格视图部分页脚。因此,我们实现了optional public func tableView(_ tableView: UITableView, viewForFooterInSection section: Int) -> UIView?,以便从xib 文件中返回UIView。这一切都很好,但最终导致页脚高度错误。如果我们使用func tableView(_ tableView: UITableView, titleForFooterInSection section: Int) -> String?,页脚显示正确,页脚高度设置正确。

所以我们还必须实现optional public func tableView(_ tableView: UITableView, estimatedHeightForFooterInSection section: Int) -> CGFloatoptional public func tableView(_ tableView: UITableView, heightForFooterInSection section: Int) -> CGFloat

    问题:如何获得实例化的 UIView 的高度? view.frame.boundsview.frame.height 始终为 0。 问题:Apple 为页脚使用的默认值是多少?它是如何计算的?我认识到,如果我们返回 table.sectionFooterHeight 作为默认值(不是我们的自定义页脚视图),那么高度是正确的。

TableViewModel:

class TableViewModel: NSObject, UITableViewDelegate, UITableViewDataSource 

    var sections: [TableViewSection] = []

    func numberOfSections(in tableView: UITableView) -> Int 
        return sections.count
    

    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int 
        return sections[section].numberOfRowsInTableView(tableView)
    

    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell 
        return sections[(indexPath as NSIndexPath).section].tableView(tableView, cellForRowAtIndexPath: indexPath)
    

    func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) 
        sections[(indexPath as NSIndexPath).section].tableView(tableView, didSelectRowAtIndexPath: indexPath)
    

    func tableView(_ tableView: UITableView, titleForHeaderInSection section: Int) -> String? 
        return sections[section].headerText
    

    func tableView(_ tableView: UITableView, willDisplayHeaderView view: UIView, forSection section: Int) 
        sections[section].tableView(tableView, willDisplayHeaderView: view)
    

    func tableView(_ tableView: UITableView, titleForFooterInSection section: Int) -> String? 
        return sections[section].footerText
    

    func tableView(_ tableView: UITableView, willDisplayFooterView view: UIView, forSection section: Int) 
        sections[section].tableView(tableView, willDisplayFooterView: view)
    

    func tableView(_ tableView: UITableView, shouldShowMenuForRowAt indexPath: IndexPath) -> Bool 
        return sections[(indexPath as NSIndexPath).section].tableView(tableView, cellContentsToCopyAtIndexPath: indexPath, withSender: nil) != nil
    

    func tableView(_ tableView: UITableView, heightForFooterInSection section: Int) -> CGFloat 
        ??
    

    func tableView(_ tableView: UITableView, viewForFooterInSection section: Int) -> UIView? 
        return sections[section].footerView
    

    func tableView(_ tableView: UITableView, estimatedHeightForFooterInSection section: Int) -> CGFloat 
        ??
    

【问题讨论】:

您在设置 titleForFooterInSection 时是否尝试将身高保存在变量中?类似于: footerHeight = section[section].footerText.frame.bounds.height ,其中 footerHeight 是控制器中的变量,您可以将其用作“heightForFooterInSection”的返回值 你的页脚有固定的高度吗?还是它们的高度是动态的并由 AutoLayout 设置? 是的,但这意味着我必须在创建页脚时设置高度。我不知道它在每台设备上的高度(包含字符串,因此动态高度取决于手机大小)不知何故苹果的 interfacebuilder 确实在默认视图上设置了高度。 【参考方案1】:

我通过添加以下方法解决了它:

func tableView(_ tableView: UITableView, heightForFooterInSection section: Int) -> CGFloat 
    return sections[section].footerViewHeight ?? UITableViewAutomaticDimension

并返回节中页脚的计算高度。它确实可以正确调整大小,我们可以对其进行调整。如前所述,返回 UITableViewAutomaticDimension 需要:

// Returning this value from tableView:heightForHeaderInSection: or tableView:heightForFooterInSection: results in a height that fits the value returned from // tableView:titleForHeaderInSection: or tableView:titleForFooterInSection: if the title is not nil.

【讨论】:

【参考方案2】:

如果您使用自动布局,您仍然可以在heightForFooterInSection 中返回UITableViewAutomaticDimension,并且将自动计算页脚的高度。与heightForRowAt 相同,heightForFooterInSection 必须与estimatedHeightForFooterInSection 结合使用。一个简单的示例如下所示:

func tableView(_ tableView: UITableView, heightForFooterInSection section: Int) -> CGFloat 
    return UITableViewAutomaticDimension


func tableView(_ tableView: UITableView, estimatedHeightForFooterInSection section: Int) -> CGFloat 
    return 18 // or whatever value

【讨论】:

感谢您的回答。我会试一试,看看它是如何工作的。 Wi 提供反馈。 // Returning this value from tableView:heightForHeaderInSection: or tableView:heightForFooterInSection: results in a height that fits the value returned from // tableView:titleForHeaderInSection: or tableView:titleForFooterInSection: if the title is not nil. @BennX 你从哪里得到这个?在我的一个项目中,这是按预期工作的 如果你按照 Xcode 来定义 UITableViewAutomaticDimension 那里写的。 @BennX 如果titleForFooterInSection 返回nil 并且estimatedHeightForFooterInSection 已实现,我可以确定UITableViewAutomaticDimensionheightForFooterInSection 一起使用。还要确保您已按照此处所述正确设置所有约束:***.com/a/30300870/1422333

以上是关于UITableView 自定义部分页脚的主要内容,如果未能解决你的问题,请参考以下文章

从自定义单元格返回 NSIndexPath ? (UITableView)

UITableView 原型单元上的自定义附件视图未触发

如何在自定义单元格的 initWithStyle: 方法中获取 UITableView 的宽高?

调整 uitableview 部分页脚的高度

返回正确大小的页脚时,部分页脚中的 UITableView 随机白线

出现键盘时,UITableView 的部分页脚正在移动