English | French | Português | 简体中文 | Español | 한국어 | বাংলা | 日本語 | Turkish | Italian
Un wrapper alrededor de InheritedWidget para hacerlo más fácil de usar y más utilizable.
Al utilizar provider
en lugar de escribir manualmente los InheritedWidget, obtendrás:
- asignación/eliminación simplificada de los recursos
- lazy-loading
- un boilerplate muy reducido en lugar de hacer una nueva clase cada vez.
- compatibilidad con devtools
- una manera común de consumir esto InheritedWidgets (ver Provider.of/Consumer/Selector)
- incrementa la escalabilidad de tus clases con un mecanismo de escucha (listener) que crece exponencialmente en complejidad (tales como ChangeNotifier, el cual es O(N²) al enviar notificaciones)
Para leer más acerca de provider
, leé su documentación.
Ver también:
- La documentación oficial de Flutter en gestión de estado, la cual muestra como usar
provider
+ ChangeNotifier - Ejemplos de arquitectura en Flutter, la cual contiene una implementación de una app usando
provider
+ ChangeNotifier - flutter_bloc y Mobx, los cuales usan
provider
en su arquitectura.
-
Los parámetros
builder
einitialBuilder
de providers son eliminados.initialBuilder
deberá ser reemplazado porcreate
.builder
de "proxy" providers deberá ser reemplazado porupdate
builder
de providers clásicos deberá ser reemplazado porcreate
.
-
las nuevas callbacks
create
/update
son lazy-loaded, lo cual significa que serán llamadas la primera vez que su valor sea leído, en lugar de la primera vez que el usuario sea creado.Si esto no se quiere, puedes deshabilitar el lazy-loading pasando el valor
lazy: false
a el provider de tu elección.FutureProvider( create: (_) async => doSomeHttpRequest(), lazy: false, child: ... )
-
ProviderNotFoundError
es renombrado aProviderNotFoundException
. -
La interface
SingleChildCloneableWidget
es eliminada y reemplazada por un nuevo tipo de widgetSingleChildWidget
.Ve este issue para detalles acerca de como migrar.
-
Selector ahora comparará más profundamente el valor anterior y nuevo si estas son colecciones.
Si este no quiere, puedes revertir el comportamiento pasando el parámetro
shouldRebuild
al SelectorSelector<Selected, Consumed>( shouldRebuild: (previous, next) => previous == next, builder: ..., )
-
DelegateWidget
y su familia han sido eliminados. En su lugar, para providers personalizados tendrán subclasses directas InheritedProvider o un provider existente.
Los providers permiten no solo exponer un valor, sino también crear/escuchar/eliminar.
Para exponer un objeto recién creado, utilice el constructor por defecto de un provider.
No uses el constructor .value
si quieres crear un objeto, o
podrías tener efectos secundarios no deseados.
Ve esta respuesta de stackoverflow
la cual explica en mayor detalle, por qué usar el constructor .value
para crear nuevos valores no es óptimo.
- DO crea un nuevo objeto dentro de
create
.
Provider(
create: (_) => MyModel(),
child: ...
)
- DON'T usar
Provider.value
para crear tu objeto.
ChangeNotifierProvider.value(
value: MyModel(),
child: ...
)
-
DON'T crear tu objeto de variables que pueden cambiar en el tiempo.
En esta situación, tu objeto nunca será actualizado cuando el valor cambie.
int count;
Provider(
create: (_) => MyModel(count),
child: ...
)
Si quieres pasar variables que puede cambiar en el tiempo de tu objeto,
considera utilizar ProxyProvider
:
int count;
ProxyProvider0(
update: (_, __) => MyModel(count),
child: ...
)
NOTA:
Cuando se utiliza la llamada de create
/update
de un provider, vale la pena señalar que en esta llamada
se llama lazy
por defecto.
Lo que esto significa es que, hasta que el valor sea solicitado al menos una vez, el create
/update
no se llamarán las callback.
Este comportamiento puede ser desactivado si quieres precalcular algo de lógica, usando el parámetro lazy
:
MyProvider(
create: (_) => Something(),
lazy: false,
)
Si ya tienes una instancia de un objeto y quieres exponerlo,
deberías usar el constructor .value
de un proveedor.
Si no lo hace, puede llamar al método dispose
de tu objeto cuando todavía está en uso.
- DO usa
ChangeNotifierProvider.value
para un provider tipo ChangeNotifier existente.
MyChangeNotifier variable;
ChangeNotifierProvider.value(
value: variable,
child: ...
)
- DON'T reutiliza un ChangeNotifier existente usando su constructor por defecto.
MyChangeNotifier variable;
ChangeNotifierProvider(
create: (_) => variable,
child: ...
)
La manera mas fácil de leer un valor es usando los métodos de su extensión en [BuildContext]:
context.watch<T>()
, el cual hace que el widget escuche los cambios enT
.context.read<T>()
, este regresaT
sin escucharlo.context.select<T, R>(R cb(T value))
, permite al widget escuchar solo una pequeña parte deT
.
O utiliza el método estático Provider.of<T>(context)
, el cual se comporta similar a watch
y cuando le pasa el valor false
al parámetro listen
como Provider.of<T>(context, liste: false)
se comportará similar a read
Vale la pena decir que context.read<T>()
no hará que tu widget haga un rebuild cuando el valor cambia y no puede ser
llamado dentro de StatelessWidget.build
/State.build
. Por otro lado, puede ser llamado libremente fuera de estos métodos.
Estos métodos buscarán en el árbol de los widgets a partir del widget asociado
con el BuildContext
pasado, y devolverá la variable de tipo más cercano
T
encontrada (o lanzara una exception
si no encuentra nada).
Vale la pena señalar que esta operación es O(1). No implica realmente recorrer en el árbol de los widgets.
Combinado con el primer ejemplo de exponer un valor, este
el widget leerá el String
expuesto y mostrará "Hola Mundo".
class Home extends StatelessWidget {
@override
Widget build(BuildContext context) {
return Text(
// Don't forget to pass the type of the object you want to obtain to `watch`!
context.watch<String>(),
);
}
}
Otra opción, en lugar de utilizar estos métodos, podemos usar Consumer y Selector.
Estos pueden ayudar a optimizar el desempeño o cuando es difícil de obtener
el BuildContext
descendiente del provider
Ve las FAQ o la documentación de Consumer y Selector para más información.
Cuando inyectamos muchos valores en una aplicación grande, Provider
puede convertirse
muy anidado rápidamente.
Provider<Something>(
create: (_) => Something(),
child: Provider<SomethingElse>(
create: (_) => SomethingElse(),
child: Provider<AnotherThing>(
create: (_) => AnotherThing(),
child: someWidget,
),
),
),
A:
MultiProvider(
providers: [
Provider<Something>(create: (_) => Something()),
Provider<SomethingElse>(create: (_) => SomethingElse()),
Provider<AnotherThing>(create: (_) => AnotherThing()),
],
child: someWidget,
)
El comportamiento de ambos ejemplos es el mismo. Multiprovider
solo cambia la
apariencia del código.
Desde la versión 3.0.0, existe un nuevo tipo de provider: ProxyProvider
.
ProxyProvider
es un provider que combina múltiples valores de otros provider
dentro de un nuevo objeto, y envía el resultado a Provider
.
Este nuevo objeto será actualizado cuando uno de los providers de los que depende sea actualizado.
El siguiente ejemplo usa ProxyProvider
para construir una traducción de acuerdo a un contador
que viene de otro provider.
Widget build(BuildContext context) {
return MultiProvider(
providers: [
ChangeNotifierProvider(create: (_) => Counter()),
ProxyProvider<Counter, Translations>(
update: (_, counter, __) => Translations(counter.value),
),
],
child: Foo(),
);
}
class Translations {
const Translations(this._value);
final int _value;
String get title => 'You clicked $_value times';
}
Viene con múltiples variaciones, tales como:
-
ProxyProvider
vsProxyProvider2
vsProxyProvider3
, ...El dígito después del nombre de la clase es el número de providers que
ProxyProvider
depende. -
ProxyProvider
vsChangeNotifierProxyProvider
vsListenableProxyProvider
, ...Todos trabajan de manera similar, pero en lugar de enviar el resultado del
Provider
, elChangeNotifierProxyProvider
enviará su valor a unChangeNotifierProvider
.
Flutter viene con devtools, el cual muestra cómo el árbol de widgets, está en un momento determinado.
Ya que los providers son widgets, también son visibles en las devtools:
De ahí, si tu presionas un provider, serás capaz de ver los valores que contiene:
(screenshot de las devtools usando la carpeta de example
)
Por defecto, devtool necesita toString
, el cual por defecto es "Instance of MyClass".
Para algo más útil, tienes dos opciones:
-
Usa el API Diagnosticable de Flutter.
Para la mayoría de los casos, esto se logrará utilizando DiagnosticableTreeMixin en tus objetos, siguiendo la implementación personalizada de debugFillProperties.
class MyClass with DiagnosticableTreeMixin { MyClass({this.a, this.b}); final int a; final String b; @override void debugFillProperties(DiagnosticPropertiesBuilder properties) { super.debugFillProperties(properties); // list all the properties of your class here. // See the documentation of debugFillProperties for more information. properties.add(IntProperty('a', a)); properties.add(StringProperty('b', b)); } }
-
override
toString
.Si no puedes usar DiagnosticableTreeMixin (como si tu clase estuviera en un paquete que no depende de Flutter), entonces puedes simplemente usar el método override
toString
.Esto es más fácil que usar DiagnosticableTreeMixin pero es menos poderoso: No podrás expandir/contraer los detalles de tu objeto.
class MyClass with DiagnosticableTreeMixin { MyClass({this.a, this.b}); final int a; final String b; @override String toString() { return '$runtimeType(a: $a, b: $b)'; } }
Esta excepción ocurre porque estás tratando de escuchar a un provider de un life-cycle que nunca jamás será llamado de nuevo.
Significa que debes usar otro ciclo de vida (build
), o explícitamente
específica que no te importan las actualizaciones.
Entonces, en lugar de:
initState() {
super.initState();
print(context.watch<Foo>().value);
}
puedes hacer esto:
Value value;
Widget build(BuildContext context) {
final value = context.watch<Foo>().value;
if (value != this.value) {
this.value = value;
print(value);
}
}
El cual imprime value
cuando sea que cambie (y solo cuando cambie)
También puedes hacer esto:
initState() {
super.initState();
print(context.read<Foo>().value);
}
El cual imprimirá value
una vez e ignorará las actualizaciones.
Puedes hacer que el objeto de tu provider implemente ReassembleHandler
:
class Example extends ChangeNotifier implements ReassembleHandler {
@override
void reassemble() {
print('Did hot-reload');
}
}
Después úsalo de forma normal con provider
:
ChangeNotifierProvider(create: (_) => Example()),
Utilizo ChangeNotifier y tengo una exception cuando se actualiza, que sucede?
Es probable que esto ocurra porque se está modificando el ChangeNotifier de uno de sus descendientes, mientras que el árbol de los widgets se está construyendo.
Una situación típica en la que esto ocurre es cuando se inicia una petición http, donde el futuro está almacenado dentro del notifier:
initState() {
super.initState();
context.read<MyNotifier>().fetchSomething();
}
Esto no está permitido, porque la actualización es inmediata.
Lo que significa que algunos widgets pueden construir antes de la mutación, mientras que otros widgets construirán después de la mutación. Esto podría causar inconsistencias en su UI y por lo tanto no está permitido.
En su lugar, deberías realizar esa mutación en un lugar que afectará a todo el árbol por igual:
-
directamente dentro del
create
de tu provider/constructor de tu modelo:class MyNotifier with ChangeNotifier { MyNotifier() { _fetchSomething(); } Future<void> _fetchSomething() async {} }
Esto es útil cuando no existe un "parámetro externo".
-
asincronamente al final del frame
initState() { super.initState(); Future.microtask(() => context.read<MyNotifier>(context).fetchSomething(someValue); ); }
Esto es ligeramente menos ideal, pero permite pasar parámetros a la mutación.
Tengo que usar ChangeNotifier para estados complejos?
No.
Puedes utilizar un objeto para representar tu estado. Por ejemplo, otra opción
es usar la arquitectura de Provider.value()
combinada con StatefulWidget
.
Aquí está un ejemplo del counter usando tal arquitectura:
class Example extends StatefulWidget {
const Example({Key key, this.child}) : super(key: key);
final Widget child;
@override
ExampleState createState() => ExampleState();
}
class ExampleState extends State<Example> {
int _count;
void increment() {
setState(() {
_count++;
});
}
@override
Widget build(BuildContext context) {
return Provider.value(
value: _count,
child: Provider.value(
value: this,
child: widget.child,
),
);
}
}
donde podemos leer el estado haciendo esto:
return Text(context.watch<int>().toString());
y modificar el estado con:
return FloatingActionButton(
onPressed: () => context.read<ExampleState>().increment(),
child: Icon(Icons.plus_one),
);
También, puedes crear tu propio provider.
Si. provider
muestra todos los pequeños componentes para hacer el tuyo.
Esto incluye:
-
SingleChildStatelessWidget
, para hacer que el widget funcione conMultiProvider
. Esta interfaz es expuesta como parte depackage:provider/single_child_widget
-
InheritedProvider, el
InheritedWidget
genérico obtenido cuando hacemoscontext.watch
.
Aquí está un ejemplo de un provider personalizado para usar ValueNotifier
como estado:
https://gist.github.com/rrousselGit/4910f3125e41600df3c2577e26967c91
En lugar de usar context.watch
, puedes usar context.select
para escuchar solo a un conjunto
específico de propiedades en el objeto obtenido
Por ejemplo, podrías escribir esto:
Widget build(BuildContext context) {
final person = context.watch<Person>();
return Text(person.name);
}
Podría provocar que el widget haga rebuild, si otro aparte de name
cambia.
En su lugar, puedes usar context.select
para escuchar solo los cambios en la propiedad name
:
Widget build(BuildContext context) {
final name = context.select((Person p) => p.name);
return Text(name);
}
De esta manera, el widget no realizará rebuilds innecesarios si algo aparte de name
cambia.
Similarmente, puedes utilizar Consumer/Selector.
Su propiedad child
opcional permite solo hacer rebuild en una parte especifica del
árbol de widgets.
Foo(
child: Consumer<A>(
builder: (_, a, child) {
return Bar(a: a, child: child);
},
child: Baz(),
),
)
En este ejemplo, solo Bar
hará un rebuild cuando A
actualice. Foo
y Baz
no harán builds
innecesarios.
No. Aunque puedes tener varios providers compartiendo el mismo tipo, un widget será capaz de obtener sólo uno de ellos: el ancestro más cercano.
En cambio, debe dar explícitamente a ambos proveedores un tipo diferente
En lugar de:
Provider<String>(
create: (_) => 'England',
child: Provider<String>(
create: (_) => 'London',
child: ...,
),
),
Utiliza:
Provider<Country>(
create: (_) => Country('England'),
child: Provider<City>(
create: (_) => City('London'),
child: ...,
),
),
Sí, se debe dar una pista del tipo al compilador para indicar que la interfaz se consumirá, con la implementación prevista en create.
abstract class ProviderInterface with ChangeNotifier {
...
}
class ProviderImplementation with ChangeNotifier implements ProviderInterface {
...
}
class Foo extends StatelessWidget {
@override
build(context) {
final provider = Provider.of<ProviderInterface>(context);
return ...
}
}
ChangeNotifierProvider<ProviderInterface>(
create: (_) => ProviderImplementation(),
child: Foo(),
),
provider
nos muestra algunos tipos diferentes de "provider" para diferentes tipos de objetos.
La lista completa de todos los objetos disponibles esta aquí
Nombre | Descripción |
---|---|
Provider | La forma más básica de provider. Toma un valor y lo expone, sea cual sea el valor. |
ListenableProvider | Un proveedor específico para el objeto Listenable. ListenableProvider escuchará el objeto y pedirá a los widgets que dependen de él que lo reconstruyan siempre que se llame al listener. |
ChangeNotifierProvider | Se especifica el tipo de ListenableProvider para ChangeNotifier. Este llamará automáticamente ChangeNotifier.dispose cuando lo necesite. |
ValueListenableProvider | Escucha al ValueListenable y solo muestra ValueListenable.value . |
StreamProvider | Escucha a un Stream y muestra el último valor emitido. |
FutureProvider | Toma un Future y actualiza a sus dependientes cuando el futuro es completado. |