Android Multimedia框架总结(二十五)MediaProjection实现手机截屏(无须root)

转载请把头部出处链接和尾部二维码一起转载,本文出自逆流的鱼yuiop:http://blog.csdn.net/hejjunlin/article/details/53966818

前言:一年半多以前,我们曾有个项目,要做一个截屏功能,当时负责调研的同事,答应了产品上这个功能,但开发一周后,发现,无法实现截取手机屏幕图像,须要root权限,才能做。因为最近研究MediaProjection,意外的发现,竟然无须root,可以轻松实现次功能。曾经被做不到的,如今做到了,很难相信此时的心情。看下今天的Agenda:

  • Android源码中使用组合键是如何实现屏幕截图功能的?
  • MediaProjection实现手机截屏效果
  • 简要思路

以我的魅族手机为例,是同时按电源键+音量下键来实现截屏,苹果手机则是电源键 + HOME键,小米手机是菜单键+音量下键,而HTC一般是按住电源键再按左下角的“主页”键。那么Android源码中使用组合键是如何实现屏幕截图功能呢?

Android源码中对按键的捕获位于文件PhoneWindowManager.java中

这个类处理所有的按键事件,其中函数interceptKeyBeforeQueueing()会对常用的按键做特殊处理。以我自己魅族手机为例,是同时按电源键和音量下键来截屏,那么在这个方法中可以看到这么如下代码:

interceptKeyBeforeQueueing

以上代码总结为:当按下音量下键时,且是down时,会进入到interceptPowerKeyDown方法中,而按下电源键时,且是down时,也会进入interceptPowerKeyDown方法中,接着进入这个方法中看下:

interceptPowerKeyDown

以上代码总结为:如果是interactive为true,且一些其他条件满足时,会记录电源键按下的时间,且进入interceptScreenshotChord方法中:

以上代码总结为:用两个布尔变量判断是否同时按了音量下键和电源键后,再计算两个按键响应Down事件之间的时间差不超过( // Time to volume and power must be pressed within this interval of each other.

private static final long SCREENSHOT_CHORD_DEBOUNCE_DELAY_MILLIS = 150;)150毫秒,也就认为是同时按了这两个键后,此种case时,才是真正的屏幕截屏的组合键。

获取到组合键后,会用一个mHandler 开始post一个runnable,进入这个runnable中:

以上代码,会执行方法takeScreenshot方法,接着进入:

以上代码总结为:使用AIDL绑定了service服务到”com.android.systemui.screenshot.TakeScreenshotService”,注意在service连接成功时,对message的msg.arg1和msg.arg2两个参数的赋值。其中在mScreenshotTimeout中对服务service做了超时处理。接着我们找到实现这个服务service的类TakeScreenshotService,该类在(frameworks/base/packages/SystemUI/src/com/android/systemui/screenshot包下):

对应代码如下:

以上代码总结为:引用了GlobalScreenshot类,调用了takeScreenshot方法,紧接着进入:

以上代码总结为:引用SurfaceControl类,调用了screenshot方法, 传入了屏幕的宽和高,这两个参数,接着进入SurfaceControl类中,位于frameworks/base/core/java/android/view目录下:

对应screenshot方法代码如下:

screenshot

最终到达native方法中nativeScreenshot

上面就是java层的部分,接着到jni层,在\frameworks\base\core\jni\android_view_SurfaceControl.cpp中:

以上代码总结为:到jni中,映射nativeScreenshot方法的是nativeScreenshotBitmap函数,这个函数代码如下:

以上代码总结为:实例化ScreenshotClient类,且调用update方法,最终通过构造一个新的bitmap,把screenshot得到的像素点,及相关信息达到bitmap,通过GraphicsJNI,重新创建一个bitmap返回给上层。

我们接着看下ScreenshotClient的声明,位于SurfaceComposerClient.h中。注意这个类是在frameworks\native\include\gui中,而前面android_view_SurfaceControl是在\frameworks\base\core\jni\下

最后辗转来到c++层,就是\frameworks\native\libs\gui下的SurfaceComposerClient.cpp中,实现ScreenshotClient声明的函数update,如下:

以上代码总结为:通过ISurfaceComposer接口,调用captureScreen函数,我们找到其对应的实现类\frameworks\native\libs\gui\ISurfaceComposer.cpp,找到captureScreen函数的实现如下:

以上代码总结为:把IGraphicBufferProducer,作为binder开始写到parcel中,最后一版看到transact,就知道当前是client端,而server端的onTransact()函数,会接收到传过来的参数。如这里的data。接着看下BnSurfaceComposer的onTransact方法,这个BnSurfaceComposer依然是在ISurfaceComposer.cpp中。

以上代码总结为:当进入到CAPTURE_SCREEN中,data会读取IGraphicBufferProducer生成出的图像buffe,接着调用 reply->writeInt32(res);返回给client.然后再回调到java层。以上就是系统截屏的原理。

那对于多媒体这块可以通过MediaProjection+VirturalDisplay+MediaProjectionManager来实现截屏,以下我的实例效果图:

效果图1:操作过程,点击浮动小剪刀,就可以实现截屏,拖动小剪刀,可到任意位置:

效果图2:截图后过程

主界面:

体验apk:

链接:http://pan.baidu.com/s/1kVugxCV 密码:97z0

实现思路:

  • MediaProjection是一个token,用户可以授予应用程序捕获屏幕内容和录制系统音频。
  • 屏幕截取,可以通过MediaProjectionManager的createScreenCaptureIntent,创建一个intent,保证有足够能力截取屏幕上的内容,但是没有系统声音。
  • 调用setUpMediaProjection()方法获取共享数据并对其进行赋值;若已初始化,则直接调用virtualDisplay()方法利用之前定义的变量对截屏环境进行初始化。

第一时间获得博客更新提醒,以及更多android干货,源码分析,欢迎关注我的微信公众号,扫一扫下方二维码或者长按识别二维码,即可关注。

如果你觉得好,随手点赞,也是对笔者的肯定,也可以分享此公众号给你更多的人,原创不易

时间: 2024-08-24 03:38:25

Android Multimedia框架总结(二十五)MediaProjection实现手机截屏(无须root)的相关文章

Android Multimedia框架总结(十四)Camera框架初识及自定义相机案例

转载请把头部出处链接和尾部二维码一起转载,本文出自逆流的鱼yuiop:http://blog.csdn.net/hejjunlin/article/details/52738492 前言:国庆节告一段落,又是新一月,上月主要是围绕MediaPlayer相关展开,从今天开始,开始分析多媒体框架中的Camera模块,看下今天的Agenda: Camera拍照 Camera录像 新API android.hardware.camera2 新旧API特点对比 Camera自定义相机 新API andro

Android Multimedia框架总结(十)Stagefright框架之音视频输出过程

转载请把头部出处链接和尾部二维码一起转载,本文出自逆流的鱼yuiop:http://blog.csdn.net/hejjunlin/article/details/52560012 前言:上篇文中最后介绍了数据解码放到Buffer过程,今天分析的是stagefright框架中音视频输出过程: 先看下今天的Agenda: 一张图回顾数据处理过程 视频渲染器构建过程 音频数据到Buffer过程 AudioPlayer在AwesomePlayer运行过程 音视频同步 音视频输出 一张图看音视频输出 一

Android项目实战(二十五):Android studio 混淆+打包+验证是否成功

前言: 单挑Android项目,最近即时通讯用到环信,集成sdk的时候 官方有一句 在 ProGuard 文件中加入以下 keep. -keep class com.hyphenate.** {*;} -dontwarn com.hyphenate.** 即:混淆规则. 自己没写过关于混淆打包的文章,在此补上. 下面了解Android studio环境下 项目混淆打包的操作. ------------------------------------------------------------

Android实例-手机安全卫士(二十五)-检查手机SIM卡是否变更

一.目标 1.利用广播接收者(手机重启时会发出相应广播)获取开机时的SIM序列号,与已经保存着的比较.根据比较结果完成相应操作. 2.(可选)强制在防盗设置向导第二步中要求绑定SIM卡后才可进行下一步设置. 3.实现从另一个Activity中获取数据,即点击”选择联系人“人后跳至新建的类(选择联系人)中,并从中获取联系人数据. 二.代码实现. 1.在程序包下新建包(取名com.example.mobilesafe.receiver),用于接收广播.在新建包下新建类(取名BootCompleteR

应用程序框架实战二十五:查询条件(规约模式应用)

前面已经做了一些准备工作,本篇将介绍查询条件的封装,它是规约模式的一个应用. 规约使用一个对象来封装谓词,我之前已经介绍过它在验证方面的应用,本篇是规约模式在查询方面的应用. 规约的强大之处在于,能够将一堆杂乱无章的条件判断或查询条件封装起来,以一个清晰的概念来表达,并使得这些谓词具备了可复用的能力. 首先在Util.Domains项目的Repositories目录中创建ICriteria接口,这个接口表示一个查询条件,代码如下. using System; using System.Linq.

Android实战技巧之二十五:模拟器如何重启?试试Genymotion!

需要测试Android 5.1系统重启后的一个功能,我手里的设备都没有这么高的版本,那么只好向模拟器求助了.就像大多网友一样,我遇到了一个大问题,如何将Android模拟器重启? 我有几个想法: * 长按模拟器的power kay,调出重启键.可惜的是系统默认只有poweroff一项. * adb reboot或者adb shell进入模拟器再执行reboot.尝试了几个模拟器,都没有成功,只听见我本子的cpu风扇在飞速的运转着. * 自己写一个重启的小应用做重启这件事.这是几年前我就想写的程序

Android学习路线(二十五)全面理解Android Navigation逻辑

应用导航的一致性是整体用户体验的重要组成部分,如果app的导航方式不一样,用户不能很快理解,那么这个应用会让用户有很大的挫败感,大大地影响了用户体验. Android 3.0后,系统像大家介绍了其在全局导航表现上的重大改变.全面地理解“Back”以及“Up”的导航效果以及意义,能够大大地减少用户的学习时间,用户在使用过程中很快能够学习如何在应用的各个界面间的切换. Android 2.3以及它之前的系统都是通过“Back”按钮来为app导航的.在Android 3.0后出现的Actionbar则

Android学习笔记二十五之ListView多布局实现

Android学习笔记二十五之ListView多布局实现 这一节是介绍ListView这个控件的最后一节,实现一个Item的多布局.像我们经常在用的各种即时通讯工具,QQ.微信等,假设他们的会话界面是ListView实现的,那么ListView就有多种Item布局,这一节,我们就来实现一个ListView的多种Item. 要实现ListView里面有多种Item,就要重写适配器的两个方法getViewTypeCount()和getItemViewType(int position),第一个方法是

android产品研发(二十二)-->android实用调试技巧

转载请标明出处:一片枫叶的专栏 上一篇文章中我们讲解了android UI优化方面的知识.我们讲解了android中的include.marge.ViewStub标签,在使用这些标签时可以简化我们的布局文件,优化组件绘制流程:讲解了android中的过度绘制相关知识点,通过优化我们的App过度绘制可以提高App的UI绘制流程与性能:我们还讲解了App中一些UI优化的小tips.更多关于android UI优化方面的知识可以参考我的:android产品研发(二十一)–>android中的UI优化