2014-09-11 14 views
18

W UIWebView, mogę JSContext poprzez:Jak zdobyć JSContext z WKWebView

[webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"] 

W ten sam sposób nie działa w WKWebView i awarii aplikacji, kiedy nadejdzie ten wiersz kodu.

Czy istnieje sposób na uzyskanie JSContext w WKWebView?

z góry dzięki.

+0

czy znalazłeś jakieś rozwiązanie dla JS ca llbacki? –

Odpowiedz

8

Nie można uzyskać kontekstu, ponieważ układ i javascript są obsługiwane w innym procesie.

Zamiast tego dodaj skrypty do konfiguracji WebView i ustaw kontroler widoku (lub inny obiekt) jako program obsługi komunikatów skryptowych.

Teraz, wysyłać wiadomości z JavaScript tak:

window.webkit.messageHandlers.interOp.postMessage(message) 

Skrypt wiadomość obsługi otrzyma zwrotnego:

- (void)userContentController:(WKUserContentController *)userContentController 
          didReceiveScriptMessage:(WKScriptMessage *)message{ 
    NSLog(@"%@", message.body); 
} 
+0

Tak, użyłem już tej metody, chociaż powiązanie obiektu macierzystego z kontekstem js jest wygodniejsze w obsłudze wywołania zwrotnego js. Mogę wywołać wywołanie zwrotne js poprzez evaluationJavaScript – Xiong

+0

Upewnij się, aby otworzyć raport o błędzie. Apple może zrobić trochę magii w środku i nadal zapewnia funkcjonalność. Ponieważ projekt jest open source, ktoś może to zrobić i wysłać żądanie ściągnięcia. –

+0

Wadą tego jest, że nie ma sposobu na uzyskanie informacji o błędach JavaScript, które występują podczas wykonywania. –

0

XWebView to dobre rozwiązanie dla Twojego problemu. Może ustanowić pomost między natywnym a JS i zapewnić wiążące stylowe API.

+0

Wygląda naprawdę fajnie, ale tylko iOS, tylko fyi ... – shmim

+0

Ale to nie daje dostępu do Javascriptcore – NaveenKumar

+0

Rzeczywiście, niemożliwe jest uzyskanie dostępu do JSContext z WKWebView, ponieważ jest on w procesie renderowania zamiast procesu App. – soflare

1

konfigurowania wkwebview tak i dodać Odpowiednio obsługi i Umieść wiadomość ze skryptu w podobny wzór

NSString *myScriptSource = @"alert('Hello, World!')"; 


WKUserScript *s = [[WKUserScript alloc] initWithSource:myScriptSource injectionTime:WKUserScriptInjectionTimeAtDocumentStart forMainFrameOnly:YES]; 
WKUserContentController *c = [[WKUserContentController alloc] init]; 
[c addUserScript:s]; 
// Add a script message handler for receiving "buttonClicked" event notifications posted from the JS document using window.webkit.messageHandlers.buttonClicked.postMessage script message 
[c addScriptMessageHandler:self name:@"buttonClicked"]; 

WKWebViewConfiguration *conf = [[WKWebViewConfiguration alloc] init]; 
conf.userContentController = c; 

WKWebView *webview = [[WKWebView alloc] initWithFrame:self.view.bounds configuration:conf]; 
[self.view addSubview:webview]; 
webview.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight; 
// Do any additional setup after loading the view, typically from a nib. 
NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://google.com"]]; 
[webview loadRequest:request]; 

wdrożenia komunikatu scenariusza obsługi „WKScriptMessageHandler” metodą nazwy

#pragma mark -WKScriptMessageHandler 
- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message   { 
if ([message.name isEqualToString:@"buttonClicked"]) { 
self.buttonClicked ++; 
} 

// JS objects are automatically mapped to ObjC objects 
id messageBody = message.body; 
if ([messageBody isKindOfClass:[NSDictionary class]]) { 
NSString* idOfTappedButton = messageBody[@"ButtonId"]; 
[self updateColorOfButtonWithId:idOfTappedButton]; 
} 
} 

i opublikować formularz wiadomość js jak to

var button = document.getElementById("clickMeButton"); 
button.addEventListener("click", function() { 
    var messgeToPost = {'ButtonId':'clickMeButton'}; 
    window.webkit.messageHandlers.buttonClicked.postMessage(messgeToPost); 
},false);