欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

解析Android截取手机屏幕两种实现方案

程序员文章站 2022-06-20 13:14:31
最近在开发的过程中,遇到了一个需要截取屏幕保存为图片的需求,具体为截取webview的视图保存图片。 方法1:首先想到的思路是利用sdk提供的view.getdrawin...

最近在开发的过程中,遇到了一个需要截取屏幕保存为图片的需求,具体为截取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打开这个()界面,截取的图片就会有问题,具体表现为网页中游动的水母没有显示在截取的图片上。

方法2:使用android系统提供的服务context.media_projection_service,进行截图操作。

github地址:https://github.com/miqt/capwindow

demo源码下载地址:capwindow_jb51.rar

关键部分代码解析:↓

发送截图请求

 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的系统设备上适用。

总结:

总而言之,这两种方法各有利弊,使用的时候要根据自己的实际需求做出选择。以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。