iOS中JS 与OC的交互(JavaScriptCore.framework)

时间:2023-02-15 23:48:41

iOS中实现js与oc的交互,目前网上也有不少流行的开源解决方案;

如:react native

当然一些轻量级的任务使用系统提供的UIWebView 以及JavaScriptCore.framework 里面提供的api就能快速满足的。

方式一:iOS6上使用UIWebView 拦截url方式

以下是演示的js和oc端的代码

<html>
<head><title>JAAndOC1</title></head>
<script>
function sdkhi()
{
alert("hello world!");
} function sdkhito(name)
{
alert("hello to "+name);
} </script>
<body> <h1>hello world</h1>
<h1>hello world</h1>
<h1>hello world</h1>
<h1>hello world</h1> <h1><a href="TCSDK://START?name=cc&age=18&phone=10086">START</a></h1>
<h1><a href="javascript:sdkhi()">SDKHI</a></h1> </body> </html>
@interface ViewController ()<UIWebViewDelegate>
{
UIWebView *theWeb;
}
@end @implementation ViewController - (void)viewDidLoad {
[super viewDidLoad]; theWeb = [[UIWebView alloc]initWithFrame:self.view.bounds];
theWeb.scalesPageToFit = YES;
theWeb.delegate = self;
[self.view addSubview:theWeb]; //本示例方法适用于iOS 6 + NSURL *reqUrl = [NSURL URLWithString:[[NSBundle mainBundle] pathForResource:@"js1.html" ofType:nil]];
[theWeb loadRequest:[NSURLRequest requestWithURL:reqUrl]]; UIButton *clickBtn = [UIButton buttonWithType:UIButtonTypeCustom];
clickBtn.frame = CGRectMake(, , , );
[clickBtn setTitle:@"clisk" forState:];
[clickBtn setTitleColor:[UIColor redColor] forState:UIControlStateNormal];
[self.view addSubview:clickBtn];
[clickBtn addTarget:self action:@selector(clickToCloseAlert) forControlEvents:UIControlEventTouchUpInside]; } //OC调JS 并传参数
- (void)clickToCloseAlert
{
//在弹出的alert中可能会有来自 xxx.html 字样 (百度去掉)
[theWeb stringByEvaluatingJavaScriptFromString:@"sdkhito('cc')"];
} //JS 调 OC 并传参数
- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType
{ //访问如下自定义协议连接
//request.URL tcsdk://START?name=cc&age=18&phone=10086
//request.URL.scheme tcsdk 取出协议
//request.URL.host START 取出host
//request.URL.query name=cc&age=18&phone=10086 取出参数
NSLog(@"%@",request.URL);
NSLog(@"%@",request.URL.scheme); return YES;
}

1. OC 调用js代码主要通过如下接口

[theWeb stringByEvaluatingJavaScriptFromString:@"sdkhito('cc')"];

流程:iOS的UIWebView 加载 包含js的网页;在js中定义好函数接口;

然后在oc上就可以通过上面接口直接调用的;

2. JS 调用OC 主要通过UIWebView的代理方法shouldStartLoadWithRequest 拦截URL

//JS 调 OC 并传参数

- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType

{

//访问如下自定义协议连接

//request.URL   tcsdk://START?name=cc&age=18&phone=10086

//request.URL.scheme   tcsdk 取出协议

//request.URL.host     START 取出host

//request.URL.query name=cc&age=18&phone=10086  取出参数

NSLog(@"%@",request.URL);

NSLog(@"%@",request.URL.scheme);

return YES;

}

流程:iOS UIWebView 加载包含js的网页;在网页中用<a> 标签名其他跳转方式改变url的操作;

然后在代理中通过拆分url的结构,分别取出协议,host,参数等;来达到js调用oc方法的的目的。

  问题:通过url传参有限制

方式二:iOS7之后使用 JavaScriptCore.framework

该框架提供完整的js运行环境,在处理oc与js的交互中非常的方法;

如下测试代码

<html>
<head><title>JAAndOC2</title></head>
<script> //以下方法由oc 调用 js
function sdkhi3TO(name)
{
alert("hello world2!"+name);
} function sdkhi2()
{
alert("hello world2!");
} function sdkhito(name)
{
alert("hello to "+name);
} </script>
<body> <h1>hello world</h1>
<h1>hello world</h1>
<h1>hello world</h1>
<h1>hello world</h1> <h1><a href="TCSDK://START?name=cc&age=18&phone=10086">START</a></h1>
<h1><a href="javascript:sdkhi2()">SDKHI</a></h1> <!-- 以下由js 调用oc 的方法 -->
<h1><a href="javascript:ocsayhi('abc')">OCHI</a></h1>
<h1><a href="javascript:alert(pl.fullyName())">OCHI</a></h1> </body> </html>
#import "ViewController.h"
#import <JavaScriptCore/JavaScriptCore.h>
#import "Person.h" @interface ViewController ()<UIWebViewDelegate>
{
UIWebView *theWeb;
JSContext *jsCXT;
} @end @implementation ViewController - (void)viewDidLoad {
[super viewDidLoad];
//以下使用javascriptcore.framewordk //适用于iOS7.0 theWeb = [[UIWebView alloc]initWithFrame:self.view.bounds];
theWeb.scalesPageToFit = YES;
theWeb.delegate = self;
[self.view addSubview:theWeb]; NSURL *reqUrl = [NSURL URLWithString:[[NSBundle mainBundle] pathForResource:@"js2.html" ofType:nil]];
[theWeb loadRequest:[NSURLRequest requestWithURL:reqUrl]]; } - (void)webViewDidFinishLoad:(UIWebView *)webView
{
//使用webview的js执行环境
jsCXT = [webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"]; //异常处理
jsCXT.exceptionHandler =
^(JSContext *context, JSValue *exceptionValue)
{
context.exception = exceptionValue;
NSLog(@"%@", exceptionValue);
}; //OC 调用JS
//[jsCXT[@"sdkhi2"] callWithArguments:nil];
//sdkhito
//[jsCXT[@"sdkhito"] callWithArguments:@[@"aa"]]; //注册一个方法由js调用
jsCXT[@"ocsayhi"] = ^(NSString *name) {
NSLog(@"say hi to %@",name);
}; Person *pp = [[Person alloc]init];
pp.firstName = @"cc";
pp.lastName = @"j";
pp.site = @"baidu.com";
jsCXT[@"pl"] = pp; [jsCXT evaluateScript:@"alert(pl.fullyName())"]; }
#import <Foundation/Foundation.h>
#import <JavaScriptCore/JavaScriptCore.h>
@protocol PersonPtl <JSExport> @property (nonatomic,copy)NSString *site;
/**
对于多参数的方法,JavaScriptCore的转换方式将Objective-C的方法每个部分都合并在一起,冒号后的字母变为大写并移除冒号。比如下边协议中的方法,在JavaScript调用就是:doFooWithBar(foo, bar); **/
- (NSString *)fullyName; @end @interface Person : NSObject<PersonPtl> @property (nonatomic, copy) NSString *firstName;
@property (nonatomic, copy) NSString *lastName; @end
#import "Person.h"

@implementation Person

@synthesize site;

- (NSString *)fullyName
{
return [NSString stringWithFormat:@"%@ %@", self.firstName, self.lastName]; } @end

1. OC调用js

流程我们创建UIWebView 在 代理中获取当前页面的js环境;就是JSContext

jsCXT = [webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];

  然后在网页js中定义的方法,在这里我们就可以直接调用了;

//OC 调用JS

[jsCXT[@"sdkhi2"] callWithArguments:nil];

[jsCXT[@"sdkhito"] callWithArguments:@[@"aa"]];

2. JS 调用 OC

这里有两个方式,我们可以通过block的方法在oc里为js 注入一下方法,让js调用

//注册一个方法由js调用

jsCXT[@"ocsayhi"] = ^(NSString *name) {

NSLog(@"say hi to %@",name);

};

另外一个方法,稍麻烦一点,我们需要先声明协议,然后实现协议方法;通过JSExport 来暴露给JS环境调用

如上面的代码中的Person类;

调用方式:

Person *pp = [[Person alloc]init];

pp.firstName = @"cc";

pp.lastName = @"j";

pp.site = @"baidu.com";

jsCXT[@"pl"] = pp;

//以下是在当前jscxt中执行代码,当然网页js中也可以直接写

[jsCXT evaluateScript:@"alert(pl.fullyName())"];

最后完整示例工程:

https://github.com/cocoajin/TDDDemo/tree/master/JSAndOC

参考:http://www.skyfox.org/ios-wkwebview-cookie-opration.html

iOS中JS 与OC的交互(JavaScriptCore.framework)的更多相关文章

  1. iOS中js与objective-c的交互&lpar;转&rpar;

    因为在iOS中没有WebKit.Framework这个库的,所以也就没有 windowScriptObject对象方法了.要是有这个的方法的话 就方便多了,(ps:MacOS中有貌似) 现在我们利用其 ...

  2. iOS开发&colon;JavaScriptCore&period;framework的简单使用--JS与OC的交互篇

    iOS7之后苹果为众猿推出了JavaScriptCore.framework这个框架,这个框架为大家在与JS交互上提供了很大帮助,可以在html界面上调用OC方法并传参,也可以在OC上调用JS方法并传 ...

  3. iOS下JS与OC互相调用(四)--JavaScriptCore

    前面讲完拦截URL的方式实现JS与OC互相调用,终于到JavaScriptCore了.它是从iOS7开始加入的,用 Objective-C 把 WebKit 的 JavaScript 引擎封装了一下, ...

  4. iOS下JS与OC互相调用(一)--UIWebView 拦截URL

    最近准备把之前用UIWebView实现的JS与原生相互调用功能,用WKWebView来替换.顺便搜索整理了一下JS 与OC 交互的方式,非常之多啊.目前我已知的JS 与 OC 交互的处理方式: * 1 ...

  5. iOS下JS与OC互相调用(六)--WKWebView &plus; WebViewJavascriptBridge

    上一篇文章介绍了UIWebView 如何通过WebViewJavascriptBridge 来实现JS 与OC 的互相调用,这一篇来介绍一下WKWebView 又是如何通过WebViewJavascr ...

  6. iOS下JS与OC互相调用(五)--UIWebView &plus; WebViewJavascriptBridge

    WebViewJavascriptBridge是一个有点年代的JS与OC交互的库,使用该库的著名应用还挺多的,目前这个库有7000+star.我去翻看了它的第一版本已经是4年前了,在版本V4.1.4以 ...

  7. iOS下JS与OC互相调用(八)--Cordova详解&plus;实战

    扯两句,可以跳过 由于项目中Cordova相关功能一直是同事在负责,所以也没有仔细的去探究Cordova到底是怎么使用的,又是如何实现JS 与 OC 的交互.所以我基本上是从零开始研究和学习Cordo ...

  8. UIWebView中JS与OC交互 WebViewJavascriptBridge的使用

    一.综述 现在很多的应用都会在多种平台上发布,所以很多程序猿们都开始使用Hybrid App的设计模式.就是在app上嵌入网页,只要写一份网页代码,就可以跑在不同的系统上.在iOS中,app多是通过W ...

  9. iOS UIWebView 中 js调用OC 打开相册 获取图片, OC调用js 将图片加载到html上

    线上html <!DOCTYPE html> <html> <head> <title>HTML中用JS调用OC方法</title> &lt ...

随机推荐

  1. poj3281 Dining

    Dining Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 14316   Accepted: 6491 Descripti ...

  2. 【CodeVS 1218】【NOIP 2012】疫情控制

    http://codevs.cn/problem/1218/ 比较显然的倍增,但是对于跨过根需要很多讨论,总体思路是贪心. 写了一上午,不想再说什么了 #include<cstdio> # ...

  3. apache日志文件太大的问题

    apache日志文件太大的问题 处理Apache日志的两种方法 rotatelogs 是 Apache 2.2 中自带的管道日志程序 rotatelogs [ -l ] logfile [ rotat ...

  4. sqlmap文件在tomcat7中运行报错原因及&lt&semi;&excl;&lbrack;CDATA&lbrack; &rsqb;&rsqb;&gt&semi;

    sqlmap在eclipse中运行,好好的.放到tomcat7中抛出如下异常: Caused by: java.lang.RuntimeException: Error occurred. Cause ...

  5. 安装ArchLinux BIOS

    VMware默认下使用BIOS 本文基于ArchLinux(https://www.archlinux.org/)Current Release: 2013.08.01的ISO写的安装教程! ISO下 ...

  6. RabbitMQ和kafka从几个角度简单的对比--转

    业界对于消息的传递有多种方案和产品,本文就比较有代表性的两个MQ(rabbitMQ,kafka)进行阐述和做简单的对比, 在应用场景方面, RabbitMQ,遵循AMQP协议,由内在高并发的erlan ...

  7. 弹出层iframe链接设置

    jQuery 比较方便就是创建删除了,所以创建一个弹出层就是当点击div的时候创建一个新的div利用固定位fixed(与浏览器窗口有关)和z-index覆盖body 并利用opacity设置其透明度产 ...

  8. Android实现购物车功能

    如图: 主要代码如下: actvity中的代码: publicclassShoppingCartActivity extendsBaseActivity {      private List< ...

  9. Log4Net 添加自定义字段并保存到数据库

    Log4Net是常用的功能强大的日志插件,该插件提供了几个默认字段 大家可能都用过Log4Net插件来记录日志,该插件默认提供了这几个字段@log_date, @thread, @log_level, ...

  10. laravel5 项目上线后务必将开发环境更改为生产环境

    如果以开发环境上线,出错信息将全通过json暴露出来了,屏蔽方式如下: .env 文件设置如下APP_ENV=productionAPP_DEBUG=false 改完设置后把缓存清理一遍 如果更改后清 ...