WKWebView使用及注意点(预防踩坑)------强烈推荐阅读!!!

iOS8之后,苹果推出了WebKit这个框架,用来替换原有的UIWebView,新的控件优点多多,不一一叙述。由于一直在适配iOS7,就没有去替换,现在仍掉了iOS7,以为很简单的就替换过来了,然而在替换的过程中,却遇到了很多坑。还有一点就是原来写过一篇文章 Objective-C与JavaScript交互的那些事以为年代久远的UIWebView已经作古,可这篇文章现在依然有一定的阅读量。所以在决定在续一篇此文,以引导大家转向WKWebView,并指出自己踩过的坑,让大家少走弯路。

此篇文章的逻辑图

1192353-3b1b2a629853d8b2.png

WKWebView简单介绍

使用及注意点

WKWebView只能用代码创建,而且自身就支持了右滑返回手势allowsBackForwardNavigationGestures和加载进度estimatedProgress等一些UIWebView不具备却非常好用的属性。在创建的时候,指定初始化方法中要求传入一个WKWebViewConfiguration对象,一般我们使用默认配置就好,但是有些地方是要根据自己的情况去做更改。比如,配置中的allowsInlineMediaPlayback这个属性,默认为NO,如果不做更改,网页中内嵌的视频就无法正常播放。

更改User-Agent

有时我们需要在User-Agent添加一些额外的信息,这时就要更改默认的User-Agent在使用UIWebView的时候,可用如下代码(在使用UIWebView之前执行)全局更改User-Agent:

1
2
3
4
5
6
7
8
9
10
11
// 获取默认
User-AgentUIWebView *webView = [[UIWebView alloc] initWithFrame:CGRectZero];
NSString *oldAgent = [webView stringByEvaluatingJavaScriptFromString:@ "navigator.userAgent" ];
 
// 给User-Agent添加额外的信息
NSString *newAgent = [NSString stringWithFormat:@ "%@;%@" , oldAgent, @ "extra_user_agent" ];
 
// 设置global User-Agent
NSDictionary *dictionnary = [[NSDictionary alloc] initWithObjectsAndKeys:newAgent, @ "UserAgent" , nil];
 
[[NSUserDefaults standardUserDefaults] registerDefaults:dictionnary];

以上代码是全局更改User-Agent,也就是说,App内所有的Web请求的User-Agent都被修改。替换为WKWebView后更改全局User-Agent可以继续使用上面的一段代码,或者改为用WKWebView获取默认的User-Agent,代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
self.wkWebView = [[WKWebView alloc] initWithFrame:CGRectZero];
// 获取默认
 
User-Agent[self.wkWebView evaluateJavaScript:@ "navigator.userAgent"  completionHandler:^(id result, NSError *error) {      
    NSString *oldAgent = result;    
 
   // 给User-Agent添加额外的信息    
   NSString *newAgent = [NSString stringWithFormat:@ "%@;%@" , oldAgent, @ "extra_user_agent" ];  
 
   // 设置global User-Agent    
   NSDictionary *dictionary = [NSDictionary dictionaryWithObjectsAndKeys:newAgent, @ "UserAgent" , nil];   
   [[NSUserDefaults standardUserDefaults] registerDefaults:dictionary];
   
}];

对比发现,这两种方法并没有本质的区别,一点小区别在于一个是用UIWebView获取的默认User-Agent,一个是用WKWebView获取的默认User-Agent。上面方法的缺点也是很明显的,就是App内所有Web请求的User-Agent全部被修改。

在iOS9,WKWebView提供了一个非常便捷的属性去更改User-Agent,就是customUserAgent属性。这样使用起来不仅方便,也不会全局更改User-Agent,可惜的是iOS9才有,如果适配iOS8,还是要使用上面的方法。

WKWebView的相关的代理方法

  • WKWebView的相关的代理方法分别在WKNavigationDelegate和WKUIDelegate以及WKScriptMessageHandler这个与JavaScript交互相关的代理方法。

  • WKNavigationDelegate: 此代理方法中除了原有的UIWebView的四个代理方法,还增加了其他的一些方法,具体可参考我下面给出的Demo。

  • WKUIDelegate: 此代理方法在使用中最好实现,否则遇到网页alert的时候,如果此代理方法没有实现,则不会出现弹框提示。

WKScriptMessageHandler: 此代理方法就是和JavaScript交互相关,具体介绍参考下面的专门讲解。

WKWebView使用过程中的坑

WKWebView下面添加自定义View

因为我们有个需求是在网页下面在添加一个View,用来展示此链接内容的相关评论。在使用UIWebView的时候,做法非常简单粗暴,在UIWebView的ScrollView后面添加一个自定义View,然后根据View的高度,在改变一下scrollView的contentSize属性。以为WKWebView也可以这样简单粗暴的去搞一下,结果却并不是这样。

首先改变WKWebView的scrollView的contentSize属性,系统会在下一次帧率刷新的时候,再给你改变回原有的,这样这条路就行不通了。我马上想到了另一个办法,改变scrollView的contentInset这个系统倒不会在变化回原来的,自以为完事大吉。后来过了两天,发现有些页面的部分区域的点击事件无法响应,百思不得其解,最后想到可能是设置的contentInset对其有了影响,事实上正是如此。查来查去,最后找到了一个解决办法是,就是当页面加载完成时,在网页下面拼一个空白的div,高度就是你添加的View的高度,让网页多出一个空白区域,自定义的View就添加在这个空白的区域上面。这样就完美解决了此问题。具体可参考Demo所写,核心代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
self.addView = [[UIView alloc] initWithFrame:CGRectMake(0, 0, ScreenWidth, addViewHeight)];
 
self.addView.backgroundColor = [UIColor redColor];
 
[self.webView.scrollView addSubview:self.addView];
 
NSString *js = [NSString stringWithFormat:@"\
                         var  appendDiv = document.getElementById(\"AppAppendDIV\");\                        
                         if  (appendDiv) {\                        
                         appendDiv.style.height = %@+\"px\";\                        
                         else  {\                        
                         var  appendDiv = document.createElement(\"div\");\                        
                         appendDiv.setAttribute(\"id\",\"AppAppendDIV\");\                        
                         appendDiv.style.width=%@+\"px\";\                        
                         appendDiv.style.height=%@+\"px\";\                        
                         document.body.appendChild(appendDiv);\                        
                         }\                        
                         ", @(addViewHeight), @(self.webView.scrollView.contentSize.width), @(addViewHeight)];
                         
[self.webView evaluateJavaScript:js completionHandler:nil];

WKWebView加载HTTPS的链接

HTTPS已经越来越被重视,前面我也写过一系列的HTTPS的相关文章HTTPS从原理到应用(四):iOS中HTTPS实际使用当加载一些HTTPS的页面的时候,如果此网站使用的根证书已经内置到了手机中这些HTTPS的链接可以正常的通过验证并正常加载。但是如果使用的证书(一般为自建证书)的根证书并没有内置到手机中,这时是链接是无法正常加载的,必须要做一个权限认证。开始在UIWebView的时候,是把请求存储下来然后使用NSURLConnection去重新发起请求,然后走NSURLConnection的权限认证通道,认证通过后,在使用UIWebView去加载这个请求。

在WKWebView中,WKNavigationDelegate中提供了一个权限认证的代理方法,这是权限认证更为便捷。代理方法如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
- (void)webView:(WKWebView *)webView didReceiveAuthenticationChallenge:(NSURLAuthenticationChallenge *)challenge completionHandler:(void (^)(NSURLSessionAuthChallengeDisposition disposition, NSURLCredential *credential))completionHandler {    
if  ([challenge.protectionSpace.authenticationMethod isEqualToString:NSURLAuthenticationMethodServerTrust]) {
         
if  ([challenge previousFailureCount] == 0) { 
            
NSURLCredential *credential = [NSURLCredential credentialForTrust:challenge.protectionSpace.serverTrust];   
          
completionHandler(NSURLSessionAuthChallengeUseCredential, credential);  
       
else  {    
         
completionHandler(NSURLSessionAuthChallengeCancelAuthenticationChallenge, nil); 
        
}    
 
else  {    
     
completionHandler(NSURLSessionAuthChallengeCancelAuthenticationChallenge, nil);   
  
}
}

这个方法比原来UIWebView的认证简单的多。但是使用中却发现了一个很蛋疼的问题,iOS8系统下,自建证书的HTTPS链接,不调用此代理方法。查来查去,原来是一个bug,在iOS9中已经修复,这明显就是不管iOS8的情况了,而且此方法也没有标记在iOS9中使用,这点让我感到有点失望。这样我就又想到了换回原来UIWebView的权限认证方式,但是试来试去,发现也不能使用了。所以关于自建证书的HTTPS链接在iOS8下面使用WKWebView加载,我没有找到很好的办法去解决此问题。这样我不得已有些链接换回了HTTP,或者在iOS8下面在换回UIWebView。如果你有解决办法,也欢迎私信我,感激不尽。

WKWebView和JavaScript交互

WKWebView和JavaScript交互,在WKUserContentController.h这个头文件中- (void)addScriptMessageHandler:(id )scriptMessageHandler name:(NSString *)name;这个方法的注释中已经明确给出了交互办法。使用起来倒是非常的简单。创建WKWebView的时候添加交互对象,并让交互对象实现WKScriptMessageHandler中的唯一的一个代理方法。具体的方式参考Demo中的使用。

1
2
3
4
5
// 添加交互对象
[config.userContentController addScriptMessageHandler:(id)self.ocjsHelper name:@ "timefor" ];
 
// 代理方法
- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message;

JavaScript调用Objective-C的时候,使用window.webkit.messageHandlers.timefor.postMessage({code: '0001', functionName: 'getdevideId'}); Objective-C自动对交互参数包装成了WKScriptMessage对象,其属性body则为传送过来的参数,name为添加交互对象的时候设置的名字,以此名字可以过滤掉不属于自己的交互方法。其中body可以为NSNumber, NSString, NSDate, NSArray, NSDictionary, and NSNull。

而Objective-C在回调JavaScript的时候,不能像我原来在 Objective-C与JavaScript交互的那些事这篇文章中写的那样,JavaScript传过来一个匿名函数,Objective-C这边直接调用一下就完事。WKWebView没有办法传过来一个匿名函数,所以回调方式,要么执行一段JavaScript代码,或者就是调用JavaScript那边的一个全局函数。一般是采用后者,至于Web端虽说暴露了一个全局函数,同样可以把这一点代码处理的很优雅。Objective-C传给JavaScript的参数,可以为Number, String, and Object。参考如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
// 数字
NSString *js = [NSString stringWithFormat:@ "globalCallback(%@)" , number];
[self.webView evaluateJavaScript:js completionHandler:nil];
// 字符串
NSString *js = [NSString stringWithFormat:@ "globalCallback(\'%@\')" , string];
[self.webView evaluateJavaScript:js completionHandler:nil];
// 对象
NSString *js = [NSString stringWithFormat:@ "globalCallback(%@)" , @{@ "name"  : @ "timefor" }];
[self.webView evaluateJavaScript:js completionHandler:nil];
// 带返回值的JS函数
[self.webView evaluateJavaScript:@ "globalCallback()"  completionHandler:^(id result, NSError * _Nullable error) { 
    
// 接受返回的参数,result中
 
}];

总结

此文主要介绍了WKWebView使用中的注意点,一般也都是常用的,还有缓存等一些不是太常用的就没有具体介绍。如果在其他方面遇到问题,也欢迎你私信我共同探讨进步。
此文的Demo地址:WKWebViewDemo 如果此文对你有所帮助,请给个star吧。


原文地址:http://www.cocoachina.com/ios/20161122/18162.html

参考

http://blog.csdn.net/shaobo8910/article/details/53352516

http://blog.csdn.net/u011619283/article/details/52352514

http://blog.csdn.net/j_av_a/article/details/52160413

http://www.jianshu.com/p/f896d73c670a

http://stackoverflow.com/questions/34693311/links-in-wkwebview-randomly-not-clickable/35100064#35100064

https://bugs.webkit.org/show_bug.cgi?id=140197







  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
YOLO高分设计资源源码,详情请查看资源内容中使用说明 YOLO高分设计资源源码,详情请查看资源内容中使用说明 YOLO高分设计资源源码,详情请查看资源内容中使用说明 YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明
Web View 是一种在原生应用程序中嵌入网页内容的组件。它允许开发人员使用 HTML、CSS 和 JavaScript 来构建应用程序的一部分或全部界面,并且可以与原生应用程序进行交互。 在 Android 平台上,可以使用 Android WebView 组件来实现 Web View 功能。要使用 Web View,在布局文件中添加一个 WebView 元素,并在代码中通过获取 WebView 实例来控制和加载网页内容。你可以使用 WebView 提供的方法来加载 URL、显示HTML内容、处理页面加载事件、与 JavaScript 进行交互等。 在 iOS 平台上,可以使用 UIWebView 或 WKWebView 来实现 Web View 功能。UIWebView 是较早的实现方式,而 WKWebView 是从 iOS 8 开始引入的新一代 Web View 组件。你可以创建一个 UIWebView 或 WKWebView 的实例,并通过设置其 delegate 来处理页面加载事件、与 JavaScript 进行交互等。通过加载 URL 或加载 HTML 字符串,你可以在应用程序中显示网页内容。 需要注意的是,Web View 的使用要遵循相关安全性和性能的最佳实践。例如,在加载外部 URL 时,应确保只加载可信任的源,并避免潜在的安全风险。此外,过多或复杂的网页内容可能会影响应用程序的性能和用户体验,因此需要进行优化和限制。 总结来说,Web View 是一种在原生应用程序中显示网页内容的组件,可以使用 WebView 或 WKWebView 在 Android 和 iOS 平台上实现。通过加载 URL 或 HTML 内容,开发人员可以在应用程序中展示并与网页内容进行交互。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值