Android开发经验分享- 用相机录制视频
xsobi 2024-12-25 16:13 1 浏览
现在的手机一般都会提供相机功能,有些相机大镜头已经支持1000万以上的像素,有些还支持光学变焦,这些手机已经变成了专业数码相机,为了充分利用手机上的相机功能,Android应用可以控制拍照和录制视频。
首先在项目中添加以下权限
<!--摄像头相关-->
<uses-permission android:name="android.permission.CAMERA" />
<uses-feature android:name="android.hardware.camera" />
<uses-feature android:name="android.hardware.camera.autofocus" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<!--音视频相关-->
<uses-permission android:name="android.permission.RECORD_AUDIO"/>
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
拍照
一、通过系统Intent进行拍照
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
创建Uri方法一:
//添加参数
ContentValues values = new ContentValues();
//设置文件名称
values.put(MediaStore.Images.Media.DISPLAY_NAME, String.valueOf(System.currentTimeMillis())+".jpg");
//设置文件类型
values.put(MediaStore.Images.Media.MIME_TYPE, "image/png");
//添加存储的位置的uri
uri = getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);
创建Uri方法二:
File file = new File(Environment.getExternalStorageDirectory().getAbsolutePath()+"/"+System.currentTimeMillis()+".jpg");
uri = Uri.fromFile(file);
//添加图像方向
intent.putExtra(MediaStore.Images.Media.ORIENTATION, 0);
//设置图像输出位置
intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
startActivityForResult(intent,1);
获取返回的图像路径
public static String getPathFromUri(Context mContext,Uri contentUri){
String[] proj = { MediaStore.Images.Media.DATA };
CursorLoader loader = new CursorLoader(mContext, contentUri, proj, null, null, null);
Cursor cursor = loader.loadInBackground();
int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
cursor.moveToFirst();
return cursor.getString(column_index);
}
通过路径加载Bitmap
Bitmap bitmap = BitmapFactory.decodeFile(getPathFromUri(this, uri));
二、利用Camrea拍照
1、调用Camera的open()打开相机
2、调用Camera的getParameters()获取拍照参数。该方法返回一个Camera.Paremeters对象
3、调用Camera.Parameters对象方法设置拍照的参数
4、调用Camera.startPreview()方法开始预览取景,在预览取景之前需要调用Camera的setPreviewDisplay(SurfaceHolder holder)方法设置使用哪个SurfaceView来显示取景图片。
5、调用Camera的takePicture()方法进行拍照
6、结束程序时,调用Camera的stopPreview()结束取景预览,并调用release()方法释放资源
public class CameraActivity extends Activity {
private SurfaceView mSurfaceView;
private View mTake,mCancle;
private SurfaceHolder mSurfaceHolder;
private boolean isPreview = false;
private Camera mCamera;
private Camera.AutoFocusCallback autoFocusCallback = new Camera.AutoFocusCallback() {
@Override
public void onAutoFocus(boolean success, Camera camera) {
if(success){
// 第一个监听器;当用户按下快门时激发该监听器
// 第二个监听器;当相机获取原始照片时激发该监听器
// 第三个监听器;当相机获取JPG照片时激发该监听器
mCamera.takePicture(new Camera.ShutterCallback() {
@Override
public void onShutter() {
//快门按下时
}
}, new Camera.PictureCallback() {
@Override
public void onPictureTaken(byte[] data, Camera camera) {
//此处代码可以决定是否需要保存原始照片信息
}
}, new Camera.PictureCallback() {
@Override
public void onPictureTaken(byte[] data, Camera camera) {
Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length);
//创建矩阵 旋转图像
Matrix matrix = new Matrix();
matrix.setRotate(90f,bitmap.getWidth()/2,bitmap.getHeight()/2);
Bitmap bit =Bitmap.createBitmap(bitmap,0,0,bitmap.getWidth(),bitmap.getHeight(),matrix,true);
//销毁原图
bitmap.recycle();
mImage.setImageBitmap(bit);
File file = new File(Environment.getExternalStorageDirectory(),System.currentTimeMillis()+".jpg");
FileOutputStream fos = null;
try {
fos = new FileOutputStream(file);
bit.compress(Bitmap.CompressFormat.JPEG,100,fos);
try {
fos.close();
} catch (IOException e) {
e.printStackTrace();
}
} catch (FileNotFoundException e) {
e.printStackTrace();
}
mCamera.stopPreview();;
mCamera.startPreview();
isPreview=true;
}
});
}
}
};
private ImageView mImage;
@Override
protected void onCre
ate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_camera);
initView();
process();
setAllClick();
}
private void setAllClick() {
mSurfaceHolder.addCallback(new SurfaceHolder.Callback() {
@Override
public void surfaceCreated(SurfaceHolder holder) {
initCamera();
}
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
if (mCamera != null) {
if (isPreview) {
mCamera.stopPreview();
mCamera.release();
mCamera = null;
}
}
}
});
mTake.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//拍照
if (mCamera != null) {
//添加自动聚焦
mCamera.autoFocus(autoFocusCallback);
// takePhoto();
}
}
});
}
private int mPreviewWidth=0;
private int mPreviewHeight=0;
/**
* 初始化相机
*/
private void initCamera() {
if(!isPreview){
//打开相机
// mCamera = Camera.open(Camera.getNumberOfCameras()-1);
mCamera = Camera.open(0);
}
if(mCamera!=null&&!isPreview){
//获取相关设置
Camera.Parameters parameters = mCamera.getParameters();
List<Camera.Size> previewsizes = parameters.getSupportedPreviewSizes();
Iterator<Camera.Size> iterator = previewsizes.iterator();
while(iterator.hasNext()){
Camera.Size next = iterator.next();
if(next.width>mPreviewWidth&&next.height>mPreviewHeight){
mPreviewWidth=next.width;
mPreviewHeight=next.height;
}
Log.i("TAG","camera->"+mPreviewWidth+":"+mPreviewHeight);
}
Log.i("TAG","surface->"+mSurfaceView.getWidth()+":"+mSurfaceView.getHeight());
//设置预览招聘时每秒显示多少帧的最小值和最大值
// parameters.setPreviewFpsRange(10, 20);
//设置图片格式
parameters.setPictureFormat(PixelFormat.JPEG);
//设置预览大小
// parameters.setPreviewSize(mPreviewWidth, mPreviewHeight);
//设置照片大小
parameters.setPictureSize(mPreviewWidth, mPreviewHeight);
//设置JPG照片的质量
parameters.set("jpeg-quality", 100);
//设置取景的SutfaceView
try {
//设置显示方向
mCamera.setDisplayOrientation(90);
mCamera.setParameters(parameters);
//设置配置
mCamera.setPreviewDisplay(mSurfaceHolder);
mCamera.startPreview();
} catch (IOException e) {
e.printStackTrace();
}
}
isPreview=true;
}
/**
* 业务过程
*/
private void process() {
mSurfaceHolder = mSurfaceView.getHolder();
//设置Surface不需要自己维护还冲区
mSurfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
}
/**
* 初始化视图
*/
private void initView() {
mSurfaceView =(SurfaceView) findViewById(R.id.camera_view);
mTake = findViewById(R.id.take_photo);
mCancle = findViewById(R.id.cancle);
mImage = (ImageView)findViewById(R.id.image);
}
}
录像
一、使用系统Intent
Intent intent = new Intent(MediaStore.ACTION_VIDEO_CAPTURE);
//限制时长 s
intent.putExtra(MediaStore.EXTRA_DURATION_LIMIT, 5);
//限制大小
intent.putExtra(MediaStore.EXTRA_SIZE_LIMIT, 1024*1024);
//设置质量
intent.putExtra(MediaStore.EXTRA_VIDEO_QUALITY, 1);
//设置输出位置
intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
startActivityForResult(intent, 1);
返回
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if(resultCode==RESULT_OK){
Uri uri = data.getData();
//视频地址
String videoPath = getPathFromUri(this,uri);
}
}
public static String getPathFromUri(Context mContext,Uri contentUri){
String[] proj = { MediaStore.Images.Media.DATA };
CursorLoader loader = new CursorLoader(mContext, contentUri, proj, null, null, null);
Cursor cursor = loader.loadInBackground();
int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
cursor.moveToFirst();
return cursor.getString(column_index);
}
二、使用MediaRecord
public class RecordeActivity extends Activity {
private SurfaceView mRecordView;
private View mStart, mStop;
private boolean isRecord = false;
private SurfaceHolder mSurfaceHolder;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_recorde);
initView();
process();
setAllClick();
}
private void setAllClick() {
mStart.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if(!isRecord){
startRecord();
}
}
});
mStop.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if(isRecord){
//停止录制
mRecorder.stop();
//释放资源
mRecorder.release();
mRecorder=null;
}
}
});
}
private File file;
private MediaRecorder mRecorder;
/**
* 开始录制
*/
private void startRecord() {
file = new File(Environment.getExternalStorageDirectory() + "/" + System.currentTimeMillis() + ".mp4");
try {
//创建MediaPay对象
mRecorder = new MediaRecorder();
mRecorder.reset();
//设置从麦克风采集声音
mRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
//设置从摄像头采集图像
mRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);
mRecorder.setOrientationHint(90);
//设置视频文件输出格式,这个设置必须在声音编码格式,图像编码格式之前设置
mRecorder.setOutputFormat(MediaRecorder.OutputFormat.MPEG_4);
//设置声音编码格式、图像编码格式
mRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.DEFAULT);
mRecorder.setVideoEncoder(MediaRecorder.VideoEncoder.MPEG_4_SP);
//设置视频大小
mRecorder.setVideoSize(1280, 960);
//设置帧率 每秒32帧
mRecorder.setVideoFrameRate(32);
//设置编码比特率
mRecorder.setVideoEncodingBitRate(5*1024*1024);
mRecorder.setOutputFile(file.getAbsolutePath());
//指定使用Surface来预览视频
mRecorder.setPreviewDisplay(mSurfaceHolder.getSurface());
//准备
mRecorder.prepare();
//开始录制
mRecorder.start();
isRecord = true;
} catch (IOException e) {
e.printStackTrace();
}
}
private void process() {
mSurfaceHolder = mRecordView.getHolder();
mSurfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
mSurfaceHolder.setKeepScreenOn(true);
}
private void initView() {
mRecordView = (SurfaceView) findViewById(R.id.video_recorde);
mStart = findViewById(R.id.start);
mStop = findViewById(R.id.stop);
}
}
更多内容关注微信公众号mjw-java或访问www.moliying.com
相关推荐
- 【互联网那些事】高效开发Android App的10个建议
-
假如要GooglePlay上做一个最失败的案例,那最好的秘诀就是界面奇慢无比、耗电、耗内存。接下来就会得到用户的消极评论,最后名声也就臭了。即使你的应用设计精良、创意无限也没用。 耗电或者内存...
- 手机APP开发方式有哪些? 手机app的开发模式有哪三种?
-
微信小程序开发定制_软件开发_APP开发_网站制作-优软软件开发...
- Android开发入门(一):Android系统简介
-
Android系统是Google公司在2008年推出的一款智能移动设备操作系统,通过不断地版本迭代,目前已经推出到Android11版本了。Android系统广泛应用在手机、平板、电视等各种电子设...
- 物联网app开发流程 物联网app开发工具
-
现在随着科技的发展,很多产品都想用一个手机app去显示他的参数数据或者通过手机app去控制它。但是很多人不知道他的流程。今天我就来说下物联网app开发流程。首先需要把物联网app开发流程分2个步骤,一...
- Android开发进阶 | 如何学习 Android Framework?
-
大部分有“如何学习Framework源码”这个疑问的,应该大都是应用层开发。应用层是被Framework层调用执行的,知道自己的代码是怎么被调用的,才能理解程序的本质,理解本质有助于解决遇到的...
- 快速实现APP混合开发(Hybrid App开发)攻略
-
前言:...
- 三个阶段带你了解一款app开发的完整流程
-
第一个阶段需求阶段:1.需求讨论--开发类型、开发平台、具体的产品功能需求、项目预计完成时间、预算2.需求评估--确认合作后评估具体的预算3.界面设计--设计部门进行产品界面设计,形成效果图...
- Android 开发中文引导-应用小部件
-
应用小部件是可以嵌入其它应用(例如主屏幕)并收到定期更新的微型应用视图。这些视图在用户界面中被叫做小部件,并可以用应用小部件提供者发布。可以容纳其他应用部件的应用组件叫做应用部件的宿主(1)。下面的截...
- 手机软件开发从零开始【Android第2篇Hello】
-
Hello,朋友们我们又见面了。上一篇我们讲到了《Android开发环境搭建【Android基础第1篇】》,错过的朋友可以点击文章末尾的“阅读原文”查看。另外需要下载JDK和ADT-bundle工具的...
- 「全栈工程师之梦的开始--安卓开发(二)」开发安卓app
-
在配置好jdk开发环境、安装好开发工具Androidstudio后,我们就可以开始开发安卓app了。首先,我们需要先了解下android的术语。...
- 二、Android界面开发 android 开发
-
学习目标了解Android常用布局了解Android常用控件...
- 如何开发一款APP既快捷也简便 开发一款app的步骤
-
具体较为简单的步骤可以选择用androidstudio开发app1、打开软件,在菜单中选择file-》newproject打开创建向导。2、配置项目,确定各个名称和存放项目的存放路径;Applic...
- 安卓开发中的“Android高手”,需要具备哪些技术?
-
前言成为一名安卓开发者很容易,但是要成为一名“Android高手”却不那么容易;...
- 移动开发(一):使用.NET MAUI开发第一个安卓APP
-
对于工作多年的C#程序员来说,近来想尝试开发一款安卓APP,考虑了很久最终选择使用.NETMAUI这个微软官方的框架来尝试体验开发安卓APP,毕竟是使用VisualStudio开发工具,使用起来也...
- 微软推出PowerApps:零基础开发Win10/iOS/安卓企业应用
-
IT之家讯微软今天面向企业宣布了全新的应用开发解决方案PowerApps,让Windows(包括Win10)、iOS以及安卓应用的开发和分发变得更加简单。PowerApps的用户界面与Office办...
- 一周热门
- 最近发表
- 标签列表
-
- grid 设置 (58)
- 移位运算 (48)
- not specified (45)
- patch补丁 (31)
- 导航栏 (58)
- context xml (46)
- scroll (43)
- element style (30)
- dedecms模版 (53)
- vs打不开 (29)
- nmap (30)
- c 视频教程下载 (33)
- paddleocr (28)
- listview排序 (33)
- firebug 使用 (31)
- transactionmanager (30)
- characterencodingfilter (33)
- getmonth (34)
- commandtimeout (30)
- hibernate教程 (31)
- label换行 (33)
- curlpost (31)
- android studio 3 0 (34)
- android开发视频 (30)
- android应用开发 (31)