iOS-WebKit学习记录

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;

所以以上方法的调用顺序是:

  1. decidePolicyForNavigationAction
  2. didStartProvisionalNavigation
  3. decidePolicyForNavigationResponse
  4. didCommitNavigation
  5. 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。

你可能感兴趣的:(iOS-WebKit学习记录)