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
e7772d0e
Unverified
Commit
e7772d0e
authored
Jan 19, 2021
by
xubaolin
Committed by
GitHub
Jan 19, 2021
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Reland "Improve the ScrollBar behavior when nested (#71843)" (#74104)
parent
d3257053
Changes
5
Show whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
152 additions
and
0 deletions
+152
-0
scrollbar.dart
packages/flutter/lib/src/cupertino/scrollbar.dart
+2
-0
scrollbar.dart
packages/flutter/lib/src/material/scrollbar.dart
+8
-0
scrollbar.dart
packages/flutter/lib/src/widgets/scrollbar.dart
+17
-0
scrollbar_test.dart
packages/flutter/test/material/scrollbar_test.dart
+62
-0
scrollbar_test.dart
packages/flutter/test/widgets/scrollbar_test.dart
+63
-0
No files found.
packages/flutter/lib/src/cupertino/scrollbar.dart
View file @
e7772d0e
...
...
@@ -61,6 +61,7 @@ class CupertinoScrollbar extends RawScrollbar {
this
.
thicknessWhileDragging
=
defaultThicknessWhileDragging
,
Radius
radius
=
defaultRadius
,
this
.
radiusWhileDragging
=
defaultRadiusWhileDragging
,
ScrollNotificationPredicate
?
notificationPredicate
,
})
:
assert
(
thickness
!=
null
),
assert
(
thickness
<
double
.
infinity
),
assert
(
thicknessWhileDragging
!=
null
),
...
...
@@ -77,6 +78,7 @@ class CupertinoScrollbar extends RawScrollbar {
fadeDuration:
_kScrollbarFadeDuration
,
timeToFade:
_kScrollbarTimeToFade
,
pressDuration:
const
Duration
(
milliseconds:
100
),
notificationPredicate:
notificationPredicate
??
defaultScrollNotificationPredicate
,
);
/// Default value for [thickness] if it's not specified in [CupertinoScrollbar].
...
...
packages/flutter/lib/src/material/scrollbar.dart
View file @
e7772d0e
...
...
@@ -69,6 +69,7 @@ class Scrollbar extends StatefulWidget {
this
.
hoverThickness
,
this
.
thickness
,
this
.
radius
,
this
.
notificationPredicate
,
})
:
super
(
key:
key
);
/// {@macro flutter.widgets.Scrollbar.child}
...
...
@@ -111,6 +112,9 @@ class Scrollbar extends StatefulWidget {
/// default [Radius.circular] of 8.0 pixels.
final
Radius
?
radius
;
/// {@macro flutter.widgets.Scrollbar.notificationPredicate}
final
ScrollNotificationPredicate
?
notificationPredicate
;
@override
_ScrollbarState
createState
()
=>
_ScrollbarState
();
}
...
...
@@ -129,6 +133,7 @@ class _ScrollbarState extends State<Scrollbar> {
radius:
widget
.
radius
??
CupertinoScrollbar
.
defaultRadius
,
radiusWhileDragging:
widget
.
radius
??
CupertinoScrollbar
.
defaultRadiusWhileDragging
,
controller:
widget
.
controller
,
notificationPredicate:
widget
.
notificationPredicate
,
);
}
return
_MaterialScrollbar
(
...
...
@@ -139,6 +144,7 @@ class _ScrollbarState extends State<Scrollbar> {
hoverThickness:
widget
.
hoverThickness
,
thickness:
widget
.
thickness
,
radius:
widget
.
radius
,
notificationPredicate:
widget
.
notificationPredicate
,
);
}
}
...
...
@@ -153,6 +159,7 @@ class _MaterialScrollbar extends RawScrollbar {
this
.
hoverThickness
,
double
?
thickness
,
Radius
?
radius
,
ScrollNotificationPredicate
?
notificationPredicate
,
})
:
super
(
key:
key
,
child:
child
,
...
...
@@ -163,6 +170,7 @@ class _MaterialScrollbar extends RawScrollbar {
fadeDuration:
_kScrollbarFadeDuration
,
timeToFade:
_kScrollbarTimeToFade
,
pressDuration:
Duration
.
zero
,
notificationPredicate:
notificationPredicate
??
defaultScrollNotificationPredicate
,
);
final
bool
?
showTrackOnHover
;
...
...
packages/flutter/lib/src/widgets/scrollbar.dart
View file @
e7772d0e
...
...
@@ -565,6 +565,10 @@ class ScrollbarPainter extends ChangeNotifier implements CustomPainter {
/// visible without the fade animation. This requires that a [ScrollController]
/// is provided to [controller], or that the [PrimaryScrollController] is available.
///
/// If the scrollbar is wrapped around multiple [ScrollView]s, it only responds to
/// the nearest scrollView and shows the corresponding scrollbar thumb by default.
/// Set [notificationPredicate] to something else for more complicated behaviors.
///
/// Scrollbars are interactive and will also use the [PrimaryScrollController] if
/// a [controller] is not set. Scrollbar thumbs can be dragged along the main axis
/// of the [ScrollView] to change the [ScrollPosition]. Tapping along the track
...
...
@@ -607,6 +611,7 @@ class RawScrollbar extends StatefulWidget {
this
.
fadeDuration
=
_kScrollbarFadeDuration
,
this
.
timeToFade
=
_kScrollbarTimeToFade
,
this
.
pressDuration
=
Duration
.
zero
,
this
.
notificationPredicate
=
defaultScrollNotificationPredicate
,
})
:
assert
(
child
!=
null
),
assert
(
fadeDuration
!=
null
),
assert
(
timeToFade
!=
null
),
...
...
@@ -767,6 +772,16 @@ class RawScrollbar extends StatefulWidget {
/// Cannot be null, defaults to [Duration.zero].
final
Duration
pressDuration
;
/// {@template flutter.widgets.Scrollbar.notificationPredicate}
/// A check that specifies whether a [ScrollNotification] should be
/// handled by this widget.
///
/// By default, checks whether `notification.depth == 0`. That means if the
/// scrollbar is wrapped around multiple [ScrollView]s, it only responds to the
/// nearest scrollView and shows the corresponding scrollbar thumb.
/// {@endtemplate}
final
ScrollNotificationPredicate
notificationPredicate
;
@override
RawScrollbarState
<
RawScrollbar
>
createState
()
=>
RawScrollbarState
<
RawScrollbar
>();
}
...
...
@@ -1031,6 +1046,8 @@ class RawScrollbarState<T extends RawScrollbar> extends State<T> with TickerProv
}
bool
_handleScrollNotification
(
ScrollNotification
notification
)
{
if
(!
widget
.
notificationPredicate
(
notification
))
return
false
;
final
ScrollMetrics
metrics
=
notification
.
metrics
;
if
(
metrics
.
maxScrollExtent
<=
metrics
.
minScrollExtent
)
...
...
packages/flutter/test/material/scrollbar_test.dart
View file @
e7772d0e
...
...
@@ -1059,4 +1059,66 @@ void main() {
final
CupertinoScrollbar
scrollbar
=
tester
.
widget
<
CupertinoScrollbar
>(
find
.
byType
(
CupertinoScrollbar
));
expect
(
scrollbar
.
controller
,
isNotNull
);
},
variant:
const
TargetPlatformVariant
(<
TargetPlatform
>{
TargetPlatform
.
iOS
}));
testWidgets
(
"Scrollbar doesn't show when scroll the inner scrollable widget"
,
(
WidgetTester
tester
)
async
{
final
GlobalKey
key1
=
GlobalKey
();
final
GlobalKey
key2
=
GlobalKey
();
final
GlobalKey
outerKey
=
GlobalKey
();
final
GlobalKey
innerKey
=
GlobalKey
();
await
tester
.
pumpWidget
(
Directionality
(
textDirection:
TextDirection
.
ltr
,
child:
MediaQuery
(
data:
const
MediaQueryData
(),
child:
Scrollbar
(
key:
key2
,
notificationPredicate:
null
,
child:
SingleChildScrollView
(
key:
outerKey
,
child:
SizedBox
(
height:
1000.0
,
width:
double
.
infinity
,
child:
Column
(
children:
<
Widget
>[
Scrollbar
(
key:
key1
,
notificationPredicate:
null
,
child:
SizedBox
(
height:
300.0
,
width:
double
.
infinity
,
child:
SingleChildScrollView
(
key:
innerKey
,
child:
const
SizedBox
(
key:
Key
(
'Inner scrollable'
),
height:
1000.0
,
width:
double
.
infinity
,
),
),
),
),
],
),
),
),
),
),
),
);
// Drag the inner scrollable widget.
await
tester
.
drag
(
find
.
byKey
(
innerKey
),
const
Offset
(
0.0
,
-
25.0
));
await
tester
.
pump
();
// Scrollbar fully showing.
await
tester
.
pump
(
const
Duration
(
milliseconds:
500
));
expect
(
tester
.
renderObject
(
find
.
byKey
(
key2
)),
paintsExactlyCountTimes
(
#drawRect
,
2
),
// Each bar will call [drawRect] twice.
);
expect
(
tester
.
renderObject
(
find
.
byKey
(
key1
)),
paintsExactlyCountTimes
(
#drawRect
,
2
),
);
},
variant:
TargetPlatformVariant
.
all
());
}
packages/flutter/test/widgets/scrollbar_test.dart
View file @
e7772d0e
...
...
@@ -804,4 +804,67 @@ void main() {
),
);
});
// Regression test for https://github.com/flutter/flutter/issues/66444
testWidgets
(
"RawScrollbar doesn't show when scroll the inner scrollable widget"
,
(
WidgetTester
tester
)
async
{
final
GlobalKey
key1
=
GlobalKey
();
final
GlobalKey
key2
=
GlobalKey
();
final
GlobalKey
outerKey
=
GlobalKey
();
final
GlobalKey
innerKey
=
GlobalKey
();
await
tester
.
pumpWidget
(
Directionality
(
textDirection:
TextDirection
.
ltr
,
child:
MediaQuery
(
data:
const
MediaQueryData
(),
child:
RawScrollbar
(
key:
key2
,
thumbColor:
const
Color
(
0x11111111
),
child:
SingleChildScrollView
(
key:
outerKey
,
child:
SizedBox
(
height:
1000.0
,
width:
double
.
infinity
,
child:
Column
(
children:
<
Widget
>[
RawScrollbar
(
key:
key1
,
thumbColor:
const
Color
(
0x22222222
),
child:
SizedBox
(
height:
300.0
,
width:
double
.
infinity
,
child:
SingleChildScrollView
(
key:
innerKey
,
child:
const
SizedBox
(
key:
Key
(
'Inner scrollable'
),
height:
1000.0
,
width:
double
.
infinity
,
),
),
),
),
],
),
),
),
),
),
),
);
// Drag the inner scrollable widget.
await
tester
.
drag
(
find
.
byKey
(
innerKey
),
const
Offset
(
0.0
,
-
25.0
));
await
tester
.
pump
();
// Scrollbar fully showing.
await
tester
.
pump
(
const
Duration
(
milliseconds:
500
));
expect
(
tester
.
renderObject
(
find
.
byKey
(
key2
)),
paintsExactlyCountTimes
(
#drawRect
,
2
),
// Each bar will call [drawRect] twice.
);
expect
(
tester
.
renderObject
(
find
.
byKey
(
key1
)),
paintsExactlyCountTimes
(
#drawRect
,
2
),
);
});
}
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