我正在尝试在React Native应用中集成Firebase支持的RecyclerView。使用硬编码数据可以很好地工作,但是在插入动态加载的行并在RecyclerView.Adapter上调用notifyItemInserted()或notifyDataSetChanged()时,RecyclerView本身并不反映这一变化。这将显示为初始空白视图,直到点击应用程序的刷新按钮,其唯一效果是从React Native端重新渲染RecyclerView,或直到滚动或屏幕方向更改为止。
在阅读本网站上的十几个或更多问题后,关于不涉及React Native的类似问题并尝试了大多数常见解决方案,我怀疑这个问题与React Native有关。
鉴于React Native的列表视图实现存在性能问题,找到解决方案可能对许多人有所帮助。
public class PostsAdapter extends RecyclerView.Adapter<PostsAdapter.ViewHolder> {
ArrayList<Post> mDataset;
public PostsAdapter(ArrayList<Post> list){
mDataset = list;
}
public static class ViewHolder extends RecyclerView.ViewHolder {
public LinearLayout mPostView;
public ViewHolder(LinearLayout v) {
super(v);
mPostView = v;
}
}
@Override
public PostsAdapter.ViewHolder onCreateViewHolder(ViewGroup parent,
int viewType) {
LinearLayout v = (LinearLayout) LayoutInflater.from(parent.getContext())
.inflate(R.layout.post, parent, false);
ViewHolder vh = new ViewHolder(v);
return vh;
}
@Override
public void onBindViewHolder(ViewHolder holder, int position) {
TextView tv = (TextView) holder.mPostView.findViewById(R.id.title);
tv.setText(mDataset.get(position).title);
}
@Override
public int getItemCount() {
return mDataset.size();
}
}
public class RNPostsViewManager extends SimpleViewManager{
private ArrayList<Post> mDataset = new ArrayList<>();
public static final String REACT_CLASS = "AndroidPostsView";
private RecyclerView mRecyclerView;
private PostsAdapter mAdapter;
private RecyclerView.LayoutManager mLayoutManager;
private DatabaseReference mDatabase;
private Query initialDataQuery;
private ChildEventListener initialDataListener;
@Override
public String getName() {
return REACT_CLASS;
}
@UiThread
public void addPost (Post p){
mDataset.add(p);
mAdapter.notifyItemInserted(mDataset.size()-1);
}
@Override
public RecyclerView createViewInstance( ThemedReactContext context) {
mAdapter = new PostsAdapter(mDataset);
mRecyclerView = new RecyclerView(context){
@Override
protected void onAttachedToWindow() {
super.onAttachedToWindow();
initialDataQuery.addChildEventListener(initialDataListener);
}
};
mLayoutManager = new LinearLayoutManager(context.getCurrentActivity(), LinearLayoutManager.VERTICAL, false);
DividerItemDecoration mDecoration = new DividerItemDecoration(context, 1);
mDecoration.setDrawable(ContextCompat.getDrawable(context.getCurrentActivity(), R.drawable.sep));
mRecyclerView.setLayoutManager(mLayoutManager);
mRecyclerView.addItemDecoration(mDecoration);
mRecyclerView.setItemAnimator(new DefaultItemAnimator());
mRecyclerView.setAdapter(mAdapter);
mDatabase = FirebaseDatabase.getInstance().getReference();
initialDataQuery = mDatabase.child("wp-posts").orderByChild("unixPostDate").limitToFirst(100);
initialDataListener = new ChildEventListener() {
@Override
public void onChildAdded(DataSnapshot dataSnapshot, String previousChildName) {
Post p = dataSnapshot.getValue(Post.class);
addPost(p);
}
@Override
public void onChildChanged(DataSnapshot dataSnapshot, String previousChildName) {
}
@Override
public void onChildRemoved(DataSnapshot dataSnapshot) {
}
@Override
public void onChildMoved(DataSnapshot dataSnapshot, String previousChildName) {
}
@Override
public void onCancelled(DatabaseError databaseError) {
}
};
return mRecyclerView;
}
}
// layout file post.xml
<?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:background="@android:color/background_light"
android:clickable="true"
android:focusable="true"
android:gravity="center_vertical"
android:orientation="horizontal"
android:padding="10dp"
android:weightSum="1">
<Button
android:id="@+id/button"
style="@style/Widget.AppCompat.Button.Small"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginRight="20dp"
android:background="@android:drawable/ic_menu_more"
android:gravity="center_vertical" />
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="fill_vertical"
android:orientation="horizontal"
android:weightSum="1">
<TextView
android:id="@+id/title"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_gravity="fill_vertical"
android:layout_weight="1"
android:gravity="top"
android:text="TextView"
android:textColor="@color/title"
android:textStyle="bold" />
</LinearLayout>
</LinearLayout>
答案 0 :(得分:7)
问题是当PS> "`u{0048}"
H
是本机UI组件时,requestLayout
无效。
以下黑客使所有这些问题消失了:
我现在覆盖RecyclerView
内的requestLayout
方法。
然后在任何RecyclerView
方法,甚至notify*
调用或任何调用重新布局的方法之前,我允许我的自定义scrollToPosition
方法强制重新布局。
最终结果如下:
requestLayout
答案 1 :(得分:0)
ReactNative中的根视图是ReactRootView,onlayout
是一个空方法。
在RecyclerView中调用notifyDatasetChanged
时,它实际上是请求layout
重新传输其子节点。布局方法将调用super.layout
来首先遍历整个视图树。因此,当根视图是ReactRootView时,这不是问题。
您可以手动调用RecyclerView.onlayout(boolean changed, int l, int t, int r, int b)
以触发其子级重新布局,以使notifyDatasetChanged正常工作。