android视频截屏&手机录屏实现代码

发布时间 - 2026-01-11 02:31:59    点击率:

本文介绍了android视频截屏&手机录屏实现代码,分享给大家,希望对大家有帮助

问题

在android中有时候我们需要对屏幕进行截屏操作,单一的截屏操作好解决可以通过activity的顶层view DecorView获取一个bitmap,得到就是当前activity上面的全部视图。

 View view = activity.getWindow().getDecorView();
    view.setDrawingCacheEnabled(true);
    view.buildDrawingCache();
    Bitmap bmp = view.getDrawingCache();
    DisplayMetrics dm = new DisplayMetrics();
    activity.getWindowManager().getDefaultDisplay().getMetrics(dm);
    Bitmap ret = Bitmap.createBitmap(bmp, 0, 0, dm.widthPixels, dm.heightPixels);
    view.destroyDrawingCache();

如果activity中包含一些视频播放器比如SurfaceView GLSurfaceView TextureView,在调用截屏代码会发现播放视频的部分是黑屏的,原因是这几种视频渲染的view通过以上代码拿到的是缓冲区不是真正的图像。

解决办法

android5.0以上系统提供了一个 MediaProjectionManager类来对手机进行录屏操作,也支持获取手机的Image图像的操作,知道了这些我们就可以通过提供的api来进行截屏操作了。

这里通过Service来操作截屏和录屏的api

1.绑定截屏的Service

 Intent intent = new Intent(this, ScreenService.class);
 bindService(intent, mServiceConnection, BIND_AUTO_CREATE);

 public void onServiceConnected(ComponentName className, IBinder service) {
       DisplayMetrics metrics = new DisplayMetrics();
       getWindowManager().getDefaultDisplay().getMetrics(metrics);
       ScreenService.RecordBinder binder = (ScreenService.RecordBinder) service;
       recordService = binder.getRecordService();
       recordService.setConfig(metrics.widthPixels, metrics.heightPixels, metrics.densityDpi);
       mButton.setEnabled(true);
       mButton.setText(recordService.isRunning() ? "结束" : "开始");
 }

2.请求权限 onActivityResult 方法中回调。

 Intent captureIntent = projectionManager.createScreenCaptureIntent();
 startActivityForResult(captureIntent, RECORD_REQUEST_CODE);

成功后

 @Override
 protected void onActivityResult(int requestCode, int resultCode, Intent data) {
     if (requestCode == RECORD_REQUEST_CODE && resultCode == RESULT_OK) {

       //######## 截屏逻辑 ########
       mediaProjection = projectionManager.getMediaProjection(resultCode, data);
       recordService.setMediaProject(mediaProjection);
       recordService.initImageReader();

     }
 }

3. 获取截屏

 @Override
 public void onClick(View view) {

  //######## 截屏逻辑 ########
   Bitmap bitmap = recordService.getBitmap();
   mImageView.setImageBitmap(bitmap);
 }

录屏

录屏需要初始化一些录屏参数,输入麦克风类型视频类型,保存路径等

 private void initRecorder() {
     mediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
     mediaRecorder.setVideoSource(MediaRecorder.VideoSource.SURFACE);
     mediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
     mediaRecorder.setOutputFile(
         getSavePath() + System.currentTimeMillis() + ".mp4");
     mediaRecorder.setVideoSize(width, height);
     mediaRecorder.setVideoEncoder(MediaRecorder.VideoEncoder.H264);
     mediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
     mediaRecorder.setVideoEncodingBitRate(5 * 1024 * 1024);
     mediaRecorder.setVideoFrameRate(30);
     try {
       mediaRecorder.prepare();
     } catch (IOException e) {
       e.printStackTrace();
     }
 }

开始录屏

mediaRecorder.start();

保存路径

完整Service代码

public class ScreenService extends Service {
private MediaRecorder mediaRecorder;
private VirtualDisplay virtualDisplay;
private boolean running;
private int width = 720;
private int height = 1080;
private int dpi;
private ImageReader mImageReader;
private MediaProjection mediaProjection;

 @Override
 public IBinder onBind(Intent intent) {
   return new RecordBinder();
 }


 @Override
 public void onCreate() {
   super.onCreate();
   running = false;
   mediaRecorder = new MediaRecorder();
 }


 @Override
 public int onStartCommand(Intent intent, int flags, int startId) {
   return super.onStartCommand(intent, flags, startId);
 }


 @Override
 public void onDestroy() {
   super.onDestroy();
 }


 public void setMediaProject(MediaProjection project) {
   mediaProjection = project;
 }


 public boolean isRunning() {
   return running;
 }


 public void setConfig(int width, int height, int dpi) {
   this.width = width;
   this.height = height;
   this.dpi = dpi;
 }


 /**
 * 开始录屏
 *
 * @return true
 */
 public boolean startRecord() {
   if (mediaProjection == null || running) {
     return false;
   }
   initRecorder();
   createVirtualDisplay();
   mediaRecorder.start();
   running = true;
   return true;
 }


 /**
 * 结束录屏
 *
 * @return true
 */
 public boolean stopRecord() {
   if (!running) {
     return false;
   }
   running = false;
   mediaRecorder.stop();
   mediaRecorder.reset();
   virtualDisplay.release();
   mediaProjection.stop();

   return true;
 }


 public void setMediaProjection(MediaProjection mediaProjection) {
   this.mediaProjection = mediaProjection;
 }


 /**
 * 初始化ImageRead参数
 */
 public void initImageReader() {
   if (mImageReader == null) {
     int maxImages = 2;
     mImageReader = ImageReader.newInstance(width, height, PixelFormat.RGBA_8888, maxImages);
     createImageVirtualDisplay();
   }
 }


 /**
 * 创建一个录屏 Virtual
 */

 private void createVirtualDisplay() {
   virtualDisplay = mediaProjection
       .createVirtualDisplay("mediaprojection", width, height, dpi, DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR, mediaRecorder
           .getSurface(), null, null);
 }


 /**
 * 创建一个ImageReader Virtual
 */
 private void createImageVirtualDisplay() {
   virtualDisplay = mediaProjection
       .createVirtualDisplay("mediaprojection", width, height, dpi,
           DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR, mImageReader
           .getSurface(), null, null);
 }


 /**
 * 初始化保存屏幕录像的参数
 */
 private void initRecorder() {
   mediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
   mediaRecorder.setVideoSource(MediaRecorder.VideoSource.SURFACE);
   mediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
   mediaRecorder.setOutputFile(
       getSavePath() + System.currentTimeMillis() + ".mp4");
   mediaRecorder.setVideoSize(width, height);
   mediaRecorder.setVideoEncoder(MediaRecorder.VideoEncoder.H264);
   mediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
   mediaRecorder.setVideoEncodingBitRate(5 * 1024 * 1024);
   mediaRecorder.setVideoFrameRate(30);
   try {
     mediaRecorder.prepare();
   } catch (IOException e) {
     e.printStackTrace();
   }
 }


 /**
 * 获取一个保存屏幕录像的路径
 *
 * @return path
 */
 public String getSavePath() {
   if (Environment.getExternalStorageState()
          .equals(Environment.MEDIA_MOUNTED)) {
     String rootDir = Environment.getExternalStorageDirectory()
                   .getAbsolutePath() + "/" +
         "ScreenRecord" + "/";

     File file = new File(rootDir);
     if (!file.exists()) {
       if (!file.mkdirs()) {
         return null;
       }
     }
     return rootDir;
   } else {
     return null;
   }
 }


 /**
 * 请求完权限后马上获取有可能为null,可以通过判断is null来重复获取。
 */
 public Bitmap getBitmap() {
   Bitmap bitmap = cutoutFrame();
   if (bitmap == null) {
     getBitmap();
   }
   return bitmap;
 }


 /**
 * 通过底层来获取下一帧的图像
 *
 * @return bitmap
 */
 public Bitmap cutoutFrame() {
   Image image = mImageReader.acquireLatestImage();
   if (image == null) {
     return null;
   }
   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);
   return Bitmap.createBitmap(bitmap, 0, 0, width, height);
 }


 public class RecordBinder extends Binder {
   public ScreenService getRecordService() {
     return ScreenService.this;
   }
 }

demo下载

github

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# android视频截屏  # android手机录屏  # android  # 截屏录屏  # Android实现截屏方式整理(总结)  # Android截屏保存png图片的实例代码  # android截屏功能实现代码  # 使用python编写android截屏脚本双击运行即可  # android实现手机截屏并保存截图功能  # Android 使用Shell脚本截屏并自动传到电脑上  # Android截屏SurfaceView黑屏问题的解决办法  # Android实现截屏并保存操作功能  # Android 屏幕截屏方法汇总  # Android实现截屏功能  # 可以通过  # 创建一个  # 的是  # 有可能  # 中有  # 作了  # 下一  # 给大家  # 几种  # 解决办法  # 绑定  # 作好  # 回调  # 大家多多  # 就可以  # 黑屏  # 知道了  # 视频播放器  # isRunning  # mButton 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 宙斯浏览器视频悬浮窗怎么开启 边看视频边操作其他应用教程  bootstrap日历插件datetimepicker使用方法  如何在新浪SAE免费搭建个人博客?  Laravel如何实现一对一模型关联?(Eloquent示例)  laravel怎么为API路由添加签名中间件保护_laravel API路由签名中间件保护方法  HTML5空格在Angular项目里怎么处理_Angular中空格的渲染问题【详解】  制作旅游网站html,怎样注册旅游网站?  如何用腾讯建站主机快速创建免费网站?  UC浏览器如何切换小说阅读源_UC浏览器阅读源切换【方法】  如何选择可靠的免备案建站服务器?  PythonWeb开发入门教程_Flask快速构建Web应用  详解免费开源的.NET多类型文件解压缩组件SharpZipLib(.NET组件介绍之七)  如何用IIS7快速搭建并优化网站站点?  如何快速生成高效建站系统源代码?  夸克浏览器网页跳转延迟怎么办 夸克浏览器跳转优化  香港服务器WordPress建站指南:SEO优化与高效部署策略  javascript中对象的定义、使用以及对象和原型链操作小结  大学网站设计制作软件有哪些,如何将网站制作成自己app?  Laravel路由Route怎么设置_Laravel基础路由定义与参数传递规则【详解】  Javascript中的事件循环是如何工作的_如何利用Javascript事件循环优化异步代码?  如何在阿里云通过域名搭建网站?  ChatGPT怎么生成Excel公式_ChatGPT公式生成方法【指南】  Laravel如何集成第三方登录_Laravel Socialite实现微信QQ微博登录  Laravel Facade的原理是什么_深入理解Laravel门面及其工作机制  Laravel中Service Container是做什么的_Laravel服务容器与依赖注入核心概念解析  HTML 中动态设置元素 name 属性的正确语法详解  JS经典正则表达式笔试题汇总  怎样使用JSON进行数据交换_它有什么限制  JavaScript中如何操作剪贴板_ClipboardAPI怎么用  东莞市网站制作公司有哪些,东莞找工作用什么网站好?  微信小程序 require机制详解及实例代码  免费网站制作appp,免费制作app哪个平台好?  Laravel Eloquent性能优化技巧_Laravel N+1查询问题解决  jimdo怎样用html5做选项卡_jimdo选项卡html5实现与切换效果【指南】  香港服务器网站生成指南:免费资源整合与高速稳定配置方案  google浏览器怎么清理缓存_谷歌浏览器清除缓存加速详细步骤  如何实现建站之星域名转发设置?  PHP怎么接收前端传的文件路径_处理文件路径参数接收方法【汇总】  如何在服务器上配置二级域名建站?  phpredis提高消息队列的实时性方法(推荐)  JavaScript实现Fly Bird小游戏  如何解决hover在ie6中的兼容性问题  Win11怎么修改DNS服务器 Win11设置DNS加速网络【指南】  Laravel观察者模式如何使用_Laravel Model Observer配置  如何用VPS主机快速搭建个人网站?  利用vue写todolist单页应用  Win11怎么关闭资讯和兴趣_Windows11任务栏设置隐藏小组件  Laravel项目如何进行性能优化_Laravel应用性能分析与优化技巧大全  如何用低价快速搭建高质量网站?  Win11怎么关闭专注助手 Win11关闭免打扰模式设置【操作】