Search code examples
flutterdartlistviewflutter-container

Flutter: Scrolling to a widget in ListView


How can I scroll to a special widget in a ListView? For instance I want to scroll automatically to some Container in the ListView if I press a specific button.

ListView(children: <Widget>[
  Container(...),
  Container(...), // scroll for example to this container 
  Container(...)
]);

Solution

  • By far, the easiest solution is to use Scrollable.ensureVisible(context). As it does everything for you and work with any widget size. Fetching the context using GlobalKey.

    The problem is that ListView won't render non-visible items. Meaning that your target most likely will not be built at all. Which means your target will have no context ; preventing you from using that method without some more work.

    In the end, the easiest solution will be to replace your ListView by a SingleChildScrollView and wrap your children into a Column. Example :

    class ScrollView extends StatelessWidget {
      final dataKey = new GlobalKey();
    
      @override
      Widget build(BuildContext context) {
        return new Scaffold(
          primary: true,
          appBar: new AppBar(
            title: const Text('Home'),
          ),
          body: new SingleChildScrollView(
            child: new Column(
              children: <Widget>[
                new SizedBox(height: 160.0, width: double.infinity, child: new Card()),
                new SizedBox(height: 160.0, width: double.infinity, child: new Card()),
                new SizedBox(height: 160.0, width: double.infinity, child: new Card()),
                // destination
                new Card(
                  key: dataKey,
                  child: new Text("data\n\n\n\n\n\ndata"),
                )
              ],
            ),
          ),
          bottomNavigationBar: new RaisedButton(
            onPressed: () => Scrollable.ensureVisible(dataKey.currentContext),
            child: new Text("Scroll to data"),
          ),
        );
      }
    }
    

    NOTE : While this allows to scroll to the desired item easily, consider this method only for small predefined lists. As for bigger lists you'll get performance problems.

    But it's possible to make Scrollable.ensureVisible work with ListView ; although it will require more work.