Android实现截屏功能(已适配Android11)

写在开头

为满足监听用户截屏并展示悬浮反馈入口的需求,对Android端的用户截屏功能进行了简单的调研。由于Android系统并没有提供截屏通知相关的API,所有需要我们自己利用系统能提供的相关特性变通实现。

通过学习,看到网上大概了提供了三种解决方案:

  1. 利用FileObserver监听某个目录中资源变化情况
  2. 利用ContentObserver监听图片资源的变化
  3. 监听截屏快捷按键 ( 由于厂商自定义Android系统的多样性,再加上快捷键的不同以及第三方应用,监听截屏快捷键这事基本不靠谱,可以直接忽略 )

这里我的实现是通过第二种方案解决,具体为什么不用1,3两种,下面的博客给了很好的解释,我不在累赘。
Android 截屏监听:如何实现截图分享功能?
但是实现方式略微不同,欢迎大家一起学习指正。

ScreenShotMaster
先给github地址,对于截屏思路有了解的同学可以直接看gitgub示例。

原理介绍

大家都知道,Android系统有一个媒体数据库,不管我们是相机拍摄的照片还是使用系统截屏截取的图片,系统都会把这张图片的详细信息加入到这个媒体数据库,并发出内容改变通知,所以我们可以利用内容观察者(ContentObserver)监听媒体数据库的变化,当数据库有变化时,获取最后插入的一条图片数据,如果该图片符合我们特定的规则,则认为被截屏了。

那么我们需要怎么做才能确定是截图呢?

  1. 监听截图的资源URI(MediaStore.Images.Media.EXTERNAL_CONTENT_URI)
  2. 因为我们要读取图片内容,所以我们需要读取SD卡的权限,android.permission.READ_EXTERNAL_STORAGE并动态申请。
  3. 获得图片信息后判断图片是否符合截图规则。

截图规则

网上绝大多数规则:

  1. 时间判断,图片的生成时间在开始监听之后, 并与当前时间相隔10秒内:开始监听后生成的图片才有意义,相隔10秒内说明是刚刚生成的。
  2. 路径判断,图片路径符合包含特定的关键词:这一点是关键,截屏图片的保存路径通常包含“screenshot”等截图string。

实现主要步骤(具体请移步Github)

###注册图片监听者

fun registerContentObserver() {
        if (contentObserver == null) {
            contentObserver =
                ScreenShotApplication.applicationContext.contentResolver.registerObserver(
                    MediaStore.Images.Media.EXTERNAL_CONTENT_URI
                ) {
                    //监听到截图后
                    _dataChanged.value = true
                }
        }
    }


/**
 * 利用ContentResolver监听照片数据的变化
 */
fun ContentResolver.registerObserver(
        uri: Uri,
        observer: (selfChange: Boolean) -> Unit
): ContentObserver {
    val contentObserver = object : ContentObserver(Handler()) {
        override fun onChange(selfChange: Boolean) {
            observer(selfChange)
        }
    }
    registerContentObserver(uri, true, contentObserver)
    return contentObserver
}

获取截图图片

因为适配Android11后,查询图片的SQL发生了变化,所以需要针对查询的方式有了一些改变,详见下面的代码和这个博客。

解析 Android 11 getContentResolver 获取多媒体图片

  /*
    *  获取截图图片
    */
    fun getScreentShotImage(bucketId: String? = null) {
        Thread {
            try {
                var data: ScreentShotInfo? = null
                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
                    //11 高版本获取图片信息
                    data = queryImagesP(bucketId)
                } else {
                    //低版本获取图片信息
                    data = queryImages(bucketId)
                }
                val imagePath = data.path?.toLowerCase()
                screenShoot.forEach {
                    if (imagePath?.contains(it)!! && (System.currentTimeMillis() / 1000 - data.addTime < 2)) {
                        _screentShotInfoData.postValue(data)
                        return@forEach
                    }
                }
            } catch (e: Exception) {
            }
        }.start()
    }

    /**
     * 只获取普通图片,不获取Gif
     */
    fun queryImages(bucketId: String?): ScreentShotInfo {
        val screentShotInfo = ScreentShotInfo()

        val uri = MediaStore.Files.getContentUri("external")
        val sortOrder = MediaStore.Images.Media._ID + " DESC limit 1 "
        var selection = (MediaStore.Files.FileColumns.MEDIA_TYPE + "="
                + MediaStore.Files.FileColumns.MEDIA_TYPE_IMAGE) +
                " AND " + MediaStore.Images.Media.MIME_TYPE + "=?" +
                " or " + MediaStore.Images.Media.MIME_TYPE + "=?"
        try {
            val data = ScreenShotApplication.applicationContext.contentResolver.query(
                uri,
                ScreenShotProjection,
                selection,
                imageType,
                sortOrder
            )

            if (data == null) {
                return screentShotInfo
            }

            if (data.moveToFirst()) {
                //查询数据
                val imageId: String =
                    data.getString(data.getColumnIndexOrThrow(ScreenShotProjection[0]))
                val imagePath: String =
                    data.getString(data.getColumnIndexOrThrow(ScreenShotProjection[1]))
                val imageSize: Long =
                    data.getLong(data.getColumnIndexOrThrow(ScreenShotProjection[2]))
                val imageWidth: Int =
                    data.getInt(data.getColumnIndexOrThrow(ScreenShotProjection[3]))
                val imageHeight: Int =
                    data.getInt(data.getColumnIndexOrThrow(ScreenShotProjection[4]))
                val imageMimeType: String =
                    data.getString(data.getColumnIndexOrThrow(ScreenShotProjection[5]))
                val imageAddTime: Long =
                    data.getLong(data.getColumnIndexOrThrow(ScreenShotProjection[6]))
                screentShotInfo.path = imagePath
                screentShotInfo.addTime = imageAddTime
            }

        } catch (e: Exception) {
            e.printStackTrace()
        }

        return screentShotInfo
    }

    /**
     * 只获取普通图片,不获取Gif(在Android11的机器中)
     * 在targetSdkVersion适配到30后  查询图片的Sql发生了变化
     */
    @RequiresApi(Build.VERSION_CODES.O)
    @WorkerThread
    fun queryImagesP(bucketId: String?): ScreentShotInfo {
        val screentShotInfo = ScreentShotInfo()
        val uri = MediaStore.Files.getContentUri("external")
        val sortOrder = MediaStore.Files.FileColumns._ID + " DESC"
        var selection = (MediaStore.Files.FileColumns.MEDIA_TYPE + "="
                + MediaStore.Files.FileColumns.MEDIA_TYPE_IMAGE) +
                " AND " + MediaStore.Images.Media.MIME_TYPE + "=?" +
                " or " + MediaStore.Images.Media.MIME_TYPE + "=?"

        val bundle = createSqlQueryBundle(selection, imageType, sortOrder, 1)

        try {
            val data = ScreenShotApplication.applicationContext.contentResolver.query(
                uri,
                ScreenShotProjection,
                bundle,
                null
            )

            if (data == null) {
                return screentShotInfo
            }

            if (data.moveToFirst()) {
                //查询数据
                val imageId: String =
                    data.getString(data.getColumnIndexOrThrow(ScreenShotProjection[0]))
                val imagePath: String =
                    data.getString(data.getColumnIndexOrThrow(ScreenShotProjection[1]))
                val imageSize: Long =
                    data.getLong(data.getColumnIndexOrThrow(ScreenShotProjection[2]))
                val imageWidth: Int =
                    data.getInt(data.getColumnIndexOrThrow(ScreenShotProjection[3]))
                val imageHeight: Int =
                    data.getInt(data.getColumnIndexOrThrow(ScreenShotProjection[4]))
                val imageMimeType: String =
                    data.getString(data.getColumnIndexOrThrow(ScreenShotProjection[5]))
                val imageAddTime: Long =
                    data.getLong(data.getColumnIndexOrThrow(ScreenShotProjection[6]))
                screentShotInfo.path = imagePath
                screentShotInfo.addTime = imageAddTime
            }

        } catch (e: Exception) {
            e.printStackTrace()
        }
        return screentShotInfo
    }

    /*
    * 创建Android11 所需要的bundle对象
    * */
    fun createSqlQueryBundle(
        selection: String,
        selectionArgs: Array<String>,
        sortOrder: String?, limitCount: Int = 0, offset: Int = 0
    ): Bundle? {
        if (selection == null && selectionArgs == null && sortOrder == null) {
            return null
        }
        val queryArgs = Bundle()
        if (selection != null) {
            queryArgs.putString(ContentResolver.QUERY_ARG_SQL_SELECTION, selection)
        }
        if (selectionArgs != null) {
            queryArgs.putStringArray(ContentResolver.QUERY_ARG_SQL_SELECTION_ARGS, selectionArgs)
        }
        if (sortOrder != null) {
            queryArgs.putString(ContentResolver.QUERY_ARG_SQL_SORT_ORDER, sortOrder)
        }
        queryArgs.putString(ContentResolver.QUERY_ARG_SQL_LIMIT, "$limitCount offset $offset")
        return queryArgs
    }

写在结尾

到此为止,实现了我们的需求。因为我们的市场是在海外,在bug检测后台发现了一些莫名其妙的空指针问题,所以代码中加了一些强制判空和try catch处理。后续有时间在优化,也希望大家可以提出宝贵的意见。

感谢

在开发和解决bug的过程中,提供帮助和解决思路的一些网站。

https://www.cnblogs.com/kaidarwang/p/6525878.html
https://blog.csdn.net/xietansheng/article/details/52692163
https://zhuanlan.zhihu.com/p/37011146
https://blog.csdn.net/wmyasw/article/details/108993757

  • 2
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
为了在Android应用程序中实现摄像头截图功能,可以按照以下步骤进行操作: 1.在AndroidManifest.xml文件中添加相机和存储权限: ```xml <uses-permission android:name="android.permission.CAMERA" /> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> ``` 2.在布局文件中添加一个SurfaceView用于预览摄像头捕获的图像: ```xml <SurfaceView android:id="@+id/surfaceView" android:layout_width="match_parent" android:layout_height="match_parent" /> ``` 3.在Activity中获取摄像头实例并启动预览: ```java private Camera mCamera; private SurfaceView mSurfaceView; private SurfaceHolder mSurfaceHolder; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); mSurfaceView = findViewById(R.id.surfaceView); mSurfaceHolder = mSurfaceView.getHolder(); mSurfaceHolder.addCallback(new SurfaceHolder.Callback() { @Override public void surfaceCreated(SurfaceHolder holder) { try { mCamera = Camera.open(); mCamera.setPreviewDisplay(holder); mCamera.startPreview(); } catch (IOException e) { e.printStackTrace(); } } @Override public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) { } @Override public void surfaceDestroyed(SurfaceHolder holder) { mCamera.stopPreview(); mCamera.release(); } }); } ``` 4.添加一个按钮用于截图,并在按钮点击事件中实现截图功能: ```java public void capture(View view) { mCamera.takePicture(null, null, new Camera.PictureCallback() { @Override public void onPictureTaken(byte[] data, Camera camera) { File pictureFile = getOutputMediaFile(); if (pictureFile == null) { return; } try { FileOutputStream fos = new FileOutputStream(pictureFile); fos.write(data); fos.close(); Toast.makeText(MainActivity.this, "截图成功", Toast.LENGTH_SHORT).show(); } catch (IOException e) { e.printStackTrace(); } mCamera.startPreview(); } }); } private File getOutputMediaFile() { File mediaStorageDir = new File(Environment.getExternalStoragePublicDirectory( Environment.DIRECTORY_PICTURES), "MyCameraApp"); if (!mediaStorageDir.exists()) { if (!mediaStorageDir.mkdirs()) { return null; } } String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss").format(new Date()); File mediaFile = new File(mediaStorageDir.getPath() + File.separator + "IMG_" + timeStamp + ".jpg"); return mediaFile; } ``` 5.在AndroidManifest.xml文件中添加文件读写权限: ```xml <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> ```

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值