android.MediaMuxer时间裁剪

2024-02-02 09:44

本文主要是介绍android.MediaMuxer时间裁剪,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

使用MediaMuxer裁剪视频_安卓muxer 裁剪视频画布-CSDN博客

关键步骤

mediaExtractor.seekTo(beginTime, MediaExtractor.SEEK_TO_PREVIOUS_SYNC);long presentTimeUs = mediaExtractor.getSampleTime();
if (presentTimeUs > endTime)break;

功能代码

VideoView videoView_video;
ImageButton imageButton_startTime, imageButton_endTime;
Button button_clip;
String path_video="";
int startTime, endTime;imageButton_startTime = (ImageButton)findViewById(R.id.imageButton_startTime);
imageButton_startTime.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {startTime = videoView_video.getCurrentPosition(); //msToast.makeText(getApplicationContext(), "开始时间:" + startTime, Toast.LENGTH_SHORT).show();}
});imageButton_endTime = (ImageButton)findViewById(R.id.imageButton_endTime);
imageButton_endTime.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {endTime = videoView_video.getCurrentPosition();Toast.makeText(getApplicationContext(), "结束时间:" + endTime, Toast.LENGTH_SHORT).show();}
});button_clip = (Button)findViewById(R.id.button_clip);
button_clip.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {if (path_video.equals("")) {Toast.makeText(getApplicationContext(), "错误:没有视频", Toast.LENGTH_SHORT).show();return;}if (startTime >= endTime) {Toast.makeText(getApplicationContext(), "错误:开始时间(" + startTime + ") > 结束时间(" + endTime + ")", Toast.LENGTH_SHORT).show();return;}String path_output = Environment.getExternalStorageDirectory().getPath() + File.separator + Environment.DIRECTORY_MOVIES + path_video.substring(path_video.lastIndexOf("/"), path_video.lastIndexOf(".")) + "_clip.mp4";clipVideo(path_video, path_output, startTime, endTime);}
});

 

void clipVideo(String path_video, String path_output, int startTime, int endTime) {MediaExtractor mediaExtractor = new MediaExtractor();try {mediaExtractor.setDataSource(path_video);} catch (Exception e) {Toast.makeText(getApplicationContext(), e.toString(), Toast.LENGTH_SHORT).show();Log.e(Thread.currentThread().getStackTrace()[2] + "", e.toString());}int videoTrack = -1,audioTrack = -1;if (mediaExtractor.getTrackCount() == 0) {Toast.makeText(getApplicationContext(), "错误:trackCount == 0", Toast.LENGTH_SHORT).show();return;}for (int i = 0; i < mediaExtractor.getTrackCount(); i++) {if (mediaExtractor.getTrackFormat(i).getString(MediaFormat.KEY_MIME).startsWith("video/")) {videoTrack = i;} else if (mediaExtractor.getTrackFormat(i).getString(MediaFormat.KEY_MIME).startsWith("audio/")) {audioTrack = i;}}Toast.makeText(getApplicationContext(), "videoTrack: " + videoTrack + "\naudioTrack: " + audioTrack, Toast.LENGTH_SHORT).show();Log.e(Thread.currentThread().getStackTrace()[2] + "", "videoTrack: " + videoTrack + ", audioTrack: " + audioTrack);MediaFormat videoFormat = mediaExtractor.getTrackFormat(videoTrack);MediaFormat audioFormat = mediaExtractor.getTrackFormat(audioTrack);MediaMuxer mediaMuxer = null;try {mediaMuxer = new MediaMuxer(path_output, MediaMuxer.OutputFormat.MUXER_OUTPUT_MPEG_4);} catch (Exception e) {Toast.makeText(getApplicationContext(), e.toString(), Toast.LENGTH_SHORT).show();Log.e(Thread.currentThread().getStackTrace()[2] + "", e.toString());}int videoTrackIndex = mediaMuxer.addTrack(videoFormat);//int rotation = videoFormat.getInteger(MediaFormat.KEY_ROTATION); //视频的rotation//mediaMuxer.setOrientationHint(rotation);int audioTrackIndex = mediaMuxer.addTrack(audioFormat);mediaMuxer.start();int bufferSize = 256 * 1024;ByteBuffer inputBuffer = ByteBuffer.allocate(bufferSize);//视频部分mediaExtractor.selectTrack(videoTrack);MediaCodec.BufferInfo videoInfo = new MediaCodec.BufferInfo();mediaExtractor.seekTo(startTime * 1000L, MediaExtractor.SEEK_TO_PREVIOUS_SYNC);while (true) {int sampleSize = mediaExtractor.readSampleData(inputBuffer, 0);long presentationTimeUs = mediaExtractor.getSampleTime();if (sampleSize < 0 || presentationTimeUs > endTime * 1000L) {mediaExtractor.unselectTrack(videoTrack);break;}//int trackIndex = mediaExtractor.getSampleTrackIndex();int sampleFlag = mediaExtractor.getSampleFlags();videoInfo.offset = 0;videoInfo.size = sampleSize;videoInfo.flags = sampleFlag;videoInfo.presentationTimeUs = presentationTimeUs;mediaMuxer.writeSampleData(videoTrackIndex, inputBuffer, videoInfo);mediaExtractor.advance();}//音频mediaExtractor.selectTrack(audioTrack);MediaCodec.BufferInfo audioInfo = new MediaCodec.BufferInfo();mediaExtractor.seekTo(startTime * 1000L, MediaExtractor.SEEK_TO_CLOSEST_SYNC);while (true) {int sampleSize = mediaExtractor.readSampleData(inputBuffer, 0);long presentationTimeUs = mediaExtractor.getSampleTime();if (sampleSize < 0 || presentationTimeUs > endTime * 1000L) {mediaExtractor.unselectTrack(audioTrack);break;}//int trackIndex = mediaExtractor.getSampleTrackIndex();int sampleFlag = mediaExtractor.getSampleFlags();audioInfo.offset = 0;audioInfo.size = sampleSize;audioInfo.flags = sampleFlag;audioInfo.presentationTimeUs = presentationTimeUs;mediaMuxer.writeSampleData(audioTrackIndex, inputBuffer, audioInfo);mediaExtractor.advance();}mediaMuxer.stop();mediaMuxer.release();mediaExtractor.release();MediaScannerConnection.scanFile(getApplicationContext(), new String[]{path_output}, null, null);Toast.makeText(getApplicationContext(), "视频剪辑完成:" + path_output, Toast.LENGTH_SHORT).show();
}

源码:GitLink | 确实开源

这篇关于android.MediaMuxer时间裁剪的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/670305

相关文章

Android实现打开本地pdf文件的两种方式

《Android实现打开本地pdf文件的两种方式》在现代应用中,PDF格式因其跨平台、稳定性好、展示内容一致等特点,在Android平台上,如何高效地打开本地PDF文件,不仅关系到用户体验,也直接影响... 目录一、项目概述二、相关知识2.1 PDF文件基本概述2.2 android 文件访问与存储权限2.

Android Studio 配置国内镜像源的实现步骤

《AndroidStudio配置国内镜像源的实现步骤》本文主要介绍了AndroidStudio配置国内镜像源的实现步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、修改 hosts,解决 SDK 下载失败的问题二、修改 gradle 地址,解决 gradle

Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码

《Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码》:本文主要介绍Java中日期时间转换的多种方法,包括将Date转换为LocalD... 目录一、Date转LocalDateTime二、Date转LocalDate三、LocalDateTim

在Android平台上实现消息推送功能

《在Android平台上实现消息推送功能》随着移动互联网应用的飞速发展,消息推送已成为移动应用中不可或缺的功能,在Android平台上,实现消息推送涉及到服务端的消息发送、客户端的消息接收、通知渠道(... 目录一、项目概述二、相关知识介绍2.1 消息推送的基本原理2.2 Firebase Cloud Me

golang获取当前时间、时间戳和时间字符串及它们之间的相互转换方法

《golang获取当前时间、时间戳和时间字符串及它们之间的相互转换方法》:本文主要介绍golang获取当前时间、时间戳和时间字符串及它们之间的相互转换,本文通过实例代码给大家介绍的非常详细,感兴趣... 目录1、获取当前时间2、获取当前时间戳3、获取当前时间的字符串格式4、它们之间的相互转化上篇文章给大家介

Feign Client超时时间设置不生效的解决方法

《FeignClient超时时间设置不生效的解决方法》这篇文章主要为大家详细介绍了FeignClient超时时间设置不生效的原因与解决方法,具有一定的的参考价值,希望对大家有一定的帮助... 在使用Feign Client时,可以通过两种方式来设置超时时间:1.针对整个Feign Client设置超时时间

springboot+dubbo实现时间轮算法

《springboot+dubbo实现时间轮算法》时间轮是一种高效利用线程资源进行批量化调度的算法,本文主要介绍了springboot+dubbo实现时间轮算法,文中通过示例代码介绍的非常详细,对大家... 目录前言一、参数说明二、具体实现1、HashedwheelTimer2、createWheel3、n

Android中Dialog的使用详解

《Android中Dialog的使用详解》Dialog(对话框)是Android中常用的UI组件,用于临时显示重要信息或获取用户输入,本文给大家介绍Android中Dialog的使用,感兴趣的朋友一起... 目录android中Dialog的使用详解1. 基本Dialog类型1.1 AlertDialog(

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda