iOS-JavaScript交互方案:网页监听APP返回键(goback)+APP监听网页返回键(JS调用OC对象方法)

1.网页监听APP返回键(原生goback)

假设需求:当APP点击原生导航栏左上角返回键时,APP并不返回上级VC,而是让UIWebView返回上级页面。

1.1首先

新建CMWebViewController,让其继承UIViewController,即:

  • CMWebViewController.h中有:
@interface CMWebViewController : UIViewController

1.2 其次

建议基于UIViewController自建一个BackButtonHandler的分类,如UIViewController+BackButtonHandler.hUIViewController+BackButtonHandler.m

  • UIViewController+BackButtonHandler.h
#import <UIKit/UIKit.h>

@protocol BackButtonHandlerProtocol <NSObject>

@optional

-(BOOL)navigationShouldPopOnBackButton;

@end

@interface UIViewController (BackButtonHandler) <BackButtonHandlerProtocol>

@end
  • UIViewController+BackButtonHandler.m
#import "UIViewController+BackButtonHandler.h"

@implementation UIViewController (BackButtonHandler)

@end

@implementation UINavigationController (ShouldPopOnBackButton)

- (BOOL)navigationBar:(UINavigationBar *)navigationBar shouldPopItem:(UINavigationItem *)item {

    if([self.viewControllers count] < [navigationBar.items count]) {
        return YES;
    }

    BOOL shouldPop = YES;
    UIViewController* vc = [self topViewController];
    if([vc respondsToSelector:@selector(navigationShouldPopOnBackButton)]) {
        shouldPop = [vc navigationShouldPopOnBackButton];
    }

    if(shouldPop) {
        dispatch_async(dispatch_get_main_queue(), ^{
            [self popViewControllerAnimated:YES];
        });
    } else {
        // Workaround for iOS7.1. Thanks to @boliva - http://stackoverflow.com/posts/comments/34452906
        for(UIView *subview in [navigationBar subviews]) {
            if(0. < subview.alpha && subview.alpha < 1.) {
                [UIView animateWithDuration:.25 animations:^{
                    subview.alpha = 1.;
                }];
            }
        }
    }

    return NO;
}

3.最后

在实现文件CMWebViewController.m导入如上分类,并实现分类中BackButtonHandlerProtocol协议的navigationShouldPopOnBackButton方法。

  • CMWebViewController.m
#import "CMWebViewController.h"
#import "UIViewController+BackButtonHandler.h"
@interface CMWebViewController ()<UIWebViewDelegate,JSObjcDelegate>
- (BOOL)navigationShouldPopOnBackButton
{
    if ([_webView canGoBack]) {
        [_webView goBack];
        return NO;
    }
    return YES;
}

2.网页监听APP返回键(OC调用JS)

假设需求:APP隐藏原生导航栏,相当于网页全屏了,当在APP中点击网页端的左上角返回键时,APP退出UIWebView并返回上级VC页面。

2.1 iOS端

CMWebViewController实现文件代码

  • 导入头文件
#import <JavaScriptCore/JavaScriptCore.h>

@protocol JSObjcDelegate <JSExport>

//iosDelegate对象调用的JavaScript方法,必须声明!!!
- (void)getCall;

@end
  • 代理及属性
@interface OpenHelpWebViewController ()<JSObjcDelegate>
@property (nonatomic, strong) JSContext *jsContext;
@property (weak, nonatomic) IBOutlet UIWebView *webView;
@end
  • 设置JS-OC交互对象
- (void)webViewDidFinishLoad:(UIWebView *)webView {
    
    // 设置javaScriptContext上下文
    self.jsContext = [webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
    // 将iosDelegate对象指向自身
    self.jsContext[@"iosDelegate"] = self;
    
    self.jsContext.exceptionHandler = ^(JSContext *context, JSValue *exceptionValue) {
        context.exception = exceptionValue;
        NSLog(@"异常信息:%@", exceptionValue);
    };
}
  • iosDelegate对象方法(恭候JS调用)
- (void)getCall{
    
    NSLog(@"call");
    // 之后在回调JavaScript的方法Callback把内容传出去
    dispatch_async(dispatch_get_main_queue(), ^{
        
        //APP返回上级页面动作
        [self.navigationController popViewControllerAnimated:YES];
    });
}

2.2 H5/JS端

  • H5关键部分:布局元素ID
<header class="header test">
        <span>
            ![](./testFile/backIcon.png)
        </span>
        <h2 class="txt_cen">网页端标题</h2>
        <div></div>
</header>
  • JS关键部分:调用OC方法
<script type="text/javascript">
 
  $("#backId").click(function(){
    var flag =  getTheFlagString("flag");
    if(flag == "h5"){
        history.go(-1);
    }else if(android){
        window.androidDelegate.getCall();
    }else if(ios){
        window.iosDelegate.getCall();
    }
  });

</script>

参考文献

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏LeeCen

WKWebView ajax请求Cookie丢失

发现H5里面 ajax请求失败302,这可能Cookie丢失或Cookie不相同了

5741
来自专栏林德熙的博客

wpf DoEvents 用法原理存在的坑推荐方法

如果在执行一段卡UI的代码,这时如何让UI响应。如果存在代码需要获得依赖属性,那么代码就需要在UI线程执行,但是这时就会卡UI,为了让UI响应,所以就需要使用D...

6941
来自专栏守望轩

html页面表格导出到excel总结

最近一个项目需要把报表的表格导入excel,在网上找了一些方法,比较研究了一下,记在这里,备忘。 表格例子如下: <table id="tableExcel" ...

4829
来自专栏TechBox

史上最全的iOS之访问自定义cell的textField.text的N种方法前言方法一(方法1请略过)方法二(发送系统通知)方法三(发送自定义通知)方法四(使用block)方法五(使用delegate

1864
来自专栏菩提树下的杨过

Flash/Flex学习笔记(16):如何做自定义Loading加载其它swf

const FILE_PATH:String="main.swf"; const CLASS_NAME:String="MainSwf"; var loade...

1835
来自专栏青玉伏案

iOS开发之自定义表情键盘(组件封装与自动布局)

  下面的东西是编写自定义的表情键盘,话不多说,开门见山吧!下面主要用到的知识有MVC, iOS开发中的自动布局,自定义组件的封装与使用,Block回调,Cor...

26310
来自专栏葡萄城控件技术团队

七天学会ASP.NET MVC (五)——Layout页面使用和用户角色管理

系列文章 七天学会ASP.NET MVC (一)——深入理解ASP.NET MVC 七天学会ASP.NET MVC (二)——ASP.NET MVC 数据传递 ...

3328
来自专栏iOS 开发杂谈

浅析 NSTimer 和 CADisplayLink 内存泄漏

谈论 NSTimer & CADisplayLink 内存泄漏,要理解 NSTimer & CADisplayLink 的基础概念,下面通过一个倒计时的实现的 ...

2561
来自专栏Python疯子

Swift - MJRefresh库的使用详解1(配置,及库自带的下拉刷新组件)

除了使用 UIRefreshControl,网上也有许多第三方刷新库可供选择。MJRefresh 是其中比较优秀的一个。

5352
来自专栏向治洪

MobX 在 React Native开发中的应用

MobX 是一款精准的状态管理工具库,如果你在 React 和 React Native 应用中使用过 Flux、Alt、Redux 和 Reflux,那毫不犹...

5338

扫码关注云+社区

领取腾讯云代金券