ML Kit条形码扫描:图像数据大小无效

时间:2018-05-16 14:43:12

标签: java android firebase android-camera2 firebase-mlkit

我想检测捕获图像中的条形码。我使用android的camera2捕获图像。在此之后,检索图像的元数据并将图像保存到设备。元数据全部传递给下一个活动,即应用程序尝试检测条形码的位置。

下一个活动从之前保存的文件中创建一个byte []。接下来,使用与intent一起传递的数据创建相关的FirebaseVision对象。最后,应用程序尝试调用detectInImage()方法,其中抛出错误:

  

" java.lang.IllegalArgumentException:图像数据大小无效。"

我怀疑这是来自捕获的图像太大,但我似乎无法弄清楚如何捕获较小的图像,我也无法在参考文档中找到任何关于允许的最大尺寸的内容。有关此错误以及如何解决此问题的信息将非常感激。以下是我认为相关的代码。

private final ImageReader.OnImageAvailableListener onImageAvailableListener
        = new ImageReader.OnImageAvailableListener() {
    @Override
    public void onImageAvailable(ImageReader imageReader) {
        try{
            // Semaphore ensures date is recorded before starting next activity
            storeData.acquire();
            Image resultImg = imageReader.acquireNextImage(); // Image from camera
            imgWidth = resultImg.getWidth();
            imgHeight = resultImg.getHeight();
            ByteBuffer buffer = resultImg.getPlanes()[0].getBuffer();
            data = new byte[buffer.remaining()]; // Byte array with the images data
            buffer.get(data);
            String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss").format(new Date());

            // Note: mediaFile directs to Pictures/"ThisProject" folder
            File media = new File(mediaFile.getPath() +
                    File.separator + "IMG_" + timeStamp + ".jpg");

            // Saving the image
            FileOutputStream fos = null;
            try {
                fos = new FileOutputStream(media);
                fos.write(data);
                uri = Uri.fromFile(media);
            } catch (IOException e) {
                Log.e(TAG, e.getMessage());
            } finally {
                if (fos != null) {
                    try {
                        fos.close();
                    } catch (IOException e) {
                        Log.e(TAG, e.getMessage());
                    }
                }
            }
            resultImg.close();
        } catch (InterruptedException e) {
            Log.e(TAG, e.getMessage());
        }
        storeData.release();

    }
};

这基本上检索图像高度& width,然后将其写入文件。 发送到下一个活动的数据包括:图像宽度,图像高度,图像旋转和指向文件的Uri。

使用此功能,我尝试使用Firebase ML Kit检测条形码:

// uri is the uri referencing the saved image
File f = new File(uri.getPath());
data = new byte[(int) f.length()];

try{
        BufferedInputStream bis = new BufferedInputStream(new FileInputStream(f));
        DataInputStream dis = new DataInputStream(bis);
        dis.readFully(data);
    } catch (IOException e) {
        Log.e(TAG, e.getMessage());
    }

FirebaseVisionBarcodeDetectorOptions options = new FirebaseVisionBarcodeDetectorOptions.Builder().setBarcodeFormats(
            FirebaseVisionBarcode.FORMAT_QR_CODE,
            FirebaseVisionBarcode.FORMAT_DATA_MATRIX
    ).build();

FirebaseVisionBarcodeDetector detector = FirebaseVision.getInstance().getVisionBarcodeDetector(options);
FirebaseVisionImage image;

int rotationResult;
        switch (imgRotation) {
            case 0: {
                rotationResult = FirebaseVisionImageMetadata.ROTATION_0;
                break;
            }
            case 90: {
                rotationResult = FirebaseVisionImageMetadata.ROTATION_90;
                break;
            }
            case 180: {
                rotationResult = FirebaseVisionImageMetadata.ROTATION_180;
                break;
            }
            case 270: {
                rotationResult = FirebaseVisionImageMetadata.ROTATION_270;
                break;
            }
            default: {
                rotationResult = FirebaseVisionImageMetadata.ROTATION_0;
                break;
            }
        }

FirebaseVisionImageMetadata metadata = new FirebaseVisionImageMetadata.Builder()
            .setWidth(imgWidth)
            .setHeight(imgHeight)
            .setFormat(FirebaseVisionImageMetadata.IMAGE_FORMAT_NV21)
            .setRotation(rotationResult)
            .build();

image = FirebaseVisionImage.fromByteArray(data, metadata);

Task<List<FirebaseVisionBarcode>> result = detector.detectInImage(image)

1 个答案:

答案 0 :(得分:5)

一些事情。

  1. 如果您使用camera2,则您的图像格式不应为NV21。有关所有camera2支持的图像格式,请参见此处: https://developer.android.com/reference/android/media/Image#getFormat() 您的byte []不是NV21而您指定了IMAGE_FORMAT_NV21并导致错误

  2. 与camera2最直观的集成如下:

    • 实例化ImageReader时指定JPEG格式。

    • onImageAvailable会返回一个android.media.Image,你可以直接使用FirebaseVisionImage.fromMediaImage(...)创建一个FirebaseVisionImage。 (您可以在官方文档here

    • 中找到如何计算轮换信息
  3. 如果您必须完成两项活动,那么您需要解决android.media.Image不能Parcelable的事实。我建议您首先将其转换为Bitmap,这是Parcelable,您可以直接将其设置为Intent extra(由您决定。从最终用户的角度来看,它是非常见的)看到条形码保存到我的图库。 因此,您可能需要考虑跳过将其保存到文件的步骤。之后,在您的第二个活动中,您可以使用FirebaseVisionImage.fromBitmap(...)