iOS WKWebView 详解
来源:互联网 发布:js里的值给input 编辑:程序博客网 时间:2024/06/08 16:11
原文地址:http://www.brighttj.com/ios/ios-wkwebview-new-features-and-use.html
在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界面的三种提示框(警告框、确认框、输入框)分别对应三种代理方法。下面只举了警告框的例子。
在WKWebview中,js的alert是不会出现任何内容的,你必须重写WKUIDelegate
委托的runJavaScriptAlertPanelWithMessage message
方法,自己处理
/** * 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对象,初始化WKWebViewConfigurationWKWebViewConfiguration *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/37657009runJavaScriptAlertPanelWithMessage:
WKWebView进度及title
WKWebView 的estimatedProgress和title 都是KVO模式,所以可以添加监控:
<code class="hljs php has-numbering"> [webView addObserver:<span class="hljs-keyword">self</span> forKeyPath:@<span class="hljs-string">"estimatedProgress"</span> options:NSKeyValueObservingOptionNew context:<span class="hljs-keyword">NULL</span>]; [webView addObserver:<span class="hljs-keyword">self</span> forKeyPath:@<span class="hljs-string">"title"</span> options:NSKeyValueObservingOptionNew context:<span class="hljs-keyword">NULL</span>];</code>监控的实现方法:
<code class="hljs objectivec has-numbering">- (<span class="hljs-keyword">void</span>)observeValueForKeyPath:(<span class="hljs-built_in">NSString</span> *)keyPath ofObject:(<span class="hljs-keyword">id</span>)object change:(<span class="hljs-built_in">NSDictionary</span> *)change context:(<span class="hljs-keyword">void</span> *)context { <span class="hljs-keyword">if</span> ([keyPath isEqualToString:@<span class="hljs-string">"estimatedProgress"</span>]) { <span class="hljs-keyword">if</span> (object == webView) { [<span class="hljs-keyword">self</span><span class="hljs-variable">.progressView</span> setAlpha:<span class="hljs-number">1.0</span>f]; [<span class="hljs-keyword">self</span><span class="hljs-variable">.progressView</span> setProgress:<span class="hljs-keyword">self</span><span class="hljs-variable">.currentSubView</span><span class="hljs-variable">.webView</span><span class="hljs-variable">.estimatedProgress</span> animated:<span class="hljs-literal">YES</span>]; <span class="hljs-keyword">if</span>(<span class="hljs-keyword">self</span><span class="hljs-variable">.currentSubView</span><span class="hljs-variable">.webView</span><span class="hljs-variable">.estimatedProgress</span> >= <span class="hljs-number">1.0</span>f) { [<span class="hljs-built_in">UIView</span> animateWithDuration:<span class="hljs-number">0.3</span> delay:<span class="hljs-number">0.3</span> options:UIViewAnimationOptionCurveEaseOut animations:^{ [<span class="hljs-keyword">self</span><span class="hljs-variable">.progressView</span> setAlpha:<span class="hljs-number">0.0</span>f]; } completion:^(<span class="hljs-built_in">BOOL</span> finished) { [<span class="hljs-keyword">self</span><span class="hljs-variable">.progressView</span> setProgress:<span class="hljs-number">0.0</span>f animated:<span class="hljs-literal">NO</span>]; }]; } } <span class="hljs-keyword">else</span> { [<span class="hljs-keyword">super</span> observeValueForKeyPath:keyPath ofObject:object change:change context:context]; } } <span class="hljs-keyword">else</span> <span class="hljs-keyword">if</span> ([keyPath isEqualToString:@<span class="hljs-string">"title"</span>]) { <span class="hljs-keyword">if</span> (object == <span class="hljs-keyword">self</span><span class="hljs-variable">.webView</span>) { <span class="hljs-keyword">self</span><span class="hljs-variable">.title</span> = <span class="hljs-keyword">self</span><span class="hljs-variable">.webView</span><span class="hljs-variable">.title</span>; } <span class="hljs-keyword">else</span> { [<span class="hljs-keyword">super</span> observeValueForKeyPath:keyPath ofObject:object change:change context:context]; } } <span class="hljs-keyword">else</span> { [<span class="hljs-keyword">super</span> observeValueForKeyPath:keyPath ofObject:object change:change context:context]; }}</code>这里的进度增加了动画,类似safari的进度效果
需要注意的是销毁的时候一定要移除监控
<code class="hljs bash has-numbering"> [webView removeObserver:self <span class="hljs-keyword">for</span>KeyPath:@<span class="hljs-string">"estimatedProgress"</span>]; [webView removeObserver:self <span class="hljs-keyword">for</span>KeyPath:@<span class="hljs-string">"title"</span>];</code>
- iOS WKWebView 详解
- ios WKWebView详解
- IOS WKWebView
- 【iOS】WKWebView
- iOS - WKWebView
- IOS-WKWebView
- iOS网络3—UIWebView与WKWebView使用详解
- iOS UIWebView和WKWebView的JS调用详解
- iOS-WKWebView使用
- IOS 进阶之 WKWebView
- iOS WKWebView的使用
- iOS开发 之 WKWebView
- iOS Objective-C WKWebView
- iOS WKWebView js交互
- ##iOS 8新增 WKWebView
- iOS-WKWebView的封装
- iOS wkwebview 弹出键盘
- ios 清 wkwebview 缓存
- 上传文件及参数
- PHP实现快速排序
- C++标准库之mutex
- Android Studio Logcat显示不全问题
- LeetCode *** 283. Move Zeroes
- iOS WKWebView 详解
- Fragment+ViewPager
- 用 Java 实现断点续传 (HTTP)
- 二叉树的java实现和遍历
- 在javascript或者jQuery中绑定按钮点击事件,和在HTML 标签中直接调用onclick属性有什么区别?
- JS中判断null、undefined与NaN的方法
- aerospike 代码阅读
- 【笔记】 《js权威指南》- 第14章 Window对象 14.8 多窗口和窗体
- Netty in Action (四)第一章节 第三部分 Netty核心组件