主布局:

 <?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context="net.bwie.surfaceviewvideolist.MainActivity"> <android.support.v7.widget.RecyclerView
android:id="@+id/recycler_view"
android:layout_width="match_parent"
android:layout_height="match_parent"
app:layoutManager="android.support.v7.widget.LinearLayoutManager"/> </RelativeLayout>

主布局的Item:

 <?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"> <TextView
android:id="@+id/progress_tv"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="当前进度"/> <Button
android:id="@+id/play_btn"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="播放"/> <TextView
android:id="@+id/title_tv"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="标题"/> <net.bwie.surfaceviewvideolist.widget.VideoSurfaceView
android:id="@+id/video_view"
android:layout_width="match_parent"
android:layout_height="200dp"
android:layout_gravity="center_horizontal"/> </LinearLayout>

主界面Activity:

 /**
* 1、封装播放视频的SurfaceView
* 2、网络请求数据(手写bean):bean、httpservice
* 3、放在RecyclerView中展示
* 4、item中放入SurfaceView
* 5、点击播放弹出通知
* 6、进度
*/
public class MainActivity extends AppCompatActivity implements Callback<KaiyanBean> { protected RecyclerView mRecyclerView;
private Object mData; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
super.setContentView(R.layout.activity_main);
initView(); initData();
} private void initData() {
Retrofit retrofit = new Retrofit.Builder()
.baseUrl("http://baobab.kaiyanapp.com/")
.addConverterFactory(GsonConverterFactory.create()).build(); retrofit.create(KaiyanHttpService.class)
.getKaiyanCall()
.enqueue(this);
} private void initView() {
mRecyclerView = (RecyclerView) findViewById(R.id.recycler_view);
} @Override
public void onResponse(Call<KaiyanBean> call, Response<KaiyanBean> response) {
KaiyanBean kaiyanBean = response.body(); List<VideoBean> videoList = new ArrayList<>(); for (KaiyanBean.ItemBean itemBean : kaiyanBean.getItemList()) {
Log.d("1507","type: " + itemBean.getType()+ ", title : " + itemBean.getData().getTitle() + "url : " + itemBean.getData().getPlayUrl()); if ("video".equals(itemBean.getType())) {
// 过滤掉type不为video的数据
VideoBean videoBean = new VideoBean(itemBean.getData().getTitle(), itemBean.getData().getPlayUrl());
videoList.add(videoBean);
} } VideoListAdapter adapter = new VideoListAdapter(this, videoList);
mRecyclerView.setAdapter(adapter);
} @Override
public void onFailure(Call<KaiyanBean> call, Throwable t) { }
}

Bean:

 package net.bwie.surfaceviewvideolist.bean;

 import java.util.List;

 public class KaiyanBean {

     private List<ItemBean> itemList;

     public List<ItemBean> getItemList() {
return itemList;
} public void setItemList(List<ItemBean> itemList) {
this.itemList = itemList;
} public static class ItemBean {
public String getType() {
return type;
} public void setType(String type) {
this.type = type;
} private String type;
private DataBean data; public DataBean getData() {
return data;
} public void setData(DataBean data) {
this.data = data;
} public static class DataBean { private String title;
private String playUrl; public String getTitle() {
return title;
} public void setTitle(String title) {
this.title = title;
} public String getPlayUrl() {
return playUrl;
} public void setPlayUrl(String playUrl) {
this.playUrl = playUrl;
}
} } }
 package net.bwie.surfaceviewvideolist.bean;

 public class VideoBean {

     private String title;
private String playUrl; public VideoBean(String title, String playUrl) {
this.title = title;
this.playUrl = playUrl;
} public String getTitle() {
return title;
} public void setTitle(String title) {
this.title = title;
} public String getPlayUrl() {
return playUrl;
} public void setPlayUrl(String playUrl) {
this.playUrl = playUrl;
}
}

Http:

 package net.bwie.surfaceviewvideolist.httpservice;

 import net.bwie.surfaceviewvideolist.bean.KaiyanBean;

 import retrofit2.Call;
import retrofit2.http.GET; public interface KaiyanHttpService { @GET("api/v4/tabs/selected?udid=11111&vc=168&vn=3.3.1&deviceModel=Huawei%36&first_channel=eyepetizer_baidu_market&last_channel=eyepetizer_baidu_market&system_version_code=20")
Call<KaiyanBean> getKaiyanCall(); }

封装类:

 /**
* 能播放视频的SurfaceView,封装MediaPlayer
* 提供一个public方法用于设置播放路径和播放的方法
*/
public class VideoSurfaceView extends SurfaceView implements SurfaceHolder.Callback, MediaPlayer.OnPreparedListener {
private SurfaceHolder mHolder;
private MediaPlayer mMediaPlayer; public VideoSurfaceView(Context context, AttributeSet attrs) {
super(context, attrs); init();
} private void init() {
// 获得缓冲区持有者
mHolder = getHolder();
// 设置生命周期的回调
mHolder.addCallback(this);
} // 设置播放路径并播放
public void playVideo(String path) {
if (mMediaPlayer == null) {
mMediaPlayer = new MediaPlayer();
// 准备完毕监听器
mMediaPlayer.setOnPreparedListener(this);
} try {
// 重置MediaPlayer
mMediaPlayer.reset();
// 设置画面播放源
mMediaPlayer.setDisplay(mHolder);
// 设置播放源
mMediaPlayer.setDataSource(path);
// 准备播放
mMediaPlayer.prepareAsync();
} catch (Exception e) {
e.printStackTrace();
Log.e("1507", e.getMessage());
} } // 停止播放
public void stop() {
if (mMediaPlayer != null) {
mMediaPlayer.stop();
}
} @Override
public void surfaceCreated(SurfaceHolder holder) { } @Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) { } @Override
public void surfaceDestroyed(SurfaceHolder holder) {
if (mMediaPlayer != null) {
mMediaPlayer.release();// 释放资源
mMediaPlayer = null;
}
} @Override
public void onPrepared(MediaPlayer mp) {
mMediaPlayer.start();
}
}

Adapter:

 public class VideoListAdapter extends RecyclerView.Adapter<VideoListAdapter.ViewHolder> {

     private Context mContext;
private List<VideoBean> mDatas; // 记录上一次播放的VideoView
private VideoSurfaceView mPlayingView; public VideoListAdapter(Context context, List<VideoBean> datas) {
mContext = context;
mDatas = datas;
} @Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View itemView = LayoutInflater.from(mContext)
.inflate(R.layout.item_video, parent, false);
return new ViewHolder(itemView);
} @Override
public void onBindViewHolder(final ViewHolder holder, int position) {
final VideoBean videoBean = mDatas.get(position); holder.mTitleTextView.setText(videoBean.getTitle()); // item刚复用进入屏幕时,无论是否播放都停止并隐藏
holder.mSurfaceView.stop();
holder.mSurfaceView.setVisibility(View.INVISIBLE); holder.mPlayBtn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) { // 下一次播放时,停止上一次播放
if (mPlayingView != null) {
mPlayingView.stop();
mPlayingView.setVisibility(View.INVISIBLE);
} // 播放当前位置的视频
holder.mSurfaceView.setVisibility(View.VISIBLE);// 显示
holder.mSurfaceView.playVideo(videoBean.getPlayUrl()); mPlayingView = holder.mSurfaceView;
}
});
} @Override
public int getItemCount() {
return mDatas == null ? 0 : mDatas.size();
} static class ViewHolder extends RecyclerView.ViewHolder { Button mPlayBtn;
TextView mTitleTextView;
TextView mProgressTextView;
VideoSurfaceView mSurfaceView; public ViewHolder(View itemView) {
super(itemView); mPlayBtn = ((Button) itemView.findViewById(R.id.play_btn));
mTitleTextView = ((TextView) itemView.findViewById(R.id.title_tv));
mProgressTextView = ((TextView) itemView.findViewById(R.id.progress_tv));
mSurfaceView = ((VideoSurfaceView) itemView.findViewById(R.id.video_view));
}
} }

别忘了加权限:

<uses-permission android:name="android.permission.INTERNET"/>

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

导包:

compile 'com.android.support:recyclerview-v7:26.0.0-alpha1'

compile 'com.squareup.retrofit2:converter-gson:2.3.0'

compile 'com.squareup.okhttp3:okhttp:3.9.0'

compile 'com.github.bumptech.glide:glide:3.8.0'

SurfaceViewVideoList网络获取视频播放的更多相关文章

  1. Android热身:通过网络获取资源并更新UI组件

    Android热身:通过网络获取资源并更新UI组件 目标 点击"发送请求"按钮,下载某网页的html源码,并显示在TextView控件上:点击"清空",清除Te ...

  2. Android代码优化----PullToRefresh+universal-image-loader实现从网络获取数据并刷新

    [声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.com/smyhvae/ 文章来源:http://www.cnblogs.com/smyhvae/p/4 ...

  3. Android ListView从网络获取图片及文字显示

    上一篇文章说的是ListView展示本地的图片以及文本,这一篇说一下如何从网络获取图片以及文本来显示.事实上,一般是先获取Josn或sml数据,然后解释显示.我们先从网上获取xml,然后对其进行解析, ...

  4. 网络获取数据的Xml的Pull解析

    网络获取的XML的Pull解析 <?xml version="1.0" encoding="utf-8" ?> - <students> ...

  5. 网络获取的XML的Pull解析

    <?xml version="1.0" encoding="utf-8" ?> - <students> - <student x ...

  6. Android ListView 和 ***Adapter 从本地/网络获取歌曲列表

    本文内容 环境 项目结构 测试数据 演示 1:SimpleAdapter 演示 2:BaseAdapter 演示 3:CustomLazyList 演示 4:CustomLazyCompleteLis ...

  7. Android LazyList 从网络获取图片并缓存

    原演示地址 本文内容 环境 演示 LazyList 从网络获取图片并缓存 参考资料 本文是 Github 上的一个演示,通过网络获取歌手专辑的缩略图,并显示在 ListView 控件中.该演示具备将缩 ...

  8. Android Volley 库通过网络获取 JSON 数据

    本文内容 什么是 Volley 库 Volley 能做什么 Volley 架构 环境 演示 Volley 库通过网络获取 JSON 数据 参考资料 Android 关于网络操作一般都会介绍 HttpC ...

  9. Recorder︱深度学习小数据集表现、优化(Active Learning)、标注集网络获取

    一.深度学习在小数据集的表现 深度学习在小数据集情况下获得好效果,可以从两个角度去解决: 1.降低偏差,图像平移等操作 2.降低方差,dropout.随机梯度下降 先来看看深度学习在小数据集上表现的具 ...

随机推荐

  1. Docker 镜像安装 GitLab 中文社区版

    docker run \ --detach \ --publish : \ --publish : \ --name gitlab \ --restart unless-stopped \ --vol ...

  2. docker学习实践之路[第二站]nginx镜像实践

    上一篇文章中已经成功的拉取的nginx的镜像 在本篇文章中则详细介绍docker利用文件卷.断后映射然后进行nginx的配置. 输入一下命令: docker run -d --name mynginx ...

  3. IE不兼容document.getElementsByClassName

    在DOM3里已经加入了getElementsByClassName这个方法,然而IE9.10以外的其它版本均不支持,这是一块伤痛啊! 目前可以这么解决,判断浏览器支不支持这个方法,如果支持就不管:如果 ...

  4. C# 多线程之List的线程安全问题

    网上关于List的线程安全问题将的很少,所以自己实验了一把,发现确实是线程不安全的.所以当你在进行多线程编程中使用了共享的List集合,必须对其进行线程安全处理. List的Add方法是线程不安全的, ...

  5. Entity framework 预热

    Entity framework  预热 对于在应用程序中定义的每个DbContext类型,在首次使用时,Entity Framework都会根据数据库中的信息在内存生成一个映射视图(mapping ...

  6. web的脚本安全-XSS

    XSS,即Cross Site Scripting,叫X是因为之前有了一个CSS.中文可以叫跨站脚本攻击.是前端工程师的一大威胁. XSS的根本,就是有恶意用户把代码植入了你要访问的页面中,从而控制你 ...

  7. windows cmd窗口提示“telnet”命令不能内部或外部命令,也不是可运行的程序

    windows cmd窗口提示“telnet”命令不能内部或外部命令,也不是可运行的程序 原因:C:\Windows\System32目录下没有telnet.exe,path系统变量的值包含了C:\W ...

  8. logstash使用template提前设置好maping同步mysql数据到Elasticsearch5.5.2

    上篇blog说到采用logstash-input-jdbc将mysql数据同步到ES(http://www.cnblogs.com/jstarseven/p/7704893.html),但是这里有一个 ...

  9. spring boot + mybatis + druid + redis

    接上篇,使用redis做缓存 新建spring boot 工程,添加pom引用 <dependency> <groupId>org.springframework.boot&l ...

  10. filebeat-2-通过kafka队列链接logstash

    filebeat 直接到logstash, 由于logstash的设计问题, 可能会出现阻塞问题, 因为中间使用消息队列分开 可以使用redis, 或者kafka, 这儿使用的是kafka 1, 安装 ...