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
9f17a43e
Unverified
Commit
9f17a43e
authored
Nov 22, 2019
by
Kate Lovett
Committed by
GitHub
Nov 22, 2019
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
SliverIgnorePointer (#45127)
parent
33d30224
Changes
5
Show whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
297 additions
and
5 deletions
+297
-5
proxy_box.dart
packages/flutter/lib/src/rendering/proxy_box.dart
+3
-3
sliver.dart
packages/flutter/lib/src/rendering/sliver.dart
+122
-0
basic.dart
packages/flutter/lib/src/widgets/basic.dart
+2
-2
sliver.dart
packages/flutter/lib/src/widgets/sliver.dart
+60
-0
slivers_test.dart
packages/flutter/test/widgets/slivers_test.dart
+110
-0
No files found.
packages/flutter/lib/src/rendering/proxy_box.dart
View file @
9f17a43e
...
@@ -3016,8 +3016,8 @@ class RenderRepaintBoundary extends RenderProxyBox {
...
@@ -3016,8 +3016,8 @@ class RenderRepaintBoundary extends RenderProxyBox {
class
RenderIgnorePointer
extends
RenderProxyBox
{
class
RenderIgnorePointer
extends
RenderProxyBox
{
/// Creates a render object that is invisible to hit testing.
/// Creates a render object that is invisible to hit testing.
///
///
/// The [ignoring] argument must not be null. If [ignoringSemantics]
, this
/// The [ignoring] argument must not be null. If [ignoringSemantics]
is null,
/// render object will be ignored for semantics if [ignoring] is true.
///
this
render object will be ignored for semantics if [ignoring] is true.
RenderIgnorePointer
({
RenderIgnorePointer
({
RenderBox
child
,
RenderBox
child
,
bool
ignoring
=
true
,
bool
ignoring
=
true
,
...
@@ -3039,7 +3039,7 @@ class RenderIgnorePointer extends RenderProxyBox {
...
@@ -3039,7 +3039,7 @@ class RenderIgnorePointer extends RenderProxyBox {
if
(
value
==
_ignoring
)
if
(
value
==
_ignoring
)
return
;
return
;
_ignoring
=
value
;
_ignoring
=
value
;
if
(
ignoringSemantics
==
null
)
if
(
_ignoringSemantics
==
null
||
!
_ignoringSemantics
)
markNeedsSemanticsUpdate
();
markNeedsSemanticsUpdate
();
}
}
...
...
packages/flutter/lib/src/rendering/sliver.dart
View file @
9f17a43e
...
@@ -1817,3 +1817,125 @@ class RenderSliverToBoxAdapter extends RenderSliverSingleBoxAdapter {
...
@@ -1817,3 +1817,125 @@ class RenderSliverToBoxAdapter extends RenderSliverSingleBoxAdapter {
setChildParentData
(
child
,
constraints
,
geometry
);
setChildParentData
(
child
,
constraints
,
geometry
);
}
}
}
}
/// A render object that is invisible during hit testing.
///
/// When [ignoring] is true, this render object (and its subtree) is invisible
/// to hit testing. It still consumes space during layout and paints its sliver
/// child as usual. It just cannot be the target of located events, because its
/// render object returns false from [hitTest].
///
/// When [ignoringSemantics] is true, the subtree will be invisible to the
/// semantics layer (and thus e.g. accessibility tools). If [ignoringSemantics]
/// is null, it uses the value of [ignoring].
class
RenderSliverIgnorePointer
extends
RenderSliver
with
RenderObjectWithChildMixin
<
RenderSliver
>
{
/// Creates a render object that is invisible to hit testing.
///
/// The [ignoring] argument must not be null. If [ignoringSemantics] is null,
/// this render object will be ignored for semantics if [ignoring] is true.
RenderSliverIgnorePointer
({
RenderSliver
sliver
,
bool
ignoring
=
true
,
bool
ignoringSemantics
,
})
:
assert
(
ignoring
!=
null
),
_ignoring
=
ignoring
,
_ignoringSemantics
=
ignoringSemantics
{
child
=
sliver
;
}
/// Whether this render object is ignored during hit testing.
///
/// Regardless of whether this render object is ignored during hit testing, it
/// will still consume space during layout and be visible during painting.
bool
get
ignoring
=>
_ignoring
;
bool
_ignoring
;
set
ignoring
(
bool
value
)
{
assert
(
value
!=
null
);
if
(
value
==
_ignoring
)
return
;
_ignoring
=
value
;
if
(
_ignoringSemantics
==
null
||
!
_ignoringSemantics
)
markNeedsSemanticsUpdate
();
}
/// Whether the semantics of this render object is ignored when compiling the
/// semantics tree.
///
/// If null, defaults to value of [ignoring].
///
/// See [SemanticsNode] for additional information about the semantics tree.
bool
get
ignoringSemantics
=>
_ignoringSemantics
;
bool
_ignoringSemantics
;
set
ignoringSemantics
(
bool
value
)
{
if
(
value
==
_ignoringSemantics
)
return
;
final
bool
oldEffectiveValue
=
_effectiveIgnoringSemantics
;
_ignoringSemantics
=
value
;
if
(
oldEffectiveValue
!=
_effectiveIgnoringSemantics
)
markNeedsSemanticsUpdate
();
}
bool
get
_effectiveIgnoringSemantics
=>
ignoringSemantics
??
ignoring
;
@override
void
setupParentData
(
RenderObject
child
)
{
if
(
child
.
parentData
is
!
SliverPhysicalParentData
)
child
.
parentData
=
SliverPhysicalParentData
();
}
@override
void
performLayout
()
{
assert
(
child
!=
null
);
child
.
layout
(
constraints
,
parentUsesSize:
true
);
geometry
=
child
.
geometry
;
}
@override
bool
hitTest
(
SliverHitTestResult
result
,
{
double
mainAxisPosition
,
double
crossAxisPosition
})
{
return
!
ignoring
&&
super
.
hitTest
(
result
,
mainAxisPosition:
mainAxisPosition
,
crossAxisPosition:
crossAxisPosition
,
);
}
@override
bool
hitTestChildren
(
SliverHitTestResult
result
,
{
double
mainAxisPosition
,
double
crossAxisPosition
})
{
return
child
!=
null
&&
child
.
geometry
.
hitTestExtent
>
0
&&
child
.
hitTest
(
result
,
mainAxisPosition:
mainAxisPosition
,
crossAxisPosition:
crossAxisPosition
,
);
}
@override
void
applyPaintTransform
(
RenderObject
child
,
Matrix4
transform
)
{
assert
(
child
!=
null
);
final
SliverPhysicalParentData
childParentData
=
child
.
parentData
;
childParentData
.
applyPaintTransform
(
transform
);
}
@override
void
visitChildrenForSemantics
(
RenderObjectVisitor
visitor
)
{
if
(
child
!=
null
&&
!
_effectiveIgnoringSemantics
)
visitor
(
child
);
}
@override
void
debugFillProperties
(
DiagnosticPropertiesBuilder
properties
)
{
super
.
debugFillProperties
(
properties
);
properties
.
add
(
DiagnosticsProperty
<
bool
>(
'ignoring'
,
ignoring
));
properties
.
add
(
DiagnosticsProperty
<
bool
>(
'ignoringSemantics'
,
_effectiveIgnoringSemantics
,
description:
ignoringSemantics
==
null
?
'implicitly
$_effectiveIgnoringSemantics
'
:
null
,
),
);
}
}
packages/flutter/lib/src/widgets/basic.dart
View file @
9f17a43e
...
@@ -6026,8 +6026,8 @@ class RepaintBoundary extends SingleChildRenderObjectWidget {
...
@@ -6026,8 +6026,8 @@ class RepaintBoundary extends SingleChildRenderObjectWidget {
class
IgnorePointer
extends
SingleChildRenderObjectWidget
{
class
IgnorePointer
extends
SingleChildRenderObjectWidget
{
/// Creates a widget that is invisible to hit testing.
/// Creates a widget that is invisible to hit testing.
///
///
/// The [ignoring] argument must not be null. If [ignoringSemantics]
, this
/// The [ignoring] argument must not be null. If [ignoringSemantics]
is null,
/// render object will be ignored for semantics if [ignoring] is true.
///
this
render object will be ignored for semantics if [ignoring] is true.
const
IgnorePointer
({
const
IgnorePointer
({
Key
key
,
Key
key
,
this
.
ignoring
=
true
,
this
.
ignoring
=
true
,
...
...
packages/flutter/lib/src/widgets/sliver.dart
View file @
9f17a43e
...
@@ -1625,6 +1625,66 @@ class SliverFillRemaining extends SingleChildRenderObjectWidget {
...
@@ -1625,6 +1625,66 @@ class SliverFillRemaining extends SingleChildRenderObjectWidget {
}
}
}
}
/// A sliver widget that is invisible during hit testing.
///
/// When [ignoring] is true, this widget (and its subtree) is invisible
/// to hit testing. It still consumes space during layout and paints its sliver
/// child as usual. It just cannot be the target of located events, because it
/// returns false from [RenderSliver.hitTest].
///
/// When [ignoringSemantics] is true, the subtree will be invisible to
/// the semantics layer (and thus e.g. accessibility tools). If
/// [ignoringSemantics] is null, it uses the value of [ignoring].
class
SliverIgnorePointer
extends
SingleChildRenderObjectWidget
{
/// Creates a sliver widget that is invisible to hit testing.
///
/// The [ignoring] argument must not be null. If [ignoringSemantics] is null,
/// this render object will be ignored for semantics if [ignoring] is true.
const
SliverIgnorePointer
({
Key
key
,
this
.
ignoring
=
true
,
this
.
ignoringSemantics
,
Widget
sliver
,
})
:
assert
(
ignoring
!=
null
),
super
(
key:
key
,
child:
sliver
);
/// Whether this sliver is ignored during hit testing.
///
/// Regardless of whether this sliver is ignored during hit testing, it will
/// still consume space during layout and be visible during painting.
final
bool
ignoring
;
/// Whether the semantics of this sliver is ignored when compiling the
/// semantics tree.
///
/// If null, defaults to value of [ignoring].
///
/// See [SemanticsNode] for additional information about the semantics tree.
final
bool
ignoringSemantics
;
@override
RenderSliverIgnorePointer
createRenderObject
(
BuildContext
context
)
{
return
RenderSliverIgnorePointer
(
ignoring:
ignoring
,
ignoringSemantics:
ignoringSemantics
,
);
}
@override
void
updateRenderObject
(
BuildContext
context
,
RenderSliverIgnorePointer
renderObject
)
{
renderObject
..
ignoring
=
ignoring
..
ignoringSemantics
=
ignoringSemantics
;
}
@override
void
debugFillProperties
(
DiagnosticPropertiesBuilder
properties
)
{
super
.
debugFillProperties
(
properties
);
properties
.
add
(
DiagnosticsProperty
<
bool
>(
'ignoring'
,
ignoring
));
properties
.
add
(
DiagnosticsProperty
<
bool
>(
'ignoringSemantics'
,
ignoringSemantics
,
defaultValue:
null
));
}
}
/// Mark a child as needing to stay alive even when it's in a lazy list that
/// Mark a child as needing to stay alive even when it's in a lazy list that
/// would otherwise remove it.
/// would otherwise remove it.
///
///
...
...
packages/flutter/test/widgets/slivers_test.dart
View file @
9f17a43e
...
@@ -3,9 +3,12 @@
...
@@ -3,9 +3,12 @@
// found in the LICENSE file.
// found in the LICENSE file.
import
'package:flutter_test/flutter_test.dart'
;
import
'package:flutter_test/flutter_test.dart'
;
import
'package:flutter/material.dart'
;
import
'package:flutter/widgets.dart'
;
import
'package:flutter/widgets.dart'
;
import
'package:flutter/rendering.dart'
;
import
'package:flutter/rendering.dart'
;
import
'semantics_tester.dart'
;
Future
<
void
>
test
(
WidgetTester
tester
,
double
offset
,
{
double
anchor
=
0.0
})
{
Future
<
void
>
test
(
WidgetTester
tester
,
double
offset
,
{
double
anchor
=
0.0
})
{
return
tester
.
pumpWidget
(
return
tester
.
pumpWidget
(
Directionality
(
Directionality
(
...
@@ -418,6 +421,113 @@ void main() {
...
@@ -418,6 +421,113 @@ void main() {
// It will be corrected after a auto scroll animation.
// It will be corrected after a auto scroll animation.
expect
(
controller
.
offset
,
800.0
);
expect
(
controller
.
offset
,
800.0
);
});
});
group
(
'SliverIgnorePointer - '
,
()
{
Widget
_boilerPlate
(
Widget
sliver
)
{
return
Localizations
(
locale:
const
Locale
(
'en'
,
'us'
),
delegates:
const
<
LocalizationsDelegate
<
dynamic
>>[
DefaultWidgetsLocalizations
.
delegate
,
DefaultMaterialLocalizations
.
delegate
,
],
child:
Directionality
(
textDirection:
TextDirection
.
ltr
,
child:
MediaQuery
(
data:
const
MediaQueryData
(),
child:
CustomScrollView
(
slivers:
<
Widget
>[
sliver
])
)
)
);
}
testWidgets
(
'ignores pointer events'
,
(
WidgetTester
tester
)
async
{
final
SemanticsTester
semantics
=
SemanticsTester
(
tester
);
final
List
<
String
>
events
=
<
String
>[];
await
tester
.
pumpWidget
(
_boilerPlate
(
SliverIgnorePointer
(
ignoring:
true
,
ignoringSemantics:
false
,
sliver:
SliverToBoxAdapter
(
child:
GestureDetector
(
child:
const
Text
(
'a'
),
onTap:
()
{
events
.
add
(
'tap'
);
},
)
)
)
));
expect
(
semantics
.
nodesWith
(
label:
'a'
),
hasLength
(
1
));
await
tester
.
tap
(
find
.
byType
(
GestureDetector
));
expect
(
events
,
equals
(<
String
>[]));
});
testWidgets
(
'ignores semantics'
,
(
WidgetTester
tester
)
async
{
final
SemanticsTester
semantics
=
SemanticsTester
(
tester
);
final
List
<
String
>
events
=
<
String
>[];
await
tester
.
pumpWidget
(
_boilerPlate
(
SliverIgnorePointer
(
ignoring:
false
,
ignoringSemantics:
true
,
sliver:
SliverToBoxAdapter
(
child:
GestureDetector
(
child:
const
Text
(
'a'
),
onTap:
()
{
events
.
add
(
'tap'
);
},
)
)
)
));
expect
(
semantics
.
nodesWith
(
label:
'a'
),
hasLength
(
0
));
await
tester
.
tap
(
find
.
byType
(
GestureDetector
));
expect
(
events
,
equals
(<
String
>[
'tap'
]));
});
testWidgets
(
'ignores pointer events & semantics'
,
(
WidgetTester
tester
)
async
{
final
SemanticsTester
semantics
=
SemanticsTester
(
tester
);
final
List
<
String
>
events
=
<
String
>[];
await
tester
.
pumpWidget
(
_boilerPlate
(
SliverIgnorePointer
(
ignoring:
true
,
ignoringSemantics:
true
,
sliver:
SliverToBoxAdapter
(
child:
GestureDetector
(
child:
const
Text
(
'a'
),
onTap:
()
{
events
.
add
(
'tap'
);
},
)
)
)
));
expect
(
semantics
.
nodesWith
(
label:
'a'
),
hasLength
(
0
));
await
tester
.
tap
(
find
.
byType
(
GestureDetector
));
expect
(
events
,
equals
(<
String
>[]));
});
testWidgets
(
'ignores nothing'
,
(
WidgetTester
tester
)
async
{
final
SemanticsTester
semantics
=
SemanticsTester
(
tester
);
final
List
<
String
>
events
=
<
String
>[];
await
tester
.
pumpWidget
(
_boilerPlate
(
SliverIgnorePointer
(
ignoring:
false
,
ignoringSemantics:
false
,
sliver:
SliverToBoxAdapter
(
child:
GestureDetector
(
child:
const
Text
(
'a'
),
onTap:
()
{
events
.
add
(
'tap'
);
},
)
)
)
));
expect
(
semantics
.
nodesWith
(
label:
'a'
),
hasLength
(
1
));
await
tester
.
tap
(
find
.
byType
(
GestureDetector
));
expect
(
events
,
equals
(<
String
>[
'tap'
]));
});
});
}
}
bool
isRight
(
Offset
a
,
Offset
b
)
=>
b
.
dx
>
a
.
dx
;
bool
isRight
(
Offset
a
,
Offset
b
)
=>
b
.
dx
>
a
.
dx
;
...
...
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