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
93a1484a
Unverified
Commit
93a1484a
authored
Aug 23, 2021
by
chunhtai
Committed by
GitHub
Aug 23, 2021
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add a hook for scroll position to notify scrolling context when dimen… (#87076)
parent
02bb8a43
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
85 additions
and
17 deletions
+85
-17
scrollable.dart
packages/flutter/lib/src/widgets/scrollable.dart
+18
-7
page_view_test.dart
packages/flutter/test/widgets/page_view_test.dart
+1
-1
range_maintaining_scroll_physics_test.dart
...r/test/widgets/range_maintaining_scroll_physics_test.dart
+4
-4
scroll_activity_test.dart
packages/flutter/test/widgets/scroll_activity_test.dart
+1
-1
scrollable_restoration_test.dart
...ges/flutter/test/widgets/scrollable_restoration_test.dart
+2
-3
scrollable_test.dart
packages/flutter/test/widgets/scrollable_test.dart
+59
-1
No files found.
packages/flutter/lib/src/widgets/scrollable.dart
View file @
93a1484a
...
@@ -525,7 +525,6 @@ class ScrollableState extends State<Scrollable> with TickerProviderStateMixin, R
...
@@ -525,7 +525,6 @@ class ScrollableState extends State<Scrollable> with TickerProviderStateMixin, R
_gestureDetectorKey
.
currentState
!.
replaceSemanticsActions
(
actions
);
_gestureDetectorKey
.
currentState
!.
replaceSemanticsActions
(
actions
);
}
}
// GESTURE RECOGNITION AND POINTER IGNORING
// GESTURE RECOGNITION AND POINTER IGNORING
final
GlobalKey
<
RawGestureDetectorState
>
_gestureDetectorKey
=
GlobalKey
<
RawGestureDetectorState
>();
final
GlobalKey
<
RawGestureDetectorState
>
_gestureDetectorKey
=
GlobalKey
<
RawGestureDetectorState
>();
...
@@ -719,6 +718,15 @@ class ScrollableState extends State<Scrollable> with TickerProviderStateMixin, R
...
@@ -719,6 +718,15 @@ class ScrollableState extends State<Scrollable> with TickerProviderStateMixin, R
}
}
}
}
bool
_handleScrollMetricsNotification
(
ScrollMetricsNotification
notification
)
{
if
(
notification
.
depth
==
0
)
{
final
RenderObject
?
scrollSemanticsRenderObject
=
_scrollSemanticsKey
.
currentContext
?.
findRenderObject
();
if
(
scrollSemanticsRenderObject
!=
null
)
scrollSemanticsRenderObject
.
markNeedsSemanticsUpdate
();
}
return
false
;
}
// DESCRIPTION
// DESCRIPTION
@override
@override
...
@@ -757,12 +765,15 @@ class ScrollableState extends State<Scrollable> with TickerProviderStateMixin, R
...
@@ -757,12 +765,15 @@ class ScrollableState extends State<Scrollable> with TickerProviderStateMixin, R
);
);
if
(!
widget
.
excludeFromSemantics
)
{
if
(!
widget
.
excludeFromSemantics
)
{
result
=
_ScrollSemantics
(
result
=
NotificationListener
<
ScrollMetricsNotification
>(
key:
_scrollSemanticsKey
,
onNotification:
_handleScrollMetricsNotification
,
position:
position
,
child:
_ScrollSemantics
(
allowImplicitScrolling:
_physics
!.
allowImplicitScrolling
,
key:
_scrollSemanticsKey
,
semanticChildCount:
widget
.
semanticChildCount
,
position:
position
,
child:
result
,
allowImplicitScrolling:
_physics
!.
allowImplicitScrolling
,
semanticChildCount:
widget
.
semanticChildCount
,
child:
result
,
)
);
);
}
}
...
...
packages/flutter/test/widgets/page_view_test.dart
View file @
93a1484a
...
@@ -787,7 +787,7 @@ void main() {
...
@@ -787,7 +787,7 @@ void main() {
));
));
expect
(
controller
.
page
,
0
);
expect
(
controller
.
page
,
0
);
controller
.
jumpToPage
(
2
);
controller
.
jumpToPage
(
2
);
expect
(
await
tester
.
pumpAndSettle
(
const
Duration
(
minutes:
1
)),
1
);
expect
(
await
tester
.
pumpAndSettle
(
const
Duration
(
minutes:
1
)),
2
);
expect
(
controller
.
page
,
2
);
expect
(
controller
.
page
,
2
);
await
tester
.
pumpWidget
(
await
tester
.
pumpWidget
(
PageStorage
(
PageStorage
(
...
...
packages/flutter/test/widgets/range_maintaining_scroll_physics_test.dart
View file @
93a1484a
...
@@ -122,7 +122,7 @@ void main() {
...
@@ -122,7 +122,7 @@ void main() {
final
TestGesture
drag1
=
await
tester
.
startGesture
(
const
Offset
(
10.0
,
500.0
));
final
TestGesture
drag1
=
await
tester
.
startGesture
(
const
Offset
(
10.0
,
500.0
));
expect
(
await
tester
.
pumpAndSettle
(),
1
);
// Nothing to animate
expect
(
await
tester
.
pumpAndSettle
(),
1
);
// Nothing to animate
await
drag1
.
moveTo
(
const
Offset
(
10.0
,
0.0
));
await
drag1
.
moveTo
(
const
Offset
(
10.0
,
0.0
));
expect
(
await
tester
.
pumpAndSettle
(),
1
);
// Nothing to anim
ate
expect
(
await
tester
.
pumpAndSettle
(),
2
);
// Nothing to animate, only one semantics upd
ate
await
drag1
.
up
();
await
drag1
.
up
();
expect
(
await
tester
.
pumpAndSettle
(),
1
);
// Nothing to animate
expect
(
await
tester
.
pumpAndSettle
(),
1
);
// Nothing to animate
expect
(
position
.
pixels
,
moreOrLessEquals
(
500.0
));
expect
(
position
.
pixels
,
moreOrLessEquals
(
500.0
));
...
@@ -132,14 +132,14 @@ void main() {
...
@@ -132,14 +132,14 @@ void main() {
final
TestGesture
drag2
=
await
tester
.
startGesture
(
const
Offset
(
10.0
,
500.0
));
final
TestGesture
drag2
=
await
tester
.
startGesture
(
const
Offset
(
10.0
,
500.0
));
expect
(
await
tester
.
pumpAndSettle
(),
1
);
// Nothing to animate
expect
(
await
tester
.
pumpAndSettle
(),
1
);
// Nothing to animate
await
drag2
.
moveTo
(
const
Offset
(
10.0
,
100.0
));
await
drag2
.
moveTo
(
const
Offset
(
10.0
,
100.0
));
expect
(
await
tester
.
pumpAndSettle
(),
1
);
// Nothing to anim
ate
expect
(
await
tester
.
pumpAndSettle
(),
2
);
// Nothing to animate, only one semantics upd
ate
expect
(
position
.
maxScrollExtent
,
900.0
);
expect
(
position
.
maxScrollExtent
,
900.0
);
expect
(
position
.
pixels
,
lessThanOrEqualTo
(
900.0
));
expect
(
position
.
pixels
,
lessThanOrEqualTo
(
900.0
));
expect
(
position
.
activity
,
isInstanceOf
<
DragScrollActivity
>());
expect
(
position
.
activity
,
isInstanceOf
<
DragScrollActivity
>());
final
_ExpandingBoxState
expandingBoxState
=
tester
.
state
<
_ExpandingBoxState
>(
find
.
byType
(
ExpandingBox
));
final
_ExpandingBoxState
expandingBoxState
=
tester
.
state
<
_ExpandingBoxState
>(
find
.
byType
(
ExpandingBox
));
expandingBoxState
.
toggleSize
();
expandingBoxState
.
toggleSize
();
expect
(
await
tester
.
pumpAndSettle
(),
1
);
// Nothing to anim
ate
expect
(
await
tester
.
pumpAndSettle
(),
2
);
// Nothing to animate, only one semantics upd
ate
expect
(
position
.
activity
,
isInstanceOf
<
DragScrollActivity
>());
expect
(
position
.
activity
,
isInstanceOf
<
DragScrollActivity
>());
expect
(
position
.
minScrollExtent
,
0.0
);
expect
(
position
.
minScrollExtent
,
0.0
);
expect
(
position
.
maxScrollExtent
,
100.0
);
expect
(
position
.
maxScrollExtent
,
100.0
);
...
@@ -150,7 +150,7 @@ void main() {
...
@@ -150,7 +150,7 @@ void main() {
expect
(
position
.
minScrollExtent
,
0.0
);
expect
(
position
.
minScrollExtent
,
0.0
);
expect
(
position
.
maxScrollExtent
,
100.0
);
expect
(
position
.
maxScrollExtent
,
100.0
);
expect
(
position
.
pixels
,
50.0
);
expect
(
position
.
pixels
,
50.0
);
expect
(
await
tester
.
pumpAndSettle
(),
1
);
// Nothing to anim
ate
expect
(
await
tester
.
pumpAndSettle
(),
2
);
// Nothing to animate, only one semantics upd
ate
expect
(
position
.
minScrollExtent
,
0.0
);
expect
(
position
.
minScrollExtent
,
0.0
);
expect
(
position
.
maxScrollExtent
,
100.0
);
expect
(
position
.
maxScrollExtent
,
100.0
);
expect
(
position
.
pixels
,
50.0
);
expect
(
position
.
pixels
,
50.0
);
...
...
packages/flutter/test/widgets/scroll_activity_test.dart
View file @
93a1484a
...
@@ -23,7 +23,7 @@ void main() {
...
@@ -23,7 +23,7 @@ void main() {
await
tester
.
pump
();
await
tester
.
pump
();
await
tester
.
pumpWidget
(
MaterialApp
(
home:
ListView
(
controller:
controller
,
children:
children
(
31
))));
await
tester
.
pumpWidget
(
MaterialApp
(
home:
ListView
(
controller:
controller
,
children:
children
(
31
))));
expect
(
controller
.
position
.
pixels
,
thirty
+
200.0
);
// same distance past the end
expect
(
controller
.
position
.
pixels
,
thirty
+
200.0
);
// same distance past the end
expect
(
await
tester
.
pumpAndSettle
(),
7
);
// now it goes ballistic...
expect
(
await
tester
.
pumpAndSettle
(),
8
);
// now it goes ballistic...
expect
(
controller
.
position
.
pixels
,
thirty
+
100.0
);
// and ends up at the end
expect
(
controller
.
position
.
pixels
,
thirty
+
100.0
);
// and ends up at the end
});
});
...
...
packages/flutter/test/widgets/scrollable_restoration_test.dart
View file @
93a1484a
...
@@ -446,7 +446,6 @@ void main() {
...
@@ -446,7 +446,6 @@ void main() {
expect
(
find
.
text
(
'Tile 12'
),
findsNothing
);
expect
(
find
.
text
(
'Tile 12'
),
findsNothing
);
final
TestRestorationData
initialData
=
await
tester
.
getRestorationData
();
final
TestRestorationData
initialData
=
await
tester
.
getRestorationData
();
final
TestGesture
gesture
=
await
tester
.
startGesture
(
tester
.
getCenter
(
find
.
byType
(
ListView
)));
final
TestGesture
gesture
=
await
tester
.
startGesture
(
tester
.
getCenter
(
find
.
byType
(
ListView
)));
await
gesture
.
moveBy
(
const
Offset
(
0
,
-
525
));
await
gesture
.
moveBy
(
const
Offset
(
0
,
-
525
));
await
tester
.
pump
();
await
tester
.
pump
();
...
@@ -457,12 +456,12 @@ void main() {
...
@@ -457,12 +456,12 @@ void main() {
expect
(
find
.
text
(
'Tile 11'
),
findsOneWidget
);
expect
(
find
.
text
(
'Tile 11'
),
findsOneWidget
);
expect
(
find
.
text
(
'Tile 12'
),
findsOneWidget
);
expect
(
find
.
text
(
'Tile 12'
),
findsOneWidget
);
// Restoration data hasn't changed
and no frame is scheduled
.
// Restoration data hasn't changed.
expect
(
await
tester
.
getRestorationData
(),
initialData
);
expect
(
await
tester
.
getRestorationData
(),
initialData
);
expect
(
tester
.
binding
.
hasScheduledFrame
,
isFalse
);
// Restoration data changes with up event.
// Restoration data changes with up event.
await
gesture
.
up
();
await
gesture
.
up
();
await
tester
.
pump
();
expect
(
await
tester
.
getRestorationData
(),
isNot
(
initialData
));
expect
(
await
tester
.
getRestorationData
(),
isNot
(
initialData
));
});
});
}
}
...
...
packages/flutter/test/widgets/scrollable_test.dart
View file @
93a1484a
...
@@ -152,7 +152,7 @@ void main() {
...
@@ -152,7 +152,7 @@ void main() {
expect
(
getScrollOffset
(
tester
),
heldPosition
);
expect
(
getScrollOffset
(
tester
),
heldPosition
);
await
gesture
.
up
();
await
gesture
.
up
();
// Once the hold is let go, it should still snap back to origin.
// Once the hold is let go, it should still snap back to origin.
expect
(
await
tester
.
pumpAndSettle
(
const
Duration
(
minutes:
1
)),
2
);
expect
(
await
tester
.
pumpAndSettle
(
const
Duration
(
minutes:
1
)),
3
);
expect
(
getScrollOffset
(
tester
),
0.0
);
expect
(
getScrollOffset
(
tester
),
0.0
);
},
variant:
const
TargetPlatformVariant
(<
TargetPlatform
>{
TargetPlatform
.
iOS
,
TargetPlatform
.
macOS
}));
},
variant:
const
TargetPlatformVariant
(<
TargetPlatform
>{
TargetPlatform
.
iOS
,
TargetPlatform
.
macOS
}));
...
@@ -1313,6 +1313,64 @@ void main() {
...
@@ -1313,6 +1313,64 @@ void main() {
await
gesture
.
removePointer
();
await
gesture
.
removePointer
();
await
tester
.
pump
();
await
tester
.
pump
();
},
variant:
const
TargetPlatformVariant
(<
TargetPlatform
>{
TargetPlatform
.
iOS
,
TargetPlatform
.
macOS
,
TargetPlatform
.
android
}));
},
variant:
const
TargetPlatformVariant
(<
TargetPlatform
>{
TargetPlatform
.
iOS
,
TargetPlatform
.
macOS
,
TargetPlatform
.
android
}));
testWidgets
(
'Updated content dimensions correctly reflect in semantics'
,
(
WidgetTester
tester
)
async
{
// Regression test for https://github.com/flutter/flutter/issues/40419.
final
SemanticsHandle
handle
=
tester
.
ensureSemantics
();
final
UniqueKey
listView
=
UniqueKey
();
await
tester
.
pumpWidget
(
MaterialApp
(
home:
TickerMode
(
enabled:
true
,
child:
ListView
.
builder
(
key:
listView
,
itemCount:
100
,
itemBuilder:
(
BuildContext
context
,
int
index
)
{
return
Text
(
'Item
$index
'
);
},
),
),
));
SemanticsNode
scrollableNode
=
tester
.
getSemantics
(
find
.
descendant
(
of:
find
.
byKey
(
listView
),
matching:
find
.
byType
(
RawGestureDetector
)));
SemanticsNode
?
syntheticScrollableNode
;
scrollableNode
.
visitChildren
((
SemanticsNode
node
)
{
syntheticScrollableNode
=
node
;
return
true
;
});
expect
(
syntheticScrollableNode
!.
hasFlag
(
ui
.
SemanticsFlag
.
hasImplicitScrolling
),
isTrue
);
// Disabled the ticker mode to trigger didChangeDependencies on Scrollable.
// This can happen when a route is push or pop from top.
// It will reconstruct the scroll position and apply content dimensions.
await
tester
.
pumpWidget
(
MaterialApp
(
home:
TickerMode
(
enabled:
false
,
child:
ListView
.
builder
(
key:
listView
,
itemCount:
100
,
itemBuilder:
(
BuildContext
context
,
int
index
)
{
return
Text
(
'Item
$index
'
);
},
),
),
));
await
tester
.
pump
();
// The correct workflow will be the following:
// 1. _RenderScrollSemantics receives a new scroll position without content
// dimensions and creates a SemanticsNode without implicit scroll.
// 2. The content dimensions are applied to the scroll position during the
// layout phase, and the scroll position marks the semantics node of
// _RenderScrollSemantics dirty.
// 3. The _RenderScrollSemantics rebuilds its semantics node with implicit
// scroll.
scrollableNode
=
tester
.
getSemantics
(
find
.
descendant
(
of:
find
.
byKey
(
listView
),
matching:
find
.
byType
(
RawGestureDetector
)));
syntheticScrollableNode
=
null
;
scrollableNode
.
visitChildren
((
SemanticsNode
node
)
{
syntheticScrollableNode
=
node
;
return
true
;
});
expect
(
syntheticScrollableNode
!.
hasFlag
(
ui
.
SemanticsFlag
.
hasImplicitScrolling
),
isTrue
);
handle
.
dispose
();
});
}
}
// ignore: must_be_immutable
// ignore: must_be_immutable
...
...
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