iOS中文行间距富文本高度与显示那些坑

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了iOS中文行间距富文本高度与显示那些坑相关的知识,希望对你有一定的参考价值。

参考技术A 前段时间来了一个新设计,将App的风格修改了一遍。在显示文字时,增加了行间距。原本以为只是展示和计算高度的时候添加上富文本的lineSpace属性即可,但是结果在显示一行中文的时候,却怎么也计算字体的时候多了一个行间距的高度,展示的Label也多了一个行间距的高度。经过查找资料,网上也有人遇到同样的坑。计算高度时,需要将判断一下是否中文与一行,如果是一行带中文并且有行间距,此时总体高度应该减去行间距。展示的Label如果用的是自适应高度,也应该做一下判断,一行中文带行间距时,不设置Label的lineSpace属性。如果你也遇到这样坑,可以通过本文章的Demo解决,并封装好一些方法,便于以后的使用。

通常计算文本的高度使用以下两种方式,通过计算出来的高度计算父控件的总高度。

对NSString进行以下方法的增加。

https://github.com/casscqt/lineSpaceTextHeightDemo

iOS11 与 iPhone X适配的那些坑(持更中...)

目录 问题列表

1.XCode9添加资源文件不能获取路径问题

2.iOS11 系统导航栏上自定义view的显示问题

3.UITableView 出现的遮挡显示问题(三行代码解决,放到基类)

4.适配iPhoneX 屏幕原则

5.适配过程一些常量的设置

6.出现UIScrollview 漂移问题(基本都是iPhoneX上)

7.iPhone X 上运行有黑色区域问题

8.有些导致iPhoneX获取网络状态崩溃的问题

9.一些关于iPhone X底部的设计示范

10.关于iPhoneX的home条(HomeIndicator)的显隐

11.关于iOS11权限的变化

12.iOS11 UICollectionview 的scroll indicator等被组视图遮挡的问题

13.iOS11 无法弹出键盘问题

14.iOS11 数组语法糖??替换方法的改变

1. XCode9添加资源文件不能获取路径问题

XCode9调用[[NSBundle mainBundle] pathForResource:resourceNameofType:nil];方法来获取文件路径,返回为nil. 添加正确方式如下图1,

 
技术分享图片
图1  添加资源选项

但是XCode9 却埋了一道坑,如下图2, 点击资源文件,找到文件选项

 
技术分享图片
图2  资源文件选项

红框区域未默认勾选, 只要把这个复选框勾选即可解决.  具体原因是:xcode9 添加文件使用addfile 拖拽的文件不会自动添加到Compile Sources 和 Copy Bundle Resources 下.

 

2. iOS11 系统导航栏上自定义view的显示问题

之前代码设置导航栏:

self.navigationItem.titleView = customView;

会出现这样的问题,如下图3,

 
技术分享图片
图3 导航栏出错

 

界面出乱的原因是因为iOS11调整了导航栏图层结构。快速简便的修改方式,直接上代码,

self.navigationItem.hidesBackButton = YES;
[self.navigationController.navigationBaraddSubview:YourCustomView];

但是需在控制器的dealloc里移除你的自定义view,

[YourCustomView removeFromSuperview]; 

下图4为修改好的自定义视图

 
技术分享图片
图4

 

3. UITableView 出现的遮挡显示问题(三行代码解决,放到基类)

// iOS11适配
self.tableView.estimatedRowHeight = 0;
self.tableView.estimatedSectionHeaderHeight = 0;
self.tableView.estimatedSectionFooterHeight = 0;

4. 适配iPhoneX 屏幕原则

 
技术分享图片
图5 iphoneX 竖屏
 
技术分享图片
图6 iPhoneX 横屏
 
技术分享图片
 

设计原则:让那行可操作、不可阻挡的控件或视图显示在图5、图6的蓝色区域(安全区域safeArea)。

顶部危险区距离:44

底部危险区距离:34

5. 适配过程一些常量的设置

#define YYISiPhoneX [[UIScreen mainScreen] bounds].size.width >=375.0f && [[UIScreen mainScreen] bounds].size.height >=812.0f&& YYIS_IPHONE
#define YYIS_IPHONE (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone)
//状态栏高度
#define kStatusBarHeight    (CGFloat)(YYISiPhoneX?(44):(20))
// 导航栏高度
#define kNavBarHBelow7      (44)
// 状态栏和导航栏总高度
#define kNavBarHAbove7      (CGFloat)(YYISiPhoneX?(88):(64))
// TabBar高度
#define kTabBarHeight      (CGFloat)(YYISiPhoneX?(49+34):(49))
// 顶部安全区域远离高度
#define kTopBarSafeHeight  (CGFloat)(YYISiPhoneX?(44):(0))
// 底部安全区域远离高度
#define kBottomSafeHeight  (CGFloat)(YYISiPhoneX?(34):(0))
// iPhoneX的状态栏高度差值
#define kTopBarDifHeight    (CGFloat)(YYISiPhoneX?(24):(0))

6. 出现UIScrollview 漂移问题(基本都是iPhoneX上)

若是UIScrollview的frame和contentSize高度或者宽度一致却还导致了漂移问题,可用以下一行代码搞定。

if(@available(iOS11,*)) {_scrollView.contentInsetAdjustmentBehavior=UIScrollViewContentInsetAdjustmentNever;}

同时,如果在push进入webview页面时,底部会有黑边一闪而过,也可用此方法解决。

若嫌麻烦,每个scrollview及其子类都要去设置的话,也可以使用以下方法

// AppDelegate 里进行全局设置 if(@available(iOS 11.0, *)){[[UIScrollView appearance] setContentInsetAdjustmentBehavior:UIScrollViewContentInsetAdjustmentNever];

7. iPhone X 上运行有黑色区域问题

启动图问题,使用LaunchScreen来做启动图 或者 修改Assets中的LaunchImage,添加iPhoneX的Launch图1125*2436(px) 竖屏模式。

8.有些导致iPhoneX获取网络状态崩溃的问题

直接上代码,无需通过KVC得到图层名获取网络状态,一劳永逸!(因还不熟悉markdown语法,以下代码需自行缩进

#import <CoreTelephony/CTTelephonyNetworkInfo.h>
+ (NSString *)getNetWorkInfo {
    NSString *strNetworkInfo = @"No Network";
    struct sockaddr_storage zeroAddress;
    bzero(&zeroAddress,sizeof(zeroAddress));
    zeroAddress.ss_len = sizeof(zeroAddress);
    zeroAddress.ss_family = AF_INET;
// Recover reachability flags
    SCNetworkReachabilityRef defaultRouteReachability = SCNetworkReachabilityCreateWithAddress(NULL,(struct sockaddr *)&zeroAddress);
    SCNetworkReachabilityFlags flags;
//获得连接的标志   
    BOOL didRetrieveFlags = SCNetworkReachabilityGetFlags(defaultRouteReachability,&flags);
    CFRelease(defaultRouteReachability);
//如果不能获取连接标志,则不能连接网络,直接返回
    if(!didRetrieveFlags){ return strNetworkInfo;}     BOOL isReachable = ((flags & kSCNetworkFlagsReachable)!=0); BOOL needsConnection = ((flags & kSCNetworkFlagsConnectionRequired)!=0); if(!isReachable || needsConnection) {return strNetworkInfo;} // 网络类型判断
    if((flags & kSCNetworkReachabilityFlagsConnectionRequired)== 0){strNetworkInfo = @"WIFI";}   
    if(((flags & kSCNetworkReachabilityFlagsConnectionOnDemand ) != 0) ||(flags & kSCNetworkReachabilityFlagsConnectionOnTraffic) != 0) { if ((flags & kSCNetworkReachabilityFlagsInterventionRequired) == 0){strNetworkInfo = @"WIFI";}}
    if ((flags & kSCNetworkReachabilityFlagsIsWWAN) ==kSCNetworkReachabilityFlagsIsWWAN) {if ([[[UIDevice currentDevice] systemVersion] floatValue] >= 7.0) {CTTelephonyNetworkInfo * info = [[CTTelephonyNetworkInfo alloc] init];NSString *currentRadioAccessTechnology = info.currentRadioAccessTechnology;if (currentRadioAccessTechnology) {if ([currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyLTE]) {strNetworkInfo =  @"4G";} else if ([currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyEdge] || [currentRadioAccessTechnologyisEqualToString:CTRadioAccessTechnologyGPRS]) {strNetworkInfo =  @"2G";} else {strNetworkInfo =  @"3G";}}} else {if((flags & kSCNetworkReachabilityFlagsReachable) == kSCNetworkReachabilityFlagsReachable) {if ((flags & kSCNetworkReachabilityFlagsTransientConnection) == kSCNetworkReachabilityFlagsTransientConnection) {if((flags & kSCNetworkReachabilityFlagsConnectionRequired) == kSCNetworkReachabilityFlagsConnectionRequired) {strNetworkInfo = @"2G";} else {strNetworkInfo = @"3G";}}}}}
    // if ([strNetworkInfo isEqualToString: @"No Network"]) {strNetworkInfo = @"WWAN";}
    return strNetworkInfo;
}

9.一些关于iPhone X底部的设计示范

 
技术分享图片
示范图一

 

 
技术分享图片
示范图二

 

 
技术分享图片
示范图三

 

10. 关于iPhoneX的home条(HomeIndicator)的显隐

先看下图7,

 
技术分享图片
图7 横屏看撩妹视频时

这样的体验其实是不好的(还怎么用英文撩妹?),所以为了以防这样的问题出现,苹果其实提供了隐藏HomeIndicator的方法,如下,

- (BOOL)prefersHomeIndicatorAutoHidden {
  return YES;
}

在VC 里边重写 prefersHomeIndicatorAutoHidden 返回 YES(默认是NO),Home指示条就能自动隐藏了,此方法是在屏幕若无交互事件响应时,延迟2秒左右会回调此方法。

11. 关于iOS11权限的变化

1.定位问题:无法定位,第一次使用时没有弹出定位请求对话框。

原因:由于iOS11的定位权限的key做了更改,在以前iOS11之前的始终允许定位NSLocationAlwaysUsageDescription基础上添加NSLocationWhenInUseUsageDescription和NSLocationAlwaysAndWhenInUsageDescription。具体参考博客:http://blog.csdn.net/dangyalingengjia/article/details/77965029

2.相册权限问题:iOS11上读写相册的照片时发生崩溃。

原因:由于iOS11相册相关权限的key发生了变化。用户在没有权限的情况下,访问相册导致崩溃。

iOS11之前对应的key是NSPhotoLibraryUsageDescription,而iOS11的Key变为NSPhotoLibraryAddUsageDescription。同定位的Key一样,由于key没有兼容性,所以需要保留原key以兼容iOS11之前的版本。

12. iOS11 UICollectionview 的scroll indicator等被组视图遮挡的问题

问题如下图8红框区域,

 
技术分享图片
图8 UICollectionview bug图

导致这个的原因是因为组视图的z坐标出现了问题, 有以下两种解决办法

第一种: 在该类中重写layoutSubviews方法<适用于小规模>

- (void)layoutSubviews {
    [super layoutSubviews];
    self.layer.zPosition = 0;
}

第二种: 自定义一个layer, 重写其zPosition的get方法。然后对组视图类添加分类方法,重写layerClass方法即可。

自定义layer:

@implementation YYZPositionZeroLayer

- (CGFloat)zPosition {
    return 0;
}
@end

添加分类重写组视图layerClass方法:

@implementationUICollectionReusableView (ZPositionZero)
+ (Class)layerClass {
  return[YYZPositionZeroLayer class];
}
@end

13. iOS11 无法弹出键盘问题

问题描述: 在某些时候你会发现调用becomeFirstResponder方法时,并未响应键盘,系统键盘无法弹出.

问题发现: 如果在调用之前使用了UIAlertView弹出对话框,就会导致系统键盘无法弹出,猜测是因为UIAlertView在消失遍历window窗口时存在windowLevel 大于 levelNormal 的window把他作为keywindow

问题解决: 如果你的项目里面还在使用UIAlertView(2_0,9_0), 建议用UIAlertController替换掉.

14.iOS11 数组语法糖??替换方法的改变

在iOS11 之前,

array[i] 语法糖的替换方法是[array objectAtIndex:i]

在iOS11 之后,

array[i] 语法糖的替换方法是[array objectAtIndexedSubscript:i]

用到动态交换方法的童鞋需要注意
























































以上是关于iOS中文行间距富文本高度与显示那些坑的主要内容,如果未能解决你的问题,请参考以下文章

iOS--UILabel设置行距和字间距,并根据文本计算高度

IOS 富文本 ,设置行间距字间距,计算高度(转载组合而成)

IOS 富文本 ,设置行间距字间距,计算高度(转载组合而成)

IOS 富文本 ,设置行间距字间距,计算高度(转载组合而成)

iOS踩过的坑之富文本计算文字高度

iOS之富文本