Android截屏

最近由于公司项目需要实现手机的截屏,在实现过程中真的可以说是历经千辛万苦,各种问题层出不穷。现写这篇博客作为见证。

通过上网搜索截屏主要有如下几种方式

使用View.getDrawingCache()方式

通过该方法可以获取到当前activity的页面的bitmap,然后进行保存,可以说是最简单的实习方式。优点是不需要root,不过缺点也比较明显只能获取当前运行的activity,无法获取其他应用,也不能用到service后台截屏。

View view = activity.getWindow().getDecorView();
view.setDrawingCacheEnabled(true);
view.buildDrawingCache();
Bitmap b1 = view.getDrawingCache();

//获取状态栏高度
Rect frame = new Rect();
activity.getWindow().getDecorView().getWindowVisibleDisplayFrame(frame);
int statusBarHeight = frame.top;

System.out.println(statusBarHeight);//获取屏幕长和高

int width = activity.getWindowManager().getDefaultDisplay().getWidth();
int height = activity.getWindowManager().getDefaultDisplay().getHeight();//去掉标题栏
Bitmap b = Bitmap.createBitmap(b1, 0, statusBarHeight, width, height - statusBarHeight);
view.destroyDrawingCache();

bufferframe读取fb0

在手机的/dev/graphics目录下的fb0文件是负责屏幕渲染的帧缓存,网上有一些教程讲如何用c将手机中的fb0转换成bmp格式的图片,但是并不死所有的手机都支持,还与Android版本有关,而且手机必须要有root权限。

因为对c与c++不是很了解,我尝试着做了,但任然会有很多的问题,我也试过用java代码读取fb0,保存bitmap图片,但都失败了,保存的图片全部都是黑色的,真的是很悲伤。

反射

通过查看Android的源码,sdk是有截屏的代码,但是隐藏的,无法调用。在frameworks/base/packages/SystemUI/src/com/android/systemui/screenshot下面有两个文件,GlobalScreenshot与TakeScreenshotService。

我们主要查看GlobalScreenshot找到takeScreenshot方法

void takeScreenshot(Runnable finisher, boolean statusBarVisible, boolean navBarVisible) {  
       // We need to orient the screenshot correctly (and the Surface api seems to take screenshots  
       // only in the natural orientation of the device :!)  
       mDisplay.getRealMetrics(mDisplayMetrics);  
       float[] dims = {mDisplayMetrics.widthPixels, mDisplayMetrics.heightPixels};  
       float degrees = getDegreesForRotation(mDisplay.getRotation());  
       boolean requiresRotation = (degrees > 0);  
       if (requiresRotation) {  
           // Get the dimensions of the device in its native orientation  
           mDisplayMatrix.reset();  
           mDisplayMatrix.preRotate(-degrees);  
           mDisplayMatrix.mapPoints(dims);  
           dims[0] = Math.abs(dims[0]);  
           dims[1] = Math.abs(dims[1]);  
       }  
       // Take the screenshot  
       mScreenBitmap = SurfaceControl.screenshot((int) dims[0], (int) dims[1]);  
       if (mScreenBitmap == null) {  
           notifyScreenshotError(mContext, mNotificationManager);  
           finisher.run();  
           return;  
       }  

       if (requiresRotation) {  
           // Rotate the screenshot to the current orientation  
           Bitmap ss = Bitmap.createBitmap(mDisplayMetrics.widthPixels,  
                   mDisplayMetrics.heightPixels, Bitmap.Config.ARGB_8888);  
           Canvas c = new Canvas(ss);  
           c.translate(ss.getWidth() / 2, ss.getHeight() / 2);  
           c.rotate(degrees);  
           c.translate(-dims[0] / 2, -dims[1] / 2);  
           c.drawBitmap(mScreenBitmap, 0, 0, null);  
           c.setBitmap(null);  
           // Recycle the previous bitmap  
           mScreenBitmap.recycle();  
           mScreenBitmap = ss;  
       }  

       // Optimizations  
       mScreenBitmap.setHasAlpha(false);  
       mScreenBitmap.prepareToDraw();  

       // Start the post-screenshot animation  
       startAnimation(finisher, mDisplayMetrics.widthPixels, mDisplayMetrics.heightPixels,  
               statusBarVisible, navBarVisible);  
   }

关键是

mScreenBitmap = SurfaceControl.screenshot((int) dims[0], (int) dims[1]);

这个就是截屏的主要代码,这个地方在不同的Android版本中有一些差别
有Surface与SurfaceControl。

因此我们通过反射机制的代码

wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
                mDisplay = wm.getDefaultDisplay();
                mDisplayMatrix = new Matrix();
                mDisplayMetrics = new DisplayMetrics();
                // We need to orient the screenshot correctly (and the Surface api seems to take screenshots
                // only in the natural orientation of the device :!)
                mDisplay.getRealMetrics(mDisplayMetrics);
                float[] dims =
                {
                        mDisplayMetrics.widthPixels, mDisplayMetrics.heightPixels
                };
                float degrees = getDegreesForRotation(mDisplay.getRotation());
                boolean requiresRotation = (degrees > 0);
                if (requiresRotation)
                {
                    // Get the dimensions of the device in its native orientation
                    mDisplayMatrix.reset();
                    mDisplayMatrix.preRotate(-degrees);
                    mDisplayMatrix.mapPoints(dims);
                    dims[0] = Math.abs(dims[0]);
                    dims[1] = Math.abs(dims[1]);
                }

                Bitmap mScreenBitmap = screenShot((int) dims[0], (int) dims[1]);
                if (requiresRotation)
                {
                    // Rotate the screenshot to the current orientation
                    Bitmap ss = Bitmap.createBitmap(mDisplayMetrics.widthPixels, mDisplayMetrics.heightPixels,
                            Bitmap.Config.ARGB_8888);
                    Canvas c = new Canvas(ss);
                    c.translate(ss.getWidth() / 2, ss.getHeight() / 2);
                    c.rotate(degrees);
                    c.translate(-dims[0] / 2, -dims[1] / 2);
                    c.drawBitmap(mScreenBitmap, 0, 0, null);
                    c.setBitmap(null);
                    mScreenBitmap = ss;
                    if (ss != null && !ss.isRecycled())
                    {
                        ss.recycle();
                    }
                }

                // If we couldn't take the screenshot, notify the user
                if (mScreenBitmap == null)
                {
                    Toast.makeText(context, "screen shot fail", Toast.LENGTH_SHORT).show();
                }

                // Optimizations
                mScreenBitmap.setHasAlpha(false);
                mScreenBitmap.prepareToDraw();
private Bitmap screenShot(int width, int height)
    {
        Log.i(TAG, "android.os.Build.VERSION.SDK : " + android.os.Build.VERSION.SDK_INT);
        Class<?> surfaceClass = null;
        Method method = null;
        try
        {
            Log.i(TAG, "width : " + width);
            Log.i(TAG, "height : " + height);
            if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.JELLY_BEAN_MR2)
            {

                surfaceClass = Class.forName("android.view.SurfaceControl");
            }
            else
            {
                surfaceClass = Class.forName("android.view.Surface");
            }
            method = surfaceClass.getDeclaredMethod(METHOD_NAME, int.class, int.class);
            method.setAccessible(true);
            return (Bitmap) method.invoke(null, width, height);
        }
        catch (NoSuchMethodException e)
        {
            Log.e(TAG, e.toString());
        }
        catch (IllegalArgumentException e)
        {
            Log.e(TAG, e.toString());
        }
        catch (IllegalAccessException e)
        {
            Log.e(TAG, e.toString());
        }
        catch (InvocationTargetException e)
        {
            Log.e(TAG, e.toString());
        }
        catch (ClassNotFoundException e)
        {
            Log.e(TAG, e.toString());
        }
        return null;
    }

遗憾的是通过测试发现mScreenBitmap是null,也就是获取截屏失败了,原因可能是因为该SurfaceControl或Surface是隐藏的,虽然编译没报错,但是截屏会失败。

重新编译Android的sdk

这个是简单又粗暴的方法,我们可以将SurfaceControl或Surface类的隐藏去掉,生成我们自己的sdk。这是一个漫长的过程,可能会出现各种错误,我自己没有尝试。

通过adb命令

截屏命令主要有两个adb shell screencap -p xxx.png 或 adb shell screenshot xxx.png

screencap是从Android 2.3开始提供的一个系统级的截图工具,通过源码可以了解到screencap的实现方式,默认会从底层UI Surface去获取屏幕截图,如果失败则从linux kernel层的display framebuffer(/dev/graphics/fb0)去获取屏幕截图。

screenshot是从Android 4.0开始提供的另一个截图的工具, 通过源码可以发现screenshot则是直接读取/dev/graphics/fb0去获取屏幕的图像数据。

Process process = Runtime.getRuntime().exec("
/system/bin/screencap -p "+ fileFullPath)
Process process = Runtime.getRuntime().exec("
/system/bin/screencap "+ fileFullPath)

该adb命令是需要root权限的,因此手机没有权限截屏也是不能成功的。

最后


由于公司项目是盒子开发,所以说天生有root权限,因此实现截屏功能来说还是较为容易的,经历了从自带–>bufferframe–>adb 的过程。