Android 5.0及以上實現屏幕截圖

在Android 5.0,API 21 之前想要截圖系統屏幕必須Root才能完成,5.0之后開放了接口,下面看我們是怎么實現的。

--

1. 涉及到的相關類

1. MediaProjectionManager

官方原話: Manages the retrieval of certain types of {@link MediaProjection} tokens.
這個類通過 Context#getSystemServiceMEDIA_PROJECTION_SERVICE 獲取,他的功能就是獲取MediaProjection

2. MediaProjection

官方原話:A token granting applications the ability to capture screen contents and/or record system audio. The exact capabilities granted depend on the type of MediaProjection.在這個類中我們能獲取到屏幕的內容

3. ImageReader

官方原話:The ImageReader class allows direct application access to image data
rendered into a {@link android.view.Surface}
通過這個類我們可以把Surface轉換成圖片

2. 上面三個類就可以完成我們截取屏幕圖片的操作,那么下面我們將解釋他們是怎么合作完成的

1. 首先獲取用戶授權,截圖屏幕需要用戶手動授權后才能操作

  @TargetApi(Build.VERSION_CODES.LOLLIPOP)
  public void requestCapturePermission() {

  if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
   //5.0 之后才允許使用屏幕截圖

  return;
  }

   MediaProjectionManager mediaProjectionManager = (MediaProjectionManager)
      getSystemService(Context.MEDIA_PROJECTION_SERVICE);
  startActivityForResult(
      mediaProjectionManager.createScreenCaptureIntent(),
      REQUEST_MEDIA_PROJECTION);
  }

這里必須使用startActivityForResult 因為在createScreenCaptureIntent() 方法中會返回用戶授權截取屏幕的結果,用戶根據下面彈窗允許或者拒絕

授權
授權

用戶選擇后在Activity 的onActivityResult 中操作返回的結果data

  @Override
  protected void onActivityResult(int requestCode, int resultCode, Intent data) {
      super.onActivityResult(requestCode, resultCode, data);

      switch (requestCode) {
          case REQUEST_MEDIA_PROJECTION:

          if (resultCode == RESULT_OK && data != null) {
              FloatWindowsService.setResultData(data);
              startService(new Intent(getApplicationContext(), FloatWindowsService.class));
          } 

      break;
      }
  }

這里我是用FloatWindowsService在桌面上顯示一個懸浮按鈕,點擊截屏,下面我們看在FloatWindowsService 是如何實現截圖

2. 截取屏幕內容生成Bitmap

首先創建ImageReader實例

  private void createImageReader() {

       mImageReader = ImageReader.newInstance(mScreenWidth, mScreenHeight, PixelFormat.RGBA_8888, 2); 

  }

然后點擊事件中觸發startScreenShot()

  private void startScreenShot() {

      mFloatView.setVisibility(View.GONE);

      Handler handler = new Handler();
      handler.postDelayed(new Runnable() {
      public void run() {
          //獲取當前屏幕內容
          startVirtual();
      }
      }, 5);

      handler.postDelayed(new Runnable() {
      public void run() {
          //生成圖片保存到本地
          startCapture();

      }
      }, 30);
  }

startVirtual() 方法中我們做一件事,就是獲取當前屏幕內容

  public void startVirtual() {
      if (mMediaProjection != null) {
       virtualDisplay();
      } else {
       setUpMediaProjection();
       virtualDisplay();
      }
  }

與此同時需要獲取MediaProjection 實例,而mResultData 是授權后返回的結果

  public void setUpMediaProjection() {
   if (mResultData == null) {
      Intent intent = new Intent(Intent.ACTION_MAIN);
      intent.addCategory(Intent.CATEGORY_LAUNCHER);
      startActivity(intent);
   } else {
        //mResultData是在Activity中用戶授權后返回的結果
       mMediaProjection = getMediaProjectionManager().getMediaProjection(Activity.RESULT_OK, mResultData);
  }
}

最終得到當前屏幕的內容,注意這里mImageReader.getSurface()被傳入,屏幕的數據也將會在ImageReader中的Surface中

private void virtualDisplay() {
    mVirtualDisplay = mMediaProjection.createVirtualDisplay("screen-mirror",
    mScreenWidth, mScreenHeight, mScreenDensity, DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR,
    mImageReader.getSurface(), null, null);
 }

最后把mImageReader得到的屏幕內容數據轉換成圖片,在AsyncTask中處理,

Image.Plane中的 buffer 數據并不是完全是Bitmap所需要的,需要注意下面3點
1. Image 設置的圖片格式與Bitmap設置的必須一致
2. 緩沖數據存在行間距,所以我們必須去除這些間距
3. Image 使用后必須調用image.close();關閉,否則再次使用會報錯
@Override
protected Bitmap doInBackground(Image... params) {

   if (params == null || params.length < 1 || params[0] == null) {

       return null;
  }

  Image image = params[0];

  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();
最后把生成的bitmap保存起來,就ok了

源碼

APK

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,333評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,491評論 3 416
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,263評論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,946評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,708評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,186評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,255評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,409評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,939評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,774評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,976評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,518評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,209評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,641評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,872評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,650評論 3 391
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,958評論 2 373

推薦閱讀更多精彩內容