一般对于一个播放器,应该支持如下几种显示模式:

  • 等比例,最大化区域显示,不裁剪
  • 等比例,最大区域显示,裁剪
  • 拉伸显示,铺满全屏

要实现这几种显示模式,其实只要对播放控件的布局进行些许调整即可.那EasyPlayer是怎么实现的呢?

EasyPlayer使用一个PlayFragment进行视频播放.该类实现了一个setScaleType的方法来进行显示模式设定(或切换).

public void setScaleType(@IntRange(from = ASPACT_RATIO_INSIDE, to = FILL_WINDOW) int type){
mRatioType = type;
if (mWidth != 0 && mHeight != 0){
onVideoSizeChange();
}
}

这里的逻辑很简单,首先保存一下type的状态,并判断视频的宽高是否已经获取到了,如果是,则立马对view的尺寸进行计算,否则等获取到了后再计算.

其参数支持如下几种:

/**
* 等比例,最大化区域显示,不裁剪
*/
public static final int ASPACT_RATIO_INSIDE = 1;
/**
* 等比例,裁剪,裁剪区域可以通过拖拽展示\隐藏
*/
public static final int ASPACT_RATIO_CROPE_MATRIX = 2;
/**
* 等比例,最大区域显示,裁剪
*/
public static final int ASPACT_RATIO_CENTER_CROPE = 3;
/**
* 拉伸显示,铺满全屏
*/
public static final int FILL_WINDOW = 4;

我们结合代码与注释,看看onVideoSizeChange()函数主要做了什么:

private void onVideoSizeChange() {
// 视频尺寸未获取成功,直接返回.
if (mWidth == 0 || mHeight == 0) return;
if (mAttacher != null){ // 如果之前的模式是ASPACT_RATIO_CROPE_MATRIX,先释放attacher.
mAttacher.cleanup();
mAttacher = null;
}
if (mRatioType == ASPACT_RATIO_CROPE_MATRIX) {
// ...
// 这种情况下,需要将显示区域铺满父窗口.并使用matrix和手势进行显示控制.
// 具体的控制方法,参考文章:http://blog.csdn.net/jyt0551/article/details/56063869
mSurfaceView.getLayoutParams().height = ViewGroup.LayoutParams.MATCH_PARENT;
mSurfaceView.getLayoutParams().width = ViewGroup.LayoutParams.MATCH_PARENT;
// 初始化新的Attacher进行显示控制.
// ...
// 使用一个角度控件来显示当前的拖动角度.
mAngleView.setVisibility(View.VISIBLE);
}else {
// 重置matrix为默认(不做显示形变)
mSurfaceView.setTransform(new Matrix());
mAngleView.setVisibility(View.GONE);
// 分别计算出显示控件的宽高比和视频的宽高比
float ratioView = getView().getWidth() * 1.0f/getView().getHeight();
float ratio = mWidth * 1.0f/mHeight; switch (mRatioType){
case ASPACT_RATIO_INSIDE: {
if (ratioView - ratio < 0){ // 屏幕宽高比相比视频的宽高比更小.表示视频是过于宽了.
// 那就以宽为基准.宽最大化
mSurfaceView.getLayoutParams().width = ViewGroup.LayoutParams.MATCH_PARENT;
// 高进行等比例缩放
mSurfaceView.getLayoutParams().height = (int) (getView().getWidth() / ratio + 0.5f);
}else{ // 视频是竖屏了.
// 以高为基准,高最大化
mSurfaceView.getLayoutParams().height = ViewGroup.LayoutParams.MATCH_PARENT;
// 宽进行等比例缩放
mSurfaceView.getLayoutParams().width = (int) (getView().getHeight() * ratio + 0.5f);
}
}
break;
case ASPACT_RATIO_CENTER_CROPE: {
// 以更短的为基准
if (ratioView - ratio < 0){ // 屏幕宽高比相比视频的宽高比更小.表示视频是过于宽了.横向裁剪
// 以高为基准.高铺满全屏
mSurfaceView.getLayoutParams().height = ViewGroup.LayoutParams.MATCH_PARENT;
// 等比例计算出横向区域..多出父控件的范围将被裁剪
mSurfaceView.getLayoutParams().width = (int) (getView().getHeight() * ratio+ 0.5f);
}else{ // 视频是竖屏了.
// 那就以宽为基准.宽铺满屏幕
mSurfaceView.getLayoutParams().width = ViewGroup.LayoutParams.MATCH_PARENT;
// 等比例计算出纵向区域,超出父控件的范围将被裁剪
mSurfaceView.getLayoutParams().height = (int) (getView().getWidth() / ratio+ 0.5f);
}
}
break;
case FILL_WINDOW:{
// 铺满父控件即可...
mSurfaceView.getLayoutParams().height = ViewGroup.LayoutParams.MATCH_PARENT;
mSurfaceView.getLayoutParams().width = ViewGroup.LayoutParams.MATCH_PARENT;
}
break;
}
}
// 布局更改,刷新下.
mSurfaceView.requestLayout();
}

onVideoSizeChange函数,除了在手动设置显示模式的时候调用外,在成功获取到了视频宽高,或者视频宽高更改之后,也会调用.如下:

if (resultCode == EasyRTSPClient.RESULT_VIDEO_DISPLAYED) {
// 视频开始显示了.
onVideoDisplayed();
} else if (resultCode == EasyRTSPClient.RESULT_VIDEO_SIZE) {
// 视频分辨率获取到,或者更改了
mWidth = resultData.getInt(EasyRTSPClient.EXTRA_VIDEO_WIDTH);
mHeight = resultData.getInt(EasyRTSPClient.EXTRA_VIDEO_HEIGHT);
onVideoSizeChange();
}

在EasyPlayer工程的demo事例里面,调用方法如下:

    public void onToggleAspectRatio(View view) {
PlayFragment f =mRenderFragment;
if (f == null) return;
f.setScaleType(++i);
switch (i){
case PlayFragment.ASPACT_RATIO_INSIDE: {
Toast.makeText(this,"等比例居中",Toast.LENGTH_SHORT).show();
}
break;
case PlayFragment.ASPACT_RATIO_CENTER_CROPE: {
Toast.makeText(this,"等比例居中裁剪视频",Toast.LENGTH_SHORT).show();
}
break;
case PlayFragment.FILL_WINDOW:{
Toast.makeText(this,"拉伸视频,铺满区域",Toast.LENGTH_SHORT).show();
}
break;
case PlayFragment.ASPACT_RATIO_CROPE_MATRIX:{
Toast.makeText(this,"等比例显示视频,可拖拽显示隐藏区域.",Toast.LENGTH_SHORT).show();
}
break;
}
if (i == PlayFragment.FILL_WINDOW){
i = 0;
}
}

让我们看看展示效果:

关于EasyPlayer流媒体播放器

An elegant, simple, fast android RTSP/RTMP/HLS/HTTP Player.EasyPlayer support RTSP(RTP over TCP/UDP)version & Pro version,cover all kinds of streaming media!EasyPlayer是一款精炼、高效、稳定的流媒体播放器,分为RTSP版、RTMP版和Pro版三个版本,支持各种各样的流媒体音视频协议和文件的播放,在安防、互联网、教育、录播、IPTV等多个领域大放异彩,广泛应用!

EasyPlayer:https://github.com/EasyDSS/EasyPlayer

点击链接加入群【EasyPlayer】:544917793

获取更多信息

邮件:support@easydarwin.org

EasyDarwin开源流媒体服务器:www.EasyDarwin.org

EasyDSS商用流媒体解决方案:www.EasyDSS.com

EasyNVR无插件直播方案:www.EasyNVR.com

Copyright © EasyDarwin Team 2012-2017

EasyPlayer RTSP 安卓Android播放器显示模式设置方法的更多相关文章

  1. EasyPlayer RTSP安卓Android播放器架构简析

    本文转自EasyDarwin开源团队成员John的博客:http://blog.csdn.net/jyt0551/article/details/73310641 EasyPlayer 是一款小而美的 ...

  2. RTSP Windows专用播放器EasyPlayer : 稳定、兼容、高效、超低延时

    EasyPlayer RTSP Windows专用播放器 EasyPlayer RTSP Windows 播放器是由EasyDarwin团队开发和维护的一个完善的RTSP流媒体播放器项目,视频编码支持 ...

  3. github上十二款最著名的Android播放器开源项目

    1.ijkplayer 项目地址: https://github.com/Bilibili/ijkplayer 介绍:Ijkplayer 是Bilibili发布的基于 FFplay 的轻量级 Andr ...

  4. GitHub上最著名的Android播放器开源项目大全

    GitHub上最著名的Android播放器开源项目大全                                                                          ...

  5. EasyPlayer安卓Android流媒体播放器实现直播过程中客户端快照功能

    本文转自:http://blog.csdn.net/jyt0551/article/details/56942795 对于一个裸的RTSP URL,存放在播放列表上略显单调与枯燥.大家可以看到Easy ...

  6. 【翻译】安卓新播放器EXOplayer介绍

    http://developer.android.com/guide/topics/media/exoplayer.html   前言: Playing videos and music is a p ...

  7. Android——播放器和图片轮播

    layout文件: <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:an ...

  8. 安卓MP3播放器开发实例(1)之音乐列表界面

    学习安卓开发有一年了,想想这一年的努力,确实也收获了不少.也找到了比較如意的工作. 今天准备分享一个以前在初学阶段练习的一个项目.通过这个项目我真正的找到了开发安卓软件的感觉,从此逐渐步入安卓开发的正 ...

  9. Android播放器推荐:可以播放本地音乐、视频、在线播放音乐、视频、网络收音机等

    下载链接:http://www.eoeandroid.com/forum.php?mod=attachment&aid=MTAxNTczfGMyNjNkMzFlfDEzNzY1MzkwNTR8 ...

随机推荐

  1. Python Challenge 第七关

    第七关,只有一张图片,右键源代码也什么都没有,只是这图片上有一行类似马赛克一样的部分.看来答案只有在这张图上找了.下载了图片,去网上搜一下有什么库可以处理图像.搜到了一个PIL,发现安装的python ...

  2. springBoot AOP环绕增强、自定义注解、log4j2、MDC

    (一)log4j2 maven配置 <dependency> <groupId>org.springframework.boot</groupId> <art ...

  3. WKWebView携带不上cookie的问题处理

    自从WKWebView推出后Apple官方及众多开发者都推荐使用它代替UIWebView,确实通过加载速度.占用内存方面的对比都要好上几个档次,索性就把项目中的浏览器控件换成了WKWebView,一开 ...

  4. Cryptography I 学习笔记 --- 数论简介

    0. Zn代表{0,1....n-1}的集合 1. 模运算符合交换律结合律 2. gcd(greatest common divisor),可以由扩展欧几里得算法快速得到. 3. 模逆(modular ...

  5. 关联分析(Apriori算法)

    两个概念: 频繁项集:常出现的物品集合 关联分析:找到诸如:尿布-->啤酒的关联,反过来则是另一条 两个控制参数: 项集的支持度(support):一个项集出现的次数在所有样本中出现的比例 可信 ...

  6. HDU 1024 Max Sum Plus Plus [动态规划+m子段和的最大值]

    Max Sum Plus Plus Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Tot ...

  7. 【共享单车】—— React后台管理系统开发手记:主页面架构设计

    前言:以下内容基于React全家桶+AntD实战课程的学习实践过程记录.最终成果github地址:https://github.com/66Web/react-antd-manager,欢迎star. ...

  8. 点击页面li显示li中文字

    <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/ ...

  9. Android世界第一个activity启动过程

    Android世界第一个activity启动过程 第一次使用Markdown,感觉不错. Android系统从按下开机键一直到launcher的出现,是一个如何的过程,中间都做出了什么操作呢.带着这些 ...

  10. APT攻击:91%的攻击是利用电子邮件

    一封假冒的"二代医疗保险补充保险费扣费说明",导致上万家中小型企业的资料被窃;一封伪装银行交易纪录的钓鱼信件,让韩国爆发史上最大黑客攻击. APT攻击通常会以电子邮件的形式出现,邮 ...