Android4.0以上版本 后台service跨进程截取当前屏幕总结

本文介绍了在Android 4.0及以上版本如何实现在后台跨进程截取屏幕,包括4.0至4.2版本使用反射调用Surface.screenshot方法,4.3版本通过获取ROOT权限执行adb shell命令。对于没有ROOT权限的情况,提供了反射调用和使用开源工具包ShellUtils的方法。
摘要由CSDN通过智能技术生成

最近个人在做一个android的APP应用,涉及到了后台截取当前屏幕的需求,如果只是在自己应用里面截取大家应该都知道如何做,关键就在于后台跨进程截取其他APP当前运行的画面遇到了问题。

本人研究了一周左右的时间终于找到了一套相对而言比较可行的方案。

在这里需要感谢 李博Garvin文章参考了他的部分代码,这位牛人也是在这方面花了不少精力。

当然,我只是初学android开发,如果里面有不正确的言论还请大家及时指出。

截取自身Activity的界面(这里只是大概的贴出了主要的功能代码,详细的自己去网上找,多的是)

import java.io.File;  
import java.io.FileNotFoundException;  
import java.io.FileOutputStream;  
import java.io.IOException;  
   
import android.app.Activity;  
import android.graphics.Bitmap;  
import android.graphics.Rect;  
import android.view.View;  
   
public class ScreenShot {  
   
    private static Bitmap takeScreenShot(Activity activity) {  
        // View是你需要截图的View  
        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;  
   
        // 获取屏幕长和高  
        int width = activity.getWindowManager().getDefaultDisplay().getWidth();  
        int height = activity.getWindowManager().getDefaultDisplay()  
                .getHeight();  
        // 去掉标题栏  
        Bitmap b = Bitmap.createBitmap(b1, 0, statusBarHeight, width, height  
                - statusBarHeight);  
        view.destroyDrawingCache();  
        return b;  
    }  
   
    private static void savePic(Bitmap b, File filePath) {  
        FileOutputStream fos = null;  
        try {  
            fos = new FileOutputStream(filePath);  
            if (null != fos) {  
                b.compress(Bitmap.CompressFormat.PNG, 100, fos);  
                fos.flush();  
                fos.close();  
            }  
        } catch (FileNotFoundException e) {  
            // e.printStackTrace();  
        } catch (IOException e) {  
            // e.printStackTrace();  
        }  
    }  
   
    public static void shoot(Activity a, File filePath) {  
        if (filePath == null) {  
            return;  
        }  
        if (!filePath.getParentFile().exists()) {  
            filePath.getParentFile().mkdirs();  
        }  
        ScreenShot.savePic(ScreenShot.takeScreenShot(a), filePath);  
    }  


后台service跨进程截屏

1、android 4.0以下版本后台截屏

通过JNI方式截取屏幕,缺点是需要源码环境,详情请参考:

Android: How to Capture Screen in Gingerbread(2.3中实现截屏)_capture2.3比例尺-CSDN博客

Android 2.3实现截屏 - 精神邋遢的民工 - 博客园

2、android 4.0~4.2  版本后台截屏

这个可以通过反射android.view.Surface类的 screenshot 方法完成。

sc = Class.forName("android.view.Surface");
method=sc.getMethod("screenshot", new Class[] {int.class, int.class});  
Object o = method.invoke(sc, new Object[]{(int) dims[0],(int) dims[1]});  
mScreenBitmap =(Bitmap)o;

该方法优点无需ROOT,缺点是只支持4.0~4.2,4.3版本就不适用了。

或者通过 bufferframe读取fb0这种方法,但是估计需要ROOT权限

3、android 4.3版本后台截屏

4.3版本其实也存在screenshot方法,只不过他所属的整个类android.view.SurfaceControl都被hide了,所以通过一般的反射是无法调用的。

这里我开始参考了CSDN 李博Garvin 这位博主的做法(android4.3 截屏功能的尝试与失败分析),就是在PC端执行adb shell  /system/bin/screencap -p /sdcard/screenshot.png命令是可以正确截取屏幕的,但是换到android端执行该命令就会出错,截出来的图片大小为0.个人推测应该是android端执行shell命令出了什么问题。

在尝试了很多办法之后,终于在网上找到了解决办法。

原因就在于我们之前执行命令之前没有获取ROOT权限,就算是当前手机已经有了ROOT权限,我们也必须要在程序里再次获取ROOT权限,

也就是先执行下 Runtime.getRuntime().exec(“su”),然后在执行我们需要的命令。

然后我们就可以看到SD卡里面生成的正常的图片了。

这方面网上已经有了相应的工具包,我们开发起来可以事半功倍。

https://github.com/Trinea/android-common

而我们要做的只是把它里面的ShellUtils引用进来,然后调用execCommand(String command, boolean isRoot) 就可以了

具体使用方法请参见:Android Java执行Shell命令

总结一下: 4.0以上版本优先考虑ROOT方法,简单,通用。如果没有ROOT的话就判断系统版本,4.3以下的通过反射调用,4.3以上暂时还不清楚

你可以使用 `WindowManager` 和 `DisplayMetrics` 获取屏幕的宽度和高度,然后使用 `Bitmap.createBitmap()` 方法创建一个新的 Bitmap,并使用 `Canvas` 将当前屏幕顶部的正方形区域绘制到 Bitmap 上。以下是实现这个功能的示例代码: ```java // 获取屏幕的宽度和高度 WindowManager windowManager = (WindowManager) getSystemService(Context.WINDOW_SERVICE); DisplayMetrics displayMetrics = new DisplayMetrics(); windowManager.getDefaultDisplay().getMetrics(displayMetrics); int screenWidth = displayMetrics.widthPixels; int screenHeight = displayMetrics.heightPixels; // 计算正方形区域的位置和大小 int squareSize = screenHeight / 2; // 正方形的大小为屏幕高度的一半 int left = (screenWidth - squareSize) / 2; // 正方形左边界的位置 int top = 0; // 正方形顶部的位置 int right = left + squareSize; // 正方形右边界的位置 int bottom = top + squareSize; // 正方形底部的位置 // 创建一个新的 Bitmap Bitmap bitmap = Bitmap.createBitmap(squareSize, squareSize, Bitmap.Config.ARGB_8888); // 使用 Canvas 将正方形区域绘制到 Bitmap 上 Canvas canvas = new Canvas(bitmap); View view = getWindow().getDecorView().getRootView(); // 获取根 View view.draw(canvas); bitmap = Bitmap.createBitmap(bitmap, left, top, squareSize, squareSize); // 截取正方形区域 // 将 Bitmap 显示出来或者保存到本地 imageView.setImageBitmap(bitmap); // 将 Bitmap 显示到 ImageView 上 ``` 上面的代码中,首先获取屏幕的宽度和高度,然后计算出正方形区域的位置和大小。接着,创建一个新的 Bitmap,并使用 Canvas 将当前屏幕顶部的正方形区域绘制到 Bitmap 上。最后,截取正方形区域并将 Bitmap 显示到 ImageView 上。 需要注意的是,上面的代码必须在主线程中执行,否则会抛出异常。
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值