211 lines
6.4 KiB
Dart
211 lines
6.4 KiB
Dart
// 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/rendering.dart';
|
|
import 'package:flutter/services.dart';
|
|
import 'package:flutter/src/widgets/basic.dart';
|
|
import 'package:flutter/src/widgets/framework.dart';
|
|
import 'package:flutter/src/widgets/layout_builder.dart';
|
|
import 'package:flutter/src/widgets/media_query.dart';
|
|
import 'package:flutter/src/widgets/scroll_view.dart';
|
|
import 'package:flutter/src/widgets/sliver_layout_builder.dart';
|
|
import 'package:flutter_test/flutter_test.dart';
|
|
|
|
class Wrapper extends StatelessWidget {
|
|
const Wrapper({
|
|
super.key,
|
|
required this.child,
|
|
});
|
|
|
|
final Widget child;
|
|
|
|
@override
|
|
Widget build(BuildContext context) => child;
|
|
}
|
|
|
|
void main() {
|
|
testWidgets('Moving a global key from another LayoutBuilder at layout time', (WidgetTester tester) async {
|
|
final GlobalKey victimKey = GlobalKey();
|
|
|
|
await tester.pumpWidget(Row(
|
|
textDirection: TextDirection.ltr,
|
|
children: <Widget>[
|
|
Wrapper(
|
|
child: LayoutBuilder(builder: (BuildContext context, BoxConstraints constraints) {
|
|
return const SizedBox();
|
|
}),
|
|
),
|
|
Wrapper(
|
|
child: Wrapper(
|
|
child: LayoutBuilder(builder: (BuildContext context, BoxConstraints constraints) {
|
|
return Wrapper(
|
|
child: SizedBox(key: victimKey),
|
|
);
|
|
}),
|
|
),
|
|
),
|
|
],
|
|
));
|
|
|
|
await tester.pumpWidget(Row(
|
|
textDirection: TextDirection.ltr,
|
|
children: <Widget>[
|
|
Wrapper(
|
|
child: LayoutBuilder(builder: (BuildContext context, BoxConstraints constraints) {
|
|
return Wrapper(
|
|
child: SizedBox(key: victimKey),
|
|
);
|
|
}),
|
|
),
|
|
Wrapper(
|
|
child: Wrapper(
|
|
child: LayoutBuilder(builder: (BuildContext context, BoxConstraints constraints) {
|
|
return const SizedBox();
|
|
}),
|
|
),
|
|
),
|
|
],
|
|
));
|
|
|
|
expect(tester.takeException(), null);
|
|
});
|
|
|
|
testWidgets('Moving a global key from another SliverLayoutBuilder at layout time', (WidgetTester tester) async {
|
|
final GlobalKey victimKey1 = GlobalKey();
|
|
final GlobalKey victimKey2 = GlobalKey();
|
|
|
|
await tester.pumpWidget(
|
|
Directionality(
|
|
textDirection: TextDirection.ltr,
|
|
child: CustomScrollView(
|
|
slivers: <Widget>[
|
|
SliverLayoutBuilder(
|
|
builder: (BuildContext context, SliverConstraints constraint) {
|
|
return SliverPadding(key: victimKey1, padding: const EdgeInsets.fromLTRB(1, 2, 3, 4));
|
|
},
|
|
),
|
|
SliverLayoutBuilder(
|
|
builder: (BuildContext context, SliverConstraints constraint) {
|
|
return SliverPadding(key: victimKey2, padding: const EdgeInsets.fromLTRB(5, 7, 11, 13));
|
|
},
|
|
),
|
|
SliverLayoutBuilder(
|
|
builder: (BuildContext context, SliverConstraints constraint) {
|
|
return const SliverPadding(padding: EdgeInsets.fromLTRB(5, 7, 11, 13));
|
|
},
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
|
|
await tester.pumpWidget(
|
|
Directionality(
|
|
textDirection: TextDirection.ltr,
|
|
child: CustomScrollView(
|
|
slivers: <Widget>[
|
|
SliverLayoutBuilder(
|
|
builder: (BuildContext context, SliverConstraints constraint) {
|
|
return SliverPadding(key: victimKey2, padding: const EdgeInsets.fromLTRB(1, 2, 3, 4));
|
|
},
|
|
),
|
|
SliverLayoutBuilder(
|
|
builder: (BuildContext context, SliverConstraints constraint) {
|
|
return const SliverPadding(padding: EdgeInsets.fromLTRB(5, 7, 11, 13));
|
|
},
|
|
),
|
|
SliverLayoutBuilder(
|
|
builder: (BuildContext context, SliverConstraints constraint) {
|
|
return SliverPadding(key: victimKey1, padding: const EdgeInsets.fromLTRB(5, 7, 11, 13));
|
|
},
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
|
|
expect(tester.takeException(), null);
|
|
});
|
|
|
|
testWidgets('LayoutBuilder does not layout twice', (WidgetTester tester) async {
|
|
// This widget marks itself dirty when the closest MediaQuery changes.
|
|
final _LayoutCount widget = _LayoutCount();
|
|
late StateSetter setState;
|
|
bool updated = false;
|
|
|
|
await tester.pumpWidget(
|
|
Directionality(
|
|
textDirection: TextDirection.ltr,
|
|
child: StatefulBuilder(
|
|
builder: (BuildContext context, StateSetter setter) {
|
|
setState = setter;
|
|
return MediaQuery(
|
|
data: updated
|
|
? const MediaQueryData(platformBrightness: Brightness.dark)
|
|
: const MediaQueryData(),
|
|
child: LayoutBuilder(
|
|
builder: (BuildContext context, BoxConstraints constraints) {
|
|
return Center(
|
|
child: SizedBox.square(
|
|
dimension: 20,
|
|
child: Center(
|
|
child: SizedBox.square(
|
|
dimension: updated ? 10 : 20,
|
|
child: widget,
|
|
),
|
|
),
|
|
),
|
|
);
|
|
},
|
|
),
|
|
);
|
|
}
|
|
),
|
|
),
|
|
);
|
|
|
|
assert(widget._renderObject.layoutCount == 1);
|
|
setState(() { updated = true; });
|
|
|
|
await tester.pump();
|
|
expect(widget._renderObject.layoutCount, 2);
|
|
});
|
|
}
|
|
|
|
class _LayoutCount extends LeafRenderObjectWidget {
|
|
late final _RenderLayoutCount _renderObject;
|
|
|
|
@override
|
|
RenderObject createRenderObject(BuildContext context) {
|
|
return _renderObject = _RenderLayoutCount(MediaQuery.of(context));
|
|
}
|
|
|
|
@override
|
|
void updateRenderObject(BuildContext context, _RenderLayoutCount renderObject) {
|
|
renderObject.mediaQuery = MediaQuery.of(context);
|
|
}
|
|
}
|
|
|
|
class _RenderLayoutCount extends RenderProxyBox {
|
|
_RenderLayoutCount(this._mediaQuery);
|
|
int layoutCount = 0;
|
|
|
|
MediaQueryData get mediaQuery => _mediaQuery;
|
|
MediaQueryData _mediaQuery;
|
|
set mediaQuery(MediaQueryData newValue) {
|
|
if (newValue != _mediaQuery) {
|
|
_mediaQuery = newValue;
|
|
markNeedsLayout();
|
|
}
|
|
}
|
|
|
|
@override
|
|
bool get sizedByParent => true;
|
|
|
|
@override
|
|
void performLayout() {
|
|
layoutCount += 1;
|
|
}
|
|
}
|