首页 > 解决方案 > 在 setState 渲染后执行函数

问题描述

在我的应用程序中,我想在添加项目后滚动到 ListView 的底部。问题是它在添加项目之前向下滚动,因此隐藏了一个项目。有没有办法在项目渲染后调用滚动函数?

示例代码:

class MyHomePage extends StatefulWidget {
  @override
  _MyHomePageState createState() => _MyHomePageState();
}

class _MyHomePageState extends State<MyHomePage> {
  var _items = [];
  var _counter = 0;

  final _listViewController = ScrollController();

  void _addItem() {
    setState(() {
      _items.add(++_counter);
    });
    _listViewController.jumpTo(_listViewController.position.maxScrollExtent);
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text('Demo'),
      ),
      body: ListView(
        padding: const EdgeInsets.all(16.0),
        controller: _listViewController,
        children: [
          ..._items.map((i) => ListTile(title: Text(i.toString()))),
        ],
      ),
      floatingActionButton: FloatingActionButton(
        onPressed: _addItem,
        child: Icon(Icons.add),
      ),
    );
  }
}

标签: flutterdart

解决方案


使用 WidgetsBinding.instance.addPersistentFrameCallback :

https://api.flutter.dev/flutter/scheduler/SchedulerBinding/addPersistentFrameCallback.html

  bool added = false;

  void initState() {
    WidgetsBinding.instance.addPersistentFrameCallback(jumpToDown);

    super.initState();
  }

  void _addItem() {
    setState(() {
      _items.add(++_counter);
      added = true;
    });
  }

  void jumpToDown([nothing]) {
    if (added) {
      _listViewController.jumpTo(_listViewController.position.maxScrollExtent);
    }
    added = false;
  }

推荐阅读