web-dev-qa-db-ja.com

FirestoreをAndroidでページ分割する方法は?

私は、Firestoreのページネーションについてのインターネット上のすべての記事(stackoverflow)を読んでいますが、運はありません。ドキュメントに正確なコードを実装しようとしましたが、何も起こりません。アイテム(1250以上)を含む基本的なデータベースがあり、それらを段階的に取得したいです。スクロールして15個のアイテムを(データベースの最後のアイテムまで)ロードします。

ドキュメントコードを使用する場合:

// Construct query for first 25 cities, ordered by population
Query first = db.collection("cities")
    .orderBy("population")
    .limit(25);

first.get()
    .addOnSuccessListener(new OnSuccessListener<QuerySnapshot>() {
    @Override
    public void onSuccess(QuerySnapshot documentSnapshots) {
        // ...

        // Get the last visible document
        DocumentSnapshot lastVisible = documentSnapshots.getDocuments()
            .get(documentSnapshots.size() -1);

        // Construct a new query starting at this document,
        // get the next 25 cities.
        Query next = db.collection("cities")
            .orderBy("population")
            .startAfter(lastVisible)
            .limit(25);

        // Use the query for pagination
        // ...
    }
});

実行する方法?ドキュメントにはあまり多くの詳細がありません。

PS:ユーザーがスクロールするときに、リストビューではなくリサイクラビューが必要です。ありがとう

15
Johans Bormman

公式ドキュメント で述べられているように、この問題を解決するための鍵は startAfter() メソッドを使用することです。したがって、クエリカーソルをlimit()メソッドと組み合わせることにより、クエリをページ分割できます。バッチ内の最後のドキュメントを次のバッチのカーソルの開始点として使用できます。

このページネーションの問題を解決するには、このpostからの回答をご覧ください。ステップごとに、Cloud Firestoreデータベースからデータを小さなチャンクでロードし、ListViewボタンクリックで表示する方法。

解決策:

Firestoreデータベースからデータを取得し、RecyclerViewの小さなチャンクで表示するには、以下の手順に従ってください。

製品を使用した上記の例を見てみましょう。製品、都市、または必要なものを使用できます。原則は同じです。ユーザーがスクロールしたときにさらに製品をロードしたい場合、RecyclerView.OnScrollListenerを使用します。

最初にRecyclerViewを定義し、レイアウトマネージャーをLinearLayoutManagerに設定してリストを作成しましょう。また、空のリストを使用してアダプターをインスタンス化し、アダプターをRecyclerViewに設定します。

RecyclerView recyclerView = findViewById(R.id.recycler_view);
recyclerView.setLayoutManager(new LinearLayoutManager(this));
List<ProductModel> list = new ArrayList<>();
ProductAdapter productAdapter = new ProductAdapter(list);
recyclerView.setAdapter(productAdapter);

次のようなデータベース構造があると仮定します。

Firestore-root
   |
   --- products (collection)
         |
         --- productId (document)
                |
                --- productName: "Product Name"

そして、次のようなモデルクラス:

public class ProductModel {
    private String productName;

    public ProductModel() {}

    public ProductModel(String productName) {this.productName = productName;}

    public String getProductName() {return productName;}
}

アダプタークラスは次のようになります。

private class ProductAdapter extends RecyclerView.Adapter<ProductViewHolder> {
    private List<ProductModel> list;

    ProductAdapter(List<ProductModel> list) {
        this.list = list;
    }

    @NonNull
    @Override
    public ProductViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
        View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_product, parent, false);
        return new ProductViewHolder(view);
    }

    @Override
    public void onBindViewHolder(@NonNull ProductViewHolder productViewHolder, int position) {
        String productName = list.get(position).getProductName();
        productViewHolder.setProductName(productName);
    }

    @Override
    public int getItemCount() {
        return list.size();
    }
}

item_productレイアウトには、TextViewという1つのビューのみが含まれます。

<TextView
    Android:layout_width="wrap_content"
    Android:layout_height="wrap_content"
    Android:id="@+id/text_view"
    Android:textSize="25sp"/>

そして、これがホルダークラスの外観です:

private class ProductViewHolder extends RecyclerView.ViewHolder {
    private View view;

    ProductViewHolder(View itemView) {
        super(itemView);
        view = itemView;
    }

    void setProductName(String productName) {
        TextView textView = view.findViewById(R.id.text_view);
        textView.setText(productName);
    }
}

次に、制限をグローバル変数として定義し、15に設定します。

private int limit = 15;

この制限を使用して、クエリを定義しましょう。

FirebaseFirestore rootRef = FirebaseFirestore.getInstance();
CollectionReference productsRef = rootRef.collection("products");
Query query = productsRef.orderBy("productName", Query.Direction.ASCENDING).limit(limit);

あなたの場合にも魔法をかけるコードは次のとおりです。

query.get().addOnCompleteListener(new OnCompleteListener<QuerySnapshot>() {
    @Override
    public void onComplete(@NonNull Task<QuerySnapshot> task) {
        if (task.isSuccessful()) {
            for (DocumentSnapshot document : task.getResult()) {
                ProductModel productModel = document.toObject(ProductModel.class);
                list.add(productModel);
            }
            productAdapter.notifyDataSetChanged();
            lastVisible = task.getResult().getDocuments().get(task.getResult().size() - 1);

            RecyclerView.OnScrollListener onScrollListener = new RecyclerView.OnScrollListener() {
                @Override
                public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
                    super.onScrollStateChanged(recyclerView, newState);
                    if (newState == AbsListView.OnScrollListener.SCROLL_STATE_TOUCH_SCROLL) {
                        isScrolling = true;
                    }
                }

                @Override
                public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
                    super.onScrolled(recyclerView, dx, dy);

                    LinearLayoutManager linearLayoutManager = ((LinearLayoutManager) recyclerView.getLayoutManager());
                    int firstVisibleItemPosition = linearLayoutManager.findFirstVisibleItemPosition();
                    int visibleItemCount = linearLayoutManager.getChildCount();
                    int totalItemCount = linearLayoutManager.getItemCount();

                    if (isScrolling && (firstVisibleItemPosition + visibleItemCount == totalItemCount) && !isLastItemReached) {
                        isScrolling = false;
                        Query nextQuery = productsRef.orderBy("productName", Query.Direction.ASCENDING).startAfter(lastVisible).limit(limit);
                        nextQuery.get().addOnCompleteListener(new OnCompleteListener<QuerySnapshot>() {
                            @Override
                            public void onComplete(@NonNull Task<QuerySnapshot> t) {
                                if (t.isSuccessful()) {
                                    for (DocumentSnapshot d : t.getResult()) {
                                        ProductModel productModel = d.toObject(ProductModel.class);
                                        list.add(productModel);
                                    }
                                    productAdapter.notifyDataSetChanged();
                                    lastVisible = t.getResult().getDocuments().get(t.getResult().size() - 1);

                                    if (t.getResult().size() < limit) {
                                        isLastItemReached = true;
                                    }
                                }
                            }
                        });
                    }
                }
            };
            recyclerView.addOnScrollListener(onScrollListener);
        }
    }
});

lastVisibleは、クエリの最後の表示項目を表すDocumentSnapshotオブジェクトです。この場合、15番目ごとに、gloabl変数として宣言されます。

private DocumentSnapshot lastVisible;

また、isScrollingisLastItemReachedもグローバル変数であり、次のように宣言されます。

private boolean isScrolling = false;
private boolean isLastItemReached = false;

リアルタイムでデータを取得したい場合は、get()呼び出しを使用する代わりに、コレクション内の複数のドキュメントの リスニングに関する公式ドキュメントで説明されているaddSnapshotListener()を使用する必要があります。

23
Alex Mamo

FirebaseUI-Androidも最近Firestore Paginatorを発表しました。

私はコードでそれを使用しましたが、うまく機能します-.addSnapshotListener()の代わりに.get()を使用して動作するため、リサイクラはリアルタイムではありません。

こちらのドキュメントをご覧ください。

https://github.com/firebase/FirebaseUI-Android/tree/master/firestore#using-the-firestorepagingadapter

5
Jeff Padgett