网站建设费属于哪个会计科目/黑马培训
我们要在播放控制栏下面加入下面一行。
这个就是标准的row。
leanback的原理
Android Leanback结构源码简析 - 简书
我们知道Row用来提供数据,row可以通过一个ObjectAdapter来管理和提供数据
我们知道presenter是一个负责将数据绑定到视图上的对象,它可以根据不同的数据类型创建不同的视图
如何做上图红框中的UI呢?
RowsSupportFragment用来干什么的呢?
RowsSupportFragment是一个Fragment,它用于显示一个垂直的列表,列表中的每一行都是由ObjectAdapter提供的数据和RowPresenter创建的视图组成。
PlaybackSupportFragment
我们知道VideoSupportFragment继承于PlaybackSupportFragment,我们接下来看下PlaybackSupportFragment的实现
public class PlaybackSupportFragment extends Fragment {//用于显示list,覆盖了PlaybackSupportFragment整个页面
RowsSupportFragment mRowsSupportFragment;//设置给mRowsSupportFragment用于显示垂直列表
ObjectAdapter mAdapter;//播放控制presenter
PlaybackRowPresenter mPresenter;//播放控制的row
Row mRow;@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState){/**
*获取RowsSupportFragment,RowsSupportFragment作为layout中的一部分
*/
mRowsSupportFragment = (RowsSupportFragment) getChildFragmentManager().findFragmentById(
R.id.playback_controls_dock);if (mRowsSupportFragment == null) {
mRowsSupportFragment = new RowsSupportFragment();
getChildFragmentManager().beginTransaction().replace(R.id.playback_controls_dock, mRowsSupportFragment).commit();}// 这里设置了adapter,如果mAdapter没有设置,mAdapter就是ArrayObjectAdapter,Prsenter是ClassPresenterSelectorif (mAdapter == null) {
setAdapter(new ArrayObjectAdapter(new ClassPresenterSelector()));} else {
mRowsSupportFragment.setAdapter(mAdapter);}
mRowsSupportFragment.setOnItemViewSelectedListener(mOnItemViewSelectedListener);
mRowsSupportFragment.setOnItemViewClickedListener(mOnItemViewClickedListener); }/**
*setAdapter可以知道都设置到了mRowsSupportFragment
*/
public void setAdapter(ObjectAdapter adapter) {
mAdapter = adapter;
setupRow();
setupPresenter();
setPlaybackRowPresenterAlignment();if (mRowsSupportFragment != null) {
mRowsSupportFragment.setAdapter(adapter);}}}/**
*setupRow可以看到是把row放到了adapter中
*/private void setupRow() {if (mAdapter instanceof ArrayObjectAdapter && mRow != null) {
ArrayObjectAdapter adapter = ((ArrayObjectAdapter) mAdapter);if (adapter.size() == 0) {
adapter.add(mRow);} else {
adapter.replace(0, mRow);}} else if (mAdapter instanceof SparseArrayObjectAdapter && mRow != null) {
SparseArrayObjectAdapter adapter = ((SparseArrayObjectAdapter) mAdapter);
adapter.set(0, mRow);}}//为adapter设置PresenterSelectorprivate void setupPresenter() {if (mAdapter != null && mRow != null && mPresenter != null) {
PresenterSelector selector = mAdapter.getPresenterSelector();if (selector == null) {
selector = new ClassPresenterSelector();((ClassPresenterSelector) selector).addClassPresenter(mRow.getClass(), mPresenter);
mAdapter.setPresenterSelector(selector);} else if (selector instanceof ClassPresenterSelector) {((ClassPresenterSelector) selector).addClassPresenter(mRow.getClass(), mPresenter);}}} public void setPlaybackRow(Row row) {this.mRow = row;
setupRow();
setupPresenter();} public void setPlaybackRowPresenter(PlaybackRowPresenter presenter) {this.mPresenter = presenter;
setupPresenter();
setPlaybackRowPresenterAlignment();}
总结下,上面的代码就是创建了RowsSupportFragment,创建了ArrayObjectAdapter,并把创建的ArrayObjectAdapter设置给RowsSupportFragment。
播放器的控制UI是如何显示在页面的呢?
我们平常使用leanback定义的播放控制UI,我们通过下面的方式使用
//mPlayerAdapter封装了player mMediaPlayerGlue = new PlaybackTransportControlGlue(getActivity(),
mPlayerAdapter);
mHost = new VideoSupportFragmentGlueHost(VideoSupportFragment);
mMediaPlayerGlue.setHost(mHost);
通过上面的代码,我们知道了mMediaPlayerGlue通过VideoSupportFragmentGlueHost持有了VideoSupportFragment的一个实例。那么播放器的控制UI又是如何添加到VideoSupportFragment中的呢?
我们接下来看下PlaybackTransportControlGlue
我们通过代码发现PlaybackTransportControlGlue继承于PlaybackBaseControlGlue我们接下来看下PlaybackBaseControlGlueclass PlaybackBaseControlGlue extends PlaybackGlue{
void onCreateDefaultControlsRow() {
if (mControlsRow == null) {
PlaybackControlsRow controlsRow = new PlaybackControlsRow(this);
setControlsRow(controlsRow);
}
} /**
*由于presenter关系到视图的样式,所以需要在子类中实现
*/
void onCreateDefaultRowPresenter() {
if (mControlsRowPresenter == null) {
setPlaybackRowPresenter(onCreateRowPresenter());
}
} /**
*这里的host就是VideoSupportFragmentGlueHost,通过查看源码发现
*setPlaybackRowPresenter和setPlaybackRow我们知道最后就设置到了PlaybackSupportFragment中的mAdapter中,mAdapter又设置给了RowsSupportFragment
*/
@Override
protected void onAttachedToHost(PlaybackGlueHost host) {
super.onAttachedToHost(host); onCreateDefaultControlsRow();
onCreateDefaultRowPresenter();
host.setPlaybackRowPresenter(getPlaybackRowPresenter());
host.setPlaybackRow(getControlsRow());
}
}public class PlaybackTransportControlGlue<T extends PlayerAdapter>
extends PlaybackBaseControlGlue<T> { @Override
protected PlaybackRowPresenter onCreateRowPresenter() { PlaybackTransportRowPresenter rowPresenter = new PlaybackTransportRowPresenter() {
@Override
protected void onBindRowViewHolder(RowPresenter.ViewHolder vh, Object item) {
super.onBindRowViewHolder(vh, item);
vh.setOnKeyListener(PlaybackTransportControlGlue.this);
}
@Override
protected void onUnbindRowViewHolder(RowPresenter.ViewHolder vh) {
super.onUnbindRowViewHolder(vh);
vh.setOnKeyListener(null);
}
};
return rowPresenter;
}
}
通过上面的分析我们就知道了PlaybackControlsRow和PlaybackTransportRowPresenter,添加到了到了VideoSupportFragment中的RowsSupportFragment中mAdapter中。
这个架构有点绕,我们再总结下。
PlaybackSupportFragment中有个成员变量RowsSupportFragment,RowsSupportFragment覆盖于PlaybackSupportFragment。在PlaybackSupportFragment创建了ArrayObjectAdapter,名字为mAdapter,并设置给了RowsSupportFragment。我们就知道了,只要把row和present添加到mAdapter中就可以实现UI的正常显示。
PlaybackTransportControlGlue最为粘合剂,实现了UI和播放器的粘合。我们在PlaybackTransportControlGlue的基类PlaybackBaseControlGlue的onAttachedToHost发现创建了播放控制的row和presenter,并通过VideoSupportFragmentGlueHost实现对PlaybackSupportFragment的操作setPlaybackRow和setPlaybackRowPresenter,并添加到了PlaybackSupportFragment的mAdapter,添加mAdapter的row和presenter。
RowsSupportFragment我们知道就是垂直的列表,列表根据row和presenter。
如何在播放控制UI下面继续添加新的UI呢?
我们通过上面分析知道了,播放控制UI就是加在了PlaybackSupportFragment的mAdapter中,那么我们也可以非常简单的新建ROW和presenter并将其加入到mAdapter中。
ListRow
//A Row composed of a optional HeaderItem, and an ObjectAdapter describing the items in the list.public class ListRow extends Row {private final ObjectAdapter mAdapter;private CharSequence mContentDescription;/**
* Returns the {@link ObjectAdapter} that represents a list of objects.
*/public final ObjectAdapter getAdapter() {return mAdapter;}public ListRow(HeaderItem header, ObjectAdapter adapter) {super(header);
mAdapter = adapter;verify();}public ListRow(long id, HeaderItem header, ObjectAdapter adapter) {super(id, header);
mAdapter = adapter;verify();}public ListRow(ObjectAdapter adapter) {super();
mAdapter = adapter;verify();}}
通过类的描述我们知道ListRow是用来显示一个标题加一个列表的样式。形如
我们看到有一个构造函数
public ListRow(HeaderItem header, ObjectAdapter adapter) {super(header);
mAdapter = adapter;verify();}
HeaderItem只要传入一个String即可创建,那么adapter怎么处理呢?
既然是列表那么adapter肯定是ArrayObjectAdapter,ArrayAdapter又如何创建呢?
public ArrayObjectAdapter(Presenter presenter) {
super(presenter);
}
Presenter使用来显示具体UI,并实现数据绑定的。我们接下来看看presenter的实现
public abstract class Presenter implements FacetProvider {/***创建viewHolder,实现view复用与避免每次执行昂贵的findviewById**/public abstract ViewHolder onCreateViewHolder(ViewGroup parent);/****数据绑定*/public abstract void onBindViewHolder(ViewHolder viewHolder, Object item);/***数据销毁*/public abstract void onUnbindViewHolder(ViewHolder viewHolder); }
presenter创建成功后,我们可以通过ArrayObjectAdapter的
public void setItems(final List itemList, final DiffCallback callback)
设置具体数据的显示。
下面是具体的demo。
leanbcakshowcase: android leanback 例子的二次修改
下面三个文件是具体的实现
CardPresenter.java
VideoConsumptionExampleFragmentV1.java
VideoMediaPlayerGlueV1.java