我正试图显示从DBFlow(sqlite)后端馈送的recyclerview列表。列表中的每个项都表示一个项,一旦选定,该项将转换为详细活动视图。与任何cursorloader一样,我希望每次在支持视图的加载程序中发生更改时,布局都会自我更新。
在棉花糖中进行测试时,这一点是正确的,但在棒棒糖和果冻豆(4.3)中均未能做到。在失败的情况下,尽管在调试器中看到onBindViewHolder()为每个项调用一次(适当的数据到达游标中,顺便说一句),但列表中仍会显示一个项。
以下是我观点中涉及的主要内容。我故意简化了代码,如果认为有必要,可以要求更多的细节:
显示列表的活动中的onResume()重写:
@Override
protected void onResume() {
super.onResume();
getSupportLoaderManager().initLoader(0, null, new LoaderManager.LoaderCallbacks<Cursor>() {
@Override
public Loader<Cursor> onCreateLoader(int arg0, Bundle cursor) {
return BackendManager.getListCursorLoader(mContext);
}
@Override
public void onLoadFinished(Loader<Cursor> arg0, Cursor cursor) {
int numItemsRetrieved = cursor.getCount();
if ((numItemsRetrieved == 1) {
Optional<Item> selectedItemOpt = BackendManager.getSingleItem();
if (selectedItemOpt.isPresent()) {
Item selectedItem = selectedItemOpt.get();
Intent t = new Intent(mContext, ItemDetailActivity.class);
Bundle b = new Bundle();
b.putSerializable("item", selectedItem);
it.putExtras(b);
startActivityForResult(it, SHOW_ITEM_DETAIL);
}
}
((ItemListRecyclerAdapter) mRecyclerListView.getAdapter()).swapCursor(cursor);
}
@Override
public void onLoaderReset(Loader<Cursor> arg0) {
((ItemListRecyclerAdapter) mRecyclerListView.getAdapter()).swapCursor(null);
}
});
}
静态管理器中的游标加载程序生成:
public static CursorLoader getListCursorLoader(Context ctx) {
return new CursorLoader(
ctx, // Context
Item.CONTENT_URI, // Uri
null, // Projection
null, // Selection
null, // selectionArgs
null // sortOrder
);
}
游标的RecyclerView适配器。当前正在使用skyfish's:
public class ItemListRecyclerAdapter extends CursorRecyclerViewAdapter<ItemListRecyclerAdapter.ViewHolder> {
private Context mContext;
public ItemListRecyclerAdapter(Context context, Cursor cursor) {
super(context, cursor);
mContext = context;
}
public static class ViewHolder extends RecyclerView.ViewHolder {
public TextView itemText;
public ImageView itemImage;
public ViewHolder(View view) {
super(view);
itemText = (TextView) view.findViewById(R.id.itemTitle);
itemImage = (ImageView) view.findViewById(R.id.itemPicture);
}
}
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View itemView = LayoutInflater.from(parent.getContext())
.inflate(R.layout.activity_6_list_item, parent, false);
ViewHolder vh = new ViewHolder(itemView);
return vh;
}
@Override
public void onBindViewHolder(ViewHolder viewHolder, Cursor cursor) {
// Find fields to populate in inflated template
String itemTextData = cursor.getString(cursor.getColumnIndex("itemText"));
viewHolder.itemText.setText(itemTextData);
// Force text fit
AutofitHelper afh = AutofitHelper.create(viewHolder.itemText);
afh.setMaxTextSize(TypedValue.COMPLEX_UNIT_SP, 22.0f);
String itemImageUrl = cursor.getString(cursor.getColumnIndex("logo"));
if ((itemImageUrl != null) && (!itemImageUrl.isEmpty())) {
Picasso.with(mContext)
.load(itemImageUrl)
.resize(300, 300)
.centerCrop()
.placeholder(R.drawable.placeholder)
.into(viewHolder.itemImage);
} else {
viewHolder.itemImage.setImageDrawable(ContextCompat.getDrawable(mContext, R.drawable.placeholder));
}
}
}
如上所述,这已通过以下测试和验证:
Nexus6上的棉花糖(6.0.1):工作
银河系s5上的棒棒糖(5.0):不起作用
Galaxy S3上的果冻豆(4.3):不起作用
我是不是丢了什么东西?有什么线索可以进一步了解吗?
最佳答案
既然没有任何答案,我会解释我最终是怎么解决这个问题的。
在完全重写之后,我去掉了skyfish的光标适配器,完全使用android的recyclerview接口。与使用游标/适配器方案从sqlite后端提取数据并在对底层存储所做的更改时交换游标不同,我只是在存储管理器中启用了get all items静态方法来检索最新的数据。为了在数据更改时更新列表,我使用dbflow的Observables来获得此类更改的通知。一旦检测到更改,我就使用这样的get all items检索更新后的数据,并相应地更新recyclerview,从而执行适当的notifyitem()调用来设置布局更改的动画。
不需要android&dbflow之外的附加依赖项或代码。这种方法已经在原始问题中描述的所有设备中得到了验证。