首页 > 解决方案 > 如何将 YUV 转换为横向位图?

问题描述

第一种方法

当我设置 javasetRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_SENSOR); java setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT);

Camera2 运行良好并从 imageavailablelistener 保存图像。

第二种方法

当我设置java setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE); java setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_REVERSE_LANDSCAPE);

它将退出应用程序并且无法保存图像

转换积分

YUV Image to Bitmap 方法归功于链接

呼叫者

    private final ImageReader.OnImageAvailableListener mOnImageAvailableListener = new ImageReader.OnImageAvailableListener() {
        @Override
        public void onImageAvailable(ImageReader reader) {
            final Image image = reader.acquireLatestImage();
            if (image == null)
                return;
            BitmapFunctions bitmapFunctions = new BitmapFunctions();
//            QrDecoder qrDecoder = new QrDecoder();

            //Convert YUV to RGB and return bitmap
            Bitmap bitmap = bitmapFunctions.mediaImageToBitmap(image, activity);

实际转换函数

public class BitmapFunctions {
    private Allocation allocationYuv;
    private Allocation allocationRgb;
    private RenderScript rs;




    public Bitmap mediaImageToBitmap(Image image, Context context) {
        final ByteBuffer yuvBytes = imageToByteBuffer(image);

        // Convert YUV to RGB
        rs = RenderScript.create(context);

        Bitmap bitmap = Bitmap.createBitmap(image.getWidth(), image.getHeight(), Bitmap.Config.ARGB_8888);
        allocationRgb = Allocation.createFromBitmap(rs, bitmap);

        allocationYuv = Allocation.createSized(rs, Element.U8(rs), yuvBytes.array().length);
        allocationYuv.copyFrom(yuvBytes.array());

        ScriptIntrinsicYuvToRGB scriptYuvToRgb = ScriptIntrinsicYuvToRGB.create(rs, Element.U8_4(rs));
        scriptYuvToRgb.setInput(allocationYuv);
        scriptYuvToRgb.forEach(allocationRgb);

        allocationRgb.copyTo(bitmap);
        return bitmap;
    }

    public void release() {
        allocationYuv.destroy();
        allocationRgb.destroy();
        rs.destroy();
    }

    
    private ByteBuffer imageToByteBuffer(final Image image) {
        final Rect crop = image.getCropRect();
        final int width = crop.width();
        final int height = crop.height();

        final Image.Plane[] planes = image.getPlanes();
        final byte[] rowData = new byte[planes[0].getRowStride()];
        final int bufferSize = width * height * ImageFormat.getBitsPerPixel(ImageFormat.YUV_420_888) / 8;
        final ByteBuffer output = ByteBuffer.allocateDirect(bufferSize);

        int channelOffset = 0;
        int outputStride = 0;

        for (int planeIndex = 0; planeIndex < 3; planeIndex++) {
            if (planeIndex == 0) {
                channelOffset = 0;
                outputStride = 1;
            } else if (planeIndex == 1) {
                channelOffset = width * height + 1;
                outputStride = 2;
            } else if (planeIndex == 2) {
                channelOffset = width * height;
                outputStride = 2;
            }

            final ByteBuffer buffer = planes[planeIndex].getBuffer();
            final int rowStride = planes[planeIndex].getRowStride();
            final int pixelStride = planes[planeIndex].getPixelStride();

            final int shift = (planeIndex == 0) ? 0 : 1;
            final int widthShifted = width >> shift;
            final int heightShifted = height >> shift;

            buffer.position(rowStride * (crop.top >> shift) + pixelStride * (crop.left >> shift));

            for (int row = 0; row < heightShifted; row++) {
                final int length;

                if (pixelStride == 1 && outputStride == 1) {
                    length = widthShifted;
                    buffer.get(output.array(), channelOffset, length);
                    channelOffset += length;
                } else {
                    length = (widthShifted - 1) * pixelStride + 1;
                    buffer.get(rowData, 0, length);

                    for (int col = 0; col < widthShifted; col++) {
                        output.array()[channelOffset] = rowData[col * pixelStride];
                        channelOffset += outputStride;
                    }
                }

                if (row < heightShifted - 1) {
                    buffer.position(buffer.position() + rowStride - length);
                }
            }
        }

        return output;
    }

错误

2019-06-19 17:04:47.551 23356-23423/com.dsonic.datasonicscanner E/AndroidRuntime: FATAL EXCEPTION: CameraBackground
    Process: com.dsonic.datasonicscanner, PID: 23356
    java.lang.IllegalStateException: buffer is inaccessible
        at java.nio.DirectByteBuffer.get(DirectByteBuffer.java:219)
        at com.dsonic.dsoniccamera2lib.Camera2.BitmapFunctions.imageToByteBuffer(BitmapFunctions.java:105)
        at com.dsonic.dsoniccamera2lib.Camera2.BitmapFunctions.mediaImageToBitmap(BitmapFunctions.java:36)
        at com.dsonic.dsoniccamera2lib.Camera2.Camera2Manager$4.onImageAvailable(Camera2Manager.java:302)
        at android.media.ImageReader$ListenerHandler.handleMessage(ImageReader.java:812)
        at android.os.Handler.dispatchMessage(Handler.java:108)
        at android.os.Looper.loop(Looper.java:166)
        at android.os.HandlerThread.run(HandlerThread.java:65)

ctrl+f找到错误行

BitmapFunctions.java:105 是 channelOffset += outputStride;

BitmapFunctions.java:36 是位图 bitmap = Bitmap.createBitmap(image.getWidth(), image.getHeight(), Bitmap.Config.ARGB_8888);

Camera2Manager.java:302 是位图 bitmap = bitmapFunctions.mediaImageToBitmap(image, activity);

2019 年 6 月 20 日更新

现在一切正常,相同的代码没有任何变化。使用相同的电缆,但电话今天早上重新启动并添加Log.e("Image Size", "Width = " + image.getWidth() + " Height = " + image.getHeight());

在我将其注释掉后,它会再次自动退出我。现在我删除了评论,它不能再工作了。怎么回事?

谁能给我解释一下?

标签: androidandroid-bitmapandroid-imageandroid-camera2bytebuffer

解决方案


setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_SENSOR)而不是设置这个。

将其更改为强制清单中的方向。不知道为什么这个工作

希望有人能给我解释一下。


推荐阅读