Android系统没有对用户截屏行为提供回调的api,所以我们只能走野路子来获取用户是否截屏了。一般大家都会采用如下两种方法

1.监听截屏图片所在目录变化(FileObserver)
2.监听媒体库的变化(ContentObserver)
上面两种方法均不是万能的,需要结合使用才能达到良好的效果,首先看看如何监控目录
在android中,我们可以通过FileObserver来监听目录变化,先来看看如何使用
private static final File DIRECTORY_PICTURES = new File(Environment.getExternalStorageDirectory(), Environment.DIRECTORY_PICTURES);
private static final File DIRECTORY_DCIM = new File(Environment.getExternalStorageDirectory(), Environment.DIRECTORY_DCIM);
if (manufacturer.equalsIgnoreCase("xiaomi")) {
DIRECTORY_SCREENSHOT = new File(DIRECTORY_DCIM, "Screenshots");
} else {
DIRECTORY_SCREENSHOT = new File(DIRECTORY_PICTURES, "Screenshots");
}
FILE_OBSERVER = new FileObserver(DIRECTORY_SCREENSHOT.getPath(), FileObserver.ALL_EVENTS) {
@Override
public void onEvent(int event, String path) {
if (event == FileObserver.CREATE) {
String newPath = new File(DIRECTORY_SCREENSHOT, path).getAbsolutePath();
Log.d(TAG, "path: " + newPath);
}
}
};
我们对指定目录的指定事件监听即可,当事件被触发时onEvent会回调。这里我们只关心目录中有没有新的文件生成。
坑1:在实践中发现,并不是所有手机都允许如此监听或者说都能收到回调。有的手机上面无法收到CREATE事件,但是可以收到其他事件。
我还发现,有的时候收到的事件并没有在FileObserver中定义,比如32768!下面是Linux中相应event对应的含义,32768=IN_IGNORED,但是为什么会ignore,并不清楚。
http://rswiki.csie.org/lxr/http/source/include/linux/inotify.h?a=m68k#L45
还遇到过1073741856(1073741856 = 0x40000000 | 0x20,即IN_OPEN | IN_ISDIR)和1073741840(1073741840 = 0x40000000 | 0x10,即IN_CLOSE_NOWRITE | IN_ISDIR)。
坑2:不同手机,监听的目录并不一致。小米需要监听Environment.DIRECTORY_DCIM,其他监听Environment.DIRECTORY_PICTURES即可。
关于FileObserver这里再多说两句,FileObserver无法进行递归监听,也就是说,我们监听的文件夹中如果有子文件夹,并且我们想知道其中变化,这种方式是不可行的。需要手动对子文件进行操作。
另外,当我们监听的目录/文件被删除后又重新建立了一个同名的目录/文件,之前的FileObserver不会继续工作,需要重新设置监听才行。
还要注意,FileObserver回调并不在主线程中,而是在FileObserver线程中。
鉴于上述原因,我们还要使用方法2,监听媒体库变化。这个方法使用ContentObserver即可。
private static final ContentObserver CONTENT_OBSERVER = new ContentObserver(HANDLER) {
@Override
public void onChange(boolean selfChange, Uri uri) {
//记得先检查读文件的权限
ContentResolver resolver = GeneralInfoHelper.getContext().getContentResolver();
if (uri.toString().matches(MediaStore.Images.Media.EXTERNAL_CONTENT_URI + "(/\\d+)?")) {
Cursor cursor = resolver.query(uri, PROJECTION, null, null, MediaStore.MediaColumns.DATE_ADDED + " DESC");
if (cursor != null && cursor.moveToFirst()) {
//完整路径
String newPath = cursor.getString(cursor.getColumnIndex(MediaStore.MediaColumns.DATA));
File file = new File(newPath);
//file.exists() 判断文件是否存在
}
if (cursor != null) {
cursor.close();
}
}
}
};
getContentResolver().registerContentObserver(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, true, CONTENT_OBSERVER);
坑3:实践中发现,并不是所有手机都是监听相同的Uri,有的带数字,有的不带。
坑4:查询数据库时记得按MediaStore.MediaColumns.DATE_ADDED字段排序,注意,这个时间单位是秒,不是毫秒
坑5:即使排了序,你拿到的仍然有可能不是正确的,在魅族E2上面出现了这个问题。但是当我删除了魅族E2截图文件夹之后,一切又恢复正常了……这里我做了一个简单的判断,如何DATE_ADDED和当前时间相差两秒以内,那么从数据库查出的这条数据我视为有效
坑6:当用户删除了截图文件夹的时候,媒体库此时会更新,所以此时onChange会收到大量回调,所以这里需要判断判断文件是否存在。
可能有人会问,为什么不直接用第二种方法?
原因有2,首先从坑5可以看出第二种方法也并非100%有效,其次,这种方法速度很慢,通常会有2-3秒的延迟。而第一种方法如果有效,通常都会比后者快很多。
好了,障碍基本扫清,下面开始融合两种方法
首先使用成员变量记录截图文件路径
private static String sScreenshotPath;
当方法1或者方法2收到结果时,用收到的结果与sScreenshotPath对比,如果是同一个文件,那么就无需再次通知了,否则则进行通知。
逻辑太简单,代码就不写了。但是实际情况是不会这么乐观的。
坑7:在实践中发现,有的系统不直接保存截图,而是先生成一个隐藏文件,比如叫.截图.jpg,然后再修改文件名(去掉“.”)。这种情况下,我们可能就会收到两次用户截图事件的回调(方法1和方法2都可能收到回调),但实际用户只截了一次。
这里我做了一个特殊处理,在判断是否是同一个文件时,只判断文件名,而不去管文件的完整路径也不管文件是否隐藏(也就是不比较文件名前面的“.”)
//仅靠文件名而不是全路径判断是否为同一个截图文件,因为有些系统对截图有move操作
private static boolean isSameFile(String newPath) {
if (TextUtils.isEmpty(sScreenshotPath)) {
return false;
}
return TextUtils.equals(removePrefixDot(new File(sScreenshotPath).getName()), removePrefixDot(new File(newPath).getName()));
}
private static String removePrefixDot(@NonNull String filename) {
if (filename.startsWith(".")) {
return filename.substring(1);
}
return filename;
}
至此,android截图事件监听基本结束,由于测试机器有限,故无法保证上述方法万无一失。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# android
# 截图监听
# android截图事件监听
# Native.js获取监听开关等操作Android蓝牙设备实例代码
# native.js获取手机硬件基本信息实例代码android版
# Dcloud的native.js直接拨打电话Android实例代码
# DCloud的native.js调用系统分享实例Android版代码
# Android中通过view方式获取当前Activity的屏幕截图实现方法
# Android中如何获取视频文件的截图、缩略图
# Android模拟器中窗口截图存成文件实现思路及代码
# 详解有关Android截图与录屏功能的学习
# Android实现截图和分享功能的代码
# Android获取常用辅助方法(获取屏幕高度、宽度、密度、通知栏高度、截图)
# Android实现拍照截图功能
# Android屏幕及view的截图实例详解
# Android截屏截图的几种方法总结
# Android实现截图分享qq 微信功能
# Android实现从相册截图的功能
# Android 中WebView 截图的实现方式
# Android App内监听截图加二维码功能代码
# Android 5.0及以上编程实现屏幕截图功能的方法
# Android仿银行客户签名并且保存签名的截图文件并命名为本地时间
# Android 截图功能源码的分析
# Android使用WebView实现截图分享功能
# Native.js屏幕截图实例代码
# 回调
# 两种
# 种方法
# 递归
# 在实践中
# 是否存在
# 不直接
# 魅族
# 都是
# 判断是否
# 就会
# 是在
# 会有
# 好了
# 我还
# 有可能
# 就不
# 都能
# 中有
# 两次
相关文章:
如何选择长沙网站建站模板?H5响应式与品牌定制哪个更优?
建站主机系统SEO优化与智能配置核心关键词操作指南
建站之星IIS配置教程:代码生成技巧与站点搭建指南
建站主机服务器选型指南与性能优化方案解析
建站之星免费模板:自助建站系统与智能响应式一键生成
香港服务器租用费用高吗?如何避免常见误区?
如何用腾讯建站主机快速创建免费网站?
武汉网站制作费用多少,在武汉武昌,建面100平方左右的房子,想装暖气片,费用大概是多少啊?
电视网站制作tvbox接口,云海电视怎样自定义添加电视源?
宝塔建站无法访问?如何排查配置与端口问题?
网站制作话术技巧,网站推广做的好怎么话术?
建站主机选购指南与交易推荐:核心配置解析
如何用y主机助手快速搭建网站?
小程序网站制作需要准备什么资料,如何制作小程序?
哈尔滨网站建设策划,哈尔滨电工证查询网站?
网页设计与网站制作内容,怎样注册网站?
c# await 一个已经完成的Task会发生什么
如何快速查询网址的建站时间与历史轨迹?
建站之星收费标准详解:套餐费用及年费价格表一览
广州网站制作的公司,现在专门做网站的公司有没有哪几家是比较好的,性价比高,模板也多的?
如何在企业微信快速生成手机电脑官网?
如何通过多用户协作模板快速搭建高效企业网站?
如何快速生成高效建站系统源代码?
再谈Python中的字符串与字符编码(推荐)
为什么Go需要go mod文件_Go go mod文件作用说明
深圳网站制作培训,深圳哪些招聘网站比较好?
c# 在高并发下使用反射发射(Reflection.Emit)的性能
制作门户网站的参考文献在哪,小说网站怎么建立?
贸易公司网站制作流程,出口贸易网站设计怎么做?
潮流网站制作头像软件下载,适合母子的网名有哪些?
制作国外网站的软件,国外有哪些比较优质的网站推荐?
如何正确下载安装西数主机建站助手?
学生网站制作软件,一个12岁的学生写小说,应该去什么样的网站?
建站之星如何修改网站生成路径?
相亲简历制作网站推荐大全,新相亲大会主持人小萍萍资料?
如何使用Golang安装API文档生成工具_快速生成接口文档
寿县云建站:智能SEO优化与多行业模板快速上线指南
清单制作人网站有哪些,近日“兴风作浪的姑奶奶”引起很多人的关注这是什么事情?
官网网站制作腾讯审核要多久,联想路由器newifi官网
如何在IIS中新建站点并解决端口绑定冲突?
网页制作模板网站推荐,网页设计海报之类的素材哪里好?
如何在Golang中实现微服务服务拆分_Golang微服务拆分与接口管理方法
建站之星图片链接生成指南:自助建站与智能设计教程
怎么将XML数据可视化 D3.js加载XML
表情包在线制作网站免费,表情包怎么弄?
如何通过网站建站时间优化SEO与用户体验?
如何通过虚拟主机空间快速建站?
如何在西部数码注册域名并快速搭建网站?
如何快速搭建高效可靠的建站解决方案?
如何撰写建站申请书?关键要点有哪些?
*请认真填写需求信息,我们会在24小时内与您取得联系。