Unverified Commit 2b2bbfa6 authored by Ethan Saadia's avatar Ethan Saadia Committed by GitHub

Add option for ExpansionTile to maintain the state of its children when collapsed (#57172)

parent 859f2f9b
...@@ -41,10 +41,12 @@ class ExpansionTile extends StatefulWidget { ...@@ -41,10 +41,12 @@ class ExpansionTile extends StatefulWidget {
this.children = const <Widget>[], this.children = const <Widget>[],
this.trailing, this.trailing,
this.initiallyExpanded = false, this.initiallyExpanded = false,
this.maintainState = false,
this.tilePadding, this.tilePadding,
this.expandedCrossAxisAlignment, this.expandedCrossAxisAlignment,
this.expandedAlignment, this.expandedAlignment,
}) : assert(initiallyExpanded != null), }) : assert(initiallyExpanded != null),
assert(maintainState != null),
assert( assert(
expandedCrossAxisAlignment != CrossAxisAlignment.baseline, expandedCrossAxisAlignment != CrossAxisAlignment.baseline,
'CrossAxisAlignment.baseline is not supported since the expanded children ' 'CrossAxisAlignment.baseline is not supported since the expanded children '
...@@ -88,6 +90,13 @@ class ExpansionTile extends StatefulWidget { ...@@ -88,6 +90,13 @@ class ExpansionTile extends StatefulWidget {
/// Specifies if the list tile is initially expanded (true) or collapsed (false, the default). /// Specifies if the list tile is initially expanded (true) or collapsed (false, the default).
final bool initiallyExpanded; final bool initiallyExpanded;
/// Specifies whether the state of the children is maintained when the tile expands and collapses.
///
/// When true, the children are kept in the tree while the tile is collapsed.
/// When false (default), the children are removed from the tree when the tile is
/// collapsed and recreated upon expansion.
final bool maintainState;
/// Specifies padding for the [ListTile]. /// Specifies padding for the [ListTile].
/// ///
/// Analogous to [ListTile.contentPadding], this property defines the insets for /// Analogous to [ListTile.contentPadding], this property defines the insets for
...@@ -253,13 +262,23 @@ class _ExpansionTileState extends State<ExpansionTile> with SingleTickerProvider ...@@ -253,13 +262,23 @@ class _ExpansionTileState extends State<ExpansionTile> with SingleTickerProvider
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final bool closed = !_isExpanded && _controller.isDismissed; final bool closed = !_isExpanded && _controller.isDismissed;
final bool shouldRemoveChildren = closed && !widget.maintainState;
final Widget result = Offstage(
child: TickerMode(
child: Column(
crossAxisAlignment: widget.expandedCrossAxisAlignment ?? CrossAxisAlignment.center,
children: widget.children,
),
enabled: !closed,
),
offstage: closed
);
return AnimatedBuilder( return AnimatedBuilder(
animation: _controller.view, animation: _controller.view,
builder: _buildChildren, builder: _buildChildren,
child: closed ? null : Column( child: shouldRemoveChildren ? null : result,
crossAxisAlignment: widget.expandedCrossAxisAlignment ?? CrossAxisAlignment.center,
children: widget.children,
),
); );
} }
......
...@@ -230,6 +230,46 @@ void main() { ...@@ -230,6 +230,46 @@ void main() {
expect(find.text('Subtitle'), findsOneWidget); expect(find.text('Subtitle'), findsOneWidget);
}); });
testWidgets('ExpansionTile maintainState', (WidgetTester tester) async {
await tester.pumpWidget(
MaterialApp(
theme: ThemeData(
platform: TargetPlatform.iOS,
dividerColor: _dividerColor,
),
home: Material(
child: SingleChildScrollView(
child: Column(
children: const <Widget>[
ExpansionTile(
title: Text('Tile 1'),
initiallyExpanded: false,
maintainState: true,
children: <Widget>[
Text('Maintaining State'),
],
),
ExpansionTile(
title: Text('Title 2'),
initiallyExpanded: false,
maintainState: false,
children: <Widget>[
Text('Discarding State'),
],
),
],
),
),
),
));
// This text should be offstage while ExpansionTile collapsed
expect(find.text('Maintaining State', skipOffstage: false), findsOneWidget);
expect(find.text('Maintaining State'), findsNothing);
// This text shouldn't be there while ExpansionTile collapsed
expect(find.text('Discarding State'), findsNothing);
});
testWidgets('ExpansionTile padding test', (WidgetTester tester) async { testWidgets('ExpansionTile padding test', (WidgetTester tester) async {
await tester.pumpWidget(const MaterialApp( await tester.pumpWidget(const MaterialApp(
home: Material( home: Material(
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment