在 WebView Android 中扫描条形码

Scan Barcode Within WebView Android

本文关键字:扫描 条形码 Android WebView      更新时间:2023-09-26

我正在尝试在用户按下 html 中的按钮时启动条形码扫描仪。但是我收到一个错误:

Uncaught TypeError: Object [object Object] has no method 'scanBarcode'", source: file:///android_asset/www/index.html (49)

在javascript中调用:

function scanner(){ 
        Android.scanBarcode();
    }

现在在 Java 文件中,我有以下代码,它声明了public class JavaScriptInterface中的函数

这是我MainActivity.java文件:

public class MainActivity extends Activity {
@SuppressWarnings("deprecation")
@SuppressLint("SetJavaScriptEnabled") @Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    requestWindowFeature(Window.FEATURE_NO_TITLE);
    setContentView(R.layout.activity_main);
    WebView myWebView = (WebView)findViewById(R.id.webView);
    myWebView.addJavascriptInterface(new WebAppInterface(this), "Android");
    myWebView.setInitialScale(1);
    WebSettings webSettings = myWebView.getSettings();
    webSettings.setJavaScriptEnabled(true);
    webSettings.setBuiltInZoomControls(false);
    webSettings.setSupportZoom(true);
    webSettings.setJavaScriptEnabled(true);
    webSettings.setLoadWithOverviewMode(true);
    webSettings.setUseWideViewPort(true);
    //webView.getSettings().setJavaScriptEnabled(true);
    myWebView.setWebChromeClient(new WebChromeClient());
    //myWebView.setWebViewClient(new WebViewClient());
    myWebView.loadUrl("file:///android_asset/www/index.html");
}
public class JavaScriptInterface {
    Context mContext;
    // Instantiate the interface and set the context
    JavaScriptInterface(Context c) {
        mContext = c;
    }
    // using Javascript to call the finish activity
    public void closeMyActivity() {
        finish();
    }
    public void scanBarcode() {
        Intent intent = new Intent("com.google.zxing.client.android.SCAN");
        intent.setPackage("com.google.zxing.client.android");
        startActivityForResult(intent, 0);
    }
}   //JavascriptInterface
public void onActivityResult(int requestCode, int resultCode, Intent intent) {
    if (requestCode == 0) {
        if (resultCode == RESULT_OK) {
            //here is where you get your result
            String barcode = intent.getStringExtra("SCAN_RESULT");
        }
    }
}

现在这是我在网上找到的代码示例,对我来说一切看起来都很好,我最初唯一要做的就是导入所有必要的库,并将barcode变量定义为字符串类型。

任何帮助将不胜感激。或者也许是从Web视图访问条形码扫描仪功能的不同方法。

...
    myWebView.addJavascriptInterface(new WebAppInterface(this), "Android");
...
public class JavaScriptInterface {

您已经调用了接口类 JavaScriptInterface,但创建了一个新的 WebAppInterface 以传递给 WebView。

编辑

上面的答案仍然成立:你永远不会用你的WebView注册接口(JavaScriptInterface(,所以你无权访问这些方法(scanBarcode(。

由于 WebView 中实际上需要两个接口,因此需要添加这两个接口

myWebView.addJavascriptInterface(new WebAppInterface(this), "Android");
myWebView.addJavascriptInterface(new JavaScriptInterface(this), "Scanner");

在你的javascript中,你将有两个对象AndroidScanner,它们对应于这两个接口。Android将具有WebAppInterface中定义的所有方法,而Scanner将具有JavaScriptInterface中定义的方法。你的 JavaScript 函数变成了

function scanner() {
    Scanner.scanBarcode();
}

编辑 2

如果您的目标是 sdk 版本 17 或更高版本,则还必须注释要在 javascript 中使用的方法

public class JavaScriptInterface {
...
    @JavascriptInterface
    public void scanBarcode() {
    ...
    }
}

奖励:一些不请自来的建议,使用更好的名称。如果 WebAppInterface 提供了与应用程序数据交互的方法,请将其称为 DataInterface(即使这可能太通用了,但我不知道您的应用程序使用哪种数据(。由于 JavaScriptInterface 启动扫描程序并关闭应用程序,也许你可以称之为 ActionInterface。将它们作为"AppData"和"AppActions"传递给您的WebView,为您提供了AppData.getCategories()AppActions.closeMyActivity()等更合乎逻辑的功能。