WKWebView使用及注意点(keng)

作者: TIME_for | 来源:发表于2016-11-19 16:31 被阅读9322次

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

    此篇文章的逻辑图

    此篇文章的逻辑图

    WKWebView使用

    WKWebView简单介绍

    使用及注意点

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

    更改User-Agent

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

    // 获取默认User-Agent
    UIWebView *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,代码如下:

    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全部被修改。

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

    WKWebView的相关的代理方法

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

    • WKNavigationDelegate: 此代理方法中除了原有的UIWebView的四个代理方法,还增加了其他的一些方法,具体可参考我下面给出的Demo
    • WKUIDelegate: 此代理方法在使用中最好实现,否则遇到网页alert的时候,如果此代理方法没有实现,则不会出现弹框提示。
    • WKScriptMessageHandler: 此代理方法就是和JavaScript交互相关,具体介绍参考下面的专门讲解。

    WKWebView使用过程中的坑

    WKWebView下面添加自定义View

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

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

    
    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中提供了一个权限认证的代理方法,这是权限认证更为便捷。代理方法如下:

    - (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加载POST请求

    非常感谢@e231e1ff5f8b的指出,原来POST请求这儿还有一个坑。自己项目中并没有这块需求,也就没有发现。加载POST请求的时候,会丢失HTTPBody。解决办法是在网页上开一个JavaScript方法,在请求POST的时候去调用JavaScript这个方法,从而完成POST请求。调用JavaScript方法参考下面交互这一章节。

    WKWebView和JavaScript交互

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

    // 添加交互对象
    [config.userContentController addScriptMessageHandler:(id)self.ocjsHelper name:@"timefor"];
    
    /** 此点后来更新,如果不移除交互对象,则导致交互对象内存常驻(2016.12.17) */
    // VC销毁时,移除交互对象
    [self.webView.configuration.userContentController removeScriptMessageHandlerForName:@"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。参考如下:

    // 数字
    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使用中的注意点,一般也都是常用的,还有缓存等一些不是太常用的就没有具体介绍。如果在其他方面遇到问题,也欢迎你私信我共同探讨进步。WKWebView确实比UIWebView有些地方好用不少,但是一些bug至今也没解决,权限挑战是在iOS9解决的,POST请求则至今没有解决,而改变contentInset导致的点击事件不准确,同样是没有解决。这些问题让开发者使用起来,有诸多不便啊。
    此文的Demo地址:WKWebViewDemo 如果此文对你有所帮助,请给个star吧。

    更新

    • 2016.12.17
    // VC销毁时,移除交互对象;如果不移除交互对象,则导致交互对象内存常驻而引起内存泄漏
    [self.webView.configuration.userContentController removeScriptMessageHandlerForName:@"timefor"];
    

    参考

    相关文章

      网友评论

      • SDBridge:分享一个Demo. WKWebView 监听JS端的所有的console.log日志
        https://www.jianshu.com/p/09e4799c5328
        https://github.com/housenkui/WKWebView-Console/
      • 转岗做JAVA:在使用WKWebView,发现加载页面比UIWebView慢,而且刷新时候渲染的帧数也比UIWebView的帧数多好几倍,导致就算是预加载了,页面渲染还是比UIWebView视觉效果差。你们有遇到这个问题吗?
      • fcf7850dd33e:请问作者,WK不能加载ios8的HTTPS怎么解决的?证书权限代理没调用
        First灬DKS:@TIME_for 好的,非常感谢!不过我估计我们后台是不同意购买证书的:joy:
        TIME_for:@First灬DKS 最后证书替换为购买的HTTPS证书了。
        First灬DKS:请问这个问题你解决了吗?
      • 聪zero:楼主,你好,想问下,如果我加载本地html文件,然后html内部有https请求,这个时候如何做认证,页面上的请求并没有进行页面跳转好像是不会触发那说的那个认证的代理方法了
      • 星辰大海_王:楼主,我调用一个js的方法,为什么得不到结果:self.webView?.evaluateJavaScript("getDataTest()", completionHandler: { (dataResult, error) in
        print("\(error)")
        })。打印error:Optional(Error Domain=WKErrorDomain Code=4 "发生 JavaScript 异常" UserInfo={WKJavaScriptExceptionLineNumber=1, WKJavaScriptExceptionMessage=ReferenceError: Can't find variable: getDataTest, WKJavaScriptExceptionSourceURL=file:///var/containers/Bundle/Application/FDB63A82-2887-4F0E-918B-D67AEA8311D7/rete.app/ptryur/details.html, NSLocalizedDescription=发生 JavaScript 异常, WKJavaScriptExceptionColumnNumber=12})
        error in __connection_block_invoke_2: Connection interrupted
        星辰大海_王:@gojan html里面有这个方法
        076f37cc3cd0:就报错信息来看,是details.html里js找不到变量getDataTest() 最好检查一下html 可以先查看是否存在function getDataTest
        ad193b2e3cf5:楼主,同问。为什么这问题没人回答
      • 木子影: 有个问题
      • 酷哥不回头看爆炸:你好 ,我想请教一下wkwebview 读取localStorage 的问题。 方便加一下qq吗 425611381 或者看一下我给你的私信
      • HuberCui:大神,如果iOS用WKWebView,交互的时候安卓应该怎么办呢?
        小千:关键是android是同步,iOS的wkwebview时异步的,之前为了兼容iOS7和减轻前端工作量,用uiwebview实现了同步,现在要换成wkwebview,前端就开始推脱不想做分平台,:joy:
        HuberCui:@TIME_for 谢谢楼主
        TIME_for:让 前端区分 iOS 和 Android。做不同的处理。
      • 3f58e786b546:知道我为啥老和你聊天么?因为我喜欢你
      • 以技术之名:你好我想问下,交互对象只能定义一个还是多个?
        TIME_for:可以定义多个。
      • 56d35fdf1a2a:大神 我又来请教了 我按照你写的WKWebview的方法进行js与oc交互 还是有问题
        Error Domain=WKErrorDomain Code=4 "A JavaScript exception occurred" UserInfo={WKJavaScriptExceptionLineNumber=1, WKJavaScriptExceptionMessage=ReferenceError: Can't find variable: showCallback, WKJavaScriptExceptionSourceURL=....(前端提供给我的url), NSLocalizedDescription=A JavaScript exception occurred, WKJavaScriptExceptionColumnNumber=13}

        showCallback是这样写的
        var showCallback = function(devideId){
        alert(devideId);
        }
        是我漏写了什么吗 这种问题怎么解决呢
        星辰大海_王:请问怎么解决的
        来自外太空:你好,我也出现这种问题,请问怎么解决的
        TIME_for:检查一下 showCallback 是否是一个全局函数。
      • 是从什么盛:关于UA问题,在替换WK之后还是使用的原有UA写入,并没有什么问题,只是改变请求头内数据内容.感觉与webview与wk容器无关
        TIME_for:使用原来的是没问题。 感觉与webview与wk容器无关,这句没太明白啥意思。。
      • 丶生如夏花:作者你好。我是昨天在伯乐上,说追加尾图有问题的人。
        你可以看一下你的追加div部分吗?执行那段JS,我从block里面打印出信息是存在类型不支持的,会不会因为这个,所以不是过于复杂的页面造成的问题,是这段JS没生效…
        丶生如夏花:@TIME_for 嗯嗯, 好的, 也是辛苦了。我发现我从服务器获得的网址, 他自身HTML代码有张图片加载不出来了, 所以导致位置页不对。但是, 去除图片后, 还是存在遮挡最后一行文字约半行左右的高度, 我也不知道为什么了。
        TIME_for:你说的那个问题,是界面缩放引起的,界面缩放的时候,这时候,位置就不准确了,登录页面进去就有缩放。这个开始还真是没有注意到。
      • XTShow:您好,有个问题想请教您一下,在您这篇文章的《WKWebView下面添加自定义View》部分中,添加了addView后调用监听,处理addView的frame,然后您的demo里紧接着打印了self.webView.scrollView.contentSize,这里我发现在处理后,会让横向也产生了一个增量变化,也就是不是屏幕宽度了,这个是为什么呢?是js语句的原因吗?(本人js0基础。。。见谅。。。)
        XTShow:@TIME_for 学习了!谢谢!:grin:
        TIME_for:自己写的 test.html 中,写的 ul li 这两个 CSS 引起的。可以不使用这两个样式试一下,即可消除这个问题。
      • 07d93406ec39:WKWebViewConfiguration 的 selectionGranularity 設置為 character 的問題,似乎從來沒有人提到
        TIME_for:@Ikaros_vv 这个还真没有太注意,多谢提醒啊。
        07d93406ec39:@TIME_for 用dynamic 的在跨段落選取時會改變選取行為,而character 則會看不到選取框。。
        TIME_for:和用默认的有什么区别?
      • steveyangwj:WkWebView不支持缓存啊,硬伤啊
        TIME_for:iOS9 之后,系统自动做了缓存 WKWebsiteDataStore 这个相关的类可对缓存进行管理。但是无法向 UIWebView 那样 拦截 NSURLProtocol 自定义缓存。
      • qingchen91:解决了很大的问题,很详细。
        TIME_for:@索马里11 共同进步~~
      • 小杨VBoy:关键的地方。cookie在wkwebview中使用并没有介绍
        零零321:请问下 cookie在wk中是怎么同步的
        TIME_for:@小杨VBoy 项目中没有用到,可以私下探讨哦。
      • KennyMcCormick:大神思维导图用什么画的,效果这么好。
        KennyMcCormick:@TIME_for 谢谢
        TIME_for:@KennyMcCormick MindLine
      • 夏妍妍:知道我为啥老和你聊天么?因为我喜欢你
        TIME_for:@夏妍妍 :sweat:
        夏妍妍:@年光逝也被僵尸号占了 哈哈 忘了
        空转风:@夏妍妍 喜欢不关注?
      • e231e1ff5f8b:楼主,你好!请教下您,能否写一篇关于WKWebView POST 带参请求的博客……
        TIME_for:@_超 是的啊。
        超_iOS:@TIME_for 这个POST 真累人啊
        TIME_for:@e231e1ff5f8b 由于没用WKWebView发送过POST的请求,不知道这儿还有个坑,http://stackoverflow.com/questions/26253133/cant-set-headers-on-my-wkwebview-post-request 这儿已经给了很好的解决方案。回头我在这篇文章和Demo中,在更新一下。非常感谢你的指出。

      本文标题:WKWebView使用及注意点(keng)

      本文链接:https://www.haomeiwen.com/subject/azpdpttx.html