热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android仿微信拍摄短视频

本文主要对Android仿微信拍摄短视频的实现方法进行介绍,其功能设置为类似于微信,点击开始拍摄,设置最长拍摄时间。具有很好的参考价值,需要的朋友一起来看下吧

近期做项目需要添加上传短视频功能,功能设置为类似于微信,点击开始拍摄,设置最长拍摄时间,经过研究最终实现了这个功能,下面就和大家分享一下,希望对你有帮助。

1.视频录制自定义控件:

/**
 * 视频播放控件
 */
public class MovieRecorderView extends LinearLayout implements OnErrorListener {
 private SurfaceView mSurfaceView;
 private SurfaceHolder mSurfaceHolder;
 private ProgressBar mProgressBar;
 private MediaRecorder mMediaRecorder;
 private Camera mCamera;
 private Timer mTimer;// 计时器
 private OnRecordFinishListener mOnRecordFinishListener;// 录制完成回调接口
 private int mWidth;// 视频分辨率宽度
 private int mHeight;// 视频分辨率高度
 private boolean isOpenCamera;// 是否一开始就打开摄像头
 private int mRecordMaxTime;// 一次拍摄最长时间
 private int mTimeCount;// 时间计数
 private File mVecordFile = null;// 文件
 public MovieRecorderView(Context context) {
 this(context, null);
 }
 public MovieRecorderView(Context context, AttributeSet attrs) {
 this(context, attrs, 0);
 }
 @SuppressLint("NewApi")
 public MovieRecorderView(Context context, AttributeSet attrs, int defStyle) {
 super(context, attrs, defStyle);
 TypedArray a = context.obtainStyledAttributes(attrs,
 R.styleable.MovieRecorderView, defStyle, 0);
 mWidth = a.getInteger(R.styleable.MovieRecorderView_width, 320);// 默认320
 mHeight = a.getInteger(R.styleable.MovieRecorderView_height, 240);// 默认240
 isOpenCamera = a.getBoolean(
 R.styleable.MovieRecorderView_is_open_camera, true);// 默认打开
 mRecordMaxTime = a.getInteger(
 R.styleable.MovieRecorderView_record_max_time, 10);// 默认为10
 LayoutInflater.from(context)
 .inflate(R.layout.movie_recorder_view, this);
 mSurfaceView = (SurfaceView) findViewById(R.id.surfaceview);
 mProgressBar = (ProgressBar) findViewById(R.id.progressBar);
 mProgressBar.setMax(mRecordMaxTime);// 设置进度条最大量
 mSurfaceHolder = mSurfaceView.getHolder();
 mSurfaceHolder.addCallback(new CustomCallBack());
 mSurfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
 a.recycle();
 }
 /**
 * 
 */
 private class CustomCallBack implements Callback {
 @Override
 public void surfaceCreated(SurfaceHolder holder) {
 if (!isOpenCamera)
 return;
 try {
 initCamera();
 } catch (IOException e) {
 e.printStackTrace();
 }
 }
 @Override
 public void surfaceChanged(SurfaceHolder holder, int format, int width,
 int height) {
 }
 @Override
 public void surfaceDestroyed(SurfaceHolder holder) {
 if (!isOpenCamera)
 return;
 freeCameraResource();
 }
 }
 /**
 * 初始化摄像头
 */
 private void initCamera() throws IOException {
 if (mCamera != null) {
 freeCameraResource();
 }
 try {
 mCamera = Camera.open();
 } catch (Exception e) {
 e.printStackTrace();
 freeCameraResource();
 }
 if (mCamera == null)
 return;
 setCameraParams();
 mCamera.setDisplayOrientation(90);
 mCamera.setPreviewDisplay(mSurfaceHolder);
 mCamera.startPreview();
 mCamera.unlock();
 }
 /**
 * 设置摄像头为竖屏
 */
 private void setCameraParams() {
 if (mCamera != null) {
 Parameters params = mCamera.getParameters();
 params.set("orientation", "portrait");
 mCamera.setParameters(params);
 }
 }
 /**
 * 释放摄像头资源
 */
 private void freeCameraResource() {
 if (mCamera != null) {
 mCamera.setPreviewCallback(null);
 mCamera.stopPreview();
 mCamera.lock();
 mCamera.release();
 mCamera = null;
 }
 }
 private void createRecordDir() {
 //录制的视频保存文件夹
 File sampleDir = new File(Environment.getExternalStorageDirectory()
 + File.separator + "ysb/video/");//录制视频的保存地址
 if (!sampleDir.exists()) {
 sampleDir.mkdirs();
 }
 File vecordDir = sampleDir;
 // 创建文件
 try {
 mVecordFile = File.createTempFile("recording", ".mp4", vecordDir);// mp4格式的录制的视频文件
 } catch (IOException e) {
 e.printStackTrace();
 }
 }
 /**
 * 初始化
 * @throws IOException
 */
 @SuppressLint("NewApi")
 private void initRecord() throws IOException {
 mMediaRecorder = new MediaRecorder();
 mMediaRecorder.reset();
 if (mCamera != null)
 mMediaRecorder.setCamera(mCamera);
 mMediaRecorder.setOnErrorListener(this);
 mMediaRecorder.setPreviewDisplay(mSurfaceHolder.getSurface());
 mMediaRecorder.setVideoSource(VideoSource.CAMERA);// 视频源
 mMediaRecorder.setAudioSource(AudioSource.MIC);// 音频源
 mMediaRecorder.setOutputFormat(OutputFormat.MPEG_4);// 视频输出格式
 mMediaRecorder.setAudioEncoder(AudioEncoder.AMR_NB);// 音频格式
 mMediaRecorder.setVideoSize(mWidth, mHeight);// 设置分辨率:
 // mMediaRecorder.setVideoFrameRate(16);// 这个我把它去掉了,感觉没什么用
 mMediaRecorder.setVideoEncodingBitRate(1 * 1024 * 1024 * 100);// 设置帧频率,然后就清晰了
 mMediaRecorder.setOrientationHint(90);// 输出旋转90度,保持竖屏录制
 mMediaRecorder.setVideoEncoder(VideoEncoder.MPEG_4_SP);// 视频录制格式
 // mediaRecorder.setMaxDuration(Constant.MAXVEDIOTIME * 1000);
 mMediaRecorder.setOutputFile(mVecordFile.getAbsolutePath());
 mMediaRecorder.prepare();
 try {
 mMediaRecorder.start();
 } catch (IllegalStateException e) {
 e.printStackTrace();
 } catch (RuntimeException e) {
 e.printStackTrace();
 } catch (Exception e) {
 e.printStackTrace();
 }
 }
 /**
 * 开始录制视频
 * @param fileName
 * 视频储存位置
 * @param onRecordFinishListener
 * 达到指定时间之后回调接口
 */
 public void record(final OnRecordFinishListener onRecordFinishListener) {
 this.mOnRecordFinishListener= onRecordFinishListener;
 createRecordDir();
 try {
 if (!isOpenCamera)// 如果未打开摄像头,则打开
 initCamera();
 initRecord();
 mTimeCount = 0;// 时间计数器重新赋值
 mTimer = new Timer();
 mTimer.schedule(new TimerTask() {
 @Override
 public void run() {
  mTimeCount++;
  mProgressBar.setProgress(mTimeCount);// 设置进度条
  if (mTimeCount == mRecordMaxTime) {// 达到指定时间,停止拍摄
  stop();
  if (mOnRecordFinishListener != null)  mOnRecordFinishListener.onRecordFinish();
  }
 }
 }, 0, 1000);
 } catch (IOException e) {
 e.printStackTrace();
 }
 }
 /**
 * 停止拍摄
 */
 public void stop() {
 stopRecord();
 releaseRecord();
 freeCameraResource();
 }
 /**
 * 停止录制
 */
 public void stopRecord() {
 mProgressBar.setProgress(0);
 if (mTimer != null)
 mTimer.cancel();
 if (mMediaRecorder != null) {
 // 设置后不会崩
 mMediaRecorder.setOnErrorListener(null);
 mMediaRecorder.setPreviewDisplay(null);
 try {
 mMediaRecorder.stop();
 } catch (IllegalStateException e) {
 e.printStackTrace();
 } catch (RuntimeException e) {
 e.printStackTrace();
 } catch (Exception e) {
 e.printStackTrace();
 }
 }
 }
 /**
 * 释放资源
 */
 private void releaseRecord() {
 if (mMediaRecorder != null) {
 mMediaRecorder.setOnErrorListener(null);
 try {
 mMediaRecorder.release();
 } catch (IllegalStateException e) {
 e.printStackTrace();
 } catch (Exception e) {
 e.printStackTrace();
 }
 }
 mMediaRecorder = null;
 }
 public int getTimeCount() {
 return mTimeCount;
 }
 //返回录制的视频文件
 public File getmVecordFile() {
 return mVecordFile;
 }
 /**
 * 录制完成回调接口
 */
 public interface OnRecordFinishListener {
 public void onRecordFinish();
 }
 @Override
 public void onError(MediaRecorder mr, int what, int extra) {
 try {
 if (mr != null)
 mr.reset();
 } catch (IllegalStateException e) {
 e.printStackTrace();
 } catch (Exception e) {
 e.printStackTrace();
 }
 }
}

2.视频录制界面文件movie_recorder_view.xml:

<&#63;xml version="1.0" encoding="utf-8"&#63;>

 
 

做好这些准备工作,下面我们就可以开始设计我们的视频录制功能了。PS:以上代码取至网上,在此向大牛致敬。

3.拍摄主界面,拍摄界面有两部分组成,上面是视频拍摄控件显示,下面是用户点击拍摄按钮,配置文件:activity_main.xml。

<&#63;xml version="1.0" encoding="utf-8"&#63;>


 

4.有了主界面的视图,下面我们就开始书写我们的Activity文件MainActivity.java:

public class MainActivity extends Activity {
 private MovieRecorderView mRecorderView;//视频录制控件
 private Button mShootBtn;//视频开始录制按钮
 private boolean isFinish = true;
 private boolean success = false;//防止录制完成后出现多次跳转事件
 @Override
 protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.activity_main);
 mRecorderView = (MovieRecorderView) findViewById(R.id.movieRecorderView);
 mShootBtn = (Button) findViewById(R.id.shoot_button);
 //用户长按事件监听
 mShootBtn.setOnTouchListener(new OnTouchListener() {
 @Override
 public boolean onTouch(View v, MotionEvent event) {
 if (event.getAction() == MotionEvent.ACTION_DOWN) {//用户按下拍摄按钮
  mShootBtn.setBackgroundResource(R.drawable.bg_movie_add_shoot_select);
  mRecorderView.record(new OnRecordFinishListener() {
  @Override
  public void onRecordFinish() {
  if(!success&&mRecorderView.getTimeCount()<10){//判断用户按下时间是否大于10秒
  success = true;
  handler.sendEmptyMessage(1);
  }
  }
  });
 } else if (event.getAction() == MotionEvent.ACTION_UP) {//用户抬起拍摄按钮
  mShootBtn.setBackgroundResource(R.drawable.bg_movie_add_shoot);
  if (mRecorderView.getTimeCount() > 3){//判断用户按下时间是否大于3秒
  if(!success){
  success = true;
  handler.sendEmptyMessage(1);
  }
  } else {
  success = false;
  if (mRecorderView.getmVecordFile() != null)
 mRecorderView.getmVecordFile().delete();//删除录制的过短视频
  mRecorderView.stop();//停止录制
  Toast.makeText(MainActivity.this, "视频录制时间太短", Toast.LENGTH_SHORT).show();
  }
 }
 return true;
 }
 });
 }
 @Override
 public void onResume() {
 super.onResume();
 isFinish = true;
 if (mRecorderView.getmVecordFile() != null)
 mRecorderView.getmVecordFile().delete();//视频使用后删除
 }
 @Override
 public void onSaveInstanceState(Bundle outState) {
 super.onSaveInstanceState(outState);
 isFinish = false;
 success = false;
 mRecorderView.stop();//停止录制
 }
 @Override
 public void onPause() {
 super.onPause();
 }
 @Override
 public void onDestroy() {
 super.onDestroy();
 }
 private Handler handler = new Handler() {
 @Override
 public void handleMessage(Message msg) {
 if(success){
 finishActivity();
 }
 }
 };
 //视频录制结束后,跳转的函数
 private void finishActivity() {
 if (isFinish) {
 mRecorderView.stop();
 Intent intent = new Intent(this, SuccessActivity.class);
 Bundle bundle = new Bundle();
 bundle.putString("text", mRecorderView.getmVecordFile().toString());
 intent.putExtras(bundle);
 startActivity(intent);
 }
 success = false;
 }
 /**
 * 录制完成回调
 */
 public interface OnShootCompletionListener {
 public void OnShootSuccess(String path, int second);
 public void OnShootFailure();
 }
}

到这里我们仿微信的短视频拍摄就已经大功告成,那么下面我们检验一下,我们录制的效果如何,下面我以Android提供的视频播放控件(VideoView)为大家介绍一下如何播放录制的短视频。

5.播放视频的配置文件activity_success.xml:

<&#63;xml version="1.0" encoding="utf-8"&#63;>

 
 
 

6.视频播放的控制代码SuccessActivity.java:

public class SuccessActivity extends Activity implements OnClickListener{
 private TextView text;//视频保存的路径
 private Button button1;//播放开关
 private Button button2;//暂停开关
 private Button button3;//重新播放开关
 private Button button4;//视频大小开关
 private VideoView videoView1;//视频播放控件
 private String file;//视频路径
 @Override
 protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.activity_success);
 Bundle bundle = getIntent().getExtras();
 file = bundle.getString("text");//获得拍摄的短视频保存地址
 init();
 setValue();
 }
 //初始化
 private void init() {
 text = (TextView) findViewById(R.id.text);
 button1 = (Button) findViewById(R.id.button1);
 button2 = (Button) findViewById(R.id.button2);
 button3 = (Button) findViewById(R.id.button3);
 button4 = (Button) findViewById(R.id.button4);
 videoView1 = (VideoView) findViewById(R.id.videoView1);
 }
 //设置
 private void setValue() {
 text.setText(file);
 button1.setOnClickListener(this);
 button2.setOnClickListener(this);
 button3.setOnClickListener(this);
 button4.setOnClickListener(this);
 videoView1.setVideoPath(file);
 }
 @Override
 public void onClick(View v) {
 switch (v.getId()) {
 case R.id.button1:
 videoView1.start();
 break;
 case R.id.button2:
 videoView1.pause(); 
 break;
 case R.id.button3:
 videoView1.resume();
 videoView1.start();
 break;
 case R.id.button4:
 Toast.makeText(this, "视频长度:"+(videoView1.getDuration()/1024)+"M", Toast.LENGTH_SHORT).show();
 break;
 default:
 break;
 }
 } 
}

7.添加权限:












功能界面截图:

好了,到这里关于拍摄短视频的知识就和大家分享完毕,具体的实现很简单,相信大家看到这里已经已经学会了,当然如果你还有什么疑问,可以留言讨论。最后给大家分享一个demo的下载地址,方便大家下载学习,下载地址:http://xiazai.jb51.net/201612/yuanma/WeChatVideoRecordDemo_jb51.rar

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持!


推荐阅读
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • 本文介绍了如何使用PHP向系统日历中添加事件的方法,通过使用PHP技术可以实现自动添加事件的功能,从而实现全局通知系统和迅速记录工具的自动化。同时还提到了系统exchange自带的日历具有同步感的特点,以及使用web技术实现自动添加事件的优势。 ... [详细]
  • Monkey《大话移动——Android与iOS应用测试指南》的预购信息发布啦!
    Monkey《大话移动——Android与iOS应用测试指南》的预购信息已经发布,可以在京东和当当网进行预购。感谢几位大牛给出的书评,并呼吁大家的支持。明天京东的链接也将发布。 ... [详细]
  • 本文介绍了使用kotlin实现动画效果的方法,包括上下移动、放大缩小、旋转等功能。通过代码示例演示了如何使用ObjectAnimator和AnimatorSet来实现动画效果,并提供了实现抖动效果的代码。同时还介绍了如何使用translationY和translationX来实现上下和左右移动的效果。最后还提供了一个anim_small.xml文件的代码示例,可以用来实现放大缩小的效果。 ... [详细]
  • 基于layUI的图片上传前预览功能的2种实现方式
    本文介绍了基于layUI的图片上传前预览功能的两种实现方式:一种是使用blob+FileReader,另一种是使用layUI自带的参数。通过选择文件后点击文件名,在页面中间弹窗内预览图片。其中,layUI自带的参数实现了图片预览功能。该功能依赖于layUI的上传模块,并使用了blob和FileReader来读取本地文件并获取图像的base64编码。点击文件名时会执行See()函数。摘要长度为169字。 ... [详细]
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 本文讨论了如何优化解决hdu 1003 java题目的动态规划方法,通过分析加法规则和最大和的性质,提出了一种优化的思路。具体方法是,当从1加到n为负时,即sum(1,n)sum(n,s),可以继续加法计算。同时,还考虑了两种特殊情况:都是负数的情况和有0的情况。最后,通过使用Scanner类来获取输入数据。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • Mac OS 升级到11.2.2 Eclipse打不开了,报错Failed to create the Java Virtual Machine
    本文介绍了在Mac OS升级到11.2.2版本后,使用Eclipse打开时出现报错Failed to create the Java Virtual Machine的问题,并提供了解决方法。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • 《数据结构》学习笔记3——串匹配算法性能评估
    本文主要讨论串匹配算法的性能评估,包括模式匹配、字符种类数量、算法复杂度等内容。通过借助C++中的头文件和库,可以实现对串的匹配操作。其中蛮力算法的复杂度为O(m*n),通过随机取出长度为m的子串作为模式P,在文本T中进行匹配,统计平均复杂度。对于成功和失败的匹配分别进行测试,分析其平均复杂度。详情请参考相关学习资源。 ... [详细]
author-avatar
LINBO-D_915
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有