Android 截屏实现、屏幕截图、MediaProjection、ImageReader
1. 第一步:调起系统捕获屏幕的Intent
MainActivity:public void goCaptureIntent() {//第一步.调起系统捕获屏幕的IntentmMediaProjectionManager = (MediaProjectionManager) getSystemService(MEDIA_PROJECTION_SERVICE);Intent captureIntent = mMediaProjectionManager.createScreenCaptureIntent();startActivityForResult(captureIntent, SCREEN_SHOT_CODE);
}
2. 第二步:通过startForegroundService来获取mediaProjection
注:sdk 28以及以下可以直接在Activity中获取,29以及以上在Activity中获取会报错
MainActivity:@RequiresApi(api = Build.VERSION_CODES.O)
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {super.onActivityResult(requestCode, resultCode, data);if(requestCode==SCREEN_SHOT_CODE && resultCode == RESULT_OK){//第二步通过startForegroundService来获取mediaProjectionIntent service = new Intent(this, ScreenShootService.class);service.putExtra("code", resultCode);service.putExtra("data", data);service.putExtra("width", getScreenWidth(this));service.putExtra("height", getScreenHeight(this));startForegroundService(service);}
}
3. 第三步:获取mediaProjection
注:AndroidManifest.xml中注册的service要加上以下属性
<service android:name=".ScreenShootService"android:enabled="true"android:foregroundServiceType="mediaProjection"></service>
第三步的代码:
ScreenShootService:@Override
public int onStartCommand(Intent intent, int flags, int startId) {int resultCode = intent.getIntExtra("code", -1);Intent data = intent.getParcelableExtra("data");width = intent.getIntExtra("width",1080);height = intent.getIntExtra("height",1920);notification();//第三步:获取mediaProjectionMediaProjectionManager mMediaProjectionManager = (MediaProjectionManager) getSystemService(MEDIA_PROJECTION_SERVICE);MediaProjection mediaProjection = mMediaProjectionManager.getMediaProjection(resultCode, data);if (mediaProjection == null) {Log.e(TAG, "media projection is null");}Bitmap bitmap = screenShot(mediaProjection);EventBus.getDefault().post(bitmap);return super.onStartCommand(intent, flags, startId);
}
4. 第四步:通过mediaProjection截图获取Image
ScreenShootService://第四步;通过mediaProjection截图获取Image
public Bitmap screenShot(MediaProjection mediaProjection){Objects.requireNonNull(mediaProjection);@SuppressLint("WrongConstant")ImageReader imageReader = ImageReader.newInstance(width, height, PixelFormat.RGBA_8888, 60);VirtualDisplay virtualDisplay = mediaProjection.createVirtualDisplay("screen", width, height, 1, DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR,imageReader.getSurface(), null, null);//取现在最新的图片SystemClock.sleep(1000);//取最新的图片Image image = imageReader.acquireLatestImage();
// Image image = imageReader.acquireNextImage();//释放 virtualDisplay,不释放会报错virtualDisplay.release();return image2Bitmap(image);
}
5. 第五步:将Image转为Bitmap
ScreenShootService://第五步:将Image转为Bitmap
public static Bitmap image2Bitmap(Image image) {if (image == null) {System.out.println("image 为空");return null;}int width = image.getWidth();int height = image.getHeight();final Image.Plane[] planes = image.getPlanes();final ByteBuffer buffer = planes[0].getBuffer();int pixelStride = planes[0].getPixelStride();int rowStride = planes[0].getRowStride();int rowPadding = rowStride - pixelStride * width;Bitmap bitmap = Bitmap.createBitmap(width+ rowPadding / pixelStride , height, Bitmap.Config.ARGB_8888);bitmap.copyPixelsFromBuffer(buffer);//截取图片
// Bitmap cutBitmap = Bitmap.createBitmap(bitmap,0,0,width/2,height/2);//压缩图片
// Matrix matrix = new Matrix();
// matrix.setScale(0.5F, 0.5F);
// System.out.println(bitmap.isMutable());
// bitmap = Bitmap.createBitmap(bitmap, 0, 0, width, height, matrix, false);image.close();return bitmap;
}
6. 第六步:展示截图结果
这里是通过EventBust从Service传到了Activity
MainActivity://接收service返回的bitmap图片数据
@Subscribe(threadMode = ThreadMode.MAIN)//在ui线程执行
public void onMoonEvent(Bitmap bitmap){//第六步:展示截图结果imgShowImage.setBackground(new BitmapDrawable(bitmap));
}
异常处理1. E/BufferQueueProducer: [ImageReader-2340x1080f1m1-24598-0](id:601600000001,api:1,p:1080,c:24598) dequeueBuffer: BufferQueue has been abandoned
分析:异常:BufferQueue已经被丢弃了,还在使用
说明有在可能是截图中使用到的某个对象被GC了,BufferQueue的对象大概有三个ImageReader 、VirtualDisplay 和 Image,Image在截图万之后已经close掉了,然后尝试截图后分别对ImageReader和VirtualDisplay进行释放,ImageReader释放问题依然存在,VirtualDisplay释放解决了该问题。
VirtualDisplay virtualDisplay = mediaProjection.createVirtualDisplay("screen", width, height, 1, DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR,imageReader.getSurface(), null, null);
...使用完Image之后...
...virtualDisplay.release();
异常处理2:Media projections require a foreground service of type ServiceInfo.FOREGROUND_SERVICE_TYPE_MEDIA_PROJECTION
sdk版本产生的适配问题
两种解决方案:
方案一:将sdk降低到28或者以下
方案二:即需要使用 startForegroundService
也就是上面demo中使用到的方法
Activity:public void goCaptureIntent() {//第一步.调起系统捕获屏幕的IntentmMediaProjectionManager = (MediaProjectionManager) getSystemService(MEDIA_PROJECTION_SERVICE);Intent captureIntent = mMediaProjectionManager.createScreenCaptureIntent();startActivityForResult(captureIntent, SCREEN_SHOT_CODE);
}@RequiresApi(api = Build.VERSION_CODES.O)
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {super.onActivityResult(requestCode, resultCode, data);if(requestCode==SCREEN_SHOT_CODE && resultCode == RESULT_OK){//第二步通过startForegroundService来获取mediaProjectionIntent service = new Intent(this, ScreenShootService.class);service.putExtra("code", resultCode);service.putExtra("data", data);service.putExtra("width", getScreenWidth(this));service.putExtra("height", getScreenHeight(this));startForegroundService(service);}
}
Service:@Override
public int onStartCommand(Intent intent, int flags, int startId) {int resultCode = intent.getIntExtra("code", -1);Intent data = intent.getParcelableExtra("data");width = intent.getIntExtra("width",1080);height = intent.getIntExtra("height",1920);notification();//第二步:创建mediaProjectionMediaProjectionManager mMediaProjectionManager = (MediaProjectionManager) getSystemService(MEDIA_PROJECTION_SERVICE);MediaProjection mediaProjection = mMediaProjectionManager.getMediaProjection(resultCode, data);if (mediaProjection == null) {Log.e(TAG, "media projection is null");}Bitmap bitmap = screenShot(mediaProjection);EventBus.getDefault().post(bitmap);return super.onStartCommand(intent, flags, startId);
}
AndroidManifest.xml:<service android:name=".ScreenShootService"android:enabled="true"android:foregroundServiceType="mediaProjection">
最终效果图:
需要源码的可以点个赞,点关注,然后评论里留下邮箱
Android 截屏实现、屏幕截图、MediaProjection、ImageReader相关推荐
- Android截屏方法总结
最近研究了一些Android的截屏方法,做一个总结. 图片剪裁方法 使用View.getDrawingCache()得到Bitmap.非常简单但是只能截图本应用的图片,并且没办法控制截图的范围. 对B ...
- Android截屏的几种实现
Android截屏的几种实现 最近我们的APP要求需要截屏功能,网上看了看大致有一下几种实现的方式,由于我们的机器是特定的设备,(类似于广告机,已经ROOT),所以就采用最简单的方式,只需在代码中执行 ...
- Android 截屏监听(截图分享功能实现)
具体来说就是,检测到了用户在应用内有截图,弹出一个分享界面, 在截图下方添加一个二维码,进行分享. ●●● 前言 Android系统没有直接对截屏事件监听的接口,也没有广播,只能自己动手来丰衣足食, ...
- android华为虚拟截屏黑屏,Android截屏表面视图显示黑屏
Android截屏表面视图显示黑屏 我试图通过代码拍摄我的游戏的截图,并通过一个意图来分享它.我能做这些事情,但是截图总是看起来是黑色的.下面是与分享截图相关的代码:View view = MainA ...
- android开发截屏代码,android截屏代码:C++实现
android截屏代码:C++实现 示例代码在: frameworks\base\services\surfaceflinger\tests\screencap\screencap.cpp /* * ...
- Android截屏分享
最近项目需要实现Android截屏分享功能,包括Android截屏获取图片.将图片保存到本地.通知系统相册更新.通过微信.QQ.微博分享截屏图片,本篇博客作为总结回顾. 一.Android截屏获取图片 ...
- android 截屏函数_android截屏功能实现代码
这篇文章主要为大家详细介绍了android截屏功能的实现代码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下 android开发中通过View的getDrawingCache方法可以达到截屏的目的, ...
- Android截屏截图的几种方法总结
Android截屏 Android截屏的原理:获取具体需要截屏的区域的Bitmap,然后绘制在画布上,保存为图片后进行分享或者其它用途 一.Activity截屏 1.截Activity界面(包含空白的 ...
- python 几种android截屏方式的学习与使用
python 几种android截屏方式的学习与使用 screencap screencap是Android的截屏工具.通过调用访问SurfaceFlinger服务或读取 /dev/graphics/ ...
- android盒子截图,Android截屏截图的几种方法总结
Android截屏 Android截屏的原理:获取具体需要截屏的区域的Bitmap,然后绘制在画布上,保存为图片后进行分享或者其它用途 一.Activity截屏 1.截Activity界面(包含空白的 ...
最新文章
- c语言中比较两个数组函数,输入两个数组,调用large函数比较,计数,输出统计结果...
- 从某一日期开始过day天的日期
- java程序初始化顺序
- python 使用文本注解绘制树节点_整理了 34 个被吹爆了的Python开源框架
- JS的instanceof
- python写mapreduce_用python写MapReduce函数——以WordCount为例
- 在原生js中的事件监听方法
- leetcode 179. 最大数(排序)
- Outlook最小到系统托盘
- 设计模式----Adapter(适配器)
- 用JSF实现页面刷新后,checkbox仍处于选中状态
- vue key重复_Vue 前端面试题
- python的分号用英文还是中文_中文的分号和英文的分号使用方法有区别吗?
- Echarts柱状图,如何基于数据进行百分比显示
- CuPy is not correctly installed
- 接口测试-什么是header头部?
- nginx新增conf文件
- 盖亚绘制的星图里,藏着银河系不为人知的混乱过去
- 查询Apple app的bundle ID
- 嵌套循环中的break和continue使用规则
热门文章
- linux服务器安装网卡驱动,Linux下如何安装网卡驱动
- mellanox 网卡驱动,Mellanox网卡OFED驱动安装
- FreeSWITCH折腾笔记9——使用FS做一个i-SBC
- android LBS模式,android: 如何开启webview的LBS功能
- 忘记电脑开机密码怎么办、win10、win7忘记开机密码怎么解决
- python上传数据到坚果云_坚果云功能
- 电脑桌面壁纸的尺寸分类
- vivado程序固化到flash
- 为什么我怎么也理解不了波粒二象性,是因为智商不够吗?
- win10c语言乱码修复方法,大神详解win10系统记事本中文变乱码的处理方案