android截图事件监听的原理与实现

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截图事件监听基本结束,由于测试机器有限,故无法保证上述方法万无一失。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android中通过view方式获取当前Activity的屏幕截图实现方法

    此方法是通过view的方式获取当前activity的屏幕截图,并不是framebuffer的方式,所以有一定的局限性.但是这种方法相对简单,容易理解. 首先通过下面的函数获取Bitmap格式的屏幕截图: 复制代码 代码如下: public Bitmap myShot(Activity activity) { // 获取windows中最顶层的view View view = activity.getWindow().getDecorView(); view.buildDrawingCache()

  • Android 截图功能源码的分析

    Android 截图功能源码的分析 一般没有修改rom的android原生系统截图功能的组合键是音量减+开机键:今天我们从源码角度来分析截图功能是如何在源码中实现的. 在android系统中,由于我们的每一个Android界面都是一个Activity,而界面的显示都是通过Window对象实现的,每个Window对象实际上都是PhoneWindow的实例,而每个PhoneWindow对象都对应一个PhoneWindowManager对象,当我们在Activity界面执行按键操作的时候,在将按键的处

  • Android实现截图和分享功能的代码

    先给大家展示下效果图吧 直接上代码: xml的布局: <Button android:id="@+id/btn_jp" android:layout_marginTop="10dip" android:layout_width="match_parent" android:layout_height="wrap_content" android:gravity="center" android:tex

  • Android截屏截图的几种方法总结

    Android截屏 Android截屏的原理:获取具体需要截屏的区域的Bitmap,然后绘制在画布上,保存为图片后进行分享或者其它用途 一.Activity截屏 1.截Activity界面(包含空白的状态栏) /** * 根据指定的Activity截图(带空白的状态栏) * * @param context 要截图的Activity * @return Bitmap */ public static Bitmap shotActivity(Activity context) { View vie

  • Android仿银行客户签名并且保存签名的截图文件并命名为本地时间

    首先需要一个自定义view用来签字使用,可以修改颜色和画笔的粗细,可以擦拭重新画 package com.android.tcm.view; import android.content.Context; import android.graphics.Bitmap; import android.graphics.Bitmap.Config; import android.graphics.Canvas; import android.graphics.Color; import androi

  • Android屏幕及view的截图实例详解

    Android屏幕及view的截图实例详解 屏幕可见区域的截图 整个屏幕截图的话可以用View view = getWindow().getDecorView(); public static Bitmap getNormalViewScreenshot(View view) { view.setDrawingCacheEnabled(true); view.buildDrawingCache(); return view.getDrawingCache(); } scrollview的整体截屏

  • Android中如何获取视频文件的截图、缩略图

    背景 公司最近要求给我负责的APP加上视频录制和发布的功能,我简单的完成了基本的录制和视频压缩功能,后来发现发布接口需要上传视频的截图,网上搜索了一下资料,在这里整理一下. 代码实现 /** * 获取视频文件截图 * * @param path 视频文件的路径 * @return Bitmap 返回获取的Bitmap */ public static Bitmap getVideoThumb(String path) { MediaMetadataRetriever media = new Me

  • android截图事件监听的原理与实现

    Android系统没有对用户截屏行为提供回调的api,所以我们只能走野路子来获取用户是否截屏了.一般大家都会采用如下两种方法 1.监听截屏图片所在目录变化(FileObserver) 2.监听媒体库的变化(ContentObserver) 上面两种方法均不是万能的,需要结合使用才能达到良好的效果,首先看看如何监控目录 在android中,我们可以通过FileObserver来监听目录变化,先来看看如何使用 private static final File DIRECTORY_PICTURES

  • Android 触摸事件监听(Activity层,ViewGroup层,View层)详细介绍

    Android不同层次的触摸事件监听 APP开发中,经常会遇到有关手势处理的操作,比如向右滑动返回上一个页面.关于触摸事件的处理,我们可以大概处理在不同的层次上. Activity层:可以看做触摸事件获取的最顶层 ViewGroup层:ViewGroup层可以自主控制是否让子View获取触摸事件 View层:可以决定自己是否真正的消费触摸事件,如果不消费抛给上层ViewGroup Activity级别的手势监听:(右滑动返回上层界面) Activity层手势监听的使用场景:一般用于当前页面中没有

  • Android App内监听截图加二维码功能代码

    Android截屏功能是一个常用的功能,可以方便的用来分享或者发送给好友,本文介绍了如何实现app内截屏监控功能,当发现用户在我们的app内进行了截屏操作时,进行对图片的二次操作,例如添加二维码,公司logo等一系列*. 项目地址 测试截图: 截屏原理 android系统并没有提供截屏通知相关的API,需要我们自己利用系统能提供的相关特性变通实现.Android系统有一个媒体数据库,每拍一张照片,或使用系统截屏截取一张图片,都会把这张图片的详细信息加入到这个媒体数据库,并发出内容改变通知,我们可

  • Android开发之button事件监听简单实例

    本文实例讲述了Android开发之button事件监听用法.分享给大家供大家参考.具体如下: 事件监听的listener,有以下几种方式: 1.声明一个普通的class,实现OnClickListener接口,然后在button的setOnClickListener中new该类的一个对象. 2.使用匿名内部类,直接 btn.setOnClickListener(new OnClickListener() { @Override public void onClick(View arg0) { S

  • Android TextView实现带链接文字事件监听的三种常用方式示例

    本文实例讲述了Android TextView实现带链接文字事件监听的三种常用方式.分享给大家供大家参考,具体如下: /** * TextView实现文字链接跳转功能 * @description: * @author ldm * @date 2016-4-21 下午4:34:05 */ public class TextViewLinkAct extends Activity { private TextView tv_3; private TextView tv_4; @Override p

  • 详解Android截屏事件监听

    1. 前言 Android系统没有直接对截屏事件监听的接口,也没有广播,只能自己动手来丰衣足食,一般有三种方法. 利用FileObserver监听某个目录中资源变化情况 利用ContentObserver监听全部资源的变化 监听截屏快捷按键 由于厂商自定义Android系统的多样性,再加上快捷键的不同以及第三方应用,监听截屏快捷键这事基本不靠谱,可以直接忽略. 本文使用的测试手机,一加2(One Plus 2). 2. FileObserver 添加权限: <uses-permission an

  • Android编程双重单选对话框布局实现与事件监听方法示例

    本文实例讲述了Android编程双重单选对话框布局实现与事件监听方法.分享给大家供大家参考,具体如下: 首先是自定义XML布局代码: <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_pare

  • 手把手教你Android全局触摸事件监听

    Android系统全局触摸事件监听 Android触摸全局监听指的是调用监听后在任何界面都能获取到触摸事件. 要实现这个功能必须要修改源码添加新的接口,因为系统默认是不暴露这个方法的. 源码 监听系统全局触摸事件的类和相关代码: frameworks\base\services\core\java\com\android\server\wm\WindowManagerService.java @Override public void registerPointerEventListener(P

  • Android 中的监听和按键处理详情

    目录 各种监听 按键处理 onKeyDown() onBackPressed() Fragment中监听Back返回键 各种监听 我们来练习下各种监听.我们在 TextView 上添加了触摸监听,在 Button 上添加了长按监听,在 Spinner 下拉框选项发生变化的时候添加了监听,在 ListView 选中选项时增加了监听. xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

  • JS 事件绑定、事件监听、事件委托详细介绍

    在JavaScript的学习中,我们经常会遇到JavaScript的事件机制,例如,事件绑定.事件监听.事件委托(事件代理)等.这些名词是什么意思呢,有什么作用呢? 事件绑定 要想让 JavaScript 对用户的操作作出响应,首先要对 DOM 元素绑定事件处理函数.所谓事件处理函数,就是处理用户操作的函数,不同的操作对应不同的名称. 在JavaScript中,有三种常用的绑定事件的方法: 在DOM元素中直接绑定: 在JavaScript代码中绑定: 绑定事件监听函数. 在DOM中直接绑定事件

随机推荐