通過ijetty獲取android設(shè)備的截圖

需求:對于無顯示設(shè)備的android設(shè)備,如何獲取到當前設(shè)備畫面

條件:首先在android設(shè)備中搭建一個web服務器,可以參考基于前面介紹的ijetty服務器

實現(xiàn):

方法1:

參考系統(tǒng)hide代碼,android.view.Surface.java

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

我們要利用web服務器把圖像傳出來,可以拿到上面代碼中的outputStream,然后拷貝到HttpServletResponse的輸出流中。
這個screenshot函數(shù)還需要一些其他數(shù)據(jù),例如display,metrics,dims等,我們參考Surface類中的實現(xiàn),得出的可用類代碼如下:

public class ScreenshotNative {

    private Context mContext;
    private WindowManager mWindowManager;
    private Display mDisplay;
    private DisplayMetrics mDisplayMetrics;
    private Matrix mDisplayMatrix;

    private Bitmap mScreenBitmap;

    private static final Object sLock = new Object();

    public ScreenshotNative(Context context) {
        mContext = context;

        mDisplayMatrix = new Matrix();
        mWindowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
        mDisplay = mWindowManager.getDefaultDisplay();
        mDisplayMetrics = new DisplayMetrics();
        mDisplay.getRealMetrics(mDisplayMetrics);
    }

    private float getDegreesForRotation(int value) {
        switch (value) {
            case Surface.ROTATION_90:
                return 360f - 90f;
            case Surface.ROTATION_180:
                return 360f - 180f;
            case Surface.ROTATION_270:
                return 360f - 270f;
        }
        return 0f;
    }

    public void takeScreenshot(OutputStream out) throws IOException {
        // We need to orient the screenshot correctly (and the Surface api seems to take screenshots
        // only in the natural orientation of the device :!)
        //
        mDisplay.getRealMetrics(mDisplayMetrics);
        float[] dims = {mDisplayMetrics.widthPixels, mDisplayMetrics.heightPixels};
        float degrees = getDegreesForRotation(mDisplay.getRotation());
        boolean requiresRotation = (degrees > 0);
        if (requiresRotation) {
            // Get the dimensions of the device in its native orientation
            mDisplayMatrix.reset();
            mDisplayMatrix.preRotate(-degrees);
            mDisplayMatrix.mapPoints(dims);
            dims[0] = Math.abs(dims[0]);
            dims[1] = Math.abs(dims[1]);
        }

        Log.d("takeScreenshot", "takeScreenshot, dims, w-h: " + dims[0] + "-" + dims[1] + "; " +
                "dm w-h: " + mDisplayMetrics.widthPixels + mDisplayMetrics.heightPixels +
                "Thread=" + Thread.currentThread().getName());
        // Take the screenshot
        synchronized (sLock) {
            mScreenBitmap = SurfaceControl.screenshot((int) dims[0], (int) dims[1]);

            if (mScreenBitmap == null) {
                throw new IOException("Bitmap is null after taking native screenshot!");

            }

            if (requiresRotation) {
                // Rotate the screenshot to the current orientation
                Bitmap ss = Bitmap.createBitmap(mDisplayMetrics.widthPixels,
                        mDisplayMetrics.heightPixels, Bitmap.Config.ARGB_8888);
                Canvas c = new Canvas(ss);
                c.translate(ss.getWidth() / 2, ss.getHeight() / 2);
                c.rotate(degrees);
                c.translate(-dims[0] / 2, -dims[1] / 2);
                c.drawBitmap(mScreenBitmap, 0, 0, null);
                c.setBitmap(null);
                // Recycle the previous bitmap
                mScreenBitmap.recycle();
                mScreenBitmap = ss;
            }

            // Optimizations
            mScreenBitmap.setHasAlpha(false);
            mScreenBitmap.prepareToDraw();

            mScreenBitmap.compress(Bitmap.CompressFormat.PNG, 100, out);
            Log.d("takeScreenshot","Thread=" + Thread.currentThread().getName());
            mScreenBitmap.recycle();
            // Clear any references to the bitmap
            mScreenBitmap = null;
        }

    }
}

使用方法如下:

    private void screenshotNative(HttpServletResponse response){
        try {
            ServletOutputStream outputStream = response.getOutputStream();
            new ScreenshotNative(mContext).takeScreenshot(outputStream);
            outputStream.flush();
            outputStream.close();

        } catch (IOException e) {
            e.printStackTrace();
        }
    }

這樣在GET請求就能拿到對應的流了 然后存在本地即可。

方法2:

使用screencap工具

public void takeScreenShot(){ 
   String mSavedPath = "/sdcard/" + "screenshot.png" ; 
   try {                     
          Process p = Runtime. getRuntime().exec("screencap -p " + mSavedPath); 
          p.waitFor();
    } catch (Exception e) { 
          e.printStackTrace();
}

代碼執(zhí)行后,再把文件拷到servlet的輸出流中就好了。screencap只支持png格式的存儲。

以上兩種方法都需要添加權(quán)限:

<uses-permission android:name="android.permission.READ_FRAME_BUFFER"/>

并使用system uid

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關(guān)閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 179,030評論 25 709
  • afinalAfinal是一個android的ioc,orm框架 https://github.com/yangf...
    passiontim閱讀 15,871評論 2 45
  • 為師,七年,厭倦了一樣的講座,一樣的教學; 那么,學生,為何要遭受一如既往的重復? 為師,奮力,想尋得一份自由的尊...
    草木吟閱讀 236評論 0 0
  • $(document).on("click", ".select_address_", function(){va...
    豹發(fā)戶閱讀 336評論 0 0
  • 讀完這本書,其實就是不少牛人針對某些問題的解答和分享。越發(fā)感覺在這個時代,分享的重要性。很多東西,你覺得自己會了,...
    樂樂樂t閱讀 245評論 0 0

友情鏈接更多精彩內(nèi)容