iOS开发 UIWebView使用

在App开发过程中经常需要加载网页,iOS开发领域加载网页主要有两种方式:一是利用UIWebView,二是利用iOS8出现的WKWebView.下面主要介绍UIWebView,WKWebView移步http://www.jianshu.com/p/5b29df96bc9f
UIWebView 主要分四个方面讲解:1.加载网页;2.回调方法;3.网页加载进度条显示;4.原生与js交互
1.加载网页:UIVWebView加载本地网页代码和非本地网页

    UIWebView *webView = [[UIWebView alloc] init];  
    webView.delegate = self;  
    webView.frame = self.view.bounds;  
    NSString *htmlPath = [[NSBundle mainBundle] pathForResource:@"index" ofType:@"html"];  
    NSString *htmlCont = [NSString stringWithContentsOfFile:htmlPath encoding:NSUTF8StringEncoding error:nil];  
    // 获取当前应用的根目录  
    NSString *path = [[NSBundle mainBundle] bundlePath];  
    NSURL *baseURL = [NSURL fileURLWithPath:path];  
    [webView loadHTMLString:htmlCont baseURL:baseURL]; 
   //加载非本地网页  
    NSURL *url = [[NSURL alloc] initWithString:@"https://www.baidu.com"];  
    NSURLRequest *request = [[NSURLRequest alloc] initWithURL:url];  
    [webView loadRequest:request];

2.回调方法,回调方法比较简单,后面再介绍与js交互的时候在介绍。

//网页开始加载时调用  
- (void)webViewDidStartLoad:(UIWebView *)webView {  
  
}  
//网页加载完成时调用,在该方法里面可以实现一些与js调用相关的代码  
- (void)webViewDidFinishLoad:(UIWebView *)webView {  
  
}  
//网页加载失败时调用,做相关提示  
- (void)webView:(UIWebView *)webView didFailLoadWithError:(NSError *)error {  
  
}  

3.网页进度条显示,网页进度条显示对于UIWebView来说,目前微信,支付宝等做的显示都是假象,即使没有网也可以跑到80%。我们可以利用UIView直接画一个,启动一个定时器不断修改它的frame,在网页加载完成的回调方法里面将其隐掉即可,也可以使用比较流行的NJKWebViewProgress。下面介绍这个第三方webView进度条显示的具体使用。
1)导入头文件:

#import "NJKWebViewProgressView.h"  
@interface ViewController ()<UIWebViewDelegate,NJKWebViewProgressDelegate>
@property (nonatomic, strong) UIWebView *webView;  
@property (nonatomic, strong) NJKWebViewProgress *webViewProgress;  
@property (nonatomic, strong) NJKWebViewProgressView *webViewProgressView;  
@end

2)初始化,注意在初始化的方法里面将webView的代理给了webViewProgress,此时webView加载完的相关调用将会被webViewProgress拦截,巧妙的是,webViewProgress拦截后又将这几个方法回调给了控制器,因此控制器里面仍然可以获取到网页的相关状态

    _webViewProgress = [[NJKWebViewProgress alloc] init];    
    _webView.delegate = _webViewProgress;    
    _webViewProgress.webViewProxyDelegate = self;    
    _webViewProgress.progressDelegate = self;    
        
    CGRect navBounds = self.navigationController.navigationBar.bounds;    
    CGRect barFrame = CGRectMake(0,    
                                 navBounds.size.height - 2,    
                                 navBounds.size.width,    
                                 2);    
    _webViewProgressView = [[NJKWebViewProgressView alloc] initWithFrame:barFrame];    
    _webViewProgressView.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleTopMargin;    
    [_webViewProgressView setProgress:0 animated:YES]

3)方法回调,控制器遵循NJKWebViewProgressDelagete和UIWebViewDelegate代理方法,虽然webView的代理方法给了webViewProgress,但在webViewProgress中又将方法回调给了控制器,因此控制器仍然是可以拿到web的回调方法。

#pragma NJKWebViewProgressDelegate  
-(void)webViewProgress:(NJKWebViewProgress *)webViewProgress updateProgress:(float)progress  {  
    [self.webViewProgressView setProgress:progress animated:YES];  
    self.title = [_webView stringByEvaluatingJavaScriptFromString:@"document.title"];  
}  
#pragma UIWebViewDelegate  
- (void) webViewDidStartLoad:(UIWebView *)webView  {  
    NSLog(@"webViewDidStartLoad");  
}  
- (void) webViewDidFinishLoad:(UIWebView *)webView  {  
    NSLog(@"webViewDidFinishLoad");  
}  
- (void) webView:(UIWebView *)webView didFailLoadWithError:(NSError *)error {  
    NSLog(@"didFailLoadWithError:%@", error);  
}  

4.原生与js交互,主要的操作是放在webViewdidFinishedLoad方法里面处理。
1)原生调用js方法:原生调用js方法,可以直接执行webView自带stringByEvaluatingByJavaScriptFromeString方法,如获取网页的title

[webView stringByEvaluatingJavaScriptFromString:@"document.title"];  

这里面有一个坑需要注意,我之前公司做的一个项目需要从h5页面中调用一个js的方法,该方法返回需要分享的数据,但不管我怎么调用,该方法都没有数据返回,因为js调用返回数据到原生是一个异步的过程,因此你直接获取返回的数据是不行的,而WKWebView可以做到这点,因为WKWebView有一个异步回调的方法获取js返回的数据。还有一点需要注意的是这个方法一定是在window下面可获取的,也就是一个全局的方法。如果真的想利用stringByEvaluatingByJavaScriptFromeString调用js方法获取相关数据,也是有办法的,前端可以在需要调用的js方法里面将需要的数据附在url后面,js方法执行完会回调 -(BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType;我们app端根据协商好的url判定,解析url获取数据。
2)js调用原生方法:有两种办法来让js调用原生方法。其一是利用block将js端的代码与原生进行关联,比如h5页面有个按钮,按钮点击会调用alert方法。

- (void)webViewDidFinishLoad:(UIWebView *)webView  {  
    // 以 html title 设置 导航栏 title  
    self.title = [webView stringByEvaluatingJavaScriptFromString:@"document.title"];  
      
    // Undocumented access to UIWebView's JSContext  
    self.context = [webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];  
      
    // 以 block 形式关联 JavaScript function  
    self.context[@"alert"] =  ^(NSString *str)  {  
        UIAlertView *alert = [[UIAlertView alloc]initWithTitle:@"msg from js" message:str delegate:nil cancelButtonTitle:@"ok" otherButtonTitles:nil, nil nil];  
        [alert show];  
    };  
}  

其二是利用-(BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType,在该方法里面进行拦截,根据url执行相关原生代码。比如h5页面有一个按钮,按钮点击要求原生弹到另外一个控制器。那我们可以和前端定义好url为secondVc://这种形式,
h5页面JS里面实现方法:

function commit() {  
     window.location.href = "secondVc://";  
}

当点击按钮的时候,requestString = "secondVc://";我们app端就可以根据url来判定执行相关动作了。

-(BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType{  
    NSString * requestString = request.URL.absoluteString;  
    NSLog(@"请求的地址:%@",requestString);  
    if ([requestString containsString:@"secondVc://"]){  
        //做你想要的操作  
        SecondVc *secondVC = [[UIViewController alloc] init];  
        [self.navigationController pushViewController:secondVC animated:YES];  
    }  
}

以上就是UIWebView的基本使用。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,816评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,729评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,300评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,780评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,890评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,084评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,151评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,912评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,355评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,666评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,809评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,504评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,150评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,121评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,628评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,724评论 2 351

推荐阅读更多精彩内容

  • 前言:iOS 开发中,h5 和原生实现通信有多种方式, JSBridge 就是最常用的一种,各 JSBridge ...
    ShannonChenCHN阅读 8,447评论 11 63
  • http://www.cnblogs.com/mddblog/p/5281748.html 一、整体介绍 UIWe...
    F麦子阅读 1,230评论 0 2
  • 前言 关于UIWebView的介绍,相信看过上文的小伙伴们,已经大概清楚了吧,如果有问题,欢迎提问。 本文是本系列...
    CoderLF阅读 8,953评论 2 12
  • 一、WebView WebView就是一个内嵌浏览器控件,在iOS中主要有两种WebView:UIWebView和...
    iOS祎阅读 1,081评论 0 2
  • 入夜,难入睡。 听着从前慢,想时代的快捷,生活得不惊不喜。 一生只够爱一人,说得看似平凡,需要的确是一辈子的相濡以...
    胡萝卜须兮阅读 96评论 0 1