移动终端高级开发Android Studio Camera2的录像功能
一:功能实现
利用MediaRecorder+Camera2框架实现基本的录像功能,当APP启动时自动打开摄像头,点击按钮开始录像,点击按钮停止录像并将录制的视频文件保存在sdcard/my_video/中
二:操作步骤
1.在activity_main.xml中添加停止按钮
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/root">
<lmh.example.zy01.AutoFitTextureView
android:id="@+id/texture"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
<ImageButton
android:id="@+id/capture"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="bottom"
android:src="@drawable/capture" />
<ImageButton
android:id="@+id/stop"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="end|bottom"
android:src="@drawable/stop"
/>
</FrameLayout>
2.修改setUpCameraOutputs函数,加入MediaRecorder的相关配置
CameraManager manager = (CameraManager) getSystemService(Context.CAMERA_SERVICE);
try {
// 获取指定摄像头的特性
CameraCharacteristics characteristics = manager.getCameraCharacteristics(mCameraId);
// 获取摄像头支持的配置属性
StreamConfigurationMap map = characteristics.get(CameraCharacteristics.
SCALER_STREAM_CONFIGURATION_MAP);
// 获取摄像头支持的最大尺寸
Size largest = Collections.max(
Arrays.asList(map.getOutputSizes(ImageFormat.JPEG)),
new CompareSizesByArea());
mVideoSize = chooseVideoSize(map.getOutputSizes(MediaRecorder.class));
mPreviewSize = chooseOptimalSize(map.getOutputSizes(SurfaceTexture.class),
width, height, mVideoSize);
// 获取最佳的预览尺寸
previewSize = chooseOptimalSize(map.getOutputSizes(SurfaceTexture.class),
width, height, largest);
// 根据选中的预览尺寸来调整预览组件(TextureView的)的长宽比
int orientation = getResources().getConfiguration().orientation;
if (orientation == Configuration.ORIENTATION_LANDSCAPE) {
textureView.setAspectRatio(previewSize.getWidth(), previewSize.getHeight());
} else {
textureView.setAspectRatio(previewSize.getHeight(), previewSize.getWidth());
}
SetpMediaRecorder();
}
catch (CameraAccessException e)
{
e.printStackTrace();
}
catch (NullPointerException e)
{
System.out.println("出现错误。");
}
3.创建startRecordingVideo()函数并绑定录像按钮,实现开始录像的功能
①创建CaptureRequest.Builder对象,请求摄像操作CameraDevice.TEMPLATE_RECORD
② 设置参数(可参考EXAM1-11中的captureStillPicture方法,并将此次请求添加一个MeidaRecorder.getSurface()对象作为图像的输出目标容器
③ 创建CaptureRequest对象
④ 创建CameraCaptureSession对象,并传入Request对象;
⑤ 在CameraCaptureSession.StateCallback()回调方法onConfigured中执行setRepeatingRequest()方法,并调用MediaRecorder的start开始录像
4.创建stopRecordingVideo()函数并绑定停止按钮,实现停止录像的功能
① 调用MediaRecorder的stop()方法停止录像,并回到预览画面
三: 实例源码
(MainActivity.java; AutoFitTextureView.java )
MainActivity.java
import android.Manifest;
import android.app.Activity;
import android.content.Context;
import android.content.pm.PackageManager;
import android.content.res.Configuration;
import android.graphics.ImageFormat;
import android.graphics.Matrix;
import android.graphics.RectF;
import android.graphics.SurfaceTexture;
import android.hardware.camera2.CameraAccessException;
import android.hardware.camera2.CameraCaptureSession;
import android.hardware.camera2.CameraCharacteristics;
import android.hardware.camera2.CameraDevice;
import android.hardware.camera2.CameraManager;
import android.hardware.camera2.CameraMetadata;
import android.hardware.camera2.CaptureRequest;
import android.hardware.camera2.params.StreamConfigurationMap;
import android.media.MediaRecorder;
import android.os.Build;
import android.os.Bundle;
import android.os.Environment;
import android.util.Log;
import android.util.Size;
import android.util.SparseIntArray;
import android.view.Surface;
import android.view.TextureView;
import android.widget.FrameLayout;
import android.widget.ImageButton;
import android.widget.Toast;
import androidx.annotation.NonNull;
import androidx.annotation.RequiresApi;
import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;
import static android.content.ContentValues.TAG;
@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
public class MainActivity extends Activity
{
private static final SparseIntArray ORIENTATIONS = new SparseIntArray();
static {
ORIENTATIONS.append(Surface.ROTATION_0, 90);
ORIENTATIONS.append<