美文网首页iOSios 知识小集使用WKWebView和H5交互
iOS开发:给UIWebview的导航栏添加返回、关闭按钮

iOS开发:给UIWebview的导航栏添加返回、关闭按钮

作者: First灬DKS | 来源:发表于2016-05-12 11:28 被阅读13803次

    在我们平日的开发中,不免有原生与H5的交互,比如说:从原生页面的一个按钮,点击之后跳转到了一个H5的页面A,A页面中又有一个按钮,点击之后,又加载了一个新的H5页面B,从B点击一个按钮,又加载一个新的H5页面C,如果此时我们点击左上角的返回按钮,会直接返回到我们的原生页面;

    是不是上面给用户的体验很不好(当然残品经理会觉得是,我们都是无所谓的啦),此时我们想要重新定制返回按钮,我们想要从C页面判断是否还有上一级H5页面可供返回,如果有上一级页面还是H5,点击左上角的返回则返回到B页面,并且在B页面的左上角加上一个关闭按钮,这个关闭按钮的作用主要是为了关闭所有的H5的页面,直接返回到我们原生的页面;如果我们不点击关闭按钮,还是点击返回,则从B页面返回到A页面;再次点击返回,则关闭了H5的页面,回到了原生的页面;

    说的也许有点儿绕,不过大致思想就是:先判断当前的H5页面是否可以返回:

    //判断当前H5是否可以返回
    [self.webView canGoBack]
    

    如果可以返回,则返回到上一个H5页面,并在左上角添加一个关闭按钮,如果不可以返回,则直接:

    //回到原生页面
    [self.navigationController popViewControllerAnimated:YES];
    

    下面是我的主要实现代码,我写一个继承与UIViewController的类,接受了UIWebviewDelegate,并定义了一个UIwebview的属性,给外面留了一个方法,只需要传递一个URL,我们就可以加载;如果有地方需要加载H5的页面,我们可以直接集成与这个类,这样的好处在于方便维护;

    当然我封装的这个类,同时也是支持HTTPS的请求的;话不多说,代码如下:

    创建一个类,继承与UIViewController,.h中的代码

    #import <UIKit/UIKit.h>
    
    @interface SYWebViewController : UIViewController<UIWebViewDelegate, NSURLConnectionDelegate>
    
    //定义一个属性,方便外接调用
    @property (nonatomic, strong) UIWebView *webView;
    
    //声明一个方法,外接调用时,只需要传递一个URL即可
    - (void)loadHTML:(NSString *)htmlString;
    
    @end
    
    

    .m中的实现如下:

    #import "SYWebViewController.h"
    
    @interface NSURLRequest (InvalidSSLCertificate)
    
    + (BOOL)allowsAnyHTTPSCertificateForHost:(NSString*)host;
    + (void)setAllowsAnyHTTPSCertificate:(BOOL)allow forHost:(NSString*)host;
    
    @end
    
    @interface SYWebViewController ()
    
    @property (nonatomic, strong) NSURLRequest *request;
    //判断是否是HTTPS的
    @property (nonatomic, assign) BOOL isAuthed;
    
    //返回按钮
    @property (nonatomic, strong) UIBarButtonItem *backItem;
    //关闭按钮
    @property (nonatomic, strong) UIBarButtonItem *closeItem;
    
    @end
    
    @implementation SYWebViewController
    
    - (void)viewDidLoad {
        [super viewDidLoad];
        self.view.backgroundColor = [UIColor whiteColor];
        self.webView = [[UIWebView alloc] initWithFrame:CGRectMake(0, 0, self.view.frame.size.width, self.view.frame.size.height - 64)];
        [self.view addSubview:self.webView];
        
        [self addLeftButton];
    }
    
    //加载URL
    - (void)loadHTML:(NSString *)htmlString
    {
        NSURL *url = [NSURL URLWithString:htmlString];
        self.request = [NSURLRequest requestWithURL:url cachePolicy:NSURLRequestReloadIgnoringLocalCacheData timeoutInterval:5.0];
        [NSURLRequest setAllowsAnyHTTPSCertificate:YES forHost:[url host]];
        [self.webView loadRequest:self.request];
    }
    
    #pragma mark - UIWebViewDelegate
    
    //开始加载
    - (BOOL)webView:(UIWebView *)awebView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType
    {
        NSString* scheme = [[request URL] scheme];
        //判断是不是https
        if ([scheme isEqualToString:@"https"]) {
            //如果是https:的话,那么就用NSURLConnection来重发请求。从而在请求的过程当中吧要请求的URL做信任处理。
            if (!self.isAuthed) {
                NSURLConnection* conn = [[NSURLConnection alloc] initWithRequest:request delegate:self];
                [conn start];
                [awebView stopLoading];
                return NO;
            }
        }
        return YES;
    }
    
    //设置webview的title为导航栏的title
    - (void)webViewDidFinishLoad:(UIWebView *)webView
    {
        self.title = [webView stringByEvaluatingJavaScriptFromString:@"document.title"];
    }
    
    #pragma mark ================= NSURLConnectionDataDelegate <NSURLConnectionDelegate>
    
    - (BOOL)connection:(NSURLConnection *)connection canAuthenticateAgainstProtectionSpace:(NSURLProtectionSpace *)protectionSpace
    {
        return [protectionSpace.authenticationMethod isEqualToString:NSURLAuthenticationMethodServerTrust];
    }
    
    - (void)connection:(NSURLConnection *)connection willSendRequestForAuthenticationChallenge:(NSURLAuthenticationChallenge *)challenge
    {
        if ([challenge previousFailureCount] == 0) {
            self.isAuthed = YES;
            //NSURLCredential 这个类是表示身份验证凭据不可变对象。凭证的实际类型声明的类的构造函数来确定。
            NSURLCredential *cre = [NSURLCredential credentialForTrust:challenge.protectionSpace.serverTrust];
            [challenge.sender useCredential:cre forAuthenticationChallenge:challenge];
        }
    }
    
    - (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
    {
        NSLog(@"网络不给力");
    }
    
    - (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
    {
        self.isAuthed = YES;
        //webview 重新加载请求。
        [self.webView loadRequest:self.request];
        [connection cancel];
    }
    
    #pragma mark - 添加关闭按钮
    
    - (void)addLeftButton
    {
        self.navigationItem.leftBarButtonItem = self.backItem;
    }
    
    //点击返回的方法
    - (void)backNative
    {
        //判断是否有上一层H5页面
        if ([self.webView canGoBack]) {
              //如果有则返回
            [self.webView goBack];
            //同时设置返回按钮和关闭按钮为导航栏左边的按钮
            self.navigationItem.leftBarButtonItems = @[self.backItem, self.closeItem];
        } else {
            [self closeNative];
        }
    }
    
    //关闭H5页面,直接回到原生页面
    - (void)closeNative
    {
        [self.navigationController popViewControllerAnimated:YES];
    }
    
    #pragma mark - init
    
    - (UIBarButtonItem *)backItem
    {
        if (!_backItem) {
            _backItem = [[UIBarButtonItem alloc] init];
            UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
            //这是一张“<”的图片,可以让美工给切一张
            UIImage *image = [UIImage imageNamed:@"sy_back"];
            [btn setImage:image forState:UIControlStateNormal];
            [btn setTitle:@"返回" forState:UIControlStateNormal];
            [btn addTarget:self action:@selector(backNative) forControlEvents:UIControlEventTouchUpInside];
            [btn.titleLabel setFont:[UIFont systemFontOfSize:17]];
            [btn setTitleColor:[UIColor sy_backColor] forState:UIControlStateNormal];
            //字体的多少为btn的大小
            [btn sizeToFit];
            //左对齐
            btn.contentHorizontalAlignment = UIControlContentHorizontalAlignmentLeft;
            //让返回按钮内容继续向左边偏移15,如果不设置的话,就会发现返回按钮离屏幕的左边的距离有点儿大,不美观
            btn.contentEdgeInsets = UIEdgeInsetsMake(0, -15, 0, 0);
            btn.frame = CGRectMake(0, 0, 40, 40);
            _backItem.customView = btn;
        }
        return _backItem;
    }
    
    - (UIBarButtonItem *)closeItem
    {
        if (!_closeItem) {
            _closeItem = [[UIBarButtonItem alloc] initWithTitle:@"关闭" style:UIBarButtonItemStylePlain target:self action:@selector(closeNative)];
        }
        return _closeItem;
    }
    
    @end
    

    具体的使用方法就是,创建一个类,继承与这类:SYWebViewController

    #import <UIKit/UIKit.h>
    #import "SYWebViewController.h"
    
    @interface SYFlashHTMLViewController : SYWebViewController
    
    @end
    
    

    然后在这个类的.m中的viewDidLoad中,调用父视图加载URL的方法,即:

    - (void)viewDidLoad {
        [super viewDidLoad];
        self.webView.delegate = self;
        [self loadHTML:self.htmlString];
    }
    

    效果如下图:打开百度(图片1),点击图片进入(图片2),点击返回返回到(图片3),图片1和图片的区别在于多了个关闭按钮:

    图片1.png 图片2.png 图片3.png

    GitHub地址:Demo
    纯手打,喜欢点个赞,希望对看到的你有所帮助!!!

    相关文章

      网友评论

      • IOS开发攻城狮_Fyc:为什么内部跳转没有关闭按钮,返回一下才出现,写的有问题呀
        First灬DKS:@IOS开发攻城狮_Fyc 那你可以在web页面加载成功之后,在finish方法里面判断当前的web是否可以返回,可以返回的话,就显示关闭按钮;
        IOS开发攻城狮_Fyc:@First灬DKS 这个逻辑是不对的那,参考微信
        First灬DKS:目前的实现思路是:进入到web页面之后,点击返回时才会判断是否还是H5页面,如果是的话,就出现关闭按钮;
      • 新校花园地:最后一个页面没有关闭按钮吗?
        First灬DKS:@韩艳威 从原生进入到H5页面,是没有关闭按钮的;如果在H5页面返回的话,如果还有H5页面,则会显示关闭按钮;
        新校花园地:还是说从原生应用进入的第一个H5页面没有关闭关闭。
      • Sanchain:学习了,谢谢!
        First灬DKS:@Sanchain :smile:
      • 李小南:楼主项目中有没有在webView侧滑返回上一级H5页面的?能否提供大概思路
        First灬DKS:我工程中目前没有涉及到H5侧滑返回上一级H5的需求;侧滑会直接返回上一级VC,关于这个问题,如果要返回H5,应该先把VC的侧滑给关闭,然后去给web加侧滑功能,当然也是我个人的想法,没有经过验证;
      • 那仅有的执著:楼主, 在页面跳转的时候 ,右侧有一个按钮,一级页面显示,在二级页面隐藏,只在一进来的一级页面显示,怎么判断啊?
        First灬DKS:大概理解你的意思,你试试用[self.webView canGoBack]这个方法,如果可以返回,说明已经不再一级页面了,此时隐藏;或者是点击H5页面按钮时,让H5页面调用你一个方法去隐藏这个按钮;
      • 是我始终拒绝成长吗:[NSURLRequest setAllowsAnyHTTPSCertificate:YES forHost:[url host]];这个是私有方法吗
        First灬DKS:@是我始终拒绝成长吗 你看我.m中的实现;
        是我始终拒绝成长吗:@First灬DKS 直接掉的话没有这个方法
        First灬DKS:不是私有方法,亲测可用!
      • 那仅有的执著:[self loadHTML:self.htmlString]; 继承的controller里找不到self.htmlString
        First灬DKS:@那仅有的执著 这个是个属性,你有没有定义呢?
      • 伟大的梦想:楼主 请问怎么知道是否需要返回上一级呢 求解答 很急 谢谢!
        First灬DKS:@伟大的梦想 点击返回按钮时, if ([self.webView canGoBack]) 成立,则返回到第一个web页面;不是这样吗?
        伟大的梦想:@First灬DKS 假如我点击了一个web view上得某个控件 重新打开了二级菜单 我怎么判断返回上一级
        First灬DKS:@伟大的梦想 if ([self.webView canGoBack]) 这个是判断是否可以返回上一级的代码,你所说的是否需要返回上一级,是需要判断什么时候点击返回按钮吗?
      • 因为太有钱:楼主的 需求 有没有 在页面 跳转的时候 导航栏的标题也跟着改变,右侧也需要一个按钮,每跳转一个页面,点击右侧的按钮也需要不同的处理方式!!
        First灬DKS:@cocoazxy 导航栏的title是可以变化的!右侧是可以添加按钮的,此时需要原生与H5的交互了!让web控制按钮的点击方法!
      • 饥人谷_张炜:来一波demo啊.. = =
        First灬DKS:@麒麟阁饿了 这块儿东西我是从工程中整理出来,没有做Demo,你只需要实现一下上面的.h和.m 就可以了,很简单的; :smiley:
      • 一杯红酒mm:请问楼主,点击图3上的返回能不能返回到原声界面
        First灬DKS:@一杯红酒mm 如果H5页面已经进入到了二级页面,只有点击关闭才可以返回到原生页面;点击返回只能返回到上一个H5的页面;回复的有些晚了,不好意思!
      • 百里玄卿:想问下楼主,怎么去掉或者隐藏webView自带的导航栏呢?还有底部的电脑版手机版之类的。
        First灬DKS:@大哥赵 这个没有遇到过,这些应该是H5那边控制的吧;
      • 9178f5dfbb83:在浏览器里面多点几次,返回按钮的功能也是沿着历史记录返回,这样不合理吧,有什么好的解决办法吗
        First灬DKS:@9178f5dfbb83 目前处理的就是按照历史记录来的,如果[self.webView canGoBack],就返回上一个webView,上面还有一个关闭按钮,是把当前的webView关闭,返回到原生页面的,如果你想不按照历史的记录来,应该可以在点击返回按钮的方法里面处理吧,或者你有什么好的方法了,也希望可以分享一下!
      • Ambitiones:为什么我的导航条标题没有根据web标题切换呢
        First灬DKS:@cocoazxy 是的?
        因为太有钱:@First灬DKS 导航栏的title 是根据web的title 改变的?
        First灬DKS:@one_person 你有没有设置web的title为导航栏的title呢?或者是你在哪里设置成了一个定值,或者是代理方法没有执行?
      • 叶小合:楼主能给个github链接吗
        叶小合:好的
        First灬DKS:@合肥叶子 不好意思啊,这篇文章没有写demo,是从我的工程中剥离出来的!你只需要把上面的.h和.m在工程中实现一下即可;

      本文标题:iOS开发:给UIWebview的导航栏添加返回、关闭按钮

      本文链接:https://www.haomeiwen.com/subject/dixprttx.html