专栏首页陈满iOSiOS-JavaScript交互方案:网页监听APP返回键(goback)+APP监听网页返回键(JS调用OC对象方法)

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 条评论
登录 后参与评论

相关文章

  • iOS开发·runtime原理与实践: 方法交换篇(Method Swizzling)(iOS“黑魔法”,埋点统计,禁止UI控件连续点击,防奔溃处理)

    Method Swizzing是发生在运行时的,主要用于在运行时将两个Method进行交换,我们可以将Method Swizzling代码写到任何地方,但是只有...

    陈满iOS
  • iOS面试系列·某安金融科技公司iOS开发面试回忆

    也许面试官自己也没自己实现过,毕竟有些东西苹果为什么这样设计,若不是苹果公司的工程师无法知道。

    陈满iOS
  • iOS·采用第三方(百度地图SDK)实现定位等功能开发

    首先,申请一个baidu账号,接着进入新建密钥入口申请成为baidu地图开发者,填写相关开发者信息和短信验证码。接下来点击创建应用,如官方文档新建密钥文档指南里...

    陈满iOS
  • 真·药神:伟哥的发明者,宣布投身人工智能

    你可能没听说过 Viagra,但一定听说过「伟哥」,也一定见过药店门口「万艾可」的海报,伟哥、万艾可,就是 Viagra 的常见的中文译名。

    HyperAI超神经
  • 业界 | 每天1800万单,1小时送到,美团外卖如何优化配送模型?

    大数据文摘
  • 自己实现一个滑动窗口

    上述计算中的alpha的值是一个0~1之间的常量,aplha值决定了一段时间内的平滑水平,alpha越趋于1,历史值对当前的平均值的影响越大,反之亦然

    Meet相识
  • C#8.0 新增功能

    可将 readonly 修饰符应用于结构的任何成员。 它指示该成员不会修改状态。 这比将 readonly 修饰符应用于 struct 声明更精细。 请考虑以下...

    张传宁老师
  • 开源库BaseRecyclerViewAdapterHelper

    相信大家RecyclerView应该不会陌生,大多数开发者应该都使用上它了,它也是google推荐替换ListView的控件,但是用过它的同学应该都知道它在某些...

    xiangzhihong
  • WPF 从触摸消息转触摸事件

    在 WPF 程序可能因为一些坑让程序触摸失效,如果此时还可以收到系统的触摸消息,那么可以通过从触摸消息转触摸事件解决程序触摸失效但不适合所有触摸失效程序

    林德熙
  • 购物车模块

    爱撒谎的男孩

扫码关注云+社区

领取腾讯云代金券