Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

InheritedWidget with Scaffold as child doesn't seem to be working

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 the '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!

like image 547
KBDrums Avatar asked Mar 19 '18 23:03

KBDrums


1 Answers

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 MaterialApp.

above material :

new MyInherited(
  child: new MaterialApp(
    // ...
  ),
)
like image 165
Rémi Rousselet Avatar answered Oct 01 '22 06:10

Rémi Rousselet