Commit 1b872603 authored by Adam Barth's avatar Adam Barth

Keyboard causes Scaffold body to resize (#3570)

This patch adds an option to Scaffold to disable resizing the body to avoid the
window padding. This lets developers create layouts that are stable when the
keyboard overlays the app.

Fixes #3565
parent 2781a086
...@@ -202,10 +202,27 @@ class _FloatingActionButtonTransitionState extends State<_FloatingActionButtonTr ...@@ -202,10 +202,27 @@ class _FloatingActionButtonTransitionState extends State<_FloatingActionButtonTr
/// Implements the basic material design visual layout structure. /// Implements the basic material design visual layout structure.
/// ///
/// This class provides APIs for showing drawers, snackbars, and bottom sheets. /// This class provides APIs for showing drawers, snack bars, and bottom sheets.
/// ///
/// See: <https://www.google.com/design/spec/layout/structure.html> /// To display a snackbar or a persistent bottom sheet, obtain the
/// [ScaffoldState] for the current [BuildContext] via [Scaffold.of] and use the
/// [ScaffoldState.showSnackBar] and [ScaffoldState.showBottomSheet] functions.
///
/// See also:
///
/// * [AppBar]
/// * [FloatingActionButton]
/// * [Drawer]
/// * [SnackBar]
/// * [BottomSheet]
/// * [ScaffoldState]
/// * <https://www.google.com/design/spec/layout/structure.html>
class Scaffold extends StatefulWidget { class Scaffold extends StatefulWidget {
/// Creates a visual scaffold for material design widgets.
///
/// By default, the [appBarBehavior] causes the [appBar] not to respond to
/// scrolling and the [body] is resized to avoid the window padding (e.g., to
/// to avoid being obscured by an onscreen keyboard).
Scaffold({ Scaffold({
Key key, Key key,
this.appBar, this.appBar,
...@@ -213,18 +230,52 @@ class Scaffold extends StatefulWidget { ...@@ -213,18 +230,52 @@ class Scaffold extends StatefulWidget {
this.floatingActionButton, this.floatingActionButton,
this.drawer, this.drawer,
this.scrollableKey, this.scrollableKey,
this.appBarBehavior: AppBarBehavior.anchor this.appBarBehavior: AppBarBehavior.anchor,
this.resizeToAvoidWindowPadding: true
}) : super(key: key) { }) : super(key: key) {
assert(scrollableKey != null ? (appBarBehavior != AppBarBehavior.anchor) : true); assert(scrollableKey != null ? (appBarBehavior != AppBarBehavior.anchor) : true);
} }
/// An app bar to display at the top of the scaffold.
final AppBar appBar; final AppBar appBar;
/// The primary content of the scaffold.
///
/// Displayed below the app bar and behind the [floatingActionButton] and
/// [drawer]. To avoid the body being resized to avoid the window padding
/// (e.g., from the onscreen keyboard), see [resizeToAvoidWindowPadding].
final Widget body; final Widget body;
/// A button displayed on top of the body.
///
/// Typically a [FloatingActionButton].
final Widget floatingActionButton; final Widget floatingActionButton;
/// A panel displayed to the side of the body, often hidden on mobile devices.
///
/// Typically a [Drawer].
final Widget drawer; final Widget drawer;
/// The key of the primary [Scrollable] widget in the [body].
///
/// Used to control scroll-linked effects, such as the collapse of the
/// [appBar].
final Key scrollableKey; final Key scrollableKey;
/// How the [appBar] should respond to scrolling.
///
/// By default, the [appBar] does not respond to scrolling.
final AppBarBehavior appBarBehavior; final AppBarBehavior appBarBehavior;
/// Whether the [body] (and other floating widgets) should size themselves to avoid the window's padding.
///
/// For example, if there is an onscreen keyboard displayed above the
/// scaffold, the body can be resized to avoid overlapping the keyboard, which
/// prevents widgets inside the body from being obscured by the keyboard.
///
/// Defaults to true.
final bool resizeToAvoidWindowPadding;
/// The state from the closest instance of this class that encloses the given context. /// The state from the closest instance of this class that encloses the given context.
static ScaffoldState of(BuildContext context) => context.ancestorStateOfType(const TypeMatcher<ScaffoldState>()); static ScaffoldState of(BuildContext context) => context.ancestorStateOfType(const TypeMatcher<ScaffoldState>());
...@@ -232,6 +283,10 @@ class Scaffold extends StatefulWidget { ...@@ -232,6 +283,10 @@ class Scaffold extends StatefulWidget {
ScaffoldState createState() => new ScaffoldState(); ScaffoldState createState() => new ScaffoldState();
} }
/// State for a [Scaffold].
///
/// Can display [SnackBar]s and [BottomSheet]s. Retrieve a [ScaffoldState] from
/// the current [BuildContext] using [Scaffold.of].
class ScaffoldState extends State<Scaffold> { class ScaffoldState extends State<Scaffold> {
// APPBAR API // APPBAR API
...@@ -510,8 +565,7 @@ class ScaffoldState extends State<Scaffold> { ...@@ -510,8 +565,7 @@ class ScaffoldState extends State<Scaffold> {
); );
} }
Widget _buildScrollableAppBar(BuildContext context) { Widget _buildScrollableAppBar(BuildContext context, EdgeInsets padding) {
final EdgeInsets padding = MediaQuery.of(context).padding;
final double expandedHeight = (config.appBar?.expandedHeight ?? 0.0) + padding.top; final double expandedHeight = (config.appBar?.expandedHeight ?? 0.0) + padding.top;
final double collapsedHeight = (config.appBar?.collapsedHeight ?? 0.0) + padding.top; final double collapsedHeight = (config.appBar?.collapsedHeight ?? 0.0) + padding.top;
final double minimumHeight = (config.appBar?.minimumHeight ?? 0.0) + padding.top; final double minimumHeight = (config.appBar?.minimumHeight ?? 0.0) + padding.top;
...@@ -558,7 +612,9 @@ class ScaffoldState extends State<Scaffold> { ...@@ -558,7 +612,9 @@ class ScaffoldState extends State<Scaffold> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final EdgeInsets padding = MediaQuery.of(context).padding; EdgeInsets padding = MediaQuery.of(context).padding;
if (!config.resizeToAvoidWindowPadding)
padding = new EdgeInsets.only(top: padding.top);
if (_snackBars.length > 0) { if (_snackBars.length > 0) {
final ModalRoute<dynamic> route = ModalRoute.of(context); final ModalRoute<dynamic> route = ModalRoute.of(context);
...@@ -581,7 +637,7 @@ class ScaffoldState extends State<Scaffold> { ...@@ -581,7 +637,7 @@ class ScaffoldState extends State<Scaffold> {
); );
_addIfNonNull(children, appBar, _ScaffoldSlot.appBar); _addIfNonNull(children, appBar, _ScaffoldSlot.appBar);
} else { } else {
children.add(new LayoutId(child: _buildScrollableAppBar(context), id: _ScaffoldSlot.appBar)); children.add(new LayoutId(child: _buildScrollableAppBar(context, padding), id: _ScaffoldSlot.appBar));
} }
// Otherwise the AppBar will be part of a [app bar, body] Stack. See AppBarBehavior.scroll below. // Otherwise the AppBar will be part of a [app bar, body] Stack. See AppBarBehavior.scroll below.
......
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import 'package:flutter_test/flutter_test.dart';
import 'package:flutter/material.dart';
import 'package:flutter/rendering.dart';
import 'package:test/test.dart';
void main() {
test('Scaffold control test', () {
testWidgets((WidgetTester tester) {
Key bodyKey = new UniqueKey();
tester.pumpWidget(new Scaffold(
appBar: new AppBar(title: new Text('Title')),
body: new Container(key: bodyKey)
));
RenderBox bodyBox = tester.renderObjectOf(find.byKey(bodyKey));
expect(bodyBox.size, equals(new Size(800.0, 544.0)));
tester.pumpWidget(new MediaQuery(
data: new MediaQueryData(padding: new EdgeInsets.only(bottom: 100.0)),
child: new Scaffold(
appBar: new AppBar(title: new Text('Title')),
body: new Container(key: bodyKey)
)
));
bodyBox = tester.renderObjectOf(find.byKey(bodyKey));
expect(bodyBox.size, equals(new Size(800.0, 444.0)));
tester.pumpWidget(new MediaQuery(
data: new MediaQueryData(padding: new EdgeInsets.only(bottom: 100.0)),
child: new Scaffold(
appBar: new AppBar(title: new Text('Title')),
body: new Container(key: bodyKey),
resizeToAvoidWindowPadding: false
)
));
bodyBox = tester.renderObjectOf(find.byKey(bodyKey));
expect(bodyBox.size, equals(new Size(800.0, 544.0)));
});
});
}
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