WKWebView的新特性与使用

在WWDC2014中,苹果推出了最新的iOS8系统,其中也伴随着很多控件的更新与升级。其中全新的WebKit库让人很是兴奋。本文也将讲解到WebKit中更新的WKWebView控件的新特性与使用方法,它很好的解决了UIWebView存在的内存、加载速度等诸多问题。

环境信息:

Mac OS X 10.10.1

Xcode 6.1.1

iOS 8.1

正文:

一、WKWebView新特性

  • 在性能、稳定性、功能方面有很大提升(最直观的体现就是加载网页是占用的内存,模拟器加载百度与开源中国网站时,WKWebView占用23M,而UIWebView占用85M);
  • 允许JavaScript的Nitro库加载并使用(UIWebView中限制);
  • 支持了更多的HTML5特性
  • 高达60fps的滚动刷新率以及内置手势
  • 将UIWebViewDelegate与UIWebView重构成了14类与3个协议查看苹果官方文档);

二、初始化

1. 首先需要引入WebKit库


#import <WebKit/WebKit.h>

2. 初始化方法分为以下两种


// 默认初始化
- (instancetype)initWithFrame:(CGRect)frame;

// 根据对webview的相关配置,进行初始化
- (instancetype)initWithFrame:(CGRect)frame configuration:(WKWebViewConfiguration *)configuration NS_DESIGNATED_INITIALIZER;

3. 加载网页与HTML代码的方式与UIWebView相同,代码如下:


WKWebView *webView = [[WKWebView alloc] initWithFrame:self.view.bounds];
[webView loadRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:@"http://www.baidu.com"]]];
[self.view addSubview:webView];

三、 WKWebView的代理方法

1. WKNavigationDelegate

该代理提供的方法,可以用来追踪加载过程(页面开始加载、加载完成、加载失败)、决定是否执行跳转。


// 页面开始加载时调用
- (void)webView:(WKWebView *)webView didStartProvisionalNavigation:(WKNavigation *)navigation;
// 当内容开始返回时调用
- (void)webView:(WKWebView *)webView didCommitNavigation:(WKNavigation *)navigation;
// 页面加载完成之后调用
- (void)webView:(WKWebView *)webView didFinishNavigation:(WKNavigation *)navigation;
// 页面加载失败时调用
- (void)webView:(WKWebView *)webView didFailProvisionalNavigation:(WKNavigation *)navigation;

页面跳转的代理方法有三种,分为(收到跳转与决定是否跳转两种)


// 接收到服务器跳转请求之后调用
- (void)webView:(WKWebView *)webView didReceiveServerRedirectForProvisionalNavigation:(WKNavigation *)navigation;
// 在收到响应后,决定是否跳转
- (void)webView:(WKWebView *)webView decidePolicyForNavigationResponse:(WKNavigationResponse *)navigationResponse decisionHandler:(void (^)(WKNavigationResponsePolicy))decisionHandler;
// 在发送请求之前,决定是否跳转
- (void)webView:(WKWebView *)webView decidePolicyForNavigationAction:(WKNavigationAction *)navigationAction decisionHandler:(void (^)(WKNavigationActionPolicy))decisionHandler;

2. WKUIDelegate


// 创建一个新的WebView
- (WKWebView *)webView:(WKWebView *)webView createWebViewWithConfiguration:(WKWebViewConfiguration *)configuration forNavigationAction:(WKNavigationAction *)navigationAction windowFeatures:(WKWindowFeatures *)windowFeatures;

剩下三个代理方法全都是与界面弹出提示框相关的,针对于web界面的三种提示框(警告框、确认框、输入框)分别对应三种代理方法。下面只举了警告框的例子。


/**
 *  web界面中有弹出警告框时调用
 *
 *  @param webView           实现该代理的webview
 *  @param message           警告框中的内容
 *  @param frame             主窗口
 *  @param completionHandler 警告框消失调用
 */
- (void)webView:(WKWebView *)webView runJavaScriptAlertPanelWithMessage:(NSString *)message initiatedByFrame:(void (^)())completionHandler;

3. WKScriptMessageHandler

这个协议中包含一个必须实现的方法,这个方法是提高App与web端交互的关键,它可以直接将接收到的JS脚本转为OC或Swift对象。(当然,在UIWebView也可以通过“曲线救国”的方式与web进行交互,著名的Cordova框架就是这种机制)


// 从web界面中接收到一个脚本时调用
- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message;

四、WKWebView加载JS


// 图片缩放的js代码
NSString *js = @"var count = document.images.length;for (var i = 0; i < count; i++) {var image = document.images[i];image.style.width=320;};window.alert('找到' + count + '张图');"; // 根据JS字符串初始化WKUserScript对象 WKUserScript *script = [[WKUserScript alloc] initWithSource:js injectionTime:WKUserScriptInjectionTimeAtDocumentEnd forMainFrameOnly:YES]; // 根据生成的WKUserScript对象,初始化WKWebViewConfiguration WKWebViewConfiguration *config = [[WKWebViewConfiguration alloc] init]; [config.userContentController addUserScript:script]; _webView = [[WKWebView alloc] initWithFrame:self.view.bounds configuration:config]; [_webView loadHTMLString:@"<head></head><img src='http://www.nsu.edu.cn/v/2014v3/img/background/3.jpg' />"baseURL:nil];
[self.view addSubview:_webView];

五、本文Demo下载

?https://github.com/saitjr/WKWebViewSimpleDemo.git

六、关于JS的加载或WebView的其他使用技巧,可查看一下文章:

使用Safari对WebView进行调试

WebView加载HTML图片大小自适应与文章自动换行

参考资料:

http://nshipster.cn/wkwebkit/

http://www.cocoachina.com/webapp/20141121/10277.html

http://blog.csdn.net/cyforce/article/details/37657009

《WKWebView的新特性与使用》有53个想法

          1. 请问一下WKWebView 怎么修改userAgent,使用UIWebView修改userAgent的方法,在WKWebView上不起作用了…

      1. 有尝试使用过window.webkit.messageHandlers..postMessage()这个方法没?尝试了一直报js失败,或者在Safari这边调试的时候也报这个window.webkit.messageHandlers不存在。

  1. 你好,我在使用WkwebView的时候,发现在IOS8下加载本地html代码,html代码会调用本地的js/css文件时,会出现js/css文件没有被调用的情况。把编译版本设置在ios9.0,就完全没有问题,可以正常调用。请问这个问题要怎么解决?是ios8下的bug?已经确认没有被xcode将js/css文件添加到compile sources中。

    1. 请问如何解决的呢?我的也是,WKWebView加载界面内,js调用新的界面没有被出发,请前辈把解决方法发到我邮箱里面954777574@qq.com

  2. WKWebView 是不是不能加载缓存图片(Cache)中的资源?如何解决?(用js替换成本地图片不显示,UIWebView 没有问题)

      1. 我也出现类似问题,页面加载出来突然就白屏了,有些手机会,有些手机不会,内存大大的时候也会白屏,白屏的时候wkwebview的url变成了nil,但是没办法kvo 这个url,请问你们有解决的思路了么?

  3. custom://js 这类的JS 在WKWebView load的时候可以用NavgationAction的request拦截到,但是通过手势pop 回来的时候 就不识别了 这是怎么回事??

  4. 非常好的文章,看了之后马上就可以实现基本的效果了。受益匪浅。以后会多关注你的文章。

  5. wkwebview 如何点击下载,跳转到App store。

    如下js
    location.href=’https://itunes.apple.com/us/app/hua-xi-zheng-quan-shou-ji/id1068034308?l=zh&ls=1&mt=8′”

  6. wkwebview 加载网页,有1个按钮点击没响应,这个问题有遇到过吗?UIWebview加载的话没有问题,求解。

  7. 你好,项目中需要在webView下方添加评论内容,我的想法是将webView放到tableView的头部,但是遇到的问题是,如何在webView加载完成之后获取到它的真实高度?用UIWebview可以获取到,但是WKWebView获取不到,一直也没找到好的方法,请问该如何解决?????

  8. 你好,请问wkwebView怎么做白名单呢?以前UIWebView 可以用NSURLProtocol 来进行拦截,WkWebView现在也不支持了。谢谢!

  9. 你好,请问wkwebview 的window.webkit.messageHandlers.getLoadAlarmPic.postMessage(devSn) 有办法改写成 xx.getLoadAlarmPic(devsn)格式么,谢谢!

发表评论

电子邮件地址不会被公开。