質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

87.49%

javaからローカルhtml/scriptのfunctionを実行したい

解決済

回答 1

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 1,365

score 12

 前提・実現したいこと

androidアプリをローカルhtmlをベースにwebviewで作成したいと思っています。
javascriptからjavaへのデータ受け渡し・操作は問題なくできたのですが、逆のjavaからjavascriptのfunctionが動かなくて行き詰まっています。
androidは初心者なので、基本的なところから間違っているのかもしれません。ご教示お願いします。

 発生している問題・エラーメッセージ

javascript --> Java すなわちappJsInterface.testMessageStart() --> testMessageStartは問題なく動きます。
Java --> javascript() すなわちwebView.loadUrl --> window.testMessageCallbackでアプリが落ちます。

androidStudio初心者で適切なエラーメッセージを提示できないのですが、Event Logにはエラーは出ていません。LogCatは以下のメッセージが流れっぱなしになっています。
コード

2018-12-02 18:41:31.446 1763-32080/? I/AudioFlinger: AudioFlinger's thread 0xe2383c40 tid=32080 ready to run
2018-12-02 18:41:31.447 2763-30854/com.google.android.googlequicksearchbox:search E/IAudioFlinger: createRecord returned error -12
2018-12-02 18:41:31.447 2763-30854/com.google.android.googlequicksearchbox:search E/AudioRecord: AudioFlinger could not create record track, status: -12
2018-12-02 18:41:31.447 2763-30854/com.google.android.googlequicksearchbox:search E/AudioRecord-JNI: Error creating AudioRecord instance: initialization check failed with status -12.
2018-12-02 18:41:31.447 2763-30854/com.google.android.googlequicksearchbox:search E/android.media.AudioRecord: Error code -20 when initializing native AudioRecord object.
2018-12-02 18:41:31.448 2763-30854/com.google.android.googlequicksearchbox:search I/MicrophoneInputStream: mic_started  SR : 16000 CC : 16 SO : 6
2018-12-02 18:41:31.449 2763-2763/com.google.android.googlequicksearchbox:search I/MicroDetectionWorker: onReady
2018-12-02 18:41:31.456 2763-30854/com.google.android.googlequicksearchbox:search I/MicrophoneInputStream: mic_close  SR : 16000 CC : 16 SO : 6
2018-12-02 18:41:31.458 2763-28065/com.google.android.googlequicksearchbox:search I/MicroRecognitionRunner: Detection finished
2018-12-02 18:41:31.458 2763-28065/com.google.android.googlequicksearchbox:search W/ErrorReporter: reportError [type: 211, code: 524300]: Error reading from input stream
2018-12-02 18:41:31.459 2763-3112/com.google.android.googlequicksearchbox:search I/MicroRecognitionRunner: Stopping hotword detection.
2018-12-02 18:41:31.459 2763-28065/com.google.android.googlequicksearchbox:search W/ErrorProcessor: onFatalError, processing error from engine(4)
    com.google.android.apps.gsa.shared.speech.b.g: Error reading from input stream
        at com.google.android.apps.gsa.staticplugins.microdetection.d.k.a(SourceFile:91)
        at com.google.android.apps.gsa.staticplugins.microdetection.d.l.run(Unknown Source:14)
        at com.google.android.libraries.gsa.runner.a.a.b(SourceFile:32)
        at com.google.android.libraries.gsa.runner.a.c.call(Unknown Source:4)
        at java.util.concurrent.FutureTask.run(FutureTask.java:266)
        at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:458)
        at java.util.concurrent.FutureTask.run(FutureTask.java:266)
        at com.google.android.apps.gsa.shared.util.concurrent.b.g.run(Unknown Source:4)
        at com.google.android.apps.gsa.shared.util.concurrent.b.aw.run(SourceFile:4)
        at com.google.android.apps.gsa.shared.util.concurrent.b.aw.run(SourceFile:4)
        at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1167)
        at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:641)
        at java.lang.Thread.run(Thread.java:764)
        at com.google.android.apps.gsa.shared.util.concurrent.b.i.run(SourceFile:6)
     Caused by: com.google.android.apps.gsa.shared.exception.GsaIOException: Error code: 393238 | Buffer overflow, no available space.
        at com.google.android.apps.gsa.speech.audio.Tee.j(SourceFile:103)
        at com.google.android.apps.gsa.speech.audio.au.read(SourceFile:2)
        at java.io.InputStream.read(InputStream.java:101)
        at com.google.android.apps.gsa.speech.audio.ao.run(SourceFile:17)
        at com.google.android.apps.gsa.speech.audio.an.run(SourceFile:2)
        at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:458)
        at java.util.concurrent.FutureTask.run(FutureTask.java:266) 
        at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:458) 
        at java.util.concurrent.FutureTask.run(FutureTask.java:266) 
        at com.google.android.apps.gsa.shared.util.concurrent.b.g.run(Unknown Source:4) 
        at com.google.android.apps.gsa.shared.util.concurrent.b.aw.run(SourceFile:4) 
        at com.google.android.apps.gsa.shared.util.concurrent.b.aw.run(SourceFile:4) 
        at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1167) 
        at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:641) 
        at java.lang.Thread.run(Thread.java:764) 
        at com.google.android.apps.gsa.shared.util.concurrent.b.i.run(SourceFile:6) 
コード

 該当のソースコード

Java

ソースコード

import android.app.AlertDialog;
import android.content.DialogInterface;
import android.content.pm.ActivityInfo;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.webkit.JavascriptInterface;
import android.webkit.WebView;
import android.webkit.WebViewClient;

@SuppressLint({"JavascriptInterface", "SetJavaScriptEnabled"})
public class MainActivity extends AppCompatActivity {

    private WebView webView;

    @SuppressLint("AddJavascriptInterface")
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        webView = findViewById(R.id.webView);
        webView.setWebViewClient(new WebViewClient());
        webView.getSettings().setJavaScriptEnabled(true);
        webView.getSettings().setDomStorageEnabled(true);

        MyJavaScriptInterface obj = new MyJavaScriptInterface();
        webView.addJavascriptInterface( obj, "appJsInterface" );

        webView.loadUrl("file:///android_asset/index.html");

    }

    // js -> ネイティブ
    class MyJavaScriptInterface {

        @JavascriptInterface
        public void showAlert( String title, String message ) {
            new AlertDialog.Builder(MainActivity.this)
                    .setTitle(title)
                    .setMessage(message)
                    .setPositiveButton("OK", null)
                    .show();
        }

        @JavascriptInterface
        public void testMessageStart() {  .....................①
            new AlertDialog.Builder(MainActivity.this)
                    .setTitle("resept")
                    .setMessage("JS --> ネイティブ OK")
                    .setPositiveButton("OK", new DialogInterface.OnClickListener() {
                        @Override
                        public void onClick(DialogInterface dialog, int which) {
                            // OK button pressed
                            testMessageCallback();

                        }
                    })
                    .setNegativeButton("Cancel", null)
                    .show();

        }
    }

    // ネイティブ -> js
    public void testMessageCallback() {

        //webView = findViewById(R.id.webView);
        final String script = "javascript:window.testMessageCallback('%s');";
        webView.loadUrl( String.format( script, "testMessageCallback" ) );  .......②

    }
コード

JavaScript

コード

var testMessageSendClick  = function () {

     appJsInterface.testMessageStart();


}

window.testMessageCallback = function (message) {

     appJsInterface.showAlert("infomation", "testOK");

}


```

 試したこと

javascriptからjavaへのデータ受け渡し・操作①、すなわちAlertDialogは問題なく表示されるのでうまくいっていると思われます。
javaからjavascriptへのデータ受け渡し・操作②、すなわちwebView.loadUrl( String.format( script, "testMessageCallback" ) );が動かず、アプリが落ちてしまいます。(testMessageCallback()は間違いなく呼ばれているのは確認しました)
webViewが認識されてないのかと思いtestMessageCallback()内でwebView = findViewById(R.id.webView);の有無は確認しましたが関係ありませんでした。
javascript内のwindow.testMessageCallback()が無いと判断されて落ちているのかとも思うのですが、原因・解決策がネット検索からはたどり着けませんでした。

 補足情報(FW/ツールのバージョンなど)

AndroidStudio 3.2.1
API Level 28
minSdkVersion 23
targetSdkVersion 28
確認はエミュレータ,Nexus5x API28

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

check解決した方法

0

その後もいろいろ試しましたが、うまくいきませんでした。
素人ながら下記のweb情報から、漠然とした原因ですが、セキュリティ上の観点から
JavascriptInterfaceでjavaScript->Javaに送って処理し、その結果をwebView.loadUrlでjava->javaScriptに送ることができないのかなと判断しました。

リンク内容

WebChromeClientのonJsConfirmを使ってなんとかやりたい機能を代替えできそうなので、とりあえずはそれで対応しますが、はっきりとした原因をお分かりの方は是非教えてください。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 87.49%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る