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:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 xmlns:tools="http://schemas.android.com/tools"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
android:background="@android:color/background_dark"
 android:orientation="vertical">
 <SurfaceView
  android:id="@+id/surfaceview"
  android:layout_width="fill_parent"
  android:layout_height="0dp"
  android:layout_weight="1"
  />
 <ProgressBar
 android:id="@+id/progressBar"
 style="?android:attr/progressBarStyleHorizontal"
 android:layout_width="match_parent"
 android:layout_height="2dp"
 />
</LinearLayout>

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

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

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 xmlns:tools="http://schemas.android.com/tools"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
 android:background="@android:color/white"
 android:orientation="vertical">
<com.example.wechatvideorecorddemo.MovieRecorderView
 android:id="@+id/movieRecorderView"
 android:layout_width="match_parent"
 android:layout_height="0dp"
 android:layout_weight="1"
 android:layout_margin="3dp" />
 <Button
 android:id="@+id/shoot_button"
 android:layout_width="wrap_content"
 android:layout_height="wrap_content"
 android:layout_gravity="center" android:background="@drawable/bg_movie_add_shoot"
 android:text="按住拍"
 android:textColor="#20b6ff"/>
</LinearLayout>

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:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 xmlns:tools="http://schemas.android.com/tools"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
 android:background="@android:color/white"
 android:orientation="vertical">
 <TextView
 android:id="@+id/text"
 android:layout_width="wrap_content"
 android:layout_height="wrap_content"
 android:layout_gravity="center"
 android:text="@string/app_name" />
 <LinearLayout
 android:layout_width="match_parent"
 android:layout_height="wrap_content"
 android:orientation="horizontal"
 >
 <Button
  android:id="@+id/button1"
  android:layout_width="match_parent"
  android:layout_height="wrap_content"
  android:layout_weight="1"
  android:gravity="center"
  android:padding="5dp"
  android:text="播放"
  />
 <Button
  android:id="@+id/button2"
  android:layout_width="match_parent"
  android:layout_height="wrap_content"
  android:layout_weight="1"
  android:gravity="center"
  android:padding="5dp"
  android:text="暂停"
  />
 <Button
  android:id="@+id/button3"
  android:layout_width="match_parent"
  android:layout_height="wrap_content"
  android:layout_weight="1"
  android:gravity="center"
  android:padding="5dp"
  android:text="重播"
  />
 <Button
  android:id="@+id/button4"
  android:layout_width="match_parent"
  android:layout_height="wrap_content"
  android:layout_weight="1"
  android:gravity="center"
  android:padding="5dp"
  android:text="视频长度"
  />
 </LinearLayout>
 <VideoView
 android:id="@+id/videoView1"
 android:layout_width="wrap_content"
 android:layout_height="500dp" />
</LinearLayout>

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.添加权限:

<!-- 视频录制的权限star -->
<!-- 摄像头 -->
<uses-permission android:name="android.permission.CAMERA" />
<!-- 音频即声音 -->
<uses-permission android:name="android.permission.RECORD_AUDIO" />
<!-- sd卡写入权限 -->
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<!-- 硬件支持 -->
<uses-feature android:name="android.hardware.camera" />
<uses-feature android:name="android.hardware.camera.autofocus" />
<!-- 视频录制的权限end -->

功能界面截图:

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

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

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索android
仿微信
仿微信拍摄、android 仿微信朋友圈、android 仿微信小视频、android仿微信、android仿微信通讯录,以便于您获取更多的相关知识。

时间: 2024-08-31 03:12:53

Android仿微信拍摄短视频_Android的相关文章

Android仿微信语音聊天功能_Android

本文实例讲述了Android仿微信语音聊天功能代码.分享给大家供大家参考.具体如下: 项目效果如下: 具体代码如下: AudioManager.java package com.xuliugen.weichat; import java.io.File; import java.io.IOException; import java.util.UUID; import android.media.MediaRecorder; public class AudioManager { private

Android仿微信主界面设计_Android

先来一张效果图 一.ActionBar的设计 首先是main.xml,先定义这些菜单,界面稍后在调整 <menu xmlns:android="http://schemas.android.com/apk/res/android" tools:context=".MainActivity"> <item android:id="@+id/action_search" android:actionViewClass="a

Android仿微信群聊头像_Android

工作中需要实现仿钉钉群头像的一个功能,就是个人的头像拼到一起显示,看了一下市场上的APP好像微信的群聊头像是组合的,QQ的头像不是,别的好像也没有了. 给大家分享一下怎么实现的吧.首先我们先看一下效果图: 好了,下面说一下具体怎么实现的: 实现思路 1.首先获取Bitmap图片(本地.网络) 2.创建一个指定大小的缩略图 3.组合Bitmap图片 很简单,本地图片需要我们从本地读取,如果是网络图片我们也可以根据URL来获取bitmap进行组合 具体实现过程 1.布局文件: <LinearLayo

Android仿微信图片点击全屏效果_Android

废话不多说先看下效果 先是微信的 再是模仿的 先说下实现原理再一步步分析 这里总共有2个Activity一个就是主页一个就是显示我们图片效果的页面参数通过Intent传送素材内容均来自网络(感谢聪明的蘑菇) 图片都是Glide异步下的下的下的重要的事情说三次然后就是用动画做放大操作然后显示出来了并没有做下载原图的实现反正也是一样 下载下来Set上去而且动画都不需要更简便. OK我们来看分析下 obj目录下分别创建了2个对象一个用来使用来处理显示页面的图片尺寸信息以及位置信息还有一个是用来附带UR

Android仿微信朋友圈图片查看器_Android

再看文章之前,希望大家先打开自己的微信点到朋友圈中去,仔细观察是不是发现朋友圈里的有个"九宫格"的图片区域,点击图片又会跳到图片的详细查看页面,并且支持图片的滑动和缩放?这个功能是不是很常用呢?!那么我今天正好做了这个Demo,下面为大家讲解一下.首先按照惯例先看一下效果图吧,尤其不会录制gif动画(哎~没办法,模拟器不支持多点触控,刚好我的手机又没有Root,不能录屏,悲催啊,大家见谅,想要看真实效果的话,烦请移到文章最下方转载文章中进行源码下载,点击下载源码,运行后再看效果哈~~)

Android仿微信雷达辐射搜索好友(逻辑清晰实现简单)_Android

不知不觉这个春节也已经过完了,遗憾家里没网,没能及时给大家送上祝福,今天回到深圳,明天就要上班了,小伙伴们是不是和我一样呢?今天讲的是一个大家都见过的动画,雷达搜索好友嘛,原理也十分的简单,你看完我的分析,也会觉得很简单了,国际惯例,无图无真相,我们先看看效果图,对了,真 测试机送人了,所讲这段时间应该一直用模拟器显示吧! 仿微信雷达扫描,仿安卓微信.云播雷达扫描动画效果点击中间的黑色圆圈开始扫描动画,再次点击复位,需要这种效果的朋友可以自己下载看一下. 效果图如下所示: 这个界面相信大家都认识

Android仿微信发表说说实现拍照、多图上传功能_Android

本文实例为大家分享了Android仿微信发表说说.心情功能,供大家参考,具体内容如下 既能实现拍照,选图库,多图案上传的案例,目前好多App都有类似微信朋友圈的功能,能过发表说说等附带图片上传.下面的就是实现该功能的过程:大家还没有看过Android Retrofit 2.0框架上传图片解决方案这篇文章,在看今天的就很容易,接在本项目中用到了一个library:photopicker,封装了图片的选择功能,是否选相机,还有选中图片后可以查看图片的功能.   一. 首先:将photopicker到

Android 仿微信朋友圈点赞和评论弹出框功能_Android

贡献/下载源码:https://github.com/mmlovesyy/PopupWindowDemo 本文简单模仿微信朋友圈的点赞和评论弹出框,布局等细节请忽略,着重实现弹出框.发评论,及弹出位置的控制. 1. 微信弹出框 微信朋友圈的点赞和评论功能,有2个组成部分: 点击左下角的"更多"按钮,弹出对话框: 点击评论,弹出输入框,添加评论并在页面中实时显示:   微信朋友圈点赞和评论功能 2. 实际效果 本文将建一个 ListView,在其 Item 中简单模仿微信的布局,然后着重

Android仿微信二维码和条形码_Android

本文实例为大家分享了Android仿微信二维码和条形码的具体代码,供大家参考,具体内容如下 package your.QRCode.namespace; import java.io.File; import java.io.FileOutputStream; import java.util.HashMap; import java.util.Map; import com.google.zxing.BarcodeFormat; import com.google.zxing.EncodeHi