这篇文章描述了一个非常相似的问题,但答案并不能解决所有问题:
我有一个可能很长的列表,用户可以在其中添加新项目(一次打开)。添加后/添加时,列表应滚动到末尾。
(顺便说一句,不,reverse: true不是一个选项)
阅读另一篇文章后,我明白使用SchedulerBinding.instance.addPostFrameCallback((_) => scrollToEnd());should 的技巧 b/c 新列表maxScrollExtent将是正确的。
但它不能可靠地工作:当已经滚动到列表末尾或接近末尾时,一切正常。但是,当添加新项目时列表滚动到其开头(或从结尾的某种方式)时,列表会滚动,但滚动位置恰好偏离了一项 - 最新的一项。
我认为这可能与未能让所有孩子活下来有关ListView.builder- 但如何解决呢?
哦,还有一个额外的问题:刚刚发现了另一个非常奇怪的行为:添加两个项目后,最后一个项目有点看不到,但列表不可滚动 - 这很奇怪。但更奇怪的是,在下一个添加项目单击时,列表会滚动一点点 - 但从未创建新项目!?
这是一个完整的例子:
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),
)
],
))),
);
}
}
Run Code Online (Sandbox Code Playgroud)
我将滚动到maxScrollExtent与它们结合起来Scrollable.ensureVisible,每个都修复了另一个的缺陷。
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),
),
));
}
}
Run Code Online (Sandbox Code Playgroud)
Scrollable.ensureVisible如果项目尚未创建,则其本身无法提供可见性,但当项目非常接近时可以处理它们。
| 归档时间: |
|
| 查看次数: |
4901 次 |
| 最近记录: |