我正在使用 SingleChildScrollView。它的 scrollDirection 设置为 Horizontal,在一个 Row Widget 中放置了 20 个以上的子 widget。我想以编程方式滚动到行中的位置小部件(即第 5 或第 6 位置)。有没有办法以编程方式做到这一点?
SingleChildScrollView(
scrollDirection: Axis.horizontal,
child: Row(
children: buttons,
),
)
最简单的方法是使用 Scrollable.ensureVisible.
ensureVisible 方法
滚动包含给定的可滚动对象 上下文以使给定的上下文可见。
请看下面的代码:
import 'package:flutter/material.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
debugShowCheckedModeBanner: false,
theme: ThemeData(
primarySwatch: Colors.blue,
),
home: const MyHomePage(title: 'Flutter Demo Home Page'),
);
}
}
class MyHomePage extends StatefulWidget {
const MyHomePage({Key key, this.title}) : super(key: key);
final String title;
@override
_MyHomePageState createState() => _MyHomePageState();
}
class _MyHomePageState extends State<MyHomePage> {
int _value = 0;
static final List<GlobalKey> _key = List.generate(20, (index) => GlobalKey());
final List<Widget> buttons = List.generate(
20,
(index) => RaisedButton(
onPressed: () {},
color: index % 2 == 0 ? Colors.grey : Colors.white,
child: Text("Button No # ${index + 1}", key: _key[index]),
),
);
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(widget.title),
),
body: Column(
children: [
SingleChildScrollView(
scrollDirection: Axis.horizontal,
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: buttons,
),
),
DropdownButton(
value: _value,
items: List.generate(
20,
(index) => DropdownMenuItem(
child: Text("Goto Button # ${index + 1}"), value: index),
),
onChanged: (value) {
setState(() {
_value = value;
print("calling");
Scrollable.ensureVisible(_key[value].currentContext);
});
},
)
],
),
);
}
}
你可以定义一个
ScrollController
:
ScrollController _controller = new ScrollController();
传递给
SingleChildScrollView
:
SingleChildScrollView(
controller: _scrollController,
scrollDirection: Axis.horizontal,
child: Row(
children: buttons,
),
),
并按如下方式以编程方式滚动它:
void scroll(double position) {
_scrollController.jumpTo(position);
}
或者,如果需要滚动动画:
void scrollAnimated(double position) {
_scrollController.animateTo(position, Duration(seconds: 1), Curves.ease);
}
如果你想在构建布局后立即自动滚动,你可以通过重写
initState
方法来实现:
@override
void initState() {
super.initState();
WidgetsBinding.instance
.addPostFrameCallback((_) => scroll(500)); // scroll automatically 500px (as an example)
}
您可以在 SingleChildScrollView 中添加 ScrollController 并滚动到您想要的特定位置
_scrollController.animateTo(
//here specifing position= 100 mean 100px
100,
curve: Curves.ease,
duration: Duration(seconds: 1),
这也可以在无状态小部件中完成,方法是在构建方法中触发它。使用
addPostFrameCallback
确保在渲染完成后完成滚动。您需要为选择的 value
添加状态管理。但随后您可以选择使用 GetX、StatefulWidget 或任何其他工具。
import 'package:flutter/material.dart';
void main() => runApp(const MyApp());
class MyApp extends StatelessWidget {
const MyApp({super.key});
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
debugShowCheckedModeBanner: false,
theme: ThemeData(
primarySwatch: Colors.blue,
),
home: MyHomePage(title: 'Flutter Demo Home Page'),
);
}
}
class MyHomePage extends StatelessWidget {
MyHomePage({Key? key, this.title}) : super(key: key);
final String? title;
static final List<GlobalKey> _key = List.generate(20, (index) => GlobalKey());
final List<Widget> buttons = List.generate(
20,
(index) => ElevatedButton(
onPressed: () {},
style: ButtonStyle(backgroundColor: MaterialStateProperty.resolveWith((_)=> index % 2 == 0 ? Colors.lightGreen : Colors.orange)),
child: Text("Button No # ${index + 1}", key: _key[index]),
),
);
@override
Widget build(BuildContext context) {
final thisWidget = Scaffold(
appBar: AppBar(
title: Text(title!),
),
body: Column(
children: [
SingleChildScrollView(
scrollDirection: Axis.horizontal,
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: buttons,
),
),
DropdownButton(
value: 0,
items: List.generate(
20,
(index) => DropdownMenuItem(
value: index,
child: Text("Goto Button # ${index + 1}")),
),
onChanged: (value) {
if (value != null) {
WidgetsBinding.instance
.addPostFrameCallback((_) => Scrollable.ensureVisible(_key[value].currentContext!));
}
},
)
],
),
);
WidgetsBinding.instance
.addPostFrameCallback((_) => Scrollable.ensureVisible(_key[15].currentContext!));
return thisWidget;
}
}