[63→100] Android仿微信录制短视频

作者: 沉思的Panda | 来源:发表于2016-06-28 21:37 被阅读6795次

微信朋友圈录制小视频,效果图如下:


拍摄小视频.png

怎么使用,大家应该不陌生了。其中关键技术有两个:

  1. 录制视频技术;
  2. “按住拍”的动画效果;

在网上搜了几个demo,最终发现下面两个开源项目比较靠谱:

  1. RecordVideoDemo ← 重点推荐
  2. WeiXinCamera

RecordVideoDemo中实现了两种录制方法:
a. 采用系统类MediaRecorder。
b. 直接采集摄像头画面和声卡的声音,再保存为视频格式。

经过统计,6s的视频,方案a获取的视频非常清晰,大小为32M,方案比为200多k。考虑到小视频上传、加载速度的要求高于清晰度,所以果断选择了方案b。

WeiXinCamera里面实现“按住拍、线条逐步变窄为0”的动画效果,抽取封装一下也可以用。

经过试验,采用动画方案反应会慢几个几秒,体验不好,在VideoCapture里面用ProgressBar来模拟,效果很好

集成步骤

  1. RecordVideoDemo中的WXLikeVideoRecorderLib拷贝到项目目录
  2. settings.gradle 中添加:
 include ':WXLikeVideoRecorderLib'
  1. app项目的build.gradle中添加依赖:
dependencies{
  compile project(':WXLikeVideoRecorderLib')
}
  1. 添加 摄像头、音频、存储器 的读写权限
<uses-permission android:name="android.permission.CAMERA" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.RECORD_AUDIO" />
  1. 修改WXLikeVideoRecorder,增加设置最长录制时间的接口。
// 最长录制时间private long maxRecordTime = 15000;
    /**
     * 设置最长录制时间
     * @param maxRecordTime
     */
    public void setMaxRecordTime(long maxRecordTime) {
        this.maxRecordTime = maxRecordTime;
    }
  1. 封装RecordFragmentHolder。
package lib;
import android.animation.Animator;
import android.animation.ValueAnimator;
import android.content.Context;
import android.hardware.Camera;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.widget.Toast;
import sz.itguy.utils.FileUtil;
import sz.itguy.wxlikevideo.camera.CameraHelper;
import sz.itguy.wxlikevideo.recorder.WXLikeVideoRecorder;
import sz.itguy.wxlikevideo.views.CameraPreviewView;
import sz.itguy.wxlikevideo.views.CircleBackgroundTextView;
/**
 * Created by shitianci on 16/6/28.
 */
public class RecordFragmentHolder {
    private static final String TAG = RecordFragmentHolder.class.getSimpleName();
    private final Context mContext;
    private final OnRecordListener mListener;
    private  Camera mCamera;
    private WXLikeVideoRecorder mRecorder;
    private boolean isCancelRecord = false;
    private ValueAnimator animation;
    // 输出宽度
    private int outputWidth = 320;
    // 输出高度
    private int outputHeight = 240;

    public interface OnRecordListener{
        void onEnd(String videoPath);
    }
    public RecordFragmentHolder(Context context, OnRecordListener listener) {
        mContext = context;
        mListener = listener;
    }
    /**
     * 初始化空间
     * @param preview 摄像头预览界面
     * @param btnRecord 录制按钮
     * @param animationLine 控制线
     * @param duration 时长
     * @return
     */
    public boolean init(CameraPreviewView preview, CircleBackgroundTextView btnRecord, final View animationLine, final long duration) {
        // Create an instance of Camera
        int cameraId = CameraHelper.getDefaultCameraID();
        mCamera = CameraHelper.getCameraInstance(cameraId);
        if (null == mCamera) {
            Toast.makeText(mContext, "打开相机失败!", Toast.LENGTH_SHORT).show();
            return false;
        }
        // 初始化录像机
        mRecorder = new WXLikeVideoRecorder(mContext, FileUtil.MEDIA_FILE_DIR);
        mRecorder.setOutputSize(outputWidth, outputHeight);
        preview.setCamera(mCamera, cameraId);
        mRecorder.setCameraPreviewView(preview);
        btnRecord.setOnTouchListener(new CircleBackgroundTextView.OnTouchListener() {
            @Override
            public void onDownListener(MotionEvent event) {
            }
            @Override
            public void onLongListener(final MotionEvent event) {
                Log.d(TAG, "onLongListener");
                isCancelRecord = false;
                startRecord();
                animation = AnimationUtil.startAnimation(animationLine, duration, new Animator.AnimatorListener() {
                    @Override
                    public void onAnimationStart(Animator animator) {
                    }
                    @Override
                    public void onAnimationEnd(Animator animator) {
                        stopRecord();
                    }
                    @Override
                    public void onAnimationCancel(Animator animator) {
                    }
                    @Override
                    public void onAnimationRepeat(Animator animator) {
                    }
                });
            }
            @Override
            public void onUpListener(MotionEvent event) {
                animation.cancel();
                stopRecord();
            }
        });
        return true;
    }

    /**
     * 设置输出的宽高
     * @param outputWidth
     * @param outputHeight
     */
    public void setOutputWidthAndHeight(int outputWidth, int outputHeight) {
        this.outputWidth = outputWidth;
        this.outputHeight = outputHeight;
    }

    public void onPause() {
        if (mRecorder != null) {
            boolean recording = mRecorder.isRecording();
            // 页面不可见就要停止录制
            mRecorder.stopRecording();
            // 录制时退出,直接舍弃视频
            if (recording) {
                FileUtil.deleteFile(mRecorder.getFilePath());
            }
        }
        releaseCamera();              // release the camera immediately on pause event
    }

    private void releaseCamera() {
        if (mCamera != null) {
            mCamera.setPreviewCallback(null);
            // 释放前先停止预览
            mCamera.stopPreview();
            mCamera.release();        // release the camera for other applications
            mCamera = null;
        }
    }

    /**
     * 开始录制
     */
    public void startRecord() {
        if (mRecorder.isRecording()) {
            Log.d(TAG, "startRecord");
            Toast.makeText(mContext, "正在录制中…", Toast.LENGTH_SHORT).show();
            return;
        }

        // initialize video camera
        if (prepareVideoRecorder()) {
            // 录制视频
            if (!mRecorder.startRecording())
                Toast.makeText(mContext, "录制失败…", Toast.LENGTH_SHORT).show();
        }
    }

    /**
     * 准备视频录制器
     *
     * @return
     */
    private boolean prepareVideoRecorder() {
        if (!FileUtil.isSDCardMounted()) {
            Toast.makeText(mContext, "SD卡不可用!", Toast.LENGTH_SHORT).show();
            return false;
        }
        return true;
    }

    /**
     * 停止录制
     */
    public void stopRecord() {
        mRecorder.stopRecording();
        String videoPath = mRecorder.getFilePath();
        mListener.onEnd(videoPath);
        // 没有录制视频
        if (null == videoPath) {
            return;
        }
        // 若取消录制,则删除文件,否则通知宿主页面发送视频
        if (isCancelRecord) {
            FileUtil.deleteFile(videoPath);
        } else {
            // 告诉宿主页面录制视频的路径
//            mContext.startActivity(new Intent(mContext, PlayVideoActiviy.class).putExtra(PlayVideoActiviy.KEY_FILE_PATH, videoPath));
        }
    }
}
  1. 在Fragment引用就可以了
package com.hbbohan.growmemory.view;
import android.Manifest;
import android.os.Bundle;
import android.view.View;
import com.hbbohan.growmemory.B;
import com.hbbohan.growmemory.R;
import java.io.File;
import butterfork.Bind;
import lib.RecordFragmentHolder;
import panda.android.lib.base.ui.fragment.BaseFragment;
import panda.android.lib.base.util.DevUtil;
import panda.android.lib.base.util.IntentUtil;
import sz.itguy.wxlikevideo.views.CameraPreviewView;
import sz.itguy.wxlikevideo.views.CircleBackgroundTextView;
/**
 * Created by shitianci on 16/6/28.
 */
public class RecordVideoFragment extends BaseFragment {
    @Bind(B.id.view_camera_preview)
    CameraPreviewView mViewCameraPreview;
    @Bind(B.id.btn_record)
    CircleBackgroundTextView mBtnRecord;
    @Bind(B.id.view_animation_line)
    View mViewAnimationLine;
    private RecordFragmentHolder mRecordFragmentHolder;
    @Override
    public String[] getPermissions() {
        return new String[]{
                Manifest.permission.CAMERA,
                Manifest.permission.WRITE_EXTERNAL_STORAGE,
                Manifest.permission.READ_EXTERNAL_STORAGE,
                Manifest.permission.RECORD_AUDIO
        };
    }
    @Override
    public void onActivityCreated(Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
        mRecordFragmentHolder = new RecordFragmentHolder(getActivity(), new RecordFragmentHolder.OnRecordListener() {
            @Override
            public void onEnd(String videoPath) {
                DevUtil.showInfo(getActivity(), "视频存放在:" + videoPath);
                IntentUtil.openFile(getActivity(), new File(videoPath));
            }
        });
        if (!mRecordFragmentHolder.init(mViewCameraPreview, mBtnRecord, mViewAnimationLine, 15000)){
            getActivity().finish();
        }
    }
    @Override
    public void onPause() {
        super.onPause();
        mRecordFragmentHolder.onPause();
        getActivity().finish();
    }
    @Override
    public int getLayoutId() {
        return R.layout.fragment_record_video;
    }
}
  1. 添加动画的引用库
package lib;
import android.animation.Animator;
import android.animation.ObjectAnimator;
import android.animation.ValueAnimator;
import android.util.Log;
import android.view.View;
import android.view.ViewGroup;
/**
 * Created by shitianci on 16/6/28.
 */
public class AnimationUtil {
    private static final String TAG = AnimationUtil.class.getSimpleName();
    /**
     * 动画效果:开始的宽度为父容器的宽度,逐步向中间缩减为0。
     * 使用场景:微信录制小视频
     *
     */
    public static ValueAnimator startAnimation(final View view, final long duration, final Animator.AnimatorListener animatorListener) {
        ValueAnimator va = ObjectAnimator.ofInt(view.getWidth(), 0);
        va.setDuration(duration);
        va.addListener(animatorListener);
        va.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {
                int value = (int) animation.getAnimatedValue();
                ViewGroup.LayoutParams params = view.getLayoutParams();
                params.width = value;
                view.setLayoutParams(params);
                view.requestLayout();
            }
        });
        //结束时恢复宽高
        final int width = view.getWidth();
        final int height = view.getHeight();
        va.addListener(new Animator.AnimatorListener() {
            @Override
            public void onAnimationStart(Animator animator) {
                Log.d(TAG, "onAnimationStart");
            }
            @Override
            public void onAnimationEnd(Animator animator) {
                Log.d(TAG, "onAnimationEnd");
                setViewLayoutParams(view, width, height);
            }
            @Override
            public void onAnimationCancel(Animator animator) {
                Log.d(TAG, "onAnimationCancel");
            }
            @Override
            public void onAnimationRepeat(Animator animator) {
                Log.d(TAG, "onAnimationRepeat");
            }
        });
        va.start();
        return va;
    }

    /**
     * 设置view的宽高
     * @param view
     * @param width
     * @param height
     */
    public static void setViewLayoutParams(View view, int width, int height) {
        ViewGroup.LayoutParams params = view.getLayoutParams();
        params.width = width;
        params.height = height;
        view.setLayoutParams(params);
        view.requestLayout();
    }
}

备注:如果采用23以上的sdk编译,在6.0设备上会碰到权限问题,具体解决方案,参考Android M上的权限获取问题

Panda
2016-06-28

相关文章

网友评论

  • 539f6bd0c3ba:楼主有没有遇到这个问题啊
    library "/system/lib/libdl.so" ("/system/lib/libdl.so") needed or dlopened by "/system/lib/libnativeloader.so" is not accessible for the namespace: [name="classloader-namespace", ld_library_paths="", default_library_paths="/data/app/com.yus.ycamera-1/lib/arm:/data/app/com.yus.ycamera-1/base.apk!/lib/armeabi-v7a:/data/app/com.yus.ycamera-1/split_lib_dependencies_apk.apk!/lib/armeabi-v7a:/data/app/com.yus.ycamera-1/split_lib_slice_0_apk.apk!/lib/armeabi-v7a:/data/app/com.yus.ycamera-1/split_lib_slice_1_apk.apk!/lib/armeabi-v7a:/data/app/com.yus.ycamera-1/split_lib_slice_2_apk.apk!/lib/armeabi-v7a:/data/app/com.yus.ycamera-1/split_lib_slice_3_apk.apk!/lib/armeabi-v7a:/data/app/com.yus.ycamera-1/split_lib_slice_4_apk.apk!/lib/armeabi-v7a:/data/app/com.yus.ycamera-1/split_lib_slice_5_apk.apk!/lib/armeabi-v7a:/data/app/com.yus.ycamera-1/split_lib_slice_6_apk.apk!/lib/armeabi-v7a:/data/app/com.yus.ycamera-1/split_lib_slice_7_apk.apk!/lib/armeabi-v7a:/data/app/com.yus.ycamera-1/split_lib_slice_8_apk.apk!/lib/armeabi
    junerver:原因是这个项目里用的库不支持23及以上的API版本,如果要用楼主提供的这个方案,只有将项目targetSdkVersion调整为22
  • Peterius的致幻脸:问个问题哈,像朋友圈发视屏类似功能,录制的原始视频是不是都要客户端进行转码和压缩?
  • 311120c733f1:大侠能把你的ScalableVideoView这个类给我一份么1784353688@qq.com。:blush: 我在代码里没找到
    311120c733f1:尴尬了没看gradle,我以为是你自己写的控件:blush:
  • 64e1a66335e1:切换为前置摄像头时,预览没问题,但是录制的影像是倒立的,有没有遇到过这个问题。
    64e1a66335e1:已解决,在设置fillter的时候 判断是否前置摄像头,前置设置为"clock_flip",后置为"clock"
  • Michael_Meng:1、视频预览界面的视频很清晰,但是录制出来之后非常模糊,这个在哪里改呢?
    2、如何将录制界面修改为全屏录制?
    2012lc:修改比特率和输出宽高
  • 老去的九零后青年:怎么修改清晰度啊
  • huangyirui:请问有仿微博那样断点录制视频的demo么
  • oO小花Oo:时间超过1分钟就会抛出异常 :joy:
  • c4f10497f2d7:你好有改进的源码参考一下吗?
    c4f10497f2d7:@沉思的Panda 921712319@qq.com
    c4f10497f2d7:@沉思的Panda 新手,可不可以提供一个你的demo呀,学习一下。
    沉思的Panda:@c4f10497f2d7 改动的代码都贴出来了
  • a3865fdefe3d:https://github.com/mabeijianxi/small-video-record,增强压缩与封装,欢迎交流
    旧时光的格子少年:@剑西 压缩之后,视频缺少。设置上限10s,可能保留下来的就只有3,4秒
  • 705311ff2337:在魅族上完美运行..但是在小米手机和华为手机上出现这个异常
    09-06 22:51:10.004 16329-16329/com.tradeaider.qcapp E/InputEventReceiver: Exception dispatching input event.
    09-06 22:51:10.004 16329-16329/com.tradeaider.qcapp E/MessageQueue-JNI: Exception in MessageQueue callback: handleReceiveCallback
    09-06 22:51:10.005 16329-16329/com.tradeaider.qcapp E/MessageQueue-JNI: java.lang.UnsatisfiedLinkError: org.bytedeco.javacpp.avutil
    at java.lang.Class.classForName(Native Method)
    at java.lang.Class.forName(Class.java:324)
    at org.bytedeco.javacpp.Loader.load(Loader.java:390)
    at org.bytedeco.javacpp.Loader.load(Loader.java:358)
    at org.bytedeco.javacpp.avcodec$AVPacket.<clinit>(avcodec.java:1407)
    at org.bytedeco.javacv.FFmpegFrameRecorder.<init>(FFmpegFrameRecorder.java:251)
    沉思的Panda:@wuf 我用小米4、小米pro测过,是正常的
  • haha__Kong:不明觉厉
  • kizi:new CircleBackgroundTextView.OnTouchListener() 里面重写的方法只有一个onTouch。。。你上面好多·。。。。。
    沉思的Panda:@kizi 在项目里,暂时还没有剥离,你看 https://github.com/szitguy/RecordVideoDemo 也一样,我只是在他上面增加了一些东西
    kizi:@沉思的Panda 方便将给个Demo吗
    沉思的Panda:@kizi 添加了一个动画
  • e26900d7381e:我用了作者推荐的RecordVideoDemo,发现有的手机(三星a8000,魅族4)录制的时候调用recorder.start后就报错了,不能录制成功。下面是错误信息,请大神指点:org.bytedeco.javacv.FrameRecorder$Exception: avio_open error() error -2: Could not open '/storage/emulated/0/cn.itguy.recordvideodemo/Media/VID_1471331018715.mp4'
    08-16 15:03:38.719 819-819/? W/System.err: at org.bytedeco.javacv.FFmpegFrameRecorder.startUnsafe(FFmpegFrameRecorder.java:741)
    08-16 15:03:38.719 819-819/? W/System.err: at org.bytedeco.javacv.FFmpegFrameRecorder.start(FFmpegFrameRecorder.java:390)
    08-16 15:03:38.719 819-819/? W/System.err: at sz.itguy.wxlikevideo.recorder.WXLikeVideoRecorder.startRecording(WXLikeVideoRecorder.java:250)
    08-16 15:03:38.719 819-819/? W/System.err: at sz.itguy.recordvideodemo.NewRecordVideoActivity.startRecord(NewRecordVideoActivity.java:108)
    08-16 15:03:38.719 819-819/? W/System.err: at sz.itguy.recordvideodemo.NewRecordVideoActivity.onTouch(NewRecordVideoActivity.java:152)
    08-16 15:03:38.719 819-819/? W/System.err: at android.view.View.dispatchTouchEvent(View.java:9181)
    08-16 15:03:38.719 819-819/? W/System.err: at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2691)
    08-16 15:03:38.719 819-819/? W/System.err: at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:2308)
    08-16 15:03:38.719 819-819/? W/System.err: at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2691)
    沉思的Panda:@e26900d7381e 那就不清楚了,从日志来看,就是文件无法打开
    e26900d7381e:@沉思的Panda 读写权限都加了,目录是在录制视频的时候创建的,其他手机都没有问题,就这个手机(root过),还是没找出原因
    沉思的Panda:@e26900d7381e 不能打开“/storage/emulated/0/cn.itguy.recordvideodemo/Media/VID_1471331018715.mp4'”这个文件,问题原因可能有两个:
    1. 目录是否存在
    2. 是否有读写的权限
  • 魏魏魏魏:正打算学习,谢谢,要是把微信定位地图指点一下使用的是地图定位的什么api就更好了
  • 捡淑:怒赞一个
    HannyYeung:怎么哪里都有你啊,这个头像也是醉了
  • Yuven:赞一个~

本文标题:[63→100] Android仿微信录制短视频

本文链接:https://www.haomeiwen.com/subject/keitjttx.html