【Android】 从头搭建视频播放器(5)——将所有放在一起

本文介绍了如何使用Android平台构建视频播放器的过程。通过整合多个组件,包括手势控制器、屏幕方向切换器等,创建了一个名为StrawMediaPlayer的完整播放器解决方案。文章详细说明了StrawMediaPlayer的职责与实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

【Android】 从头搭建视频播放器(5)——将所有放在一起


转载请注明出处http://blog.youkuaiyun.com/arnozhang12/article/details/48736363
 
 

1、简介

        结合前面的四篇文章,我们基本上已经了解了 Android 平台上面实现一款视频播放器所需要的方方面面。其中一些基础的控件实现,比如顶部的 Bar、底部的进度展示、全屏按钮等 View 的 Bar,由于实现起来非常简单,在此也没必要做过多的介绍。

        接下来,我们将之前做的 SystemMediaPlayerImpl、MediaPlayerGestureController、ScreenOrientationSwitcher 等等结合起来,完成最终提供给上层业务使用的 StrawMediaPlayer

2、StrawMediaPlayer 的职责

        有了前面介绍的基础模块,StrawMediaPlayer 的实现就变得异常简单了,只需要从 FrameLayout 派生一个 StrawMediaPlayer,把所有的 View(RootView、SurfaceView、PlayerTopControl、PlayerBottomControl、LoadingLayout等等)结合起来,按 BaseMediaPlayerListener 的回调进行相应的界面更新及处理即可。

        StrawMediaPlayer 做进行的处理如下:

  • 包含了前面所有介绍的相关模块,协调管理各个模块;
  • 继承自 FrameLayout,里面包含了所有的播放器相关的 ViewGroup 及 View,并监听处理这些 View 的操作事件以及更新这些 View(比如更新按钮的 drawable、更新 SeekBar 的 Progress、更新 Loading 画面等等);
  • 处理 onTouchEvent,分发 MotionEvent 到触摸事件处理模块(MediaPlayerGestureController);
  • 对上层提供一些接口,方便上层进行访问(如 playMedia、pauseMedia、resumeMedia 等等);
  • 接受上层提供的 Listener,将上层关心的一系列事件通知给上层(setPlayerListener)。

3、模块职责图回顾

        可以看出,我们在实现整个播放器的过程中,采用的思路是:先用 自顶向下 的思维进行需求分析,细化功能点,抽离出主要的模块及职责,确定出各个模块之间的交互关系。然后在实现的时候采用 自底向上 的方法,将各个模块一一实现,最终按预先设想的交互关系组合在一起,得到了我们的最终结果。

        我们再次回顾一下 第一篇 中的提到的模块结构图,以便对 StrawMediaPlayer 的职责做进一步的深化了解:

模块职责图

        再贴一张最终的实现图吧:

最终实现图

4、StrawMediaPlayer 的声明 & 部分实现

/**
 * StrawMediaPlayer.java
 *
 * @author arnozhang
 * @email  zyfgood12@163.com
 * @date   2015.9.25
 */
public class StrawMediaPlayer extends FrameLayout {

    private Context mContext;
    private ScreenOrientationSwitcher mScreenOrientationSwitcher;
    private MediaPlayerGestureController mGestureController;
    private PlayerBottomControl mPlayerBottomControl;


    public StrawMediaPlayer(Context context) {
        this(context, (AttributeSet) null, 0);
    }

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

    public StrawMediaPlayer(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        init(context);
    }

        private void init(Context context) {
        if (isInEditMode()) {
            return;
        }

        mContext = context;
        View.inflate(context, R.layout.layout_straw_media_player, this);

        initView();
        initMediaPlayer();
        initGestureController();
        initOrientationSwitcher();

        requestFocus();
    }

    public void doDestroy() {
        LogUtils.e(TAG, "MediaPlayer Will **Destroy**!!");

        unScheduleAuditionCheck();
        mScreenOrientationSwitcher.disable();
        mLockOrientationPanel.doDestroy();
        mPlayerBottomControl.doDestroy();
        mMediaPlayer.doDestroy();
    }

    private void initMediaPlayer() {
        SurfaceView surfaceView = (SurfaceView) findViewById(R.id.player_surface);
        mMediaPlayer = new SystemMediaPlayerImpl(mContext, surfaceView);
        mMediaPlayer.addPlayerListener(mBasePlayerListener);

        // ...
    }

    public boolean onTouchEvent(MotionEvent ev) {
        mGestureController.handleTouchEvent(ev);
        return true;
    }

    public void setPlayerListener(MediaPlayerListener listener) {
        mPlayerListener = listener;
        mPlayerBottomControl.setPlayerListener(listener);
        mMediaPlayer.addPlayerListener(listener);
    }

    private BaseMediaPlayerListener mBasePlayerListener = new BaseMediaPlayerListener() {
        @Override
        public void onLoading() {
            showLoading(R.string.qcloud_player_dialog_video_loading);
        }

        @Override
        public void onLoadFailed() {
            showLoading(R.string.qcloud_player_video_load_time_out);
        }

        @Override
        public void onFinishLoading() {
            mLoadingContainer.setVisibility(View.GONE);
            mPlayerBottomControl.switchViewState(mIsFullscreen);
        }

        // ...
        // ...
    };

    public void pauseMedia() {
        if (isLoading()) {
            return;
        }

        mMediaPlayer.pause();
    }

    public void resumeMedia() {
        if (isLoading()) {
            return;
        }

        mMediaPlayer.resume();
    }

    public void stopMedia() {
        unScheduleAuditionCheck();
        mMediaPlayer.stop();
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        mMediaPlayer.updateSurfaceSize(getMeasuredWidth());
    }

    public void playMedia(VideoInfo info, int startMillSeconds {
           mMediaPlayer.play(info);

           // ...
    }

    // ...
}
课程介绍:欢迎加入《Electron实战:基于Electron从零开发本地音乐播放器》视频课程,这是一门专为前端开发者、全栈工程师以及对桌面应用开发感兴趣的学习者量身打造的实战课程。本课程不仅教授如何使用Electron框架构建跨平台的桌面应用程序,更通过一个具体的项目——本地音乐播放器,带你深入了解Electron的强大功能和灵活性。 教学设计:本课程采用项目驱动的教学模式,从零开始,逐步引导你构建一个功能齐全的本地音乐播放器。课程内容涵盖了Electron的基础知识、核心模块的使用、与Node.js和Chromium的集成、以及如何处理文件系统、音频播放等实际问题。每个章节都紧密衔接,确保你在实践中掌握知识点。内容特色:实战导向: 通过实际项目,让你在动手操作中学习,避免枯燥的理论灌输。模块化教学:课程内容分为多个模块,每个模块解决一个核心问题,便于理解和吸收。最新技术栈:使用最新的Electron版本,确保你学习的技术不落伍。代码质量:强调良好的编码习惯和代码组织,提升你的软件工程能力。用户界面设计: 结合现代前端技术,如React或Vue,打造美观且用户友好的界面。 讲解方式:直观演示: 通过屏幕录制和实时编码,让你直观地看到每一步的实现过程。详细解说: 每一步操作都配有详细的语音解说,确保你理解背后的原理。互动问答: 提供在线问答环节,解答你在学习过程中遇到的问题。实战练习: 每个模块结束后都有相应的练习,巩固所学知识。 差异化特点:项目驱动: 与其他课程相比,本课程更注重项目的完整性和实用性,让你学以致用。技术深度: 深入探讨Electron的高级特性,如进程间通信、性能优化等,提升你的技术深度。社区支持: 提供社区交流平台,让你在学习过程中能够与其他学习者交流心得,共同进步。加入我们,一起用Electron打造你的第一个桌面应用,开启跨平台开发的新篇章!
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值