关于Graywater的系列文章
- RecyclerView的超强辅助Graywater——理论篇
- RecyclerView的超强辅助Graywater——基础实操篇
- RecyclerView的超强辅助Graywater——点击事件
- RecyclerView的超强辅助Graywater——综合实操篇
看到上一篇Graywater文章是11月8日写的,也是拖很久了,这篇是Graywater系列文章最后一篇完结篇,把自己挖的坑给填了。
这篇文章是Graywater综合练习文章,包括了复杂视图、基本增删改功能的实现,文末附有Demo地址,同样Demo也只是起一个抛砖引玉的作用,实际上还有很大的延展空间。
先上Demo效果图:
GraywaterDemo综合.gif老规矩,还是先说说本文目录:
1. 复杂视图的实现
2. 数据的插入
3. 数据的删除
4. 数据的更新
5. 注意点
1. 复杂视图的实现
从GIF图中可以看到,该RecyclerView分为2大模块,娱乐新闻模块和体育新闻模块,每个模块对应一个数据源Primitive和一个管理器ItemBinder,数据源就对应了该模块所有数据内容,管理器对该模块所有视图进行管理。
比如娱乐新闻模块对应了一个EntertainPrimitive和EntertainItemBinder,包含了该模块的所有数据,同时还包含了娱乐新闻标题Binder和娱乐新闻内容的Binder(娱乐新闻内容的Binder使用List集合保存)。
GraywaterDemo模块图.png接着往下看,在RecyclerView里有3种视图:标题视图、娱乐新闻视图、体育新闻视图。每种视图都有一组自己对应的Binder、ViewHolder和ViewHolderCreator以及自己的布局文件。
看看实现娱乐新闻视图的核心代码:
EntertainItemBinder.java
@NonNull
@Override
public List<GraywaterAdapter.Binder<? super EntertainPrimitive, ? extends PrimitiveViewHolder>> getBinderList(@NonNull final EntertainPrimitive model, int position) {
return new ArrayList<GraywaterAdapter.Binder<? super EntertainPrimitive, ? extends PrimitiveViewHolder>>() {{
add(titleBinder); //添加娱乐新闻标题
for (EntertainItem entertainItem : model.getEntertainItems()) {
add(entertainBinder); //添加娱乐新闻内容
}
}};
}
所以在ItemBinder中可以添加多种类型的binder,就可以实现不同的视图了。
2. 数据的插入
插入娱乐新闻的核心代码:
@Override
public void onClickaddNews(String type) {
Toast.makeText(this, "add " + type + " success", Toast.LENGTH_SHORT).show();
switch (type) {
case TYPE_ENTERTAIN:
addEntertainNews();
mPrimitiveAdapter.remove(POS_ENTERTAIN);
mPrimitiveAdapter.add(POS_ENTERTAIN, mEntertainPrimitive);
//不需要mPrimitiveAdapter.notifyDataSetChanged();在add()方法中已经包含了更新操作
break;
case TYPE_SPORT:
addSportNews();
mPrimitiveAdapter.remove(POS_SPORT);
mPrimitiveAdapter.add(POS_SPORT, mSportPrimitive);
break;
}
}
private void addEntertainNews() {
if (entertainItems == null) {
entertainItems = new ArrayList<>();
}
EntertainItem entertain = new EntertainItem();
entertain.setId("" + (entertainItems.size() + 1));
entertain.setUrl("https://img8.ccnxs.cn/uploadfile/hbase/201901/0129/HBC5C4FABFA51855.png");
entertain.setTitle("item " + (entertainItems.size() + 1) + " : " + "胡海泉独自现身丽江 面带微笑任拍照");
entertainItems.add(entertain);
}
3. 数据的删除
删除娱乐新闻的核心代码:
@Override
public void onClickDeleteEntertain(EntertainItem entertain) {
boolean needRefresh = false;
Iterator iterator = entertainItems.iterator();
while (iterator.hasNext()) {
EntertainItem entertainItem = (EntertainItem) iterator.next();
if (entertainItem.getId().equals(entertain.getId())) {
iterator.remove();
needRefresh = true;
break;
}
}
if (needRefresh) {
mPrimitiveAdapter.remove(POS_ENTERTAIN);
mPrimitiveAdapter.add(POS_ENTERTAIN, mEntertainPrimitive);
}
}
4. 数据的更新
更新娱乐新闻的核心代码:
点击娱乐新闻跳转到编辑页进行编辑,编辑成功返回MainActivity,在onActivityResult()方法中进行处理,实际上娱乐新闻和体育新闻的编辑是不应该放在同一个编辑页的,我这里图方便就写在一起了。
@Override
public void onClickEditEntertain(EntertainItem entertain) {
Intent intent = new Intent(this, EditActivity.class);
intent.putExtra("type", TYPE_ENTERTAIN);
intent.putExtra("obj", entertain);
startActivityForResult(intent, 11);
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
switch (resultCode) {
case 100:
String type = data.getStringExtra("type");
if (MainActivity.TYPE_ENTERTAIN.equals(type)) {
updateEntertainItem((EntertainItem) data.getSerializableExtra("obj"));
} else {
updateSportItem((SportItem) data.getSerializableExtra("obj"));
}
break;
}
}
private void updateEntertainItem(EntertainItem item) {
boolean needRefresh = false;
Iterator iterator = entertainItems.iterator();
while (iterator.hasNext()) {
EntertainItem entertainItem = (EntertainItem) iterator.next();
if (entertainItem.getId().equals(item.getId())) {
entertainItem.setTitle(item.getTitle());
needRefresh = true;
break;
}
}
if (needRefresh) {
mPrimitiveAdapter.remove(POS_ENTERTAIN);
mPrimitiveAdapter.add(POS_ENTERTAIN, mEntertainPrimitive);
}
}
5. 注意点
· binderIndex
看一下娱乐新闻内容EntertainBinder中的关键代码
@Override
public void bind(@NonNull EntertainPrimitive model, @NonNull EntertainViewHolder holder, @NonNull List<GraywaterAdapter.Binder<? super EntertainPrimitive, ? extends EntertainViewHolder>>
binders, int binderIndex, @NonNull GraywaterAdapter.ActionListener<EntertainPrimitive, EntertainViewHolder> actionListener) {
EntertainItem entertain = model.getEntertainItems().get(binderIndex - 1); //因为第一个数据是新闻标题,所以新闻内容数据获取时要减一
Picasso.get().load(entertain.getUrl()).placeholder(R.mipmap.ic_launcher).into(holder.getImg());
holder.getTitle().setText(entertain.getTitle());
holder.getmActionListenerDelegate().update(actionListener, model, holder, binders, binderIndex, null);
holder.getMainLayoutView().setOnClickListener(holder.getmActionListenerDelegate());
holder.getDelete().setOnClickListener(holder.getmActionListenerDelegate());
}
注意这句代码:
EntertainItem entertain = model.getEntertainItems().get(binderIndex - 1);
有没有疑惑为什么要binderIndex要减1?上文也提到EntertainItemBinder是对标题和内容统一进行管理,而binderIndex返回的是binder的下标。看一下断点截图就明白了:
断点.png因为包含了标题TitleBinder,所以第一条新闻内容对应的binderIndex对应的下标是1,但新闻内容的数据集合与新闻标题的数据是相互独立,没有关系的,所以第一条新闻内容在数据集合中的位置是0,所以binderIndex需要减1。
补充:
如果对你有帮助的话,点赞、评论、赞赏都是对我的鼓励,也是支持我写下去的动力,谢谢!
网友评论