裁剪视频帧显示在滚动条,按住配音滚动条自动滚动,视频同时播放

先放预览图


裁剪视频帧显示在滚动条,按住配音滚动条自动滚动,视频同时播放_第1张图片
demo.png

本文主要实现的是按住按钮,开始录音同时滚动条滚动,视频同时播放,当松开按钮滚动条停止录音停止,将刚才录音的区域显示在滚动条上
首先整理思路,长按事件,因为setOnLongClickListener不能得到松开的时间 所以重写系统OnTouchListener是最好的办法。
先放代码


import android.app.Activity;
import android.os.Handler;
import android.os.Message;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;

/**
 * 提供View的单击和长按事件的判断回调
 */
public class TouchUtils {

    private static String TAG = "TouchUtils";
    private static long DELAYED_TIME = 20;

    public static void setTouchEventListener(final View view, final OnTouchEventListener onTouchEventListener) {
        setTouchEventListener(view, DELAYED_TIME, onTouchEventListener);
    }

    /**
     * 注册touch事件
     */
    public static void setTouchEventListener(final View view, final long delayedTime, final OnTouchEventListener onTouchEventListener) {

        final Handler uiHandler = new Handler() {
            @Override
            public void handleMessage(Message msg) {
                if (view.getContext() instanceof Activity) {
                    Activity activity = (Activity) view.getContext();
                    if (activity.isFinishing()) {
                        removeCallbacksAndMessages(null);
                    }else {
                        onTouchEventListener.onLongTouch(view);
                        sendEmptyMessageDelayed(1, delayedTime);
                    }
                } else {
                   throw  new RuntimeException("必须绑定Activity");
                }

                Log.i(TAG, "onLongTouch: ");
            }
        };

        view.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {

                uiHandler.removeCallbacksAndMessages(null);
                String touchTag = (String) view.getTag(Integer.MAX_VALUE - 1);
                if (touchTag == null) {
                    Log.i(TAG, "onSingleTouch: ");
                    onTouchEventListener.onSingleTouch(view);
                }
                onTouchEventListener.onTouchEnd(view);
                Log.i(TAG, "onTouchEnd: ");

            }
        });

        /* 下面的不用管,值关注 OnclickListener 就可以了 */
        view.setOnTouchListener(new View.OnTouchListener() {
            @Override
            public boolean onTouch(View v, MotionEvent event) {
                if (event.getAction() == MotionEvent.ACTION_DOWN) {
                    view.setTag(Integer.MAX_VALUE - 1, null);
                    onTouchEventListener.onTouchStart(view);
                    Log.i(TAG, "onTouchStart: ");
                } else if (event.getAction() == MotionEvent.ACTION_UP) {
                    uiHandler.removeCallbacksAndMessages(null);
                } else if (event.getAction() == MotionEvent.ACTION_CANCEL) {
                    uiHandler.removeCallbacksAndMessages(null);
                }

                return false;
            }
        });

        view.setOnLongClickListener(new View.OnLongClickListener() {
            @Override
            public boolean onLongClick(View v) {
                view.setTag(Integer.MAX_VALUE - 1, "onLongTouch");
                uiHandler.sendEmptyMessage(1);

                return false;
            }
        });
    }

    public interface OnTouchEventListener {

        void onTouchStart(View view);//触发按动

        void onSingleTouch(View view);//单次点击

        void onLongTouch(View view);//长按

        void onTouchEnd(View view);//触摸事件结束

    }
}
 这个类使用Handler实现了长按时间的监听,拿到长按的监听,接下来需要做的就是滚动条滚动

接下来要实现的就是滚动条和蓝色覆盖了
首先分析一下组成
1、整个滚动条分成了2个部分
第一部分是RecyclerView实现的视频每一帧的图
第二部分是蓝色覆盖条
我们把两部分加在一起封装成一个DubScrollView


import android.content.Context;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.util.AttributeSet;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.widget.Button;
import android.widget.FrameLayout;

import com.wanyueliang.avm.R;
import com.wanyueliang.avm.config.AppFilmQuickConfig;
import com.wanyueliang.avm.model.databean.DubBean;
import com.wanyueliang.avm.ui.create.editor.dub.adapter.DubAdapter;
import com.wanyueliang.avm.utils.click_util.onFastDisableClickListener;
import com.wanyueliang.avm.utils.log.AppLog;
import com.wanyueliang.avm.widget.slider.DoubleSliderView;
import com.wanyueliang.avm.widget.slider.scroll_slider.MaterialBean;
import com.wanyueliang.avm.widget.slider.scroll_slider.RangeDoubleSliderView;
import com.wanyueliang.avm.widget.textview.TimeTextView;

import org.greenrobot.greendao.annotation.NotNull;

import java.util.List;


public class DubScrollView extends FrameLayout {

    private final String TAG = getClass().getSimpleName();
    private Context mContext;
    /*View*/
    private TimeTextView mTvCurrentTime;
    private RecyclerView mRvMaterial;
    private Button mBtnDelete;
    private DubMarkView mDmvMarkView;


    /*data*/
    private float mViewWidth;//整体的宽度
    private float editViewHeight;//核心数据----编辑的View的高度,recyclerView和DoubleSliderView
    private float relativeDuration = 5f;//核心数据----每一份editViewHeight对应的时间
    private float mTotalTime;//核心数据----总时间
    private float precisionValue = 1000f;//核心数据----精度

    private RangeDoubleSliderView.Builder builder;
    private List mEditBeans;//阴影相关
    private DubBean mShowEditBean;

    private float sliderWidth;//滑块的宽度的高度,默认为DoubleSliderView高度的三分之一 editViewHeight/3f
    private float scrollCurrentTime;//当前中心轴对应的时间
    private int mOffsetX;//记录RecyclerView总偏移量

    private DubAdapter materialAdapter;
    private List mData;
    private View headView;
    private View footerView;

    private boolean mLimitValue;//是否有限制
    private boolean isRecording;//是否正在录制配音
    private float startTime;

    public DubScrollView(Context context) {
        this(context, null);
    }

    public DubScrollView(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public DubScrollView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init();
    }

    private void init() {
        mContext = getContext();
        LayoutInflater.from(mContext).inflate(R.layout.view_dub_scroll_view_layout, this, true);
        mTvCurrentTime = (TimeTextView) findViewById(R.id.tv_current_time);
        mRvMaterial = (RecyclerView) findViewById(R.id.rv_material);
        mRvMaterial = (RecyclerView) findViewById(R.id.rv_material);
        mTvCurrentTime = (TimeTextView) findViewById(R.id.tv_current_time);
        mBtnDelete = (Button) findViewById(R.id.btn_delete);
        mDmvMarkView = (DubMarkView) findViewById(R.id.dmv_mark_view);
        mDmvMarkView.hideSlider();
    }

    /*底层素材的数据*/
    public void setMaterialData(List materialBeans, float totalTime) {
        this.mData = materialBeans;
        this.mTotalTime = totalTime;
        if (materialAdapter != null) {
            materialAdapter.setData(mData);
            materialAdapter.notifyDataSetChanged();
        }
    }

    /*添加的素材区域数据*/
    public void setEditData(List editBeans) {
        this.mEditBeans = editBeans;
        if (mDmvMarkView != null) {
            mDmvMarkView.setEditBeans(mEditBeans);
            mDmvMarkView.notifyDataChange();
        }
        //第一次进入,先匹配
        //检查是否有匹配到的区间
        checkMatching();
    }

    /*更新添加的素材区域数据*/
    public void notifyDataSetChange() {
        if (mDmvMarkView != null) {
            mDmvMarkView.notifyDataChange();
        }
        //第一次进入,先匹配
        //检查是否有匹配到的区间
        checkMatching();
    }

    public boolean isRecording() {
        return isRecording;
    }

    public void setRecordingAndStartTime(boolean recording, float startTime) {
        isRecording = recording;
        this.startTime = startTime;
    }

    public void setBtnDeleteVisibility(int visibility) {
        mBtnDelete.setVisibility(visibility);
    }

    public void setBtnDeleteOnClickListener(onFastDisableClickListener onClickListener) {
        mBtnDelete.setOnClickListener(onClickListener);
    }

    /**
     * 界面生成,初始化数据
     */
    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        super.onSizeChanged(w, h, oldw, oldh);
        if (w > 0 && h > 0) {
            mViewWidth = w;
            editViewHeight = mRvMaterial.getMeasuredHeight();
            Log.i(TAG, "onSizeChanged: editViewHeight=" + editViewHeight);
            initData();
            initListener();
        }
    }


    private void initData() {
        mEditBeans = AppFilmQuickConfig.getFilmDub();

        materialAdapter = new DubAdapter(mContext, relativeDuration, editViewHeight);//设置底层的素材预览图

        materialAdapter.setData(this.mData);

        mDmvMarkView.setEditBeans(mEditBeans);

        mTvCurrentTime.setTime(0, (long) (balanceValue(mTotalTime) * precisionValue));

        mRvMaterial.setLayoutManager(new LinearLayoutManager(mContext, RecyclerView.HORIZONTAL, false));

        /*在RecyclerView添加前面和后面的占位View*/
        addPlaceholderView();

        mRvMaterial.setAdapter(materialAdapter);
        // 滑块的宽度等于滑块的高度/3f
        sliderWidth = editViewHeight / 3f;

        float totalCurrent = ((mViewWidth - sliderWidth * 2f) / editViewHeight) * relativeDuration;

        int sliderMinCurrent = (int) (1 * precisionValue);//为了精准度,设置的数值增大10倍
        int sliderTotalCurrent = (int) (totalCurrent * precisionValue + 0.5f);//为了精准度,设置的数值增大10倍

        builder = new RangeDoubleSliderView.Builder();
        builder.setMinCurrent(sliderMinCurrent)
                .setStartCurrent(sliderTotalCurrent / 2)//起点为中心轴
                .setDurationCurrent(sliderTotalCurrent / 2)//持续时间
                .setTotalCurrent(sliderTotalCurrent);

        mDmvMarkView.setBuilder(builder);//设置DoubleSliderView的参数

        mDmvMarkView.setLimitView(headView, footerView);//设置限制左右滑块的坐标相关的View

        //第一次进入,先匹配
        //检查是否有匹配到的区间
        checkMatching();
    }


    /**
     * 检查是否有匹配到的区间
     */
    private void checkMatching() {
        if (mEditBeans != null && mEditBeans.size() > 0) {
            int editSize = mEditBeans.size();
            float leftSeekTo;
            float rightSeekTo;
            for (int i = editSize - 1; i >= 0; i--) {
                DubBean dubBean = mEditBeans.get(i);
                float startTime = Float.valueOf(dubBean.getOffsetStartTime());
                float endTime = Float.valueOf(dubBean.getTimeLength()) + startTime;
                if (startTime <= scrollCurrentTime && endTime >= scrollCurrentTime) {
                    leftSeekTo = startTime / relativeDuration * editViewHeight + mViewWidth / 2 - mOffsetX - sliderWidth;
                    rightSeekTo = leftSeekTo + (endTime - startTime) / relativeDuration * editViewHeight + sliderWidth;
                    mDmvMarkView.setTran(leftSeekTo, rightSeekTo);
                    mShowEditBean = dubBean;
                    break;
                }
            }

            if (mOnScrollChangeListener != null) {
                mOnScrollChangeListener.onMatchEditSelected(mShowEditBean);
            }

        }
    }


    /**
     * 在RecyclerView添加前面和后面的占位View
     */
    private void addPlaceholderView() {
        headView = LayoutInflater.from(mContext).inflate(R.layout.item_placeholder_layout, mRvMaterial, false);
        setViewWidth(headView, mViewWidth / 2f);
        materialAdapter.addHeaderView(headView);

        footerView = LayoutInflater.from(mContext).inflate(R.layout.item_placeholder_layout, mRvMaterial, false);
        setViewWidth(footerView, mViewWidth / 2f);
        materialAdapter.addFooterView(footerView);
    }

    /*占位View的宽度为重宽度的一半*/
    private void setViewWidth(View view, float w) {

        MarginLayoutParams marginLayoutParams = (MarginLayoutParams) view.getLayoutParams();
        marginLayoutParams.width = (int) (w + 0.5f);
        view.setLayoutParams(marginLayoutParams);
    }

    /*设置监听*/
    private void initListener() {

        mRvMaterial.addOnScrollListener(new RecyclerView.OnScrollListener() {
            @Override
            public void onScrollStateChanged(RecyclerView recyclerView, int newState) {

                if (newState == RecyclerView.SCROLL_STATE_IDLE) {//停止滑动的时候,寻找对应的View
                    //检查是否有匹配到的区间
                    if (mOnUseScrollChangeListener != null) {
                        mOnUseScrollChangeListener.onStateIdle();
                    }
                    checkMatching();
                } else if (newState == RecyclerView.SCROLL_STATE_DRAGGING) {
                    if (mOnUseScrollChangeListener != null) {
                        mOnUseScrollChangeListener.onUserDragging();
                    }
                }

                if (mOnScrollChangeListener != null) {
                    mOnScrollChangeListener.onScrollStateChanged(recyclerView, newState);
                }

            }

            @Override
            public void onScrolled(RecyclerView recyclerView, int dx, int dy) {

                mOffsetX += dx;//RecyclerView整体的滑动偏移量


                //得到中心轴对应的时间
                scrollCurrentTime = balanceValue(mOffsetX / editViewHeight/* * relativeDuration*/ * precisionValue) / precisionValue;
                AppLog.i(TAG + "_AVM", "onScrolled:预览条的偏移量 mOffsetX=" + mOffsetX + "scrollCurrentTime == " + scrollCurrentTime + "startTime" + startTime);

                //设置中心轴对应的时间
                mTvCurrentTime.setTime((long) (scrollCurrentTime * precisionValue), (long) (balanceValue(mTotalTime) * precisionValue));

                //使ShadowDoubleSliderViewView跟随滚动
                mDmvMarkView.setCustomScrollX(scrollCurrentTime, mOffsetX, dx);
                mDmvMarkView.setRecordingAndStartTime(isRecording, startTime);

                if (mOnScrollChangeListener != null) {
                    mOnScrollChangeListener.onScrolled(recyclerView, dx, dy);
                    mOnScrollChangeListener.onCurrentTime(scrollCurrentTime);
                }
            }
        });

    }


    public void setLimit(boolean mLimitValue) {
        this.mLimitValue = mLimitValue;
    }

    public boolean getLimit() {
        return mLimitValue;
    }


    /*使中心轴滚动到对应的时间*/
    public void seekTo(float currentTime) {
        mRvMaterial.scrollBy((int) (-(scrollCurrentTime - currentTime) * editViewHeight / relativeDuration), 0);
    }

    public void smoothScrollToPosition(int position) {
        mRvMaterial.smoothScrollToPosition(position);
    }

    /*平衡误差*/
    private int balanceValue(float value) {
        if (value % 0.5f != 0) {
            value += 0.5f;
        } else {
            //Nothing
            Log.i(TAG, "balanceValue: 有等于0的value=" + value);
        }
        return (int) value;
    }

    public float getCurrentTime() {
        return scrollCurrentTime;
    }

    public float getTotalTime() {
        return mTotalTime;
    }

    public List getEditBeans() {
        return mEditBeans;
    }

    private OnScrollChangeListener mOnScrollChangeListener;

    public void setOnEditScrollChangeListener(OnScrollChangeListener onScrollChangeListener) {
        this.mOnScrollChangeListener = onScrollChangeListener;
    }


    private OnUseScrollChangeListener mOnUseScrollChangeListener;

    public void setOnUseScrollChangeListener(OnUseScrollChangeListener onUseScrollChangeListener) {
        this.mOnUseScrollChangeListener = onUseScrollChangeListener;
    }

    /**
     * 为了可能的需要,该回调结合了{@link RecyclerView.OnScrollListener} 和 {@link DoubleSliderView.OnSliderChangerListener}
     * 并且实现了接口的方法,但是没有做处理。外部有需要的可以选择复写特定的方法来处理
     */
    public abstract static class OnScrollChangeListener extends RecyclerView.OnScrollListener implements DoubleSliderView.OnSliderChangerListener {

        //匹配到对应的编辑区域
        public abstract void onMatchEditSelected(DubBean editBean);

        public abstract void onMatchEditChange(DubBean editBean);

        public void onCurrentTime(float currentTime) {

        }

        @Override
        public void onStartTouch(int touchThumb) {
        }

        @Override
        public void onLeftSliderChange(long leftCurrent, long rightCurrent, long totalCurrent) {
        }

        @Override
        public void onRightSliderChange(long leftCurrent, long rightCurrent, long totalCurrent) {
        }

        @Override
        public void onStopTouch(int touchThumb) {
        }

        protected abstract void onEndCrop(@NotNull DubBean showEditBean);
    }

    public interface OnUseScrollChangeListener {

        void onUserDragging();//人为滑动

        void onStateIdle();//停止拖动

    }
}

上面这个类更多的是处理逻辑其实可以不用去关注太多,重要的是DubMarkView


import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.Rect;
import android.util.AttributeSet;

import com.wanyueliang.avm.R;
import com.wanyueliang.avm.model.databean.DubBean;
import com.wanyueliang.avm.utils.log.AppLog;
import com.wanyueliang.avm.widget.slider.scroll_slider.RangeDoubleSliderView;

import java.util.List;

public class DubMarkView extends RangeDoubleSliderView {

    private final String TAG = getClass().getSimpleName();
    private Context mContext;

    /*data*/
    //参数
    private float precisionValue = 100f;//核心数据----精度
    private float mCurrentTime;//核心数据----精度
    //绘制数据
    private Paint mShadowPaint;//线条的画笔颜色
    private Paint mLinePaint;//线条的画笔颜色

    private float mLineSize;//线条的画笔宽度
    private int mShadowColor;//线条的画笔颜色
    protected float mViewWidth;//View的宽度
    protected float mViewHeight;//View的高度
    protected Rect rect = new Rect();
    private List dubBeans;
    private boolean isRecording;
    private float startTime;

    public DubMarkView(Context context) {
        this(context, null);
    }

    public DubMarkView(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public DubMarkView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        initialize();
    }

    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        super.onSizeChanged(w, h, oldw, oldh);
        if (w > 0 && h > 0) {
            mViewWidth = w;
            mViewHeight = h;
        }
    }

    private void initialize() {
        mContext = getContext();

        mShadowColor = getResources().getColor(R.color.colorPrimary);

        mShadowPaint = new Paint();
        mShadowPaint.setColor(mShadowColor);
        mShadowPaint.setAlpha((int) (255 * 0.4));
        mShadowPaint.setAntiAlias(true);
        mShadowPaint.setStyle(Paint.Style.FILL);

        mLineSize = 8;

        int lineColor = getResources().getColor(R.color.colorPrimary);

        mLinePaint = new Paint();
        mLinePaint.setColor(lineColor);
        mLinePaint.setStrokeWidth(mLineSize);
        mLinePaint.setAlpha((int) (255 * 0.4));
        mLinePaint.setAntiAlias(true);

    }

    public void setEditBeans(List dubBeans) {
        this.dubBeans = dubBeans;
        invalidate();
    }

    public boolean isRecording() {
        return isRecording;
    }

    public void setRecordingAndStartTime(boolean recording, float startTime) {
        isRecording = recording;
        this.startTime = startTime;
    }

    public void notifyDataChange() {
        invalidate();
    }

    private float mOffsetX;


    @Override
    public void setCustomScrollX(float currentTime, int offsetX, int dxValue) {
        super.setCustomScrollX(currentTime, offsetX, dxValue);
        mCurrentTime = currentTime;
        mOffsetX = offsetX;

        invalidate();
    }


    @Override
    public void onDraw(Canvas canvas) {
        //
        if (isRecording) {
            mShadowPaint.setColor(mShadowColor);

            mShadowPaint.setAlpha((int) (255 * 0.4));

            rect.left = (int) ((mViewWidth / 2f + startTime * mViewHeight) - mOffsetX);
            rect.right = (int) (mViewWidth / 2f + mCurrentTime * mViewHeight - mOffsetX) + 1;

            AppLog.i("DubMarkView_AVM" + "_AVM", "Recording !!!! === startTime == " + mCurrentTime + "mCurrentTime == " + mCurrentTime);

            rect.top = 0;
            rect.bottom = (int) mViewHeight;

            canvas.drawLine(rect.left + mLineSize / 2f, 0, rect.left + mLineSize / 2f, mViewHeight, mLinePaint);

            canvas.drawLine(rect.right - mLineSize / 2f, 0, rect.right - mLineSize / 2f, mViewHeight, mLinePaint);
            canvas.drawRect(rect, mShadowPaint);

            drawEditRect(canvas);

        } else {
            drawEditRect(canvas);
        }

        super.onDraw(canvas);
    }

    private void drawEditRect(Canvas canvas) {
        if (dubBeans != null) {

            for (int i = 0; i < dubBeans.size(); i++) {
                DubBean dubBean = dubBeans.get(i);

                mShadowPaint.setColor(mShadowColor);

                mShadowPaint.setAlpha((int) (255 * 0.4));

                rect.left = (int) (mViewWidth / 2f + (Float.valueOf(dubBean.getOffsetStartTime()) * mViewHeight) - mOffsetX);
                rect.right = (int) (mViewWidth / 2f + ((Float.valueOf(dubBean.getOffsetStartTime()) + Float.valueOf(dubBean.getTimeLength())) * mViewHeight) - mOffsetX) + 1;

                AppLog.i("DubMarkView_AVM" + "_AVM",
                        "drawEditRect === startTime == " + Float.valueOf(dubBean.getOffsetStartTime())
                                + "mCurrentTime == " + (Float.valueOf(dubBean.getOffsetStartTime()) + Float.valueOf(dubBean.getTimeLength()))
                );

                rect.top = 0;
                rect.bottom = (int) mViewHeight;

                canvas.drawLine(rect.left + mLineSize / 2f, 0, rect.left + mLineSize / 2f, mViewHeight, mLinePaint);

                canvas.drawLine(rect.right - mLineSize / 2f, 0, rect.right - mLineSize / 2f, mViewHeight, mLinePaint);
                canvas.drawRect(rect, mShadowPaint);
            }
        }
    }
}

核心代码就是onDraw方法在录制的时候通过外部传入的当前时间绘制两条线然后绘制Rect画出蓝色条形图
RecyclerView这里就不详细描述了我们要说的是
RangeDoubleSliderView这个类更多的是实现了一个两边可滑动条,其实我们这个功能中用不上可以继承View不用继承RangeDoubleSliderView也没关系

你可能感兴趣的:(裁剪视频帧显示在滚动条,按住配音滚动条自动滚动,视频同时播放)