Android實(shí)現(xiàn)截屏方式整理(總結(jié))
本文介紹了Android 實(shí)現(xiàn)截屏方式整理,分享給大家。希望對大家有幫助
可能的需求:
- 截自己的屏
- 截所有的屏
- 帶導(dǎo)航欄截屏
- 不帶導(dǎo)航欄截屏
- 截屏并編輯選取一部分
- 自動(dòng)截取某個(gè)空間或者布局
- 截取長圖
- 在后臺(tái)去截屏
1.只截取自己應(yīng)用內(nèi)部界面
1.1 截取除了導(dǎo)航欄之外的屏幕
View dView = getWindow().getDecorView(); dView.setDrawingCacheEnabled(true); dView.buildDrawingCache(); Bitmap bitmap = Bitmap.createBitmap(dView.getDrawingCache()); if (bitmap != null) { try { // 獲取內(nèi)置SD卡路徑 String sdCardPath = Environment.getExternalStorageDirectory().getPath(); // 圖片文件路徑 String filePath = sdCardPath + File.separator + "screenshot.png"; File file = new File(filePath); FileOutputStream os = new FileOutputStream(file); bitmap.compress(Bitmap.CompressFormat.PNG, 100, os); os.flush(); os.close(); DebugLog.d("a7888", "存儲(chǔ)完成"); } catch (Exception e) { } }
1.2 截取某個(gè)控件或者區(qū)域
兩種方案:
跟上面差不多,只不過view不適用根view,而是使用某個(gè)某個(gè)控件。
View dView = title; dView.setDrawingCacheEnabled(true); dView.buildDrawingCache(); Bitmap bitmap = Bitmap.createBitmap(dView.getDrawingCache());
手動(dòng)draw
View dView = titleTv; Bitmap bitmap = Bitmap.createBitmap(dView.getWidth(), dView.getHeight(), Bitmap.Config.ARGB_8888); //使用Canvas,調(diào)用自定義view控件的onDraw方法,繪制圖片 Canvas canvas = new Canvas(bitmap); dView.draw(canvas);
1.3 截取帶導(dǎo)航欄的整個(gè)屏幕
這一小節(jié)會(huì)將一些理論上可以,但是實(shí)踐會(huì)特別復(fù)雜,不太推薦使用。可以學(xué)習(xí)了解。
adb 命令
這里指的不是連接電腦進(jìn)行adb操控,而是在App內(nèi)部實(shí)現(xiàn)adb命令的操控
在APK中調(diào)用“adb shell screencap -pfilepath” 命令
該命令讀取系統(tǒng)的framebuffer,需要獲得系統(tǒng)權(quán)限:
(1). 在AndroidManifest.xml文件中添加
<uses-permissionandroid:name="android.permission.READ_FRAME_BUFFER"/>
(2). 修改APK為系統(tǒng)權(quán)限,將APK放到源碼中編譯, 修改Android.mk
LOCAL_CERTIFICATE := platform publicvoid takeScreenShot(){ String mSavedPath = Environment.getExternalStorageDirectory()+File. separator + "screenshot.png" ; try { Runtime. getRuntime().exec("screencap -p " + mSavedPath); } catch (Exception e) { e.printStackTrace(); }
利用系統(tǒng)的隱藏API,實(shí)現(xiàn)Screenshot,這部分代碼是系統(tǒng)隱藏的,需要在源碼下編譯。
1).修改Android.mk, 添加系統(tǒng)權(quán)限
LOCAL_CERTIFICATE := platform
2).修改AndroidManifest.xml 文件,添加權(quán)限
<uses-permissionandroid:name="android.permission.READ_FRAME_BUFFER"/>
public boolean takeScreenShot(String imagePath){ if(imagePath.equals("" )){ imagePath = Environment.getExternalStorageDirectory()+File. separator+"Screenshot.png" ; } Bitmap mScreenBitmap; WindowManager mWindowManager; DisplayMetrics mDisplayMetrics; Display mDisplay; mWindowManager = (WindowManager) mcontext.getSystemService(Context.WINDOW_SERVICE); mDisplay = mWindowManager.getDefaultDisplay(); mDisplayMetrics = new DisplayMetrics(); mDisplay.getRealMetrics(mDisplayMetrics); float[] dims = {mDisplayMetrics.widthPixels , mDisplayMetrics.heightPixels }; mScreenBitmap = Surface. screenshot((int) dims[0], ( int) dims[1]); if (mScreenBitmap == null) { return false ; } try { FileOutputStream out = new FileOutputStream(imagePath); mScreenBitmap.compress(Bitmap.CompressFormat. PNG, 100, out); } catch (Exception e) { return false ; } return true ; }
Android本地編程(Native Programming)讀取framebuffer
命令行,框架的截屏功能是通過framebuffer來實(shí)現(xiàn)的,所以我們先來介紹一下framebuffer。
framebuffer介紹
幀緩沖(framebuffer)是Linux為顯示設(shè)備提供的一個(gè)接口,把顯存抽象后的一種設(shè)備,他允許上層應(yīng)用程序在圖形模式下直接對顯示緩沖區(qū)進(jìn)行 讀寫操作。這種操作是抽象的,統(tǒng)一的。用戶不必關(guān)心物理顯存的位置、換頁機(jī)制等等具體細(xì)節(jié)。這些都是由Framebuffer設(shè)備驅(qū)動(dòng)來完成的。
linux FrameBuffer 本質(zhì)上只是提供了對圖形設(shè)備的硬件抽象,在開發(fā)者看來,F(xiàn)rameBuffer 是一塊顯示緩存,往顯示緩存中寫入特定格式的數(shù)據(jù)就意味著向屏幕輸出內(nèi)容。所以說FrameBuffer就是一塊白板。例如對于初始化為16 位色的FrameBuffer 來說, FrameBuffer中的兩個(gè)字節(jié)代表屏幕上一個(gè)點(diǎn),從上到下,從左至右,屏幕位置與內(nèi)存地址是順序的線性關(guān)系。
幀緩存有個(gè)地址,是在內(nèi)存里。我們通過不停的向frame buffer中寫入數(shù)據(jù), 顯示控制器就自動(dòng)的從frame buffer中取數(shù)據(jù)并顯示出來。全部的圖形都共享內(nèi)存中同一個(gè)幀緩存。
android截屏實(shí)現(xiàn)思路
Android系統(tǒng)是基于Linux內(nèi)核的,所以也存在framebuffer這個(gè)設(shè)備,我們要實(shí)現(xiàn)截屏的話只要能獲取到framebuffer中的數(shù)據(jù),然后把數(shù)據(jù)轉(zhuǎn)換成圖片就可以了,android中的framebuffer數(shù)據(jù)是存放在 /dev/graphics/fb0 文件中的,所以我們只需要來獲取這個(gè)文件的數(shù)據(jù)就可以得到當(dāng)前屏幕的內(nèi)容。
現(xiàn)在我們的測試代碼運(yùn)行時(shí)候是通過RC(remote controller)方式來運(yùn)行被測應(yīng)用的,那就需要在PC機(jī)上來訪問模擬器或者真機(jī)上的framebuffer數(shù)據(jù),這個(gè)的話可以通過android的ADB命令來實(shí)現(xiàn)。
各大手機(jī)自帶的按鍵組合進(jìn)行截屏
Android源碼中對按鍵的捕獲位于文件PhoneWindowManager.java(alps\frameworks\base\policy\src\com\android\internal\policy\impl)中,這個(gè)類處理所有的鍵盤輸入事件,其中函數(shù)interceptKeyBeforeQueueing()會(huì)對常用的按鍵做特殊處理。
2. 截取非含當(dāng)前應(yīng)用的屏幕部分(最佳官方方案)
Android 在5.0 之后支持了實(shí)時(shí)錄屏的功能。通過實(shí)時(shí)錄屏我們可以拿到截屏的圖像。同時(shí)可以通過在Service中處理實(shí)現(xiàn)后臺(tái)的錄屏。具體的類講解大家自行網(wǎng)上查閱。
大體步驟:
1.初始化一個(gè)MediaProjectionManager。
MediaProjectionManager mMediaProjectionManager = (MediaProjectionManager)getApplication().getSystemService(Context.MEDIA_PROJECTION_SERVICE);
2.創(chuàng)建intent,并啟動(dòng)Intent。注意這里是startActivityForResult
startActivityForResult(mMediaProjectionManager.createScreenCaptureIntent(), REQUEST_MEDIA_PROJECTION);
3.在onActivityResult中拿到Mediaprojection。
mResultCode = resultCode; mResultData = data; mMediaProjection = mMediaProjectionManager.getMediaProjection(mResultCode, mResultData);
4.設(shè)置VirtualDisplay 將圖像和展示的View關(guān)聯(lián)起來。一般來說我們會(huì)將圖像展示到SurfaceView,這里為了為了便于拿到截圖,我們使用ImageReader,他內(nèi)置有SurfaceView。
mImageReader = ImageReader.newInstance(windowWidth, windowHeight, 0x1, 2); //ImageFormat.RGB_565 mVirtualDisplay = mMediaProjection.createVirtualDisplay("screen-mirror", windowWidth, windowHeight, mScreenDensity, DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR, mImageReader.getSurface(), null, null);
5.通過ImageReader拿到截圖
strDate = dateFormat.format(new java.util.Date()); nameImage = pathImage+strDate+".png"; Image image = mImageReader.acquireLatestImage(); int width = image.getWidth(); int height = image.getHeight(); final Image.Plane[] planes = image.getPlanes(); final ByteBuffer buffer = planes[0].getBuffer(); int pixelStride = planes[0].getPixelStride(); int rowStride = planes[0].getRowStride(); int rowPadding = rowStride - pixelStride * width; Bitmap bitmap = Bitmap.createBitmap(width+rowPadding/pixelStride, height, Bitmap.Config.ARGB_8888); bitmap.copyPixelsFromBuffer(buffer); bitmap = Bitmap.createBitmap(bitmap, 0, 0,width, height); image.close();
6.注意截屏之后要及時(shí)關(guān)閉VirtualDisplay ,因?yàn)閂irtualDisplay 是十分消耗內(nèi)存和電量的。
if (mVirtualDisplay == null) { return; } mVirtualDisplay.release(); mVirtualDisplay = null;
ps: 具體可以參考Google官方給的demo以及其他開發(fā)者寫的Demo
https://github.com/googlesamples/android-ScreenCapture
https://github.com/VincentWYJ/CaptureScreen
3. 截取長屏
截取長屏其實(shí)原理就是截取整個(gè)ScrollView或者ListView的視圖,因此實(shí)現(xiàn)原理跟上面中提到的截取某個(gè)控件的View基本一致。
ScrollView 實(shí)現(xiàn)截屏
/** * 截取scrollview的屏幕 * **/ public static Bitmap getScrollViewBitmap(ScrollView scrollView) { int h = 0; Bitmap bitmap; for (int i = 0; i < scrollView.getChildCount(); i++) { h += scrollView.getChildAt(i).getHeight(); } // 創(chuàng)建對應(yīng)大小的bitmap bitmap = Bitmap.createBitmap(scrollView.getWidth(), h, Bitmap.Config.ARGB_8888); final Canvas canvas = new Canvas(bitmap); scrollView.draw(canvas); return bitmap; }
ListView實(shí)現(xiàn)截屏
/** * 截圖listview * **/ public static Bitmap getListViewBitmap(ListView listView,String picpath) { int h = 0; Bitmap bitmap; // 獲取listView實(shí)際高度 for (int i = 0; i < listView.getChildCount(); i++) { h += listView.getChildAt(i).getHeight(); } Log.d(TAG, "實(shí)際高度:" + h); Log.d(TAG, "list 高度:" + listView.getHeight()); // 創(chuàng)建對應(yīng)大小的bitmap bitmap = Bitmap.createBitmap(listView.getWidth(), h, Bitmap.Config.ARGB_8888); final Canvas canvas = new Canvas(bitmap); listView.draw(canvas); return bitmap; }
WebView實(shí)現(xiàn)截屏
//這是webview的,利用了webview的api private static Bitmap captureWebView(WebView webView) { Picture snapShot = webView.capturePicture(); Bitmap bmp = Bitmap.createBitmap(snapShot.getWidth(), snapShot.getHeight(), Bitmap.Config.ARGB_8888); Canvas canvas = new Canvas(bmp); snapShot.draw(canvas); return bmp; }
有時(shí)候我們可能需要去滾動(dòng)屏幕,然后再滾動(dòng)到某一個(gè)地方時(shí)停止截屏,這樣就會(huì)去截取從開始到滾動(dòng)結(jié)束位置的view,而不是整個(gè)ScrollView,這個(gè)時(shí)候就需要進(jìn)行一些控制,具體原理跟上面講的差不多,可以參考一下下面的實(shí)現(xiàn):
https://android-notes.github.io/2016/12/03/android%E9%95%BF%E6%88%AA%E5%B1%8F%E5%8E%9F%E7%90%86/
4. 實(shí)時(shí)截屏
可參考2中Android 在5.0的做法,進(jìn)行實(shí)時(shí)錄制。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android UI系列-----ScrollView和HorizontalScrollView的詳解
本篇文章主要是介紹的Android UI系列-----ScrollView和HorizontalScrollView,ScrollView和HorizontalScrollView都是布局容器,有需要的可以了解一下。2016-11-11Android開發(fā)中關(guān)于獲取當(dāng)前Activity的一些思考
這篇文章主要為大家詳細(xì)介紹了Android開發(fā)過程中,關(guān)于獲取當(dāng)前Activity的一些思考,感興趣的小伙伴們可以參考一下2016-02-02Android中利用zxing實(shí)現(xiàn)自己的二維碼掃描識(shí)別詳解
這篇文章主要給大家介紹了關(guān)于Android中利用zxing實(shí)現(xiàn)自己的二維碼掃描識(shí)別的相關(guān)資料,文中通過圖文介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用zxing具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起看看吧。2017-09-09Android實(shí)現(xiàn)帶有指示器的自定義底部導(dǎo)航欄
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)帶有指示器的自定義底部導(dǎo)航欄,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-04-04Android 仿抖音的評論列表的UI和效果的實(shí)現(xiàn)代碼
抖音是一款音樂創(chuàng)意短視頻社交軟件,此app已在android各大應(yīng)用商店和app store 上線。下面小編給大家?guī)砹薃ndroid 仿抖音的評論列表的UI和效果的實(shí)現(xiàn)代碼,感興趣的朋友參考下吧2018-03-03Android實(shí)現(xiàn)左滑關(guān)閉窗口
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)左滑關(guān)閉窗口,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04Android實(shí)現(xiàn)圓形ProgressBar停止轉(zhuǎn)動(dòng)的方法詳解
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)圓形ProgressBar停止轉(zhuǎn)動(dòng)方法的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-03-03