- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在尝试将 Firebase 支持的 RecyclerView 集成到 React Native 应用程序中。对于硬编码数据,它运行良好,但是在插入动态加载的行并在 RecyclerView.Adapter 上调用 notifyItemInserted() 或 notifyDataSetChanged() 时,RecyclerView 本身不会反射(reflect)更改。这表现为一个初始的空白 View ,直到应用程序的刷新按钮被点击,这具有从 React Native 端重新渲染 RecyclerView 的唯一效果,或者直到滚动或屏幕方向改变。
在阅读了本网站上关于不涉及 React Native 的类似问题的十几个或更多问题并尝试了大多数常见解决方案后,我怀疑这个问题与 React Native 特别相关。
鉴于 React Native 的 ListView 实现的持续性能问题,找到解决方案可能对许多人有所帮助。
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>
最佳答案
问题是当 RecyclerView
是原生 UI 组件时,requestLayout
无法正常工作。
以下 hack 解决了所有这些问题:
我现在覆盖 RecyclerView
中的 requestLayout
方法。然后在任何 notify*
方法,甚至 scrollToPosition
调用或任何调用重新布局的方法之前,我允许我的自定义 requestLayout
方法强制重新布局。
最终结果是这样的:
private boolean mRequestedLayout = false;
public void aMethodThatUpdatesStuff(int indexToUpdate, ReadableMap updatedChild) {
final SPAdapter adapter = (SPAdapter) getAdapter();
mRequestedLayout = false;
adapter.updateDataAtIndex(indexToUpdate, updatedChild); // <-- this runs notifyItemChanged inside
}
@Override
public void requestLayout() {
super.requestLayout();
// We need to intercept this method because if we don't our children will never update
// Check https://stackoverflow.com/questions/49371866/recyclerview-wont-update-child-until-i-scroll
if (!mRequestedLayout) {
mRequestedLayout = true;
this.post(new Runnable() {
@SuppressLint("WrongCall")
@Override
public void run() {
mRequestedLayout = false;
layout(getLeft(), getTop(), getRight(), getBottom());
onLayout(false, getLeft(), getTop(), getRight(), getBottom());
}
});
}
}
关于android - React Native 中的 RecyclerView : notifyItemInserted() and notifyDataSetChanged() have no effect,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44868899/
我正在将我的应用程序从 sqlite 迁移到 Firebase。以前我会从数据库读取 N 项到数组列表并调用 notifyItemRangeInserted。现在从 Firebase 获取数据最方便的
我需要做的是将项目插入到 RecyclerView 的顶部。我有一个适配器,它保留了我的项目的 List,当我想添加到 RecyclerView 的顶部时,我只是使用这段代码: mItems.add(
我有一个 RecyclerView.Adapter 带有 hasStableIds(true); 和一个 LinearLayoutManager 带有 reverseLayout(true); 我向适
我正在使用 RecyclerView,.notifyItemInserted 仅将一项插入 RecyclerView。对于多个项目,我该怎么做?我想避免使用 .notifyDataSetChanged
必须notifyItemRangeChanged调用后被调用notifyItemInserted在 RecyclerView.Adapter 中?SO 上有很多示例,他们总是在 notifyItemI
当我添加一个新项目时,我正在调用方法 public void addItem(ArrayList> arrayList, int position){ this.arrayList=array
如果我们使用这两种方法来告诉适配器您指向的数据已更改,我很困惑,那么它们之间有什么区别。 最佳答案 notifyDataSetChanged() 可以被认为是一个“主要”变化。您是在告诉适配器数据集中
所以出于某种原因,每当我向我的 ArrayList 添加一个新项目并通知 recyclerview 它已被添加时,它会复制前一个条目的数据。因此, View 已创建,但使用的是最后一个条目中的数据,而
出于某种原因,当向 RecyclerView 添加一个新项目时(应该插入到列表的顶部),它不会显示,除非我向下滚动列表并返回到顶部,并且没有任何动画任何一个。 (只是出现在列表的顶部,就好像它一直都在
我正在尝试在回收站 View 中的第一个位置插入一个项目,即 0 索引,如下所示 try { commentList.add(new Comment( Prefer
我有一个要求,我应该在滚动和更新列表的同时下载广告项。由于调用 notifyDatasetChnaged() 会重置所有内容,因此我正在调用 notifyItemInserted(position)。
我正在将我的适配器移植到 RecyclerView.Adapter 我想要达到的目标:当用户在接近尾声时向下滚动时,我想开始获取数据,我还想在最后添加我的 ProgressBar View ,让用户知
我对 Android 还很陌生,目前正在开发我的第一个基于 RecyclerView 的严肃应用程序。当我插入或更新现有行时,我注意到我的应用程序中有一些有问题的行为:即使我在添加或 notifyIt
例如,您有一个适配器,并在 onBindViewHolder 方法中将 OnClickListener 设置为某些 View (并根据 View 位置在那里执行一些操作)。您应该将 final 分配给
最近我将我的 recyclerview-v7:23 升级到 recyclerview-v7:24.2.0。我的应用程序有一个无穷无尽的滚动列表。当我将加载 View 添加到 RecyclerView
我正在使用 data.add(0,item); notifyItemInserted(0); 只要项目空间没有被填满,这就可以正常工作。之后就看不到动画了。 当我这样做时动画效果很好 data.add
我有一个带有水平线性布局管理器的 RecyclerView,声明如下: RecyclerView graph = (RecyclerView) findViewById(R.id.graph); Re
我正在尝试将 Firebase 支持的 RecyclerView 集成到 React Native 应用程序中。对于硬编码数据,它运行良好,但是在插入动态加载的行并在 RecyclerView.Ada
我是一名优秀的程序员,十分优秀!