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
fc23277d
Commit
fc23277d
authored
Feb 11, 2016
by
Adam Barth
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Cleanup MixedViewport
This patch fixes a couple minor bugs and cleans up MixedViewport a bit.
parent
3fa3d226
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
88 additions
and
41 deletions
+88
-41
mixed_viewport.dart
packages/flutter/lib/src/widgets/mixed_viewport.dart
+42
-41
mixed_viewport_test.dart
packages/flutter/test/widget/mixed_viewport_test.dart
+46
-0
No files found.
packages/flutter/lib/src/widgets/mixed_viewport.dart
View file @
fc23277d
...
@@ -103,7 +103,7 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
...
@@ -103,7 +103,7 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
/// Returns false if any of the previously-cached offsets have been marked as
/// Returns false if any of the previously-cached offsets have been marked as
/// invalid and need to be updated.
/// invalid and need to be updated.
bool
get
isValid
=>
_invalidIndices
.
length
==
0
;
bool
get
_isValid
=>
_invalidIndices
.
isEmpty
;
/// The constraints for which the current offsets are valid.
/// The constraints for which the current offsets are valid.
BoxConstraints
_lastLayoutConstraints
;
BoxConstraints
_lastLayoutConstraints
;
...
@@ -139,17 +139,20 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
...
@@ -139,17 +139,20 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
void
mount
(
Element
parent
,
dynamic
newSlot
)
{
void
mount
(
Element
parent
,
dynamic
newSlot
)
{
super
.
mount
(
parent
,
newSlot
);
super
.
mount
(
parent
,
newSlot
);
renderObject
.
callback
=
layout
;
renderObject
renderObject
.
totalExtentCallback
=
_noIntrinsicExtent
;
..
direction
=
widget
.
direction
renderObject
.
maxCrossAxisExtentCallback
=
_noIntrinsicExtent
;
..
callback
=
layout
renderObject
.
minCrossAxisExtentCallback
=
_noIntrinsicExtent
;
..
totalExtentCallback
=
_noIntrinsicExtent
..
maxCrossAxisExtentCallback
=
_noIntrinsicExtent
..
minCrossAxisExtentCallback
=
_noIntrinsicExtent
;
}
}
void
unmount
()
{
void
unmount
()
{
renderObject
.
callback
=
null
;
renderObject
renderObject
.
totalExtentCallback
=
null
;
..
callback
=
null
renderObject
.
minCrossAxisExtentCallback
=
null
;
..
totalExtentCallback
=
null
renderObject
.
maxCrossAxisExtentCallback
=
null
;
..
minCrossAxisExtentCallback
=
null
..
maxCrossAxisExtentCallback
=
null
;
super
.
unmount
();
super
.
unmount
();
}
}
...
@@ -167,14 +170,15 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
...
@@ -167,14 +170,15 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
return
null
;
return
null
;
}
}
static
const
Object
_omit
=
const
Object
();
// used as a slot when it's not yet time to attach the child
static
final
Object
_omit
=
new
Object
();
// used as a slot when it's not yet time to attach the child
void
update
(
MixedViewport
newWidget
)
{
void
update
(
MixedViewport
newWidget
)
{
_ChangeDescription
changes
=
newWidget
.
evaluateChangesFrom
(
widget
);
_ChangeDescription
changes
=
newWidget
.
evaluateChangesFrom
(
widget
);
super
.
update
(
newWidget
);
super
.
update
(
newWidget
);
renderObject
.
direction
=
widget
.
direction
;
if
(
changes
==
_ChangeDescription
.
resized
)
if
(
changes
==
_ChangeDescription
.
resized
)
_resetCache
();
_resetCache
();
if
(
changes
!=
_ChangeDescription
.
none
||
!
isValid
)
{
if
(
changes
!=
_ChangeDescription
.
none
||
!
_
isValid
)
{
renderObject
.
markNeedsLayout
();
renderObject
.
markNeedsLayout
();
}
else
{
}
else
{
// We have to reinvoke our builders because they might return new data.
// We have to reinvoke our builders because they might return new data.
...
@@ -197,20 +201,20 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
...
@@ -197,20 +201,20 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
assert
(
renderObject
!=
null
);
assert
(
renderObject
!=
null
);
final
int
startIndex
=
_firstVisibleChildIndex
;
final
int
startIndex
=
_firstVisibleChildIndex
;
int
lastIndex
=
startIndex
+
_childrenByKey
.
length
-
1
;
int
lastIndex
=
startIndex
+
_childrenByKey
.
length
-
1
;
Element
nextSibling
=
null
;
Element
previousChild
=
null
;
for
(
int
index
=
lastIndex
;
index
>=
startIndex
;
index
-
=
1
)
{
for
(
int
index
=
startIndex
;
index
<=
lastIndex
;
index
+
=
1
)
{
final
Widget
newWidget
=
_buildWidgetAt
(
index
);
final
Widget
newWidget
=
_buildWidgetAt
(
index
);
final
_ChildKey
key
=
new
_ChildKey
.
fromWidget
(
newWidget
);
final
_ChildKey
key
=
new
_ChildKey
.
fromWidget
(
newWidget
);
final
Element
oldElement
=
_childrenByKey
[
key
];
final
Element
oldElement
=
_childrenByKey
[
key
];
assert
(
oldElement
!=
null
);
assert
(
oldElement
!=
null
);
final
Element
newElement
=
updateChild
(
oldElement
,
newWidget
,
nextSibling
);
final
Element
newElement
=
updateChild
(
oldElement
,
newWidget
,
previousChild
);
assert
(
newElement
!=
null
);
assert
(
newElement
!=
null
);
_childrenByKey
[
key
]
=
newElement
;
_childrenByKey
[
key
]
=
newElement
;
// Verify that it hasn't changed size.
// Verify that it hasn't changed size.
// If this assertion fires, it means you didn't call "invalidate"
// If this assertion fires, it means you didn't call "invalidate"
// before changing the size of one of your items.
// before changing the size of one of your items.
assert
(
_debugIsSameSize
(
newElement
,
index
,
_lastLayoutConstraints
));
assert
(
_debugIsSameSize
(
newElement
,
index
,
_lastLayoutConstraints
));
nextSibling
=
newElement
;
previousChild
=
newElement
;
}
}
}
}
}
}
...
@@ -283,8 +287,7 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
...
@@ -283,8 +287,7 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
Element
_getElement
(
int
index
,
BoxConstraints
innerConstraints
)
{
Element
_getElement
(
int
index
,
BoxConstraints
innerConstraints
)
{
assert
(
index
<=
_childOffsets
.
length
-
1
);
assert
(
index
<=
_childOffsets
.
length
-
1
);
final
Widget
newWidget
=
_buildWidgetAt
(
index
);
final
Widget
newWidget
=
_buildWidgetAt
(
index
);
final
Element
newElement
=
_inflateOrUpdateWidget
(
newWidget
);
return
_inflateOrUpdateWidget
(
newWidget
);
return
newElement
;
}
}
// Build the widget at index.
// Build the widget at index.
...
@@ -293,8 +296,7 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
...
@@ -293,8 +296,7 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
final
Widget
newWidget
=
_maybeBuildWidgetAt
(
index
);
final
Widget
newWidget
=
_maybeBuildWidgetAt
(
index
);
if
(
newWidget
==
null
)
if
(
newWidget
==
null
)
return
null
;
return
null
;
final
Element
newElement
=
_inflateOrUpdateWidget
(
newWidget
);
return
_inflateOrUpdateWidget
(
newWidget
);
return
newElement
;
}
}
// Build the widget at index, handling the case where there is no such widget.
// Build the widget at index, handling the case where there is no such widget.
...
@@ -349,39 +351,39 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
...
@@ -349,39 +351,39 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
return
result
;
return
result
;
}
}
/// This is the core lazy-build algorithm. It builds widgets incrementally
double
_getMaxExtent
(
BoxConstraints
constraints
)
{
/// from index 0 until it has built enough widgets to cover itself, and
/// discards any widgets that are not displayed.
void
_doLayout
(
BoxConstraints
constraints
)
{
Map
<
_ChildKey
,
Element
>
newChildren
=
new
Map
<
_ChildKey
,
Element
>();
Map
<
int
,
Element
>
builtChildren
=
new
Map
<
int
,
Element
>();
// Establish the start and end offsets based on our current constraints.
double
extent
;
switch
(
widget
.
direction
)
{
switch
(
widget
.
direction
)
{
case
Axis
.
vertical
:
case
Axis
.
vertical
:
extent
=
constraints
.
maxHeight
;
assert
(
constraints
.
maxHeight
<
double
.
INFINITY
&&
assert
(
extent
<
double
.
INFINITY
&&
'There is no point putting a lazily-built vertical MixedViewport inside a box with infinite internal '
+
'There is no point putting a lazily-built vertical MixedViewport inside a box with infinite internal '
+
'height (e.g. inside something else that scrolls vertically), because it would then just eagerly build '
+
'height (e.g. inside something else that scrolls vertically), because it would then just eagerly build '
+
'all the children. You probably want to put the MixedViewport inside a Container with a fixed height.'
is
String
);
'all the children. You probably want to put the MixedViewport inside a Container with a fixed height.'
is
String
);
break
;
return
constraints
.
maxHeight
;
case
Axis
.
horizontal
:
case
Axis
.
horizontal
:
extent
=
constraints
.
maxWidth
;
assert
(
constraints
.
maxWidth
<
double
.
INFINITY
&&
assert
(
extent
<
double
.
INFINITY
&&
'There is no point putting a lazily-built horizontal MixedViewport inside a box with infinite internal '
+
'There is no point putting a lazily-built horizontal MixedViewport inside a box with infinite internal '
+
'width (e.g. inside something else that scrolls horizontally), because it would then just eagerly build '
+
'width (e.g. inside something else that scrolls horizontally), because it would then just eagerly build '
+
'all the children. You probably want to put the MixedViewport inside a Container with a fixed width.'
is
String
);
'all the children. You probably want to put the MixedViewport inside a Container with a fixed width.'
is
String
);
break
;
return
constraints
.
maxWidth
;
}
}
}
final
double
endOffset
=
widget
.
startOffset
+
extent
;
/// This is the core lazy-build algorithm. It builds widgets incrementally
/// from index 0 until it has built enough widgets to cover itself, and
/// discards any widgets that are not displayed.
void
_doLayout
(
BoxConstraints
constraints
)
{
final
Map
<
_ChildKey
,
Element
>
newChildren
=
new
Map
<
_ChildKey
,
Element
>();
final
Map
<
int
,
Element
>
builtChildren
=
new
Map
<
int
,
Element
>();
// Establish the start and end offsets based on our current constraints.
final
double
endOffset
=
widget
.
startOffset
+
_getMaxExtent
(
constraints
);
// Create the constraints that we will use to measure the children.
// Create the constraints that we will use to measure the children.
final
BoxConstraints
innerConstraints
=
_getInnerConstraints
(
constraints
);
final
BoxConstraints
innerConstraints
=
_getInnerConstraints
(
constraints
);
// Before doing the actual layout, fix the offsets for the widgets whose
// Before doing the actual layout, fix the offsets for the widgets whose
// size has apparently changed.
// size has apparently changed.
if
(!
isValid
)
{
if
(!
_
isValid
)
{
assert
(
_childOffsets
.
length
>
0
);
assert
(
_childOffsets
.
length
>
0
);
assert
(
_childOffsets
.
length
==
_childExtents
.
length
+
1
);
assert
(
_childOffsets
.
length
==
_childExtents
.
length
+
1
);
List
<
int
>
invalidIndices
=
_invalidIndices
.
toList
();
List
<
int
>
invalidIndices
=
_invalidIndices
.
toList
();
...
@@ -512,13 +514,12 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
...
@@ -512,13 +514,12 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
assert
(!
haveChildren
||
startIndex
<
_childExtents
.
length
);
assert
(!
haveChildren
||
startIndex
<
_childExtents
.
length
);
// Build the other widgets that are visible.
// Build the other widgets that are visible.
int
index
=
startIndex
;
int
index
;
if
(
haveChildren
)
{
if
(
haveChildren
)
{
// Update the renderObject configuration
// Update the renderObject configuration
renderObject
.
direction
=
renderObject
.
direction
;
renderObject
.
startOffset
=
_childOffsets
[
startIndex
]
-
widget
.
startOffset
;
renderObject
.
startOffset
=
_childOffsets
[
index
]
-
widget
.
startOffset
;
// Build all the widgets we still need.
// Build all the widgets we still need.
while
(
_childOffsets
[
index
]
<
endOffset
)
{
for
(
index
=
startIndex
;
_childOffsets
[
index
]
<
endOffset
;
index
+=
1
)
{
if
(!
builtChildren
.
containsKey
(
index
))
{
if
(!
builtChildren
.
containsKey
(
index
))
{
Element
element
=
_maybeGetElement
(
index
,
innerConstraints
);
Element
element
=
_maybeGetElement
(
index
,
innerConstraints
);
if
(
element
==
null
)
{
if
(
element
==
null
)
{
...
@@ -543,7 +544,6 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
...
@@ -543,7 +544,6 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
builtChildren
[
index
]
=
element
;
builtChildren
[
index
]
=
element
;
}
}
assert
(
builtChildren
[
index
]
!=
null
);
assert
(
builtChildren
[
index
]
!=
null
);
index
+=
1
;
}
}
}
}
...
@@ -554,6 +554,7 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
...
@@ -554,6 +554,7 @@ class _MixedViewportElement extends RenderObjectElement<MixedViewport> {
}
}
if
(
haveChildren
)
{
if
(
haveChildren
)
{
assert
(
index
!=
null
);
// Place all our children in our RenderObject.
// Place all our children in our RenderObject.
// All the children we are placing are in builtChildren and newChildren.
// All the children we are placing are in builtChildren and newChildren.
Element
previousChild
=
null
;
Element
previousChild
=
null
;
...
...
packages/flutter/test/widget/mixed_viewport_test.dart
View file @
fc23277d
...
@@ -150,4 +150,50 @@ void main() {
...
@@ -150,4 +150,50 @@ void main() {
callbackTracker
.
clear
();
callbackTracker
.
clear
();
});
});
});
});
test
(
'MixedViewport reinvoke builders'
,
()
{
testWidgets
((
WidgetTester
tester
)
{
List
<
int
>
callbackTracker
=
<
int
>[];
List
<
String
>
text
=
<
String
>[];
IndexedBuilder
itemBuilder
=
(
BuildContext
context
,
int
i
)
{
callbackTracker
.
add
(
i
);
return
new
Container
(
key:
new
ValueKey
<
int
>(
i
),
width:
500.0
,
// this should be ignored
height:
220.0
,
child:
new
Text
(
"
$i
"
)
);
};
ElementVisitor
collectText
=
(
Element
element
)
{
final
Widget
widget
=
element
.
widget
;
if
(
widget
is
Text
)
text
.
add
(
widget
.
data
);
};
Widget
builder
()
{
return
new
MixedViewport
(
builder:
itemBuilder
,
startOffset:
0.0
);
}
tester
.
pumpWidget
(
builder
());
expect
(
callbackTracker
,
equals
([
0
,
1
,
2
]));
callbackTracker
.
clear
();
tester
.
walkElements
(
collectText
);
expect
(
text
,
equals
([
'0'
,
'1'
,
'2'
]));
text
.
clear
();
tester
.
pumpWidget
(
builder
());
expect
(
callbackTracker
,
equals
([
0
,
1
,
2
]));
callbackTracker
.
clear
();
tester
.
walkElements
(
collectText
);
expect
(
text
,
equals
([
'0'
,
'1'
,
'2'
]));
text
.
clear
();
});
});
}
}
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