Asynchronous JavaScript calls from Android WebView

2020-03-15 01:30发布

I'm building hybrid Android app with WebView that communicates with the device with JavaScriptInterface annotation

From the WebView:

webView.addJavascriptInterface(someService, "someService");

The service implementation:

@JavascriptInterface
public void someMethod() {
    //do some business logic..
}

Problem is that from the JavaScript I run it like this:

function callSomeMethod() {
    someService.someMethod()
};

This call is synchronous, and would like something that will run asynchronously like:

function callSomeMethod(callback) {
    someService.someMethod(function(result) {
        if (result == 'success')
            callback();
    })
};

Preferably using promise:

function callSomeMethod() {
    return someService.someMethod()
    //someMethod returns promise
};

Does Android WebView has built in support for running JavaScript code asynchronously?

1条回答
爱情/是我丢掉的垃圾
2楼-- · 2020-03-15 01:54

That solely depends on you. You just need to return immediately from the injected method, but be able to call JS code when the execution is complete. Something like this (note that it's only a rough sketch):

private WebView mWebView;
private final Object mLock = new Object();
private String mJsCallbackCode;

@JavascriptInterface
public void someMethod(String jsCallbackCode) {
    synchronized (mLock) {
        mJsCallbackCode = jsCallbackCode;
    }
    // Start some business logic asynchronously, and return back here immediately.
    return;
}

public void onBusinessLogicCompleted(bool success) {
    String jsCallbackCode;
    synchronized (mLock) {
        jsCallbackCode = mJsCallbackCode;
    }
    mWebView.loadUrl("javascript:" + jsCallbackCode + "(" + success + ");void(0);");
}

And in JavaScript you use it like this:

function callSomeMethod(callback) {
    window._someMethodCallback = callback;
    someService.someMethod(
        '(function(success){' +
        '    if (success) window._someMethodCallback();' +
        '    delete window._someMethodCallback;' +
        '})'
    );
};    

So the idea is that you pass the JS code you need to be called back as a string (because you can't pass a real JS object). This code will be called in the global context.

Locking in Java is needed because methods called from JS run on a dedicated thread, not on your app's UI thread.

Note that in M preview, an API for postMessage has been added to WebView, enabling to post asynchronous messages between Java and JS code.

查看更多
登录 后发表回答