zoukankan      html  css  js  c++  java
  • iOS导航标题不居中问题(转载)

    前言

    一直以来都让我很头痛的一个问题:系统自带的导航条,在标题文字很长时,进入到下一个界面,而下一个界面的标题也很长时,就会出现标题不居中显示。

    曾经,我尝试过很多种办法,但是都没有从根上解决问题。下面笔者分别说说用过哪些方案。

    方案一(不可行)

    这个方案是不使用系统自带默认的backButtonItem,而是使用leftBarButtonItem

    这样做的好处是:解决了本界面标题过长,而上一个界面的标题也很长时,本界面的标题不居中显示的问题。

    这样做的坏处是:系统自带的右滑返回手势就没有了。

    方案二(不可行)

    这里我们自定义一个继承于UINavigationController的类,然后所有使用导航类的地方都使用我们所定义的导航控制器类。在这个类中,我们通过配置全局的导航条相关全局属性

     1 - (void)config {
     2   NSString *backImageName = @"default_back";
     3   
     4   if (kIsIOS7OrLater) {
     5     UIImage *image = [UIImage imageNamed:backImageName];
     6     CGFloat w = image.size.width;
     7     if (kScreenWidth <= 320) {
     8       w = 30;
     9     }
    10     self.navigationBar.backIndicatorImage = image;
    11     UIImage *backButtonImage = [image resizableImageWithCapInsets:UIEdgeInsetsMake(0, w, 0, -w)];
    12     [[UIBarButtonItem appearance] setBackButtonBackgroundImage:backButtonImage
    13                                                       forState:UIControlStateNormal
    14                                                     barMetrics:UIBarMetricsDefault];
    15     // 将返回按钮的文字position设置不在屏幕上显示
    16     [[UIBarButtonItem appearance] setBackButtonTitlePositionAdjustment:UIOffsetMake(NSIntegerMin, NSIntegerMin) forBarMetrics:UIBarMetricsDefault];
    17     self.interactivePopGestureRecognizer.enabled = YES;
    18   } else {
    19 #if __IPHONE_OS_VERSION_MIN_REQUIRED < __IPHONE_7_0
    20     UIBarButtonItem *item = [[UIBarButtonItem alloc] init];
    21     UIImage *image = [UIImage imageNamed:backImageName];
    22     [item setBackButtonBackgroundImage:[image resizableImageWithCapInsets:UIEdgeInsetsMake(0, image.size.width, 0, 0)] forState:UIControlStateNormal barMetrics:UIBarMetricsDefault];
    23     [item setBackgroundImage:image forState:UIControlStateNormal barMetrics:UIBarMetricsDefault];
    24     [item setBackButtonTitlePositionAdjustment:UIOffsetMake(NSIntegerMin, NSIntegerMin) forBarMetrics:UIBarMetricsDefault];
    25     self.navigationItem.backBarButtonItem = item;
    26 #endif
    27   }
    28   
    29   return;
    30 }

    因此这里,我们需要使用自定义的返回箭头,而且不显示返回按钮的文字。因此,我们需要设置如下:

    [[UIBarButtonItem appearance] setBackButtonTitlePositionAdjustment:UIOffsetMake(NSIntegerMin, NSIntegerMin) forBarMetrics:UIBarMetricsDefault];
    

    但是,仅仅部分标题是正常显示,还是有一些地方有不居中的。因此,我们还需要再添加一些代码解决。

    下面,我们使用了全局设置UIBarButtonItem的字体大小为1

    NSDictionary *attributes = @{NSFontAttributeName : [UIFont systemFontOfSize:1]};
    [[UIBarButtonItem appearance] setTitleTextAttributes:attributes
                                                forState:UIControlStateNormal];

    这样确实解决了我们的问题。标题没有出现不居中显示的了,但是又引发了新的问题。因为我们是全局将UIBarButtonItem的字体大小修改为1,那么所有使用了这个控件的地方,都会不显示了。除非我们每个使用的地方再重新设置其字体大小。

    因此,此方案也不可行。

    方案三(可行,但不太好)

    应用中始终隐藏系统自带的导航,然后自定义一个UIView作为导航,这样就解决居中显示的问题,但是没有了返回手势,因此需要借助第三方库追加右滑返回手势功能,但是体验不如原生的好。

    方案四(当前最终方案)

    自定义一个继承于UINavigationController的类,然后所有使用导航类的地方都使用我们所定义的导航控制器类。在这个类中,我们通过配置全局的导航条相关全局属性。

     1 - (void)config {
     2   NSString *backImageName = @"default_back";
     3   
     4   if (kIsIOS7OrLater) {
     5     UIImage *image = [UIImage imageNamed:backImageName];
     6     CGFloat w = image.size.width;
     7     if (kScreenWidth <= 320) {
     8       w = 30;
     9     }
    10     self.navigationBar.backIndicatorImage = image;
    11     UIImage *backButtonImage = [image resizableImageWithCapInsets:UIEdgeInsetsMake(0, w, 0, -w)];
    12     [[UIBarButtonItem appearance] setBackButtonBackgroundImage:backButtonImage
    13                                                       forState:UIControlStateNormal
    14                                                     barMetrics:UIBarMetricsDefault];
    15     // 将返回按钮的文字position设置不在屏幕上显示
    16     [[UIBarButtonItem appearance] setBackButtonTitlePositionAdjustment:UIOffsetMake(NSIntegerMin, NSIntegerMin) forBarMetrics:UIBarMetricsDefault];
    17     self.interactivePopGestureRecognizer.enabled = YES;
    18   } else {
    19 #if __IPHONE_OS_VERSION_MIN_REQUIRED < __IPHONE_7_0
    20     UIBarButtonItem *item = [[UIBarButtonItem alloc] init];
    21     UIImage *image = [UIImage imageNamed:backImageName];
    22     [item setBackButtonBackgroundImage:[image resizableImageWithCapInsets:UIEdgeInsetsMake(0, image.size.width, 0, 0)] forState:UIControlStateNormal barMetrics:UIBarMetricsDefault];
    23     [item setBackgroundImage:image forState:UIControlStateNormal barMetrics:UIBarMetricsDefault];
    24     [item setBackButtonTitlePositionAdjustment:UIOffsetMake(NSIntegerMin, NSIntegerMin) forBarMetrics:UIBarMetricsDefault];
    25     self.navigationItem.backBarButtonItem = item;
    26 #endif
    27   }
    28   
    29   return;
    30 }

    然后,我们在每个控制器的viewDidLoad方法中,调用这么个方法:

    - (void)resetBackButtonItem {
      NSArray *viewControllerArray = [self.navigationController viewControllers];
      
      long previousViewControllerIndex = [viewControllerArray indexOfObject:self] - 1;
      UIViewController *previous;
      
      if (previousViewControllerIndex >= 0) {
        previous = [viewControllerArray objectAtIndex:previousViewControllerIndex];
        previous.navigationItem.backBarButtonItem = [[UIBarButtonItem alloc]
                                                     initWithTitle:@""
                                                     style:UIBarButtonItemStylePlain
                                                     target:self
                                                     action:nil];
      }
    }

    这个方法是我们用于判断是否有上一个界面,如果有,则将上一个界面的返回按钮的标题设置为空,那么在本界面的返回按钮就不会有标题,如此一来,就解决了上个界面的标题过长,而本界面标题也很长时,导致本界面的标题不居中显示的问题

    建议

    我们可以将这resetBackButtonItem方法放到基类控制器中,然后在基类控制器中的viewDidLoad方法中调用,就不需要各个类都调用了。当然,如果现在类似笔者当前的情景,就需要手动各个界面都调用了。当前笔者的情景是,项目是由很多个团队敏捷迭代开发的,如果全局统一改,很有可能会影响到他人的版本,因此不得不只在需要处理的控制器调用。

    转载地址:http://blog.csdn.net/woaifen3344/article/details/50035609

      

  • 相关阅读:
    17.1.2?Replication Formats 复制格式:
    17.1.1.9 Introducing Additional Slaves to an Existing Replication Environment 引入额外的Slaves 到一个存在的复制
    17.1.1.9 Introducing Additional Slaves to an Existing Replication Environment 引入额外的Slaves 到一个存在的复制
    mysql读写分离
    mysql读写分离
    curl 返回响应头
    01_什么是Elasticsearch
    01_什么是Elasticsearch
    perl 自动发产品
    Jzoj1164求和
  • 原文地址:https://www.cnblogs.com/zj901203/p/5415949.html
Copyright © 2011-2022 走看看