国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統(tǒng) > Android > 正文

android截圖事件監(jiān)聽的原理與實(shí)現(xiàn)

2019-12-12 02:26:04
字體:
供稿:網(wǎng)友

Android系統(tǒng)沒有對用戶截屏行為提供回調(diào)的api,所以我們只能走野路子來獲取用戶是否截屏了。一般大家都會采用如下兩種方法

1.監(jiān)聽截屏圖片所在目錄變化(FileObserver)

2.監(jiān)聽媒體庫的變化(ContentObserver) 

上面兩種方法均不是萬能的,需要結(jié)合使用才能達(dá)到良好的效果,首先看看如何監(jiān)控目錄

在android中,我們可以通過FileObserver來監(jiān)聽目錄變化,先來看看如何使用

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);     }   } }; 

我們對指定目錄的指定事件監(jiān)聽即可,當(dāng)事件被觸發(fā)時onEvent會回調(diào)。這里我們只關(guān)心目錄中有沒有新的文件生成。

坑1:在實(shí)踐中發(fā)現(xiàn),并不是所有手機(jī)都允許如此監(jiān)聽或者說都能收到回調(diào)。有的手機(jī)上面無法收到CREATE事件,但是可以收到其他事件。

我還發(fā)現(xiàn),有的時候收到的事件并沒有在FileObserver中定義,比如32768!下面是Linux中相應(yīng)event對應(yīng)的含義,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:不同手機(jī),監(jiān)聽的目錄并不一致。小米需要監(jiān)聽Environment.DIRECTORY_DCIM,其他監(jiān)聽Environment.DIRECTORY_PICTURES即可。

關(guān)于FileObserver這里再多說兩句,F(xiàn)ileObserver無法進(jìn)行遞歸監(jiān)聽,也就是說,我們監(jiān)聽的文件夾中如果有子文件夾,并且我們想知道其中變化,這種方式是不可行的。需要手動對子文件進(jìn)行操作。

另外,當(dāng)我們監(jiān)聽的目錄/文件被刪除后又重新建立了一個同名的目錄/文件,之前的FileObserver不會繼續(xù)工作,需要重新設(shè)置監(jiān)聽才行。

還要注意,F(xiàn)ileObserver回調(diào)并不在主線程中,而是在FileObserver線程中。

鑒于上述原因,我們還要使用方法2,監(jiān)聽媒體庫變化。這個方法使用ContentObserver即可。

private static final ContentObserver CONTENT_OBSERVER = new ContentObserver(HANDLER) {   @Override   public void onChange(boolean selfChange, Uri uri) {     //記得先檢查讀文件的權(quán)限     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:實(shí)踐中發(fā)現(xiàn),并不是所有手機(jī)都是監(jiān)聽相同的Uri,有的帶數(shù)字,有的不帶。

坑4:查詢數(shù)據(jù)庫時記得按MediaStore.MediaColumns.DATE_ADDED字段排序,注意,這個時間單位是秒,不是毫秒

坑5:即使排了序,你拿到的仍然有可能不是正確的,在魅族E2上面出現(xiàn)了這個問題。但是當(dāng)我刪除了魅族E2截圖文件夾之后,一切又恢復(fù)正常了……這里我做了一個簡單的判斷,如何DATE_ADDED和當(dāng)前時間相差兩秒以內(nèi),那么從數(shù)據(jù)庫查出的這條數(shù)據(jù)我視為有效

坑6:當(dāng)用戶刪除了截圖文件夾的時候,媒體庫此時會更新,所以此時onChange會收到大量回調(diào),所以這里需要判斷判斷文件是否存在。

可能有人會問,為什么不直接用第二種方法?

原因有2,首先從坑5可以看出第二種方法也并非100%有效,其次,這種方法速度很慢,通常會有2-3秒的延遲。而第一種方法如果有效,通常都會比后者快很多。

好了,障礙基本掃清,下面開始融合兩種方法

首先使用成員變量記錄截圖文件路徑

private static String sScreenshotPath; 

當(dāng)方法1或者方法2收到結(jié)果時,用收到的結(jié)果與sScreenshotPath對比,如果是同一個文件,那么就無需再次通知了,否則則進(jìn)行通知。

邏輯太簡單,代碼就不寫了。但是實(shí)際情況是不會這么樂觀的。

坑7:在實(shí)踐中發(fā)現(xiàn),有的系統(tǒng)不直接保存截圖,而是先生成一個隱藏文件,比如叫.截圖.jpg,然后再修改文件名(去掉“.”)。這種情況下,我們可能就會收到兩次用戶截圖事件的回調(diào)(方法1和方法2都可能收到回調(diào)),但實(shí)際用戶只截了一次。

這里我做了一個特殊處理,在判斷是否是同一個文件時,只判斷文件名,而不去管文件的完整路徑也不管文件是否隱藏(也就是不比較文件名前面的“.”)

//僅靠文件名而不是全路徑判斷是否為同一個截圖文件,因為有些系統(tǒng)對截圖有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截圖事件監(jiān)聽基本結(jié)束,由于測試機(jī)器有限,故無法保證上述方法萬無一失。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持武林網(wǎng)。

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 永德县| 涟源市| 柏乡县| 平阳县| 阿拉善左旗| 偏关县| 兴海县| 华池县| 苗栗县| 龙江县| 台安县| 莒南县| 汝州市| 漠河县| 长阳| 苏尼特左旗| 阜城县| 新安县| 新兴县| 阜城县| 南溪县| 榆树市| 景泰县| 乐亭县| 汝南县| 陆良县| 锦屏县| 左权县| 平远县| 即墨市| 吉林省| 秦安县| 惠安县| 喀喇沁旗| 博野县| 普陀区| 获嘉县| 湖南省| 北流市| 民县| 翁牛特旗|