日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

iOS WKWebView適配實(shí)戰(zhàn)篇

瀏覽:13日期:2022-09-17 09:17:51

一、Cookie適配

1.現(xiàn)狀

WKWebView適配中最麻煩的就是cookie同步問題

WKWebView采用了獨(dú)立存儲控件,因此和以往的UIWebView并不互通

雖然iOS11以后,iOS開放了WKHTTPCookieStore讓開發(fā)者去同步,但是還是需要考慮低版本的 同步問題,本章節(jié)從各個(gè)角度切入考慮cookie同步問題

2.同步cookie(NSHTTPCookieStorage->WKHTTPCookieStore)

iOS11+

可以直接使用WKHTTPCookieStore遍歷方式設(shè)值,可以在創(chuàng)建wkwebview時(shí)候就同步也可以是請求時(shí)候

// iOS11同步 HTTPCookieStorag到WKHTTPCookieStoreWKHTTPCookieStore *cookieStore = self.wkWebView.configuration.websiteDataStore.httpCookieStore;- (void)syncCookiesToWKCookieStore:(WKHTTPCookieStore *)cookieStore API_AVAILABLE(ios(11.0)){ NSArray *cookies = [[NSHTTPCookieStorage sharedHTTPCookieStorage] cookies]; if (cookies.count == 0) return; for (NSHTTPCookie *cookie in cookies) { [cookieStore setCookie:cookie completionHandler:^{ if ([cookies.lastObject isEqual:cookie]) {[self wkwebviewSetCookieSuccess]; } }]; }}

同步cookie可以在初始化wkwebview的時(shí)候,也可以在請求的時(shí)候。初始化時(shí)候同步可以確保發(fā)起html頁面請求的時(shí)候帶上cookie

例如:請求在線頁面時(shí)候要通過cookie來認(rèn)證身份,如果不是初始化時(shí)同步,可能請求頁面時(shí)就是401了

iOS11-

通過前端執(zhí)行js注入cookie,在請求時(shí)候執(zhí)行

//wkwebview執(zhí)行JS- (void)injectCookiesLT11 { WKUserScript * cookieScript = [[WKUserScript alloc] initWithSource:[self cookieString] injectionTime:WKUserScriptInjectionTimeAtDocumentStart forMainFrameOnly:NO]; [self.wkWebView.configuration.userContentController addUserScript:cookieScript];}//遍歷NSHTTPCookieStorage,拼裝JS并執(zhí)行- (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) { // Skip cookies that will break our script if ([cookie.value rangeOfString:@'’'].location != NSNotFound) { continue; } [script appendFormat:@'if (cookieNames.indexOf(’%@’) == -1) { document.cookie=’%@’; };n', cookie.name, [self formatCookie:cookie]]; } return script;}//Format cookie的js方法- (NSString *)formatCookie:(NSHTTPCookie *)cookie { NSString *string = [NSString stringWithFormat:@'%@=%@;domain=%@;path=%@', cookie.name, cookie.value, cookie.domain, cookie.path ?: @'/']; if (cookie.secure) { string = [string stringByAppendingString:@';secure=true']; } return string;}

但是上面方法執(zhí)行js,也無法保證第一個(gè)頁面請求帶有cookie

所以請求時(shí)候創(chuàng)建request需要設(shè)置cookie,并且loadRequest

-(void)injectRequestCookieLT11:(NSMutableURLRequest*)mutableRequest { // iOS11以下,手動同步所有cookie NSArray *cookies = NSHTTPCookieStorage.sharedHTTPCookieStorage.cookies; NSMutableArray *mutableCookies = @[].mutableCopy; for (NSHTTPCookie *cookie in cookies) { [mutableCookies addObject:cookie]; } // Cookies數(shù)組轉(zhuǎn)換為requestHeaderFields NSDictionary *requestHeaderFields = [NSHTTPCookie requestHeaderFieldsWithCookies:(NSArray *)mutableCookies]; // 設(shè)置請求頭 mutableRequest.allHTTPHeaderFields = requestHeaderFields;}

3.反向同步cookie(WKHTTPCookieStore->NSHTTPCookieStorage)

wkwebview產(chǎn)生的cookie也可能在某些場景需要同步給NSHTTPCookieStorage

iOS11+可以直接用WKHTTPCookieStore去同步,

iOS11-可以采用js端獲取,觸發(fā)bridge同步給NSHTTPCookieStorage

但是js同步方式無法同步httpOnly,所以真的遇到了,還是要結(jié)合服務(wù)器等方式去做這個(gè)同步。

二、JS和Native通信

1.Native調(diào)用JS

將代碼準(zhǔn)備完畢后調(diào)用API即可,回調(diào)函數(shù)可以接收js執(zhí)行結(jié)果或者錯(cuò)誤信息,So Easy。

[self.wkWebView evaluateJavaScript:jsCode completionHandler:^(id object, NSError *error){}];

2.注入JS

其實(shí)就是提前注入一些JS方法,可以提供給JS端調(diào)用。

比如有的框架會將bridge直接通過這種方式注入到WK的執(zhí)行環(huán)境中,而不是從前端引入JS,這種好處就是假設(shè)前端的JS是在線加載,JS服務(wù)器掛了或者網(wǎng)絡(luò)問題,這樣前端頁面就失去了Naitve的Bridge通信能力了。

-(instancetype)initWithSource:(NSString *)source injectionTime:(WKUserScriptInjectionTime)injectionTime forMainFrameOnly:(BOOL)forMainFrameOnly;//WKUserScriptInjectionTime說明typedef NS_ENUM(NSInteger, WKUserScriptInjectionTime) { WKUserScriptInjectionTimeAtDocumentStart, /**文檔開始時(shí)候就注入**/ WKUserScriptInjectionTimeAtDocumentEnd /**文檔加載完成時(shí)注入**/} API_AVAILABLE(macos(10.10), ios(8.0));

3.JS調(diào)用Native

3-1.準(zhǔn)備代理類

代理類要實(shí)現(xiàn)WKScriptMessageHandler

@interface WeakScriptMessageDelegate : NSObject<WKScriptMessageHandler> @property (nonatomic, weak) id<WKScriptMessageHandler> scriptDelegate; - (instancetype)initWithDelegate:(id<WKScriptMessageHandler>)scriptDelegate;@end

WKScriptMessageHandler就一個(gè)方法

@implementation WeakScriptMessageDelegate- (instancetype)initWithDelegate:(id<WKScriptMessageHandler>)scriptDelegate { self = [super init]; if (self) { _scriptDelegate = scriptDelegate; } return self;}- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message { [self.scriptDelegate userContentController:userContentController didReceiveScriptMessage:message];}

3-2.設(shè)置代理類

合適時(shí)機(jī)(一般初始化)設(shè)置代理類,并且指定name

NSString* MessageHandlerName = @'bridge';[config.userContentController addScriptMessageHandler:[[WeakScriptMessageDelegate alloc] initWithDelegate:self] name:MessageHandlerName];

3-3.bridge的使用(JS端)

執(zhí)行完上面語句后就會在JS端注入了一個(gè)對象'window.webkit.messageHandlers.bridge'

//JS端發(fā)送消息,參數(shù)最好選用String,比較通用window.webkit.messageHandlers.bridge.postMessage('type');

3-4.Native端消息的接收

然后native端可以通過WKScriptMessage的body屬性中獲得傳入的值

- (void)userContentController:(WKUserContentController*)userContentController didReceiveScriptMessage:(WKScriptMessage *)message{ if ([message.name isEqualToString:HistoryBridageName]) { } else if ([message.name isEqualToString:MessageHandlerName]) { [self jsToNativeImpl:message.body]; }}

3-5.思考題

這里我們?yōu)槭裁匆褂肳eakScriptMessageDelegate,并且再設(shè)置個(gè)delegate指向self(controller),為什么不直接指向?

提示:可以參考NSTimer的循環(huán)引用問題

3-6.完整的示例

-(void)_defaultConfig{ WKWebViewConfiguration* config = [WKWebViewConfiguration new]; …… …… …… …… WKUserContentController* userController = [[WKUserContentController alloc] init]; config.userContentController = userController; [self injectHistoryBridge:config]; …… …… …… …… }-(void)injectHistoryBridge:(WKWebViewConfiguration*)config{ [config.userContentController addScriptMessageHandler:[[WeakScriptMessageDelegate alloc] initWithDelegate:self] name:HistoryBridageName]; NSString *_jsSource = [NSString stringWithFormat: @'(function(history) {n' ' function notify(type) {n' ' setTimeout(function() {n' ' window.webkit.messageHandlers.%@.postMessage(type)n' ' }, 0)n' ' }n' ' function shim(f) {n' ' return function pushState() {n' ' notify(’other’)n' ' return f.apply(history, arguments)n' ' }n' ' }n' ' history.pushState = shim(history.pushState)n' ' history.replaceState = shim(history.replaceState)n' ' window.addEventListener(’popstate’, function() {n' ' notify(’backforward’)n' ' })n' '})(window.history)n', HistoryBridageName ]; WKUserScript *script = [[WKUserScript alloc] initWithSource:_jsSource injectionTime:WKUserScriptInjectionTimeAtDocumentStart forMainFrameOnly:YES]; [config.userContentController addUserScript:script];}

3-7.其它問題

在iOS8 beta5前,JS和Native這樣通信設(shè)置是不行的,所以可以采用生命周期中做URL的攔截去解析數(shù)據(jù)來達(dá)到效果,這里不做贅述,可以自行參考網(wǎng)上類似UIWebview的橋接原理文章

三、實(shí)戰(zhàn)技巧

1.UserAgent的設(shè)置

添加UA

實(shí)際過程中最好只是原有UA上做添加操作,全部替換可能導(dǎo)致服務(wù)器的拒絕(安全策略)

iOS WKWebView適配實(shí)戰(zhàn)篇

日志中紅線部分是整個(gè)模擬器的UA,綠色部門是UA中的ApplicationName部分

iOS9上,WKWebview提供了API可以設(shè)置ua中的ApplicationName

config.applicationNameForUserAgent = [NSString stringWithFormat:@'%@ %@', config.applicationNameForUserAgent, @'arleneConfig'];

全部替換UA

iOS9以上直接可以指定wkwebview的customUserAgent,iOS9以下的話,設(shè)置NSUserDefaults

if (@available(iOS 9.0, *)) { self.wkWebView.customUserAgent = @'Hello My UserAgent';}else{ [[NSUserDefaults standardUserDefaults] registerDefaults:@{@'UserAgent':@'Hello My UserAgent'}]; [[NSUserDefaults standardUserDefaults] synchronize];}

2.監(jiān)聽進(jìn)度和頁面的title變化

wkwebview可以監(jiān)控頁面加載進(jìn)度,類似瀏覽器中打開頁面中的進(jìn)度條的顯示

頁面切換的時(shí)候也會自動更新頁面中設(shè)置的title,可以在實(shí)際項(xiàng)目中動態(tài)切換容器的title,比如根據(jù)切換的title設(shè)置navigationItem.title

原理直接通過KVO方式監(jiān)聽值的變化,然后在回調(diào)中處理相關(guān)邏輯

//kvo 加載進(jìn)度[self.webView addObserver:self forKeyPath:@'estimatedProgress' options:NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew context:nil];//kvo title[self.webView addObserver:self forKeyPath:@'title' options:NSKeyValueObservingOptionNew context:nil];/** KVO 監(jiān)聽具體回調(diào)**/- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{ if ([keyPath isEqual:@'estimatedProgress'] && object == self.webView) { ALLOGF(@'Progress--->%@',[NSNumber numberWithDouble:self.webView.estimatedProgress]); }else if([keyPath isEqualToString:@'title'] && object == self.webview){ self.navigationItem.title = self.webView.title; }else{ [super observeValueForKeyPath:keyPath ofObject:object change:change context:context]; }}/**銷毀時(shí)候記得移除**/[self.webView removeObserver:self forKeyPath:NSStringFromSelector(@selector(estimatedProgress))];[self.webView removeObserver:self forKeyPath:NSStringFromSelector(@selector(title))];

3.Bridge通信實(shí)戰(zhàn)

下面介紹自己實(shí)現(xiàn)的bridge通信框架,前端無需關(guān)心所在容器,框架層做適配。

import {WebBridge} from ’XXX’/*** 方法: WebBridge.call(taskName,options,callback)* 參數(shù)說明: * taskName String task的名字,用于Native處理分發(fā)任務(wù)的標(biāo)識* options Object 傳遞的其它參數(shù)* callback function 回調(diào)函數(shù)*. 回調(diào)參數(shù)* json object native返回的內(nèi)容**/WebBridge.call('Alert',{'content':'彈框內(nèi)容','btn':'btn內(nèi)容'},function(json){ console.log('call back is here',JSON.stringify(json));});

上面調(diào)用了Native的Alert控件,然后返回調(diào)用結(jié)果。

調(diào)用到的Native代碼如下:

//AlertTask.m#import 'AlertTask.h'#import <lib-base/ALBaseConstants.h>@interface AlertTask (){}@property (nonatomic,weak) ArleneWebViewController* mCtrl;@end@implementation AlertTask-(instancetype)initWithContext:(ArleneWebViewController*)controller{ self = [super init]; self.mCtrl = controller; return self;}-(NSString*)taskName{ return @'Alert';}-(void)doTask:(NSDictionary*)params{ ALShowAlert(@'Title',@'message');//彈出Alert NSMutableDictionary* callback = [ArleneTaskUtils basicCallback:params];//獲取callback [callback addEntriesFromDictionary:params]; [self.mCtrl callJS:callback];//執(zhí)行回調(diào)}@end

具體實(shí)現(xiàn)原理可以點(diǎn)擊下方視頻鏈接:

點(diǎn)擊獲取框架原理視頻

到此這篇關(guān)于iOS WKWebView適配實(shí)戰(zhàn)篇的文章就介紹到這了,更多相關(guān)iOS WKWebView適配 內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: IOS
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
性欧美精品高清| 亚洲欧美日韩国产一区| 深夜福利亚洲| 欧美亚洲三区| 久久99性xxx老妇胖精品| 麻豆91在线播放| 韩国三级一区| 亚洲欧美不卡| 国产精品红桃| 日韩中文字幕高清在线观看| 久久美女性网| 国产亚洲在线观看| 久久国际精品| 中文字幕在线视频网站| 欧美精品一区二区久久| 久色成人在线| 国产精品久久久一区二区| 国产第一亚洲| 美女精品在线| 国产精品xxxav免费视频| 日韩av福利| 伊人久久成人| 国产精品亚洲产品| 欧美sss在线视频| 亚洲毛片网站| 国产精品亚洲欧美| 日韩高清中文字幕一区二区| 99在线|亚洲一区二区| 丝袜美腿高跟呻吟高潮一区| 亚洲五月综合| 国产精品主播| 9999国产精品| 亚洲一区二区毛片| 久久亚洲人体| 亚洲欧美日韩国产一区| 国产精品一页| japanese国产精品| 国产精品久久亚洲不卡| 欧美日韩激情| 国内在线观看一区二区三区| 一区二区视频欧美| 日本v片在线高清不卡在线观看| 精品久久久网| 免播放器亚洲| 欧美aa在线观看| 欧美天堂在线| 亚洲神马久久| 免费亚洲一区| 日韩免费精品| 欧美日韩国产高清电影| 精品一区二区三区免费看| 蜜臀久久99精品久久久久久9| 中文字幕色婷婷在线视频| 欧美中文一区| 玖玖玖国产精品| 欧美sss在线视频| 麻豆高清免费国产一区| 亚洲另类视频| 99热精品在线| 91精品在线观看国产| 久久免费精品| 欧美欧美黄在线二区| 99成人在线| 免费污视频在线一区| 欧美激情亚洲| 日本一区二区三区中文字幕| 欧美日韩国产在线观看网站| 91亚洲自偷观看高清| 日韩国产成人精品| 国产综合婷婷| 国产欧美69| 亚洲午夜免费| 日韩欧美一区二区三区在线观看| 亚洲中字黄色| 欧美片第1页| 日韩av中文字幕一区二区| 亚洲综合不卡| 女人天堂亚洲aⅴ在线观看| 蜜桃成人精品| 欧美丰满日韩| 美女高潮久久久| 欧美精品三级在线| 日韩超碰人人爽人人做人人添| 老牛影视一区二区三区| 欧美日韩精品一本二本三本| 欧美三区四区| 日韩大片在线| 日本午夜大片a在线观看| 国产精品极品国产中出| 免费一级欧美片在线观看网站| 国产欧美午夜| 国产精品探花在线观看| 久久精品99国产国产精| 日本va欧美va瓶| 亚洲91网站| 日韩三区四区| 日韩精品成人| 欧美中文高清| 欧美国产三级| 麻豆国产精品| 成人高清一区| 一区二区精品伦理...| 亚洲综合电影| 亚洲午夜久久久久久尤物| 伊人精品一区| 极品日韩av| 亚洲一级网站| 美日韩精品视频| 亚洲三级网址| 国产欧美自拍| 丰满少妇一区| 日韩综合精品| 欧美日韩视频一区二区三区| 亚洲综合电影一区二区三区| 亚洲精品在线二区| 欧美亚洲免费| 国产一区二区三区四区| 超碰超碰人人人人精品| 女主播福利一区| 免费久久99精品国产| 日韩中文字幕在线一区| 久久av网址| 成人免费电影网址| 中国女人久久久| 日韩欧美2区| 精品国产麻豆| 91九色精品| 亚洲精品女人| 美女久久久久久| 91精品久久久久久久久久不卡| 日韩视频二区| 欧美专区一区| 91av亚洲| 亚洲视频播放| 国产乱码精品一区二区亚洲| 国产美女高潮在线观看| 999在线观看精品免费不卡网站| 日韩高清电影免费| 日本蜜桃在线观看视频| 麻豆成人在线| 精品网站aaa| 狠狠爱www人成狠狠爱综合网| 日韩精品一页| 成人片免费看| 亚洲资源在线| 成人污污视频| 蜜臀精品一区二区三区在线观看 | 亚洲人成亚洲精品| 欧美aaaaaa午夜精品| 97se综合| 亚洲精品乱码久久久久久蜜桃麻豆| 国产高清日韩| 午夜久久tv| 国产精品4hu.www| 欧美在线亚洲| 久久99蜜桃| 日本欧美在线看| 国产精品专区免费| 日本视频在线一区| 久久高清精品| 国产精品嫩模av在线| 国产综合欧美| 精品国产一区二| 在线观看亚洲精品福利片| av中文字幕在线观看第一页| 视频一区中文字幕| 日本不良网站在线观看| 日本久久二区| 婷婷亚洲五月| 精品国产网站| 久久精品xxxxx| 国产午夜久久| 成人看片网站| 国产毛片久久久| 亚洲综合另类| 久久国产欧美| 国产精品蜜月aⅴ在线| 久久国产高清| 亚洲精品在线影院| 美女久久99| 人人爱人人干婷婷丁香亚洲| 日韩亚洲在线| 国产99精品| 中文字幕在线高清| 国产精品白丝一区二区三区| 亚洲开心激情| 日韩一级精品| 999久久久91| 国产色播av在线| 国产精品激情| 日本伊人久久| 亚洲综合精品四区| 亚洲午夜av| 国产一区2区| 国产精品成人国产| 欧美伊人久久| 婷婷综合福利| 免费人成黄页网站在线一区二区| 久久影院一区| 久久精品123|