ios中webView和JavaScript交互总结

前言:js交互问题在很多app种都能用到,面试中也经常被问到,我做的项目中也用到了交互,今天就总结下,希望能帮助到有需要的人.


js交互页面.png

上图中就是我所做的app中涉及到交互的页面,点击不同的按钮能跳转到不同的页面,下面做具体的分析.

理论段落
在ios7以前,苹果还没有推出JavaScriptCore的时候,我们对js的操作只有在webview里面的一个函数stringByEvaluatingJavaScriptFromString,js对oc的回调都是基于URL的拦截进行的操作,一般我们都借用一些三方库,我当时用的是WebViewJavascriptBridge.那在ios7之后,我们对js的操作就会相对方便.我先介绍几个关键词
JSContext:JS的执行环境,是通过JSVirtualMachine管理这所有对象的生命周期,每个JSValue都和JSContext相关联着
JSValue: JS对象在JSVirtualMachine中的一个强引用.我们对JS的操作都是通过它的.并且每个JSValue都强引用这一个JSContext.同时OC和JS对象之间的转换也是通过它.
JSVirtualMachine: JS运行的虚拟机,有独立的堆空间和垃圾回收机制.
JSExport:一个协议,如果JS想直接调用oc对象里面的方法和属性,那么这个OC对象只需要实现这个JSExport协议方法就可以
============================我是分割线=============================

1.OC调用JS(我们不在通过URL拦截,我们直接取UIWebView的context,然后进行js操作)

- (void)webViewDidFinishLoad:(UIWebView *)webView {
    [SVProgressHUD dismiss];
    
    //oc通过stringByEvaluatingJavaScriptFromString这个方法来调用js的
    self.navigationItem.title = [webView stringByEvaluatingJavaScriptFromString:@"document.title"];
    //获取js的上下文
    self.context = [webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
    //给js传用户id
    [self appToJs:@"user_id" data:SALFSTR(User_ID)];
    //给js传版本号
    [self appToJs:@"versionCode" data:[UIApplication sharedApplication].appVersion];
    //给js传token
    [self appToJs:@"token" data:SALFSTR(User_Token)];
    
    //设置 app 交互代理
    self.context[@"app"] = self;
    
    self.context.exceptionHandler = ^(JSContext *context, JSValue *exceptionValue) {
        context.exception = exceptionValue;
        //打印异常信息
        NSLog(@"异常信息:%@", exceptionValue);
    };
}
//交互的方法
- (void)appToJs:(NSString *)type data:(NSString *)data{
    
    NSString * H5Str = [NSString stringWithFormat:@"appToJs('%@','%@')",type,data];
    [self.context evaluateScript:H5Str];
    
}

2.JS调用OC有两种方法:block和JSExport协议,我用的是后者

.h文件中
#import "BaseViewController.h"
#import <JavaScriptCore/JavaScriptCore.h>

@protocol HomeWebViewJSExport <JSExport>

//协议方法
- (void)js:(NSString *)type App:(NSString *)data;

@end

@interface HomeWebViewController : BaseViewController<UIWebViewDelegate,HomeWebViewJSExport>

@property(strong,nonatomic)NSString *webUrl;

@end
.m文件中 主要是对协议方法的实现
- (void)js:(NSString *)type App:(NSString *)data {
    
    NSLog(@"%@ %@",type,data);
    
    dispatch_async(dispatch_get_main_queue(), ^{
        
        if ([type isEqualToString:@"scenic"]) {
            //进入景区详情
            [self jumpScenicViewWithID:data];
        }else if ([type isEqualToString:@"url"]){
            //进入专题页
            [self pushToBannerWithID:data];
        }
    }); 
}

参考资料:葵花宝典书籍.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 本文由我们团队的 纠结伦 童鞋撰写。 写在前面 本篇文章是对我一次组内分享的整理,大部分图片都是直接从keynot...
    知识小集阅读 15,286评论 11 172
  • 写在前面 本篇文章是对我一次组内分享的整理,大部分图片都是直接从keynote上截图下来的,本来有很多炫酷动效的,...
    等开会阅读 14,509评论 6 69
  • 注:本文copy自http://www.jianshu.com/p/ac534f508fb0,纯属当笔记使用。 概...
    BookKeeping阅读 742评论 1 3
  • 项目中涉及OC与网页的交互,查找资料时看到了JavaScriptCore.framework,就对照文章ios7 ...
    YaoYaoX阅读 2,379评论 7 11
  • 那天晚上,我爱上了你, 因为我知道,有了你,我的生活, 会有更多的精彩,风雨共济,一路有你! 致敬,简书APP 【...
    A伟东阅读 297评论 0 0