詳解Android截屏事件監(jiān)聽
1. 前言
Android系統(tǒng)沒有直接對截屏事件監(jiān)聽的接口,也沒有廣播,只能自己動手來豐衣足食,一般有三種方法。
- 利用FileObserver監(jiān)聽某個目錄中資源變化情況
- 利用ContentObserver監(jiān)聽全部資源的變化
- 監(jiān)聽截屏快捷按鍵
由于廠商自定義Android系統(tǒng)的多樣性,再加上快捷鍵的不同以及第三方應(yīng)用,監(jiān)聽截屏快捷鍵這事基本不靠譜,可以直接忽略。
本文使用的測試手機(jī),一加2(One Plus 2)。
2. FileObserver
添加權(quán)限:
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
代碼示例:
public class ScreenshotActivity extends AppCompatActivity { private final String TAG = "Screenshot"; private static final String PATH = Environment.getExternalStorageDirectory() + File.separator + Environment.DIRECTORY_PICTURES + File.separator + "Screenshots" + File.separator; protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_screenshot); mFileObserver = new CustomFileObserver(PATH); } @Override protected void onResume() { super.onResume(); mFileObserver.startWatching(); Log.d(TAG, PATH); } @Override protected void onStop() { super.onStop(); mFileObserver.stopWatching(); } /** * 目錄監(jiān)聽器 */ private class CustomFileObserver extends FileObserver { private String mPath; public CustomFileObserver(String path) { super(path); this.mPath = path; } public CustomFileObserver(String path, int mask) { super(path, mask); this.mPath = path; } @Override public void onEvent(int event, String path) { Log.d(TAG, path + " " + event); // 監(jiān)聽到事件,做一些過濾去重處理操作 } } }
打印的日志:
一加2
D/Screenshot: Screenshot_2016-12-16-17-49-18.png 256 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 32 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 2 D/Screenshot: Screenshot_2016-12-16-17-49-18.png 8
三星 S4
D/Screenshot: Screenshot_2016-12-16-19-01-08.png 256 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 32 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 2 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 8 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 32 D/Screenshot: Screenshot_2016-12-16-19-01-08.png 16
可以通過指定構(gòu)造函數(shù)中的mask,監(jiān)聽某一個事件。
類型 | int值 | 說明 |
---|---|---|
FileObserver.ACCESS | 1 | 讀取某個文件 |
FileObserver.MODIFY | 2 | 向某個文件寫入數(shù)據(jù) |
FileObserver.ATTRIB | 4 | 文件的屬性被修改(權(quán)限/日期/擁有者) |
FileObserver.CLOSE_WRITE | 8 | 寫入數(shù)據(jù)后關(guān)閉 |
FileObserver.CLOSE_NOWRITE | 16 | 只讀模式打開文件后關(guān)閉 |
FileObserver.OPEN | 32 | 打開某個文件 |
FileObserver.MOVED_FROM | 64 | 有文件或者文件夾從被監(jiān)聽的文件夾中移走 |
FileObserver.MOVED_TO | 128 | 有文件或者文件夾移動到被監(jiān)聽的文件夾 |
FileObserver.CREATE | 256 | 文件或者文件夾被創(chuàng)建 |
FileObserver.DELETE | 512 | 文件被刪除 |
FileObserver.DELETE_SELF | 1024 | 被監(jiān)聽的文件或者目錄被刪除 |
FileObserver.MOVE_SELF | 2048 | 被監(jiān)聽的文件或者目錄被移走 |
幾點注意事項:
- 每一次截屏,有多個事件回調(diào)
- 每一次截屏,不同的手機(jī),事件回調(diào)可能有些不同,參考上述日志
- 不同的手機(jī),默認(rèn)截屏圖片儲存的文件夾可能不同
- FileObserver只能監(jiān)聽文件夾中子文件和子文件夾的變化情況,不能監(jiān)聽子文件夾內(nèi)部的資源變化
- 需要<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>權(quán)限,否則可能收不到事件
基于第三點和第四點,這種方法并不能適用于所有的機(jī)型。
注意:如果自己寫Demo沒有收到事件,檢查一下權(quán)限和監(jiān)聽的目錄
3. ContentObserver
ContentObserver用來監(jiān)聽指定uri的所有資源變化,我們可以用它來監(jiān)聽圖片資源變化情況,然后做過濾。
添加權(quán)限
<uses-permission android:name="MediaStore.Images.Media.INTERNAL_CONTENT_URI"/> <uses-permission android:name="MediaStore.Images.Media.EXTERNAL_CONTENT_URI"/>
代碼示例:
public class ScreenshotActivity extends AppCompatActivity { private static final String[] KEYWORDS = { "screenshot", "screen_shot", "screen-shot", "screen shot", "screencapture", "screen_capture", "screen-capture", "screen capture", "screencap", "screen_cap", "screen-cap", "screen cap" }; /** 讀取媒體數(shù)據(jù)庫時需要讀取的列 */ private static final String[] MEDIA_PROJECTIONS = { MediaStore.Images.ImageColumns.DATA, MediaStore.Images.ImageColumns.DATE_TAKEN, }; /** 內(nèi)部存儲器內(nèi)容觀察者 */ private ContentObserver mInternalObserver; /** 外部存儲器內(nèi)容觀察者 */ private ContentObserver mExternalObserver; private HandlerThread mHandlerThread; private Handler mHandler; protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_screenshot); mHandlerThread = new HandlerThread("Screenshot_Observer"); mHandlerThread.start(); mHandler = new Handler(mHandlerThread.getLooper()); // 初始化 mInternalObserver = new MediaContentObserver(MediaStore.Images.Media.INTERNAL_CONTENT_URI, mHandler); mExternalObserver = new MediaContentObserver(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, mHandler); // 添加監(jiān)聽 this.getContentResolver().registerContentObserver( MediaStore.Images.Media.INTERNAL_CONTENT_URI, false, mInternalObserver ); this.getContentResolver().registerContentObserver( MediaStore.Images.Media.EXTERNAL_CONTENT_URI, false, mExternalObserver ); } protected void onDestroy() { super.onDestroy(); // 注銷監(jiān)聽 this.getContentResolver().unregisterContentObserver(mInternalObserver); this.getContentResolver().unregisterContentObserver(mExternalObserver); } private void handleMediaContentChange(Uri contentUri) { Cursor cursor = null; try { // 數(shù)據(jù)改變時查詢數(shù)據(jù)庫中最后加入的一條數(shù)據(jù) cursor = this.getContentResolver().query( contentUri, MEDIA_PROJECTIONS, null, null, MediaStore.Images.ImageColumns.DATE_ADDED + " desc limit 1" ); if (cursor == null) { return; } if (!cursor.moveToFirst()) { return; } // 獲取各列的索引 int dataIndex = cursor.getColumnIndex(MediaStore.Images.ImageColumns.DATA); int dateTakenIndex = cursor.getColumnIndex(MediaStore.Images.ImageColumns.DATE_TAKEN); // 獲取行數(shù)據(jù) String data = cursor.getString(dataIndex); long dateTaken = cursor.getLong(dateTakenIndex); // 處理獲取到的第一行數(shù)據(jù) handleMediaRowData(data, dateTaken); } catch (Exception e) { e.printStackTrace(); } finally { if (cursor != null && !cursor.isClosed()) { cursor.close(); } } } /** * 處理監(jiān)聽到的資源 */ private void handleMediaRowData(String data, long dateTaken) { if (checkScreenShot(data, dateTaken)) { Log.d(TAG, data + " " + dateTaken); } else { Log.d(TAG, "Not screenshot event"); } } /** * 判斷是否是截屏 */ private boolean checkScreenShot(String data, long dateTaken) { data = data.toLowerCase(); // 判斷圖片路徑是否含有指定的關(guān)鍵字之一, 如果有, 則認(rèn)為當(dāng)前截屏了 for (String keyWork : KEYWORDS) { if (data.contains(keyWork)) { return true; } } return false; } /** * 媒體內(nèi)容觀察者(觀察媒體數(shù)據(jù)庫的改變) */ private class MediaContentObserver extends ContentObserver { private Uri mContentUri; public MediaContentObserver(Uri contentUri, Handler handler) { super(handler); mContentUri = contentUri; } @Override public void onChange(boolean selfChange) { super.onChange(selfChange); Log.d(TAG, mContentUri.toString()); handleMediaContentChange(mContentUri); } } }
日志:
D/Screenshot: content://media/external/images/media D/Screenshot: /storage/emulated/0/Pictures/Screenshots/Screenshot_2016-12-19-11-24-02.png 1482117842287
注意事項:
- ContentObserver會監(jiān)聽到所有圖片資源的變化情況,要做好去重過濾工作
- 根據(jù)uri去讀取ContentProvider內(nèi)容時候,記得關(guān)閉cursor,防止內(nèi)存泄漏
- 關(guān)鍵字可擴(kuò)展,大大增加的監(jiān)聽的范圍,比FileObserver好用多了,但是去重過濾會比FileObserver復(fù)雜一些。
4. 總結(jié)
目前這是在網(wǎng)上搜索到的關(guān)于截屏監(jiān)聽方法的總結(jié),如果大家還有什么比較好的監(jiān)聽方法,歡迎分享。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android四種數(shù)據(jù)存儲的應(yīng)用方式
這篇文章主要介紹了Android四種數(shù)據(jù)存儲的應(yīng)用方式的相關(guān)資料,希望通過本文能幫助到大家,讓大家理解掌握Android存儲數(shù)據(jù)的方法,需要的朋友可以參考下2017-10-10Android自定義實現(xiàn)BaseAdapter的普通實現(xiàn)
這篇文章主要為大家詳細(xì)介紹了Android自定義實現(xiàn)BaseAdapter的普通實現(xiàn),感興趣的小伙伴們可以參考一下2016-08-08Android仿新浪微博發(fā)布微博界面設(shè)計(5)
這篇文章主要為大家詳細(xì)介紹了Android仿新浪微博發(fā)布微博界面設(shè)計方案,具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-11-11Android簡單創(chuàng)建一個Activity的方法
這篇文章主要介紹了Android簡單創(chuàng)建一個Activity的方法,結(jié)合圖文形式分析了Android創(chuàng)建Activity的具體步驟與實現(xiàn)技巧,需要的朋友可以參考下2016-04-04Android編程之listView中checkbox用法實例分析
這篇文章主要介紹了Android編程之listView中checkbox用法,結(jié)合實例形式分析了Android中checkbox的頁面布局及功能實現(xiàn)相關(guān)技巧,需要的朋友可以參考下2016-01-01