# Flutter State Management made easy with provider 2: Provider as a Dependency Injection framework and MultiProvider

This is the second part of this tutorial series on managing Flutter state with the provider package. In [the first part](https://davidserrano.io/flutter-state-management-made-easy-with-provider-flutter-tutorial-for-beginners) we learned how we can manage the state of our widgets from model classes that inherit from `ChangeNotifier`, as well as the use of `ChangeNotifierProvider` to be able to provide our widget tree with said classes, and finally the use of the `Consumer` class to be able to read these state changes and react to them.

> 📽 Video version available on [YouTube](https://youtu.be/ZerlzX8FMBE) and [Odysee](https://odysee.com/@svprdga:d/flutter-state-management-made-easy-with-provider-2-provider-as-a-dependency-injection-framework-and-multiprovider)

%[https://youtu.be/ZerlzX8FMBE]

In this second part we will delve into the following concepts:
- First we'll see how to make multiple widgets react based on a single state provider
- We will learn how to use provider as a dependency injection framework
- Finally, we will use `MultiProvider` to make our code more readable

If you want to learn how to manage state with provider and haven't read the first part yet, I suggest you take a look at it [here](https://davidserrano.io/flutter-state-management-made-easy-with-provider-flutter-tutorial-for-beginners) first. If you have already read it and are ready to continue learning, here we go.

### Manage the state of multiple widgets with a single state provider

Let's start with the test app from the previous tutorial, you can clone, download or fork it from [here](https://github.com/svprdga/Flutter-State-Management-Basic-Provider).

We have the basic counter app. What I want to achieve is to be able to keep that numeric count and show it in different widgets, and I also want to be able to increase it from all those widgets.

First, copy `MyHomePage` from `main.dart` into a new `second_counter.dart` file, and rename it to `SecondCounter`, like so:

```dart
// lib/second_counter.dart

import 'package:flutter/material.dart';
import 'package:flutter_state_management_easy_provider/main_model.dart';
import 'package:provider/provider.dart';

class SecondCounter extends StatefulWidget {
  const SecondCounter({Key? key, required this.title}) : super(key: key);

  final String title;

  @override
  State<SecondCounter> createState() => _SecondCounterState();
}

class _SecondCounterState extends State<SecondCounter> {
  @override
  Widget build(BuildContext context) {
    return ChangeNotifierProvider(
      create: (_) => MainModel(),
      child: Consumer<MainModel>(
        builder: (context, model, child) => Scaffold(
          appBar: AppBar(
            title: Text(widget.title),
          ),
          body: Center(
            child: Column(
              mainAxisAlignment: MainAxisAlignment.center,
              children: <Widget>[
                const Text(
                  'You have pushed the button this many times:',
                ),
                Text(
                  '${model.counter}',
                  style: Theme.of(context).textTheme.headline4,
                ),
              ],
            ),
          ),
          floatingActionButton: FloatingActionButton(
            onPressed: () => model.counter++,
            tooltip: 'Increment',
            child: const Icon(Icons.add),
          ),
        ),
      ),
    );
  }
}
```

Now add a new element to the `Column` of `MyHomePage` in the form of a button to open this new screen:

```dart
// main.dart

// [...]

class MyHomePage extends StatefulWidget {
  const MyHomePage({Key? key, required this.title}) : super(key: key);

  final String title;

  @override
  State<MyHomePage> createState() => _MyHomePageState();
}

class _MyHomePageState extends State<MyHomePage> {
  @override
  Widget build(BuildContext context) {
    return ChangeNotifierProvider(
      create: (_) => MainModel(),
      child: Consumer<MainModel>(
        builder: (context, model, child) => Scaffold(
          appBar: AppBar(
            title: Text(widget.title),
          ),
          body: Center(
            child: Column(
              mainAxisAlignment: MainAxisAlignment.center,
              children: <Widget>[
                const Text(
                  'You have pushed the button this many times:',
                ),
                Text(
                  '${model.counter}',
                  style: Theme.of(context).textTheme.headline4,
                ),
                // Add this button to open SecondCounter
                Padding(
                  padding: const EdgeInsets.only(top: 20.0),
                  child: TextButton(
                    child: const Text('Open counter 2'),
                    onPressed: () => Navigator.push(
                      context,
                      MaterialPageRoute(
                        builder: (context) => const SecondCounter(
                          title: 'Second counter',
                        ),
                      ),
                    ),
                  ),
                )
              ],
            ),
          ),
          floatingActionButton: FloatingActionButton(
            onPressed: () => model.counter++,
            tooltip: 'Increment',
            child: const Icon(Icons.add),
          ),
        ),
      ),
    );
  }
}
```

Try running the app. Try incrementing the counter value, and then go to the second screen and try incrementing the counter value from there. What's the result? Well, we have two screens and each one with an independent value, and that is not what we want. Why is this happening? Well, the reason is that we are actually creating two instances of `MainModel`, one in `MyHomePage` and a different one in `SecondCounter`, each of which is provided to its descendant widgets via a `ChangeNotifierProvider`.

In order to have a unique count in the entire application, what we will have to do is move that `ChangeNotifierProvider` one level up until both screens hang from it, for example we could place it just above `MyHomePage`, in the class that wraps `MaterialApp` (`main.dart` file):

```dart
// main.dart

// [...]

void main() {
  runApp(const MyApp());
}

class MyApp extends StatelessWidget {
  const MyApp({Key? key}) : super(key: key);

  @override
  Widget build(BuildContext context) {
    return ChangeNotifierProvider<MainModel>(
      create: (_) => MainModel(),
      child: MaterialApp(
        title: 'Flutter State Management Basic',
        theme: ThemeData(
          primarySwatch: Colors.blue,
        ),
        home: const MyHomePage(title: 'Flutter State Management Basic'),
      ),
    );
  }
}

// [...]
```

Now, remove the `ChangeNotifierProvider` from `MyHomePage` and `SecondCounter`, so that the `build()` methods directly return the `Consumer<MainModel>`.

Run the app again and try to increment the counts, you will see that now there is only one unique count. This is because there is only one `MainModel` instance, and any widget below the `ChangeNotifierProvider` that provides it can access it to consume its state.

### Provider as a dependency injection framework

_Dependency injection_ is a software development technique in which the dependencies of a class are injected from the outside instead of being instantiated inside it. I could write several articles about this technique and its applications, but in order to understand each other and to focus on the topic at hand, we are going to learn how to use `provider` to create an instance and include it in the widget tree so that it can be used from anywhere in the app.

To do this, I am going to give an example of the logs package that I use, [logger](https://pub.dev/packages/logger). What I want to do is create an instance and write a log trace when the counter is incremented on the first screen and also on the second screen, and I want that instance to be shared across the app. The first thing will be to install the package:

```bash
flutter pub add logger
```

We are now going to use the `Provider` class to accomplish this task. This is the basic class that gives this package its name, in fact both `ChangeNotifierProvider` and other types of providers that we are going to see in this series of tutorials are variations or forms of this original class:

```dart
// main.dart

// [...]

void main() {
  runApp(const MyApp());
}

class MyApp extends StatelessWidget {
  const MyApp({Key? key}) : super(key: key);

  @override
  Widget build(BuildContext context) {
    return ChangeNotifierProvider<MainModel>(
      create: (_) => MainModel(),
      child: Provider<Logger>(
        create: (_) => Logger(),
        child: MaterialApp(
          title: 'Flutter State Management Basic',
          theme: ThemeData(
            primarySwatch: Colors.blue,
          ),
          home: const MyHomePage(title: 'Flutter State Management Basic'),
        ),
      ),
    );
  }
}

// [...]
```

As you can see, to make a 'hole' for this new provider, what we do is chain one inside the other. Now we are going to modify the `FloatingActionButton` of each screen to, in addition to increasing the count, print a log trace. To get this `Logger` we just created we use `Provider.of<Logger>(context, listen: false)`. Through this invocation we are indicating the type of the class we want to obtain, we pass the current `context` and we indicate `listen: false`. We'll see why this last argument has this value later in this tutorial series, for now leave it set to `false`:

```dart
// main.dart

// Alter the existent FloatingActionButton:
floatingActionButton: FloatingActionButton(
  onPressed: () {
    model.counter++;
    Provider.of<Logger>(context, listen: false).d(
      'Increment counter from MyHomePage',
    );
  },
  tooltip: 'Increment',
  child: const Icon(Icons.add),
),
```

```dart
// second_counter.dart

// Alter the existent FloatingActionButton:
floatingActionButton: FloatingActionButton(
  onPressed: () {
    model.counter++;
    Provider.of<Logger>(context, listen: false).d(
      'Increment counter from SecondCounter',
    );
  },
  tooltip: 'Increment',
  child: const Icon(Icons.add),
),
```

As you can see, through the `Provider` class you can provide the instances of the classes that you want to any Widget of your application.

### Make the code more readable using MultiProvider

What we have seen today is very good and very useful, but note that normal applications do not have only 2 dependencies as we have in this example, they can have 5, 10, 100 or more different dependencies. Imagine how each of them would be included in the next one, as we have now in `MyApp`, it would be very difficult to maintain and read, and this is why `MultiProvider` exists.

![nested_providers_blog.png](https://cdn.hashnode.com/res/hashnode/image/upload/v1651569766652/0-4GuvQ51.png align="left")

`MultiProvider` is a way to declare all your providers in an array. Let's see it in action, modify `MyApp` as follows:

```dart
// main.dart

// [...]

void main() {
  runApp(const MyApp());
}

class MyApp extends StatelessWidget {
  const MyApp({Key? key}) : super(key: key);

  @override
  Widget build(BuildContext context) {
    return MultiProvider(
      providers: [
        ChangeNotifierProvider<MainModel>(create: (_) => MainModel()),
        Provider<Logger>(
          create: (_) => Logger(),
        ),
      ],
      child: MaterialApp(
        title: 'Flutter State Management Basic',
        theme: ThemeData(
          primarySwatch: Colors.blue,
        ),
        home: const MyHomePage(title: 'Flutter State Management Basic'),
      ),
    );
  }
}

// [...]
```

That's much better. Each provider is an element of the array provided in the `providers` argument, so your code will be much better organized. Each of the providers specified in that array will be available to your application, exactly as before.

### Conclusion

In this second tutorial we have delved into the basic concepts of the provider package. We have seen how to manage state at different points using a single `ChangeNotifier` located at the top of the widget tree. We have also used `Provider` to provide single instances to the application and have arranged all our providers with `MultiProvider`.

The `provider` package contains other providers that can make your life easier while developing, I will cover them in the following chapters of this tutorial series.

If you have any questions or suggestions do not hesitate to leave me a comment below.

You can find the final app of this tutorial [here](https://github.com/svprdga/Flutter-State-Management-Easy-Provider-2).

Thanks for reading and happy coding :)
