让WKWebView支持NSURLProtocol
生活随笔
收集整理的這篇文章主要介紹了
让WKWebView支持NSURLProtocol
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
NSURLProtocol能夠攔截UIWebView內所有的請求,但是WKWebView 中的請求卻完全不遵從這一規則,只是象征性調用canInitWithRequest方法,之后的整個請求流程似乎就與 NSURLProtocol 完全無關了。使我一度認為WKWebView請求不遵守NSURLProtocol協議。
其實是能夠做到的,因為WKWebView沒有開放相關的API,所以我們需要使用私有API調用來實現。使用WKBrowsingContextController和registerSchemeForCustomProtocol。 通過反射的方式拿到了私有的 class/selector。通過kvc取到browsingContextController。通過把注冊把 http 和 https 請求交給 NSURLProtocol 處理。
[NSURLProtocol wk_registerScheme:@"http"]; [NSURLProtocol wk_registerScheme:@"https"];下面直接上源代碼吧
//FOUNDATION_STATIC_INLINE 屬于屬于runtime范疇,你的.m文件需要頻繁調用一個函數,可以用static inline來聲明。在SDWebImage讀取內存的緩存也用到這個聲明。 FOUNDATION_STATIC_INLINE Class ContextControllerClass() { static Class cls; if (!cls) { cls = [[[WKWebView new] valueForKey:@"browsingContextController"] class]; } return cls; }FOUNDATION_STATIC_INLINE SEL RegisterSchemeSelector() { return NSSelectorFromString(@"registerSchemeForCustomProtocol:"); }FOUNDATION_STATIC_INLINE SEL UnregisterSchemeSelector() { return NSSelectorFromString(@"unregisterSchemeForCustomProtocol:"); }@implementation NSURLProtocol (WebKitSupport)+ (void)wk_registerScheme:(NSString *)scheme { Class cls = ContextControllerClass(); SEL sel = RegisterSchemeSelector(); if ([(id)cls respondsToSelector:sel]) { // 放棄編輯器警告 #pragma clang diagnostic push #pragma clang diagnostic ignored "-Warc-performSelector-leaks" [(id)cls performSelector:sel withObject:scheme]; #pragma clang diagnostic pop } }+ (void)wk_unregisterScheme:(NSString *)scheme { Class cls = ContextControllerClass(); SEL sel = UnregisterSchemeSelector(); if ([(id)cls respondsToSelector:sel]) { // 放棄編輯器警告 #pragma clang diagnostic push #pragma clang diagnostic ignored "-Warc-performSelector-leaks" [(id)cls performSelector:sel withObject:scheme]; #pragma clang diagnostic pop } }注冊后,WKWebView內的請求也會走canInitWithRequest方法了。
轉載:https://www.jianshu.com/p/4fc13d4d5607
總結
以上是生活随笔為你收集整理的让WKWebView支持NSURLProtocol的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: aix 超过一天的文件_Aix 6.1下
- 下一篇: 三维转换矩阵解释