我一直在开发一个概念验证应用程序,该应用程序利用 WebKit 在 Objective C (iOS 7) 和 JavaScript 之间进行双向通信JavaScript核心框架。我终于能够让它按预期工作,但遇到了 UIWebView 丢失对我通过 JSContext 创建的 iOS 对象的引用的情况。
该应用程序有点复杂,以下是基础知识:
- 我正在 iOS 设备上运行 Web 服务器 (CocoaHTTPServer)
- UIWebView 最初加载远程 URL,然后重定向回
localhost
作为应用程序流程的一部分(想想 OAuth)
- 应用程序托管的 HTML 页面(位于本地主机)具有应该与我的 iOS 代码对话的 JavaScript
这是 iOS 端,我的 ViewController 的.h
:
#import <UIKit/UIKit.h>
#import <JavaScriptCore/JavaScriptCore.h>
// These methods will be exposed to JS
@protocol DemoJSExports <JSExport>
-(void)jsLog:(NSString*)msg;
@end
@interface Demo : UIViewController <UserInfoJSExports, UIWebViewDelegate>
@property (nonatomic, readwrite, strong) JSContext *js;
@property (strong, nonatomic) IBOutlet UIWebView *webView;
@end
以及 ViewController 的相关部分.m
:
-(void)viewDidLoad {
[super viewDidLoad];
// Retrieve and initialize our JS context
NSLog(@"Initializing JavaScript context");
self.js = [self.webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
// Provide an object for JS to access our exported methods by
self.js[@"ios"] = self;
// Additional UIWebView setup done here...
}
// Allow JavaScript to log to the Xcode console
-(void)jsLog(str) {
NSLog(@"JavaScript: %@", str);
}
这是(为了这个问题而简化的)HTML/JS 方面:
<html>
<head>
<title>Demo</title>
<script type="text/javascript">
function setContent(c, noLog){
with(document){
open();
write('<p>' + c + '</p>');
close();
}
// Write content to Xcode console
noLog || ios.jsLog(c);
}
</script>
</head>
<body onload="javascript:setContent('ios is: ' + typeof ios)">
</body>
</html>
现在,在几乎所有情况下,这都效果很好,我明白了ios is: object
在 UIWebView 和 Xcode 控制台中。很酷。但在一个特定的场景中,100% 的情况下,在 UIWebView 中进行一定数量的重定向后会失败,一旦上面的页面最终加载,它就会显示:
ios 是:未定义
...其余的 JS 逻辑退出,因为后续调用ios.jsLog
in the setContent
函数导致未定义对象异常。
最后我的问题是:什么可能/可能导致 JSContext 丢失?我深入研究了 JavaScriptCore 的 .h 文件中的“文档”,发现发生这种情况的唯一方法是如果没有更多strong
参考文献JSContext
,但就我而言,我有自己的一个,所以这似乎不对。
我唯一的另一个假设是,这与我获取信息的方式有关。JSContext
参考:
self.js = [self.webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
我知道这未必得到苹果的官方支持,尽管我确实发现至少有一个 SO'er 说他有一个苹果批准的应用程序使用了这种方法。
EDIT
我应该提到,我实施了UIWebViewDelegate
在 UIWebView 中每次重定向后检查 JSContext:
-(void)webViewDidFinishLoad:(UIWebView *)view{
// Write to Xcode console via our JSContent - is it still valid?
[self.js evaluateScript:@"ios.jsLog('Can see JS from obj c');"];
}
这适用于所有情况,即使我的网页最终加载并报告ios is: undefined
上述方法同时写入Can see JS from obj c
到 Xcode 控制台。这似乎表明 JSContext 仍然有效,并且由于某种原因它不再从 JS 中可见。
对于这个冗长的问题,我深表歉意,这方面的文档太少了,我认为我能提供的信息越多越好。