Android10适配-限制对屏幕内容的访问

AndroidQ版本官方限制了对屏幕内容的访问。

为了保护用户的屏幕内容,Android 10 更改了 READ_FRAME_BUFFER、CAPTURE_VIDEO_OUTPUT 和 CAPTURE_SECURE_VIDEO_OUTPUT 权限的作用域,从而禁止以静默方式访问设备的屏幕内容。从 Android 10 开始,这些权限只能通过签名访问。
需要访问设备屏幕内容的应用应使用 MediaProjection API,此 API 会显示提示,要求用户同意访问。

如果还使用之前的逻辑访问屏幕内容的话,会报如下错误:

java.lang.SecurityException: Media projections require a foreground service of type ServiceInfo.FOREGROUND_SERVICE_TYPE_MEDIA_PROJECTION

那么如何获取屏幕内容呢?你需要以下几步。

1、请求用户授权

this.projectionManager = (MediaProjectionManager)activity.getSystemService(Context.MEDIA_PROJECTION_SERVICE);
activity.startActivityForResult(this.projectionManager.createScreenCaptureIntent(), 1000);

执行上面的代码,会弹出系统的提示用户弹框,选择“立即开始”或者“取消”后,会在onActivityResult中进行回调。

2、新建一个CaptureScreenService

@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
public class CaptureScreenService extends Service {
    private int mResultCode;
    private Intent mResultData;
    private MediaProjectionManager projectionManager;
    private MediaProjection mMediaProjection;

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        createNotificationChannel();
        mResultCode = intent.getIntExtra("code", -1);
        mResultData = intent.getParcelableExtra("data");

        this.projectionManager = (MediaProjectionManager)getSystemService(Context.MEDIA_PROJECTION_SERVICE);
        mMediaProjection = projectionManager.getMediaProjection(mResultCode, Objects.requireNonNull(mResultData));

        ScreenCaptureManager.getInstance().start(mMediaProjection);
        return super.onStartCommand(intent, flags, startId);
    }

    private void createNotificationChannel() {
        Notification.Builder builder = new Notification.Builder(this.getApplicationContext()); //获取一个Notification构造器
        Intent nfIntent = new Intent(this, MainActivity.class); //点击后跳转的界面,可以设置跳转数据

        builder.setContentIntent(PendingIntent.getActivity(this, 0, nfIntent, 0)) // 设置PendingIntent
                .setLargeIcon(BitmapFactory.decodeResource(this.getResources(), R.mipmap.ic_launcher)) // 设置下拉列表中的图标(大图标)
                //.setContentTitle("SMI InstantView") // 设置下拉列表里的标题
                .setSmallIcon(R.mipmap.ic_launcher) // 设置状态栏内的小图标
                .setContentText("is running......") // 设置上下文内容
                .setWhen(System.currentTimeMillis()); // 设置该通知发生的时间

        /*以下是对Android 8.0的适配*/
        //普通notification适配
        if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            builder.setChannelId("notification_id");
        }
        //前台服务notification适配
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            NotificationManager notificationManager = (NotificationManager)getSystemService(NOTIFICATION_SERVICE);
            NotificationChannel channel = new NotificationChannel("notification_id", "notification_name", NotificationManager.IMPORTANCE_LOW);
            notificationManager.createNotificationChannel(channel);
        }

        Notification notification = builder.build(); // 获取构建好的Notification
        notification.defaults = Notification.DEFAULT_SOUND; //设置为默认的声音
        startForeground(110, notification);

    }

    @Override
    public void onDestroy() {
    	super.onDestroy();
        stopForeground(true);
    }
}

3、注册CaptureScreenService,并添加foregroundServiceType属性。

<service android:name=".CaptureScreenService"
            android:enabled="true"
            android:foregroundServiceType="mediaProjection"/>

4、在activity的onActivityResult中启动前台服务

	@Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if(resultCode == RESULT_OK && requestCode == 1000) {
        	//判断系统版本选择不同处理方法
            if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
            	//启动前台服务
                service = new Intent(this, CaptureScreenService.class);
                service.putExtra("code", resultCode);
                service.putExtra("data", data);
                startForegroundService(service);
            }else {
                ScreenCaptureManager.getInstance().start(resultCode, data);
            }
        }
    }

5、在CaptureScreenService的onStartCommand执行访问屏幕内容的逻辑。

	@Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        createNotificationChannel();
        mResultCode = intent.getIntExtra("code", -1);
        mResultData = intent.getParcelableExtra("data");

        this.projectionManager = (MediaProjectionManager)getSystemService(Context.MEDIA_PROJECTION_SERVICE);
        mMediaProjection = projectionManager.getMediaProjection(mResultCode, Objects.requireNonNull(mResultData));

        ScreenCaptureManager.getInstance().start(mMediaProjection);
        return super.onStartCommand(intent, flags, startId);
    }

需要在前台服务启动后5s内调用startForeground方法,否则也会报错。
所以先执行createNotificationChannel(),启动startForeground。
获取屏幕内容的逻辑封装在ScreenCaptureManager内,如下:

   public void start(MediaProjection mediaProjection) {
        this.mediaProjection = mediaProjection;
        if (this.projectionManager != null) {
            this.initVirtualDisplay(this.activity);
            this.mediaProjection.registerCallback(new ScreenCaptureManager.MediaProjectionStopCallback(), (Handler)null);
        }
    }
    
    public void initVirtualDisplay(Activity activity) {
        DisplayMetrics metrics = activity.getResources().getDisplayMetrics();
        int density = metrics.densityDpi;
        Point size = new Point();
        activity.getWindowManager().getDefaultDisplay().getRealSize(size);
        int width = size.x;
        int height = size.y;
        this.imageReader = ImageReader.newInstance(width, height, 1, 1);
        this.imageReader.setOnImageAvailableListener(new ScreenCaptureManager.ImageAvailableListener(), (Handler)null);
        String virtualDisplayName = "Screenshot";
        int flags = 9;
        this.virtualDisplay = this.mediaProjection.createVirtualDisplay(virtualDisplayName, width, height, density, flags, this.imageReader.getSurface(), (VirtualDisplay.Callback)null, (Handler)null);
    }

在ScreenCaptureManager.ImageAvailableListener()的onImageAvailable(ImageReader reader)就可以拿到屏幕的bitmap信息了。

6、记得关闭服务。

	//这是CaptureScreenService的onDestroy()方法
 	@Override
    public void onDestroy() {
    	super.onDestroy();
        stopForeground(true);
    }

stopForeground中的参数表示:是否移除之前的通知。

	//这是activity的onDestroy()
	@Override
    protected void onDestroy() {
        super.onDestroy();
        //停止获取屏幕内容
        ScreenCaptureManager.getInstance().stop();
        //关闭服务
        Intent service = new Intent(this, CaptureScreenService.class);
        stopService(service);
    }

7、记得加上FOREGROUND_SERVICE权限。

<uses-permission android:name="android.permission.FOREGROUND_SERVICE"/>

8、compileSdkVersion设置为29,否则manifest中没有foregroundServiceType属性。

至此你就可以愉快的使用屏幕内容了!

demo地址:限制对屏幕内容的访问
参考文章:
MediaProjections in Android Q
Android Foreground Service (前台服务)
官方文档

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Android屏幕适配是指在开发Android应用时,考虑到不同屏幕尺寸、屏幕分辨率和像素密度的问题,以保证应用在各种Android设备上都能正常显示和运行。由于Android设备存在多种尺寸和分辨率,不同设备上同样的元素可能会呈现不同的效果,因此屏幕适配工作变得越来越重要。 为了进行屏幕适配,开发人员需要考虑以下几个方面: 1. 使用适当的尺寸单位:在Android开发中,可以使用不同的尺寸单位来适配不同的屏幕。常用的尺寸单位包括dp(density-independent pixels)和sp(scaled pixels)。使用dp作为尺寸单位可以使得元素在不同像素密度的设备上显示一致。 2. 使用限定符进行适配Android提供了多种限定符,如smallestWidth限定符,可以根据设备的最小宽度来进行适配。通过添加少量的dimens.xml文件,可以实现对不同屏幕尺寸的适配。 3. 使用可伸缩的布局:为了适配不同屏幕尺寸,可以使用可伸缩的布局,并使用权重属性来控制元素的大小和位置。这样可以确保在不同屏幕上元素的布局能够自适应调整。 4. 使用多布局文件:根据不同的屏幕尺寸和分辨率,可以使用不同的布局文件来适配不同的设备。通过在res目录中创建不同的布局文件夹,可以实现对不同屏幕适配。 总结来说,Android屏幕适配是为了确保应用在不同的Android设备上都能够正常显示和运行,需要考虑到屏幕尺寸、屏幕分辨率和像素密度等因素。通过使用适当的尺寸单位、限定符、可伸缩的布局和多布局文件等方法,可以实现对不同屏幕适配。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [Android屏幕尺寸适配常见方案smallestWidth](https://blog.csdn.net/cat_is_so_cute/article/details/124458433)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [【Android】之屏幕适配](https://blog.csdn.net/yang553566463/article/details/127029556)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值