带有 ZXing 1D 条形码 reader 的 CameraX 实现,ImageProxy 从目标旋转 90 度,无法找到旋转它的方法

CameraX implementation with ZXing 1D barcode reader, ImageProxy is rotated 90 degrees from the target and can't find a way to rotate it

我正在尝试使用带有 ZXing 库的 CameraX 来识别条形码, 我正在使用 ImageAnalyzer 获取 ImageProxy 并将其字节数组提供给 PlanarYUVLuminanceSource 以通过 ZXing 处理它。 我的目标旋转是 0,来自相机的 imageProxy 是 90 度,所以条形码在我旋转 90 度之前是不可读的。 这是代码:

private void buildAnalyzer() {
    if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.LOLLIPOP) {
        imageAnalysis = new ImageAnalysis.Builder()
                .setTargetResolution(new Size(1080, 720))
                .setTargetRotation(Surface.ROTATION_0)
                .setBackpressureStrategy(ImageAnalysis.STRATEGY_KEEP_ONLY_LATEST)
                .build();
    }

    MultiFormatReader reader = new MultiFormatReader();

    Hashtable<DecodeHintType, Object> hints = new Hashtable<DecodeHintType, Object>(
            2);
    Vector<BarcodeFormat> decodeFormats = new Vector<BarcodeFormat>();

    decodeFormats = new Vector<BarcodeFormat>();

    decodeFormats.add(BarcodeFormat.EAN_13);

    hints.put(DecodeHintType.POSSIBLE_FORMATS, decodeFormats);
    hints.put(DecodeHintType.TRY_HARDER, Boolean.TRUE);

    reader.setHints(hints);


    imageAnalysis.setAnalyzer(Executors.newFixedThreadPool(1), new ImageAnalysis.Analyzer() {
        @Override
        public void analyze(@NonNull ImageProxy imageProxy) {

            int rotationDegrees = imageProxy.getImageInfo().getRotationDegrees();
            Log.v(TAG, "Rotation + " + rotationDegrees);
            Log.v(TAG, "format + " + imageProxy.getFormat());


            
           
            ImageProxy.PlaneProxy[] planes = imageProxy.getPlanes();
            Log.v(TAG, "planes " + planes.length);

            ByteBuffer yBuffer = planes[0].getBuffer();
            ByteBuffer uBuffer = planes[1].getBuffer();
            ByteBuffer vBuffer = planes[2].getBuffer();

            int ySize = yBuffer.remaining();
            int uSize = uBuffer.remaining();
            int vSize = vBuffer.remaining();

            byte[] data = new byte[ySize + uSize + vSize];
            //U and V are swapped
            yBuffer.get(data, 0, ySize);
            vBuffer.get(data, ySize, vSize);
            uBuffer.get(data, ySize + vSize, uSize);

            Log.v(TAG, "width : " + imageProxy.getWidth());
            Log.v(TAG, "height : " + imageProxy.getHeight());


           
            Log.v(TAG, "planes 1 size : " + ySize + " remaining: " + yBuffer.remaining());
            Log.v(TAG, "planes 2 size : " + uSize + " remaining: " + uBuffer.remaining());
            Log.v(TAG, "planes 3 size : " + vSize + " remaining: " + vBuffer.remaining());


            Log.v(TAG, "data length : " + data.length);


            PlanarYUVLuminanceSource source = new PlanarYUVLuminanceSource(data,
                    imageProxy.getWidth(),
                    imageProxy.getHeight()
                    , 0, 0,
                    imageProxy.getWidth(),
                    imageProxy.getHeight()
                    , false);
            BinaryBitmap binary = new BinaryBitmap(new HybridBinarizer(source));


            try {
                Result result = reader.decodeWithState(binary);
                Log.d(TAG, "reader read " + result.getText());
                Looper looper = Looper.getMainLooper();
                Handler handler = new Handler(looper);
                handler.post(new Runnable() {
                    @Override
                    public void run() {
                        Toast.makeText(con, result.getText() + "\n" + result.getNumBits(), Toast.LENGTH_LONG).show();

                    }
                });

            } catch (NotFoundException e) {
                Log.d(TAG, "exception  " + e.getMessage());

            }



            imageProxy.close();
        }
    });
}

上面的代码工作正常,这表明字节没有问题,也就是说,这是我尝试过但从未奏效的解决方案:

1- 旋转字节数组:我使用此方法 来旋转我从 ImageProxy 中提取的字节数组“数据”,但它返回了一个错误数组超出范围的异常, 此方法适用于因子为 4 的数组,而我的字节数组不满足此条件。 从逻辑上讲,字节数组是 yuv420 图像格式,由具有一定宽度和高度的 3 层组成。 但是让我发疯的是宽度和高度与数组不兼容:

宽度 = 1440 , 高度 = 1080 , 飞机 1 尺寸:1589728, 飞机 2 尺寸:794847, 飞机 3 尺寸:794847

看看这些数字,它们不能被 1440 或 1080 整除,也不能被 4 整除。

2- 从字节数组创建位图并旋转它,然后从新位图中提取旋转后的字节数组: 也不起作用,因为宽度和高度是未知的,就像之前一样,分辨率与字节数组编号不兼容,结果位图是一堆扭曲的绿色混乱。

3- activity 被清单中的纵向模式锁定 android:screenOrientation="portrait" :同样的问题,imageProxy 旋转仍然是 90

4- PlanarYUVLuminanceSource 不支持旋转:(不知道为什么。

我几乎尝试了 Whosebug 上的所有解决方案并阅读了 cameraX 的所有文档,但似乎没有任何方法可以解决这个问题。 如果有办法知道 byteArray 的实际分辨率,我想它会解决问题。

byte[] data = ImageUtil.imageToJpegByteArray(imageProxy);
Bitmap bitmap= BitmapFactory.decodeByteArray(data, 0, data.size)

您可以通过上面的代码获取位图。 ImageUtil class 在 androidx.camera.core

androidx.camera.core.internal.utils.ImageUtil

经过大量研究,我实现了这个扩展功能:

fun ImageProxy.toBitmap(): Bitmap {
    val buffer = planes[0].buffer.apply { rewind() }
    val bytes = ByteArray(buffer.capacity())

    // Get bitmap
    buffer.get(bytes)
    val bitmap = BitmapFactory.decodeByteArray(bytes, 0, bytes.size)

    // Fix rotation if needed
    val angle = imageInfo.rotationDegrees.toFloat()
    val matrix = Matrix().apply { postRotate(angle) }

    // Return rotated bitmap
    return Bitmap.createBitmap(bitmap, 0, 0, bitmap.width, bitmap.height, matrix, true)
}

您可以通过从 android camerax 库中调用 takePicture 来获取 ImageProxy

imageCapture.takePicture(cameraExecutor, object : ImageCapture.OnImageCapturedCallback() {
    override fun onCaptureSuccess(imageProxy: ImageProxy) {
        val bitmap = imageProxy.toBitmap()
        imageProxy.close()
    }
})