最近一直在作有關Swift和JavaScript交互的程序,因此有關UIWebView和WKWebView在使用上的差異在此總結下:html
UIWebView:java
(1)建立git
var webView: UIWebView! self.webView = UIWebView.init(frame: CGRect.init(x: 0, y: 0, width: kScreenWidth, height: kScreenHeight)) self.view.addSubview(self.webView!) self.webView.delegate = self
(2)請求github
let url = "" let request = NSURLRequest(URL: NSURL(string: url)!) self.webView.loadRequest(request)
(3)經常使用代理web
UIWebViewDelegate func webViewDidStartLoad(webView: UIWebView) { } func webView(webView: UIWebView, didFailLoadWithError error: NSError?) { } func webViewDidFinishLoad(webView: UIWebView) { }
UIWebView最屌的固然仍是結合傳說中的JavaScriptCore的Hybrid方式了cookie
(4)在UIWebView中注入(執行)JS (注意要在webViewDidFinishLoad回調,也就是加載完後才能執行)框架
//第一種方法,直接webview執行(應該是封裝了下面的方法二) let resultStr="1234" self.webView.stringByEvaluatingJavaScriptFromString("theFunc('"+resultStr+"');")! as String //第二種,使用kvc來取得mainframe的context,而後執行 let context = webView.valueForKeyPath("documentView.webView.mainFrame.javaScriptContext") as! JSContext context.evaluateScript(theScript)
(5)在UIWebView中監聽JS的函數,這種方式能夠經過原生return來call back,不過return是一種同步的call back。不是很是好。
異步
let context = webView.valueForKeyPath("documentView.webView.mainFrame.javaScriptContext") as! JSContext //假如JS傳給原生一個JSON的字符串 let accountSerial: @convention(block) String -> () = { _ in let value = JSContext.currentArguments().first let data = value?.toString().dataUsingEncoding(NSUTF8StringEncoding) let accountInfoDic = try? NSJSONSerialization.JSONObjectWithData(data!, options:[]) as! NSDictionary } context.setObject(unsafeBitCast(accountSerial, AnyObject.self), forKeyedSubscript: "saveAccountInfo") //假如須要原生return一個JSON的字符串 let cookiesSerial : @convention(block) () -> String = { let cookies = NSHTTPCookieStorage.sharedHTTPCookieStorage().cookiesForURL((webView.request?.URL)!) let header=NSHTTPCookie.requestHeaderFieldsWithCookies(cookies!) as NSDictionary if(header.objectForKey("Cookie") != nil) { return cookies.toJsonString() } }
(6)另一種奇葩的經過URL攔截的方式,讓JS調用原生ide
http://www.cnblogs.com/rayshen/p/4560728.html函數
WKWebView:
(1)建立,須要import的框架和繼承的協議
//須要導入 import WebKit class MXWebDemoController:UIViewController,WKNavigationDelegate,WKUIDelegate,WKScriptMessageHandler{} private var webView:WKWebView! private var progressView:UIProgressView! //進度條 progressView = UIProgressView(progressViewStyle: UIProgressViewStyle.Default) progressView.frame = CGRectMake(0, 0, UIScreen.mainScreen().bounds.width, 22) progressView.tintColor = UIColor.blueColor() self.view.addSubview(progressView) //初始化 let conf = WKWebViewConfiguration() webView = WKWebView(frame: CGRectMake(0,0,kScreenWidth,kScreenHeight), configuration: conf) self.view.insertSubview(webView, belowSubview: progressView)
(2)請求
let request = NSURLRequest(URL: NSURL(string:baseUrl)!, cachePolicy: .ReloadIgnoringLocalCacheData, timeoutInterval: 10)
self.webView.loadRequest(request)
(3)經常使用代理
//webview加載完,重置進度條 func webView(webView: WKWebView, didFinishNavigation navigation: WKNavigation!) { progressView.setProgress(0.0, animated: false) } //kvo監聽 override func observeValueForKeyPath(keyPath: String?, ofObject object: AnyObject?, change: [String : AnyObject]?, context: UnsafeMutablePointer<Void>) { if keyPath == "estimatedProgress" { progressView.hidden = webView.estimatedProgress == 1 progressView.setProgress(Float(webView.estimatedProgress), animated: true) } } //和JS消息交互 // MARK: - WKScriptMessageHandler func userContentController(userContentController: WKUserContentController, didReceiveScriptMessage message: WKScriptMessage) { if(message.name == "mxBack"){ self.navigationController?.popViewControllerAnimated(true) } }
(4)JS注入,最基本的就是evaluateJavaScript函數,在頁面加載完後去執行。再而,WKWebView新增了JS注入的接口。
//方法1 let jsStr = "" webView.evaluateJavaScript(jsStr),completionHandler: { (object, error) in if(error != nil){ print(error) } }) //方法2 let script = WKUserScript(source:self.scriptStr,injectionTime: .AtDocumentStart,forMainFrameOnly: true)
(5)接受來自JS的消息
//1 conf.userContentController.addScriptMessageHandler(self, name:"mxBack") //2 func userContentController(userContentController: WKUserContentController, didReceiveScriptMessage message: WKScriptMessage) { weak var weakself = self if(message.name == "mxBack"){ self.hasGetResult = true self.gotoBack() } }
有關WKWebView,有個缺點,就是和JS之間沒有同步的函數(依靠return來回傳的方式)。
以前大部分iOS項目都是使用UIWebView,如今iOS8後Webkit良好的性能取代了UIWebView,但它的異步對不少舊的Web工程都有點影響。
別人的Demo:
https://github.com/CoderJackyHuang/WKWebViewTestDemo
下週有空我會寫個更好的,更詳細的介紹JS和原生之間的交互。