flutter_requery

Creator: coderz1093

Last updated:

0 purchases

TODO
Add to Cart

Description:

flutter requery

Flutter library for fetching, caching and invalidating asynchronous data
Quick Features #

Fetch asynchronous data
Data invalidation
Optimistic response
Reset cache

Motivation #
How to do API calls in Flutter? Probably, majority would answer by using Dio.
But the real question would be, how to integrate API calls in Flutter arhitecture seamless?
One way would be to use FutureBuilder
or maybe Bloc like most of the community do.
The thing is, both FutureBuilder and Bloc have flaws.
For example, FutureBuilder is too simple. It does provide status of your query, but communicating with some FutureBuilder is impossible outside of the scope of the current screen.
Problem with Bloc is that will include so many boilerplate code for a simple API call but again, talking to the Bloc is impossible from other screens. Bloc should be
number one choice for complicated workflows, but what if you don't have any complex business logic?
flutter_requery to the rescue!
Example #
Let's start with this simple example.
/// define data and cache key
final List<String> data = ["Hello"];
final String cacheKey = 'strKey';

/// simulates API call, therefore response is delayed
Future<List<String>> _getData() async {
await Future.delayed(Duration(seconds: 1));
return data;
}

/// later, used in the build method
Query<List<String>>(
cacheKey,
future: _getData,
builder: (context, response) {
if (response.error != null) {
return Text('Error');
}
if (response.loading) {
return Text('Loading...');
}
final children = response.data.map((str) => Text(str)).toList();
return ListView(
children: children
);
},
);

/// and later when you want to invalidate your data
void _onPress() async {
await Future.delayed(Duration(seconds: 1));
data.add("World");
queryCache.invalidateQueries(cacheKey);
}
copied to clipboard
And that's the gist of it. You can find complete example on pub.dev.
Usage #
Cache key #
Every query needs a cache key.
Under this key your data will be stored in cache.
It can be string, int or list of strings and ints.
/// good
const k1 = 'myKey';
const k2 = 1;
const k3 = ["data", 1]

/// bad
const k1 = 1.0;
const k2 = true;
const k3 = ["data", true];
copied to clipboard
Idea behind having keys specified as list is that you can invalidate your queries more intelligently.
Take a look at invalidation chapter for more details.
Query #
Once the cache key is defined, next step is to write the query.
Query takes 3 arguments:

cacheKey - look here for more info.
future - async action which will be executed.
builder - follows the Flutter builder pattern. First parameter is BuildContext followed by QueryResponse object.

QueryResponse manages query status. It also has 3 properties:

data - response received from the future or null if the exception occured.
loading - boolean, true if the query is running. Otherwise, it's false.
error - represents an error received from the future. If the future was successful, error will be null.

// use Query widget in the build method
Query<List<String>>(
'myCacheKey',
future: ()async {
await Future.delayed(Duration(seconds: 1));
return ["Hello"]
}
builder: (context, QueryResponse response) {
/// error state
if (response.error != null) {
return Text('Error');
}
/// loading state
if (response.loading) {
return Text('Loading...');
}
final children = response.data.map((str) => Text(str)).toList();
return ListView(
children: children
);
},
);
copied to clipboard
Invalidation #
Data invalidation can come in two different forms.
You can either afford to wait for the API response or you simply need to show the newest data as soon as possible.
If you are interested in following, check the next chapter.
Waiting for the API response is more common and flutter_requery supports this by using the queryCache instance.
It's global and already defined by the library. Invalidate your query by passing the cache keys.
// invalidates strKey query
queryCache.invalidateQueries('strKey');

// support for bulk invalidation
queryCache.invalidateQueries(['strKey1', 'strKey2']);

// if your keys are lists, end result would be similar to
queryCache.invalidateQueries([
['strKey', 1],
['strKey2', 2]
]);
copied to clipboard
Once query is invalidated, every Query widget subscribed for that query
will execute future again and rebuild its children with the new data.
For cache-level invalidation use:
// invalidate every query stored in cache
queryCache.invalidateAll()
copied to clipboard
Invalidation works in pair with the keys defined as lists.
Cache keys defined as list must be looked upon in a hierarchical manner where the list elements defined before are ancestors of the elements that come after.
For example:
// requests is ancestor of 1
const key1 = ["requests", 1]
copied to clipboard
Reasoning behind this is to support hierarchical invalidation.
Sometimes it can get cumbersome managing invalidations and therefore developer can decide to cleverly name keys to support this.
For example:
const k1 = ["requests", 1]
const k2 = ["requests", 2]
const k3 = "requests"

// without hierarchical invalidation you need to call
queryCache.invalidateQueries([
["requests", 1], ["requests", 2], "requests"
]);

// but with hierarchical invalidation you only need to call
queryCache.invalidateQueries("requests");
copied to clipboard
Optimistic response #
Sometimes waiting period for the API response to be available is too long.
Therefore you can immediately update the cache data and rebuild your widget tree by using the optimistic response.
Make sure to remove await keyword before the API call since this will block the thread.
queryCache.setOptimistic("requests", [...oldData, newData]);
copied to clipboard
Reset #
In short, reset can be explained as cache-level invalidation without rebuilding the widget tree.
Also, async actions won't be ran immediately but only when the new Query widget is mounted or the cacheKey has changed.
This is particularly useful for the log out action.
queryCache.reset();
copied to clipboard
Summary #



API
Description




Query
Flutter widget used for data-fetching operations.


queryCache.invalidateQueries
Invalidates query specified by cache key and rebuilds the widget tree.


queryCache.invalidateAll
Invalidates every query stored in cache and rebuilds the widget tree.


queryCache.setOptimistic
Set cache data manually and rebuild the widget tree.


queryCache.reset
Invalidates every query stored in cache without rebuilding the widget tree.

License

For personal and professional use. You cannot resell or redistribute these repositories in their original state.

Files:

Customer Reviews

There are no reviews.