亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > iOS > 正文

iOs遷至WKWebView跨過的一些坑

2020-07-26 02:19:10
字體:
來源:轉載
供稿:網友

前言

在iOS中有兩種網頁視圖可以加載網頁除了系統的那個控制器。一種是UIWebView,另一種是WKWebView,其實WKWebView就是想替代UIWebView的,因為我們都知道UIWebView非常占內存等一些問題,但是現在很多人還在使用UIWebView這是為啥呢?而且官方也宣布在iOS12中廢棄了UIWebView讓我們盡快使用WKWebView。其實也就是這些東西:**頁面尺寸問題、JS交互、請求攔截、cookie帶不上的問題。**所以有時想要遷移還得解決這些問題,所以還是很煩的,所以一一解決嘍。

頁面尺寸的問題

我們知道有些網頁在UIWebView上顯示好好地,使用WKWebView就會出現尺寸的問題,這時很納悶,安卓也不會,你總不說是前端的問題吧?但其實是WKWebView中網頁是需要適配一下,所以自己添加JS吧,當然和前端關系好就可以叫他加的。下面通過設置配置中的userContentController來添加JS。

WKWebViewConfiguration *configuration = [[WKWebViewConfiguration alloc] init];NSString *jScript = @"var meta = document.createElement('meta'); meta.setAttribute('name', 'viewport'); meta.setAttribute('content', 'width=device-width'); document.getElementsByTagName('head')[0].appendChild(meta);";WKUserScript *wkUScript = [[WKUserScript alloc] initWithSource:jScript injectionTime:WKUserScriptInjectionTimeAtDocumentEnd forMainFrameOnly:YES];WKUserContentController *wkUController = [[WKUserContentController alloc] init];[wkUController addUserScript:wkUScript];configuration.userContentController = wkUController;

JS交互

我們都知道在UIWebView中可以使用自家的JavaScriptCore來進行交互非常的方便。在JavaScriptCore中有三者比較常用那就是JSContext(上下文)、JSValue(類型轉換)、JSExport(js調OC模型方法)。

在UIWebView中的便利交互方法

//JSContext就為其提供著運行環境 H5上下文- (void)webViewDidFinishLoad:(UIWebView *)webView{ JSContext *jsContext = [self.webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"]; self.jsContext = jsContext;}
// 執行腳本增加js全局變量[self.jsContext evaluateScript:@"var arr = [3, '3', 'abc'];"];
// ⚠️添加JS方法,需要注意的是添加的方法會覆蓋原有的JS方法,因為我們是在網頁加載成功后獲取上下文來操作的。// 無參數的self.jsContext[@"alertMessage"] = ^() { NSLog(@"JS端調用alertMessage時就會跑到這里來!");};// 帶參數的,值必須進行轉換 self.jsContext[@"showDict"] = ^(JSValue *value) { NSArray *args = [JSContext currentArguments]; JSValue *dictValue = args[0]; NSDictionary *dict = dictValue.toDictionary; NSLog(@"%@",dict); };
// 獲取JS中的arr數據JSValue *arrValue = self.jsContext[@"arr"];
// 異常捕獲self.jsContext.exceptionHandler = ^(JSContext *context, JSValue *exception) { weakSelf.jsContext.exception = exception; NSLog(@"exception == %@",exception);};
// 給JS中的對象重新賦值OMJSObject *omObject = [[OMJSObject alloc] init];self.jsContext[@"omObject"] = omObject;NSLog(@"omObject == %d",[omObject getSum:20 num2:40]);// 我們都知道object必須要遵守JSExport協議時,js可以直接調用object中的方法,并且需要把函數名取個別名。在JS端可以調用getS,OC可以繼續使用這個getSum這個方法@protocol OMProtocol <JSExport>// 協議 - 協議方法 JSExportAs(getS, -(int)getSum:(int)num1 num2:(int)num2);@end

在WKWebView中如何做呢?

不能像上面那樣,系統提供的是通過以下兩種方法,所以是比較難受,而且還得前端使用messageHandler來調用,即安卓和iOS分開處理。

// 直接調用jsNSString *jsStr = @"var arr = [3, '3', 'abc']; ";[self.webView evaluateJavaScript:jsStr completionHandler:^(id _Nullable result, NSError * _Nullable error) { NSLog(@"%@----%@",result, error);}];
// 下面是注冊名稱后,js使用messageHandlers調用了指定名稱就會進入到代理中// OC我們添加了js名稱后- (void)viewDidLoad{ //... [wkUController addScriptMessageHandler:self name:@"showtime"]; configuration.userContentController = wkUController;}// JS中messageHandlers調用我們在OC中的名稱一致時就會進入后面的到OC的代理window.webkit.messageHandlers.showtime.postMessage('');// 代理,判斷邏輯- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message{ if ([message.name isEqualToString:@"showtime"]) {  NSLog(@"來了!"); } NSLog(@"message == %@ --- %@",message.name,message.body); }// 最后在dealloc必須移除[self.userContentController removeScriptMessageHandlerForName:@"showtime"];
//如果是彈窗的必須自己實現代理方法- (void)webView:(WKWebView *)webView runJavaScriptAlertPanelWithMessage:(NSString *)message initiatedByFrame:(WKFrameInfo *)frame completionHandler:(void (^)(void))completionHandler{ UIAlertController *alert = [UIAlertController alertControllerWithTitle:@"提醒" message:message preferredStyle:UIAlertControllerStyleAlert]; [alert addAction:[UIAlertAction actionWithTitle:@"知道了" style:UIAlertActionStyleCancel handler:^(UIAlertAction * _Nonnull action) {  completionHandler(); }]];  [self presentViewController:alert animated:YES completion:nil];}

一直用一直爽的交互

我們上面寫了兩者的一些交互,雖然可以用呢,但是沒有帶一種很簡單很輕松的境界,所以有一個開源庫:WebViewJavaScriptBridge。這個開源庫可以同時兼容兩者,而且交互很簡單,但是你必須得前端一起,否則就哦豁了。

// 使用self.wjb = [WebViewJavascriptBridge bridgeForWebView:self.webView];// 如果你要在VC中實現 UIWebView的代理方法 就實現下面的代碼(否則省略)[self.wjb setWebViewDelegate:self];// 注冊js方法名稱[self.wjb registerHandler:@"jsCallsOC" handler:^(id data, WVJBResponseCallback responseCallback) { NSLog(@"currentThread == %@",[NSThread currentThread]); NSLog(@"data == %@ -- %@",data,responseCallback);}];// 調用JSdispatch_async(dispatch_get_global_queue(0, 0), ^{  [self.wjb callHandler:@"OCCallJSFunction" data:@"OC調用JS" responseCallback:^(id responseData) {   NSLog(@"currentThread == %@",[NSThread currentThread]);   NSLog(@"調用完JS后的回調:%@",responseData);  }];});

前端使用實例如下,具體使用方法可以查看WebViewJavaScriptBridge。

function setupWebViewJavascriptBridge(callback) {	if (window.WebViewJavascriptBridge) { return callback(WebViewJavascriptBridge); }	if (window.WVJBCallbacks) { return window.WVJBCallbacks.push(callback); }	window.WVJBCallbacks = [callback];	var WVJBIframe = document.createElement('iframe');	WVJBIframe.style.display = 'none';	WVJBIframe.src = 'https://__bridge_loaded__';	document.documentElement.appendChild(WVJBIframe);	setTimeout(function() { document.documentElement.removeChild(WVJBIframe) }, 0)}setupWebViewJavascriptBridge(function(bridge) {		/* Initialize your app here */	bridge.registerHandler('JS Echo', function(data, responseCallback) {		console.log("JS Echo called with:", data)		responseCallback(data)	})	bridge.callHandler('ObjC Echo', {'key':'value'}, function responseCallback(responseData) {		console.log("JS received response:", responseData)	})})

請求攔截

我們UIWebView在早期是使用- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType來根據scheme、host、pathComponents進行攔截做自定義邏輯處理。但是這種方法不是很靈活,于是就使用NSURLProtocol來進行攔截,例如微信攔截淘寶一樣,直接顯示一個提示。又或者是攔截請求調用本地的接口,打開相機、錄音、相冊等功能。還能直接攔截后改變原有的request,直接返回數據或者其他的url,在一些去除廣告時可以的用得上。

我們使用的時候必須要使用NSURLProtocol的子類來進行一些操作。并在使用前需要注冊自定義的Class。攔截后記得進行標記一下,防止自循環多執行??上У氖窃赪KWebView中不能進行攔截后處理的操作,只能監聽卻改變不了。源于WKWebView采用的是webkit加載,和系統的瀏覽器一樣的機制。

// 子類@interface OMURLProtocol : NSURLProtocol<NSURLSessionDataDelegate>@property (nonatomic, strong) NSURLSession *session;@end// 注冊[NSURLProtocol registerClass:[OMURLProtocol class]];
// 1. 首先會在這里來進行攔截,返回YES則表示需要經過我們自定義處理,NO則走系統處理+ (BOOL)canInitWithRequest:(NSURLRequest *)request;// 2.攔截處理將會進入下一個環節, 返回一個標準化的request,可以在這里進行重定向+ (NSURLRequest *)canonicalRequestForRequest:(NSURLRequest *)request;// 3.是否成功攔截都會走這個方法, 可以在這里進行一些自定義處理- (void)startLoading;// 4. 任何網絡請求都會走上面的攔截處理,即使我們重定向后還會再走一次或多次流程,需要標記來處理// 根據request獲取標記值來決定是否需要攔截,在canInitWithRequest內處理+ (nullable id)propertyForKey:(NSString *)key inRequest:(NSURLRequest *)request;// 標記+ (void)setProperty:(id)value forKey:(NSString *)key inRequest:(NSMutableURLRequest *)request;// 移除標記+ (void)removePropertyForKey:(NSString *)key inRequest:(NSMutableURLRequest *)request;

請求頭或數據混亂問題

還需要注意的一點是,如果實現線了攔截處理的話,我們在使用AFN和URLSession進行訪問的時候攔截會發現數據或請求頭可能和你攔截處理后的數據或請求不符合預期,這是因為我們在攔截的時候只是先請求了A后請求了B,這是不符合預期的,雖然URLConnection不會但是已被廢棄不值得提倡使用。我們通過在攔截的時候通過LLDB打印session中配置的協議時,發現是這樣的沒有包含我們自定義的協議,我們通過Runtime交換方法交換protocolClasses方法,我們實現我們自己的protocolClasses方法。但是為了保證系統原有的屬性,我們應該在系統原有的協議表上加上我們的協議類。在當前我們雖然可以通過[NSURLSession sharedSession].configuration.protocolClasses;獲取系統默認的協議類,但是如果我們在當前自定義的類里protocolClasses寫的話會造成死循環,因為我們交換了該屬性的getter方法。我們使用保存類名然后存儲至NSUserDefaults,取值時在還原class。

po session.configuration.protocolClasses<__NSArrayI 0x600001442d00>(_NSURLHTTPProtocol,_NSURLDataProtocol,_NSURLFTPProtocol,_NSURLFileProtocol,NSAboutURLProtocol)
// 自定義返回我們的協議類- (NSArray *)protocolClasses { NSArray *originalProtocols = [OMURLProtocol readOriginalProtocols]; NSMutableArray *newProtocols = [NSMutableArray arrayWithArray:originalProtocols]; [newProtocols addObject:[OMURLProtocol class]]; return newProtocols;}// 我們再次打印時發現已經加上我們自定義的協議類了po session.configuration.protocolClasses<__NSArrayM 0x60000041a4f0>(_NSURLHTTPProtocol,_NSURLDataProtocol,_NSURLFTPProtocol,_NSURLFileProtocol,NSAboutURLProtocol,OMURLProtocol)
// 存儲系統原有的協議類+ (void)saveOriginalProtocols: (NSArray<Class> *)protocols{ NSMutableArray *protocolNameArray = [NSMutableArray array]; for (Class protocol in protocols){  [protocolNameArray addObject:NSStringFromClass(protocol)]; } NSLog(@"協議數組為: %@", protocolNameArray); [[NSUserDefaults standardUserDefaults] setObject:protocolNameArray forKey:originalProtocolsKey]; [[NSUserDefaults standardUserDefaults] synchronize];}// 獲取系統原有的協議類+ (NSArray<Class> *)readOriginalProtocols{ NSArray *classNames = [[NSUserDefaults standardUserDefaults] valueForKey:originalProtocolsKey]; NSMutableArray *origianlProtocols = [NSMutableArray array]; for (NSString *name in classNames){  Class class = NSClassFromString(name);  [origianlProtocols addObject: class]; } return origianlProtocols;}
+ (void)hookNSURLSessionConfiguration{ NSArray *originalProtocols = [NSURLSession sharedSession].configuration.protocolClasses; [self saveOriginalProtocols:originalProtocols]; Class cls = NSClassFromString(@"__NSCFURLSessionConfiguration") ?: NSClassFromString(@"NSURLSessionConfiguration"); Method originalMethod = class_getInstanceMethod(cls, @selector(protocolClasses)); Method stubMethod = class_getInstanceMethod([self class], @selector(protocolClasses)); if (!originalMethod || !stubMethod) {  [NSException raise:NSInternalInconsistencyException format:@"沒有這個方法 無法交換"]; } method_exchangeImplementations(originalMethod, stubMethod);}

Cookie的攜帶問題

很多應用場景中需要使用session來進行處理,在UIWebView中很容易做到攜帶這些Cookie,但是由于WKWebView的機制不一樣,跨域會出現丟失cookie的情況是很糟糕的。目前有兩種用法:腳本和手動添加cookie。腳本不太靠譜,建議使用手動添加更為保險。

// 使用腳本來添加cookie// 獲取去cookie數據- (NSString *)cookieString{ NSMutableString *script = [NSMutableString string]; [script appendString:@"var cookieNames = document.cookie.split('; ').map(function(cookie) { return cookie.split('=')[0] } );/n"]; for (NSHTTPCookie *cookie in [[NSHTTPCookieStorage sharedHTTPCookieStorage] cookies]) {  if ([cookie.value rangeOfString:@"'"].location != NSNotFound) {   continue;  }  [script appendFormat:@"if (cookieNames.indexOf('%@') == -1) { document.cookie='%@'; };/n", cookie.name, cookie.kc_formatCookieString]; } return script;}// 添加cookieWKUserScript * cookieScript = [[WKUserScript alloc] initWithSource:[self cookieString] injectionTime:WKUserScriptInjectionTimeAtDocumentStart forMainFrameOnly:NO];[[[WKUserContentController alloc] init] addUserScript: cookieScript];
// 添加一個分類來修復cookie丟失的問題@interface NSURLRequest (Cookie)- (NSURLRequest *)fixCookie;@end@implementation NSURLRequest (Cookie)- (NSURLRequest *)fixCookie{ NSMutableURLRequest *fixedRequest; if ([self isKindOfClass:[NSMutableURLRequest class]]) {  fixedRequest = (NSMutableURLRequest *)self; } else {  fixedRequest = self.mutableCopy; } //防止Cookie丟失 NSDictionary *dict = [NSHTTPCookie requestHeaderFieldsWithCookies:[NSHTTPCookieStorage sharedHTTPCookieStorage].cookies]; if (dict.count) {  NSMutableDictionary *mDict = self.allHTTPHeaderFields.mutableCopy;  [mDict setValuesForKeysWithDictionary:dict];  fixedRequest.allHTTPHeaderFields = mDict; } return fixedRequest;}@end   // 使用場景- (void)webView:(WKWebView *)webView decidePolicyForNavigationAction:(WKNavigationAction *)navigationAction decisionHandler:(void (^)(WKNavigationActionPolicy))decisionHandler { [navigationAction.request fixCookie]; decisionHandler(WKNavigationActionPolicyAllow);}

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對武林網的支持。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲大胆人体在线| 在线播放亚洲激情| 日韩高清免费在线| 欧美激情乱人伦| 久久99久久99精品免观看粉嫩| 精品成人69xx.xyz| 中文字幕日韩欧美| 欧美有码在线观看视频| 色综合亚洲精品激情狠狠| 国产精品1234| 国产欧美日韩精品专区| 精品香蕉在线观看视频一| 国产一区二区三区在线观看视频| 最近免费中文字幕视频2019| 国产精品电影网站| 成人精品久久一区二区三区| 91麻豆桃色免费看| 在线播放日韩欧美| 午夜精品美女自拍福到在线| 国产国语刺激对白av不卡| 久久综合九色九九| 久久99热精品这里久久精品| 久久久噜噜噜久久久| 欧美性感美女h网站在线观看免费| 亚洲va欧美va在线观看| 国产日韩欧美在线观看| 国产日韩在线免费| 国产精品免费一区| 亚洲黄页视频免费观看| 亚洲国产成人精品电影| 91精品国产网站| 岛国av午夜精品| 亚洲毛片一区二区| 国产日韩欧美在线播放| 国产精品精品一区二区三区午夜版| 搡老女人一区二区三区视频tv| 欧美日韩一区二区在线| 国产日韩欧美另类| 88xx成人精品| 欧美日韩一区二区精品| 大胆欧美人体视频| 亚洲一区二区精品| 日韩免费在线观看视频| 98精品在线视频| 日韩中文视频免费在线观看| 国产网站欧美日韩免费精品在线观看| 日韩av中文字幕在线播放| 午夜精品视频在线| 欧美精品激情在线| 中文字幕一区二区精品| 欧美日韩裸体免费视频| 亚洲欧美一区二区三区久久| 在线电影中文日韩| 亚洲一区二区免费| 日韩av在线免费观看| 国产一区视频在线| 97精品在线观看| 2019精品视频| 久久国产精品久久国产精品| 欧美精品激情在线观看| 亚洲欧美一区二区三区在线| 欧美高清视频一区二区| 国产精品青草久久久久福利99| 欧美国产在线视频| 亚洲欧美激情四射在线日| 日韩激情视频在线播放| 国产精品久久不能| 精品国产依人香蕉在线精品| 成人激情春色网| 日韩69视频在线观看| 日本高清+成人网在线观看| 一区二区三区亚洲| 国产精品91在线观看| 九九热精品视频国产| 精品久久久999| 久热精品视频在线免费观看| 深夜福利日韩在线看| 国产精品久久久久久久久借妻| 欧美成人精品在线视频| 国产成人精品一区二区三区| 亚洲女人天堂色在线7777| 在线成人激情黄色| 在线a欧美视频| 欧美电影《睫毛膏》| 欧美日韩免费观看中文| 亚洲精品美女在线观看播放| 国产在线拍偷自揄拍精品| 国产欧美日韩精品丝袜高跟鞋| 亚洲欧美日韩一区二区三区在线| 国产精品三级网站| 亚洲人永久免费| 91精品国产自产在线老师啪| 97在线观看视频国产| 亚洲国产一区二区三区四区| 日韩av在线网| 久久夜色精品国产欧美乱| 91精品国产高清久久久久久久久| 国产亚洲aⅴaaaaaa毛片| 中文字幕亚洲专区| 精品久久久久久久久久| www.日韩免费| 97视频网站入口| 精品亚洲一区二区三区| 亚洲一区二区三区视频播放| 国产精品 欧美在线| 色婷婷综合成人| 日韩美女写真福利在线观看| 精品久久香蕉国产线看观看亚洲| 国产91色在线免费| 免费97视频在线精品国自产拍| 正在播放欧美视频| 亚洲经典中文字幕| 疯狂蹂躏欧美一区二区精品| 亚洲男人天堂网站| 国产日韩精品在线播放| 欧美精品在线第一页| 在线日韩日本国产亚洲| 欧美亚洲国产成人精品| 亚洲国产高潮在线观看| 久久国产视频网站| 中文字幕av日韩| 伊人激情综合网| 成人黄色影片在线| 久热爱精品视频线路一| 91亚洲精品在线| 91午夜在线播放| 久久在线视频在线| 国产精品免费一区二区三区都可以| 久久色免费在线视频| 欧美精品情趣视频| 国产成人97精品免费看片| 成人动漫网站在线观看| 精品人伦一区二区三区蜜桃网站| 91色视频在线导航| 久久综合伊人77777蜜臀| 国模叶桐国产精品一区| 日韩免费在线免费观看| 97在线免费观看| 国产亚洲激情视频在线| 亚洲第一页在线| 国产成人高清激情视频在线观看| 国产精品三级美女白浆呻吟| 日韩欧美在线中文字幕| 国产欧美精品一区二区| 韩国精品美女www爽爽爽视频| 亚洲天堂男人天堂| 亚洲日本中文字幕| 91av视频在线观看| 国产精品久久久久影院日本| 国产一区二区三区高清在线观看| 91产国在线观看动作片喷水| 日韩av不卡电影| 国产免费一区二区三区香蕉精| 在线观看精品国产视频| 日韩精品欧美国产精品忘忧草| 91av中文字幕| 亚洲国产高潮在线观看| 欧美交受高潮1| 亚洲高清久久网| 欧美国产日韩一区二区在线观看| 成人黄色片网站| 亚洲欧美日韩精品久久亚洲区| 久久艳片www.17c.com| 欧美激情精品久久久久久大尺度|