วิธีการ WebView บนข้อผิดพลาดเธรดเดียวกัน


88

ฉันมีโปรแกรม Android (Java + html ใน webview) ฉันสามารถเรียกจากจาวาสคริปต์ไปยังรหัส Java แต่ในทางกลับกันหยุดทำงาน (หลังจากอัปเดตใน eclipse)

นี่คือสิ่งที่ฉันพยายามทำ

  • สร้าง Webview (ใช้งานได้)
  • โทรในจาวาสคริปต์ไปยัง AndroidFunction.test (); (ทำงาน)
  • java test () เรียกใช้ฟังก์ชัน webView.loadUrl ("javascript: helloBack ()"); (! ไม่ทำงานอีกต่อไป)

ฉันพยายามปล่อยให้มันทำงานกับ WebView ใน MainActivity แต่มันไม่ได้ผล

MainActivity.java

public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        final WebView webView = (WebView)findViewById(R.id.webView);
        webView.getSettings().setJavaScriptEnabled(true);
        webView.setWebChromeClient(new WebChromeClient());
        WebSettings webSettings = webView.getSettings();
        webSettings.setJavaScriptEnabled(true);

        javascr = new Javascript(this, webView);
        webView.addJavascriptInterface(javascr, "AndroidFunction");
        webView.loadUrl("file:///android_asset/www/index.html");

        ....
}

Javascript.java

public class Javascript {   
    Context cont;
    WebView webView;

    Javascript(Context c, WebView w) {
        cont = c;
        webView = w;
    }

    // function called in the javascript by AndroidFunction.test();
    public void test() {
          // Breaking point!!!
        webView.loadUrl("javascript:helloBack()");
    }

ข้อผิดพลาด:

03-24 11:47:50.103: W/WebView(21026):   at com.android.org.chromium.base.SystemMessageHandler.handleMessage(SystemMessageHandler.java:27)
03-24 11:47:50.103: W/WebView(21026):   java.lang.Throwable: A WebView method was called on thread 'JavaBridge'. All WebView methods must be called on the same thread. (Expected Looper Looper{41ab68f8} called on Looper{41bb70a8}, FYI main Looper is Looper{41ab68f8})

03-24 11:47:50.103: W/WebView(21026):   at android.webkit.WebView.checkThread(WebView.java:2063)
03-24 11:47:50.103: W/WebView(21026):   at android.webkit.WebView.loadUrl(WebView.java:794)
03-24 11:47:50.103: W/WebView(21026):   at com.example.hellobt.Javascript.test(Javascript.java:24)

03-24 11:47:50.103: W/WebView(21026):   at com.android.org.chromium.base.SystemMessageHandler.nativeDoRunLoopOnce(Native Method)
03-24 11:47:50.103: W/WebView(21026):   at com.android.org.chromium.base.SystemMessageHandler.handleMessage(SystemMessageHandler.java:27)
03-24 11:47:50.103: W/WebView(21026):   at android.os.Handler.dispatchMessage(Handler.java:102)

03-24 11:47:50.103: W/WebView(21026):   at android.os.Looper.loop(Looper.java:137)
03-24 11:47:50.103: W/WebView(21026):   at android.os.HandlerThread.run(HandlerThread.java:61)

ขอบคุณสำหรับคำตอบ. ฉันแก้ไขฟังก์ชันในไฟล์ Javascript ของฉันดังนี้:

private void test(final String s) {
        webView.post(new Runnable() {
            public void run() {
                webView.loadUrl("javascript:" + s + ";");
            }
        });
        System.out.println("javscript done..");
    }

ในการแก้ไขของคุณคุณระบุว่าคุณได้เปลี่ยนรหัสของคุณตาม (หนึ่งใน) คำตอบทำให้ดูเหมือนว่าจะยังไม่สามารถแก้ปัญหาของคุณได้ - แต่มีคำตอบที่ยอมรับได้ ฉันขอแนะนำให้คุณระบุอย่างชัดเจนว่าวิธีนี้ช่วยแก้ปัญหาได้หรือไม่ (หรือเพียงแค่เลิกทำการแก้ไข)
ทอม

คำตอบ:


214

เมธอด JavaScript ถูกเรียกใช้บนเธรดพื้นหลัง (เช่นไม่ใช่ UI) คุณต้องเรียกใช้วิธีที่เกี่ยวข้องกับ Android View ทั้งหมดในเธรด UI คุณสามารถบรรลุสิ่งที่คุณต้องการด้วย:

mWebView.post(new Runnable() {
    @Override
    public void run() {
        mWebView.loadUrl(...).
    }
});

ซึ่งจะโพสต์งานเพื่อรันบนเธรด UI


คุณเพิ่มข้อมูลโค้ด (mWebView) ลงในโค้ดเดิมของคุณที่ไหน
Bowie

ในฟังก์ชัน test () โปรดดูคำถามข้างต้นของฉัน
Johan Hoeksma

จะเกิดอะไรขึ้นหากคุณต้องการคืนค่า เช่นกลับ webview.getUrl? stackoverflow.com/questions/29748782/… @JohanHoeksma
Suresh Subedi

คุณรู้หรือไม่ว่าปัญหานี้เกิดขึ้นได้อย่างไรและเมื่อใด ใน Nexus 5X (Android 7.1.1) ของฉันใช้ได้ แต่อุปกรณ์บางอย่างขัดข้อง
Ellie Zou

จะส่งตัวแปรได้อย่างไร? ฉันพยายามเพิ่มตัวแปรใน mWebView.loadUrl ("javascript: test ('" + MyData + "');"); แต่ค่าไม่เคยส่งผ่านไปยังจาวาสคริปต์!
user1788736

15

ในกรณีของฉันไม่มีอะไรปรากฏใน WebView ดังนั้นฉันจึงชอบวิธีอื่น:

runOnUiThread(new Runnable() {
    @Override
    public void run() {
        final WebView webView = (WebView) findViewById(R.id.map);
        webView.loadDataWithBaseURL(...);
    }
});

3

สิ่งนี้สามารถสรุปได้โดยใช้วิธีการโพสต์ กรุณากรอกรหัสด้านล่าง

 m_targetView.post(new Runnable() {
                        @Override
                        public void run() {
                            m_targetView.loadUrl(".....");
                        }
                    });

3

Javaรุ่น: คุณต้องใช้Runnableอินเตอร์เฟซและโพสต์ไปจัดการ

webView.post(new Runnable() {
          @Override
          public void run() {
             webView.loadUrl("file:///android_asset/www/index.html");
          }
       });

Kotlinเวอร์ชัน:

webView.post(new Runnable {
   webView.loadUrl("file:///android_asset/www/index.html")
})
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.