Android WebView中的addJavascriptInterface方法在WebChromeClient#onCreateWindow回调中创建的WebView中无法使用。

16
以下是我的测试代码。我的问题是在第二个页面中我无法引用AndroidFunction2。我正在Nexus 7上测试,使用的是Android 4.4。但在使用Android 4.0的sumsang i9100上是可以的。 我做错了什么吗?还是这是Android的一个bug? MainActivity
public class MainActivity extends Activity {
    WebView mWebView1;
    WebView mWebView2;

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

        final FrameLayout mainFrame = (FrameLayout) this.findViewById(R.id.mainFrame);

        mWebView1 = new WebView(this);
        mWebView1.getSettings().setJavaScriptEnabled(true);
        mWebView1.getSettings().setSupportMultipleWindows(true);
        mWebView1.setWebViewClient(new WebViewClient() {
            @Override
            public boolean shouldOverrideUrlLoading(WebView view, String url) {
                return false;
            }
        });
        mWebView1.setWebChromeClient(new WebChromeClient() {
            @Override
            public boolean onCreateWindow(WebView view, boolean isDialog,
                    boolean isUserGesture, Message resultMsg) {
                mWebView2 = new WebView(MainActivity.this);
                mWebView2.getSettings().setJavaScriptEnabled(true);
                mWebView2.getSettings().setSupportMultipleWindows(true);
                mWebView2.setWebChromeClient(new WebChromeClient() {
                    @Override
                    public void onConsoleMessage(String message, int lineNumber, String sourceID) {
                        Log.d("WebView", "Line: " + lineNumber + ", " + message);
                    }
                });
                mWebView2.addJavascriptInterface(new Object() {
                    @JavascriptInterface
                    public void hello2() {
                    }
                }, "AndroidFunction2");

                (( WebViewTransport )resultMsg.obj).setWebView(mWebView2);
                resultMsg.sendToTarget();
                mainFrame.addView(mWebView2);
                return true;
            }
        });
        mWebView1.addJavascriptInterface(new Object() {
            @JavascriptInterface
            public void hello1() {
            }
        }, "AndroidFunction1");
        mWebView1.loadUrl("file:///sdcard/test_1.html");

        mainFrame.addView(mWebView1);
    }
}

这两个网页是

test_1.html:

<html>
<body>
    <a href="test_2.html" target="_blank">goto test 2</a>
    <div><a href="javascript:alert(typeof AndroidFunction1);"> alert(typeof AndroidFunction1);</a> </div>
    <div><a href="javascript:alert(typeof window.AndroidFunction1);"> alert(typeof window.AndroidFunction1);</a> </div>
</body>
</html>

test_2.html

<html>
<body>
    <div><a href="javascript:alert(AndroidFunction2);"> alert(AndroidFunction2);</a> </div>
    <div><a href="javascript:alert(typeof window.AndroidFunction2);"> alert(typeof window.AndroidFunction2);</a> </div>
</body>
</html>

1
我遇到了同样的问题,你最终解决了吗? - casolorz
@mntgoat 抱歉,我还没有解决。这个问题在我的项目中仍然存在。 - zhang
@张,你找到处理内部“AndroidFunction2”接口的方法了吗? - Anukool srivastav
1个回答

51

我也遇到了同样的问题,在重新查看文档后,我发现如果您将targetSdkVersion设置为17或更高版本,则必须向您想要在JavaScript中使用的任何方法添加@JavascriptInterface注释(该方法还必须是公共的)。

如果您不提供此注释,则当在Android 4.2或更高版本上运行时,该方法将无法被您的网页访问。

以下示例(摘自http://developer.android.com/guide/webapps/webview.html)显示您可以在Android应用程序中包含以下类:

public class WebAppInterface {
    Context mContext;

    /** Instantiate the interface and set the context */
    WebAppInterface(Context c) {
        mContext = c;
    }

    @JavascriptInterface   // must be added for API 17 or higher
    public void showToast(String toast) {
        Toast.makeText(mContext, toast, Toast.LENGTH_SHORT).show();
    }
}

并且绑定为

WebView webView = (WebView) findViewById(R.id.webview);
webView.addJavascriptInterface(new WebAppInterface(this), "Android");

然后在 WebView 中的 HTML 代码中调用:

<input type="button" value="Say hello" onClick="showAndroidToast('Hello Android!')" />

<script type="text/javascript">
    function showAndroidToast(toast) {
        Android.showToast(toast);
    }
</script>

5
将来,请勿未经明确标注引用就复制他人内容,这被视为剽窃。请参见 http://stackoverflow.com/help/referencing。 - Matt
4
@Matt,我在第一行提到了“查阅文档”,这意味着我的答案是从官方文档中得出的。我并不反对你的观点,但我也没有抄袭内容的意图。 - Ram Patra
2
@Ramswaroop:我承认有一些努力是为了引用文档,但在数据被复制的情况下(就像示例中一样),我们真的需要更清晰的归属。 - Matt
请注意,如果您正在使用Java接口:注释接口方法并不重要;您必须注释方法的实现。 - ITJscott
4
这只适用于主 WebView,而不适用于通过 onCreateWindow 回调方法创建的 WebView。已尝试添加 @JavascriptInterface。 - Anukool srivastav
3
这个解决方案似乎不适用于从onCreateWindow方法创建的WebView。我想知道这怎么成为一个被接受的答案!!! - nagu

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接