Saltar al contenido principal

StateNotifierProvider

StateNotifierProvider es un provider que se usa para escuchar y exponer un StateNotifier (del paquete state_notifier, que Riverpod reexporta).
StateNotifierProvider junto con StateNotifier es la solución recomendada por Riverpod para administrar el estado que puede cambiar en reacción a una interacción del usuario.

Normalmente se utiliza para:

  • exponer un estado inmutable que puede cambiar con el tiempo después de reaccionar a eventos personalizados.
  • centralizar la lógica para modificar algún estado (también conocido como "business logic") en un solo lugar, mejorando la capacidad de mantenimiento con el tiempo.

Como ejemplo de uso, podríamos usar StateNotifierProvider para implementar una lista de tareas pendientes. Hacerlo nos permitiría exponer métodos como addTodo para permitir que la interfaz de usuario modifique la lista de todos en las interacciones del usuario:


// El estado de nuestro StateNotifier debe ser inmutable.
// También podríamos usar paquetes como Freezed para ayudar con la implementación.

class Todo {
const Todo({required this.id, required this.description, required this.completed});

// Todas las propiedades deben ser `final` en nuestra clase.
final String id;
final String description;
final bool completed;

// Como `Todo` es inmutable, implementamos un método que permite clonar el
// `Todo` con un contenido ligeramente diferente.
Todo copyWith({String? id, String? description, bool? completed}) {
return Todo(
id: id ?? this.id,
description: description ?? this.description,
completed: completed ?? this.completed,
);
}
}

// La clase StateNotifier que se pasará a nuestro StateNotifierProvider.
// Esta clase no debe exponer el estado fuera de su propiedad "estado", lo que significa
// ¡sin getters/propiedades públicas!
// Los métodos públicos en esta clase serán los que permitirán
// que la interfaz de usuario modifique el estado.
class TodosNotifier extends StateNotifier<List<Todo>> {
// Inicializamos la lista de `todos` como una lista vacía
TodosNotifier(): super([]);

// Permitamos que la interfaz de usuario agregue todos.
void addTodo(Todo todo) {
// Ya que nuestro estado es inmutable, no podemos hacer `state.add(todo)`.
// En su lugar, debemos crear una nueva lista de todos que contenga la anterior
// elementos y el nuevo.
// ¡Usar el spread operator de Dart aquí es útil!
state = [...state, todo];
// No es necesario llamar a "notifyListeners" o algo similar. Llamando a "state ="
// reconstruirá automáticamente la interfaz de usuario cuando sea necesario.
}

// Permitamos eliminar `todos`
void removeTodo(String todoId) {
// Nuevamente, nuestro estado es inmutable. Así que estamos haciendo
// una nueva lista en lugar de cambiar la lista existente.
state = [
for (final todo in state)
if (todo.id != todoId) todo,
];
}

// Marcamos una `todo` como completada
void toggle(String todoId) {
state = [
for (final todo in state)
// Estamos marcando solo el `todo` coincidente como completada
if (todo.id == todoId)
// Una vez más, dado que nuestro estado es inmutable, necesitamos hacer una copia
// del `todo`. Estamos usando nuestro método `copyWith` implementado antes
// para ayudar con eso.
todo.copyWith(completed: !todo.completed)
else
// otros `todos` no se modifican
todo,
];
}
}

// Finalmente, estamos usando StateNotifierProvider para permitir que la
// interfaz de usuario interactúe con nuestra clase TodosNotifier.
final todosProvider = StateNotifierProvider<TodosNotifier, List<Todo>>((ref) {
return TodosNotifier();
});

Ahora que hemos definido un StateNotifierProvider, podemos usarlo para interactuar con la lista de todos en nuestra interfaz de usuario:


class TodoListView extends ConsumerWidget {
const TodoListView({super.key});


Widget build(BuildContext context, WidgetRef ref) {
// Reconstruir el widget cuando cambia la lista de `todos`
List<Todo> todos = ref.watch(todosProvider);

// Vamos a representar los `todos` en un ListView
return ListView(
children: [
for (final todo in todos)
CheckboxListTile(
value: todo.completed,
// Al tocar un `todo`, cambie su estado a completado
onChanged: (value) => ref.read(todosProvider.notifier).toggle(todo.id),
title: Text(todo.description),
),
],
);
}
}