WKNavigationDelegate
- (void)webView:(WKWebView *)webView decidePolicyForNavigationAction:(WKNavigationAction *)navigationAction decisionHandler:(void (^)(WKNavigationActionPolicy))decisionHandler;
webview跳转之前调用,可以根据navigationAction决定是否要进行跳转,即webview是否需要加载新的request。
官方解释:
If you do not implement this method, the web view will load the request or, if appropriate, forward it to another application.
如果不实现这个代理方法,webview会默认加载这个request,或者如果可以的话,打开另一个APP。
虽然这个方法是optional的,但是在实际使用中,应该实现这个方法。并且应该调用decisionHandler回调。否则会出现加载不出页面等其他意想不到的情况。
- (void)webView:(WKWebView *)webView decidePolicyForNavigationResponse:(WKNavigationResponse *)navigationResponse decisionHandler:(void (^)(WKNavigationResponsePolicy))decisionHandler;
webview在获取到页面返回信息后决定是否跳转的代理方法。如果此时decisionHandler(WKNavigationResponsePolicyCancel),则webview不加载新的请求,不显示新的界面。
官方解释:
If you do not implement this method, the web view will allow the response, if the web view can show it.
如果不实现这个代理方法,webview会在能够显示的时候响应返回信息,加载新界面。但是实际使用中应该主动实现这个方法,并且调用decisionHandler(WKNavigationResponsePolicyAllow)。否则会出现无法加载新界面或者其他意向不到的情况。
webview开始加载新页面时调用此方法,该方法调用时页面还没有变化
- (void)webView:(WKWebView *)webView didStartProvisionalNavigation:(null_unspecified WKNavigation *)navigation;
webview开始加载新页面时调用此方法,当进入新页面(显示新页面)时,此方法被调用
- (void)webView:(WKWebView *)webView didCommitNavigation:(null_unspecified WKNavigation *)navigation;
webView新页面加载完成,页面元素完全显示后调用此方法。
- (void)webView:(WKWebView *)webView didFinishNavigation:(null_unspecified WKNavigation *)navigation;
所以以上方法的调用顺序是:
- decidePolicyForNavigationAction
- didStartProvisionalNavigation
- decidePolicyForNavigationResponse
- didCommitNavigation
- didFinishNavigation
WKUIDelegate
- (nullable WKWebView *)webView:(WKWebView *)webView createWebViewWithConfiguration:(WKWebViewConfiguration *)configuration forNavigationAction:(WKNavigationAction *)navigationAction windowFeatures:(WKWindowFeatures *)windowFeatures
对于webview的跳转,会调用navigationDelete协议组中的代理方法。当调用decidePolicyForNavigationAction方法时,会带有参数navigationAction,这个参数里面会有sourceFrame和targetFrame。当targetFrame的mainFrame属性是NO时,表明这个跳转要打开一个新的页面。这个时候回调用以上的UIdelegate代理方法。如果,没有实现以上的方法,则不会生成新的webview,不会建立新的页面。用户的交互会被取消掉,什么也不会发生。
WKUIDelegate类里有三个关于弹框的代理方法,如果需要JS进行弹框提示,就要实现这三个方法中对应的那个。否则无法看到弹框提示。使用举例如下:
- (void)webView:(WKWebView *)webView runJavaScriptConfirmPanelWithMessage:(NSString *)message initiatedByFrame:(WKFrameInfo *)frame completionHandler:(void (^)(BOOL))completionHandler{
// DLOG(@"msg = %@ frmae = %@",message,frame);
UIAlertController *alertController = [UIAlertController alertControllerWithTitle:@"提示" message:message?:@"" preferredStyle:UIAlertControllerStyleAlert];
[alertController addAction:([UIAlertAction actionWithTitle:@"取消" style:UIAlertActionStyleCancel handler:^(UIAlertAction * _Nonnull action) {
completionHandler(NO);
}])];
[alertController addAction:([UIAlertAction actionWithTitle:@"确认" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
completionHandler(YES);
}])];
[self presentViewController:alertController animated:YES completion:nil];
}
WKUserContentController
WKUserContentController相当于一个调度器。可以通过给这个contentController添加handler,达到给JS提供调用OC的方法的目的。
比如:OC端添加[userContentController addScriptMessageHandler:self name:@"JSCallOC"];
,并且定义一个JSCallOC的方法。OC端还要在合适的controller遵守WKScriptMessageHandler协议。实现方法
- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message{
NSLog(@"方法名:%@", message.name);
NSLog(@"参数:%@", message.body);
// 方法名
NSString *methods = [NSString stringWithFormat:@"%@:", message.name];
SEL selector = NSSelectorFromString(methods);
// 调用方法
if ([self respondsToSelector:selector]) {
[self performSelector:selector withObject:message.body];
} else {
NSLog(@"未实行方法:%@", methods);
}
}
这样在JS端调用window.webkit.messageHandlers.jsCallOC.postMessage({});
此处需要在调用时传递一个对象过去,否则方法不会被调用。
一次JS调用OC就实现了。
OC调用JS,简单的实现就调用方法
- (void)evaluateJavaScript:(NSString *)javaScriptString completionHandler:(void (^ _Nullable)(_Nullable id, NSError * _Nullable error))completionHandler;
这个方法相比较UIWebview的evaluateJavaScript,有了回调,就能够知道调用的结果。
userAgent
属性是一个只读的字符串,声明了浏览器用于 HTTP 请求的用户代理头的值。使得服务器能够识别客户使用的操作系统及版本、CPU 类型、浏览器及版本、浏览器渲染引擎、浏览器语言、浏览器插件等。
userAgent实例:Mozilla/5.0 (iPhone; CPU iPhone OS 9_1 like Mac OS X) AppleWebKit/601.1.46 (KHTML, like Gecko) Version/9.0 Mobile/13B143 Safari/601.1
cookies
WKwebview 在加载HTTP请求时不会同步 NSHTTPCookiesStorage中已经存在的cookies,这会导致页面无法获取到cookies等问题。解决的方法简单来说就是绕过NSHTTPCookiesStorage或改造加强NSHTTPCookiesStorage。
首先,把自己关心的cookies存到全局变量中。
其次,在需要加载请求时,通过设置request.allHTTPHeaderFields(此时request是NSURLMutableRequest类型)。把自己保存的Cookie加载上,确保网页能获取到cookie。
再次,网页本身进行ajax请求时,可能会获取不到cookies。可以把本地的Cookie通过userContentController addUserScript的方法,添加给网页。
具体的实现方法见:http://www.jianshu.com/p/870dba42ec15
tips
-
WKWebview的bacForwardList属性是用来存放浏览器前进后退的条目的。这个list包括当前的WKBackForWordListItem和前一页面的WKBackForWordListItem(如果有的话),以及后一页面的
@property (readonly, copy) NSURL \*URL
;和请求页面时的初始请求URL.@property (readonly, copy) NSURL \*initialURL
; -
WKFrameInfo是一个存储webview相关信息的类。这个类是临时性的。包括
//@abstract A Boolean value indicating whether the frame is the main frame
// or a subframe.
@property (nonatomic, readonly, getter=isMainFrame) BOOL mainFrame;判断当前的webview是不是主frame,当有多个webview时起作用。类似于浏览器有多个窗口,判断是不是主窗口。
//@abstract The frame's current request.
@property (nonatomic, readonly, copy) NSURLRequest *request;
//@abstract The frame's current security origin.
@property (nonatomic, readonly) WKSecurityOrigin *securityOrigin API_AVAILABLE(macosx(10.11), ios(9.0)//个人理解因为原先获取request属性时,有可能出现获取失败,拿不到数据的情况。所以添加这个属性。保证能够安全的得到请求的request。WKSecurityOrigin包括request的protocol,host,port。但是不包括path。
网友评论