WKWebView中Native與Web交互

數(shù)據(jù)交互的途徑

  1. 攔截URL
  2. MessageHandler
  3. 注入JS代碼
  4. evalueJavascript函數(shù)
  5. cookie
  6. 攔截http請求

攔截URL

實現(xiàn)關鍵: WKNavigationDelegate.

通過實現(xiàn)WKNavigationDelegate的回調(diào)函數(shù)webView(_:decidePolicyFor:decisionHandler:)來攔截Web頁面跳轉的URL, Web端可以把數(shù)據(jù)放在URL中.

MessageHandler

實現(xiàn)關鍵: WKScriptMessageHandler, WKWebViewConfiguration.

通過WKWebViewConfiguration, 向webView添加實現(xiàn)了WKScriptMessageHandler協(xié)議的對象, 監(jiān)聽來自JS端的調(diào)用.
必須在WKWebView初始化的時候完成添加.

let config = WKWebViewConfiguration()
// someHandler是實現(xiàn)了WKScriptMessageHandler的實例化對象
config.userContentController.add(someHandler, name: "nameOfSomeHandler")
let web = WKWebView(frame: .zero, configuration: config)

注入JS代碼

實現(xiàn)關鍵: WKUserScript, WKWebViewConfiguration.
WKUserScript包裝腳本代碼, 通過WKWebViewConfiguration, 向webView添加可供Web端執(zhí)行的代碼.

let config = WKWebViewConfiguration()
// someScriptString是腳本代碼字符串
let someScriptString = "..."
config.userContentController.addUserScript(WKUserScript(source: someScriptString, injectionTime: .atDocumentStart, forMainFrameOnly: true))
let web = WKWebView(frame: .zero, configuration: config)

evalueJavascript

通過WKWebView提供的函數(shù)evalueJavascript(_:completionHandler:), 從Native端調(diào)用Web端的JS函數(shù).

/*
@abstract Evaluates the given JavaScript string.
@param javaScriptString The JavaScript string to evaluate.
@param completionHandler A block to invoke when script evaluation completes or fails.
@discussion The completionHandler is passed the result of the script evaluation or an error.
*/
open func evaluateJavaScript(_ javaScriptString: String, completionHandler: ((Any?, Error?) -> Void)? = nil)

cookie

iOS 中 cookie 有專門的類型HTTPCookie, iOS11 以前WKWebView的 cookie 和 app 共享, 在 iOS11 以后WKWebView的 cookie 會單獨自己管理.
獲取cookie
WKNavigationDelegate的回調(diào)webView(_:decidePolicyFor:decisionHandler:)參數(shù)中, 有一個WKNavigationResponse對象, 有幾種方式可以從中取出cookie:

  1. HTTPCookie的靜態(tài)函數(shù)cookies(withResponseHeaderFields:for:)
  2. HTTPCookieStorage(iOS 11以前和 WebKit 共享, iOS11以后WebKit 不包含在內(nèi))
  3. WKHTTPCookieStore(iOS 11以后)
func webView(_ webView: WKWebView, decidePolicyFor navigationResponse: WKNavigationResponse, decisionHandler: @escaping (WKNavigationResponsePolicy) -> Void) {
  if
    let response = navigationResponse.response as? HTTPURLResponse,
    let headers = response.allHeaderFields as? [String : String],
    let url = response.url
  {
      // 1.通過HTTPCookie獲取
      HTTPCookie.cookies(withResponseHeaderFields: headers, for: url) // 
      // 2.通過HTTPCookieStorage獲取
      HTTPCookieStorage.shared.cookies(for: url) // 獲取對應此 URL 的 cookie
      HTTPCookieStorage.shared.cookies // 獲取全部 cookie
  }
  // 3.通過WKHTTPCookieStore獲取
  webView.configuration.websiteDataStore.httpCookieStore.getAllCookies { (cookies) in

  }
  decisionHandler(.allow)
}

設置 cookie
可以通過以下方式:

  1. HTTPCookieStorage(iOS 11以前)
  2. WKHTTPCookieStore(iOS 11以后)
let url = URL(string: "http://www.reibang.com")!
        
if let cookie = HTTPCookie(properties: [HTTPCookiePropertyKey.commentURL: url]) {
     
    if #available(iOS 11.0, *) {
        // 2.WKHTTPCookieStore設置
        webView.configuration.websiteDataStore.httpCookieStore.setCookie(cookie, completionHandler: nil)
    }else {
        // 1.HTTPCookieStorage設置
        HTTPCookieStorage.shared.setCookie(cookie)
        HTTPCookieStorage.shared.setCookies([cookie], for: url, mainDocumentURL: url) // 單獨針對某 URL 設置
    }
}

補充: UIWebViewWKWebView 不一樣, 它是通過HTTPCookieStorage和 APP 以及系統(tǒng)共享 cookie 的.

攔截自定義http請求

iOS11以后可以使用WKURLSchemeHandler協(xié)議攔截指定scheme的http請求.
在iOS13以前可以通過私有API攔截scheme為http/https, iOS13后無法再訪問私有變量, 因而無法實現(xiàn)此類請求的攔截.

關鍵: WKWebViewConfiguration, WKURLSchemeHandler, WKURLSchemeTask.

  • 自定義一個實現(xiàn)了WKURLSchemeHandler協(xié)議的類型, 在回調(diào)中處理對應scheme的請求.
class SomeURLSchemeHandler: NSObject, WKURLSchemeHandler {
    func webView(_ webView: WKWebView, start urlSchemeTask: WKURLSchemeTask) {
        // 可以自己 直接創(chuàng)建 URLResponse 做回調(diào), 或者另外調(diào)用網(wǎng)絡請求獲得 URLResponse
        ...
    }
    // 任務被告知要停止
    func webView(_ webView: WKWebView, stop urlSchemeTask: WKURLSchemeTask) {
        // 可以在此處把 webView(_:start:) 處發(fā)起的異步任務停掉
        ...
    }
}
  • 自定義類需要在回調(diào)webView(_:start:)中操作WKURLSchemeTask對象, 根據(jù)情況調(diào)用對應的函數(shù):
/*
  如果任務已經(jīng)被停止, 調(diào)用以下函數(shù)就會拋出異常
*/

/*
  把自定義的響應傳遞過去, 必須調(diào)用至少1次.
  如果任務已經(jīng)標記完成后再調(diào)用, 會拋出異常.
*/
func didReceive(_ response: URLResponse)
/*
  把響應數(shù)據(jù)傳遞進去, 必須在didReceive(_ response: URLResponse)之后調(diào)用
  如果任務已經(jīng)標記完成后再調(diào)用, 會拋出異常.
*/
func didReceive(_ data: Data)
/*
  在響應成功后調(diào)用
  如果任務已經(jīng)標記完成或者失敗后再調(diào)用, 會拋出異常
*/
func didFinish()
/*
  在響應出錯后調(diào)用
  如果任務已經(jīng)標記完成或者失敗后再調(diào)用, 會拋出異常
*/
func didFailWithError(_ error: Error)
  • WKWebView初始化的時候, 把自定義請求攔截配置好.
let config = WKWebViewConfiguration()
config.setURLSchemeHandler(someURLSchemeHandler, forURLScheme: "someScheme")
let webView = WKWebView(frame: .zero, configuration: config)

小結

  • MessageHandler/JS 代碼注入/請求攔截, 這3個方案都需要依賴WKWebViewConfiguration, 在 WebView 初始化的時候就要完成配置.
  • iOS13后, 無法通過私有API攔截 http/httpsscheme 的請求, 所以 hybrid 的時候很難做到對 web 端無入侵.
  • 為了做加載優(yōu)化(資源, 數(shù)據(jù)), 可以使用自定義 scheme 的請求, 或者 web 端調(diào)用 MessageHandler 告知 native 端, 這需要兩端協(xié)商加載的方案.
  • iOS11以后, WKWebView 的cookie不再是共享的, 需要單獨針對維護.
  • 獲取WKWebView的cookie可以從 WKNavigationResponse中獲取.
最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末峰弹,一起剝皮案震驚了整個濱河市寝并,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌塑陵,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,548評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件蜡励,死亡現(xiàn)場離奇詭異令花,居然都是意外死亡,警方通過查閱死者的電腦和手機凉倚,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,497評論 3 399
  • 文/潘曉璐 我一進店門兼都,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人稽寒,你說我怎么就攤上這事扮碧。” “怎么了杏糙?”我有些...
    開封第一講書人閱讀 167,990評論 0 360
  • 文/不壞的土叔 我叫張陵慎王,是天一觀的道長。 經(jīng)常有香客問我宏侍,道長赖淤,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,618評論 1 296
  • 正文 為了忘掉前任谅河,我火速辦了婚禮咱旱,結果婚禮上,老公的妹妹穿的比我還像新娘绷耍。我一直安慰自己吐限,他們只是感情好,可當我...
    茶點故事閱讀 68,618評論 6 397
  • 文/花漫 我一把揭開白布褂始。 她就那樣靜靜地躺著诸典,像睡著了一般。 火紅的嫁衣襯著肌膚如雪崎苗。 梳的紋絲不亂的頭發(fā)上搂赋,一...
    開封第一講書人閱讀 52,246評論 1 308
  • 那天,我揣著相機與錄音益缠,去河邊找鬼脑奠。 笑死,一個胖子當著我的面吹牛幅慌,可吹牛的內(nèi)容都是我干的宋欺。 我是一名探鬼主播,決...
    沈念sama閱讀 40,819評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼齿诞!你這毒婦竟也來了酸休?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 39,725評論 0 276
  • 序言:老撾萬榮一對情侶失蹤祷杈,失蹤者是張志新(化名)和其女友劉穎斑司,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體但汞,經(jīng)...
    沈念sama閱讀 46,268評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡宿刮,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,356評論 3 340
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了私蕾。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片僵缺。...
    茶點故事閱讀 40,488評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖踩叭,靈堂內(nèi)的尸體忽然破棺而出磕潮,到底是詐尸還是另有隱情,我是刑警寧澤容贝,帶...
    沈念sama閱讀 36,181評論 5 350
  • 正文 年R本政府宣布自脯,位于F島的核電站,受9級特大地震影響斤富,放射性物質發(fā)生泄漏冤今。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,862評論 3 333
  • 文/蒙蒙 一茂缚、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧屋谭,春花似錦脚囊、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,331評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至我擂,卻和暖如春衬以,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背校摩。 一陣腳步聲響...
    開封第一講書人閱讀 33,445評論 1 272
  • 我被黑心中介騙來泰國打工看峻, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人衙吩。 一個月前我還...
    沈念sama閱讀 48,897評論 3 376
  • 正文 我出身青樓互妓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子冯勉,可洞房花燭夜當晚...
    茶點故事閱讀 45,500評論 2 359