Android短视频滑动播放(一)



本文主要介绍采用RecyclerView配合PagerSnapHelper实现短视频滑动播放内容。

1. 主页内容构建

主页布局文件定义RecyclerView,为RecyclerView建立对应适配器。

<androidx.recyclerview.widget.RecyclerView
    android:id="@+id/rv_little_video"
    android:layout_width="match_parent"
    android:layout_height="match_parent" />

适配器条目中添加视频播放容器FrameLayout及封面ImageVIew.

<FrameLayout
    android:id="@+id/fl_content_item"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@android:color/black">

<ImageView
    android:id="@+id/iv_thumb_item"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:scaleType="centerCrop" />

</FrameLayout>

2.定义RecyclerView滑动管理

PagerSnapHelper 结合 LinearLayoutManager 实现滑动管理,实现监听任务。

PagerSnapHelper介绍

PagerSnapHelper can help achieve a similar behavior to
ViewPager. Set both RecyclerView and the items of the RecyclerView.Adapter to have android.view.ViewGroup.LayoutParams#MATCH_PARENT height and width and then attach PagerSnapHelper to the RecyclerView using #attachToRecyclerView(RecyclerView)}.

翻译:PagerSnapHelper可以帮助实现与以下类似的行为
ViewPager。 将RecyclerView和RecyclerView.Adapter的项目都设置为具有android.view.ViewGroup.LayoutParams#MATCH_PARENT的高度和宽度,然后使用#attachToRecyclerView(RecyclerView)}将PagerSnapHelper附加到RecyclerView。


自定义RecyclerView管理器

RecyclerView管理器为LinearLayoutManager 时,默认为纵向滑动,如果想采用横向滑动,就设置其滑动方向为RecyclerView.HORIZONTAL。同理,我们也可以这样采用setOrientation(RecyclerView.HORIZONTAL) 方法去改变滑动方向。

public class PagerLayoutManager extends LinearLayoutManager implements RecyclerView.OnChildAttachStateChangeListener {

    private OnPageChangedListener mOnPageChangedListener;

    private PagerSnapHelper mSnapHelper;

    /**
     * 移动方向标记
     */
    private int direction;

    public PagerLayoutManager(Context context) {
        super(context);
        mSnapHelper = new PagerSnapHelper();
    }

    /**
     * PagerSnapHelper绑定RecyclerView,同时为监听RecyclerView子布局附着,脱离,进行滑动页面内容控制
     *
     * @param view RecyclerView
     */
    @Override
    public void onAttachedToWindow(RecyclerView view) {
        super.onAttachedToWindow(view);
        mSnapHelper.attachToRecyclerView(view);
        view.addOnChildAttachStateChangeListener(this);
    }

    /**
     * 滑动状态改变监听,滑动完毕后进行播放控制
     *
     * @param state 滑动状态
     */
    @Override
    public void onScrollStateChanged(int state) {
    super.onScrollStateChanged(state);

        if (state == RecyclerView.SCROLL_STATE_IDLE) {
            View view = mSnapHelper.findSnapView(this);
            if (view == null) {
                return;
            }
            int position = getPosition(view);
            if (mOnPageChangedListener != null && getChildCount() == 1) {
                mOnPageChangedListener.onPageSelected(position, position == getItemCount() - 1);
            }
        }

    }

    @Override
    public int scrollHorizontallyBy(int dx, RecyclerView.Recycler recycler, RecyclerView.State state) {
        direction = dx;
        return super.scrollHorizontallyBy(dx, recycler, state);
    }

    @Override
    public int scrollVerticallyBy(int dy, RecyclerView.Recycler recycler, RecyclerView.State state) {
        direction = dy;
        return super.scrollVerticallyBy(dy, recycler, state);
    }

    @Override
    public void onChildViewAttachedToWindow(@NonNull View view) {
        if (mOnPageChangedListener != null && getChildCount() == 1) {
            mOnPageChangedListener.onPageInitComplete();
        }
    }

    @Override
    public void onChildViewDetachedFromWindow(@NonNull View view) {
        if (mOnPageChangedListener != null) {
            mOnPageChangedListener.onPageRelease(getPosition(view), direction >= 0);
        }
    }

    public void setOnPageChangedListener(OnPageChangedListener mOnPageChangedListener) {
        this.mOnPageChangedListener = mOnPageChangedListener;
    }

    public interface OnPageChangedListener {

        /**
         * 初始化子布局加载完成
         */
        void onPageInitComplete();

        /**
         * 子布局脱离
         *
         * @param position 子布局在RecyclerView位置
         * @param isNext   是否有下一个
         */
        void onPageRelease(int position, boolean isNext);

        /**
         * 子布局附着
         *
         * @param position 子布局在RecyclerView位置
         * @param isLast   是否最后一个
         */
        void onPageSelected(int position, boolean isLast);
    }

}

3. 视频滑动实现

为RecyclerView 设置管理器PagerLayoutManager,设置其Adapter数据内容,进行封面展示,且此时会回调onPageInitComplete()方法,进行首个视频播放。对RecyclerView进行滑动,当页面滑动结束后,会先回调管理器中onPageRelease()方法,此时可对进行中播放器进行停止释放;然后,回调onPageSelected()方法,对选中页面内容进行展示播放。

当滑动后取消时,要进行判断当前位置,避免当前页视频停止或重复播放。

mPagerLayoutManager = new PagerLayoutManager(this);
mPagerLayoutManager.setOnPageChangedListener(new PagerLayoutManager.OnPageChangedListener() {
@Override
public void onPageInitComplete() {
    int position = mPagerLayoutManager.findFirstVisibleItemPosition();
    if (position != -1) {
        mCurrentPosition = position;
    }
    startPlay(mCurrentPosition);
}

@Override
public void onPageRelease(int position, boolean isNext) {
    if (mCurrentPosition == position) {
        stopPlay();
    BaseViewHolder viewHolder = (BaseViewHolder) mRvLittleVideo.findViewHolderForLayoutPosition(mCurrentPosition);
    if (viewHolder != null) {
        ImageView mVideoThumb = viewHolder.getView(R.id.iv_thumb_item);
    if (mVideoThumb != null) {
        mVideoThumb.setVisibility(View.VISIBLE);
    }
    }
    }
}

@Override
public void onPageSelected(int position, boolean isLast) {
    if (mCurrentPosition == position) {
        return;
    }
    startPlay(position);
    mCurrentPosition = position;
    }
});

mRvLittleVideo.setLayoutManager(mPagerLayoutManager);

mLittleVideoAdapter = new LittleVideoAdapter();
mRvLittleVideo.setAdapter(mLittleVideoAdapter);

3. 播放器初始化及停止、播放

初始化播放器内容

private void initVideo() {

    mVideoView = new LittleVideoView(this);
    GSYVideoType.setShowType(GSYVideoType.SCREEN_TYPE_FULL);
    mGsySmallVideoHelperBuilder = new GSYVideoOptionBuilder();
    mGsySmallVideoHelperBuilder
            .setLooping(true)
            .setCacheWithPlay(true)
            .setIsTouchWiget(false)
            .setVideoAllCallBack(new GSYSampleCallBack() {
                @Override
                public void onPrepared(String url, Object... objects) {
                    super.onPrepared(url, objects);

                    new Handler().postDelayed(new Runnable() {
                        @Override
                        public void run() {
                            BaseViewHolder viewHolder = (BaseViewHolder) mRvLittleVideo.findViewHolderForLayoutPosition(mCurrentPosition);
                            if (viewHolder != null) {
                                ImageView mVideoThumb = viewHolder.getView(R.id.iv_thumb_item);
                                if (mVideoThumb != null) {
                                    mVideoThumb.setVisibility(View.INVISIBLE);
                                }
                            }
                        }
                    }, 100);
                }
            });
}

开始播放视频内容,进行播放器视图加载

private void startPlay(int position) {
    if (position < 0 || position >= mLittleVideoAdapter.getData().size()) {
        return;
    }
    BaseViewHolder holder = (BaseViewHolder) mRvLittleVideo.findViewHolderForLayoutPosition(position);
    ViewParent parent = mVideoView.getParent();
    if (parent instanceof FrameLayout) {
        ((ViewGroup) parent).removeView(mVideoView);
    }
    if (holder != null) {
        FrameLayout mVideoContent = holder.getView(R.id.fl_content_item);
        mVideoContent.addView(mVideoView, 0);
        mGsySmallVideoHelperBuilder.setUrl(mLittleVideoAdapter.getData().get(position).getUrl());
        mGsySmallVideoHelperBuilder.build(mVideoView);
        mVideoView.startPlayLogic();
    }
}

停止播放,移除视图

private void stopPlay() {
    mVideoView.release();
    ViewParent parent = mVideoView.getParent();
    if (parent instanceof FrameLayout) {
        ((FrameLayout) parent).removeView(mVideoView);
    }
}

3. 播放器内容

例子中采用了自定义空布局的播放器继承自GSY开源播放器,单纯进行视频播放,当然也可以采用其它的播放器饺子或者IjkPlayer等。

public class LittleVideoView extends StandardGSYVideoPlayer {

    public LittleVideoView(Context context, Boolean fullFlag) {
        super(context, fullFlag);
    }

    public LittleVideoView(Context context) {
        super(context);
    }

    public LittleVideoView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    @Override
    public int getLayoutId() {
        return R.layout.empty_control_video;
    }
}

布局文件

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@android:color/black">

    <FrameLayout
    android:id="@+id/surface_container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:gravity="center">

    </FrameLayout>

</RelativeLayout>

简单的滑动播放这些就完成了,例子也只是仅仅提供了实现的方法和思路,供大家进行学习参考,实际使用中可以对其进一步地进行封装及处理,接下来也会补充一些滑动播放适配器的数据加载处理以及多布局内容展示等内容。

Android短视频滑动播放(二)

关注公众号:几圈年轮,查看更多有趣的技术、工具、闲言、资源。

原文地址:https://www.cnblogs.com/jqnl/p/12131965.html

时间: 2024-08-25 23:04:35

Android短视频滑动播放(一)的相关文章

Android短视频开发

Android短视频录制 又拍云Android短视频录制SDK功能说明: Android 短视频播放 又拍云 Android 短视频播放器SDK功能说明:完美兼容Android2.3 (API)及其以上 支持多种视频协议:HLS,RTMP,HTTP-FLV等,支持HLS多种分辨率切换: 支持本地视频播放: 支持自由设置短视频播放窗口大小 支持全屏播放 支持缓冲区大小设置 提供UpVideoView控件 支持ARM.ARMv7a.ARM64v8a.X86等主流芯片体系架构. 进入又拍云短视频服务页

Android 实现视频录制播放

实现类似微信的小视频功能,可以录制一个视频然后播放该视频. 视频录制,使用一个自定义控件. <span style="font-size:14px;">/** * 视频录制控件 * * @author lip * * @date 2015-3-16 */ public class MovieRecorderView extends LinearLayout implements OnErrorListener { private SurfaceView mSurfaceVi

深度解析!短视频如何成为现象级产品

欢迎大家前往腾讯云+社区,获取更多腾讯海量技术实践干货哦~ 本文由 BestSDK 团队发布在腾讯云+社区 2016 年中国移动短视频用户数为 1.5 亿,今年预计会达到 2.4 亿,增长率高达 58.2%,可见短视频的热度在一直提升:近几年,短视频的生产模式在不断演进,从 UGC 到 PGC,再到最新的 MCN,内容的产能和质量均得到了巨大提升. 短视频发展史 ?图 1 图 1 所示是短视频及直播的发展史,众所周知,2016 年是直播元年,在这期间诞生了很多直播平台,比如熊猫.映客.斗鱼等:而

关于短视频

移动互联网用户的普及和流量增长,使得碎片化时代的体验得到保障,人们可以打破时间和空间限制观看浏览视频,因此短视频成为互联网时代更加便捷的内容形式.那么,接下来就给大家分享一些Android短视频开发方面的干货内容. 码率:数据传输时单位时间传送的数据位数,.码率越大,体积越大,清晰度越高:体积=码率*时间. VBR编码:画面质量高,体积小,编码时间慢. CBR编码:静态码率,编码速度快一些,文件大一些: 分辨率:分辨率越高,图像越大 清晰度:再码率一定情况下,分辨率越高图片越不清晰(本人调用的视

短视频繁荣的背后是遍体鳞伤?

近年来,短视频风口渐盛,引来腾讯.阿里.微博.今日头条等众多巨头投身红海.为了能在短视频领域中拥有绝对的话语权,一场巨头短视频争夺大战就这样轰轰烈烈地展开了. 头条系短视频点燃短视频大战,但受限于格局 早在2015年的时候,今日头条就进行"千人万元"的短视频扶持计划了,而在去年9月份,今日头条再次宣布投入10亿元来对"头条号"进行短视频内容的扶持,今年5月16日,今日头条又宣布投入10亿的资金来扶持火山小视频,由此可看出,自始至终,今日头条都是通过补贴来扶持短视频内

2018亚太CDN峰会开幕,阿里云王海华解读云+端+AI的短视频最佳实践

摘要: 4月11-12日,2018亚太CDN峰会在北京隆重召开,在11日下午的短视频论坛中,阿里云高级技术专家王海华进行了<短视频最佳实践:云+端+AI>的主题演讲,分享了短视频的生命周期关键点和阿里云技术解决方案. 4月11-12日,2018亚太CDN峰会在北京隆重召开,在11日下午的短视频论坛中,阿里云高级技术专家王海华进行了<短视频最佳实践:云+端+AI>的主题演讲,分享了短视频的生命周期关键点和阿里云技术解决方案.以下全文为演讲内容. 每个人都将成名十五秒钟 短视频已经走进

android webview 通过html5播放在线视频 切换大屏

1.添加网络访问权限 <uses-permission android:name="android.permission.INTERNET" /> 2.webview添加全屏支持 developer官方文档关于html5支持视频播放描述如下:In order to support inline HTML5 video in your application, you need to have hardware acceleration turned on, and set

android 使用WebView 支持播放优酷视频,土豆视频

看了很多文章和所谓的解决android WebView播放优酷,土豆等视频的办法,都是什么 setPluginsEnabled,在android 4.x之后都不好使,压根就没这函数,因为android 4.x 都去除了flash的支持.就这东西弄的我折腾了半天时间,现在的网站都是采集复制,浪费时间. 只有自己操刀了, 下面给出我的解决办法, 1. 修改AndroidManifest.xml 在Application节点上增加 ? 1 android:hardwareAccelerated="tr

Android MediaStore检索视频并播放

该文章是为了检索手机上sd卡中的视频,然后将检索出来的相应视频的缩略图,名称等视频信息显示在ListView上.点击每个item后播放相应的视频. 源代码: 布局文件: activity_main: <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:la