通知 RecyclerView 中 Firestore 数据的变化

Notify changes of Firestore data in RecyclerView

在我的应用程序中有一个片段,其中包含一个可扩展的 RecyclerView 从 firebase firestore 获取数据,其中的每个项目都有一个按钮,单击该按钮会更改 Firestore 中的一些数据,以某种方式影响项目。 现在的问题是:当 运行 应用程序并单击按钮时,数据不会更改,直到我刷新片段..

fragment.java:

View root = inflater.inflate(R.layout.fragment_dashboard, container, false);
final String UserId = currentUser.getUid();

final RecyclerView recyclerView = root.findViewById(R.id.recyclerview);
recyclerView.setLayoutManager(new LinearLayoutManager(getContext()));

db.collectionGroup("Id").whereEqualTo("id", UserId).get()
  .addOnCompleteListener(new OnCompleteListener < QuerySnapshot > () {
    @Override
    public void onComplete(@NonNull Task < QuerySnapshot > task) {
        final List < Company > Parent = new ArrayList < > ();

        for (QueryDocumentSnapshot document: task.getResult()) {
            final List < String > grades = (List < String > ) document.get("tagsG");

            CollectionReference collectionReference =
                document.getReference().getParent();
            final List < Product > Child = new ArrayList < > ();
            for (int i = 0; i < grades.size(); i++) {
                final String gg = grades.get(i);
                collectionReference.document(UserId).collection(gg).whereEqualTo("approvedStd",0).get()
                    .addOnCompleteListener(new OnCompleteListener < QuerySnapshot > () {
                        @Override
                        public void onComplete(@NonNull Task < QuerySnapshot > task) {
                            for (QueryDocumentSnapshot document: task.getResult()) {
                                String name = document.getString("std_name");
                                String sub = document.getString("std_subject");
                                String region = document.getString("std_region");
                                Child.add(new Product(name, sub, region, gg));
                            }
                        }
                    });
            }
            Parent.add(new Company("new", Child));
            productAdapter = new ProductAdapter(Parent);
            recyclerView.setAdapter(productAdapter);
        }
    }
});
return root;

ViewHolder.java

public class ProductViewHolder extends ChildViewHolder {
    private FirebaseFirestore db = FirebaseFirestore.getInstance();
    private FirebaseAuth mAuth = FirebaseAuth.getInstance();
    private FirebaseUser currentUser = mAuth.getCurrentUser();
    private String UserId = currentUser.getUid();
    private TextView name;
    private TextView sTextView;
    private TextView region;
    private Button addButton;
    private String phoneOfstd;
    private String grade;
    private ProductAdapter productAdapter;

    public ProductViewHolder(View itemView) {
        super(itemView);
        name = itemView.findViewById(R.id.name);
        sTextView = itemView.findViewById(R.id.sub);
        region = itemView.findViewById(R.id.region);
    }

    public void bind(Product product) {
        name.setText(product.name);
        sTextView.setText(product.sub);
        region.setText(product.region);
        grade = product.grade;

        addButton.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {

                CollectionReference questionsRef = db.collection("Teachers checked").document("courses").collection("Id");
                DocumentReference docRef = questionsRef.document(UserId).collection(grade).document(phoneOfstd);
                docRef.update("approvedStd", 1);
                productAdapter.notifyDataSetChanged();
            }
        });
    }

}

ProductAdapter.java

public class ProductAdapter extends
 ExpandableRecyclerViewAdapter < CompanyViewHolder, ProductViewHolder > {
    public ProductAdapter(List < ? extends ExpandableGroup > groups) {
        super(groups);
    }

    @Override
    public CompanyViewHolder onCreateGroupViewHolder(ViewGroup parent, int viewType) {
        View v = LayoutInflater.from(parent.getContext()).inflate(R.layout.expandable_recyclerview_company, parent, false);
        return new CompanyViewHolder(v);
    }

    @Override
    public ProductViewHolder onCreateChildViewHolder(ViewGroup parent, int viewType) {
        View v = LayoutInflater.from(parent.getContext()).inflate(R.layout.expandable_recyclerview_product, parent, false);
        return new ProductViewHolder(v);
    }

    @Override
    public void onBindChildViewHolder(ProductViewHolder holder, int flatPosition, ExpandableGroup group, int childIndex) {
        final Product product = (Product) group.getItems().get(childIndex);
        holder.bind(product);
    }

    @Override
    public void onBindGroupViewHolder(CompanyViewHolder holder, int flatPosition, ExpandableGroup group) {
        final Company company = (Company) group;
        holder.bind(company);
    }
}

编辑:

我试图在 recyclerView.setAdapter(productAdapter); 之后的片段中写 productAdapter.notifyDataSetChanged();但是应用程序崩溃并且 logcat 显示此错误:

FATAL EXCEPTION: main java.lang.NullPointerException: Attempt to invoke virtual method 'void com.example.exercise.ExpandableRecyclerView.ProductAdapter. notifyDataSetChanged()' on a null object reference

你可以在你的适配器中放置一个读取你的数据的函数来确保你的数据被更新。我在这里看不到你的适配器的代码,但试试这个,让我知道它是否已解决。

这里可能是你的错误docRef.update("approvedStd", 1); productAdapter.notifyDataSetChanged();

您更新了 firestore 中的对象,但您没有实施 EventListener 以在发生任何更改后检索数据,因此您必须使用选项一来更新 arrayList 中的此对象并调用 notifydatasetChanged 或添加 Snapshotlistener 以在执行任何更改后检索数据

首先,您的 productAdapter 等于 null 因为您没有为其分配任何值,您应该将对适配器的引用传递给视图持有者。

其次,主要问题是您需要在单击项目中的视图时更新视图。

1- 在绑定方法中绑定视图时处理更改。

2- 当您单击 btn 或视图时,您应该根据需要显示的更改来更新对象。

3- 如果更改影响当前项目只使用

notifyItemChanged(position, object)

如果它对您应该使用的许多项目有影响

notifyDataSetChanged()

例如: 如果你有复选框,你需要让它在你的回收站中工作。

1-在bind方法中绑定checkbox

checkbox.setChecked(product.isSelected());

2- 在选中的更改事件中,我们将更新产品对象

product.setSelected(isChecked);

3- 在检查更改事件结束时

notifyItemChanged(getAdapterPosition(), product);