12

将 prefersLargeTitles 用于 aUINavigationControllerUINavigationBariOS 11导航栏会增加高度。我检查过的 iPhone 从 44 增加到 96,但我认为这些数字可能会因设备而异(或者至少我们需要像可以编码一样进行编码)。

我想以编程方式找到“额外”高度 -UINavigationBar显示大标题时在传统下方添加的大标题区域的高度。我可以很容易地找到显示大标题的栏的整个高度,但是有没有办法以编程方式单独找到栏的大标题部分的高度(没有任何硬编码)?

我需要这个的原因是有时我想以编程方式滚动到 UITableView 的顶部,拉下大标题(它在“常规高度”导航栏下向上滚动)以便它显示,并且我需要的内容偏移是导航栏的额外高度。我可以使用导航栏的总高度,但这会将 UITableView 拉得太低。现在要做到这一点,我需要硬编码如下:

[self.tableView setContentOffset:CGPointMake(0, -52) animated:NO];
4

5 回答 5

10

我认为您将找不到明确解决问题的方法。您要求的是导航栏内部的一部分,它不向用户公开。但是,我想我可以为您提供一种解决方法。

为了理解这一点,让我们看一下视图调试器中的导航栏(启用了大标题)。正如您在下图中看到的,有一个额外的视图,其中包含特定视图控制器的大标题。当您不支持大标题时,此视图不存在。

基本上,您想知道的是该视图的高度。

导航栏的调试器预览

这是我提出的解决方案/解决方法

extension UINavigationBar
{
    var largeTitleHeight: CGFloat {
        let maxSize = self.subviews
            .filter { $0.frame.origin.y > 0 }
            .max { $0.frame.origin.y < $1.frame.origin.y }
            .map { $0.frame.size }
        return maxSize?.height ?? 0
    }
}

它能做什么

  1. 过滤掉从导航栏顶部开始的子视图。它们很可能是背景或导航区域,我们不需要它们。
  2. 在导航栏中查找最低的子视图。
  3. 将生成的子视图帧映射到帧大小。
  4. 返回找到的子视图的高度,如果没有找到,则返回 0。

这种方法的明显缺点是它与导航栏的结构紧密耦合,未来可能会发生变化。但是,您不太可能得到不止一个或另一个肮脏的把戏。

执行结果应如下所示。

print("Extra height: \(navigationController!.navigationBar.lagreTitleHeight)")
Extra height: 52.0
于 2017-10-27T21:45:09.267 回答
1

据我了解,您的 tableView 原点位于导航栏下,位于 UIViewController 视图中的 y = 0 处。

您的 tableView 应该将其顶部绑定修复到顶部布局指南,或者使用新的安全区域。这样您就不必以编程方式计算导航栏的大小。

如果您从未使用过它,请查看自动布局。

于 2017-10-26T15:22:24.660 回答
0

您可以简单地滚动到表格视图的顶部,而无需知道导航栏的大小。UITableView附带了一个 API。

选项1

// Get the first index path. You might want to check here whether data source item exists.
NSIndexPath *firstIndexPath = [NSIndexPath indexPathForRow:0
                                                 inSection:0];
// Scroll to it. You can play with scroll position to get what you want
[tableView scrollToRowAtIndexPath:firstIndexPath
                 atScrollPosition:UITableViewScrollPositionTop
                         animated:YES];

选项 2

CGPoint offset = CGPointMake(0.0, -tableView.contentInset.top);
[tableView setContentOffset:offset animated:YES];
于 2017-10-23T10:03:47.837 回答
0

我刚刚遇到了同样的问题。

 -(void)layoutSubviews{
            [super layoutSubviews];
            CGRect rectStatusBar = [[UIApplication sharedApplication] statusBarFrame];
            if (rectStatusBar.size.height==44.f) {
               //statusBar shows
            }else{
                if (@available(iOS 11.0, *)) {
                    for ( UIView*view in self.subviews) {
                        if ([NSStringFromClass(view.classForCoder) isEqualToString:@"_UINavigationBarContentView"]) {
                            view.frame = CGRectMake( 0,20,view.frame.size.width,44);
                        }
                        else if ([NSStringFromClass(view.classForCoder) isEqualToString:@"_UIBarBackground"]) {
                            view.frame = CGRectMake(0,0,view.frame.size.width, 64);
                        }
                    }
                }
            }
        }
于 2017-10-25T06:02:52.107 回答
0

似乎调用会navigationBar.sizeToFit()强制navigationBar调整其大小,因为它显示大标题。因此,您可以轻松计算顶部安全区域。我们提出的解决方案是下一个:

self.navigationController?.navigationBar.sizeToFit()
let navigationBarOffset = navigationController?.navigationBar.frame.origin.y ?? 0
let navigationBarHeight = navigationController?.navigationBar.frame.height ?? 0
let offset = -navigationBarOffset - navigationBarHeight
if collectionView.contentOffset.y > offset {
let contentOffset = CGPoint(x: 0, y: offset)
   collectionView.setContentOffset(contentOffset, animated: true)
}
于 2021-07-30T15:34:59.193 回答