A set of Widgets to paginate through data from any data source!
The most basic Widget in this package contains the following features.
- Select your own List Widget (
ListView.builder
,AnimatableIndexedWidgetBuilder
, your custom built Widget, etc.) - Use your own Widgets for each item
- Work with item data, not indexes
- Easily integrates with your API or provide pagination logic for local data
- Get notified when the list rebuilds or items are received
- Replaceable default Widgets for empty, loading, and error states
- Automatic item loader while new data loads
- Debuggable print statements
- Specify your chunk limit (how many items you want back at a time)
- Specify when a new chunk gets requested by setting a threshold
- Insert items into both sides of the list (
listStartChangeStream
)
Sometimes you just want to know item came before, and after your current item. With the Paginated Comparator, this is all done for you!
Note: the first item's previous
value will be the same as the current
, and the last item's next
will be the same as current
.
- All the same features as the Paginated Builder
- Access previous, current and next items in the current item's builder
Data chunking is the mechanism used to paginate. Provide a limit (defaults to 50) and watch as chunks roll in!
Note: pagination metadata can be accessed directly through a GlobalKey
(see examples)
Paginated Builder | Paginated Comparator |
---|---|
Basic error widgets are provided, but it's highly suggested to implement your own custom error widgets.
Item Error | Page Error |
---|---|
Adaptive defaults are provided, but it's highly suggested to implement your own custom loading widgets.
Item Loader | Page Loader |
---|---|
Custom loaders in the examples are courtesy of the shimmer example package.
A very basic default is provided, but it's highly suggested to implement your own custom empty widget.
Install the latest version of the package with flutter pub add paginated_builder
Use the PaginatedBuilder
or PaginatedComparator
and provide the required arguments.
This is going to be the most common use case for this package. See the extended examples in the repo for an example using bloc
and the JSON placeholder API.
Your API must support pagination for this to work. Integration will be specific to your use case.
This code is not specific to this package
import 'dart:convert';
import 'package:http/http.dart' as http;
// ...
This method gets the next chunk of data from the data source
Future<List<Post>> fetchPosts(int? cursor, int limit) async {
final startIndex = cursor ?? 0;
final response = await httpClient.get(
Uri.https(
'jsonplaceholder.typicode.com',
'/posts',
<String, String>{'_start': '$startIndex', '_limit': '$limit'},
),
);
if (response.statusCode == 200) {
final body = json.decode(response.body) as List;
return body.map((dynamic json) {
final map = json as Map<String, dynamic>;
return Post.fromJson(map);
}).toList();
}
throw Exception('error fetching posts');
}
class PostsList extends StatelessWidget {
const PostsList({super.key});
@override
Widget build(BuildContext context) {
final bloc = context.read<PostBloc>();
return PaginatedBuilder<Post, int>(
// * Required when using `ListView.builder` Widget or new items won't
// show up
rebuildListWhenChunkIsCached: true,
// Required arguments
listBuilder: (initialItemCount, paginatedItemBuilder) {
return ListView.builder(
itemBuilder: paginatedItemBuilder,
itemCount: initialItemCount,
);
},
itemBuilder: (context, data, [animation]) {
return PostListItem(post: data.item);
},
dataChunker: bloc.fetchPosts, // Call the method from above
// Optional arguments
emptyWidget: const Center(child: Text('no posts')),
cursorSelector: (Post post) => post.id,
);
}
}
This example works well when you would like to have pagination from a design perspective, but have the entire list on hand.
Refer to the full example app under the example
directory in the repo.
First we'll generate some data to show.
// As an example, we'll generate a bunch of fake posts
final allPosts = List.generate(itemCount, (index) {
final location = index + 1;
// Post is defined as a model elsewhere (see example app in repo)
return Post(
id: location,
title: 'post $location',
body: 'post body',
);
});
Future<List<Post>> _handleGetNext(Post? cursor, int limit) async {
// If the cursor is null it means there was no previous chunk
final isFirstRun = cursor == null;
final data = isFirstRun
// starting at the beginning of the list, get the maximum # of items
? widget.allPosts.take(limit)
// otherwise, skip the ones we've already returned and get # of items
: widget.allPosts
.skipWhile((post) => post != cursor)
.skip(1) // Start after the previous cursor
.take(limit);
// Adds artificial network delay to show item loading widget
return Future.delayed(const Duration(seconds: 1), data.toList);
}
PaginatedComparator<Post, Post>(
dataChunker: _handleGetNext, // Defined above!
listBuilder: _listBuilder, // Defined below!
itemBuilder: _itemBuilder, // Defined below!
// Required when using a List widget that doesn't allow
// item insertion
rebuildListWhenChunkIsCached: true,
)
///Controls what Widget is used to display the items being paginated through
Widget _listBuilder(
int? initialItemCount,
NullableIndexedWidgetBuilder paginatedItemBuilder,
) {
return ListView.builder(
itemCount: initialItemCount,
itemBuilder: paginatedItemBuilder,
);
}
/// Creates each item shown in the list
///
/// Called for each item in the list. This will most likely be called multiple
/// times for each item because the list we're using in the [_listBuilder]
/// will remove items as they're scrolled off the screen and recreate them as
/// they are scrolled back into view.
Widget _itemBuilder(
BuildContext context,
ItemComparator<Post> comparator, [
Animation<double>? animation,
]) {
// A local function used below to create columns for each item
Widget toColumn(Post post, String position) {
return Column(
mainAxisSize: MainAxisSize.min,
key: Key('${post.id}_$position'),
children: [
Text(
position,
style: Theme.of(context).textTheme.bodySmall,
),
Text(
post.title,
style: Theme.of(context).textTheme.titleLarge,
)
],
);
}
// Show a Card Widget (provided by Flutter) for each fake Post
return Card(
margin: const EdgeInsets.all(12),
child: Padding(
padding: const EdgeInsets.all(24),
child: Column(
children: [
Text(
'Comparator ${comparator.currentItem.index + 1}',
style: Theme.of(context).textTheme.headlineSmall,
),
const SizedBox(height: 24),
Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [
toColumn(comparator.previous, 'previous'),
toColumn(comparator.current, 'current'),
toColumn(comparator.next, 'next'),
],
),
],
),
),
);
}
The function used to generate the Widget containing all available items.
Commonly a ListView.builder
Widget will be returned as you can directly replace ListView
s required itemBuilder
argument with the provided paginatedItemBuilder parameter.
Warning: ensure you set the
rebuildListWhenChunkIsCached
totrue
if using a Widget that doesn't allow explicit insertion into the list. akaListView
The AnimatableIndexedWidgetBuilder
is the paginated item builder provided by this widget. Use it as a direct replacement for any regular or animated itemBuilder. This is the preferred implementation method since the entire list does not need to be rebuilt when a new chunk is received.
The item builder is the same callback used with [ListView.builder] with one exception. Normally you receive an index, whereas with item builders you receive the item from the index.
With this, you receive your converted item at that index instead.
This item is retrieved from the in-memory cache located in the [PaginatedBuilderState.cachedItems] property of the State class.
With this you receive converted items instead of an index. The converted items are the previous item, the current item, and the next item.
Items are retrieved from the in-memory cache located in the [PaginatedComparatorState.cachedItems] property of the State class.
Called to retrieve the next n
number of items from your data source.
Use the provided cursor
and limit
to skip and get the next 'n' number of items. The cursor will be the identifier selected using the cursorSelector
from the last time a chunk was retrieved.
If the cursor
is null
, this is the first time the method is being run for this data source.
The limit
is the maximum amount of items the method expects to receive when being invoked.
Warning: To avoid duplicate items, ensure you're getting the
limit
number of items AFTER thecursor
.
The default value used to define how far the user can scroll before the next chunk of data is retrieved.
The stream listened to once the initial page load happens.
When items are added to this stream, they will be added to the beginning of the cache and onItemReceived
will be called with a zero index.
Invoked when data from a new chunk is received
The callback will be called for every item received in each chunk
Also supports updating and deleting items. Use the SnapshotState parameter to decide how to handle the item received.
Invoked when the list rebuilds
The callback will be called for every rebuild of the list
Used to limit the amount of data returned with each chunk
Whether to enable print statements or not
Normally set to use kIsDebug
so logs are printed while you're working, but not in production. This value can be set explicitly.
Whether to recreate the the Widget provided in the listBuilder
after a change comes through on the listStartChangeStream
.
Whether to recreate the Widget provided in the listBuilder
when items from a new chunk is added to the in-memory cache
By default, the list created by the listBuilder
is only ever built once on initialization. Every time the list is re-built, all items need to be recreated using the item builder. Therefore, it is recommended to use a list that allows you to add in the items as they come in through the onItemReceived
callback.
However, when using a standard ListView
, there is no mechanism to insert items into the list without rebuilding the entire list. Because of this, you can set this value to true
and the list will re-initialize with all of the cached items retrieved so far.
It's recommended to use a AnimatedList
to insert and removes items from the state using a GlobalKey
or the static of
method (see AnimatedList's doc comments for details).
Whether to replace the last item in the list with a loading Widget when a new chunk is being retrieved.
Defaults to true
Used to select the value to passed into the dataChunker
the next time it's called.
The cursor will come from the last item in the data returned by the dataChunker
. The cursor should be used to skip any records previously retrieved before getting the next n
records.
n
being the number of records specified by the limit provided to the dataChunker
callback.