// Copyright 2014 The Flutter 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/gestures.dart'; import 'package:flutter/material.dart'; import 'package:flutter/rendering.dart'; import 'package:flutter_test/flutter_test.dart'; class _MockRenderSliver extends RenderSliver { @override void performLayout() { geometry = const SliverGeometry(paintOrigin: 10, paintExtent: 10, maxPaintExtent: 10); } } Future test( WidgetTester tester, double offset, EdgeInsetsGeometry padding, AxisDirection axisDirection, TextDirection textDirection, ) { final viewportOffset = ViewportOffset.fixed(offset); addTearDown(viewportOffset.dispose); return tester.pumpWidget( Directionality( textDirection: textDirection, child: Viewport( offset: viewportOffset, axisDirection: axisDirection, slivers: [ const SliverToBoxAdapter( child: SizedBox(width: 400.0, height: 400.0, child: Text('before')), ), SliverPadding( padding: padding, sliver: const SliverToBoxAdapter( child: SizedBox(width: 400.0, height: 400.0, child: Text('padded')), ), ), const SliverToBoxAdapter( child: SizedBox(width: 400.0, height: 400.0, child: Text('after')), ), ], ), ), ); } void verify(WidgetTester tester, List answerKey) { final List testAnswers = tester .renderObjectList(find.byType(SizedBox, skipOffstage: false)) .map((RenderBox target) { final Offset topLeft = target.localToGlobal(Offset.zero); final Offset bottomRight = target.localToGlobal(target.size.bottomRight(Offset.zero)); return Rect.fromPoints(topLeft, bottomRight); }) .toList(); expect(testAnswers, equals(answerKey)); } void main() { testWidgets('Viewport+SliverPadding basic test (VISUAL)', (WidgetTester tester) async { const padding = EdgeInsets.fromLTRB(25.0, 20.0, 15.0, 35.0); await test(tester, 0.0, padding, AxisDirection.down, TextDirection.ltr); expect( tester.renderObject(find.byType(Viewport)).size, equals(const Size(800.0, 600.0)), ); verify(tester, [ const Rect.fromLTWH(0.0, 0.0, 800.0, 400.0), const Rect.fromLTWH(25.0, 420.0, 760.0, 400.0), const Rect.fromLTWH(0.0, 855.0, 800.0, 400.0), ]); await test(tester, 200.0, padding, AxisDirection.down, TextDirection.ltr); verify(tester, [ const Rect.fromLTWH(0.0, -200.0, 800.0, 400.0), const Rect.fromLTWH(25.0, 220.0, 760.0, 400.0), const Rect.fromLTWH(0.0, 655.0, 800.0, 400.0), ]); await test(tester, 390.0, padding, AxisDirection.down, TextDirection.ltr); verify(tester, [ const Rect.fromLTWH(0.0, -390.0, 800.0, 400.0), const Rect.fromLTWH(25.0, 30.0, 760.0, 400.0), const Rect.fromLTWH(0.0, 465.0, 800.0, 400.0), ]); await test(tester, 490.0, padding, AxisDirection.down, TextDirection.ltr); verify(tester, [ const Rect.fromLTWH(0.0, -490.0, 800.0, 400.0), const Rect.fromLTWH(25.0, -70.0, 760.0, 400.0), const Rect.fromLTWH(0.0, 365.0, 800.0, 400.0), ]); await test(tester, 10000.0, padding, AxisDirection.down, TextDirection.ltr); verify(tester, [ const Rect.fromLTWH(0.0, -10000.0, 800.0, 400.0), const Rect.fromLTWH(25.0, -9580.0, 760.0, 400.0), const Rect.fromLTWH(0.0, -9145.0, 800.0, 400.0), ]); }); testWidgets('Viewport+SliverPadding basic test (LTR)', (WidgetTester tester) async { const padding = EdgeInsetsDirectional.fromSTEB(25.0, 20.0, 15.0, 35.0); await test(tester, 0.0, padding, AxisDirection.down, TextDirection.ltr); expect( tester.renderObject(find.byType(Viewport)).size, equals(const Size(800.0, 600.0)), ); verify(tester, [ const Rect.fromLTWH(0.0, 0.0, 800.0, 400.0), const Rect.fromLTWH(25.0, 420.0, 760.0, 400.0), const Rect.fromLTWH(0.0, 855.0, 800.0, 400.0), ]); await test(tester, 200.0, padding, AxisDirection.down, TextDirection.ltr); verify(tester, [ const Rect.fromLTWH(0.0, -200.0, 800.0, 400.0), const Rect.fromLTWH(25.0, 220.0, 760.0, 400.0), const Rect.fromLTWH(0.0, 655.0, 800.0, 400.0), ]); await test(tester, 390.0, padding, AxisDirection.down, TextDirection.ltr); verify(tester, [ const Rect.fromLTWH(0.0, -390.0, 800.0, 400.0), const Rect.fromLTWH(25.0, 30.0, 760.0, 400.0), const Rect.fromLTWH(0.0, 465.0, 800.0, 400.0), ]); await test(tester, 490.0, padding, AxisDirection.down, TextDirection.ltr); verify(tester, [ const Rect.fromLTWH(0.0, -490.0, 800.0, 400.0), const Rect.fromLTWH(25.0, -70.0, 760.0, 400.0), const Rect.fromLTWH(0.0, 365.0, 800.0, 400.0), ]); await test(tester, 10000.0, padding, AxisDirection.down, TextDirection.ltr); verify(tester, [ const Rect.fromLTWH(0.0, -10000.0, 800.0, 400.0), const Rect.fromLTWH(25.0, -9580.0, 760.0, 400.0), const Rect.fromLTWH(0.0, -9145.0, 800.0, 400.0), ]); }); testWidgets('Viewport+SliverPadding basic test (RTL)', (WidgetTester tester) async { const padding = EdgeInsetsDirectional.fromSTEB(25.0, 20.0, 15.0, 35.0); await test(tester, 0.0, padding, AxisDirection.down, TextDirection.rtl); expect( tester.renderObject(find.byType(Viewport)).size, equals(const Size(800.0, 600.0)), ); verify(tester, [ const Rect.fromLTWH(0.0, 0.0, 800.0, 400.0), const Rect.fromLTWH(15.0, 420.0, 760.0, 400.0), const Rect.fromLTWH(0.0, 855.0, 800.0, 400.0), ]); await test(tester, 200.0, padding, AxisDirection.down, TextDirection.rtl); verify(tester, [ const Rect.fromLTWH(0.0, -200.0, 800.0, 400.0), const Rect.fromLTWH(15.0, 220.0, 760.0, 400.0), const Rect.fromLTWH(0.0, 655.0, 800.0, 400.0), ]); await test(tester, 390.0, padding, AxisDirection.down, TextDirection.rtl); verify(tester, [ const Rect.fromLTWH(0.0, -390.0, 800.0, 400.0), const Rect.fromLTWH(15.0, 30.0, 760.0, 400.0), const Rect.fromLTWH(0.0, 465.0, 800.0, 400.0), ]); await test(tester, 490.0, padding, AxisDirection.down, TextDirection.rtl); verify(tester, [ const Rect.fromLTWH(0.0, -490.0, 800.0, 400.0), const Rect.fromLTWH(15.0, -70.0, 760.0, 400.0), const Rect.fromLTWH(0.0, 365.0, 800.0, 400.0), ]); await test(tester, 10000.0, padding, AxisDirection.down, TextDirection.rtl); verify(tester, [ const Rect.fromLTWH(0.0, -10000.0, 800.0, 400.0), const Rect.fromLTWH(15.0, -9580.0, 760.0, 400.0), const Rect.fromLTWH(0.0, -9145.0, 800.0, 400.0), ]); }); testWidgets('Viewport+SliverPadding hit testing', (WidgetTester tester) async { const padding = EdgeInsets.all(30.0); await test(tester, 350.0, padding, AxisDirection.down, TextDirection.ltr); expect( tester.renderObject(find.byType(Viewport)).size, equals(const Size(800.0, 600.0)), ); verify(tester, [ const Rect.fromLTWH(0.0, -350.0, 800.0, 400.0), const Rect.fromLTWH(30.0, 80.0, 740.0, 400.0), const Rect.fromLTWH(0.0, 510.0, 800.0, 400.0), ]); HitTestResult result; result = tester.hitTestOnBinding(const Offset(10.0, 10.0)); hitsText(result, 'before'); result = tester.hitTestOnBinding(const Offset(10.0, 60.0)); expect(result.path.first.target, isA()); result = tester.hitTestOnBinding(const Offset(100.0, 100.0)); hitsText(result, 'padded'); expect(result.path.any((HitTestEntry entry) => entry.target is RenderSliverPadding), isTrue); result = tester.hitTestOnBinding(const Offset(100.0, 490.0)); expect(result.path.first.target, isA()); result = tester.hitTestOnBinding(const Offset(10.0, 520.0)); hitsText(result, 'after'); }); testWidgets('Viewport+SliverPadding hit testing up', (WidgetTester tester) async { const padding = EdgeInsets.all(30.0); await test(tester, 350.0, padding, AxisDirection.up, TextDirection.ltr); expect( tester.renderObject(find.byType(Viewport)).size, equals(const Size(800.0, 600.0)), ); verify(tester, [ const Rect.fromLTWH(0.0, 600.0 + 350.0 - 400.0, 800.0, 400.0), const Rect.fromLTWH(30.0, 600.0 - 80.0 - 400.0, 740.0, 400.0), const Rect.fromLTWH(0.0, 600.0 - 510.0 - 400.0, 800.0, 400.0), ]); HitTestResult result; result = tester.hitTestOnBinding(const Offset(10.0, 600.0 - 10.0)); hitsText(result, 'before'); result = tester.hitTestOnBinding(const Offset(10.0, 600.0 - 60.0)); expect(result.path.first.target, isA()); result = tester.hitTestOnBinding(const Offset(100.0, 600.0 - 100.0)); hitsText(result, 'padded'); result = tester.hitTestOnBinding(const Offset(100.0, 600.0 - 490.0)); expect(result.path.first.target, isA()); result = tester.hitTestOnBinding(const Offset(10.0, 600.0 - 520.0)); hitsText(result, 'after'); }); testWidgets('Viewport+SliverPadding hit testing left', (WidgetTester tester) async { const padding = EdgeInsets.all(30.0); await test(tester, 350.0, padding, AxisDirection.left, TextDirection.ltr); expect( tester.renderObject(find.byType(Viewport)).size, equals(const Size(800.0, 600.0)), ); verify(tester, [ const Rect.fromLTWH(800.0 + 350.0 - 400.0, 0.0, 400.0, 600.0), const Rect.fromLTWH(800.0 - 80.0 - 400.0, 30.0, 400.0, 540.0), const Rect.fromLTWH(800.0 - 510.0 - 400.0, 0.0, 400.0, 600.0), ]); HitTestResult result; result = tester.hitTestOnBinding(const Offset(800.0 - 10.0, 10.0)); hitsText(result, 'before'); result = tester.hitTestOnBinding(const Offset(800.0 - 60.0, 10.0)); expect(result.path.first.target, isA()); result = tester.hitTestOnBinding(const Offset(800.0 - 100.0, 100.0)); hitsText(result, 'padded'); result = tester.hitTestOnBinding(const Offset(800.0 - 490.0, 100.0)); expect(result.path.first.target, isA()); result = tester.hitTestOnBinding(const Offset(800.0 - 520.0, 10.0)); hitsText(result, 'after'); }); testWidgets('Viewport+SliverPadding hit testing right', (WidgetTester tester) async { const padding = EdgeInsets.all(30.0); await test(tester, 350.0, padding, AxisDirection.right, TextDirection.ltr); expect( tester.renderObject(find.byType(Viewport)).size, equals(const Size(800.0, 600.0)), ); verify(tester, [ const Rect.fromLTWH(-350.0, 0.0, 400.0, 600.0), const Rect.fromLTWH(80.0, 30.0, 400.0, 540.0), const Rect.fromLTWH(510.0, 0.0, 400.0, 600.0), ]); HitTestResult result; result = tester.hitTestOnBinding(const Offset(10.0, 10.0)); hitsText(result, 'before'); result = tester.hitTestOnBinding(const Offset(60.0, 10.0)); expect(result.path.first.target, isA()); result = tester.hitTestOnBinding(const Offset(100.0, 100.0)); hitsText(result, 'padded'); result = tester.hitTestOnBinding(const Offset(490.0, 100.0)); expect(result.path.first.target, isA()); result = tester.hitTestOnBinding(const Offset(520.0, 10.0)); hitsText(result, 'after'); }); testWidgets('Viewport+SliverPadding no child', (WidgetTester tester) async { final offset = ViewportOffset.fixed(0.0); addTearDown(offset.dispose); await tester.pumpWidget( Directionality( textDirection: TextDirection.ltr, child: Viewport( offset: offset, slivers: const [ SliverPadding(padding: EdgeInsets.all(100.0)), SliverToBoxAdapter(child: SizedBox(width: 400.0, height: 400.0, child: Text('x'))), ], ), ), ); expect( tester.renderObject(find.text('x')).localToGlobal(Offset.zero), const Offset(0.0, 200.0), ); }); testWidgets('SliverPadding with no child reports correct geometry as scroll offset changes', ( WidgetTester tester, ) async { // Regression test for https://github.com/flutter/flutter/issues/64506 final controller = ScrollController(); addTearDown(controller.dispose); await tester.pumpWidget( Directionality( textDirection: TextDirection.ltr, child: CustomScrollView( controller: controller, slivers: const [ SliverPadding(padding: EdgeInsets.all(100.0)), SliverToBoxAdapter(child: SizedBox(width: 400.0, height: 400.0, child: Text('x'))), ], ), ), ); expect( tester.renderObject(find.text('x')).localToGlobal(Offset.zero), const Offset(0.0, 200.0), ); expect( tester.renderObject(find.byType(SliverPadding)).geometry!.paintExtent, 200.0, ); controller.jumpTo(50.0); await tester.pump(); expect( tester.renderObject(find.byType(SliverPadding)).geometry!.paintExtent, 150.0, ); }); testWidgets('Viewport+SliverPadding changing padding', (WidgetTester tester) async { final offset1 = ViewportOffset.fixed(0.0); addTearDown(offset1.dispose); await tester.pumpWidget( Directionality( textDirection: TextDirection.ltr, child: Viewport( axisDirection: AxisDirection.left, offset: offset1, slivers: const [ SliverPadding(padding: EdgeInsets.fromLTRB(90.0, 1.0, 110.0, 2.0)), SliverToBoxAdapter(child: SizedBox(width: 201.0, child: Text('x'))), ], ), ), ); expect( tester.renderObject(find.text('x')).localToGlobal(Offset.zero), const Offset(399.0, 0.0), ); final offset2 = ViewportOffset.fixed(0.0); addTearDown(offset2.dispose); await tester.pumpWidget( Directionality( textDirection: TextDirection.ltr, child: Viewport( axisDirection: AxisDirection.left, offset: offset2, slivers: const [ SliverPadding(padding: EdgeInsets.fromLTRB(110.0, 1.0, 80.0, 2.0)), SliverToBoxAdapter(child: SizedBox(width: 201.0, child: Text('x'))), ], ), ), ); expect( tester.renderObject(find.text('x')).localToGlobal(Offset.zero), const Offset(409.0, 0.0), ); }); testWidgets('Viewport+SliverPadding changing direction', (WidgetTester tester) async { final offset1 = ViewportOffset.fixed(0.0); addTearDown(offset1.dispose); await tester.pumpWidget( Directionality( textDirection: TextDirection.ltr, child: Viewport( axisDirection: AxisDirection.up, offset: offset1, slivers: const [SliverPadding(padding: EdgeInsets.fromLTRB(1.0, 2.0, 4.0, 8.0))], ), ), ); expect(tester.renderObject(find.byType(SliverPadding)).afterPadding, 2.0); final offset2 = ViewportOffset.fixed(0.0); addTearDown(offset2.dispose); await tester.pumpWidget( Directionality( textDirection: TextDirection.ltr, child: Viewport( offset: offset2, slivers: const [SliverPadding(padding: EdgeInsets.fromLTRB(1.0, 2.0, 4.0, 8.0))], ), ), ); expect(tester.renderObject(find.byType(SliverPadding)).afterPadding, 8.0); final offset3 = ViewportOffset.fixed(0.0); addTearDown(offset3.dispose); await tester.pumpWidget( Directionality( textDirection: TextDirection.ltr, child: Viewport( axisDirection: AxisDirection.right, offset: offset3, slivers: const [SliverPadding(padding: EdgeInsets.fromLTRB(1.0, 2.0, 4.0, 8.0))], ), ), ); expect(tester.renderObject(find.byType(SliverPadding)).afterPadding, 4.0); final offset4 = ViewportOffset.fixed(0.0); addTearDown(offset4.dispose); await tester.pumpWidget( Directionality( textDirection: TextDirection.ltr, child: Viewport( axisDirection: AxisDirection.left, offset: offset4, slivers: const [SliverPadding(padding: EdgeInsets.fromLTRB(1.0, 2.0, 4.0, 8.0))], ), ), ); expect(tester.renderObject(find.byType(SliverPadding)).afterPadding, 1.0); final offset5 = ViewportOffset.fixed(99999.9); addTearDown(offset5.dispose); await tester.pumpWidget( Directionality( textDirection: TextDirection.ltr, child: Viewport( axisDirection: AxisDirection.left, offset: offset5, slivers: const [SliverPadding(padding: EdgeInsets.fromLTRB(1.0, 2.0, 4.0, 8.0))], ), ), ); expect( tester .renderObject(find.byType(SliverPadding, skipOffstage: false)) .afterPadding, 1.0, ); }); testWidgets('SliverPadding propagates geometry offset corrections', (WidgetTester tester) async { Widget listBuilder(IndexedWidgetBuilder sliverChildBuilder) { return Directionality( textDirection: TextDirection.ltr, child: CustomScrollView( cacheExtent: 0.0, slivers: [ SliverPadding( padding: EdgeInsets.zero, sliver: SliverList( delegate: SliverChildBuilderDelegate(sliverChildBuilder, childCount: 10), ), ), ], ), ); } await tester.pumpWidget( listBuilder((BuildContext context, int index) { return SizedBox(height: 200.0, child: Center(child: Text(index.toString()))); }), ); await tester.drag(find.text('2'), const Offset(0.0, -300.0)); await tester.pump(); expect( tester.getRect(find.widgetWithText(SizedBox, '2')), const Rect.fromLTRB(0.0, 100.0, 800.0, 300.0), ); // Now item 0 is 400.0px and going back will underflow. await tester.pumpWidget( listBuilder((BuildContext context, int index) { return SizedBox( height: index == 0 ? 400.0 : 200.0, child: Center(child: Text(index.toString())), ); }), ); await tester.drag(find.text('2'), const Offset(0.0, 300.0)); // On this one frame, the scroll correction must properly propagate. await tester.pump(); expect( tester.getRect(find.widgetWithText(SizedBox, '0')), const Rect.fromLTRB(0.0, -200.0, 800.0, 200.0), ); }); testWidgets( 'SliverPadding includes preceding padding in the precedingScrollExtent provided to child', (WidgetTester tester) async { // Regression test for https://github.com/flutter/flutter/issues/49195 final key = UniqueKey(); await tester.pumpWidget( Directionality( textDirection: TextDirection.ltr, child: CustomScrollView( slivers: [ SliverPadding( padding: const EdgeInsets.only(top: 30), sliver: SliverFillRemaining( hasScrollBody: false, child: Container(key: key, color: Colors.red), ), ), ], ), ), ); await tester.pump(); // The value of 570 is expected since SliverFillRemaining will fill all of // the space available to it. In this test, the extent of the viewport is // 600 pixels. If the SliverPadding widget provides the right constraints // to SliverFillRemaining, with 30 pixels preceding it, it should only have // a height of 570. expect(tester.renderObject(find.byKey(key)).size.height, equals(570)); }, ); testWidgets( "SliverPadding consumes only its padding from the overlap of its parent's constraints", (WidgetTester tester) async { final mock = _MockRenderSliver(); addTearDown(mock.dispose); final renderObject = RenderSliverPadding(padding: const EdgeInsets.only(top: 20)); addTearDown(renderObject.dispose); renderObject.child = mock; renderObject.layout( const SliverConstraints( viewportMainAxisExtent: 100.0, overlap: 100.0, cacheOrigin: 0.0, scrollOffset: 0.0, axisDirection: AxisDirection.down, growthDirection: GrowthDirection.forward, crossAxisExtent: 100.0, crossAxisDirection: AxisDirection.right, userScrollDirection: ScrollDirection.idle, remainingPaintExtent: 100.0, remainingCacheExtent: 100.0, precedingScrollExtent: 0.0, ), parentUsesSize: true, ); expect(mock.constraints.overlap, 80.0); }, ); testWidgets("SliverPadding passes the overlap to the child if it's negative", ( WidgetTester tester, ) async { final mock = _MockRenderSliver(); addTearDown(mock.dispose); final renderObject = RenderSliverPadding(padding: const EdgeInsets.only(top: 20)); addTearDown(renderObject.dispose); renderObject.child = mock; renderObject.layout( const SliverConstraints( viewportMainAxisExtent: 100.0, overlap: -100.0, cacheOrigin: 0.0, scrollOffset: 0.0, axisDirection: AxisDirection.down, growthDirection: GrowthDirection.forward, crossAxisExtent: 100.0, crossAxisDirection: AxisDirection.right, userScrollDirection: ScrollDirection.idle, remainingPaintExtent: 100.0, remainingCacheExtent: 100.0, precedingScrollExtent: 0.0, ), parentUsesSize: true, ); expect(mock.constraints.overlap, -100.0); }); testWidgets('SliverPadding passes the paintOrigin of the child on', (WidgetTester tester) async { final mock = _MockRenderSliver(); addTearDown(mock.dispose); final renderObject = RenderSliverPadding(padding: const EdgeInsets.only(top: 20)); addTearDown(renderObject.dispose); renderObject.child = mock; renderObject.layout( const SliverConstraints( viewportMainAxisExtent: 100.0, overlap: 100.0, cacheOrigin: 0.0, scrollOffset: 0.0, axisDirection: AxisDirection.down, growthDirection: GrowthDirection.forward, crossAxisExtent: 100.0, crossAxisDirection: AxisDirection.right, userScrollDirection: ScrollDirection.idle, remainingPaintExtent: 100.0, remainingCacheExtent: 100.0, precedingScrollExtent: 0.0, ), parentUsesSize: true, ); expect(renderObject.geometry!.paintOrigin, 10.0); }); } void hitsText(HitTestResult hitTestResult, String text) { switch (hitTestResult.path.first.target) { case final TextSpan span: expect(span.text, text); case final RenderParagraph paragraph: final InlineSpan span = paragraph.text; expect(span, isA()); expect((span as TextSpan).text, text); case final HitTestTarget target: fail('$target is not a TextSpan or a RenderParagraph.'); } }