Android 截取手机屏幕两种实现方法

 更新时间:2017年05月09日 14:26:03   投稿:lqh  
这篇文章主要介绍了Android 截取手机屏幕两种实现方法的相关资料,需要的朋友可以参考下

Android 截取手机屏幕两种实现方法

最近在开发的过程中,遇到了一个需要截取屏幕保存为图片的需求,具体为截取webview的视图保存图片。

方法1:首先想到的思路是利用SDK提供的View.getDrawingCache()方法:

public void printScreen(View view) {
    String imgPath = "/sdcard/test.png";
    view.setDrawingCacheEnabled(true);
    view.buildDrawingCache();
    Bitmap bitmap = view.getDrawingCache();
    if (bitmap != null) {
      try {
        FileOutputStream out = new FileOutputStream(imgPath);
        bitmap.compress(Bitmap.CompressFormat.PNG, 100,
            out);
      } catch (Exception e) {
        e.printStackTrace();
      }
    }
  }

这个方法在很多情况下都是没有问题的,比如说截取imageview,TextView,甚至otherview.getRootView();都没问题,但在WebView上就会出现webview的部分截取完缺少页面里的一些内容的情况,比如说用webview打开这个(https://miqt.github.io/jellyfish/)界面,截取的图片就会有问题,具体表现为网页中游动的水母没有显示在截取的图片上。

方法2:使用Android系统提供的服务Context.MEDIA_PROJECTION_SERVICE,进行截图操作。

Demo源码:https://github.com/miqt/CapWindow

关键部分代码解析:↓

发送截图请求

final MediaProjectionManager projectionManager = (MediaProjectionManager)
        getSystemService(Context.MEDIA_PROJECTION_SERVICE);
 Intent intent = projectionManager.createScreenCaptureIntent();
 startActivityForResult(intent, REQUEST_CODE);

接收返回的结果:

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

    onScreenshotTaskBegan();
    final MediaProjectionManager projectionManager = (MediaProjectionManager)
        getSystemService(Context.MEDIA_PROJECTION_SERVICE);
    final MediaProjection mProjection = projectionManager.getMediaProjection(resultCode, data);
    Point size = Utils.getScreenSize(this);
    final int mWidth = size.x;
    final int mHeight = size.y;
    final ImageReader mImageReader = ImageReader.newInstance(mWidth, mHeight, PixelFormat
        .RGBA_8888, 2);
    final VirtualDisplay display = mProjection.createVirtualDisplay("screen-mirror", mWidth,
        mHeight, DisplayMetrics.DENSITY_MEDIUM,
        DisplayManager.VIRTUAL_DISPLAY_FLAG_PRESENTATION, mImageReader.getSurface(),
        null, null);

    mImageReader.setOnImageAvailableListener(new ImageReader.OnImageAvailableListener() {
      @Override
      public void onImageAvailable(ImageReader mImageReader) {

        Image image = null;
        try {
          image = mImageReader.acquireLatestImage();
          if (image != null) {
            final Image.Plane[] planes = image.getPlanes();
            if (planes.length > 0) {
              final ByteBuffer buffer = planes[0].getBuffer();
              int pixelStride = planes[0].getPixelStride();
              int rowStride = planes[0].getRowStride();
              int rowPadding = rowStride - pixelStride * mWidth;


              // create bitmap
              Bitmap bmp = Bitmap.createBitmap(mWidth + rowPadding / pixelStride,
                  mHeight, Bitmap.Config.ARGB_8888);
              bmp.copyPixelsFromBuffer(buffer);

              Bitmap croppedBitmap = Bitmap.createBitmap(bmp, 0, 0, mWidth, mHeight);

              saveBitmap(croppedBitmap);//保存图片

              if (croppedBitmap != null) {
                croppedBitmap.recycle();
              }
              if (bmp != null) {
                bmp.recycle();
              }
            }
          }

        } catch (Exception e) {
          e.printStackTrace();
        } finally {
          if (image != null) {
            image.close();
          }
          if (mImageReader != null) {
            mImageReader.close();
          }
          if (display != null) {
            display.release();
          }

          mImageReader.setOnImageAvailableListener(null, null);
          mProjection.stop();

          onScreenshotTaskOver();
        }

      }
    }, getBackgroundHandler());
  }

这个方法类似使用手机的系统截屏(音量下键+电源键),能够完美的吧当前原模原样的屏幕截取下来,并且修改保存方法的话甚至可以屏幕录像,但相比于第一种方法,它的缺点是完全和界面上的view没有关系,并且在调用这个服务的时候,会弹出一个权限确认的弹框。另外需要注意,这一方法只能在Android 5.0的系统设备上适用。

总结:

总而言之,这两种方法各有利弊,使用的时候要根据自己的实际需求做出选择。

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

相关文章

  • android异步加载图片并缓存到本地实现方法

    android异步加载图片并缓存到本地实现方法

    图片过多造成内存溢出,这个是最不容易解决的,要想一些好的缓存策略,比如大图片使用LRU缓存策略或懒加载缓存策略。今天首先介绍一下本地缓存图片
    2013-01-01
  • Android使用Notification在状态栏上显示通知

    Android使用Notification在状态栏上显示通知

    这篇文章主要为大家详细介绍了Android使用Notification在状态栏上显示通知,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-12-12
  • Android实现倒计时效果

    Android实现倒计时效果

    这篇文章主要为大家详细介绍了Android实现倒计时效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-10-10
  • Android 实现自定义圆形进度条的功能

    Android 实现自定义圆形进度条的功能

    这篇文章主要介绍了Android 实现自定义圆形进度条的功能的相关资料,开发Android应用的朋友肯定对自定义View不陌生,很多都有重新写的,这里就对实现圆形进度条介绍下,需要的朋友可以参考下
    2016-11-11
  • Android如何使用Bmob后端云实现失物招领功能

    Android如何使用Bmob后端云实现失物招领功能

    这篇文章主要介绍了Android如何使用Bmob后端云实现失物招领功能,帮助大家更好的理解和学习使用Android,感兴趣的朋友可以了解下
    2021-03-03
  • android 关于webview 加载h5网页开启定位的方法

    android 关于webview 加载h5网页开启定位的方法

    今天小编就为大家分享一篇android 关于webview 加载h5网页开启定位的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-07-07
  • Android手机卫士之设置密码对话框

    Android手机卫士之设置密码对话框

    这篇文章主要为大家详细介绍了Android手机卫士之设置密码对话框,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-10-10
  • android如何获取view在布局中的高度与宽度详解

    android如何获取view在布局中的高度与宽度详解

    这篇文章主要给大家介绍了关于android如何获取view在布局中的高度与宽度的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-10-10
  • Android自定义实现侧滑菜单效果

    Android自定义实现侧滑菜单效果

    这篇文章主要为大家详细介绍了Android自定义实现侧滑菜单效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-01-01
  • Android如何判断一个点在不在多边形区域内

    Android如何判断一个点在不在多边形区域内

    这篇文章主要为大家详细介绍了Android判断一个点在不在多边形区域内的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-05-05

最新评论