viewport.dart 11.5 KB
Newer Older
1 2 3 4 5 6 7
// 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/foundation.dart';
import 'package:flutter/rendering.dart';

8
import 'basic.dart';
9 10 11 12 13 14
import 'framework.dart';

export 'package:flutter/rendering.dart' show
  AxisDirection,
  GrowthDirection;

15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
AxisDirection _getDefaultCrossAxisDirection(BuildContext context, AxisDirection axisDirection) {
  assert(axisDirection != null);
  switch (axisDirection) {
    case AxisDirection.up:
      return textDirectionToAxisDirection(Directionality.of(context));
    case AxisDirection.right:
      return AxisDirection.down;
    case AxisDirection.down:
      return textDirectionToAxisDirection(Directionality.of(context));
    case AxisDirection.left:
      return AxisDirection.down;
  }
  return null;
}

30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60
/// A widget that is bigger on the inside.
///
/// [Viewport] is the visual workhorse of the scrolling machinery. It displays a
/// subset of its children according to its own dimensions and the given
/// [offset]. As the offset varies, different children are visible through
/// the viewport.
///
/// [Viewport] hosts a bidirectional list of slivers, anchored on a [center]
/// sliver, which is placed at the zero scroll offset. The center widget is
/// displayed in the viewport according to the [anchor] property.
///
/// Slivers that are earlier in the child list than [center] are displayed in
/// reverse order in the reverse [axisDirection] starting from the [center]. For
/// example, if the [axisDirection] is [AxisDirection.down], the first sliver
/// before [center] is placed above the [center]. The slivers that are later in
/// the child list than [center] are placed in order in the [axisDirection]. For
/// example, in the preceeding scenario, the first sliver after [center] is
/// placed below the [center].
///
/// [Viewport] cannot contain box children directly. Instead, use a
/// [SliverList], [SliverFixedExtentList], [SliverGrid], or a
/// [SliverToBoxAdapter], for example.
///
/// See also:
///
///  * [ListView], [PageView], [GridView], and [CustomScrollView], which combine
///    [Scrollable] and [Viewport] into widgets that are easier to use.
///  * [SliverToBoxAdapter], which allows a box widget to be placed inside a
///    sliver context (the opposite of this widget).
///  * [ShrinkWrappingViewport], a variant of [Viewport] that shrink-wraps its
///    contents along the main axis.
Adam Barth's avatar
Adam Barth committed
61
class Viewport extends MultiChildRenderObjectWidget {
62 63 64 65 66 67
  /// Creates a widget that is bigger on the inside.
  ///
  /// The viewport listens to the [offset], which means you do not need to
  /// rebuild this widget when the [offset] changes.
  ///
  /// The [offset] argument must not be null.
Adam Barth's avatar
Adam Barth committed
68
  Viewport({
69 70
    Key key,
    this.axisDirection: AxisDirection.down,
71
    this.crossAxisDirection,
72
    this.anchor: 0.0,
Ian Hickson's avatar
Ian Hickson committed
73
    @required this.offset,
74
    this.center,
75
    List<Widget> slivers: const <Widget>[],
76 77 78 79
  }) : assert(offset != null),
       assert(slivers != null),
       assert(center == null || slivers.where((Widget child) => child.key == center).length == 1),
       super(key: key, children: slivers);
80

81
  /// The direction in which the [offset]'s [ViewportOffset.pixels] increases.
82 83 84 85
  ///
  /// For example, if the [axisDirection] is [AxisDirection.down], a scroll
  /// offset of zero is at the top of the viewport and increases towards the
  /// bottom of the viewport.
86
  final AxisDirection axisDirection;
87

88 89 90 91 92 93 94 95 96 97 98
  /// The direction in which child should be laid out in the cross axis.
  ///
  /// If the [axisDirection] is [AxisDirection.down] or [AxisDirection.up], this
  /// property defaults to [AxisDirection.left] if the ambient [Directionality]
  /// is [TextDirection.rtl] and [AxisDirection.right] if the ambient
  /// [Directionality] is [TextDirection.ltr].
  ///
  /// If the [axisDirection] is [AxisDirection.left] or [AxisDirection.right],
  /// this property defaults to [AxisDirection.down].
  final AxisDirection crossAxisDirection;

99 100 101 102 103 104 105
  /// The relative position of the zero scroll offset.
  ///
  /// For example, if [anchor] is 0.5 and the [axisDirection] is
  /// [AxisDirection.down] or [AxisDirection.up], then the zero scroll offset is
  /// vertically centered within the viewport. If the [anchor] is 1.0, and the
  /// [axisDirection] is [AxisDirection.right], then the zero scroll offset is
  /// on the left edge of the viewport.
106
  final double anchor;
107 108 109 110 111 112 113 114 115

  /// Which part of the content inside the viewport should be visible.
  ///
  /// The [ViewportOffset.pixels] value determines the scroll offset that the
  /// viewport uses to select which part of its content to display. As the user
  /// scrolls the viewport, this value changes, which changes the content that
  /// is displayed.
  ///
  /// Typically a [ScrollPosition].
116
  final ViewportOffset offset;
117 118 119 120 121 122 123 124

  /// The first child in the [GrowthDirection.forward] growth direction.
  ///
  /// Children after [center] will be placed in the [axisDirection] relative to
  /// the [center]. Children before [center] will be placed in the opposite of
  /// the [axisDirection] relative to the [center].
  ///
  /// The [center] must be the key of a child of the viewport.
125 126 127
  final Key center;

  @override
Adam Barth's avatar
Adam Barth committed
128 129
  RenderViewport createRenderObject(BuildContext context) {
    return new RenderViewport(
130
      axisDirection: axisDirection,
131
      crossAxisDirection: crossAxisDirection ?? _getDefaultCrossAxisDirection(context, axisDirection),
132 133 134 135 136 137
      anchor: anchor,
      offset: offset,
    );
  }

  @override
Adam Barth's avatar
Adam Barth committed
138
  void updateRenderObject(BuildContext context, RenderViewport renderObject) {
139 140 141 142 143
    renderObject
      ..axisDirection = axisDirection
      ..crossAxisDirection = crossAxisDirection ?? _getDefaultCrossAxisDirection(context, axisDirection)
      ..anchor = anchor
      ..offset = offset;
144 145 146
  }

  @override
147
  _ViewportElement createElement() => new _ViewportElement(this);
148 149

  @override
150
  void debugFillProperties(DiagnosticPropertiesBuilder description) {
151 152
    super.debugFillProperties(description);
    description.add(new EnumProperty<AxisDirection>('axisDirection', axisDirection));
153
    description.add(new EnumProperty<AxisDirection>('crossAxisDirection', crossAxisDirection, defaultValue: null));
154 155
    description.add(new DoubleProperty('anchor', anchor));
    description.add(new DiagnosticsProperty<ViewportOffset>('offset', offset));
156
    if (center != null) {
157
      description.add(new DiagnosticsProperty<Key>('center', center));
158
    } else if (children.isNotEmpty && children.first.key != null) {
159
      description.add(new DiagnosticsProperty<Key>('center', children.first.key, tooltip: 'implicit'));
160 161 162 163
    }
  }
}

164
class _ViewportElement extends MultiChildRenderObjectElement {
165
  /// Creates an element that uses the given widget as its configuration.
166
  _ViewportElement(Viewport widget) : super(widget);
167 168

  @override
Adam Barth's avatar
Adam Barth committed
169
  Viewport get widget => super.widget;
170 171

  @override
Adam Barth's avatar
Adam Barth committed
172
  RenderViewport get renderObject => super.renderObject;
173 174 175 176

  @override
  void mount(Element parent, dynamic newSlot) {
    super.mount(parent, newSlot);
177
    _updateCenter();
178 179 180 181 182
  }

  @override
  void update(MultiChildRenderObjectWidget newWidget) {
    super.update(newWidget);
183
    _updateCenter();
184 185
  }

186
  void _updateCenter() {
187 188 189 190 191 192 193 194 195 196 197 198
    // TODO(ianh): cache the keys to make this faster
    if (widget.center != null) {
      renderObject.center = children.singleWhere(
        (Element element) => element.widget.key == widget.center
      ).renderObject;
    } else if (children.isNotEmpty) {
      renderObject.center = children.first.renderObject;
    } else {
      renderObject.center = null;
    }
  }
}
199

200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224
/// A widget that is bigger on the inside and shrink wraps its children in the
/// main axis.
///
/// [ShrinkWrappingViewport] displays a subset of its children according to its
/// own dimensions and the given [offset]. As the offset varies, different
/// children are visible through the viewport.
///
/// [ShrinkWrappingViewport] differs from [Viewport] in that [Viewport] expands
/// to fill the main axis whereas [ShrinkWrappingViewport] sizes itself to match
/// its children in the main axis. This shrink wrapping behavior is expensive
/// because the children, and hence the viewport, could potentially change size
/// whenever the [offset] changes (e.g., because of a collapsing header).
///
/// [ShrinkWrappingViewport] cannot contain box children directly. Instead, use
/// a [SliverList], [SliverFixedExtentList], [SliverGrid], or a
/// [SliverToBoxAdapter], for example.
///
/// See also:
///
///  * [ListView], [PageView], [GridView], and [CustomScrollView], which combine
///    [Scrollable] and [ShrinkWrappingViewport] into widgets that are easier to
///    use.
///  * [SliverToBoxAdapter], which allows a box widget to be placed inside a
///    sliver context (the opposite of this widget).
///  * [Viewport], a viewport that does not shrink-wrap its contents
225
class ShrinkWrappingViewport extends MultiChildRenderObjectWidget {
226 227 228 229 230 231 232
  /// Creates a widget that is bigger on the inside and shrink wraps its
  /// children in the main axis.
  ///
  /// The viewport listens to the [offset], which means you do not need to
  /// rebuild this widget when the [offset] changes.
  ///
  /// The [offset] argument must not be null.
233 234 235
  ShrinkWrappingViewport({
    Key key,
    this.axisDirection: AxisDirection.down,
236
    this.crossAxisDirection,
237 238
    @required this.offset,
    List<Widget> slivers: const <Widget>[],
239 240
  }) : assert(offset != null),
       super(key: key, children: slivers);
241

242
  /// The direction in which the [offset]'s [ViewportOffset.pixels] increases.
243 244 245 246
  ///
  /// For example, if the [axisDirection] is [AxisDirection.down], a scroll
  /// offset of zero is at the top of the viewport and increases towards the
  /// bottom of the viewport.
247
  final AxisDirection axisDirection;
248

249 250 251 252 253 254 255 256 257 258 259
  /// The direction in which child should be laid out in the cross axis.
  ///
  /// If the [axisDirection] is [AxisDirection.down] or [AxisDirection.up], this
  /// property defaults to [AxisDirection.left] if the ambient [Directionality]
  /// is [TextDirection.rtl] and [AxisDirection.right] if the ambient
  /// [Directionality] is [TextDirection.ltr].
  ///
  /// If the [axisDirection] is [AxisDirection.left] or [AxisDirection.right],
  /// this property defaults to [AxisDirection.down].
  final AxisDirection crossAxisDirection;

260 261 262 263 264 265 266 267
  /// Which part of the content inside the viewport should be visible.
  ///
  /// The [ViewportOffset.pixels] value determines the scroll offset that the
  /// viewport uses to select which part of its content to display. As the user
  /// scrolls the viewport, this value changes, which changes the content that
  /// is displayed.
  ///
  /// Typically a [ScrollPosition].
268 269 270 271 272 273
  final ViewportOffset offset;

  @override
  RenderShrinkWrappingViewport createRenderObject(BuildContext context) {
    return new RenderShrinkWrappingViewport(
      axisDirection: axisDirection,
274
      crossAxisDirection: crossAxisDirection ?? _getDefaultCrossAxisDirection(context, axisDirection),
275 276 277 278 279 280 281 282
      offset: offset,
    );
  }

  @override
  void updateRenderObject(BuildContext context, RenderShrinkWrappingViewport renderObject) {
    renderObject
      ..axisDirection = axisDirection
283
      ..crossAxisDirection = crossAxisDirection ?? _getDefaultCrossAxisDirection(context, axisDirection)
284 285 286 287
      ..offset = offset;
  }

  @override
288
  void debugFillProperties(DiagnosticPropertiesBuilder description) {
289 290
    super.debugFillProperties(description);
    description.add(new EnumProperty<AxisDirection>('axisDirection', axisDirection));
291
    description.add(new EnumProperty<AxisDirection>('crossAxisDirection', crossAxisDirection, defaultValue: null));
292
    description.add(new DiagnosticsProperty<ViewportOffset>('offset', offset));
293 294
  }
}