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
48b51c3f
Unverified
Commit
48b51c3f
authored
Oct 20, 2021
by
Kate Lovett
Committed by
GitHub
Oct 20, 2021
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fix ScrollBehavior copyWith (#91834)
parent
b11e9620
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
205 additions
and
23 deletions
+205
-23
scroll_configuration.dart
packages/flutter/lib/src/widgets/scroll_configuration.dart
+28
-23
scroll_behavior_test.dart
packages/flutter/test/widgets/scroll_behavior_test.dart
+177
-0
No files found.
packages/flutter/lib/src/widgets/scroll_configuration.dart
View file @
48b51c3f
...
@@ -85,19 +85,21 @@ class ScrollBehavior {
...
@@ -85,19 +85,21 @@ class ScrollBehavior {
/// the widget level, like [PageView.scrollBehavior], in order to change the
/// the widget level, like [PageView.scrollBehavior], in order to change the
/// default.
/// default.
ScrollBehavior
copyWith
({
ScrollBehavior
copyWith
({
bool
scrollbars
=
true
,
bool
?
scrollbars
,
bool
overscroll
=
true
,
bool
?
overscroll
,
Set
<
PointerDeviceKind
>?
dragDevices
,
Set
<
PointerDeviceKind
>?
dragDevices
,
ScrollPhysics
?
physics
,
ScrollPhysics
?
physics
,
TargetPlatform
?
platform
,
TargetPlatform
?
platform
,
AndroidOverscrollIndicator
?
androidOverscrollIndicator
,
})
{
})
{
return
_WrappedScrollBehavior
(
return
_WrappedScrollBehavior
(
delegate:
this
,
delegate:
this
,
scrollbar
:
scrollbars
,
scrollbar
s:
scrollbars
??
true
,
overscroll
Indicator:
overscroll
,
overscroll
:
overscroll
??
true
,
physics:
physics
,
physics:
physics
,
platform:
platform
,
platform:
platform
,
dragDevices:
dragDevices
,
dragDevices:
dragDevices
,
androidOverscrollIndicator:
androidOverscrollIndicator
);
);
}
}
...
@@ -251,38 +253,40 @@ class ScrollBehavior {
...
@@ -251,38 +253,40 @@ class ScrollBehavior {
class
_WrappedScrollBehavior
implements
ScrollBehavior
{
class
_WrappedScrollBehavior
implements
ScrollBehavior
{
const
_WrappedScrollBehavior
({
const
_WrappedScrollBehavior
({
required
this
.
delegate
,
required
this
.
delegate
,
this
.
scrollbar
=
true
,
this
.
scrollbar
s
=
true
,
this
.
overscroll
Indicator
=
true
,
this
.
overscroll
=
true
,
this
.
physics
,
this
.
physics
,
this
.
platform
,
this
.
platform
,
Set
<
PointerDeviceKind
>?
dragDevices
,
Set
<
PointerDeviceKind
>?
dragDevices
,
})
:
_dragDevices
=
dragDevices
;
AndroidOverscrollIndicator
?
androidOverscrollIndicator
,
})
:
_androidOverscrollIndicator
=
androidOverscrollIndicator
,
_dragDevices
=
dragDevices
;
final
ScrollBehavior
delegate
;
final
ScrollBehavior
delegate
;
final
bool
scrollbar
;
final
bool
scrollbar
s
;
final
bool
overscroll
Indicator
;
final
bool
overscroll
;
final
ScrollPhysics
?
physics
;
final
ScrollPhysics
?
physics
;
final
TargetPlatform
?
platform
;
final
TargetPlatform
?
platform
;
final
Set
<
PointerDeviceKind
>?
_dragDevices
;
final
Set
<
PointerDeviceKind
>?
_dragDevices
;
@override
final
AndroidOverscrollIndicator
?
_androidOverscrollIndicator
;
@override
@override
Set
<
PointerDeviceKind
>
get
dragDevices
=>
_dragDevices
??
delegate
.
dragDevices
;
Set
<
PointerDeviceKind
>
get
dragDevices
=>
_dragDevices
??
delegate
.
dragDevices
;
@override
@override
AndroidOverscrollIndicator
get
androidOverscrollIndicator
=>
delegate
.
androidOverscrollIndicator
;
AndroidOverscrollIndicator
get
androidOverscrollIndicator
=>
_androidOverscrollIndicator
??
delegate
.
androidOverscrollIndicator
;
@override
AndroidOverscrollIndicator
?
get
_androidOverscrollIndicator
=>
throw
UnimplementedError
();
@override
@override
Widget
buildOverscrollIndicator
(
BuildContext
context
,
Widget
child
,
ScrollableDetails
details
)
{
Widget
buildOverscrollIndicator
(
BuildContext
context
,
Widget
child
,
ScrollableDetails
details
)
{
if
(
overscroll
Indicator
)
if
(
overscroll
)
return
delegate
.
buildOverscrollIndicator
(
context
,
child
,
details
);
return
delegate
.
buildOverscrollIndicator
(
context
,
child
,
details
);
return
child
;
return
child
;
}
}
@override
@override
Widget
buildScrollbar
(
BuildContext
context
,
Widget
child
,
ScrollableDetails
details
)
{
Widget
buildScrollbar
(
BuildContext
context
,
Widget
child
,
ScrollableDetails
details
)
{
if
(
scrollbar
)
if
(
scrollbar
s
)
return
delegate
.
buildScrollbar
(
context
,
child
,
details
);
return
delegate
.
buildScrollbar
(
context
,
child
,
details
);
return
child
;
return
child
;
}
}
...
@@ -294,19 +298,20 @@ class _WrappedScrollBehavior implements ScrollBehavior {
...
@@ -294,19 +298,20 @@ class _WrappedScrollBehavior implements ScrollBehavior {
@override
@override
ScrollBehavior
copyWith
({
ScrollBehavior
copyWith
({
bool
scrollbars
=
true
,
bool
?
scrollbars
,
bool
overscroll
=
true
,
bool
?
overscroll
,
ScrollPhysics
?
physics
,
ScrollPhysics
?
physics
,
TargetPlatform
?
platform
,
TargetPlatform
?
platform
,
Set
<
PointerDeviceKind
>?
dragDevices
,
Set
<
PointerDeviceKind
>?
dragDevices
,
AndroidOverscrollIndicator
?
androidOverscrollIndicator
AndroidOverscrollIndicator
?
androidOverscrollIndicator
})
{
})
{
return
delegate
.
copyWith
(
return
delegate
.
copyWith
(
scrollbars:
scrollbars
,
scrollbars:
scrollbars
??
this
.
scrollbars
,
overscroll:
overscroll
,
overscroll:
overscroll
??
this
.
overscroll
,
physics:
physics
,
physics:
physics
??
this
.
physics
,
platform:
platform
,
platform:
platform
??
this
.
platform
,
dragDevices:
dragDevices
,
dragDevices:
dragDevices
??
this
.
dragDevices
,
androidOverscrollIndicator:
androidOverscrollIndicator
??
this
.
androidOverscrollIndicator
,
);
);
}
}
...
@@ -323,8 +328,8 @@ class _WrappedScrollBehavior implements ScrollBehavior {
...
@@ -323,8 +328,8 @@ class _WrappedScrollBehavior implements ScrollBehavior {
@override
@override
bool
shouldNotify
(
_WrappedScrollBehavior
oldDelegate
)
{
bool
shouldNotify
(
_WrappedScrollBehavior
oldDelegate
)
{
return
oldDelegate
.
delegate
.
runtimeType
!=
delegate
.
runtimeType
return
oldDelegate
.
delegate
.
runtimeType
!=
delegate
.
runtimeType
||
oldDelegate
.
scrollbar
!=
scrollbar
||
oldDelegate
.
scrollbar
s
!=
scrollbars
||
oldDelegate
.
overscroll
Indicator
!=
overscrollIndicator
||
oldDelegate
.
overscroll
!=
overscroll
||
oldDelegate
.
physics
!=
physics
||
oldDelegate
.
physics
!=
physics
||
oldDelegate
.
platform
!=
platform
||
oldDelegate
.
platform
!=
platform
||
setEquals
<
PointerDeviceKind
>(
oldDelegate
.
dragDevices
,
dragDevices
)
||
setEquals
<
PointerDeviceKind
>(
oldDelegate
.
dragDevices
,
dragDevices
)
...
...
packages/flutter/test/widgets/scroll_behavior_test.dart
View file @
48b51c3f
...
@@ -125,4 +125,181 @@ void main() {
...
@@ -125,4 +125,181 @@ void main() {
expect
(
find
.
byType
(
StretchingOverscrollIndicator
),
findsOneWidget
);
expect
(
find
.
byType
(
StretchingOverscrollIndicator
),
findsOneWidget
);
expect
(
find
.
byType
(
GlowingOverscrollIndicator
),
findsNothing
);
expect
(
find
.
byType
(
GlowingOverscrollIndicator
),
findsNothing
);
},
variant:
TargetPlatformVariant
.
only
(
TargetPlatform
.
android
));
},
variant:
TargetPlatformVariant
.
only
(
TargetPlatform
.
android
));
group
(
'ScrollBehavior configuration is maintained over multiple copies'
,
()
{
testWidgets
(
'dragDevices'
,
(
WidgetTester
tester
)
async
{
// Regression test for https://github.com/flutter/flutter/issues/91673
const
ScrollBehavior
defaultBehavior
=
ScrollBehavior
();
expect
(
defaultBehavior
.
dragDevices
,
<
PointerDeviceKind
>{
PointerDeviceKind
.
touch
,
PointerDeviceKind
.
stylus
,
PointerDeviceKind
.
invertedStylus
,
});
// Use copyWith to modify drag devices
final
ScrollBehavior
onceCopiedBehavior
=
defaultBehavior
.
copyWith
(
dragDevices:
PointerDeviceKind
.
values
.
toSet
(),
);
expect
(
onceCopiedBehavior
.
dragDevices
,
PointerDeviceKind
.
values
.
toSet
());
// Copy again. The previously modified drag devices should carry over.
final
ScrollBehavior
twiceCopiedBehavior
=
onceCopiedBehavior
.
copyWith
();
expect
(
twiceCopiedBehavior
.
dragDevices
,
PointerDeviceKind
.
values
.
toSet
());
});
testWidgets
(
'androidOverscrollIndicator'
,
(
WidgetTester
tester
)
async
{
// Regression test for https://github.com/flutter/flutter/issues/91673
const
ScrollBehavior
defaultBehavior
=
ScrollBehavior
();
expect
(
defaultBehavior
.
androidOverscrollIndicator
,
AndroidOverscrollIndicator
.
glow
);
// Use copyWith to modify androidOverscrollIndicator
final
ScrollBehavior
onceCopiedBehavior
=
defaultBehavior
.
copyWith
(
androidOverscrollIndicator:
AndroidOverscrollIndicator
.
stretch
,
);
expect
(
onceCopiedBehavior
.
androidOverscrollIndicator
,
AndroidOverscrollIndicator
.
stretch
);
// Copy again. The previously modified value should carry over.
final
ScrollBehavior
twiceCopiedBehavior
=
onceCopiedBehavior
.
copyWith
();
expect
(
twiceCopiedBehavior
.
androidOverscrollIndicator
,
AndroidOverscrollIndicator
.
stretch
);
});
testWidgets
(
'physics'
,
(
WidgetTester
tester
)
async
{
// Regression test for https://github.com/flutter/flutter/issues/91673
late
ScrollPhysics
defaultPhysics
;
late
ScrollPhysics
onceCopiedPhysics
;
late
ScrollPhysics
twiceCopiedPhysics
;
await
tester
.
pumpWidget
(
ScrollConfiguration
(
// Default ScrollBehavior
behavior:
const
ScrollBehavior
(),
child:
Builder
(
builder:
(
BuildContext
context
)
{
final
ScrollBehavior
defaultBehavior
=
ScrollConfiguration
.
of
(
context
);
// Copy once to change physics
defaultPhysics
=
defaultBehavior
.
getScrollPhysics
(
context
);
return
ScrollConfiguration
(
behavior:
defaultBehavior
.
copyWith
(
physics:
const
BouncingScrollPhysics
()),
child:
Builder
(
builder:
(
BuildContext
context
)
{
final
ScrollBehavior
onceCopiedBehavior
=
ScrollConfiguration
.
of
(
context
);
onceCopiedPhysics
=
onceCopiedBehavior
.
getScrollPhysics
(
context
);
return
ScrollConfiguration
(
// Copy again, physics should follow
behavior:
onceCopiedBehavior
.
copyWith
(),
child:
Builder
(
builder:
(
BuildContext
context
)
{
twiceCopiedPhysics
=
ScrollConfiguration
.
of
(
context
).
getScrollPhysics
(
context
);
return
SingleChildScrollView
(
child:
Container
(
height:
1000
));
}
)
);
}
)
);
}
),
));
expect
(
defaultPhysics
,
const
ClampingScrollPhysics
(
parent:
RangeMaintainingScrollPhysics
()));
expect
(
onceCopiedPhysics
,
const
BouncingScrollPhysics
());
expect
(
twiceCopiedPhysics
,
const
BouncingScrollPhysics
());
});
testWidgets
(
'platform'
,
(
WidgetTester
tester
)
async
{
// Regression test for https://github.com/flutter/flutter/issues/91673
late
TargetPlatform
defaultPlatform
;
late
TargetPlatform
onceCopiedPlatform
;
late
TargetPlatform
twiceCopiedPlatform
;
await
tester
.
pumpWidget
(
ScrollConfiguration
(
// Default ScrollBehavior
behavior:
const
ScrollBehavior
(),
child:
Builder
(
builder:
(
BuildContext
context
)
{
final
ScrollBehavior
defaultBehavior
=
ScrollConfiguration
.
of
(
context
);
// Copy once to change physics
defaultPlatform
=
defaultBehavior
.
getPlatform
(
context
);
return
ScrollConfiguration
(
behavior:
defaultBehavior
.
copyWith
(
platform:
TargetPlatform
.
fuchsia
),
child:
Builder
(
builder:
(
BuildContext
context
)
{
final
ScrollBehavior
onceCopiedBehavior
=
ScrollConfiguration
.
of
(
context
);
onceCopiedPlatform
=
onceCopiedBehavior
.
getPlatform
(
context
);
return
ScrollConfiguration
(
// Copy again, physics should follow
behavior:
onceCopiedBehavior
.
copyWith
(),
child:
Builder
(
builder:
(
BuildContext
context
)
{
twiceCopiedPlatform
=
ScrollConfiguration
.
of
(
context
).
getPlatform
(
context
);
return
SingleChildScrollView
(
child:
Container
(
height:
1000
));
}
)
);
}
)
);
}
),
));
expect
(
defaultPlatform
,
TargetPlatform
.
android
);
expect
(
onceCopiedPlatform
,
TargetPlatform
.
fuchsia
);
expect
(
twiceCopiedPlatform
,
TargetPlatform
.
fuchsia
);
});
Widget
wrap
(
ScrollBehavior
behavior
)
{
return
Directionality
(
textDirection:
TextDirection
.
ltr
,
child:
MediaQuery
(
data:
const
MediaQueryData
(
size:
Size
(
500
,
500
)),
child:
ScrollConfiguration
(
behavior:
behavior
,
child:
Builder
(
builder:
(
BuildContext
context
)
=>
SingleChildScrollView
(
child:
Container
(
height:
1000
))
)
),
)
);
}
testWidgets
(
'scrollbar'
,
(
WidgetTester
tester
)
async
{
// Regression test for https://github.com/flutter/flutter/issues/91673
const
ScrollBehavior
defaultBehavior
=
ScrollBehavior
();
await
tester
.
pumpWidget
(
wrap
(
defaultBehavior
));
// Default adds a scrollbar
expect
(
find
.
byType
(
RawScrollbar
),
findsOneWidget
);
final
ScrollBehavior
onceCopiedBehavior
=
defaultBehavior
.
copyWith
(
scrollbars:
false
);
await
tester
.
pumpWidget
(
wrap
(
onceCopiedBehavior
));
// Copy does not add scrollbar
expect
(
find
.
byType
(
RawScrollbar
),
findsNothing
);
final
ScrollBehavior
twiceCopiedBehavior
=
onceCopiedBehavior
.
copyWith
();
await
tester
.
pumpWidget
(
wrap
(
twiceCopiedBehavior
));
// Second copy maintains scrollbar setting
expect
(
find
.
byType
(
RawScrollbar
),
findsNothing
);
// For default scrollbars
},
variant:
TargetPlatformVariant
.
desktop
());
testWidgets
(
'overscroll'
,
(
WidgetTester
tester
)
async
{
// Regression test for https://github.com/flutter/flutter/issues/91673
const
ScrollBehavior
defaultBehavior
=
ScrollBehavior
();
await
tester
.
pumpWidget
(
wrap
(
defaultBehavior
));
// Default adds a glowing overscroll indicator
expect
(
find
.
byType
(
GlowingOverscrollIndicator
),
findsOneWidget
);
final
ScrollBehavior
onceCopiedBehavior
=
defaultBehavior
.
copyWith
(
overscroll:
false
);
await
tester
.
pumpWidget
(
wrap
(
onceCopiedBehavior
));
// Copy does not add indicator
expect
(
find
.
byType
(
GlowingOverscrollIndicator
),
findsNothing
);
final
ScrollBehavior
twiceCopiedBehavior
=
onceCopiedBehavior
.
copyWith
();
await
tester
.
pumpWidget
(
wrap
(
twiceCopiedBehavior
));
// Second copy maintains overscroll setting
expect
(
find
.
byType
(
GlowingOverscrollIndicator
),
findsNothing
);
// For default glowing indicator
},
variant:
TargetPlatformVariant
.
only
(
TargetPlatform
.
android
));
});
}
}
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