iOS适配
1.iOS10的适配
(1).Notification(通知)
自从Notification被引入之后,苹果就不断的更新优化,但这些更新优化只是小打小闹,直至现在iOS 10开始真正的进行大改重构,这让开发者也体会到UserNotifications的易用,功能也变得非常强大。
类型 | 作用
iOS 9 以前的通知 | 1.在调用方法时,有些方法让人很难区分,容易 写错方法,这让开发者有时候很苦恼。2.应用在运行时和非运行时捕获通知的路径还不一致。3.应用在前台时,是无法直接显示远程通知,还需要进一步处理。 4.已经发出的通知是不能更新的,内容发出时是不能改变的,并且只有简单文本展示方式,扩展性根本不是很好
iOS10 | 1.所有相关通知被统一到了UserNotifications.framework框架中。2.增加了撤销、更新、中途还可以修改通知的内容。3.通知不在是简单的文本了,可以加入视频、图片,自定义通知的展示等等。4.iOS 10相对之前的通知来说更加好用易于管理,并且进行了大规模优化,对于开发者来说是一件好事。5.iOS 10开始对于权限问题进行了优化,申请权限就比较简单了(本地与远程通知集成在一个方法中)。
(2).ATS的问题
iOS 9中默认HTTP的网络是不推荐使用的,当然我们也可以把NSAllowsArbitraryLoads设置为YES禁用ATS。不过iOS 10从2017年1月1日起苹果不允许我们通过这个方法跳过ATS,也就是说强制我们用HTTPS,如果不这样的话提交App可能会被拒绝。但是我们可以通过NSExceptionDomains来针对特定的域名开放HTTP可以容易通过审核。
(3).iOS10隐私权限设置
iOS 10 开始对隐私权限更加严格,如果你不设置就会直接崩溃,现在很多遇到崩溃问题了,一般解决办法都是在info.plist文件添加对应的Key-Value就可以了。
[图片上传失败...(image-a417ac-1526873782268)]
以上Value值,圈出的红线部分的文字是展示给用户看的,需要自己添加规范的提示说明,不能为空。
(4).Xcode8运行一堆没用的logs解决办法
[图片上传失败...(image-1e742e-1526873782268)]
解决办法是设置OS_ACTIVITY_MODE : disable如下图:
[图片上传失败...(image-46e3bf-1526873782268)]
(5).iOS10UIStatusBar方法过期
[图片上传失败...(image-7ca3db-1526873782268)]
在我们开发中有可能用到UIStatusBar一些属性,在iOS 10 中这些方法已经过期了,如果你的项目中有用的话就得需要适配。上面的图片也能发现,如果在iOS 10中你需要使用preferredStatusBar比如这样:
//iOS 10
- (UIStatusBarStyle)preferredStatusBarStyle {
return UIStatusBarStyleDefault;}
(6).iOS 10 UIColor 新增方法
因为之前我们都是用RGB来设置颜色,反正用起来也不是特别多样化,这次新增的方法应该就是一个弥补吧。所以在iOS 10 苹果官方建议我们使用sRGB,因为它性能更好,色彩更丰富。如果你自己为UIColor写了一套分类的话也可尝试替换为sRGB,UIColor类中新增了两个Api如下:
+ (UIColor *)colorWithDisplayP3Red:(CGFloat)displayP3Red green:(CGFloat)green blue:(CGFloat)blue alpha:(CGFloat)alpha NS_AVAILABLE_IOS(10_0);
- (UIColor *)initWithDisplayP3Red:(CGFloat)displayP3Red green:(CGFloat)green blue:(CGFloat)blue alpha:(CGFloat)alpha NS_AVAILABLE_IOS(10_0);
(7).iOS10UITextContentType
@property(nonatomic,copy) UITextContentType textContentType NS_AVAILABLE_IOS(10_0); // default is nil
在iOS 10 UITextField添加了textContentType枚举,指示文本输入区域所期望的语义意义。
使用此属性可以给键盘和系统信息,关于用户输入的内容的预期的语义意义。例如,您可以指定一个文本字段,用户填写收到一封电子邮件确认uitextcontenttypeemailaddress。当您提供有关您期望用户在文本输入区域中输入的内容的信息时,系统可以在某些情况下自动选择适当的键盘,并提高键盘修正和主动与其他文本输入机会的整合。
(8).iOS 10 字体随着手机系统字体而改变
当我们手机系统字体改变了之后,那我们App的label也会跟着一起变化,这需要我们写很多代码来进一步处理才能实现,但是iOS 10 提供了这样的属性adjustsFontForContentSizeCategory来设置。
UILabel *myLabel = [UILabel new];
myLabel.font =[UIFont preferredFontForTextStyle: UIFontTextStyleHeadline];
myLabel.adjustsFontForContentSizeCategory = YES;
(9).iOS10UIScrollView新增refreshControl
//iOS 10 以后只要是继承UIScrollView那么就支持刷新功能:
@property (nonatomic, strong, nullable) UIRefreshControl *refreshControl NS_AVAILABLE_IOS(10_0) __TVOS_PROHIBITED;
(10).Xcode 8使用Xib awakeFromNib的警告问题
在Xcode 8之前我们使用Xib初始化- (void)awakeFromNib {}都是这么写也没什么问题,但是在Xcode 8会有如下警告,如果不喜欢这个警告的话,应该明确的加上[super awakeFromNib];
[图片上传失败...(image-f7efe2-1526873782268)]
(11)、权限以及相关设置
我们需要打开info.plist文件添加相应权限的说明,否则程序在iOS10上会出现崩溃。
具体如下图:
[图片上传失败...(image-f48930-1526873782268)]
麦克风权限:Privacy - Microphone Usage Description 是否允许此App使用你的麦克风?
相机权限: Privacy - Camera Usage Description 是否允许此App使用你的相机?
相册权限: Privacy - Photo Library Usage Description 是否允许此App访问你的媒体资料库?通讯录权限: Privacy - Contacts Usage Description 是否允许此App访问你的通讯录?
蓝牙权限:Privacy - Bluetooth Peripheral Usage Description 是否许允此App使用蓝牙?
语音转文字权限:Privacy - Speech Recognition Usage Description 是否允许此App使用语音识别?
日历权限:Privacy - Calendars Usage Description 是否允许此App使用日历?
定位权限:Privacy - Location When In Use Usage Description 我们需要通过您的地理位置信息获取您周边的相关数据
定位权限: Privacy - Location Always Usage Description 我们需要通过您的地理位置信息获取您周边的相关数据,定位的需要这么写,防止上架被拒。
2.frame屏幕适配
(1)用frame模型实现cell的自动行高.
//1)cell的frame模型
// 这个模型对象专门用来存放cell内部所有的子控件的frame数据 + cell的高度+数据模型,一个cell拥有一个双层嵌套的Frame模型
#import <Foundation/Foundation.h>
@class GJStatus;
@interface GJStatusFrame : NSObject
// 头像的frame
@property (nonatomic, assign, readonly) CGRect iconF;
// 配图的frame
@property (nonatomic, assign, readonly) CGRect pictureF;
// cell的高度
@property (nonatomic, assign, readonly) CGFloat cellHeight;
//模型数据
@property (nonatomic, strong) GJStatus *status;
@end
//2).懒加载调用:
@property (nonatomic, strong) NSArray *statusFrames;
@implementation GJViewController
- (NSArray *)statusFrames
{
if (_statusFrames == nil) {
// 初始化
// 1.获得plist的全路径
NSString *path = [[NSBundle mainBundle] pathForResource:@"statuses.plist" ofType:nil];
// 2.加载数组
NSArray *dictArray = [NSArray arrayWithContentsOfFile:path];
// 3.将dictArray里面的所有字典转成模型对象,放到新的数组中
NSMutableArray *statusFrameArray = [NSMutableArray array];
for (NSDictionary *dict in dictArray) {
// 3.1.创建GJStatus模型对象
GJStatus *status = [GJStatus statusWithDict:dict];
// 3.2.创建GJStatusFrame模型对象
GJStatusFrame *statusFrame = [[GJStatusFrame alloc] init];
statusFrame.status = status;
// 3.2.添加模型对象到数组中
[statusFrameArray addObject:statusFrame];
}
// 4.赋值
_statusFrames = statusFrameArray;
}
return _statusFrames;
}
//3).实现tableView的行高方法
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
{
// 取出这行对应的frame模型
GJStatusFrame *statusFrame = self.statusFrames[indexPath.row];
return statusFrame.cellHeight;
}
@end
(2)如何用frame适配4.0寸,4.7寸,5.5寸的屏幕.
类型 | 作用
4.0寸 | 320/568=0.563
4.7寸 | 375/667=0.562
5.5寸 | 414/736=0.5625
很明显能看出这三种屏幕的尺寸宽高比是差不多的,因此可以在4.0寸的基础上,按比例放大来适配4.7寸和5.5寸的屏幕.
// 在AppDelegate.h中
@property float autoSizeScaleX;
@property float autoSizeScaleY;
// 在AppDelegate.m中
#define ScreenHeight [[UIScreen mainScreen] bounds].size.height
#define ScreenWidth [[UIScreen mainScreen] bounds].size.width
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
AppDelegate *myDelegate = [[UIApplication sharedApplication] delegate];
if(ScreenHeight > 480){ // 这里以(iPhone4S)为准
myDelegate.autoSizeScaleX = ScreenWidth/320;
myDelegate.autoSizeScaleY = ScreenHeight/568;
}else{
myDelegate.autoSizeScaleX = 1.0;
myDelegate.autoSizeScaleY = 1.0;
}
}
因为iPhone4s屏幕的高度是480, 因此当屏幕尺寸大于iPhone4时, autoSizeScaleX和autoSizeScaleY即为当前屏幕和iPhone5尺寸的宽高比, 比如,
如果是5,autoSizeScaleX=1,autoSizeScaleY=1;
如果是6,autoSizeScaleX=1.171875,autoSizeScaleY=1.17429577;
如果是6Plus,autoSizeScaleX=1.29375,autoSizeScaleY=1.2957;
现在我们获取了比例关系后,先来看一下如何解决代码设置界面时的适配。CGRectMake(CGFloat x, CGFloat y, CGFloat width, CGFloat height)这个方法使我们常用的设置尺寸的方法,现在我设置了一个类似于这样的方法。
CG_INLINE CGRect
TS_CGRectMake(CGFloat x, CGFloat y, CGFloat width, CGFloat height)
{
AppDelegate *myDelegate = [[UIApplication sharedApplication] delegate];
CGRect rect;
rect.origin.x = x * myDelegate.autoSizeScaleX;
rect.origin.y = y * myDelegate.autoSizeScaleY;
rect.size.width = width * myDelegate.autoSizeScaleX;
rect.size.height = height * myDelegate.autoSizeScaleY;
return rect;
}
UIImageView *imageview = [[UIImageView alloc] initWithFrame:TS_CGRectMake(100, 100, 50, 50)];
// 这样我们得出的就是转换后的坐标了.这个imageview在4.0,4.7和5.5的位置和尺寸比例都是一样的.