gpt4 book ai didi

Android 捕获屏幕到 ImageReader 的表面

转载 作者:塔克拉玛干 更新时间:2023-11-03 00:07:47 25 4
gpt4 key购买 nike

我目前正在学习一个github项目ScreenCapture的代码,它可以捕获屏幕并在surfaceview中显示图像,这里是项目https://github.com/Charlesjean/android-ScreenCapture .我尝试使用以下代码将 SurfaceView 的表面替换为 ImageReader 对象的表面:

 mImgReader = ImageReader.newInstance(mWidth, mHeight, ImageFormat.JPEG, 5);
mSurface = mImgReader.getSurface();// mSurfaceView.getHolder().getSurface();
mImgReader.setOnImageAvailableListener(new ImageReader.OnImageAvailableListener() {
@Override
public void onImageAvailable(ImageReader reader) {
Log.i(TAG, "in OnImageAvailable");

}
}, mHandler);

并像这样创建 VirtualDisplay:

mVirtualDisplay = mMediaProjection.createVirtualDisplay("ScreenCapture",
mWidth, mHeight, mScreenDensity,
DisplayManager.VIRTUAL_DISPLAY_FLAG_OWN_CONTENT_ONLY |
DisplayManager.VIRTUAL_DISPLAY_FLAG_PUBLIC,
mSurface, new VirtualDisplay.Callback() {
@Override
public void onResumed() {
Log.i(TAG, "onResumed");
super.onResumed();
}

@Override
public void onPaused() {
Log.i(TAG, "onPaused");
super.onPaused();
}
}, mHandler);

但是从未调用过onImageAvailable 方法,有人对此有任何经验吗?我不明白为什么这不起作用。

最佳答案

谢谢 Simon,我通过将图像格式更改为 PixelFormat.RGBA_8888 解决了这个问题,但是当您执行我所做的操作时,还有其他一些需要注意的地方,我将其张贴在这里以防它对其他人有所帮助 future 。

Image.Plane 的数据缓冲区与Bitmap 需要的数据缓冲区不完全相同:
1. 我们用来创建的图像格式 ImageReader是PixelFormat.RGBA_8888,所以Image.Plane的buffer会先放R(ed) channel ,再放G(reen) channel ,依此类推。为了将此缓冲区转换为位图,我们需要像这样创建位图
bitmap = Bitmap.createBitmap(metrics,width, height, Bitmap.Config.ARGB_8888);
而位图的缓冲区需要把Alpha channel 放在最前面。
2.我们从Image.Plane得到的buffer每一行都有一些padding,个人认为这是硬件设备用来加速buffer操作或者对齐。所以为了复制这个缓冲区,我们需要删除这些填充。

要理解这两点,请看下面的代码:

 public void onImageAvailable(ImageReader reader) {
Log.i(TAG, "in OnImageAvailable");
FileOutputStream fos = null;
Bitmap bitmap = null;
Image img = null;
try {
img = reader.acquireLatestImage();
if (img != null) {
Image.Plane[] planes = img.getPlanes();
if (planes[0].getBuffer() == null) {
return;
}
int width = img.getWidth();
int height = img.getHeight();
int pixelStride = planes[0].getPixelStride();
int rowStride = planes[0].getRowStride();
int rowPadding = rowStride - pixelStride * width;
byte[] newData = new byte[width * height * 4];

int offset = 0;
bitmap = Bitmap.createBitmap(metrics,width, height, Bitmap.Config.ARGB_8888);
ByteBuffer buffer = planes[0].getBuffer();
for (int i = 0; i < height; ++i) {
for (int j = 0; j < width; ++j) {
int pixel = 0;
pixel |= (buffer.get(offset) & 0xff) << 16; // R
pixel |= (buffer.get(offset + 1) & 0xff) << 8; // G
pixel |= (buffer.get(offset + 2) & 0xff); // B
pixel |= (buffer.get(offset + 3) & 0xff) << 24; // A
bitmap.setPixel(j, i, pixel);
offset += pixelStride;
}
offset += rowPadding;
}
String name = "/myscreen" + count + ".png";
count++;
File file = new File(Environment.getExternalStorageDirectory(), name);
fos = new FileOutputStream(file);
bitmap.compress(Bitmap.CompressFormat.JPEG, 100, fos);
Log.i(TAG, "image saved in" + Environment.getExternalStorageDirectory() + name);
img.close();
}

} catch (Exception e) {
e.printStackTrace();
} finally {
if (null != fos) {
try {
fos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (null != bitmap) {
bitmap.recycle();
}
if (null != img) {
img.close();
}

}
}

关于Android 捕获屏幕到 ImageReader 的表面,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27581750/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com