Android 오디 오 비디오 의 전체 코드
카 메 라 를 켜다
private void openCamera(int position) {
if (mCamera == null) {
mCamera = Camera.open(position);
int degree = getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE ? 0 : 90;
mCamera.setDisplayOrientation(degree);
}
}
camera 는 기본적으로 가로로 되 어 있 기 때문에 세로 로 녹음 하려 면 90 도 를 돌려 야 합 니 다.
int degree = getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE ? 0 : 90;
mCamera.setDisplayOrientation(degree);
카메라 미리보기우 리 는 우리 가 표시 하고 자 하 는 SurfaceView 크기 와 가장 가 까 운 camera 미리 보기 크기 를 선택 하려 고 합 니 다.여 기 는 특히 camera 가 지원 하 는 너비 가 높 은 것 보다 넓 은 것 을 주의해 야 합 니 다.
그래서 다음 선택 코드 가 생 겼 어 요.
private Size getBestCameraResolution(Camera.Parameters parameters, Size screenResolution) {
float tmp = 0f;
float mindiff = 100f;
Log.e("yuanVideo", "screen width=" + screenResolution.getWidth() + " height=" + screenResolution.getHeight());
float width_d_height;
if (screenResolution.getWidth() > screenResolution.getHeight()) {
width_d_height = (float) screenResolution.getWidth() / (float) screenResolution.getHeight();
} else {
width_d_height = (float) screenResolution.getHeight() / (float) screenResolution.getWidth();
}
Log.e("yuanVideo", "width_d_height=" + width_d_height);
Camera.Size best = null;
List<Camera.Size> supportedPreviewSizes = parameters.getSupportedPreviewSizes();
for (Camera.Size s : supportedPreviewSizes) {
tmp = Math.abs(((float) s.width / (float) s.height) - width_d_height);
Log.e("yuanVideo", "support width=" + s.width + " height=" + s.height + " ratio=" + tmp);
if (tmp < mindiff) {
mindiff = tmp;
best = s;
}
}
Log.e("yuanVideo", "best width=" + best.width + " height=" + best.height);
return new Size(best.width, best.height);
}
MediaRecorder 초기 화
private boolean prepareMediaRecorder() {
// MediaPlayer
mCamera.unlock();
mRecorder = new MediaRecorder();
mRecorder.reset();
mRecorder.setCamera(mCamera);
// ( AudioSource.CAMCORDER)
mRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
//
mRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);
Log.e("yuanProfile", "QUALITY_LOW=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_LOW));
Log.e("yuanProfile", "QUALITY_HIGH=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_HIGH));
Log.e("yuanProfile", "QUALITY_QCIF=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_QCIF));
Log.e("yuanProfile", "QUALITY_480P=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_480P));
Log.e("yuanProfile", "QUALITY_720P=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_720P));
Log.e("yuanProfile", "QUALITY_1080P=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_1080P));
Log.e("yuanProfile", "QUALITY_QVGA=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_QVGA));
Log.e("yuanProfile", "QUALITY_2160P=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_2160P));
Log.e("yuanProfile", "QUALITY_VGA=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_VGA));
Log.e("yuanProfile", "QUALITY_4KDCI=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_4KDCI));
Log.e("yuanProfile", "QUALITY_QHD=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_QHD));
Log.e("yuanProfile", "QUALITY_2K=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_2K));
if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_720P)) {
mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_720P));
} else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_480P)) {
mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_1080P));
} else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_1080P)) {
mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_480P));
} else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_LOW)) {
mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_LOW));
} else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_HIGH)) {
mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_HIGH));
} else {
return false;
}
// mTempList.add(mCurrentTempRecordData);
mRecorder.setOutputFile(mCurPath);
mRecorder.setPreviewDisplay(activtityVideoRecordBinding.sView.getHolder().getSurface()); // ①
int degree;
if(getResources().getConfiguration().orientation != Configuration.ORIENTATION_LANDSCAPE){
int degree ;
if (cameraPosition == Camera.CameraInfo.CAMERA_FACING_FRONT) {
degree = 270;
} else {
degree = 90;
}
mRecorder.setOrientationHint(degree);
}
try {
mRecorder.prepare();
} catch (Exception e) {
e.printStackTrace();
return false;
}
return true;
}
여기 도 동 영상의 회전 파 라 메 터 를 설정 해 야 합 니 다.
if(getResources().getConfiguration().orientation != Configuration.ORIENTATION_LANDSCAPE){
int degree ;
if (cameraPosition == Camera.CameraInfo.CAMERA_FACING_FRONT) {
degree = 270;
} else {
degree = 90;
}
mRecorder.setOrientationHint(degree);
}
다음은 완전한 코드 입 니 다.
package com.yuanxuzhen.ffmpeg;
import android.app.Activity;
import android.content.Context;
import android.content.res.Configuration;
import android.graphics.PixelFormat;
import android.hardware.Camera;
import android.media.CamcorderProfile;
import android.media.MediaRecorder;
import android.os.Build;
import android.os.Bundle;
import android.util.DisplayMetrics;
import android.util.Log;
import android.util.Size;
import android.view.SurfaceHolder;
import android.view.View;
import android.view.Window;
import android.view.WindowManager;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.annotation.RequiresApi;
import com.yuanxuzhen.ffmpeg.databinding.ActivtityVideoRecordBinding;
import java.io.File;
import java.io.IOException;
import java.util.List;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class VideoRecordActivity extends Activity {
ActivtityVideoRecordBinding activtityVideoRecordBinding;
MediaRecorder mRecorder;
private boolean isRecording = false;
private int cameraPosition = Camera.CameraInfo.CAMERA_FACING_FRONT;//0 ,1
private Camera mCamera;
private Camera.Parameters mParameters;
private String mCurPath = null;
private VideoTempRecordData mCurrentTempRecordData = null;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
requestWindowFeature(Window.FEATURE_NO_TITLE);
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
WindowManager.LayoutParams.FLAG_FULLSCREEN);
getWindow().setFormat(PixelFormat.TRANSLUCENT);
mCurPath = DirUtil.getCacheDir(this) + File.separator + "out.mp4";
activtityVideoRecordBinding = ActivtityVideoRecordBinding.inflate(getLayoutInflater());
setContentView(activtityVideoRecordBinding.getRoot());
activtityVideoRecordBinding.sView.getHolder().setKeepScreenOn(true);
activtityVideoRecordBinding.sView.getHolder().addCallback(new SurfaceHolder.Callback() {
@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
@Override
public void surfaceCreated(@NonNull SurfaceHolder holder) {
openPreView();
}
@Override
public void surfaceChanged(SurfaceHolder surfaceHolder, int format, int width, int height) {
Log.e("yuanVideo", "surfaceChanged width=" + width + " height=" + height);
}
@Override
public void surfaceDestroyed(@NonNull SurfaceHolder holder) {
}
});
activtityVideoRecordBinding.recordOrStop.setText(" ");
activtityVideoRecordBinding.recordOrStop.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (isRecording) {
Log.d("TAG", " ");
stopRecord();
} else {
startRecord();
}
}
});
activtityVideoRecordBinding.change.setOnClickListener(new View.OnClickListener() {
@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
@Override
public void onClick(View v) {
if(isRecording){
return;
}
releaseCamera();
cameraPosition = cameraPosition == Camera.CameraInfo.CAMERA_FACING_FRONT ? Camera.CameraInfo.CAMERA_FACING_BACK : Camera.CameraInfo.CAMERA_FACING_FRONT;
openCamera(cameraPosition);
openPreView();
}
});
}
/**
* 1.
*/
private void openCamera(int position) {
if (mCamera == null) {
mCamera = Camera.open(position);
int degree = getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE ? 0 : 90;
mCamera.setDisplayOrientation(degree);
}
}
/**
* initCameraAndSurfaceViewHolder hoder
* 2.
*/
@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
private void openPreView() {
try {
if (mCamera != null) {
mParameters = mCamera.getParameters();
mCamera.setPreviewDisplay(activtityVideoRecordBinding.sView.getHolder());
Size screenPoint = getScreenMetrics(VideoRecordActivity.this);
Size bestPreviewSize = getBestCameraResolution(mCamera.getParameters(), screenPoint);
mParameters.setPreviewSize(bestPreviewSize.getWidth(), bestPreviewSize.getHeight());
mCamera.setParameters(mParameters);
mCamera.startPreview();
mCamera.autoFocus(new Camera.AutoFocusCallback() {
@Override
public void onAutoFocus(boolean success, Camera camera) {
Log.e("yuanVideo", "autoFocus success=" + success);
}
});
mCamera.setPreviewCallback(new Camera.PreviewCallback() {
@Override
public void onPreviewFrame(byte[] data, Camera camera) {
Log.i("TAG", " ...");
Log.d("TAG", " :" + String.valueOf(data.length));
}
});
}
} catch (IOException e) {
e.printStackTrace();
}
}
private boolean prepareMediaRecorder() {
// MediaPlayer
mCamera.unlock();
mRecorder = new MediaRecorder();
mRecorder.reset();
mRecorder.setCamera(mCamera);
// ( AudioSource.CAMCORDER)
mRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
//
mRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);
Log.e("yuanProfile", "QUALITY_LOW=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_LOW));
Log.e("yuanProfile", "QUALITY_HIGH=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_HIGH));
Log.e("yuanProfile", "QUALITY_QCIF=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_QCIF));
Log.e("yuanProfile", "QUALITY_480P=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_480P));
Log.e("yuanProfile", "QUALITY_720P=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_720P));
Log.e("yuanProfile", "QUALITY_1080P=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_1080P));
Log.e("yuanProfile", "QUALITY_QVGA=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_QVGA));
Log.e("yuanProfile", "QUALITY_2160P=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_2160P));
Log.e("yuanProfile", "QUALITY_VGA=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_VGA));
Log.e("yuanProfile", "QUALITY_4KDCI=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_4KDCI));
Log.e("yuanProfile", "QUALITY_QHD=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_QHD));
Log.e("yuanProfile", "QUALITY_2K=" + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_2K));
if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_720P)) {
mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_720P));
} else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_480P)) {
mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_1080P));
} else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_1080P)) {
mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_480P));
} else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_LOW)) {
mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_LOW));
} else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_HIGH)) {
mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_HIGH));
} else {
return false;
}
// mTempList.add(mCurrentTempRecordData);
mRecorder.setOutputFile(mCurPath);
mRecorder.setPreviewDisplay(activtityVideoRecordBinding.sView.getHolder().getSurface()); // ①
if(getResources().getConfiguration().orientation != Configuration.ORIENTATION_LANDSCAPE){
int degree ;
if (cameraPosition == Camera.CameraInfo.CAMERA_FACING_FRONT) {
degree = 270;
} else {
degree = 90;
}
mRecorder.setOrientationHint(degree);
}
try {
mRecorder.prepare();
} catch (Exception e) {
e.printStackTrace();
return false;
}
return true;
}
private void startRecord() {
if (prepareMediaRecorder()) {
mRecorder.start();
isRecording = true;
activtityVideoRecordBinding.recordOrStop.setText(" ");
} else {
releaseMediaRecorder();
isRecording = false;
activtityVideoRecordBinding.recordOrStop.setText(" ");
}
}
private void stopRecord() {
if (mRecorder == null) {
return;
}
mRecorder.stop();
releaseMediaRecorder();
isRecording = false;
activtityVideoRecordBinding.recordOrStop.setText(" ");
}
@Nullable
@Override
public CharSequence onCreateDescription() {
return super.onCreateDescription();
}
@Override
protected void onDestroy() {
releaseCamera();
releaseMediaRecorder();
super.onDestroy();
}
/**
*
*/
private void releaseCamera() {
if (mCamera != null) {
mCamera.setPreviewCallback(null);
mCamera.stopPreview();
mCamera.release();
mCamera = null;
}
}
private void releaseMediaRecorder() {
if (mRecorder != null) {
mRecorder.reset();
mRecorder.release();
mRecorder = null;
mCamera.lock();
}
}
@Override
protected void onResume() {
super.onResume();
openCamera(cameraPosition);
}
@Override
protected void onPause() {
super.onPause();
releaseMediaRecorder();
releaseCamera();
}
/**
*
*
* @param parameters
* @param screenResolution
* @return
*/
@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
private Size getBestCameraResolution(Camera.Parameters parameters, Size screenResolution) {
float tmp = 0f;
float mindiff = 100f;
Log.e("yuanVideo", "screen width=" + screenResolution.getWidth() + " height=" + screenResolution.getHeight());
float width_d_height;
if (screenResolution.getWidth() > screenResolution.getHeight()) {
width_d_height = (float) screenResolution.getWidth() / (float) screenResolution.getHeight();
} else {
width_d_height = (float) screenResolution.getHeight() / (float) screenResolution.getWidth();
}
Log.e("yuanVideo", "width_d_height=" + width_d_height);
Camera.Size best = null;
List<Camera.Size> supportedPreviewSizes = parameters.getSupportedPreviewSizes();
for (Camera.Size s : supportedPreviewSizes) {
tmp = Math.abs(((float) s.width / (float) s.height) - width_d_height);
Log.e("yuanVideo", "support width=" + s.width + " height=" + s.height + " ratio=" + tmp);
if (tmp < mindiff) {
mindiff = tmp;
best = s;
}
}
Log.e("yuanVideo", "best width=" + best.width + " height=" + best.height);
return new Size(best.width, best.height);
}
/**
* , px
*
* @param context
* @return
*/
@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
public static Size getScreenMetrics(Context context) {
DisplayMetrics dm = context.getResources().getDisplayMetrics();
int w_screen = dm.widthPixels;
int h_screen = dm.heightPixels;
return new Size(w_screen, h_screen);
}
}
배치
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">
<!-- SurfaceView -->
<com.yuanxuzhen.ffmpeg.ResizeAbleSurfaceView
android:id="@+id/sView"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_centerInParent="true"/>
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_alignParentTop="true"
>
<TextView
android:id="@+id/tv_time"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="0 "
android:layout_centerInParent="true"
android:textColor="@color/white"
/>
<Button
android:id="@+id/change"
android:layout_width="wrap_content"
android:layout_height="66dp"
android:text=" "
android:layout_alignParentEnd="true"
/>
</RelativeLayout>
<LinearLayout
android:orientation="horizontal"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:gravity="center_horizontal"
android:layout_alignParentBottom="true"
android:layout_centerHorizontal="true">
<Button
android:id="@+id/record_or_stop"
android:layout_width="66dp"
android:layout_height="66dp"
android:text=" "
/>
<Button
android:id="@+id/save"
android:layout_width="66dp"
android:layout_height="66dp"
android:text=" "
/>
</LinearLayout>
</RelativeLayout>
package com.yuanxuzhen.ffmpeg;
import android.content.Context;
import android.util.AttributeSet;
import android.view.SurfaceView;
public class ResizeAbleSurfaceView extends SurfaceView {
private int mWidth = -1;
private int mHeight = -1;
public ResizeAbleSurfaceView(Context context) {
super(context);
}
public ResizeAbleSurfaceView(Context context, AttributeSet attrs) {
super(context, attrs);
}
public ResizeAbleSurfaceView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
if (-1 == mWidth || -1 == mHeight) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
}
else {
setMeasuredDimension(mWidth, mHeight);
}
}
public void resize(int width, int height) {
mWidth = width;
mHeight = height;
getHolder().setFixedSize(width, height);
requestLayout();
invalidate();
}
}
package com.yuanxuzhen.ffmpeg;
import android.content.Context;
import android.os.Environment;
import java.io.File;
public class DirUtil {
public static final String WEBVIEW_CACHE = ".webviewCache";
public static final String IMAGE_PATH = "image";
public static final String DOWNLOAD_PATH = "download";
public static final String VIDEO_PATH = ".video";
public static final String NET_PATH = ".net";
//image
public static String getImageDir(Context context) {
return getCacheDir(context) + File.separator + IMAGE_PATH;
}
//webview
public static String getWebviewCache(Context context) {
return getCacheDir(context) + File.separator + WEBVIEW_CACHE;
}
//download
public static String getDownloadDir(Context context) {
return getCacheDir(context) + File.separator + DOWNLOAD_PATH;
}
//video
public static String getVideoPath(Context context) {
return getCacheDir(context) + File.separator + VIDEO_PATH;
}
//net
public static String getNetPath(Context context) {
return getCacheDir(context) + File.separator + NET_PATH;
}
public static String getCacheDir(Context context) {
if (context == null) {
return "";
}
String path = null;
if (context.getExternalCacheDir() != null
&& (Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())
|| !Environment.isExternalStorageRemovable())) {
//
path = context.getExternalCacheDir().getPath();
} else {
//
path = context.getCacheDir().getPath();
}
return path;
}
}
이상 은 안 드 로 이 드 녹음 동 영상의 상세 한 내용 입 니 다.안 드 로 이 드 녹음 동 영상 에 관 한 자 료 는 저희 의 다른 관련 글 을 주목 해 주 십시오!
이 내용에 흥미가 있습니까?
현재 기사가 여러분의 문제를 해결하지 못하는 경우 AI 엔진은 머신러닝 분석(스마트 모델이 방금 만들어져 부정확한 경우가 있을 수 있음)을 통해 가장 유사한 기사를 추천합니다:
Bitrise에서 배포 어플리케이션 설정 테스트하기이 글은 Bitrise 광고 달력의 23일째 글입니다. 자체 또는 당사 등에서 Bitrise 구축 서비스를 사용합니다. 그나저나 며칠 전 Bitrise User Group Meetup #3에서 아래 슬라이드를 발표했...
텍스트를 자유롭게 공유하거나 복사할 수 있습니다.하지만 이 문서의 URL은 참조 URL로 남겨 두십시오.
CC BY-SA 2.5, CC BY-SA 3.0 및 CC BY-SA 4.0에 따라 라이센스가 부여됩니다.