Android 播放视频之VideoView

这篇具有很好参考价值的文章主要介绍了Android 播放视频之VideoView。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

1. VideoView使用

VideoView继承SurfaceView,并且实现了MediaPlayerControl接口,相当于MediaPlayer+SurfaceView的组合。关于MediaPlayer可参考这里。

在布局文件里添加VideoView

<VideoView
      android:id="@+id/video_view"
      android:layout_width="match_parent"
      android:layout_height="match_parent"/>

Activity里面播放视频

var mVideoView = findViewById(R.id.video_view)

mVideoView.setVideoURI(Uri.parse(uri))
// 设置控制器
mVideoView.setMediaController(MediaController(this))
mVideoView.start()

2. VideoView的主要方法

主要方法

// 设置视频源
void setVideoPath(String path)
void setVideoURI(Uri uri)
void setVideoURI(Uri uri, Map<String, String> headers)

// 恢复,从头播放
void resume()
// 播放
void start()
// 暂停
void pause()
// 释放资源
void suspend()
// 停止播放
void stopPlayback()

// 视频时长
int getDuration()
// 当前位置
int getCurrentPosition()
// 跳转到指定时间
void seekTo(int msec)

// 是否正在播放
boolean isPlaying()
// 设置控制器
void setMediaController(MediaController controller)

3. VideoView源码分析

VideoView里面有mCurrentStatemTargetState两个状态,当前状态和目标状态,初始值都是STATE_IDLE,播放的时候都与这两个状态有关。

VideoView里面的setVideoXXXresume方法都直接调用openVideo,在openVideo方法里面,调用MediaPlayer播放视频。
mPreparedListenerMediaPlayer准备好播放资源后,如果mSeekWhenPrepared被设置,跳转到指定时间。而如果mTargetState被设置为STATE_PLAYING,直接播放。

public void setVideoPath(String path) {
    setVideoURI(Uri.parse(path));
}

public void setVideoURI(Uri uri) {
    setVideoURI(uri, null);
}

public void setVideoURI(Uri uri, Map<String, String> headers) {
    mUri = uri;
    mHeaders = headers;
    mSeekWhenPrepared = 0;
    openVideo();
    requestLayout();
    invalidate();
}

public void resume() {
    openVideo();
}

private void openVideo() {
    if (mUri == null || mSurfaceHolder == null) {
        // not ready for playback just yet, will try again later
        return;
    }

    // 清空mCurrentState,但保留mTargetState
    release(false);

    mMediaPlayer = new MediaPlayer();

    mMediaPlayer.setOnPreparedListener(mPreparedListener);
    mMediaPlayer.setOnVideoSizeChangedListener(mSizeChangedListener);
    mMediaPlayer.setOnCompletionListener(mCompletionListener);
    mMediaPlayer.setDataSource(mContext, mUri, mHeaders);
    mMediaPlayer.setDisplay(mSurfaceHolder);
    mMediaPlayer.prepareAsync();

    mCurrentState = STATE_PREPARING;
}

private void release(boolean cleartargetstate) {
    if (mMediaPlayer != null) {
        mMediaPlayer.reset();
        mMediaPlayer.release();
        mMediaPlayer = null;
        mPendingSubtitleTracks.clear();
        mCurrentState = STATE_IDLE;
        if (cleartargetstate) {
            mTargetState  = STATE_IDLE;
        }
        if (mAudioFocusType != AudioManager.AUDIOFOCUS_NONE) {
            mAudioManager.abandonAudioFocus(null);
        }
    }
}

MediaPlayer.OnPreparedListener mPreparedListener = new MediaPlayer.OnPreparedListener() {
    public void onPrepared(MediaPlayer mp) {
        mCurrentState = STATE_PREPARED;

        if (mOnPreparedListener != null) {
            mOnPreparedListener.onPrepared(mMediaPlayer);
        }

        int seekToPosition = mSeekWhenPrepared;
        if (seekToPosition != 0) {
            seekTo(seekToPosition);
        }
        if (mVideoWidth != 0 && mVideoHeight != 0) {
            getHolder().setFixedSize(mVideoWidth, mVideoHeight);
            if (mSurfaceWidth == mVideoWidth && mSurfaceHeight == mVideoHeight) {
                if (mTargetState == STATE_PLAYING) {
                    start();
                } else if (!isPlaying() &&
                           (seekToPosition != 0 || getCurrentPosition() > 0)) {
                   if (mMediaController != null) {
                       mMediaController.show(0);
                   }
               }
            }
        } else {
            if (mTargetState == STATE_PLAYING) {
                start();
            }
        }
    }
};

VideoView里面startpause方法都是直接调用MediaPlayer的对应方法。suspendstopPlayback都是停止播放并释放资源,但stopPlayback会修改mTargetState

@Override
public void start() {
    if (isInPlaybackState()) {
        mMediaPlayer.start();
        mCurrentState = STATE_PLAYING;
    }
    mTargetState = STATE_PLAYING;
}

@Override
public void pause() {
    if (isInPlaybackState()) {
        if (mMediaPlayer.isPlaying()) {
            mMediaPlayer.pause();
            mCurrentState = STATE_PAUSED;
        }
    }
    mTargetState = STATE_PAUSED;
}

public void suspend() {
    release(false);
}

public void stopPlayback() {
    if (mMediaPlayer != null) {
        mMediaPlayer.stop();
        mMediaPlayer.release();
        mMediaPlayer = null;
        mCurrentState = STATE_IDLE;
        mTargetState  = STATE_IDLE;
        mAudioManager.abandonAudioFocus(null);
    }
}

getDurationgetCurrentPositionseekTo也都是调用MediaPlayer的对应方法,只是包装了一层状态的校验,比直接调用更合理。

public int getDuration() {
    if (isInPlaybackState()) {
        return mMediaPlayer.getDuration();
    }

    return -1;
}

@Override
public int getCurrentPosition() {
    if (isInPlaybackState()) {
        return mMediaPlayer.getCurrentPosition();
    }
    return 0;
}

@Override
public void seekTo(int msec) {
    if (isInPlaybackState()) {
        mMediaPlayer.seekTo(msec);
        mSeekWhenPrepared = 0;
    } else {
        mSeekWhenPrepared = msec;
    }
}

4. MediaController控制器

MediaController控制器是VideoView默认的控制器。

public void setMediaController(MediaController controller) {
    if (mMediaController != null) {
        mMediaController.hide();
    }
    mMediaController = controller;
    attachMediaController();
}

private void attachMediaController() {
    if (mMediaPlayer != null && mMediaController != null) {
        mMediaController.setMediaPlayer(this);
        View anchorView = this.getParent() instanceof View ?
                (View)this.getParent() : this;
        mMediaController.setAnchorView(anchorView);
        mMediaController.setEnabled(isInPlaybackState());
    }
}

MediaController创建界面,一般包含3个按钮,从左往右分别是倒退、播放/暂停和快进。底部从左到右是当前时间、进度条和总播放时间。拖动进度条可以跳转到指定时间。

public void setMediaPlayer(MediaPlayerControl player) {
    mPlayer = player;
    updatePausePlay();
}

public void setAnchorView(View view) {
    if (mAnchor != null) {
        mAnchor.removeOnLayoutChangeListener(mLayoutChangeListener);
    }
    mAnchor = view;
    if (mAnchor != null) {
        mAnchor.addOnLayoutChangeListener(mLayoutChangeListener);
    }

    FrameLayout.LayoutParams frameParams = new FrameLayout.LayoutParams(
            ViewGroup.LayoutParams.MATCH_PARENT,
            ViewGroup.LayoutParams.MATCH_PARENT
    );

    removeAllViews();
    View v = makeControllerView();
    addView(v, frameParams);
}

调用setPrevNextListeners 方法后,根据nextprev的值,在两边添加按钮。

public void setPrevNextListeners(View.OnClickListener next, View.OnClickListener prev)

显示如下
Android 播放视频之VideoView文章来源地址https://www.toymoban.com/news/detail-433802.html

到了这里,关于Android 播放视频之VideoView的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 使用 LibVLC 构建自定义 Android 视频播放器:分步指南,降低延迟/图像失帧(附源码)

    前言 在这篇博文中,我们将深入探讨使用 LibVLC 库的自定义 Android 视频播放器的实现细节。本分步指南将涵盖设置播放器、处理各种事件以及在您的 Android 应用程序中提供无缝视频播放体验的基本方面。 一、LibVLC 概述: 视频播放是许多 Android 应用程序中的常见功能,从流媒

    2024年02月02日
    浏览(86)
  • Android 使用ViewPager2+ExoPlayer+VideoCache 实现仿抖音视频翻页播放

    1. 实现效果    效果图中,视频没有铺满 是因为使用了ExoPlayer的RESIZE_MODE_FIT模式, 虽然使用RESIZE_MODE_FILL模式可以填充整个父布局,但是本Demo中使用的视频源本身就不适合全屏,会把视频拉伸,效果不好。 抖音上的视频源应该都有严格的宽高尺寸,才能做到全屏有很好的效

    2023年04月09日
    浏览(41)
  • Android修行手册 - VideoView全解-上

    往期文章分享 点击跳转=《导航贴》- Unity手册,系统实战学习 点击跳转=《导航贴》- Android手册,重温移动开发 本文约4.8千字,新手阅读需要9分钟,复习需要3分钟 【 收藏随时查阅不再迷路 】 众所周知,人生是一个漫长的流程,不断 克服困难 ,不断反思前进的过程。在这

    2024年02月01日
    浏览(43)
  • 【Android从零单排系列十八】《Android视图控件——VideoView》

    目录 前言 一 VideoView基本介绍 二 VideoView使用方法 三 VideoView常见属性及方法 四 VideoView简单Demo 五 总结 小伙伴们,在上文中我们介绍了Android视图组件WebView,本文我们继续盘点,介绍一下视图控件的VideoView。 videoView是Android平台上用于播放视频的控件,它提供了一些常见属性

    2024年02月09日
    浏览(43)
  • Unity VideoPlayer使用url方式,Android平台下无法播放http链接的视频(黑屏、无反应、无报错、无log输出)...

    以下内容基于 Unity 2019.4.28f1c1,未来可能有变化 (以下 Other Settings 供参考,未必影响) Edit - Project Settings - Player - Android平台 - Other Settings :设置如下 Rendering Auto Graphics API: true Configuration Scripting Backend: IL2CPP Install Location: Prefer External Internet Access: Auto 主要原因: 默认情况下,不允

    2023年04月18日
    浏览(44)
  • Android 视频播放

    1.用MEDIAPLAYER+TEXTUREVIEW封装好的视频播放器,可直接使用(附DEMO) 地址: 【Android 前沿技术】用MediaPlayer+TextureView封装好的视频播放器,可直接使用(附demo) - 灰信网(软件开发博客聚合)  https://www.freesion.com/article/3306560066/   下载地址: GitHub - colinNaive/VideoApplication: Androi

    2023年04月08日
    浏览(30)
  • Android 播放视频之ExoPlayer

    在上一篇中,我们了解了系统自带的 VideoView 来播放视频,但其支持的视频格式偏少,我们需要其他的视频播放器。 ExoPlayer 是Google的开源的应用级媒体播放器项目,支持多种视频格式和流媒体播放,GitHub地址。 导入 ExoPlayer 在布局文件里添加 StyledPlayerView 在 Activity 里面播放

    2024年02月15日
    浏览(49)
  • Android之播放本地视频和Url视频方法

    根据文件路径在浏览器中播放,可用于视频预览等场景 效果: 用浏览器播放本地视频 文件路径例子: /storage/emulated/0/Android/data/com.custom.jfrb/files/Movies/1697687179497.mp4 效果: 播放Url视频 1、导入依赖 2、xml文件 3、Activity中调用 (1) 绑定控件 (2)播放视频 视频Url例子:https:

    2024年02月08日
    浏览(52)
  • Android开发之视频播放器

    今天教大家做一个简单的视频播放器hh 至此,结束!

    2023年04月09日
    浏览(53)
  • Android 视频播放器dkplayer

    gihub地址 : https://github.com/Doikki/DKVideoPlayer GitHub - Doikki/DKVideoPlayer: Android Video Player. 安卓视频播放器,封装MediaPlayer、ExoPlayer、IjkPlayer。模仿抖音并实现预加载,列表播放,悬浮播放,广告播放,弹幕,视频水印,视频滤镜 列表播放如图所示: 一、依赖 二、工具类 Tag.java Vi

    2024年02月13日
    浏览(49)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包