前言:
web頁(yè)面和app的直接的交互是很常見的東西,在ios8之前袖裕,用的是uiwebview溉瓶,但是在ios8之后堰酿,蘋果推出了WebKit這個(gè)框架,用來(lái)替代原有的UIWebView灾锯。以前的時(shí)候需要適配iOS7嗅榕,現(xiàn)在扔掉了iOS7,所以在做和web頁(yè)面的交互的時(shí)候兼雄,可以用wkwebview替代原有的uiwebview帽蝶。
WKWebView的特點(diǎn):
- 性能高励稳,穩(wěn)定性好,占用的內(nèi)存比較小趣避,
- 支持JS交互
- 支持HTML5 新特性
- 可以添加進(jìn)度條新翎。
- 支持內(nèi)建手勢(shì),
- 據(jù)說高達(dá)60fps的刷新頻率
使用及注意點(diǎn)
WKWebView只能用代碼創(chuàng)建愁拭,而且自身就支持了右滑返回手勢(shì)allowsBackForwardNavigationGestures和加載進(jìn)度estimatedProgress等一些UIWebView不具備卻非常好用的屬性亏吝。在創(chuàng)建的時(shí)候,指定初始化方法中要求傳入一個(gè)WKWebViewConfiguration對(duì)象枫攀,一般我們使用默認(rèn)配置就好来涨,但是有些地方是要根據(jù)自己的情況去做更改。比如技羔,配置中的allowsInlineMediaPlayback這個(gè)屬性卧抗,默認(rèn)為NO社裆,如果不做更改,網(wǎng)頁(yè)中內(nèi)嵌的視頻就無(wú)法正常播放标沪。
WKWebView的相關(guān)的代理方法
WKWebView的相關(guān)的代理方法分別在WKNavigationDelegate和WKUIDelegate以及WKScriptMessageHandler這個(gè)與JavaScript交互相關(guān)的代理方法嗜傅。
- WKNavigationDelegate
該代理提供的方法吕嘀,可以用來(lái)追蹤加載過程(頁(yè)面開始加載、加載完成趁曼、加載失敽ぁ)瘾敢、決定是否執(zhí)行跳轉(zhuǎn)簇抵。具體的使用可以看下面的代碼示例 - WKUIDelegate
這個(gè)代理方法全都是與界面彈出提示框相關(guān)的,所以最好是實(shí)現(xiàn)的晃财,否則的話,遇到網(wǎng)頁(yè)alert的時(shí)候罗洗,如果此代理方法沒有實(shí)現(xiàn)钢猛,則不會(huì)出現(xiàn)彈框提示命迈。 - WKScriptMessageHandler
這個(gè)代理方法就是實(shí)現(xiàn)和JavaScript交互相關(guān)的。下面會(huì)介紹js調(diào)用oc淑倾,oc調(diào)用js的具體的使用征椒。
具體的使用的過程
創(chuàng)建WKWebView
- 我這里是定義了全局的變量
@interface ACViewController ()<WKNavigationDelegate,WKUIDelegate,WKScriptMessageHandler>
@property (strong, nonatomic) WKWebView *wkwebView;
@property (strong, nonatomic) UIProgressView *progressView;//這個(gè)是加載頁(yè)面的進(jìn)度條
@end
@implementation ACViewController
{
UILabel * label ;
WKUserContentController * userContentController;
}
- 創(chuàng)建
#pragma mark 初始化webview
-(void)initWKWebView
{
WKWebViewConfiguration * configuration = [[WKWebViewConfiguration alloc]init];//先實(shí)例化配置類 以前UIWebView的屬性有的放到了這里
//注冊(cè)供js調(diào)用的方法
userContentController =[[WKUserContentController alloc]init];
//彈出登錄
[userContentController addScriptMessageHandler:self name:@"loginVC"];
//加載首頁(yè)
[userContentController addScriptMessageHandler:self name:@"gotoFirstVC"];
//進(jìn)入詳情頁(yè)
[userContentController addScriptMessageHandler:self name:@"gotodetailVC"];
configuration.userContentController = userContentController;
configuration.preferences.javaScriptEnabled = YES;//打開js交互
_wkwebView = [[WKWebView alloc] initWithFrame:CGRectMake(0, 0.0f, screen_width, screen_height) configuration:configuration];
_wkwebView.backgroundColor = [UIColor clearColor];
_wkwebView.UIDelegate = self;
_wkwebView.navigationDelegate = self;
_wkwebView.allowsBackForwardNavigationGestures =YES;//打開網(wǎng)頁(yè)間的 滑動(dòng)返回
_wkwebView.allowsLinkPreview = YES;//允許預(yù)覽鏈接
[self initProgressView];
[_wkwebView addObserver:self forKeyPath:@"estimatedProgress" options:NSKeyValueObservingOptionNew context:nil];//注冊(cè)observer 拿到加載進(jìn)度
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:具體的網(wǎng)址]];
[_wkwebView loadRequest:request];
[self.view addSubview:_wkwebView];
}
```
- 關(guān)于進(jìn)度條的監(jiān)聽
pragma mark --這個(gè)就是設(shè)置的上面的那個(gè)加載的進(jìn)度
-(void)initProgressView
{
_progressView =[[UIProgressView alloc]initWithFrame:CGRectMake(0,0, screen_width, 10.0f)];
_progressView.tintColor = [UIColor blueColor];
_progressView.trackTintColor = [UIColor whiteColor];
[self.view addSubview:_progressView];
}
//這個(gè)是檢測(cè)那個(gè)進(jìn)度條的迂尝,顯示完成之后剪芥,進(jìn)度條就隱藏了
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context
{
if([keyPath isEqualToString:@"estimatedProgress"])
{
_progressView.hidden = NO;
CGFloat progress = [ change[@"new"] floatValue];
[_progressView setProgress:progress];
if(progress==1.0)
{
_progressView.hidden =YES;
}
}
}
- 和js交互
1.js調(diào)用oc的方法
WKScriptMessage有兩個(gè)關(guān)鍵屬性name 和 body税肪。因?yàn)槲覀兘o每一個(gè)OC方法取了一個(gè)name,所以就可以根據(jù)name 來(lái)區(qū)分執(zhí)行不同的方法锻梳。body 中存著JS 要給OC 傳的參數(shù)疑枯。
在這里我是在前面注冊(cè)了三個(gè)供js調(diào)用的方法蛔六,loginVC,gotoFirstVC具钥,gotodetailVC液兽。就是name。
在這個(gè)方法里面實(shí)現(xiàn)的
-(void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message{};
具體的代碼如下
-(void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message
{
if(message.name ==nil || [message.name isEqualToString:@""])
return;
//message body : js 傳過來(lái)值
NSLog(@"message.body ==%@",message.body);
if ([message.name isEqualToString:@"loginVC"])
{
[self showLoginView];
}
else if ([message.name isEqualToString:@"gotoFirstVC"])
{
[self showFirstVC];
}
//進(jìn)入詳情頁(yè)
else if ([message.name isEqualToString:@"gotodetailVC"])
{
NSString*url = [message.body objectForKey:@"body"];
[self gotodetial:url];
}
}
接下來(lái)就是具體的調(diào)用方法的實(shí)現(xiàn)了粗恢。
//彈出登陸頁(yè)面
- (void)showLoginView
{
//具體看項(xiàng)目中的實(shí)現(xiàn)了
}
/**
跳轉(zhuǎn)到首頁(yè)
/
-(void)showFirstVC{
//具體看項(xiàng)目中的實(shí)現(xiàn)了
}
/*
跳轉(zhuǎn)到詳情頁(yè)
*/
-(void)gotodetial:(NSString *)url {
//具體看項(xiàng)目中的實(shí)現(xiàn)了
}
2.oc調(diào)用js的方法欧瘪,給js傳值
oc調(diào)用js是通過evaluateJavaScript恋追,這個(gè)方法里面的實(shí)現(xiàn)的
我這里處理的時(shí)候,是在頁(yè)面加載完成的時(shí)候嗅绸,oc給js傳值
比如我需要給js傳用戶的登錄狀態(tài)和用戶的信息撕彤,只要在加載完成的時(shí)候,調(diào)用方法蚀狰,直接給值就行职员。
代碼示例
//加載頁(yè)面數(shù)據(jù)完成
-(void)webView:(WKWebView )webView didFinishNavigation:(null_unspecified WKNavigation )navigation
{
[_wkwebView evaluateJavaScript:@"document.getElementById("content").offsetHeight;" completionHandler:^(id _Nullable result, NSError * _Nullable error) {
//獲取頁(yè)面高度焊切,并重置webview的frame
CGFloat documentHeight = [result doubleValue];
CGRect frame = _wkwebView.frame;
frame.size.height = documentHeight+[UIScreen mainScreen].bounds.size.height-58 /顯示不全/;
_wkwebView.frame = frame;
}];
NSString *js = [NSString stringWithFormat:@"isLogin(%d)",[[UserInfoSingleton shareUserInfoSingleton] isLogin]];
// NSLog(@"%@",message.name);
//[self.wkwebView evaluateJavaScript:js completionHandler:nil];
[self.wkwebView evaluateJavaScript:js completionHandler:^(id item, NSError * _Nullable error) {
}];
NSString *js1 = [NSString stringWithFormat:@"userId(\'%@\')",[UserInfoSingleton shareUserInfoSingleton].userId];
// [self.wkwebView evaluateJavaScript:js1 completionHandler:nil];
[self.wkwebView evaluateJavaScript:js1 completionHandler:^(id item, NSError * _Nullable error) {
}];
}
- 彈窗
代碼示例
//彈窗
-(void)webView:(WKWebView *)webView runJavaScriptAlertPanelWithMessage:(NSString *)message initiatedByFrame:(WKFrameInfo *)frame completionHandler:(void (^)(void))completionHandler{
UIAlertController *alertController = [UIAlertController alertControllerWithTitle:@"提示" message:message?:@"" preferredStyle:UIAlertControllerStyleAlert];
[alertController addAction:([UIAlertAction actionWithTitle:@"確認(rèn)" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
completionHandler();
}])];
[self presentViewController:alertController animated:YES completion:nil];
}
-(void)webView:(WKWebView *)webView runJavaScriptConfirmPanelWithMessage:(NSString *)message initiatedByFrame:(WKFrameInfo *)frame completionHandler:(void (^)(BOOL))completionHandler{
// DLOG(@"msg = %@ frmae = %@",message,frame);
UIAlertController *alertController = [UIAlertController alertControllerWithTitle:@"提示" message:message?:@"" preferredStyle:UIAlertControllerStyleAlert];
[alertController addAction:([UIAlertAction actionWithTitle:@"取消" style:UIAlertActionStyleCancel handler:^(UIAlertAction * _Nonnull action) {
completionHandler(NO);
}])];
[alertController addAction:([UIAlertAction actionWithTitle:@"確認(rèn)" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
completionHandler(YES);
}])];
[self presentViewController:alertController animated:YES completion:nil];
}
-(void)webView:(WKWebView *)webView runJavaScriptTextInputPanelWithPrompt:(NSString *)prompt defaultText:(NSString *)defaultText initiatedByFrame:(WKFrameInfo *)frame completionHandler:(void (^)(NSString * _Nullable))completionHandler{
UIAlertController *alertController = [UIAlertController alertControllerWithTitle:prompt message:@"" preferredStyle:UIAlertControllerStyleAlert];
[alertController addTextFieldWithConfigurationHandler:^(UITextField * _Nonnull textField) {
textField.text = defaultText;
}];
[alertController addAction:([UIAlertAction actionWithTitle:@"完成" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
completionHandler(alertController.textFields[0].text?:@"");
}])];
}
- wkwebview所有的代理方法
1.WKNavigationDelegate來(lái)追蹤加載過程
// 頁(yè)面開始加載時(shí)調(diào)用
-(void)webView:(WKWebView *)webView didStartProvisionalNavigation:(WKNavigation *)navigation;
// 當(dāng)內(nèi)容開始返回時(shí)調(diào)用
-(void)webView:(WKWebView *)webView didCommitNavigation:(WKNavigation *)navigation;
// 頁(yè)面加載完成之后調(diào)用
-(void)webView:(WKWebView *)webView didFinishNavigation:(WKNavigation *)navigation;
// 頁(yè)面加載失敗時(shí)調(diào)用
-(void)webView:(WKWebView *)webView didFailProvisionalNavigation:(WKNavigation *)navigation;
2. WKNavigtionDelegate來(lái)進(jìn)行頁(yè)面跳轉(zhuǎn)
// 接收到服務(wù)器跳轉(zhuǎn)請(qǐng)求之后再執(zhí)行
-(void)webView:(WKWebView *)webView didReceiveServerRedirectForProvisionalNavigation:(WKNavigation *)navigation;
// 在收到響應(yīng)后专肪,決定是否跳轉(zhuǎn)
-(void)webView:(WKWebView *)webView decidePolicyForNavigationResponse:(WKNavigationResponse *)navigationResponse decisionHandler:(void (^)(WKNavigationResponsePolicy))decisionHandler;
// 在發(fā)送請(qǐng)求之前刹勃,決定是否跳轉(zhuǎn)
-(void)webView:(WKWebView *)webView decidePolicyForNavigationAction:(WKNavigationAction *)navigationAction decisionHandler:(void (^)(WKNavigationActionPolicy))decisionHandler;
```
- WKUIDelegate
//1.創(chuàng)建一個(gè)新的WebVeiw
-(nullable WKWebView *)webView:(WKWebView *)webView createWebViewWithConfiguration:(WKWebViewConfiguration *)configuration forNavigationAction:(WKNavigationAction *)navigationAction windowFeatures:(WKWindowFeatures *)windowFeatures;
//2.WebVeiw關(guān)閉(9.0中的新方法)
-(void)webViewDidClose:(WKWebView *)webView NS_AVAILABLE(10_11, 9_0);
//3.顯示一個(gè)JS的Alert(與JS交互)
-(void)webView:(WKWebView *)webView runJavaScriptAlertPanelWithMessage:(NSString *)message initiatedByFrame:(WKFrameInfo *)frame completionHandler:(void (^)(void))completionHandler;
//4.彈出一個(gè)輸入框(與JS交互的)
-(void)webView:(WKWebView *)webView runJavaScriptTextInputPanelWithPrompt:(NSString *)prompt defaultText:(nullable NSString *)defaultText initiatedByFrame:(WKFrameInfo *)frame completionHandler:(void (^)(NSString * __nullable result))completionHandler;
//5.顯示一個(gè)確認(rèn)框(JS的)
-(void)webView:(WKWebView *)webView runJavaScriptConfirmPanelWithMessage:(NSString *)message initiatedByFrame:(WKFrameInfo *)frame completionHandler:(void (^)(BOOL result))completionHandler;
```
以上的所有的代理方法,根據(jù)自己的使用情況調(diào)用就行嚎尤。
#注意
- Objective-C在回調(diào)JavaScript的時(shí)候荔仁,WKWebView沒有辦法傳過來(lái)一個(gè)匿名函數(shù),所以回調(diào)方式芽死,要么執(zhí)行一段JavaScript代碼乏梁,或者就是調(diào)用JavaScript那邊的一個(gè)全局函數(shù)。一般是采用后者收奔,至于Web端雖說暴露了一個(gè)全局函數(shù),同樣可以把這一點(diǎn)代碼處理的很優(yōu)雅坪哄。Objective-C傳給JavaScript的參數(shù),可以為Number, String, and Object。參考如下:
// 數(shù)字
NSString *js = [NSString stringWithFormat:@"方法名(%@)", number];
[self.webView evaluateJavaScript:js completionHandler:nil];
// 字符串
NSString *js = [NSString stringWithFormat:@"方法名('%@')", string];
[self.webView evaluateJavaScript:js completionHandler:nil];
// 對(duì)象
NSString *js = [NSString stringWithFormat:@"方法名(%@)", @{@"name" : @"timefor"}];
[self.webView evaluateJavaScript:js completionHandler:nil];
// 帶返回值的JS函數(shù)
[self.webView evaluateJavaScript:@"方法名()" completionHandler:^(id result, NSError * _Nullable error) {
// 接受返回的參數(shù)翩肌,result中
}];
```
- 我在本文中寫的模暗,都是根據(jù)自己項(xiàng)目中用到的寫的,主要是在和h5交互的這塊念祭,互相傳值兑宇,調(diào)用的,都是用的自己項(xiàng)目中的方法名粱坤×ジ猓可能會(huì)好理解一些。
- 關(guān)于導(dǎo)航欄處理的之類的站玄,這次項(xiàng)目中沒用用到枚驻,所以沒有做深入的研究,只是研究了項(xiàng)目中的問題株旷。