詳解Android截屏事件監(jiān)聽
1. 前言
Android系統(tǒng)沒有直接對(duì)截屏事件監(jiān)聽的接口,也沒有廣播,只能自己動(dòng)手來豐衣足食,一般有三種方法。
- 利用FileObserver監(jiān)聽某個(gè)目錄中資源變化情況
- 利用ContentObserver監(jiān)聽全部資源的變化
- 監(jiān)聽截屏快捷按鍵
由于廠商自定義Android系統(tǒng)的多樣性,再加上快捷鍵的不同以及第三方應(yīng)用,監(jiān)聽截屏快捷鍵這事基本不靠譜,可以直接忽略。
本文使用的測(cè)試手機(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)聽某一個(gè)事件。
類型 | int值 | 說明 |
---|---|---|
FileObserver.ACCESS | 1 | 讀取某個(gè)文件 |
FileObserver.MODIFY | 2 | 向某個(gè)文件寫入數(shù)據(jù) |
FileObserver.ATTRIB | 4 | 文件的屬性被修改(權(quán)限/日期/擁有者) |
FileObserver.CLOSE_WRITE | 8 | 寫入數(shù)據(jù)后關(guān)閉 |
FileObserver.CLOSE_NOWRITE | 16 | 只讀模式打開文件后關(guān)閉 |
FileObserver.OPEN | 32 | 打開某個(gè)文件 |
FileObserver.MOVED_FROM | 64 | 有文件或者文件夾從被監(jiān)聽的文件夾中移走 |
FileObserver.MOVED_TO | 128 | 有文件或者文件夾移動(dòng)到被監(jiān)聽的文件夾 |
FileObserver.CREATE | 256 | 文件或者文件夾被創(chuàng)建 |
FileObserver.DELETE | 512 | 文件被刪除 |
FileObserver.DELETE_SELF | 1024 | 被監(jiān)聽的文件或者目錄被刪除 |
FileObserver.MOVE_SELF | 2048 | 被監(jiān)聽的文件或者目錄被移走 |
幾點(diǎn)注意事項(xiàng):
- 每一次截屏,有多個(gè)事件回調(diào)
- 每一次截屏,不同的手機(jī),事件回調(diào)可能有些不同,參考上述日志
- 不同的手機(jī),默認(rèn)截屏圖片儲(chǔ)存的文件夾可能不同
- FileObserver只能監(jiān)聽文件夾中子文件和子文件夾的變化情況,不能監(jiān)聽子文件夾內(nèi)部的資源變化
- 需要<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>權(quán)限,否則可能收不到事件
基于第三點(diǎn)和第四點(diǎ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ù)庫時(shí)需要讀取的列 */ private static final String[] MEDIA_PROJECTIONS = { MediaStore.Images.ImageColumns.DATA, MediaStore.Images.ImageColumns.DATE_TAKEN, }; /** 內(nèi)部存儲(chǔ)器內(nèi)容觀察者 */ private ContentObserver mInternalObserver; /** 外部存儲(chǔ)器內(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í)查詢數(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
注意事項(xiàng):
- ContentObserver會(huì)監(jiān)聽到所有圖片資源的變化情況,要做好去重過濾工作
- 根據(jù)uri去讀取ContentProvider內(nèi)容時(shí)候,記得關(guān)閉cursor,防止內(nèi)存泄漏
- 關(guān)鍵字可擴(kuò)展,大大增加的監(jiān)聽的范圍,比FileObserver好用多了,但是去重過濾會(huì)比FileObserver復(fù)雜一些。
4. 總結(jié)
目前這是在網(wǎng)上搜索到的關(guān)于截屏監(jiān)聽方法的總結(jié),如果大家還有什么比較好的監(jiān)聽方法,歡迎分享。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android截屏保存png圖片的實(shí)例代碼
- Android實(shí)現(xiàn)截屏方式整理(總結(jié))
- JavaScript截屏功能的實(shí)現(xiàn)代碼
- java編程實(shí)現(xiàn)屏幕截圖(截屏)代碼總結(jié)
- android截屏功能實(shí)現(xiàn)代碼
- 使用python編寫android截屏腳本雙擊運(yùn)行即可
- c#根據(jù)網(wǎng)址抓取網(wǎng)頁截屏生成圖片的示例
- Android 使用Shell腳本截屏并自動(dòng)傳到電腦上
- asp.net截屏功能實(shí)現(xiàn)截取web頁面
- Android實(shí)現(xiàn)截屏并保存操作功能
- Android源碼解析之截屏事件流程
相關(guān)文章
Android四種數(shù)據(jù)存儲(chǔ)的應(yīng)用方式
這篇文章主要介紹了Android四種數(shù)據(jù)存儲(chǔ)的應(yīng)用方式的相關(guān)資料,希望通過本文能幫助到大家,讓大家理解掌握Android存儲(chǔ)數(shù)據(jù)的方法,需要的朋友可以參考下2017-10-10Android自定義實(shí)現(xiàn)BaseAdapter的普通實(shí)現(xiàn)
這篇文章主要為大家詳細(xì)介紹了Android自定義實(shí)現(xiàn)BaseAdapter的普通實(shí)現(xiàn),感興趣的小伙伴們可以參考一下2016-08-08Android實(shí)現(xiàn)登錄注冊(cè)頁面(下)
這篇文章主要介紹了Android實(shí)現(xiàn)登錄注冊(cè)頁面的第二篇,實(shí)現(xiàn)驗(yàn)證登錄和記住密碼功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04Android項(xiàng)目基本結(jié)構(gòu)詳解
這篇文章主要為大家詳細(xì)介紹了Android項(xiàng)目基本結(jié)構(gòu),從最基本的內(nèi)容講起,帶你逐步進(jìn)入用C#進(jìn)行Android應(yīng)用開發(fā)的樂園,感興趣的小伙伴們可以參考一下2016-06-06Android仿新浪微博發(fā)布微博界面設(shè)計(jì)(5)
這篇文章主要為大家詳細(xì)介紹了Android仿新浪微博發(fā)布微博界面設(shè)計(jì)方案,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-11-11Android實(shí)現(xiàn)簡單計(jì)時(shí)器功能
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)簡單計(jì)時(shí)器功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-10-10Android簡單創(chuàng)建一個(gè)Activity的方法
這篇文章主要介紹了Android簡單創(chuàng)建一個(gè)Activity的方法,結(jié)合圖文形式分析了Android創(chuàng)建Activity的具體步驟與實(shí)現(xiàn)技巧,需要的朋友可以參考下2016-04-04Android編程之listView中checkbox用法實(shí)例分析
這篇文章主要介紹了Android編程之listView中checkbox用法,結(jié)合實(shí)例形式分析了Android中checkbox的頁面布局及功能實(shí)現(xiàn)相關(guān)技巧,需要的朋友可以參考下2016-01-01