flow_test.dart 6.28 KB
Newer Older
Ian Hickson's avatar
Ian Hickson committed
1
// Copyright 2014 The Flutter Authors. All rights reserved.
Adam Barth's avatar
Adam Barth committed
2 3 4
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

5
import 'package:flutter/rendering.dart';
6 7
import 'package:flutter/widgets.dart';
import 'package:flutter_test/flutter_test.dart';
Adam Barth's avatar
Adam Barth committed
8 9

class TestFlowDelegate extends FlowDelegate {
10
  TestFlowDelegate({required this.startOffset}) : super(repaint: startOffset);
Adam Barth's avatar
Adam Barth committed
11 12 13 14 15 16 17 18 19 20 21 22

  final Animation<double> startOffset;

  @override
  BoxConstraints getConstraintsForChild(int i, BoxConstraints constraints) {
    return constraints.loosen();
  }

  @override
  void paintChildren(FlowPaintingContext context) {
    double dy = startOffset.value;
    for (int i = 0; i < context.childCount; ++i) {
23
      context.paintChild(i, transform: Matrix4.translationValues(0.0, dy, 0.0));
24
      dy += 0.75 * context.getChildSize(i)!.height;
Adam Barth's avatar
Adam Barth committed
25 26 27 28 29 30 31
    }
  }

  @override
  bool shouldRepaint(TestFlowDelegate oldDelegate) => startOffset == oldDelegate.startOffset;
}

32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47
class OpacityFlowDelegate extends FlowDelegate {
  OpacityFlowDelegate(this.opacity);

  double opacity;

  @override
  void paintChildren(FlowPaintingContext context) {
    for (int i = 0; i < context.childCount; ++i) {
      context.paintChild(i, opacity: opacity);
    }
  }

  @override
  bool shouldRepaint(OpacityFlowDelegate oldDelegate) => opacity != oldDelegate.opacity;
}

48 49
// OpacityFlowDelegate that paints one of its children twice
class DuplicatePainterOpacityFlowDelegate extends OpacityFlowDelegate {
50
  DuplicatePainterOpacityFlowDelegate(super.opacity);
51 52 53 54 55 56 57 58 59 60 61 62

  @override
  void paintChildren(FlowPaintingContext context) {
    for (int i = 0; i < context.childCount; ++i) {
      context.paintChild(i, opacity: opacity);
    }
    if (context.childCount > 0) {
      context.paintChild(0, opacity: opacity);
    }
  }
}

Adam Barth's avatar
Adam Barth committed
63
void main() {
64
  testWidgets('Flow control test', (WidgetTester tester) async {
65
    final AnimationController startOffset = AnimationController.unbounded(
66 67
      vsync: tester,
    );
68
    final List<int> log = <int>[];
Adam Barth's avatar
Adam Barth committed
69

70
    Widget buildBox(int i) {
71
      return GestureDetector(
72 73 74
        onTap: () {
          log.add(i);
        },
75
        child: Container(
76 77
          width: 100.0,
          height: 100.0,
78
          color: const Color(0xFF0000FF),
79 80
          child: Text('$i', textDirection: TextDirection.ltr),
        ),
Adam Barth's avatar
Adam Barth committed
81
      );
82 83
    }

84
    await tester.pumpWidget(
85 86
      Flow(
        delegate: TestFlowDelegate(startOffset: startOffset),
87 88 89 90 91 92 93 94
        children: <Widget>[
          buildBox(0),
          buildBox(1),
          buildBox(2),
          buildBox(3),
          buildBox(4),
          buildBox(5),
          buildBox(6),
95
        ],
96
      ),
97
    );
Adam Barth's avatar
Adam Barth committed
98

99
    await tester.tap(find.text('0'));
100
    expect(log, equals(<int>[0]));
101
    await tester.tap(find.text('1'));
102
    expect(log, equals(<int>[0, 1]));
103
    await tester.tap(find.text('2'));
104
    expect(log, equals(<int>[0, 1, 2]));
Adam Barth's avatar
Adam Barth committed
105

106
    log.clear();
107
    await tester.tapAt(const Offset(20.0, 90.0));
108
    expect(log, equals(<int>[1]));
Adam Barth's avatar
Adam Barth committed
109

110
    startOffset.value = 50.0;
111
    await tester.pump();
Adam Barth's avatar
Adam Barth committed
112

113
    log.clear();
114
    await tester.tapAt(const Offset(20.0, 90.0));
115
    expect(log, equals(<int>[0]));
Adam Barth's avatar
Adam Barth committed
116
  });
117

118 119 120 121
  testWidgets('paintChild gets called twice', (WidgetTester tester) async {
    await tester.pumpWidget(
      Flow(
        delegate: DuplicatePainterOpacityFlowDelegate(1.0),
122 123 124
        children: const <Widget>[
          SizedBox(width: 100.0, height: 100.0),
          SizedBox(width: 100.0, height: 100.0),
125 126 127 128 129
        ],
      ),
    );
    final dynamic exception = tester.takeException();
    expect(exception, isFlutterError);
130
    final FlutterError error = exception as FlutterError;
131 132 133 134 135
    expect(error.toStringDeep(), equalsIgnoringHashCodes(
      'FlutterError\n'
      '   Cannot call paintChild twice for the same child.\n'
      '   The flow delegate of type DuplicatePainterOpacityFlowDelegate\n'
      '   attempted to paint child 0 multiple times, which is not\n'
136
      '   permitted.\n',
137 138 139
    ));
  });

140 141 142
  testWidgets('Flow opacity layer', (WidgetTester tester) async {
    const double opacity = 0.2;
    await tester.pumpWidget(
143 144
      Flow(
        delegate: OpacityFlowDelegate(opacity),
145 146
        children: const <Widget>[
          SizedBox(width: 100.0, height: 100.0),
147
        ],
148
      ),
149
    );
150
    ContainerLayer? layer = RendererBinding.instance.renderView.debugLayer;
151
    while (layer != null && layer is! OpacityLayer)
152
      layer = layer.firstChild as ContainerLayer?;
Dan Field's avatar
Dan Field committed
153
    expect(layer, isA<OpacityLayer>());
154 155 156
    final OpacityLayer? opacityLayer = layer as OpacityLayer?;
    expect(opacityLayer!.alpha, equals(opacity * 255));
    expect(layer!.firstChild, isA<TransformLayer>());
157
  });
158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177

  testWidgets('Flow can set and update clipBehavior', (WidgetTester tester) async {
    const double opacity = 0.2;
    await tester.pumpWidget(
      Flow(
        delegate: OpacityFlowDelegate(opacity),
        children: const <Widget>[
          SizedBox(width: 100.0, height: 100.0),
        ],
      ),
    );

    // By default, clipBehavior should be Clip.hardEdge
    final RenderFlow renderObject = tester.renderObject(find.byType(Flow));
    expect(renderObject.clipBehavior, equals(Clip.hardEdge));

    for(final Clip clip in Clip.values) {
      await tester.pumpWidget(
        Flow(
          delegate: OpacityFlowDelegate(opacity),
178
          clipBehavior: clip,
179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206
          children: const <Widget>[
            SizedBox(width: 100.0, height: 100.0),
          ],
        ),
      );
      expect(renderObject.clipBehavior, clip);
    }
  });

  testWidgets('Flow.unwrapped can set and update clipBehavior', (WidgetTester tester) async {
    const double opacity = 0.2;
    await tester.pumpWidget(
      Flow.unwrapped(
        delegate: OpacityFlowDelegate(opacity),
        children: const <Widget>[
          SizedBox(width: 100.0, height: 100.0),
        ],
      ),
    );

    // By default, clipBehavior should be Clip.hardEdge
    final RenderFlow renderObject = tester.renderObject(find.byType(Flow));
    expect(renderObject.clipBehavior, equals(Clip.hardEdge));

    for(final Clip clip in Clip.values) {
      await tester.pumpWidget(
        Flow.unwrapped(
          delegate: OpacityFlowDelegate(opacity),
207
          clipBehavior: clip,
208 209 210 211 212 213 214 215
          children: const <Widget>[
            SizedBox(width: 100.0, height: 100.0),
          ],
        ),
      );
      expect(renderObject.clipBehavior, clip);
    }
  });
Adam Barth's avatar
Adam Barth committed
216
}