-------------------------------------------------
很多关于objc 与 js 交互的文章都比较适用于 mac开发,iOS的webview 还是有所不一样,
本文提供了一个很好解决UIWebView内js和objc 交互的思路。
自然,从oc到js,可以使用 stringByEvaluatingJavaScriptFromString: 来实现。
从js到oc,采用比较巧妙的设计,UIWebView浏览器拦截 url请求,自定义url的方式拦截交互请。
-------------------------------------------------
-------------------------------------------------

UIWebView是IOS SDK中渲染网面的控件,在显示网页的时候,我们可以hack网页然后显示想显示的内容。其中就要用到javascript的知识,而UIWebView与javascript交互的方法就是stringByEvaluatingJavaScriptFromString:

有了这个方法我们可以通过objc调用javascript,可以注入javascript。

首先我们来看一下,如何调用javascript:

 
  1. [webView stringByEvaluatingJavaScriptFromString:@"myFunction();"];

这儿myFunction()就是我们的javascript方法。

再来看看入何注入javascript,我们先写一个需要注入的javascript:

 
  1. function showAlert() {
  2. alert('in show alert');
  3. }

保存为test.js,然后拖到xcode 的resource分组下。再用代码在初始化的时候注入这个js(如在viewDidLoad方法里)。

 
  1. NSString *filePath = [[NSBundle mainBundle] pathForResource:@"test" ofType:@"js"];
  2. NSString *jsString = [[NSString alloc] initWithContentsOfFile:filePath];
  3. [webView stringByEvaluatingJavaScriptFromString:jsString];

这样就注入了上面的js,那么我们可以随时调用js的方法,如何调用,上面有介绍。

那么我们能不能通过js来调用objc的方法呢。 当然可以,原理就是利用UIWebView重定向请求,传一些命令到我们的UIWebView,在UIWebView的delegate的方法中接收这些命令,并根据命令执行相应的objc方法。这样就相当于在javascript中调用objc的方法。说起来有点抽象,看看代码一下就明白。

首先我们写一个javascript 方法如下:

[javascript] 
  1. function sendCommand(cmd,param){
  2. var url="testapp:"+cmd+":"+param;
  3. document.location = url;
  4. }
  5. function clickLink(){
  6. sendCommand("alert","你好吗?");
  7. }

然后在你的html里调用这个js方法 如:

[javascript] 
  1. "button" value="Click me!" onclick="clickLink()" />

最后我们在UIWebVew中截获这个重定向请求:

 
  1. #pragma mark --
  2. #pragma mark UIWebViewDelegate
  3. - (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType {
  4. NSString *requestString = [[request URL] absoluteString];
  5. NSArray *components = [requestString componentsSeparatedByString:@":"];
  6. if ([components count] > 1 && [(NSString *)[components objectAtIndex:0] isEqualToString:@"testapp"]) {
  7. if([(NSString *)[components objectAtIndex:1] isEqualToString:@"alert"])
  8. {
  9. UIAlertView *alert = [[UIAlertView alloc]
  10. initWithTitle:@"Alert from Cocoa Touch" message:[components objectAtIndex:2]
  11. delegate:self cancelButtonTitle:nil
  12. otherButtonTitles:@"OK", nil];
  13. [alert show];
  14. }
  15. return NO;
  16. }
  17. return YES;
  18. }

不过有一个开源工程大家可以看看,它允许javascript调用objective_c的方法。叫

jsbridge-to-cocoa   http://code.google.com/p/jsbridge-to-cocoa/

还有两个相关工程

WebViewJavascriptBridge 与 GAJavaScript 值得大家慢慢研究。

其他

插入js代码

上面的功能我们可以封装到一个js函数中,将这个函数插入到页面上执行,代码如下:

if ([title compare: @"Google"]==NSOrderedSame ) { 
[webView stringByEvaluatingJavaScriptFromString:@"var script = document_createElement_x_x('script');" 
"script.type = 'text/javascript';" 
"script.text = "function myFunction() { " 
"var field = document.getElementsByName('q')[0];" 
"field.value='朱祁林';" 
"document.forms[0].submit();" 
"}";" 
"document.getElementsByTagName_r('head')[0].a(script);"]; 
[webView stringByEvaluatingJavaScriptFromString:@"myFunction();"]; 
}

看上面的代码:

a、首先通过js创建一个script的标签,type为'text/javascript'。

b、然后在这个标签中插入一段字符串,这段字符串就是一个函数:myFunction,这个函数实现google自动搜索关键字的功能。

c、然后使用stringByEvaluatingJavaScriptFromString执行myFunction函数。

1. 一般调用

将本地数据,封装,直接作为JS的返回值。如:获取软件的APPCode

//获取APPCode

NSArray *_plist_paths=NSSearchPathForDirectoriesInDomains(NSDocumentDirectory,NSUserDomainMask,YES);
 NSString *_plist_paths_path=[_plist_paths objectAtIndex:0];
 NSArray *_plist_array= [_plist_paths_path componentsSeparatedByString:@"/"];
 NSString *_appcode=[[NSString alloc]init];
 for (NSString *item in _plist_array) {
  if ([item length]==36) {
   _appcode=item;
   break;
  }
 }
 NSLog(@"current appcode:%@",_appcode);

//注入到js中
 NSMutableString *_getApkCode=[[NSMutableString alloc]init];
 [_getApkCode appendFormat:@" function  _getApkCode(){"];
 [_getApkCode appendFormat:@"return '%@';",_appcode];
 [_getApkCode appendString:@" }"];
 [self.webView stringByEvaluatingJavaScriptFromString:_getApkCode];
 [_getApkCode release];

2.需要跟平台进行交互调用

思路:

1.制造含有一定含义的请求如:(location.href="download");

2.在方法:-(BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType中,拦截:

//testMall:http://192.168.1.20:8083本地测试页面地址

NSString *pre_download=[NSString stringWithFormat:@"%@downLoad",testMall];
 if([url hasPrefix:pre_download])
 {

//下载代码。。。。

}

3. 注意事项

a.存在Iframe嵌套的页面,js注入

页面注入JS是注入到,浏览器的html中,对于内部嵌套iframe框架的页面,则无法调用到js。此时相当于调用父页面的JS。

可以通过parent+方法名,来调用你注入的JS。parent.parent的使用个数,可以是多个,不影响js的执行,如果少用parent,可能会导致,调不到你注入的JS

b.存在交互的处理方法。推荐使用方法,iphone只负责提供js接口,不调用html内部或其他的js接口

示例:

html

function addDownload()

{

url='www.XXX.XXX.zip';

download(url);//调用iphone提供的js接口

addDownloadTask_ret();//获取iphone下载接口执行的下载结果,此处调的是本地的一个延迟方法

}

//获取iphone下载接口执行的下载结果

function addDownloadTask_ret()
{

var obj=getDownloadTaskResult();//此处为iphone提供的接口,负责返回当前下载执行情况的结果if(''!=obj||undefined!=obj)

{

//调用本地的一些后续处理方法。

}

else

{

setTimeout("addDownloadTask_ret2();",1000);

}

}

-------------------------------------------------

【iOS开发】UIWebView与JavaScript(JS) 回调交互的更多相关文章

  1. UIWebView与JavaScript(JS) 回调交互 -备

    很多关于objc 与 js 交互的文章都比较适用于 mac开发,iOS的webview 还是有所不一样, 参考:http://blog.sina.com.cn/s/blog_693de6100102v ...

  2. iOS UIWebView中javascript与Objective-C交互、获取摄像头

    UIWebView是iOS开发中常用的一个视图控件,多数情况下,它被用来显示HTML格式的内容. 支持的文档格式 除了HTML以外,UIWebView还支持iWork, Office等文档格式: Ex ...

  3. iOS 利用UIWebView与JavaScript交互的最简单办法

    这里说的是针对iOS的!并且方法很简单!!并且验证可行的!!! 1, UIWebView调用 JavaScript 的函数: NSString* strValue = [webView stringB ...

  4. UIWebView中javascript与Objective-C交互、获取摄像头

    UIWebView是iOS开发中常用的一个视图控件,多数情况下,它被用来显示HTML格式的内容. 支持的文档格式 除了HTML以外,UIWebView还支持iWork, Office等文档格式: Ex ...

  5. ios开发:如何用js调用ios

    本文转载至 :http://blog.chinaunix.net/uid-29415710-id-4058564.html - (BOOL)webView:(UIWebView *)webView s ...

  6. iOS开发 UIWebView+JavaScript 交互总结

    算是个人项目经验的,印象比较深的Web+JS交互的使用 iOS原生应用与Web页面元素交互方式有很多,JavaScriptCore.拦截协议.第三方框架WebViewJavaScriptBridge. ...

  7. 在Ios里UIWebView参入js

    //修改图片大小适应webView宽高度            [webView stringByEvaluatingJavaScriptFromString:       @"var sc ...

  8. .Net魔法堂:史上最全的ActiveX开发教程——ActiveX与JS间交互篇

    一.前言 经过上几篇的学习,现在我们已经掌握了ActiveX的整个开发过程,但要发挥ActiveX的真正威力,必须依靠JS.下面一起来学习吧! 二.JS调用ActiveX方法 只需在UserContr ...

  9. iOS开发-UIWebView加载本地和网络数据

    UIWebView是内置的浏览器控件,可以用它来浏览网页.打开文档,关于浏览网页榜样可以参考UC,手机必备浏览器,至于文档浏览的手机很多图书阅读软件,UIWebView是一个混合体,具体的功能控件内置 ...

随机推荐

  1. Lua面向对象设计

    首先对于Lua语言,它没有打算被用来进行大型的程序设计,相反,Lua目标定于小型到中型的程序设计,通常是作为大型系统的一部分,所以它只提供了一套精简的元素,很多高级语言的概念都没有.这样Lua就成为了 ...

  2. 将PDM文件导出成CHM帮助文件

    实际开发中常常将维护数据库字段用 powerdesigner 维护起来,而实际要查阅 数据库用了什么字段是,常常又不方便去打开PDM 文件去找. 下面分享一个小工具,效果看下图: 有这个小工具, PD ...

  3. 爬虫技术 -- 基础学习(四)HtmlParser基本认识

    利用爬虫技术获取网页源代码后,针对网页抽取出它的特定文本内容,利用正则表达式和抽取工具,能够更好地抽取这些内容. 下面介绍一种抽取工具 -- HtmlParser HtmlParser是一个用来解析H ...

  4. 【JVM学习笔记一】JVM内存分布

    Overview 学习JVM首先需要了解一下JVM管理的内存是如何分布的,在看了<深入理解Java虚拟机>和一些博文之后,我准备自己记录一下学习的过程. 下图是JVM中运行时数据区的大致示 ...

  5. 在eclipse中配置python插件

    最好离线下载python的离线包.名字叫——org.python.pydev.feature-1.6.3.2010100513 此包里面有两个文件夹 features 和 plugins,分别把2包中 ...

  6. onhashchange事件,只需要修改hash值即可响应onhashchange事件中的函数(适用于上一题下一题和跳转页面等功能)

    使用实例: 使用onhashchange事件做一个简单的上一页下一页功能,并且当刷新页面时停留在当前页 html: <!DOCTYPE html><html><body& ...

  7. Android学习笔记之ConnectivityManager+NetWorkInfo

    PS:眼看就要开学了,该收收心了. 学习内容: 1.ConnecivityManager 2.NetWorkInfo   ConnectivityManger:网络连接管理者,用于管理Android设 ...

  8. Linux文件打包压缩、解压缩、备份命令使用方法(转载)

    对于刚刚接触Linux的人来说,一定会给Linux下一大堆各式各样的文件名给搞晕.别个不说,单单就压缩文件为例,我们知道在Windows下最常见的压缩文件就只有两种,一是,zip,另一个是.rar.可 ...

  9. spring日记------部署环境、写demo

    一.安装jdk1.7 祥见http://zhinan.sogou.com/guide/detail/?id=1610006590 二.创建web项目 略 三.配置ssm环境 3.1添加spring.m ...

  10. 重构第15天 移除重复的代码(Remove Duplication)

    理解:移除重复的代码,顾名思义就是把多处重复的代码搬移到一个公共的地方,来减少代码量,提高代码可维护性. 详解:看下面的例子就很容易理解 重构前code using System; using Sys ...