Skip to content

[RFC] Fixing Riverpod's redundant "isLoading" gap when combining providers #4009

Open
@rrousselGit

Description

@rrousselGit

This is a follow-up to #4008
It details more the ref.setData part, to determine if this is valuable or we'd rather stick to the usual return data.

The problem

One of Riverpod's big issue when combining providers has always been:
Chaining asynchronous providers makes providers down the chain update 1+ frame after its dependency has updated.

Consider:

final valueProvider = FutureProvider<int>(...);

final doubleValue = FutureProvider<int>((ref) async {
  final value = await ref.watch(valueProvider.future);
  return value * 2;
}

...
ref.listen(doubleValue, (a, b) => print('${DateTime.now()} $b'));

In this example, updating value will print:

[00:00:00] AsyncLoading<int>()
[00:00:02] AsyncData(value: <new value>)

The reason for this AsyncLoading is twofold:

  • When doing return value in an async function, Riverpod doesn't get access to the value until one frame has passed.
  • Doing await ref.watch(valueProvider.future) adds another delay of one frame ; even if the value is already accessible.

As such, our doubleValue updates in a different frame than valueProvider. This is problematic if you were to render both in the same page, as an attentive eye would see one value change before the other.

Say you had your UI do:

Text('value: ${ref.watch(valueProvider).value}');
Text('value * 2: ${ref.watch(doubleValue).value}');

Then your UI may show:

// Initial render:
value: 1
value * 2: 2

// We update value
value: 5
value * 2: 2
// For a small amount of time, value and doubleValue are out of sync

// Last frame:
value: 5
value * 2: 10
// After a little bit, the values re-synchronize

It isn't horrifying, and it's very hard to notice. But it is a problem.

Proposal

To fix this issue, we'd have to change how FutureProvider emits its value and how we await other providers.

Removing the gap caused by return

Instead of return value * 2, we could do:

final doubleValue = FutureProvider<int>((ref) async {
  final value = await ref.watch(valueProvider.future);
  ref.state = AsyncData(value * 2);
}

This technically removes the delay introduced by return in async functions.
The problem is, it's easy to forget to call ref.state=.

Nothing forces you to ever set a state. If you forget to do so, your UI will permanently show a progress indicator. This sucks.
To fix this, we could instead do:

final doubleValue = FutureProvider<int>((ref) async {
  final value = await ref.watch(valueProvider.future);
  ref.returns(value * 2);
}

The idea is:

  • FutureProvider would no-longer care about the return value. Instead we'd only react to ref.state changes.
  • As such, FutureProvider's callback wouldn't be expected to return a Future<int> anymore, but rather a Future<Event>
  • You would have no way to create an Event instance. As such, the only way to have the program compile would be to:
    • call ref.setData/setError and return the result
    • call ref.returns
    • throw
  • ref.returns is sugar for ref.state = AsyncData(); and returns Never ; which aborts the execution of the function.
  • We can also offer a ref.emit(stream) to fuse StreamProvider/FutureProvider while we're at it.

This likely sounds confusing. The idea is:
In practice, the compiler will force you to specify one return. It is a compilation error to not specify a return thanks to the fact that we need to return an Event.

As such, this isn't legal:

final doubleValue = FutureProvider<int>((ref) async {
  ref.setData(42);
  // Compilation error, no return
}

But this is:

final doubleValue = FutureProvider<int>((ref) async {
  ref.returns(42);
}

And this is too:

final doubleValue = FutureProvider<int>((ref) async {
  Event event;
   // Simulate some complex initialization logic:
  if (something) event = ref.setData(42);
  return event;
}

So all forms of initialization should be covered :)

Solving await ref.watch(provider.future)

For this, the solution is fairly simple: Use SynchronousFuture from Flutter
When a provider watches another provider and the value is already available, we'd be returning a SynchronousFuture.
This enables Dart to skip the await

The drawback is that this is a breaking change. The main reason is because SynchronousFuture only really works with await.

For example, you can't do:

Future.wait([
  ref.watch(provider.future)
])

Future.wait doesn't understand SynchronousFuture and will break.

Of course, we could consider offering two options: A ref.watch(provider.future) which returns a normal future. And a ref.watch(provider.syncFuture) (name to be determined),
Or we could have a lint rule which enforces await ref.watch(provider.future) and warn if you use the Future in other contexts.

Conclusion

What's described here is a bit of an advanced Dart topic, so it may be difficult to grasp.
But the idea is that with a small syntax tweak, we could remofe that pesky delay when combining providers.

The question that remains is:

  • Is the gain worth the complexity?
    Folks are used to return value in their providers.
  • Should synchronous providers use the same syntax, for consistency?
    final provider = Provider((ref) => ref.returns(42));

Metadata

Metadata

Assignees

Labels

enhancementNew feature or request

Projects

No projects

Milestone

No milestone

Relationships

None yet

Development

No branches or pull requests

Issue actions