Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Sign in
Toggle navigation
F
Front-End
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
abdullh.alsoleman
Front-End
Commits
784520bd
Unverified
Commit
784520bd
authored
Jun 09, 2022
by
Kate Lovett
Committed by
GitHub
Jun 09, 2022
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Updating PrimaryScrollController for Desktop (#102099)
parent
b73be72d
Changes
31
Hide whitespace changes
Inline
Side-by-side
Showing
31 changed files
with
497 additions
and
87 deletions
+497
-87
colors_demo.dart
...tegration_tests/flutter_gallery/lib/demo/colors_demo.dart
+1
-0
cupertino_alert_demo.dart
...tter_gallery/lib/demo/cupertino/cupertino_alert_demo.dart
+1
-0
cupertino_navigation_demo.dart
...gallery/lib/demo/cupertino/cupertino_navigation_demo.dart
+1
-0
cupertino_text_field_demo.dart
...gallery/lib/demo/cupertino/cupertino_text_field_demo.dart
+1
-0
backdrop_demo.dart
...ests/flutter_gallery/lib/demo/material/backdrop_demo.dart
+1
-0
bottom_app_bar_demo.dart
...lutter_gallery/lib/demo/material/bottom_app_bar_demo.dart
+1
-0
cards_demo.dart
...n_tests/flutter_gallery/lib/demo/material/cards_demo.dart
+1
-0
chip_demo.dart
...on_tests/flutter_gallery/lib/demo/material/chip_demo.dart
+6
-1
data_table_demo.dart
...ts/flutter_gallery/lib/demo/material/data_table_demo.dart
+1
-0
elevation_demo.dart
...sts/flutter_gallery/lib/demo/material/elevation_demo.dart
+6
-1
expansion_tile_list_demo.dart
...r_gallery/lib/demo/material/expansion_tile_list_demo.dart
+1
-0
full_screen_dialog_demo.dart
...er_gallery/lib/demo/material/full_screen_dialog_demo.dart
+1
-0
icons_demo.dart
...n_tests/flutter_gallery/lib/demo/material/icons_demo.dart
+1
-0
leave_behind_demo.dart
.../flutter_gallery/lib/demo/material/leave_behind_demo.dart
+1
-0
list_demo.dart
...on_tests/flutter_gallery/lib/demo/material/list_demo.dart
+1
-0
overscroll_demo.dart
...ts/flutter_gallery/lib/demo/material/overscroll_demo.dart
+1
-0
reorderable_list_demo.dart
...tter_gallery/lib/demo/material/reorderable_list_demo.dart
+1
-0
text_form_field_demo.dart
...utter_gallery/lib/demo/material/text_form_field_demo.dart
+1
-0
typography_demo.dart
...ation_tests/flutter_gallery/lib/demo/typography_demo.dart
+6
-1
video_demo.dart
...ntegration_tests/flutter_gallery/lib/demo/video_demo.dart
+1
-0
raw_scrollbar.0.dart
examples/api/lib/widgets/scrollbar/raw_scrollbar.0.dart
+9
-5
raw_scrollbar.shape.0.dart
...ples/api/lib/widgets/scrollbar/raw_scrollbar.shape.0.dart
+5
-0
dropdown.dart
packages/flutter/lib/src/material/dropdown.dart
+2
-0
nested_scroll_view.dart
packages/flutter/lib/src/widgets/nested_scroll_view.dart
+12
-0
primary_scroll_controller.dart
...es/flutter/lib/src/widgets/primary_scroll_controller.dart
+76
-7
scroll_view.dart
packages/flutter/lib/src/widgets/scroll_view.dart
+33
-12
scrollable.dart
packages/flutter/lib/src/widgets/scrollable.dart
+30
-8
scrollbar.dart
packages/flutter/lib/src/widgets/scrollbar.dart
+22
-18
range_maintaining_scroll_physics_test.dart
...r/test/widgets/range_maintaining_scroll_physics_test.dart
+2
-1
scroll_view_test.dart
packages/flutter/test/widgets/scroll_view_test.dart
+187
-31
scrollbar_test.dart
packages/flutter/test/widgets/scrollbar_test.dart
+84
-2
No files found.
dev/integration_tests/flutter_gallery/lib/demo/colors_demo.dart
View file @
784520bd
...
@@ -96,6 +96,7 @@ class PaletteTabView extends StatelessWidget {
...
@@ -96,6 +96,7 @@ class PaletteTabView extends StatelessWidget {
final
TextStyle
blackTextStyle
=
textTheme
.
bodyText2
!.
copyWith
(
color:
Colors
.
black
);
final
TextStyle
blackTextStyle
=
textTheme
.
bodyText2
!.
copyWith
(
color:
Colors
.
black
);
return
Scrollbar
(
return
Scrollbar
(
child:
ListView
(
child:
ListView
(
primary:
true
,
itemExtent:
kColorItemHeight
,
itemExtent:
kColorItemHeight
,
children:
<
Widget
>[
children:
<
Widget
>[
...
primaryKeys
.
map
<
Widget
>((
int
index
)
{
...
primaryKeys
.
map
<
Widget
>((
int
index
)
{
...
...
dev/integration_tests/flutter_gallery/lib/demo/cupertino/cupertino_alert_demo.dart
View file @
784520bd
...
@@ -60,6 +60,7 @@ class _CupertinoAlertDemoState extends State<CupertinoAlertDemo> {
...
@@ -60,6 +60,7 @@ class _CupertinoAlertDemoState extends State<CupertinoAlertDemo> {
children:
<
Widget
>[
children:
<
Widget
>[
CupertinoScrollbar
(
CupertinoScrollbar
(
child:
ListView
(
child:
ListView
(
primary:
true
,
// Add more padding to the normal safe area.
// Add more padding to the normal safe area.
padding:
const
EdgeInsets
.
symmetric
(
vertical:
24.0
,
horizontal:
72.0
)
padding:
const
EdgeInsets
.
symmetric
(
vertical:
24.0
,
horizontal:
72.0
)
+
MediaQuery
.
of
(
context
).
padding
,
+
MediaQuery
.
of
(
context
).
padding
,
...
...
dev/integration_tests/flutter_gallery/lib/demo/cupertino/cupertino_navigation_demo.dart
View file @
784520bd
...
@@ -446,6 +446,7 @@ class CupertinoDemoTab2 extends StatelessWidget {
...
@@ -446,6 +446,7 @@ class CupertinoDemoTab2 extends StatelessWidget {
),
),
child:
CupertinoScrollbar
(
child:
CupertinoScrollbar
(
child:
ListView
(
child:
ListView
(
primary:
true
,
children:
<
Widget
>[
children:
<
Widget
>[
const
CupertinoUserInterfaceLevel
(
const
CupertinoUserInterfaceLevel
(
data:
CupertinoUserInterfaceLevelData
.
elevated
,
data:
CupertinoUserInterfaceLevelData
.
elevated
,
...
...
dev/integration_tests/flutter_gallery/lib/demo/cupertino/cupertino_text_field_demo.dart
View file @
784520bd
...
@@ -168,6 +168,7 @@ class _CupertinoTextFieldDemoState extends State<CupertinoTextFieldDemo> {
...
@@ -168,6 +168,7 @@ class _CupertinoTextFieldDemoState extends State<CupertinoTextFieldDemo> {
),
),
child:
CupertinoScrollbar
(
child:
CupertinoScrollbar
(
child:
ListView
(
child:
ListView
(
primary:
true
,
children:
<
Widget
>[
children:
<
Widget
>[
Padding
(
Padding
(
padding:
const
EdgeInsets
.
symmetric
(
vertical:
32.0
,
horizontal:
16.0
),
padding:
const
EdgeInsets
.
symmetric
(
vertical:
32.0
,
horizontal:
16.0
),
...
...
dev/integration_tests/flutter_gallery/lib/demo/material/backdrop_demo.dart
View file @
784520bd
...
@@ -104,6 +104,7 @@ class CategoryView extends StatelessWidget {
...
@@ -104,6 +104,7 @@ class CategoryView extends StatelessWidget {
final
ThemeData
theme
=
Theme
.
of
(
context
);
final
ThemeData
theme
=
Theme
.
of
(
context
);
return
Scrollbar
(
return
Scrollbar
(
child:
ListView
(
child:
ListView
(
primary:
true
,
key:
PageStorageKey
<
Category
?>(
category
),
key:
PageStorageKey
<
Category
?>(
category
),
padding:
const
EdgeInsets
.
symmetric
(
padding:
const
EdgeInsets
.
symmetric
(
vertical:
16.0
,
vertical:
16.0
,
...
...
dev/integration_tests/flutter_gallery/lib/demo/material/bottom_app_bar_demo.dart
View file @
784520bd
...
@@ -161,6 +161,7 @@ class _BottomAppBarDemoState extends State<BottomAppBarDemo> {
...
@@ -161,6 +161,7 @@ class _BottomAppBarDemoState extends State<BottomAppBarDemo> {
),
),
body:
Scrollbar
(
body:
Scrollbar
(
child:
ListView
(
child:
ListView
(
primary:
true
,
padding:
const
EdgeInsets
.
only
(
bottom:
88.0
),
padding:
const
EdgeInsets
.
only
(
bottom:
88.0
),
children:
<
Widget
>[
children:
<
Widget
>[
const
_Heading
(
'FAB Shape'
),
const
_Heading
(
'FAB Shape'
),
...
...
dev/integration_tests/flutter_gallery/lib/demo/material/cards_demo.dart
View file @
784520bd
...
@@ -375,6 +375,7 @@ class _CardsDemoState extends State<CardsDemo> {
...
@@ -375,6 +375,7 @@ class _CardsDemoState extends State<CardsDemo> {
),
),
body:
Scrollbar
(
body:
Scrollbar
(
child:
ListView
(
child:
ListView
(
primary:
true
,
padding:
const
EdgeInsets
.
only
(
top:
8.0
,
left:
8.0
,
right:
8.0
),
padding:
const
EdgeInsets
.
only
(
top:
8.0
,
left:
8.0
,
right:
8.0
),
children:
destinations
.
map
<
Widget
>((
TravelDestination
destination
)
{
children:
destinations
.
map
<
Widget
>((
TravelDestination
destination
)
{
Widget
?
child
;
Widget
?
child
;
...
...
dev/integration_tests/flutter_gallery/lib/demo/material/chip_demo.dart
View file @
784520bd
...
@@ -347,7 +347,12 @@ class _ChipDemoState extends State<ChipDemo> {
...
@@ -347,7 +347,12 @@ class _ChipDemoState extends State<ChipDemo> {
borderRadius:
BorderRadius
.
circular
(
10.0
),
borderRadius:
BorderRadius
.
circular
(
10.0
),
))
))
:
theme
.
chipTheme
,
:
theme
.
chipTheme
,
child:
Scrollbar
(
child:
ListView
(
children:
tiles
)),
child:
Scrollbar
(
child:
ListView
(
primary:
true
,
children:
tiles
,
)
),
),
),
floatingActionButton:
FloatingActionButton
(
floatingActionButton:
FloatingActionButton
(
onPressed:
()
=>
setState
(
_reset
),
onPressed:
()
=>
setState
(
_reset
),
...
...
dev/integration_tests/flutter_gallery/lib/demo/material/data_table_demo.dart
View file @
784520bd
...
@@ -175,6 +175,7 @@ class _DataTableDemoState extends State<DataTableDemo> {
...
@@ -175,6 +175,7 @@ class _DataTableDemoState extends State<DataTableDemo> {
),
),
body:
Scrollbar
(
body:
Scrollbar
(
child:
ListView
(
child:
ListView
(
primary:
true
,
padding:
const
EdgeInsets
.
all
(
20.0
),
padding:
const
EdgeInsets
.
all
(
20.0
),
children:
<
Widget
>[
children:
<
Widget
>[
PaginatedDataTable
(
PaginatedDataTable
(
...
...
dev/integration_tests/flutter_gallery/lib/demo/material/elevation_demo.dart
View file @
784520bd
...
@@ -64,7 +64,12 @@ class _ElevationDemoState extends State<ElevationDemo> {
...
@@ -64,7 +64,12 @@ class _ElevationDemoState extends State<ElevationDemo> {
),
),
],
],
),
),
body:
Scrollbar
(
child:
ListView
(
children:
buildCards
())),
body:
Scrollbar
(
child:
ListView
(
primary:
true
,
children:
buildCards
(),
),
),
);
);
}
}
}
}
dev/integration_tests/flutter_gallery/lib/demo/material/expansion_tile_list_demo.dart
View file @
784520bd
...
@@ -20,6 +20,7 @@ class ExpansionTileListDemo extends StatelessWidget {
...
@@ -20,6 +20,7 @@ class ExpansionTileListDemo extends StatelessWidget {
),
),
body:
Scrollbar
(
body:
Scrollbar
(
child:
ListView
(
child:
ListView
(
primary:
true
,
children:
<
Widget
>[
children:
<
Widget
>[
const
ListTile
(
title:
Text
(
'Top'
)),
const
ListTile
(
title:
Text
(
'Top'
)),
ExpansionTile
(
ExpansionTile
(
...
...
dev/integration_tests/flutter_gallery/lib/demo/material/full_screen_dialog_demo.dart
View file @
784520bd
...
@@ -162,6 +162,7 @@ class FullScreenDialogDemoState extends State<FullScreenDialogDemo> {
...
@@ -162,6 +162,7 @@ class FullScreenDialogDemoState extends State<FullScreenDialogDemo> {
onWillPop:
_onWillPop
,
onWillPop:
_onWillPop
,
child:
Scrollbar
(
child:
Scrollbar
(
child:
ListView
(
child:
ListView
(
primary:
true
,
padding:
const
EdgeInsets
.
all
(
16.0
),
padding:
const
EdgeInsets
.
all
(
16.0
),
children:
<
Widget
>[
children:
<
Widget
>[
Container
(
Container
(
...
...
dev/integration_tests/flutter_gallery/lib/demo/material/icons_demo.dart
View file @
784520bd
...
@@ -62,6 +62,7 @@ class IconsDemoState extends State<IconsDemo> {
...
@@ -62,6 +62,7 @@ class IconsDemoState extends State<IconsDemo> {
bottom:
false
,
bottom:
false
,
child:
Scrollbar
(
child:
Scrollbar
(
child:
ListView
(
child:
ListView
(
primary:
true
,
padding:
const
EdgeInsets
.
all
(
24.0
),
padding:
const
EdgeInsets
.
all
(
24.0
),
children:
<
Widget
>[
children:
<
Widget
>[
_IconsDemoCard
(
handleIconButtonPress
,
Icons
.
face
),
// direction-agnostic icon
_IconsDemoCard
(
handleIconButtonPress
,
Icons
.
face
),
// direction-agnostic icon
...
...
dev/integration_tests/flutter_gallery/lib/demo/material/leave_behind_demo.dart
View file @
784520bd
...
@@ -131,6 +131,7 @@ class LeaveBehindDemoState extends State<LeaveBehindDemo> {
...
@@ -131,6 +131,7 @@ class LeaveBehindDemoState extends State<LeaveBehindDemo> {
}
else
{
}
else
{
body
=
Scrollbar
(
body
=
Scrollbar
(
child:
ListView
(
child:
ListView
(
primary:
true
,
children:
leaveBehindItems
.
map
<
Widget
>((
LeaveBehindItem
item
)
{
children:
leaveBehindItems
.
map
<
Widget
>((
LeaveBehindItem
item
)
{
return
_LeaveBehindListItem
(
return
_LeaveBehindListItem
(
confirmDismiss:
_confirmDismiss
,
confirmDismiss:
_confirmDismiss
,
...
...
dev/integration_tests/flutter_gallery/lib/demo/material/list_demo.dart
View file @
784520bd
...
@@ -256,6 +256,7 @@ class _ListDemoState extends State<ListDemo> {
...
@@ -256,6 +256,7 @@ class _ListDemoState extends State<ListDemo> {
),
),
body:
Scrollbar
(
body:
Scrollbar
(
child:
ListView
(
child:
ListView
(
primary:
true
,
padding:
EdgeInsets
.
symmetric
(
vertical:
_dense
!=
null
?
4.0
:
8.0
),
padding:
EdgeInsets
.
symmetric
(
vertical:
_dense
!=
null
?
4.0
:
8.0
),
children:
listTiles
.
toList
(),
children:
listTiles
.
toList
(),
),
),
...
...
dev/integration_tests/flutter_gallery/lib/demo/material/overscroll_demo.dart
View file @
784520bd
...
@@ -64,6 +64,7 @@ class OverscrollDemoState extends State<OverscrollDemo> {
...
@@ -64,6 +64,7 @@ class OverscrollDemoState extends State<OverscrollDemo> {
onRefresh:
_handleRefresh
,
onRefresh:
_handleRefresh
,
child:
Scrollbar
(
child:
Scrollbar
(
child:
ListView
.
builder
(
child:
ListView
.
builder
(
primary:
true
,
padding:
kMaterialListPadding
,
padding:
kMaterialListPadding
,
itemCount:
_items
.
length
,
itemCount:
_items
.
length
,
itemBuilder:
(
BuildContext
context
,
int
index
)
{
itemBuilder:
(
BuildContext
context
,
int
index
)
{
...
...
dev/integration_tests/flutter_gallery/lib/demo/material/reorderable_list_demo.dart
View file @
784520bd
...
@@ -208,6 +208,7 @@ class _ListDemoState extends State<ReorderableListDemo> {
...
@@ -208,6 +208,7 @@ class _ListDemoState extends State<ReorderableListDemo> {
),
),
body:
Scrollbar
(
body:
Scrollbar
(
child:
ReorderableListView
(
child:
ReorderableListView
(
primary:
true
,
header:
_itemType
!=
_ReorderableListType
.
threeLine
header:
_itemType
!=
_ReorderableListType
.
threeLine
?
Padding
(
?
Padding
(
padding:
const
EdgeInsets
.
all
(
8.0
),
padding:
const
EdgeInsets
.
all
(
8.0
),
...
...
dev/integration_tests/flutter_gallery/lib/demo/material/text_form_field_demo.dart
View file @
784520bd
...
@@ -182,6 +182,7 @@ class TextFormFieldDemoState extends State<TextFormFieldDemo> {
...
@@ -182,6 +182,7 @@ class TextFormFieldDemoState extends State<TextFormFieldDemo> {
onWillPop:
_warnUserAboutInvalidData
,
onWillPop:
_warnUserAboutInvalidData
,
child:
Scrollbar
(
child:
Scrollbar
(
child:
SingleChildScrollView
(
child:
SingleChildScrollView
(
primary:
true
,
dragStartBehavior:
DragStartBehavior
.
down
,
dragStartBehavior:
DragStartBehavior
.
down
,
padding:
const
EdgeInsets
.
symmetric
(
horizontal:
16.0
),
padding:
const
EdgeInsets
.
symmetric
(
horizontal:
16.0
),
child:
Column
(
child:
Column
(
...
...
dev/integration_tests/flutter_gallery/lib/demo/typography_demo.dart
View file @
784520bd
...
@@ -66,7 +66,12 @@ class TypographyDemo extends StatelessWidget {
...
@@ -66,7 +66,12 @@ class TypographyDemo extends StatelessWidget {
body:
SafeArea
(
body:
SafeArea
(
top:
false
,
top:
false
,
bottom:
false
,
bottom:
false
,
child:
Scrollbar
(
child:
ListView
(
children:
styleItems
)),
child:
Scrollbar
(
child:
ListView
(
primary:
true
,
children:
styleItems
,
),
),
),
),
);
);
}
}
...
...
dev/integration_tests/flutter_gallery/lib/demo/video_demo.dart
View file @
784520bd
...
@@ -418,6 +418,7 @@ class _VideoDemoState extends State<VideoDemo> with SingleTickerProviderStateMix
...
@@ -418,6 +418,7 @@ class _VideoDemoState extends State<VideoDemo> with SingleTickerProviderStateMix
connectedCompleter:
connectedCompleter
,
connectedCompleter:
connectedCompleter
,
child:
Scrollbar
(
child:
Scrollbar
(
child:
ListView
(
child:
ListView
(
primary:
true
,
children:
<
Widget
>[
children:
<
Widget
>[
VideoCard
(
VideoCard
(
title:
'Butterfly'
,
title:
'Butterfly'
,
...
...
examples/api/lib/widgets/scrollbar/raw_scrollbar.0.dart
View file @
784520bd
...
@@ -45,8 +45,9 @@ class _MyStatefulWidgetState extends State<MyStatefulWidget> {
...
@@ -45,8 +45,9 @@ class _MyStatefulWidgetState extends State<MyStatefulWidget> {
children:
<
Widget
>[
children:
<
Widget
>[
SizedBox
(
SizedBox
(
width:
constraints
.
maxWidth
/
2
,
width:
constraints
.
maxWidth
/
2
,
// Only one scroll position can be attached to the
// When using the PrimaryScrollController and a Scrollbar
// PrimaryScrollController if using Scrollbars. Providing a
// together, only one ScrollPosition can be attached to the
// PrimaryScrollController at a time. Providing a
// unique scroll controller to this scroll view prevents it
// unique scroll controller to this scroll view prevents it
// from attaching to the PrimaryScrollController.
// from attaching to the PrimaryScrollController.
child:
Scrollbar
(
child:
Scrollbar
(
...
@@ -64,12 +65,15 @@ class _MyStatefulWidgetState extends State<MyStatefulWidget> {
...
@@ -64,12 +65,15 @@ class _MyStatefulWidgetState extends State<MyStatefulWidget> {
)),
)),
SizedBox
(
SizedBox
(
width:
constraints
.
maxWidth
/
2
,
width:
constraints
.
maxWidth
/
2
,
// This vertical scroll view has not been provided a
// This vertical scroll view has primary set to true, so it is
// ScrollController, so it is using the
// using the PrimaryScrollController. On mobile platforms, the
// PrimaryScrollController.
// PrimaryScrollController automatically attaches to vertical
// ScrollViews, unlike on Desktop platforms, where the primary
// parameter is required.
child:
Scrollbar
(
child:
Scrollbar
(
thumbVisibility:
true
,
thumbVisibility:
true
,
child:
ListView
.
builder
(
child:
ListView
.
builder
(
primary:
true
,
itemCount:
100
,
itemCount:
100
,
itemBuilder:
(
BuildContext
context
,
int
index
)
{
itemBuilder:
(
BuildContext
context
,
int
index
)
{
return
Container
(
return
Container
(
...
...
examples/api/lib/widgets/scrollbar/raw_scrollbar.shape.0.dart
View file @
784520bd
...
@@ -35,6 +35,11 @@ class MyStatelessWidget extends StatelessWidget {
...
@@ -35,6 +35,11 @@ class MyStatelessWidget extends StatelessWidget {
thumbColor:
Colors
.
blue
,
thumbColor:
Colors
.
blue
,
thumbVisibility:
true
,
thumbVisibility:
true
,
child:
ListView
(
child:
ListView
(
// On mobile platforms, setting primary to true is not required, as
// the PrimaryScrollController automatically attaches to vertical
// ScrollPositions. On desktop platforms however, using the
// PrimaryScrollController requires ScrollView.primary be set.
primary:
true
,
physics:
const
BouncingScrollPhysics
(),
physics:
const
BouncingScrollPhysics
(),
children:
List
<
Text
>.
generate
(
children:
List
<
Text
>.
generate
(
100
,
(
int
index
)
=>
Text
((
index
*
index
).
toString
())),
100
,
(
int
index
)
=>
Text
((
index
*
index
).
toString
())),
...
...
packages/flutter/lib/src/material/dropdown.dart
View file @
784520bd
...
@@ -309,6 +309,8 @@ class _DropdownMenuState<T> extends State<_DropdownMenu<T>> {
...
@@ -309,6 +309,8 @@ class _DropdownMenuState<T> extends State<_DropdownMenu<T>> {
child:
Scrollbar
(
child:
Scrollbar
(
thumbVisibility:
true
,
thumbVisibility:
true
,
child:
ListView
(
child:
ListView
(
// Ensure this always inherits the PrimaryScrollController
primary:
true
,
padding:
kMaterialListPadding
,
padding:
kMaterialListPadding
,
shrinkWrap:
true
,
shrinkWrap:
true
,
children:
children
,
children:
children
,
...
...
packages/flutter/lib/src/widgets/nested_scroll_view.dart
View file @
784520bd
...
@@ -312,7 +312,19 @@ class NestedScrollView extends StatefulWidget {
...
@@ -312,7 +312,19 @@ class NestedScrollView extends StatefulWidget {
return
<
Widget
>[
return
<
Widget
>[
...
headerSliverBuilder
(
context
,
bodyIsScrolled
),
...
headerSliverBuilder
(
context
,
bodyIsScrolled
),
SliverFillRemaining
(
SliverFillRemaining
(
// The inner (body) scroll view must use this scroll controller so that
// the independent scroll positions can be kept in sync.
child:
PrimaryScrollController
(
child:
PrimaryScrollController
(
// The inner scroll view should always inherit this
// PrimaryScrollController, on every platform.
automaticallyInheritForPlatforms:
TargetPlatform
.
values
.
toSet
(),
// `PrimaryScrollController.scrollDirection` is not set, and so it is
// restricted to the default Axis.vertical.
// Ideally the inner and outer views would have the same
// scroll direction, and so we could assume
// `NestedScrollView.scrollDirection` for the PrimaryScrollController,
// but use cases already exist where the axes are mismatched.
// https://github.com/flutter/flutter/issues/102001
controller:
innerController
,
controller:
innerController
,
child:
body
,
child:
body
,
),
),
...
...
packages/flutter/lib/src/widgets/primary_scroll_controller.dart
View file @
784520bd
...
@@ -3,19 +3,33 @@
...
@@ -3,19 +3,33 @@
// found in the LICENSE file.
// found in the LICENSE file.
import
'package:flutter/foundation.dart'
;
import
'package:flutter/foundation.dart'
;
import
'package:flutter/painting.dart'
;
import
'framework.dart'
;
import
'framework.dart'
;
import
'scroll_configuration.dart'
;
import
'scroll_controller.dart'
;
import
'scroll_controller.dart'
;
const
Set
<
TargetPlatform
>
_kMobilePlatforms
=
<
TargetPlatform
>{
TargetPlatform
.
android
,
TargetPlatform
.
iOS
,
TargetPlatform
.
fuchsia
,
};
/// Associates a [ScrollController] with a subtree.
/// Associates a [ScrollController] with a subtree.
///
///
/// When a [ScrollView] has [ScrollView.primary] set to true and is not given
/// When a [ScrollView] has [ScrollView.primary] set to true, the [ScrollView]
/// an explicit [ScrollController], the [ScrollView] uses [of] to find the
/// uses [of] to inherit the [PrimaryScrollController] associated with its
/// [ScrollController] associated with its subtree.
/// subtree.
///
/// A ScrollView that doesn't have a controller or the primary flag set will
/// inherit the PrimarySCrollController, if [shouldInherit] allows it. By
/// default [shouldInherit] is true for mobile platforms when the ScrollView has
/// a scroll direction of [Axis.vertical]. This automatic inheritance can be
/// configured with [automaticallyInheritForPlatforms] and [scrollDirection].
///
///
///
This mechanism can be used to provide default behavior for scroll views in a
///
Inheriting this ScrollController can provide default behavior for scroll
///
subtree. For example, the [Scaffold] uses this mechanism to implement the
///
views in a subtree. For example, the [Scaffold] uses this mechanism to
/// scroll-to-top gesture on iOS.
///
implement the
scroll-to-top gesture on iOS.
///
///
/// Another default behavior handled by the PrimaryScrollController is default
/// Another default behavior handled by the PrimaryScrollController is default
/// [ScrollAction]s. If a ScrollAction is not handled by an otherwise focused
/// [ScrollAction]s. If a ScrollAction is not handled by an otherwise focused
...
@@ -34,6 +48,8 @@ class PrimaryScrollController extends InheritedWidget {
...
@@ -34,6 +48,8 @@ class PrimaryScrollController extends InheritedWidget {
const
PrimaryScrollController
({
const
PrimaryScrollController
({
super
.
key
,
super
.
key
,
required
ScrollController
this
.
controller
,
required
ScrollController
this
.
controller
,
this
.
automaticallyInheritForPlatforms
=
_kMobilePlatforms
,
this
.
scrollDirection
=
Axis
.
vertical
,
required
super
.
child
,
required
super
.
child
,
})
:
assert
(
controller
!=
null
);
})
:
assert
(
controller
!=
null
);
...
@@ -41,7 +57,9 @@ class PrimaryScrollController extends InheritedWidget {
...
@@ -41,7 +57,9 @@ class PrimaryScrollController extends InheritedWidget {
const
PrimaryScrollController
.
none
({
const
PrimaryScrollController
.
none
({
super
.
key
,
super
.
key
,
required
super
.
child
,
required
super
.
child
,
})
:
controller
=
null
;
})
:
automaticallyInheritForPlatforms
=
const
<
TargetPlatform
>{},
scrollDirection
=
null
,
controller
=
null
;
/// The [ScrollController] associated with the subtree.
/// The [ScrollController] associated with the subtree.
///
///
...
@@ -51,6 +69,57 @@ class PrimaryScrollController extends InheritedWidget {
...
@@ -51,6 +69,57 @@ class PrimaryScrollController extends InheritedWidget {
/// scroll controller.
/// scroll controller.
final
ScrollController
?
controller
;
final
ScrollController
?
controller
;
/// The [Axis] this controller is configured for [ScrollView]s to
/// automatically inherit.
///
/// Used in conjunction with [automaticallyInheritForPlatforms]. If the
/// current [TargetPlatform] is not included in
/// [automaticallyInheritForPlatforms], this is ignored.
///
/// When null, no [ScrollView] in any Axis will automatically inherit this
/// controller. This is dissimilar to [PrimaryScrollController.none]. When a
/// PrimaryScrollController is inherited, ScrollView will insert
/// PrimaryScrollController.none into the tree to prevent further descendant
/// ScrollViews from inheriting the current PrimaryScrollController.
///
/// Defaults to [Axis.vertical].
final
Axis
?
scrollDirection
;
/// The [TargetPlatform]s this controller is configured for [ScrollView]s to
/// automatically inherit.
///
/// Used in conjunction with [scrollDirection]. If the [Axis] provided to
/// [shouldInherit] is not [scrollDirection], this is ignored.
///
/// When empty, no ScrollView in any Axis will automatically inherit this
/// controller. Defaults to [TargetPlatformVariant.mobile].
final
Set
<
TargetPlatform
>
automaticallyInheritForPlatforms
;
/// Returns true if this PrimaryScrollController is configured to be
/// automatically inherited for the current [TargetPlatform] and the given
/// [Axis].
///
/// This method is typically not called directly. [ScrollView] will call this
/// method if it has not been provided a [ScrollController] and
/// [ScrollView.primary] is unset.
///
/// If a ScrollController has already been provided to
/// [ScrollView.controller], or [ScrollView.primary] is set, this is method is
/// not called by ScrollView as it will have determined whether or not to
/// inherit the PrimaryScrollController.
static
bool
shouldInherit
(
BuildContext
context
,
Axis
scrollDirection
)
{
final
PrimaryScrollController
?
result
=
context
.
findAncestorWidgetOfExactType
<
PrimaryScrollController
>();
if
(
result
==
null
)
{
return
false
;
}
final
TargetPlatform
platform
=
ScrollConfiguration
.
of
(
context
).
getPlatform
(
context
);
if
(
result
.
automaticallyInheritForPlatforms
.
contains
(
platform
))
{
return
result
.
scrollDirection
==
scrollDirection
;
}
return
false
;
}
/// Returns the [ScrollController] most closely associated with the given
/// Returns the [ScrollController] most closely associated with the given
/// context.
/// context.
///
///
...
...
packages/flutter/lib/src/widgets/scroll_view.dart
View file @
784520bd
...
@@ -87,7 +87,7 @@ abstract class ScrollView extends StatelessWidget {
...
@@ -87,7 +87,7 @@ abstract class ScrollView extends StatelessWidget {
this
.
scrollDirection
=
Axis
.
vertical
,
this
.
scrollDirection
=
Axis
.
vertical
,
this
.
reverse
=
false
,
this
.
reverse
=
false
,
this
.
controller
,
this
.
controller
,
bool
?
primary
,
this
.
primary
,
ScrollPhysics
?
physics
,
ScrollPhysics
?
physics
,
this
.
scrollBehavior
,
this
.
scrollBehavior
,
this
.
shrinkWrap
=
false
,
this
.
shrinkWrap
=
false
,
...
@@ -104,15 +104,16 @@ abstract class ScrollView extends StatelessWidget {
...
@@ -104,15 +104,16 @@ abstract class ScrollView extends StatelessWidget {
assert
(
shrinkWrap
!=
null
),
assert
(
shrinkWrap
!=
null
),
assert
(
dragStartBehavior
!=
null
),
assert
(
dragStartBehavior
!=
null
),
assert
(
clipBehavior
!=
null
),
assert
(
clipBehavior
!=
null
),
assert
(!(
controller
!=
null
&&
(
primary
??
false
)),
assert
(
'Primary ScrollViews obtain their ScrollController via inheritance from a PrimaryScrollController widget. '
!(
controller
!=
null
&&
(
primary
??
false
)),
'You cannot both set primary to true and pass an explicit controller.'
,
'Primary ScrollViews obtain their ScrollController via inheritance '
'from a PrimaryScrollController widget. You cannot both set primary to '
'true and pass an explicit controller.'
,
),
),
assert
(!
shrinkWrap
||
center
==
null
),
assert
(!
shrinkWrap
||
center
==
null
),
assert
(
anchor
!=
null
),
assert
(
anchor
!=
null
),
assert
(
anchor
>=
0.0
&&
anchor
<=
1.0
),
assert
(
anchor
>=
0.0
&&
anchor
<=
1.0
),
assert
(
semanticChildCount
==
null
||
semanticChildCount
>=
0
),
assert
(
semanticChildCount
==
null
||
semanticChildCount
>=
0
),
primary
=
primary
??
controller
==
null
&&
identical
(
scrollDirection
,
Axis
.
vertical
),
physics
=
physics
??
((
primary
??
false
)
||
(
primary
==
null
&&
controller
==
null
&&
identical
(
scrollDirection
,
Axis
.
vertical
))
?
const
AlwaysScrollableScrollPhysics
()
:
null
);
physics
=
physics
??
((
primary
??
false
)
||
(
primary
==
null
&&
controller
==
null
&&
identical
(
scrollDirection
,
Axis
.
vertical
))
?
const
AlwaysScrollableScrollPhysics
()
:
null
);
/// {@template flutter.widgets.scroll_view.scrollDirection}
/// {@template flutter.widgets.scroll_view.scrollDirection}
...
@@ -169,11 +170,24 @@ abstract class ScrollView extends StatelessWidget {
...
@@ -169,11 +170,24 @@ abstract class ScrollView extends StatelessWidget {
///
///
/// On iOS, this also identifies the scroll view that will scroll to top in
/// On iOS, this also identifies the scroll view that will scroll to top in
/// response to a tap in the status bar.
/// response to a tap in the status bar.
/// {@endtemplate}
///
///
/// Defaults to true when [scrollDirection] is [Axis.vertical] and
/// Cannot be true while a [ScrollController] is provided to `controller`,
/// [controller] is null.
/// only one ScrollController can be associated with a ScrollView.
final
bool
primary
;
///
/// Setting to false will explicitly prevent inheriting any
/// [PrimaryScrollController].
///
/// Defaults to null. When null, and a controller is not provided,
/// [PrimaryScrollController.shouldInherit] is used to decide automatic
/// inheritance.
///
/// By default, the [PrimaryScrollController] that is injected by each
/// [ModalRoute] is configured to automatically be inherited on
/// [TargetPlatformVariant.mobile] for ScrollViews in the [Axis.vertical]
/// scroll direction. Adding another to your app will override the
/// PrimaryScrollController above it.
/// {@endtemplate}
final
bool
?
primary
;
/// {@template flutter.widgets.scroll_view.physics}
/// {@template flutter.widgets.scroll_view.physics}
/// How the scroll view should respond to user input.
/// How the scroll view should respond to user input.
...
@@ -393,8 +407,13 @@ abstract class ScrollView extends StatelessWidget {
...
@@ -393,8 +407,13 @@ abstract class ScrollView extends StatelessWidget {
final
List
<
Widget
>
slivers
=
buildSlivers
(
context
);
final
List
<
Widget
>
slivers
=
buildSlivers
(
context
);
final
AxisDirection
axisDirection
=
getDirection
(
context
);
final
AxisDirection
axisDirection
=
getDirection
(
context
);
final
ScrollController
?
scrollController
=
final
bool
effectivePrimary
=
primary
primary
?
PrimaryScrollController
.
of
(
context
)
:
controller
;
??
controller
==
null
&&
PrimaryScrollController
.
shouldInherit
(
context
,
scrollDirection
);
final
ScrollController
?
scrollController
=
effectivePrimary
?
PrimaryScrollController
.
of
(
context
)
:
controller
;
final
Scrollable
scrollable
=
Scrollable
(
final
Scrollable
scrollable
=
Scrollable
(
dragStartBehavior:
dragStartBehavior
,
dragStartBehavior:
dragStartBehavior
,
axisDirection:
axisDirection
,
axisDirection:
axisDirection
,
...
@@ -407,7 +426,9 @@ abstract class ScrollView extends StatelessWidget {
...
@@ -407,7 +426,9 @@ abstract class ScrollView extends StatelessWidget {
return
buildViewport
(
context
,
offset
,
axisDirection
,
slivers
);
return
buildViewport
(
context
,
offset
,
axisDirection
,
slivers
);
},
},
);
);
final
Widget
scrollableResult
=
primary
&&
scrollController
!=
null
final
Widget
scrollableResult
=
effectivePrimary
&&
scrollController
!=
null
// Further descendant ScrollViews will not inherit the same PrimaryScrollController
?
PrimaryScrollController
.
none
(
child:
scrollable
)
?
PrimaryScrollController
.
none
(
child:
scrollable
)
:
scrollable
;
:
scrollable
;
...
...
packages/flutter/lib/src/widgets/scrollable.dart
View file @
784520bd
...
@@ -1586,13 +1586,8 @@ class ScrollAction extends Action<ScrollIntent> {
...
@@ -1586,13 +1586,8 @@ class ScrollAction extends Action<ScrollIntent> {
return
true
;
return
true
;
}
}
// Check for fallback scrollable with context from PrimaryScrollController
// Check for fallback scrollable with context from PrimaryScrollController
if
(
PrimaryScrollController
.
of
(
focus
.
context
!)
!=
null
)
{
final
ScrollController
?
primaryScrollController
=
PrimaryScrollController
.
of
(
focus
.
context
!);
final
ScrollController
?
primaryScrollController
=
PrimaryScrollController
.
of
(
focus
.
context
!);
return
primaryScrollController
!=
null
&&
primaryScrollController
.
hasClients
;
return
primaryScrollController
!=
null
&&
primaryScrollController
.
hasClients
&&
primaryScrollController
.
position
.
context
.
notificationContext
!=
null
&&
Scrollable
.
of
(
primaryScrollController
.
position
.
context
.
notificationContext
!)
!=
null
;
}
}
}
return
false
;
return
false
;
}
}
...
@@ -1681,7 +1676,34 @@ class ScrollAction extends Action<ScrollIntent> {
...
@@ -1681,7 +1676,34 @@ class ScrollAction extends Action<ScrollIntent> {
ScrollableState
?
state
=
Scrollable
.
of
(
primaryFocus
!.
context
!);
ScrollableState
?
state
=
Scrollable
.
of
(
primaryFocus
!.
context
!);
if
(
state
==
null
)
{
if
(
state
==
null
)
{
final
ScrollController
?
primaryScrollController
=
PrimaryScrollController
.
of
(
primaryFocus
!.
context
!);
final
ScrollController
?
primaryScrollController
=
PrimaryScrollController
.
of
(
primaryFocus
!.
context
!);
state
=
Scrollable
.
of
(
primaryScrollController
!.
position
.
context
.
notificationContext
!);
assert
(()
{
if
(
primaryScrollController
!.
positions
.
length
!=
1
)
{
throw
FlutterError
.
fromParts
(<
DiagnosticsNode
>[
ErrorSummary
(
'A ScrollAction was invoked with the PrimaryScrollController, but '
'more than one ScrollPosition is attached.'
,
),
ErrorDescription
(
'Only one ScrollPosition can be manipulated by a ScrollAction at '
'a time.'
,
),
ErrorHint
(
'The PrimaryScrollController can be inherited automatically by '
'descendant ScrollViews based on the TargetPlatform and scroll '
'direction. By default, the PrimaryScrollController is '
'automatically inherited on mobile platforms for vertical '
'ScrollViews. ScrollView.primary can also override this behavior.'
,
),
]);
}
return
true
;
}());
if
(
primaryScrollController
!.
position
.
context
.
notificationContext
==
null
&&
Scrollable
.
of
(
primaryScrollController
.
position
.
context
.
notificationContext
!)
==
null
)
{
return
;
}
state
=
Scrollable
.
of
(
primaryScrollController
.
position
.
context
.
notificationContext
!);
}
}
assert
(
state
!=
null
,
'
$ScrollAction
was invoked on a context that has no scrollable parent'
);
assert
(
state
!=
null
,
'
$ScrollAction
was invoked on a context that has no scrollable parent'
);
assert
(
state
!.
position
.
hasPixels
,
'Scrollable must be laid out before it can be scrolled via a ScrollAction'
);
assert
(
state
!.
position
.
hasPixels
,
'Scrollable must be laid out before it can be scrolled via a ScrollAction'
);
...
...
packages/flutter/lib/src/widgets/scrollbar.dart
View file @
784520bd
...
@@ -849,9 +849,12 @@ class ScrollbarPainter extends ChangeNotifier implements CustomPainter {
...
@@ -849,9 +849,12 @@ class ScrollbarPainter extends ChangeNotifier implements CustomPainter {
/// This sample shows an app with two scrollables in the same route. Since by
/// This sample shows an app with two scrollables in the same route. Since by
/// default, there is one [PrimaryScrollController] per route, and they both have a
/// default, there is one [PrimaryScrollController] per route, and they both have a
/// scroll direction of [Axis.vertical], they would both try to attach to that
/// scroll direction of [Axis.vertical], they would both try to attach to that
/// controller. The [Scrollbar] cannot support multiple positions attached to
/// controller on mobile platforms. The [Scrollbar] cannot support multiple
/// the same controller, so one [ListView], and its [Scrollbar] have been
/// positions attached to the same controller, so one [ListView], and its
/// provided a unique [ScrollController].
/// [Scrollbar] have been provided a unique [ScrollController]. Desktop
/// platforms do not automatically attach to the PrimaryScrollController,
/// requiring [ScrollView.primary] to be true instead in order to use the
/// PrimaryScrollController.
///
///
/// Alternatively, a new PrimaryScrollController could be created above one of
/// Alternatively, a new PrimaryScrollController could be created above one of
/// the [ListView]s.
/// the [ListView]s.
...
@@ -1507,14 +1510,14 @@ class RawScrollbarState<T extends RawScrollbar> extends State<T> with TickerProv
...
@@ -1507,14 +1510,14 @@ class RawScrollbarState<T extends RawScrollbar> extends State<T> with TickerProv
ErrorHint
(
ErrorHint
(
'The Scrollbar attempted to use the
$controllerForError
. This '
'The Scrollbar attempted to use the
$controllerForError
. This '
'ScrollController should be associated with the ScrollView that '
'ScrollController should be associated with the ScrollView that '
'the Scrollbar is being applied to.
'
'the Scrollbar is being applied to.'
'
${tryPrimary
'
${tryPrimary
? 'A ScrollView with an Axis.vertical '
? 'A ScrollView with an Axis.vertical
ScrollDirection on mobile
'
'
ScrollDirection
will automatically use the '
'
platforms
will automatically use the '
'PrimaryScrollController if the user has not provided a '
'PrimaryScrollController if the user has not provided a '
'ScrollController
, but a ScrollDirection of Axis.horizontal will
'
'ScrollController
. To use the PrimaryScrollController
'
'
not. To use the PrimaryScrollController explicitly, set ScrollView.primary
'
'
explicitly, set ScrollView.primary to true for the Scrollable
'
'
to true for the Scrollable
widget.'
'widget.'
: 'When providing your own ScrollController, ensure both the '
: 'When providing your own ScrollController, ensure both the '
'Scrollbar and the Scrollable widget use the same one.'
'Scrollbar and the Scrollable widget use the same one.'
}
'
,
}
'
,
...
@@ -1539,16 +1542,17 @@ class RawScrollbarState<T extends RawScrollbar> extends State<T> with TickerProv
...
@@ -1539,16 +1542,17 @@ class RawScrollbarState<T extends RawScrollbar> extends State<T> with TickerProv
'The Scrollbar requires a single ScrollPosition in order to be painted.'
,
'The Scrollbar requires a single ScrollPosition in order to be painted.'
,
),
),
ErrorHint
(
ErrorHint
(
'When
$when
, the associated Scroll
abl
e '
'When
$when
, the associated Scroll
Controller must only have on
e '
'
widgets must have unique ScrollControllers.
'
'
ScrollPosition attached.
'
'
${tryPrimary
'
${tryPrimary
? 'The PrimaryScrollController is used by default for '
? 'If a ScrollController has not been provided, the '
'ScrollViews with an Axis.vertical ScrollDirection, '
'PrimaryScrollController is used by default on mobile platforms '
'unless the ScrollView has been provided its own '
'for ScrollViews with an Axis.vertical scroll direction. More '
'ScrollController. More than one Scrollable may have tried '
'than one ScrollView may have tried to use the '
'to use the PrimaryScrollController of the current context.'
'PrimaryScrollController of the current context. '
: 'The provided ScrollController must be unique to a '
'ScrollView.primary can override this behavior.'
'Scrollable widget.'
: 'The provided ScrollController must be unique to one '
'ScrollView widget.'
}
'
,
}
'
,
),
),
]);
]);
...
...
packages/flutter/test/widgets/range_maintaining_scroll_physics_test.dart
View file @
784520bd
...
@@ -2,6 +2,7 @@
...
@@ -2,6 +2,7 @@
// Use of this source code is governed by a BSD-style license that can be
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// found in the LICENSE file.
import
'package:flutter/foundation.dart'
;
import
'package:flutter/gestures.dart'
;
import
'package:flutter/gestures.dart'
;
import
'package:flutter/material.dart'
;
import
'package:flutter/material.dart'
;
import
'package:flutter_test/flutter_test.dart'
;
import
'package:flutter_test/flutter_test.dart'
;
...
@@ -377,7 +378,7 @@ class RangeMaintainingTestScrollBehavior extends ScrollBehavior {
...
@@ -377,7 +378,7 @@ class RangeMaintainingTestScrollBehavior extends ScrollBehavior {
const
RangeMaintainingTestScrollBehavior
();
const
RangeMaintainingTestScrollBehavior
();
@override
@override
TargetPlatform
getPlatform
(
BuildContext
context
)
=>
throw
'should not be called'
;
TargetPlatform
getPlatform
(
BuildContext
context
)
=>
defaultTargetPlatform
;
@override
@override
Widget
buildOverscrollIndicator
(
BuildContext
context
,
Widget
child
,
ScrollableDetails
details
)
{
Widget
buildOverscrollIndicator
(
BuildContext
context
,
Widget
child
,
ScrollableDetails
details
)
{
...
...
packages/flutter/test/widgets/scroll_view_test.dart
View file @
784520bd
...
@@ -54,6 +54,19 @@ Widget textFieldBoilerplate({ required Widget child }) {
...
@@ -54,6 +54,19 @@ Widget textFieldBoilerplate({ required Widget child }) {
);
);
}
}
Widget
primaryScrollControllerBoilerplate
(
{
required
Widget
child
,
required
ScrollController
controller
})
{
return
Directionality
(
textDirection:
TextDirection
.
ltr
,
child:
MediaQuery
(
data:
const
MediaQueryData
(),
child:
PrimaryScrollController
(
controller:
controller
,
child:
child
,
),
),
);
}
void
main
(
)
{
void
main
(
)
{
testWidgets
(
'ListView control test'
,
(
WidgetTester
tester
)
async
{
testWidgets
(
'ListView control test'
,
(
WidgetTester
tester
)
async
{
final
List
<
String
>
log
=
<
String
>[];
final
List
<
String
>
log
=
<
String
>[];
...
@@ -887,61 +900,155 @@ void main() {
...
@@ -887,61 +900,155 @@ void main() {
expect
(
log
,
isEmpty
);
expect
(
log
,
isEmpty
);
});
});
testWidgets
(
'Vertical CustomScrollViews are primary by default'
,
(
WidgetTester
tester
)
async
{
test
(
'PrimaryScrollController.automaticallyInheritOnPlatforms defaults to all mobile platforms'
,
(){
final
PrimaryScrollController
primaryScrollController
=
PrimaryScrollController
(
controller:
ScrollController
(),
child:
const
SizedBox
(),
);
expect
(
primaryScrollController
.
automaticallyInheritForPlatforms
,
TargetPlatformVariant
.
mobile
().
values
,
);
});
testWidgets
(
'Vertical CustomScrollViews are not primary by default'
,
(
WidgetTester
tester
)
async
{
const
CustomScrollView
view
=
CustomScrollView
();
const
CustomScrollView
view
=
CustomScrollView
();
expect
(
view
.
primary
,
is
True
);
expect
(
view
.
primary
,
is
Null
);
});
});
testWidgets
(
'Vertical ListViews are primary by default'
,
(
WidgetTester
tester
)
async
{
testWidgets
(
'Vertical CustomScrollViews use PrimaryScrollController by default on mobile'
,
(
WidgetTester
tester
)
async
{
final
ScrollController
controller
=
ScrollController
();
await
tester
.
pumpWidget
(
primaryScrollControllerBoilerplate
(
child:
const
CustomScrollView
(),
controller:
controller
,
));
expect
(
controller
.
hasClients
,
isTrue
);
},
variant:
TargetPlatformVariant
.
mobile
());
testWidgets
(
"Vertical CustomScrollViews don't use PrimaryScrollController by default on desktop"
,
(
WidgetTester
tester
)
async
{
final
ScrollController
controller
=
ScrollController
();
await
tester
.
pumpWidget
(
primaryScrollControllerBoilerplate
(
child:
const
CustomScrollView
(),
controller:
controller
,
));
expect
(
controller
.
hasClients
,
isFalse
);
},
variant:
TargetPlatformVariant
.
desktop
());
testWidgets
(
'Vertical ListViews are not primary by default'
,
(
WidgetTester
tester
)
async
{
final
ListView
view
=
ListView
();
final
ListView
view
=
ListView
();
expect
(
view
.
primary
,
is
True
);
expect
(
view
.
primary
,
is
Null
);
});
});
testWidgets
(
'Vertical GridViews are primary by default'
,
(
WidgetTester
tester
)
async
{
testWidgets
(
'Vertical ListViews use PrimaryScrollController by default on mobile'
,
(
WidgetTester
tester
)
async
{
final
GridView
view
=
GridView
.
count
(
final
ScrollController
controller
=
ScrollController
();
crossAxisCount:
1
,
await
tester
.
pumpWidget
(
primaryScrollControllerBoilerplate
(
);
child:
ListView
(),
expect
(
view
.
primary
,
isTrue
);
controller:
controller
,
));
expect
(
controller
.
hasClients
,
isTrue
);
},
variant:
TargetPlatformVariant
.
mobile
());
testWidgets
(
"Vertical ListViews don't use PrimaryScrollController by default on desktop"
,
(
WidgetTester
tester
)
async
{
final
ScrollController
controller
=
ScrollController
();
await
tester
.
pumpWidget
(
primaryScrollControllerBoilerplate
(
child:
ListView
(),
controller:
controller
,
));
expect
(
controller
.
hasClients
,
isFalse
);
},
variant:
TargetPlatformVariant
.
desktop
());
testWidgets
(
'Vertical GridViews are not primary by default'
,
(
WidgetTester
tester
)
async
{
final
GridView
view
=
GridView
.
count
(
crossAxisCount:
1
);
expect
(
view
.
primary
,
isNull
);
});
});
testWidgets
(
'Vertical GridViews use PrimaryScrollController by default on mobile'
,
(
WidgetTester
tester
)
async
{
final
ScrollController
controller
=
ScrollController
();
await
tester
.
pumpWidget
(
primaryScrollControllerBoilerplate
(
child:
GridView
.
count
(
crossAxisCount:
1
),
controller:
controller
,
));
expect
(
controller
.
hasClients
,
isTrue
);
},
variant:
TargetPlatformVariant
.
mobile
());
testWidgets
(
"Vertical GridViews don't use PrimaryScrollController by default on desktop"
,
(
WidgetTester
tester
)
async
{
final
ScrollController
controller
=
ScrollController
();
await
tester
.
pumpWidget
(
primaryScrollControllerBoilerplate
(
child:
GridView
.
count
(
crossAxisCount:
1
),
controller:
controller
,
));
expect
(
controller
.
hasClients
,
isFalse
);
},
variant:
TargetPlatformVariant
.
desktop
());
testWidgets
(
'Horizontal CustomScrollViews are non-primary by default'
,
(
WidgetTester
tester
)
async
{
testWidgets
(
'Horizontal CustomScrollViews are non-primary by default'
,
(
WidgetTester
tester
)
async
{
const
CustomScrollView
view
=
CustomScrollView
(
scrollDirection:
Axis
.
horizontal
);
final
ScrollController
controller
=
ScrollController
();
expect
(
view
.
primary
,
isFalse
);
await
tester
.
pumpWidget
(
primaryScrollControllerBoilerplate
(
child:
CustomScrollView
(
scrollDirection:
Axis
.
horizontal
,
controller:
ScrollController
(),
),
controller:
controller
,
));
expect
(
controller
.
hasClients
,
isFalse
);
});
});
testWidgets
(
'Horizontal ListViews are non-primary by default'
,
(
WidgetTester
tester
)
async
{
testWidgets
(
'Horizontal ListViews are non-primary by default'
,
(
WidgetTester
tester
)
async
{
final
ListView
view
=
ListView
(
scrollDirection:
Axis
.
horizontal
);
final
ScrollController
controller
=
ScrollController
();
expect
(
view
.
primary
,
isFalse
);
await
tester
.
pumpWidget
(
primaryScrollControllerBoilerplate
(
child:
ListView
(
scrollDirection:
Axis
.
horizontal
,
controller:
ScrollController
(),
),
controller:
controller
,
));
expect
(
controller
.
hasClients
,
isFalse
);
});
});
testWidgets
(
'Horizontal GridViews are non-primary by default'
,
(
WidgetTester
tester
)
async
{
testWidgets
(
'Horizontal GridViews are non-primary by default'
,
(
WidgetTester
tester
)
async
{
final
GridView
view
=
GridView
.
count
(
final
ScrollController
controller
=
ScrollController
();
scrollDirection:
Axis
.
horizontal
,
await
tester
.
pumpWidget
(
primaryScrollControllerBoilerplate
(
crossAxisCount:
1
,
child:
GridView
.
count
(
);
scrollDirection:
Axis
.
horizontal
,
expect
(
view
.
primary
,
isFalse
);
controller:
ScrollController
(),
crossAxisCount:
1
,
),
controller:
controller
,
));
expect
(
controller
.
hasClients
,
isFalse
);
});
});
testWidgets
(
'CustomScrollViews with controllers are non-primary by default'
,
(
WidgetTester
tester
)
async
{
testWidgets
(
'CustomScrollViews with controllers are non-primary by default'
,
(
WidgetTester
tester
)
async
{
final
CustomScrollView
view
=
CustomScrollView
(
final
ScrollController
controller
=
ScrollController
();
controller:
ScrollController
(),
await
tester
.
pumpWidget
(
primaryScrollControllerBoilerplate
(
);
child:
CustomScrollView
(
expect
(
view
.
primary
,
isFalse
);
controller:
ScrollController
(),
),
controller:
controller
,
));
expect
(
controller
.
hasClients
,
isFalse
);
});
});
testWidgets
(
'ListViews with controllers are non-primary by default'
,
(
WidgetTester
tester
)
async
{
testWidgets
(
'ListViews with controllers are non-primary by default'
,
(
WidgetTester
tester
)
async
{
final
ListView
view
=
ListView
(
final
ScrollController
controller
=
ScrollController
();
controller:
ScrollController
(),
await
tester
.
pumpWidget
(
primaryScrollControllerBoilerplate
(
);
child:
ListView
(
expect
(
view
.
primary
,
isFalse
);
controller:
ScrollController
(),
),
controller:
controller
,
));
expect
(
controller
.
hasClients
,
isFalse
);
});
});
testWidgets
(
'GridViews with controllers are non-primary by default'
,
(
WidgetTester
tester
)
async
{
testWidgets
(
'GridViews with controllers are non-primary by default'
,
(
WidgetTester
tester
)
async
{
final
GridView
view
=
GridView
.
count
(
final
ScrollController
controller
=
ScrollController
();
controller:
ScrollController
(),
await
tester
.
pumpWidget
(
primaryScrollControllerBoilerplate
(
crossAxisCount:
1
,
child:
GridView
.
count
(
);
controller:
ScrollController
(),
expect
(
view
.
primary
,
isFalse
);
crossAxisCount:
1
,
),
controller:
controller
,
));
expect
(
controller
.
hasClients
,
isFalse
);
});
});
testWidgets
(
'CustomScrollView sets PrimaryScrollController when primary'
,
(
WidgetTester
tester
)
async
{
testWidgets
(
'CustomScrollView sets PrimaryScrollController when primary'
,
(
WidgetTester
tester
)
async
{
...
@@ -1318,6 +1425,55 @@ void main() {
...
@@ -1318,6 +1425,55 @@ void main() {
);
);
});
});
testWidgets
(
'Fallback ScrollActions handle too many positions with error message'
,
(
WidgetTester
tester
)
async
{
Widget
getScrollView
()
{
return
SizedBox
(
width:
400.0
,
child:
CustomScrollView
(
primary:
true
,
slivers:
List
<
Widget
>.
generate
(
20
,
(
int
index
)
{
return
SliverToBoxAdapter
(
child:
Focus
(
child:
SizedBox
(
key:
ValueKey
<
String
>(
'Box
$index
'
),
height:
50.0
),
),
);
},
),
),
);
}
await
tester
.
pumpWidget
(
MaterialApp
(
home:
Row
(
children:
<
Widget
>[
getScrollView
(),
getScrollView
(),
],
),
),
);
await
tester
.
pumpAndSettle
();
expect
(
tester
.
getRect
(
find
.
byKey
(
const
ValueKey
<
String
>(
'Box 0'
),
skipOffstage:
false
).
first
),
equals
(
const
Rect
.
fromLTRB
(
0.0
,
0.0
,
400.0
,
50.0
)),
);
await
tester
.
sendKeyEvent
(
LogicalKeyboardKey
.
pageDown
);
final
AssertionError
exception
=
tester
.
takeException
()
as
AssertionError
;
expect
(
exception
,
isAssertionError
);
expect
(
exception
.
message
,
contains
(
'A ScrollAction was invoked with the PrimaryScrollController, but '
'more than one ScrollPosition is attached.'
),
);
});
testWidgets
(
'if itemExtent is non-null, children have same extent in the scroll direction'
,
(
WidgetTester
tester
)
async
{
testWidgets
(
'if itemExtent is non-null, children have same extent in the scroll direction'
,
(
WidgetTester
tester
)
async
{
final
List
<
int
>
numbers
=
<
int
>[
0
,
1
,
2
];
final
List
<
int
>
numbers
=
<
int
>[
0
,
1
,
2
];
...
...
packages/flutter/test/widgets/scrollbar_test.dart
View file @
784520bd
...
@@ -2203,7 +2203,7 @@ void main() {
...
@@ -2203,7 +2203,7 @@ void main() {
await
tester
.
pumpAndSettle
();
await
tester
.
pumpAndSettle
();
});
});
testWidgets
(
'Scrollbar thumb can be dragged when the scrollable widget has a negative minScrollExtent'
,
(
WidgetTester
tester
)
async
{
testWidgets
(
'Scrollbar thumb can be dragged when the scrollable widget has a negative minScrollExtent
- desktop
'
,
(
WidgetTester
tester
)
async
{
// Regression test for https://github.com/flutter/flutter/issues/95840
// Regression test for https://github.com/flutter/flutter/issues/95840
final
ScrollController
scrollController
=
ScrollController
();
final
ScrollController
scrollController
=
ScrollController
();
...
@@ -2223,6 +2223,7 @@ void main() {
...
@@ -2223,6 +2223,7 @@ void main() {
isAlwaysShown:
true
,
isAlwaysShown:
true
,
controller:
scrollController
,
controller:
scrollController
,
child:
CustomScrollView
(
child:
CustomScrollView
(
primary:
true
,
center:
uniqueKey
,
center:
uniqueKey
,
slivers:
<
Widget
>[
slivers:
<
Widget
>[
SliverToBoxAdapter
(
SliverToBoxAdapter
(
...
@@ -2282,7 +2283,88 @@ void main() {
...
@@ -2282,7 +2283,88 @@ void main() {
color:
const
Color
(
0x66BCBCBC
),
color:
const
Color
(
0x66BCBCBC
),
),
),
);
);
},
variant:
TargetPlatformVariant
.
all
());
},
variant:
TargetPlatformVariant
.
desktop
());
testWidgets
(
'Scrollbar thumb can be dragged when the scrollable widget has a negative minScrollExtent - mobile'
,
(
WidgetTester
tester
)
async
{
// Regression test for https://github.com/flutter/flutter/issues/95840
final
ScrollController
scrollController
=
ScrollController
();
final
UniqueKey
uniqueKey
=
UniqueKey
();
await
tester
.
pumpWidget
(
Directionality
(
textDirection:
TextDirection
.
ltr
,
child:
MediaQuery
(
data:
const
MediaQueryData
(),
child:
ScrollConfiguration
(
behavior:
const
ScrollBehavior
().
copyWith
(
scrollbars:
false
,
),
child:
PrimaryScrollController
(
controller:
scrollController
,
child:
RawScrollbar
(
isAlwaysShown:
true
,
controller:
scrollController
,
child:
CustomScrollView
(
center:
uniqueKey
,
slivers:
<
Widget
>[
SliverToBoxAdapter
(
child:
Container
(
height:
600.0
,
),
),
SliverToBoxAdapter
(
key:
uniqueKey
,
child:
Container
(
height:
600.0
,
),
),
SliverToBoxAdapter
(
child:
Container
(
height:
600.0
,
),
),
],
),
),
),
),
),
),
);
await
tester
.
pumpAndSettle
();
expect
(
scrollController
.
offset
,
0.0
);
expect
(
find
.
byType
(
RawScrollbar
),
paints
..
rect
(
rect:
const
Rect
.
fromLTRB
(
794.0
,
0.0
,
800.0
,
600.0
))
..
rect
(
rect:
const
Rect
.
fromLTRB
(
794.0
,
200.0
,
800.0
,
400.0
),
color:
const
Color
(
0x66BCBCBC
),
),
);
// Drag the thumb up to scroll up.
const
double
scrollAmount
=
-
10.0
;
final
TestGesture
dragScrollbarGesture
=
await
tester
.
startGesture
(
const
Offset
(
797.0
,
300.0
));
await
tester
.
pumpAndSettle
();
await
dragScrollbarGesture
.
moveBy
(
const
Offset
(
0.0
,
scrollAmount
));
await
tester
.
pumpAndSettle
();
await
dragScrollbarGesture
.
up
();
await
tester
.
pumpAndSettle
();
// The view has scrolled more than it would have by a swipe gesture of the
// same distance.
expect
(
scrollController
.
offset
,
lessThan
(
scrollAmount
*
2
));
expect
(
find
.
byType
(
RawScrollbar
),
paints
..
rect
(
rect:
const
Rect
.
fromLTRB
(
794.0
,
0.0
,
800.0
,
600.0
))
..
rect
(
rect:
const
Rect
.
fromLTRB
(
794.0
,
190.0
,
800.0
,
390.0
),
color:
const
Color
(
0x66BCBCBC
),
),
);
},
variant:
TargetPlatformVariant
.
mobile
());
test
(
'ScrollbarPainter.shouldRepaint returns true when any of the properties changes'
,
()
{
test
(
'ScrollbarPainter.shouldRepaint returns true when any of the properties changes'
,
()
{
ScrollbarPainter
createPainter
({
ScrollbarPainter
createPainter
({
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment