首页 热点资讯 义务教育 高等教育 出国留学 考研考公
您的当前位置:首页正文

iOS适配

2024-12-20 来源:化拓教育网

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的位置和尺寸比例都是一样的.
显示全文