Как получить возвращаемое значение из javascript в WebView Android?

83

Я хочу получить возвращаемое значение из Javascript в Android. Я могу сделать это с iPhone, но не с Android. Я использовал loadUrl, но он вернул void вместо объекта. Кто-нибудь может мне помочь?

Куонг Та
источник

Ответы:

79

То же, Keithно более короткий ответ

webView.addJavascriptInterface(this, "android");
webView.loadUrl("javascript:android.onData(functionThatReturnsSomething)");

И реализуем функцию

@JavascriptInterface
public void onData(String value) {
   //.. do something with the data
}

Не забудьте удалить onDataиз proguardсписка (если у вас включен прогард)

Кирилл Кулаков
источник
1
можешь уточнить со proguardстороны?
eugene
@eugene, если вы не знаете, что это такое, вы, вероятно, не используете его. В любом случае это то, что нужно включить, что усложняет обратную разработку вашего приложения
Кирилл Кулаков
1
Благодарю. Я знаю, что в корне проекта есть proguard-project.txt, но не более того. Я спрашиваю, потому что ваше решение работает, но я получаю SIGSEGV. Я пытаюсь document.getElementById("my-div").scrollTopполучить позицию прокрутки SwipeRefreshLayout::canChildScrollUp(). Я подозреваю, что это может быть из-за того, что звонок звонят слишком много раз за короткий период времени. или proguardчто я подозреваю, потому что я просто не знаю, что это ..
Евгений
Это должен быть принятый ответ, а не все комментируемые здесь хаки
Ромит Кумар
64

Вот как это сделать:

Добавьте этого клиента в свой WebView:

final class MyWebChromeClient extends WebChromeClient {
        @Override
        public boolean onJsAlert(WebView view, String url, String message, JsResult result) {
            Log.d("LogTag", message);
            result.confirm();
            return true;
        }
    }

Теперь в вашем вызове javascript выполните:

webView.loadUrl("javascript:alert(functionThatReturnsSomething)");

Теперь в вызове onJsAlert "сообщение" будет содержать возвращаемое значение.

Феликс Хазин
источник
17
Это чрезвычайно ценный прием; особенно когда вы создаете приложение, в котором у вас нет контроля над javascript и вам приходится обходиться существующими функциями. Обратите внимание, что вы можете получить тот же результат без перехвата предупреждений JS, используя public boolean onConsoleMessage(ConsoleMessage consoleMessage)вместо, onJsAlertа затем в вызове javascript, который вы делаете webView.loadUrl("javascript:console.log(functionThatReturnsSomething)");- таким образом вы оставляете предупреждения JS в покое.
Mick Byrne
Решение WebChromeClient.onConsoleMessageкажется более чистым, но имейте в виду, что оно не работает с некоторыми устройствами HTC. См. Этот вопрос для получения дополнительной информации.
Петр
3
По какой причине использование addJavascriptInterface не было бы предпочтительнее?
Кит
@Keith: вам понадобится объект, в который javascript записывает результаты, но поскольку вы не можете коснуться существующего javascript в этом вопросе, а существующий js не записывает в такой объект, метод интерфейса js здесь не помогает ( вот как я это понял).
Ray
1
@RayKoopa Не обязательно верно. Хотя вы не сможете редактировать существующий javascript на целевой веб-странице, вы все равно можете вызывать методы для переданного объекта mWebView.addJavascriptInterface(YourObject mYourObject,String yourNameForObject);, вызывая функциюmWebView.loadUrl("javascript:yourNameForObject.yourSetterMethod(valueToSet)")
Крис - младший,
21

Используйте addJavascriptInterface()для добавления объекта Java в среду Javascript. Пусть ваш Javascript вызовет метод этого объекта Java для предоставления его «возвращаемого значения».

CommonsWare
источник
как я могу сделать это в такой ситуации, как stackoverflow.com/questions/5521191/…
vnshetty
Это не поможет, если ваш js не записывается в этот объект, и вы не можете прикоснуться к js.
Ray
7
@PacMani: используйте loadUrl("javascript:...")(API Level 1-18) или evaluateJavascript()(API Level 19+), чтобы оценить свой собственный JavaScript в контексте загруженной в данный момент веб-страницы.
CommonsWare
@CommonsWare: Спасибо, я понял;) однако менеджер решил, что js можно изменить вместо «злоупотребления предупреждениями javascript» (eyeroll), поэтому я получил это с помощью метода интерфейса.
Ray
12

Вот что я придумал сегодня. Это потокобезопасный, достаточно эффективный и позволяет синхронное выполнение Javascript из Java для Android WebView .

Работает с Android 2.2 и выше. (Требуется commons-lang, потому что мне нужны мои фрагменты кода, переданные в eval () в виде строки Javascript. Вы можете удалить эту зависимость, заключив код не в кавычки, а в function(){})

Сначала добавьте это в свой файл Javascript:

function evalJsForAndroid(evalJs_index, jsString) {
    var evalJs_result = "";
    try {
        evalJs_result = ""+eval(jsString);
    } catch (e) {
        console.log(e);
    }
    androidInterface.processReturnValue(evalJs_index, evalJs_result);
}

Затем добавьте это в свою деятельность Android:

private Handler handler = new Handler();
private final AtomicInteger evalJsIndex = new AtomicInteger(0);
private final Map<Integer, String> jsReturnValues = new HashMap<Integer, String>();
private final Object jsReturnValueLock = new Object();
private WebView webView;

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    webView = (WebView) findViewById(R.id.webView);
    webView.addJavascriptInterface(new MyJavascriptInterface(this), "androidInterface");
}

public String evalJs(final String js) {
    final int index = evalJsIndex.incrementAndGet();
    handler.post(new Runnable() {
        public void run() {
            webView.loadUrl("javascript:evalJsForAndroid(" + index + ", " +
                                    "\"" + StringEscapeUtils.escapeEcmaScript(js) + "\")");
        }
    });
    return waitForJsReturnValue(index, 10000);
}

private String waitForJsReturnValue(int index, int waitMs) {
    long start = System.currentTimeMillis();

    while (true) {
        long elapsed = System.currentTimeMillis() - start;
        if (elapsed > waitMs)
            break;
        synchronized (jsReturnValueLock) {
            String value = jsReturnValues.remove(index);
            if (value != null)
                return value;

            long toWait = waitMs - (System.currentTimeMillis() - start);
            if (toWait > 0)
                try {
                    jsReturnValueLock.wait(toWait);
                } catch (InterruptedException e) {
                    break;
                }
            else
                break;
        }
    }
    Log.e("MyActivity", "Giving up; waited " + (waitMs/1000) + "sec for return value " + index);
    return "";
}

private void processJsReturnValue(int index, String value) {
    synchronized (jsReturnValueLock) {
        jsReturnValues.put(index, value);
        jsReturnValueLock.notifyAll();
    }
}

private static class MyJavascriptInterface {
    private MyActivity activity;

    public MyJavascriptInterface(MyActivity activity) {
        this.activity = activity;
    }

    // this annotation is required in Jelly Bean and later:
    @JavascriptInterface
    public void processReturnValue(int index, String value) {
        activity.processJsReturnValue(index, value);
    }
}
Кит
источник
2
Спасибо за приведенный выше код - я применил его в своем проекте Android. Однако в этом есть ловушка из-за плохого дизайна Java API. Строка: jsReturnValueLock.wait (waitMs - (System.currentTimeMillis () - start)); Иногда может случиться, что значение аргумента функции wait () будет равно 0. А это означает «ждать бесконечно» - я получал случайные ошибки «нет ответа». Я решил это, проверив: if (elapsed> = waitMS) break; и не вызывать снова System.currentTimeMillis () ниже, но с использованием прошедшего значения. Лучше всего отредактировать и исправить приведенный выше код.
gregko
Вау, большое спасибо! Я тоже видел это зависание и никогда не понимал, откуда это.
Кейт
1
Но если я запустил evalJS в слушателе button.click и т. Д. waitForJsReturnValue может привести к зависанию evalJs (). Таким образом, webView.loadUrl () внутри handler.post () может не иметь возможности выполнить, потому что прослушиватель button.click не вернулся.
victorwoo
Но как обернуть код не в кавычки, а в function(){}? Как разместить jsString {}?
victorwoo
7

На API 19+, лучший способ сделать это , чтобы позвонить evaluateJavascript на вашем WebView:

webView.evaluateJavascript("foo.bar()", new ValueCallback<String>() {
    @Override public void onReceiveValue(String value) {
        // value is the result returned by the Javascript as JSON
    }
});

Связанный ответ с более подробной информацией: https://stackoverflow.com/a/20377857

Дэвид
источник
5

Решение, предложенное @Felix Khazin, работает, но отсутствует один ключевой момент.

Вызов javascript должен выполняться после WebViewзагрузки веб-страницы в . Добавьте это WebViewClientк WebView, наряду с WebChromeClient.

Полный пример:

@Override
public void onCreate(Bundle savedInstanceState) {
    ...
    WebView webView = (WebView) findViewById(R.id.web_view);
    webView.getSettings().setJavaScriptEnabled(true);
    webView.setWebViewClient(new MyWebViewClient());
    webView.setWebChromeClient(new MyWebChromeClient());
    webView.loadUrl("http://example.com");
}

private class MyWebViewClient extends WebViewClient {
    @Override
    public void onPageFinished (WebView view, String url){
        view.loadUrl("javascript:alert(functionThatReturnsSomething())");
    }
    @Override
    public boolean shouldOverrideUrlLoading(WebView view, String url) {
        return false;
    }
}

private class MyWebChromeClient extends WebChromeClient {
@Override
public boolean onJsAlert(WebView view, String url, String message, JsResult result) {
    Log.d("LogTag", message);
        result.confirm();
        return true;
    }
}
Митко Кацев
источник
1

В качестве альтернативного варианта, использующего индивидуальную схему:

Основное занятие

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            WebView.setWebContentsDebuggingEnabled(true);
        }

        final WebView webview = new CustomWebView(this);

        setContentView(webview);

        webview.loadUrl("file:///android_asset/customPage.html");

        webview.postDelayed(new Runnable() {
            @Override
            public void run() {

                //Android -> JS
                webview.loadUrl("javascript:showToast()");
            }
        }, 1000);
    }
}

CustomWebView

public class CustomWebView extends WebView {
    public CustomWebView(Context context) {
        super(context);

        setup();
    }

    @SuppressLint("SetJavaScriptEnabled")
    private void setup() {
        setWebViewClient(new AdWebViewClient());

        getSettings().setJavaScriptEnabled(true);

    }

    private class AdWebViewClient extends WebViewClient {

        @Override
        public boolean shouldOverrideUrlLoading(WebView view, String url) {

            if (url.startsWith("customschema://")) {
                //parse uri
                Toast.makeText(CustomWebView.this.getContext(), "event was received", Toast.LENGTH_SHORT).show();

                return true;
            }


            return false;
        }

    }
}

customPage.html (находится в свернутых активах)

<!DOCTYPE html>
<html>
<head>
    <title>JavaScript View</title>

    <script type="text/javascript">

        <!--JS -> Android-->
        function showToast() {
            window.location = "customschema://goto/";
        }

    </script>
</head>

<body>

</body>
</html>
yoAlex5
источник
0

Сделать это можно так:

[Activity(Label = "@string/app_name", Theme = "@style/AppTheme", MainLauncher = true)]
public class MainActivity : AppCompatActivity
{
    public WebView web_view;
    public static TextView textView;

    protected override void OnCreate(Bundle savedInstanceState)
    {
        base.OnCreate(savedInstanceState);
        Xamarin.Essentials.Platform.Init(this, savedInstanceState);

        Window.AddFlags(WindowManagerFlags.Fullscreen);
        Window.ClearFlags(WindowManagerFlags.ForceNotFullscreen);

        // Set our view from the "main" layout resource
        SetContentView (Resource.Layout.main);

        web_view = FindViewById<WebView>(Resource.Id.webView);
        textView = FindViewById<TextView>(Resource.Id.textView);

        web_view.Settings.JavaScriptEnabled = true;
        web_view.SetWebViewClient(new SMOSWebViewClient());
        web_view.LoadUrl("https://stns.egyptair.com");
    }

    public override void OnRequestPermissionsResult(int requestCode, string[] permissions, [GeneratedEnum] Android.Content.PM.Permission[] grantResults)
    {
        Xamarin.Essentials.Platform.OnRequestPermissionsResult(requestCode, permissions, grantResults);

        base.OnRequestPermissionsResult(requestCode, permissions, grantResults);
    }
}

public class SMOSWebViewClient : WebViewClient
{
    public override bool ShouldOverrideUrlLoading(WebView view, IWebResourceRequest request)
    {
        view.LoadUrl(request.Url.ToString());
        return false;
    }

    public override void OnPageFinished(WebView view, string url)
    {
        view.EvaluateJavascript("document.getElementsByClassName('notf')[0].innerHTML;", new JavascriptResult());
    }
}

public class JavascriptResult : Java.Lang.Object, IValueCallback
{
    public string Result;

    public void OnReceiveValue(Java.Lang.Object result)
    {
        string json = ((Java.Lang.String)result).ToString();
        Result = json;
        MainActivity.textView.Text = Result.Replace("\"", string.Empty);
    }
}
Шериф Риад
источник