https://github.com/mabeijianxi/small-video-record.
List rates = mParameters.getSupportedPreviewFrameRates();
if (rates != null) {
if (rates.contains(MAX_FRAME_RATE)) {
mFrameRate = MAX_FRAME_RATE;
} else {
boolean findFrame = false;
Collections.sort(rates);
for (int i = rates.size() - 1; i >= 0; i--) {
if (rates.get(i) <= MAX_FRAME_RATE) {
mFrameRate = rates.get(i);
findFrame = true;
break;
}
}
if (!findFrame) {
mFrameRate = rates.get(0);
}
}
}
mParameters.setPreviewFrameRate(mFrameRate);
boolean findWidth = false;
for (int i = mSupportedPreviewSizes.size() - 1; i >= 0; i--) {
Size size = mSupportedPreviewSizes.get(i);
if (size.height == SMALL_VIDEO_WIDTH) {
mSupportedPreviewWidth = size.width;
findWidth = true;
break;
}
}
if (!findWidth) {
Log.e(getClass().getSimpleName(), "传入高度不支持或未找到对应宽度,请按照要求重新设置,否则会出现一些严重问题");
mSupportedPreviewWidth = 640;
SMALL_VIDEO_WIDTH = 480;
SMALL_VIDEO_HEIGHT = 360;
}
mParameters.setPreviewSize(mSupportedPreviewWidth, SMALL_VIDEO_WIDTH);
- -vf 可以添加滤镜,特别强大,可以旋转缩放剪切等等,我们需要用到旋转和剪切(我一直考虑需不需要用缩放的方式,因为这样可以在预览界面设置高分辨率看着清晰一些)。
transpose,旋转,对应的值有0、1、2、3,0:逆时针旋转90°然后垂直翻转1:顺时针旋转90°,2:逆时针旋转90°,3:顺时针旋转90°然后水平翻转。
剪切,关键字是crop,其有四个参数,分别是宽度、高度、其实剪切位置的X值与Y值,如ffmpeg -i a.mp4 -vf crop=480:360:0:0...;
-vcodec 指定视频编解码器;
-acodec 指定音频编解码器;
vbr 动态码率;
cbr 静态码率;
-crf 视频质量等级0~51,越大质量越差,建议18~28即可,与cbr模式不兼容;
-preset 转码速度,快慢的优劣应该都懂的,可根据自己业务场景设置,具体有:ultrafast、superfast、veryfast、faster、fast、medium、slow、slower、veryslow、placebo;
-i 指定输入;
-x264opts 配置其编解码参数;
maxrate 最大码率;
bitrate 固定码率;
-f 输出格式;
-s 设置帧大小。格式为 ‘wxh’;
-ss 指定开始时间;
-vframes 指定多少帧;
接着皆可在录制前配置我们的录制参数:
String cmd = String.format("filename = \"%s\"; ", result.mediaPath);
if (mCameraId == Camera.CameraInfo.CAMERA_FACING_BACK) {
cmd += String.format("addcmd = %s; ", " -vf \"transpose=1,crop=" + SMALL_VIDEO_WIDTH + ":" + SMALL_VIDEO_HEIGHT + ":0:0\" "
+getBitrateCrfSize(mediaRecorderConfig,"",true)
+getBitrateVelocity(mediaRecorderConfig,"",true)
+getBitrateModeCommand(mediaRecorderConfig,"",true));
} else {
cmd += String.format("addcmd = %s; ", " -vf \"transpose=2,crop=" + SMALL_VIDEO_WIDTH + ":" + SMALL_VIDEO_HEIGHT + ":0:0\" "
+getBitrateCrfSize(mediaRecorderConfig,"",true)
+getBitrateVelocity(mediaRecorderConfig,"",true)
+getBitrateModeCommand(mediaRecorderConfig,"",true));
}
我们这里设置了旋转滤镜与剪切滤镜,由于我们录制竖屏视频所以旋转90°,然后剪切为我们制定的视频尺寸。当然里面还有三个get函数,分别是视频质量等级、转码速度、码率模式。
- 视频质量等级命令为-crf [size]:
protected String getBitrateCrfSize(BaseMediaBitrateConfig config, String defualtCmd, boolean nendSymbol) { if (TextUtils.isEmpty(defualtCmd)) { defualtCmd = ""; } String add = ""; if (config != null && config.getMode() == BaseMediaBitrateConfig.MODE.AUTO_VBR && config.getCrfSize() > 0) { if (nendSymbol) { add = String.format("-crf \"%d\" ", config.getCrfSize()); } else { add = String.format("-crf %d ", config.getCrfSize()); } } else { return defualtCmd; } return add;
- 转码速度命令为-preset [what]:
protected String getBitrateVelocity(BaseMediaBitrateConfig config, String defualtCmd, boolean nendSymbol) { if (TextUtils.isEmpty(defualtCmd)) { defualtCmd = ""; } String add = ""; if (config != null && !TextUtils.isEmpty(config.getVelocity())) { if (nendSymbol) { add = String.format("-preset \"%s\" ", config.getVelocity()); } else { add = String.format("-preset %s ", config.getVelocity()); } } else { return defualtCmd; } return add;
- 码率模式: 码率模式分为vbr与cbr,我在里面加了三个类AutoVBRMode、VBRMode、CBRMode,三者都可传入转码速度。如果不想管那么多那么只需传入无参的AutoVBRMode对象即可,只有AutoVBRMode模式下可以传入视频质量等级值,这个值将最大程度上控制视频质量。VBRMode模式下可以指定最大码率与额定码率。、CBRMode模式下出入一个固定码率即可。
protected String getBitrateModeCommand(BaseMediaBitrateConfig config, String defualtCmd, boolean needSymbol) { String add = ""; if (TextUtils.isEmpty(defualtCmd)) { defualtCmd = ""; } if (config != null) { if (config.getMode() == BaseMediaBitrateConfig.MODE.VBR) { if (needSymbol) { add = String.format(" -x264opts \"bitrate=%d:vbv-maxrate=%d\" ", config.getBitrate(), config.getMaxBitrate()); } else { add = String.format(" -x264opts bitrate=%d:vbv-maxrate=%d ", config.getBitrate(), config.getMaxBitrate()); } return add; } else if (mediaRecorderConfig.getMode() == BaseMediaBitrateConfig.MODE.CBR) { if (needSymbol) { add = String.format(" -x264opts \"bitrate=%d:vbv-bufsize=%d:nal_hrd=cbr\" ", config.getBitrate(), config.getBufSize()); } else { add = String.format(" -x264opts bitrate=%d:vbv-bufsize=%d:nal_hrd=cbr ", config.getBitrate(), config.getBufSize()); } return add; } } return defualtCmd; }
配置好后即可开始录制,在camera的数据回调里面把数据转入底层。
@Override
public void onPreviewFrame(byte[] data, Camera camera) {
if (mRecording) {
UtilityAdapter.RenderDataYuv(data);
}
super.onPreviewFrame(data, camera);
}
3、多段视频合并
录制过程中我们可以暂停录制,这个可能生成n段短视频,这个我们就需要合并视频了,利用FFmpeg命令也可以轻松实现:
//合并ts流
String cmd = String.format("ffmpeg %s -i \"%s\" -vcodec copy -acodec copy -absf aac_adtstoasc -f mp4 -movflags faststart \"%s\" ",
FFMpegUtils.getLogCommand(),
mMediaObject.getConcatYUV(),
mMediaObject.getOutputTempVideoPath());
boolean mergeFlag = UtilityAdapter.FFmpegRun("", cmd) == 0;
4、进一步转码压缩
如果没有设置 doH264Compress 参数那么将不执行以下逻辑
String vbr = " -vbr 4 ";
if (compressConfig != null && compressConfig.getMode()==BaseMediaBitrateConfig.MODE.CBR) {
vbr = "";
}
String cmd_transcoding = String.format("ffmpeg -i %s -c:v libx264 %s %s %s -c:a libfdk_aac %s %s",
mMediaObject.getOutputTempVideoPath(),
getBitrateModeCommand(compressConfig,"",false),
getBitrateCrfSize(compressConfig, "-crf 28", false),
getBitrateVelocity(compressConfig, "-preset:v veryfast", false),
vbr,
mMediaObject.getOutputTempTranscodingVideoPath()
);
boolean transcodingFlag = UtilityAdapter.FFmpegRun("", cmd_transcoding) == 0;
5、截取视频中的一帧作为封面
public static boolean captureThumbnails(String videoPath, String outputPath, String wh, String ss) {
if (ss == null)
ss = "";
else
ss = " -ss " + ss;
String cmd = String.format("ffmpeg -i \"%s\"%s -s %s -vframes 1 \"%s\"", videoPath, ss, wh, outputPath);
return UtilityAdapter.FFmpegRun("", cmd) == 0;
}
本库的优点是简单便捷,可控性强,后期将继续维护。缺点是FFmpeg优点老,后期会考虑自己编译一份,那时利用FFmpeg玩转Android视频录制与压缩(二)也就出来了,有点开始期待了,有兴趣的同学欢迎到我github上指教https://github.com/mabeijianxi/small-video-record.
现已分享:利用FFmpeg玩转Android视频录制与压缩(二)