- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
与 MediaProjection
Android L 中可用的 API
capture the contents of the main screen (the default display) into a Surface object, which your app can then send across the network
VirtualDisplay
工作,还有我的
SurfaceView
正确显示屏幕内容。
Surface
中显示的帧,并将其打印到文件中。我尝试了以下方法,但得到的只是一个黑色文件:
Bitmap bitmap = Bitmap.createBitmap
(surfaceView.getWidth(), surfaceView.getHeight(), Bitmap.Config.ARGB_8888);
Canvas canvas = new Canvas(bitmap);
surfaceView.draw(canvas);
printBitmapToFile(bitmap);
Surface
中检索显示数据的任何想法?
VirtualDisplay
使用
Surface
的
ImageReader
:
Display display = getWindowManager().getDefaultDisplay();
Point size = new Point();
display.getSize(size);
displayWidth = size.x;
displayHeight = size.y;
imageReader = ImageReader.newInstance(displayWidth, displayHeight, ImageFormat.JPEG, 5);
Surface
的虚拟显示到
MediaProjection
:
int flags = DisplayManager.VIRTUAL_DISPLAY_FLAG_OWN_CONTENT_ONLY | DisplayManager.VIRTUAL_DISPLAY_FLAG_PUBLIC;
DisplayMetrics metrics = getResources().getDisplayMetrics();
int density = metrics.densityDpi;
mediaProjection.createVirtualDisplay("test", displayWidth, displayHeight, density, flags,
imageReader.getSurface(), null, projectionHandler);
Image
来自
ImageReader
并从中读取数据:
Image image = imageReader.acquireLatestImage();
byte[] data = getDataFromImage(image);
Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length);
null
.
getDataFromImage
方法:
public static byte[] getDataFromImage(Image image) {
Image.Plane[] planes = image.getPlanes();
ByteBuffer buffer = planes[0].getBuffer();
byte[] data = new byte[buffer.capacity()];
buffer.get(data);
return data;
}
Image
从 acquireLatestImage
返回总是有默认大小为 7672320 的数据,解码返回 null
.
ImageReader
尝试获取图像,状态
ACQUIRE_NO_BUFS
被退回。
最佳答案
在花了一些时间并了解了一些超出预期的 Android 图形架构之后,我已经开始使用它了。所有必要的部分都有详细记录,但如果您还不熟悉 OpenGL,可能会引起头痛,所以这里有一个很好的总结“傻瓜”。
我假设你
ImageReader
是关于。该类一开始名字很糟糕——最好称它为“ImageReceiver”——一个围绕 BufferQueue 接收端的愚蠢包装器(无法通过任何其他公共(public) API 访问)。不要被愚弄:它不执行任何转换。它不允许查询格式,生产者支持,即使 C++ BufferQueue 在内部公开该信息。在简单的情况下它可能会失败,例如,如果生产者使用自定义的、晦涩的格式(例如 BGRA)。
Surface
,由 ImageReader/MediaCodec 返回。没什么特别的,只是 SurfaceTexture 上的普通 Surface 有两个陷阱:
OES_EGL_image_external
和
EGL_ANDROID_recordable
.
OnFrameAvailableListener
回调将包含以下步骤:
EglCore eglCore;
Surface producerSide;
SurfaceTexture texture;
int textureId;
OffscreenSurface consumerSide;
ByteBuffer buf;
Texture2dProgram shader;
FullFrameRect screen;
...
// dimensions of the Display, or whatever you wanted to read from
int w, h = ...
// feel free to try FLAG_RECORDABLE if you want
eglCore = new EglCore(null, EglCore.FLAG_TRY_GLES3);
consumerSide = new OffscreenSurface(eglCore, w, h);
consumerSide.makeCurrent();
shader = new Texture2dProgram(Texture2dProgram.ProgramType.TEXTURE_EXT)
screen = new FullFrameRect(shader);
texture = new SurfaceTexture(textureId = screen.createTextureObject(), false);
texture.setDefaultBufferSize(reqWidth, reqHeight);
producerSide = new Surface(texture);
texture.setOnFrameAvailableListener(this);
buf = ByteBuffer.allocateDirect(w * h * 4);
buf.order(ByteOrder.nativeOrder());
currentBitmap = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888);
只有在完成上述所有操作后,您才能使用
producerSide
初始化您的 VirtualDisplay。表面。
float[] matrix = new float[16];
boolean closed;
public void onFrameAvailable(SurfaceTexture surfaceTexture) {
// there may still be pending callbacks after shutting down EGL
if (closed) return;
consumerSide.makeCurrent();
texture.updateTexImage();
texture.getTransformMatrix(matrix);
consumerSide.makeCurrent();
// draw the image to framebuffer object
screen.drawFrame(textureId, matrix);
consumerSide.swapBuffers();
buffer.rewind();
GLES20.glReadPixels(0, 0, w, h, GLES10.GL_RGBA, GLES20.GL_UNSIGNED_BYTE, buf);
buffer.rewind();
currentBitmap.copyPixelsFromBuffer(buffer);
// congrats, you should have your image in the Bitmap
// you can release the resources or continue to obtain
// frames for whatever poor-man's video recorder you are writing
}
上面的代码是方法的大大简化版本,可在
this Github project 中找到,但所有引用的类都直接来自
Grafika .
String VERTEX_SHADER_FLIPPED =
"uniform mat4 uMVPMatrix;\n" +
"uniform mat4 uTexMatrix;\n" +
"attribute vec4 aPosition;\n" +
"attribute vec4 aTextureCoord;\n" +
"varying vec2 vTextureCoord;\n" +
"void main() {\n" +
" gl_Position = uMVPMatrix * aPosition;\n" +
" vec2 coordInterm = (uTexMatrix * aTextureCoord).xy;\n" +
// "OpenGL ES: how flip the Y-coordinate: 6542nd edition"
" vTextureCoord = vec2(coordInterm.x, 1.0 - coordInterm.y);\n" +
"}\n";
离别的话
EXT_texture_format_BGRA8888
支持(可能会),分配屏幕外缓冲区并使用 glReadPixels 以这种格式检索图像。
关于android - 使用 MediaProjection 截取屏幕截图,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26545970/
我设法得到截图,但结果是这样的: 原来的: 这是我从几个来源获取的代码: final ImageReader ir = ImageReader.newInstance(width, height, P
我目前使用的是 Android 5.0 MediaProjection API。我已经成功地能够从我的应用程序启动投影 session ,但是我注意到用户可以随时转到通知栏并停止媒体投影。我的目标是捕
我正在使用 MediaProjection 录制我的屏幕如下 Display display = getWindowManager().getDefaultDisplay(); Point size
与 MediaProjection Android L 中可用的 API capture the contents of the main screen (the default display) i
我正在使用 MediaProjection 截取屏幕截图。这就是我正在做的。我使用服务创建了一个覆盖图标。单击覆盖图标会截取屏幕截图。问题是,无论何时通过按后退按钮终止应用程序或通过滑动它手动丢失 M
在对这个主题进行了大量研究之后,虽然我找到了一些答案,但我无法理解 MediaProjection API 的工作原理。 我想从后台服务类中获取设备的屏幕截图。有没有可能做到。我有一个 MainAct
我正在使用以下代码获取屏幕截图。我得到了屏幕截图,但图像上附有黑色填充。这里使用网上找到的截图和去除填充的代码示例: // requesting permission MediaPro
我正在使用 MediaProjection API 创建一个屏幕录像机应用程序。在我的应用程序中,我将停止按钮显示为一个小的覆盖窗口。我已将此 View 保持为安全状态,因此它不会出现在最终录制的视频
随着 Android Q 生效的新隐私变更,现在任何使用 MediaProjection API 的应用都必须指定 android:foregroundServiceType list 下服务标记中的
我正在尝试查找有关与受 DRM 保护的内容和 MediaProjection API 的兼容性的信息。 具体来说,我想知道我是否可以从 Netflix 或其他实现 DRM 保护的应用程序录制视频? 最
我在想出有效的解决方案时遇到了一些麻烦。我预见到一些问题,第一个是...... OOM 预防 如果我想要过去 30 秒甚至 5 分钟,这是可行的,但如果我想要过去 30 分钟或整整一个小时,或者可能是
我正在开发一个需要将屏幕捕获到位图以进行传输的应用程序。我正在尝试使用新的 Android 5.0 android.media.projection APIs进行屏幕捕获。 此 API 的工作流以调用
我仍然是 Android APP 开发的初学者,我正在尝试使用 Mediaprojection API 来录制我的屏幕..我现在面临的问题是..录制后我无法在定义的位置找到视频文件(sdcard/ca
拜托,我想知道如何使用 Android 的 MediaProjection API 来录制在 Android 设备中输出的系统声音?我要录制的系统声音是从某些用户应用程序、扬声器或耳机中从 Andro
我是一名优秀的程序员,十分优秀!