listviewflutter

Flutter: Listview get full size of scrollcontroller after adding item to list & scroll to end


This post describes a very similar problem, but the answer there doesn't solve all problems:

I have a potentially long List, where the user can add new items (on at a time). After/On add, the list should scroll to its end.

(Btw no, reverse: true is not an option)

After reading the other post, I understood using SchedulerBinding.instance.addPostFrameCallback((_) => scrollToEnd()); should to the trick b/c the new lists maxScrollExtent will be correct.

But it doesn't work reliably: When already scrolled to the end of the list or near the end everything's ok. But when the list is scrolled to its start (or some way from the end) when adding a new item, the list gets scrolled, but the scrollposition is off by exactly one item - the newest one.

I think it might have something to do with the ListView.builder not keeping all children alive - but how to solve it?

Oh and bonus question: just discovered another very strange behaviour: after adding two items the last one is a little bit out of view but the list isn't scrollable - which is strange. But even stranger is that on the next add-item-click the list scrolls this tiny bit - but without ever creating the new item!?

Here a complete example:

import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/scheduler.dart';

void main() {
  runApp(MyList());
}

class MyList extends StatefulWidget {
  MyList({Key key}) : super(key: key);

  @override
  _MyListState createState() => _MyListState();
}

var items = List<String>.generate(8, (i) => "Item $i");

class _MyListState extends State<MyList> {
  static ScrollController _scrollController = ScrollController();

  void add() {
    setState(() {
      items.add("new Item ${items.length}");
      print(items.length);
    });
    SchedulerBinding.instance.addPostFrameCallback((_) => scrollToEnd());
  }

  void scrollToEnd() {
    _scrollController.animateTo(_scrollController.position.maxScrollExtent,
        duration: const Duration(milliseconds: 350), curve: Curves.easeOut);
  }

  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: "List",
      home: Scaffold(
          appBar: AppBar(
            title: Text("List"),
          ),
          body: ListView.builder(
            controller: _scrollController,
            itemCount: items.length,
            shrinkWrap: true,
            itemBuilder: (context, index) {
              return ListTile(
                title: Text('${items[index]}'),
              );
            },
          ),
          bottomSheet: Container(
              decoration: BoxDecoration(
                  border:
                      Border(top: BorderSide(color: Colors.black, width: 1))),
              child: Row(
                mainAxisAlignment: MainAxisAlignment.end,
                children: [
                  FloatingActionButton(
                    onPressed: () {
                      add();
                    },
                    child: Icon(Icons.add),
                  )
                ],
              ))),
    );
  }
}

Solution

  • I combined scroll to maxScrollExtent with Scrollable.ensureVisible and each of them fixed the flaws of the other.

    import 'package:flutter/material.dart';
    import 'package:flutter/scheduler.dart';
    
    class MyList extends StatefulWidget {
      MyList({Key key}) : super(key: key);
    
      @override
      _MyListState createState() => _MyListState();
    }
    
    class _MyListState extends State<MyList> {
      final ScrollController _scrollController = ScrollController();
      final lastKey = GlobalKey();
      List<String> items;
    
      @override
      void initState() {
        super.initState();
        items = List<String>.generate(8, (i) => "Item $i");
      }
    
      void add() {
        setState(() {
          items.add("new Item ${items.length}");
        });
        SchedulerBinding.instance.addPostFrameCallback((_) => scrollToEnd());
      }
    
      void scrollToEnd() async {
        await _scrollController.animateTo(
            _scrollController.position.maxScrollExtent,
            duration: const Duration(milliseconds: 350),
            curve: Curves.easeOut);
        Scrollable.ensureVisible(lastKey.currentContext);
      }
    
      @override
      Widget build(BuildContext context) {
        return MaterialApp(
            title: "List",
            home: Scaffold(
              body: ListView.builder(
                controller: _scrollController,
                itemCount: items.length,
                shrinkWrap: true,
                itemBuilder: (context, index) {
                  return ListTile(
                    title: Text('${items[index]}'),
                    key: index == items.length - 1 ? lastKey : null,
                  );
                },
              ),
              floatingActionButton: FloatingActionButton(
                onPressed: () {
                  add();
                },
                child: Icon(Icons.add),
              ),
            ));
      }
    }
    

    Scrollable.ensureVisible itself cannot provide visibility if the item has not yet been created, but copes with them when item is very close.