InheritedWidget with Scaffold as child doesn't

2019-01-11 21:44发布

问题:

I was hoping to use InheritedWidget at the root level of my Flutter application to ensure that an authenticated user's details are available to all child widgets. Essentially making the Scaffold the child of the IW like this:

@override
Widget build(BuildContext context) {
return new AuthenticatedWidget(
    user: _user,
    child: new Scaffold(
      appBar: new AppBar(
        title: 'My App',
      ),
      body: new MyHome(),
      drawer: new MyDrawer(),
    ));
}

This works as expected on app start so on the surface it seems that I have implemented the InheritedWidget pattern correctly in my AuthenticatedWidget, but when I return back to the home page (MyHome) from elsewhere like this:

Navigator.popAndPushNamed(context, '/home');

This call in the build method of MyHome (which worked previously) then results in authWidget being null:

final authWidget = AuthenticatedWidget.of(context);

Entirely possible I'm missing some nuances of how to properly implement an IW but again, it does work initially and I also see others raising the same question (i.e. here under 'Inherited Widgets' heading).

Is it therefore not possible to use a Scaffold or a MaterialApp as the child of an InheritedWidget? Or is this maybe a bug to be raised? Thanks in advance!

回答1:

This is a problem many are struggling to understand when beginning in flutter.

MyInherited.of(context) will basically look into the parent of the current context to see if there's a MyInherited instantiated.

The problem is : Your inherited widget is instantiated within the current context.

=> No MyInherited as parent

=> crash

The trick is to use a different context. There are many solutions there. You could instantiate MyInherited in another widget, so that the context of your build method will have a MyInherited as parent.

Or you could potentially use a Builder to introduce a fake widget that will pass you it's context.

Example of builder :

return new MyInheritedWidget(
  child: new Builder(
    builder: (context) => new Scaffold(),
  ),
);

Another problem, for the same reasons, is that if you insert an inheritedWidget inside a route, it will not be available outside of this route.

The solution is simple here ! Put your MyInheritedWidget above or _inside MaterialApp.

Two examples :

above material :

new MyInherited(
  child: new MaterialApp(
    // ...
  ),
)

inside material :

new MaterialApp(
  builder: (context, child) {
    return MyInherited(child: child);
  },
)

The second example is useful when your InheritedWidget needs to access Navigator to push pages/dialogs. Such as login for example.



回答2:

Is it therefore not possible to use a Scaffold or a MaterialApp as the child of an InheritedWidget?

It is very possible to do this. I was struggling with this earlier and posted some details and sample code here.

You might want to make your App-level InheritedWidget the parent of the MaterialApp rather than the Scaffold widget.

I think this has more to do with how you are setting up your MaterialWidget, but I can't quite tell from the code snippets you have provided.

If you can add some more context, I will see if I can provide more.



标签: flutter