Уведомлять об изменениях данных Firestore в RecyclerView - PullRequest
0 голосов
/ 23 марта 2020

В моем Приложении есть фрагмент, содержащий Expandable RecyclerView, который получает данные из Firebase Firestore, у каждого элемента в нем есть кнопка, которая при нажатии изменяет некоторые данные в Firestore, которые так или иначе влияют на элементы. Теперь проблема в том, что при запуске приложения и нажатии кнопки данные не изменяются, пока я не перефразирую sh фрагмент ..

фрагмент. 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);
    }
}

РЕДАКТИРОВАНИЕ:

Я пытался написать productAdapter .notifyDataSetChanged (); во фрагменте после recyclerView.setAdapter (productAdapter); , но приложение вылетает, и logcat показывает эту ошибку:

ИСКЛЮЧИТЕЛЬНОЕ ИСКЛЮЧЕНИЕ: main java .lang.NullPointerException: попытка вызвать виртуальный метод void com.example.exercise.ExpandableRecyclerView.ProductAdapter. notifyDataSetChanged () 'для ссылки на пустой объект

Ответы [ 3 ]

1 голос
/ 25 марта 2020

может быть вашей ошибкой здесь docRef.update("approvedStd", 1); productAdapter.notifyDataSetChanged();

вы обновляете объект в firestore, но вы не реализуете EventListener для извлечения данных после любого изменения, поэтому у вас есть вариант буксировки один для обновления этого объекта в arrayList и вызовите notifydatasetChanged или добавьте Snapshotlistener для извлечения данных после любого выполненного изменения

1 голос
/ 25 марта 2020

Во-первых, ваш productAdapter равен null, поскольку вы не присвоили ему никакого значения, вы должны передать ссылку на ваш адаптер в держатель представления.

Во-вторых, основная проблема заключается в вас необходимо обновить представление при щелчке на представлении в элементе.

1 - обрабатывать изменения при привязке представления в методе связывания.

2- при нажатии на btn или на представление Вы должны обновить объект, изменив его, чтобы отобразить.

3 - если изменения влияют на текущий элемент, используйте

notifyItemChanged(position, object)

, если он влияет на многие элементы, которые вы должны использовать

notifyDataSetChanged()

Пример: если у вас есть флажок, и вам нужно, чтобы он работал в утилизаторе.

1 - привязать флажок в методе связывания

checkbox.setChecked(product.isSelected());

2 - при проверенном событии изменения мы обновим объект продукта

product.setSelected(isChecked);

3- в конце проверенного события изменения

notifyItemChanged(getAdapterPosition(), product);
0 голосов
/ 23 марта 2020

U может поместить функцию, которая читает ваши данные в вашем адаптере, чтобы убедиться, что ваши данные обновляются. Я не вижу код вашего адаптера здесь, но попробуйте это и дайте мне знать, если он решен или нет.

...