Ответ 1
Вы можете получить JSContext из UIWebView с помощью ключевого пути:
UIWebView *webView = [[UIWebView alloc] init];
JSContext *ctx = [webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
ctx[@"document"][@"body"][@"style"][@"background"] = @"steelblue";
Apple никогда не собиралась документировать какие-либо новые API JavaScriptCore, поэтому я не уверен, что это считается внутренним/недокументированным API или нет. У меня есть одобренное приложение, которое использует этот метод.
Обновление. Еще одно альтернативное решение, предложенное в https://github.com/TomSwift/UIWebView-TS_JavaScriptContext, - это сделать категорию на NSObject
и использовать чтобы реализовать WebKit документированный ответный вызов делегата didCreateJavaScriptContext
. Чтобы перефразировать эту реализацию, вы можете просто вызвать [NSObject contextForWebView:myWebView]
, чтобы захватить JSContext для UIWebView:
@implementation NSObject(JSContextTracker)
+ (NSMapTable *)JSContextTrackerMap {
static NSMapTable *contextTracker;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
contextTracker = [NSMapTable strongToWeakObjectsMapTable];
});
return contextTracker;
}
- (void)webView:(id)unused didCreateJavaScriptContext:(JSContext *)ctx forFrame:(id)alsoUnused {
NSAssert([ctx isKindOfClass:[JSContext class]], @"bad context");
if (!ctx)
return;
NSMapTable *map = [NSObject JSContextTrackerMap];
static long contexts = 0;
NSString *contextKey = [NSString stringWithFormat:@"jsctx_%@", @(contexts++)];
[map setObject:ctx forKey:contextKey];
ctx[@"JSContextTrackerMapKey"] = contextKey; // store the key to the map in the context itself
}
+ (JSContext *)contextForWebView:(UIWebView *)webView {
// this will trigger didCreateJavaScriptContext if it hasn't already been called
NSString *contextKey = [webView stringByEvaluatingJavaScriptFromString:@"JSContextTrackerMapKey"];
JSContext *ctx = [[NSObject JSContextTrackerMap] objectForKey:contextKey];
return ctx;
}
@end