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
d5deea49
Commit
d5deea49
authored
Mar 29, 2016
by
Ian Hickson
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #2942 from Hixie/layout-asserts
More elaborate exceptions.
parents
e1ac331b
aedf41bf
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
218 additions
and
91 deletions
+218
-91
box.dart
packages/flutter/lib/src/rendering/box.dart
+152
-85
flex.dart
packages/flutter/lib/src/rendering/flex.dart
+63
-4
object.dart
packages/flutter/lib/src/rendering/object.dart
+3
-2
No files found.
packages/flutter/lib/src/rendering/box.dart
View file @
d5deea49
...
@@ -520,8 +520,36 @@ abstract class RenderBox extends RenderObject {
...
@@ -520,8 +520,36 @@ abstract class RenderBox extends RenderObject {
bool
get
hasSize
=>
_size
!=
null
;
bool
get
hasSize
=>
_size
!=
null
;
Size
_size
;
Size
_size
;
void
set
size
(
Size
value
)
{
void
set
size
(
Size
value
)
{
assert
((
sizedByParent
&&
debugDoingThisResize
)
||
assert
(!(
debugDoingThisResize
&&
debugDoingThisLayout
));
(!
sizedByParent
&&
debugDoingThisLayout
));
assert
(
sizedByParent
||
!
debugDoingThisResize
);
assert
(()
{
if
((
sizedByParent
&&
debugDoingThisResize
)
||
(!
sizedByParent
&&
debugDoingThisLayout
))
return
true
;
assert
(!
debugDoingThisResize
);
String
contract
,
violation
,
hint
;
if
(
debugDoingThisLayout
)
{
assert
(
sizedByParent
);
violation
=
'It appears that the size setter was called from performLayout().'
;
hint
=
''
;
}
else
{
violation
=
'The size setter was called from outside layout (neither performResize() nor performLayout() were being run for this object).'
;
if
(
owner
!=
null
&&
owner
.
debugDoingLayout
)
hint
=
'Only the object itself can set its size. It is a contract violation for other objects to set it.'
;
}
if
(
sizedByParent
)
contract
=
'Because this RenderBox has sizedByParent set to true, it must set its size in performResize().'
;
else
contract
=
'Because this RenderBox has sizedByParent set to false, it must set its size in performLayout().'
;
throw
new
FlutterError
(
'RenderBox size setter called incorrectly.
\n
'
'
$violation
\n
'
'
$hint
\n
'
'
$contract
\n
'
'The RenderBox in question is:
\n
'
'
$this
'
);
});
assert
(()
{
assert
(()
{
if
(
value
is
_DebugSize
)
{
if
(
value
is
_DebugSize
)
{
if
(
value
.
_owner
!=
this
)
{
if
(
value
.
_owner
!=
this
)
{
...
@@ -628,7 +656,22 @@ abstract class RenderBox extends RenderObject {
...
@@ -628,7 +656,22 @@ abstract class RenderBox extends RenderObject {
@override
@override
void
debugAssertDoesMeetConstraints
()
{
void
debugAssertDoesMeetConstraints
()
{
assert
(
constraints
!=
null
);
assert
(
constraints
!=
null
);
assert
(
_size
!=
null
);
assert
(()
{
if
(!
hasSize
)
{
assert
(!
needsLayout
);
// this is called in the size= setter during layout, but in that case we have a size
String
contract
;
if
(
sizedByParent
)
contract
=
'Because this RenderBox has sizedByParent set to true, it must set its size in performResize().
\n
'
;
else
contract
=
'Because this RenderBox has sizedByParent set to false, it must set its size in performLayout().
\n
'
;
throw
new
FlutterError
(
'RenderBox did not set its size during layout.
\n
'
'
$contract
'
'It appears that this did not happen; layout completed, but the size property is still null.
\n
'
'The RenderBox in question is:
\n
'
'
$this
'
);
}
// verify that the size is not infinite
// verify that the size is not infinite
if
(
_size
.
isInfinite
)
{
if
(
_size
.
isInfinite
)
{
StringBuffer
information
=
new
StringBuffer
();
StringBuffer
information
=
new
StringBuffer
();
...
@@ -656,8 +699,8 @@ abstract class RenderBox extends RenderObject {
...
@@ -656,8 +699,8 @@ abstract class RenderBox extends RenderObject {
}
}
throw
new
FlutterError
(
throw
new
FlutterError
(
'
$runtimeType
object was given an infinite size during layout.
\n
'
'
$runtimeType
object was given an infinite size during layout.
\n
'
'This probably means that it is a render object that tries to be
\n
'
'This probably means that it is a render object that tries to be
'
'as big as possible, but it was put inside another render object
\n
'
'as big as possible, but it was put inside another render object
'
'that allows its children to pick their own size.
\n
'
'that allows its children to pick their own size.
\n
'
'
$information
'
'
$information
'
'See https://flutter.io/layout/ for more information.'
'See https://flutter.io/layout/ for more information.'
...
@@ -669,7 +712,7 @@ abstract class RenderBox extends RenderObject {
...
@@ -669,7 +712,7 @@ abstract class RenderBox extends RenderObject {
'
$runtimeType
does not meet its constraints.
\n
'
'
$runtimeType
does not meet its constraints.
\n
'
'Constraints:
$constraints
\n
'
'Constraints:
$constraints
\n
'
'Size:
$_size
\n
'
'Size:
$_size
\n
'
'If you are not writing your own RenderBox subclass, then this is not
\n
'
'If you are not writing your own RenderBox subclass, then this is not
'
'your fault. Contact support: https://github.com/flutter/flutter/issues/new'
'your fault. Contact support: https://github.com/flutter/flutter/issues/new'
);
);
}
}
...
@@ -711,6 +754,8 @@ abstract class RenderBox extends RenderObject {
...
@@ -711,6 +754,8 @@ abstract class RenderBox extends RenderObject {
'your fault. Contact support: https://github.com/flutter/flutter/issues/new'
'your fault. Contact support: https://github.com/flutter/flutter/issues/new'
);
);
}
}
return
true
;
});
}
}
@override
@override
...
@@ -763,8 +808,30 @@ abstract class RenderBox extends RenderObject {
...
@@ -763,8 +808,30 @@ abstract class RenderBox extends RenderObject {
/// coordinate space of the callee. The callee is responsible for checking
/// coordinate space of the callee. The callee is responsible for checking
/// whether the given position is within its bounds.
/// whether the given position is within its bounds.
bool
hitTest
(
HitTestResult
result
,
{
Point
position
})
{
bool
hitTest
(
HitTestResult
result
,
{
Point
position
})
{
assert
(!
needsLayout
);
assert
(()
{
assert
(
_size
!=
null
&&
'Missing size. Did you set a size during layout?'
!=
null
);
if
(
needsLayout
)
{
throw
new
FlutterError
(
'Cannot hit test a dirty render box.
\n
'
'The hitTest() method was invoked on this RenderBox:
\n
'
'
$this
\n
'
'Unfortunately, since this object has been marked as needing layout, its geometry is not known at this time. '
'This means it cannot be accurately hit-tested. Make sure to only mark nodes as needing layout during a pipeline '
'flush, so that it is marked clean before any event handling occurs. If you are trying to perform a hit test '
'during the layout phase itself, make sure you only hit test nodes that have completed layout (e.g. the node
\'
s '
'children, after their layout() method has been called).'
);
}
if
(!
hasSize
)
{
throw
new
FlutterError
(
'Cannot hit test a render box with no size.
\n
'
'The hitTest() method was invoked on this RenderBox:
\n
'
'
$this
\n
'
'Although this node is not marked as needing layout, its size is not set. A RenderBox object must have an '
'explicit size before it can be hit-tested. Make sure that the RenderBox in question sets its size during layout.'
);
}
return
true
;
});
if
(
position
.
x
>=
0.0
&&
position
.
x
<
_size
.
width
&&
if
(
position
.
x
>=
0.0
&&
position
.
x
<
_size
.
width
&&
position
.
y
>=
0.0
&&
position
.
y
<
_size
.
height
)
{
position
.
y
>=
0.0
&&
position
.
y
<
_size
.
height
)
{
if
(
hitTestChildren
(
result
,
position:
position
)
||
hitTestSelf
(
position
))
{
if
(
hitTestChildren
(
result
,
position:
position
)
||
hitTestSelf
(
position
))
{
...
...
packages/flutter/lib/src/rendering/flex.dart
View file @
d5deea49
...
@@ -353,10 +353,69 @@ class RenderFlex extends RenderBox with ContainerRenderObjectMixin<RenderBox, Fl
...
@@ -353,10 +353,69 @@ class RenderFlex extends RenderBox with ContainerRenderObjectMixin<RenderBox, Fl
totalChildren
++;
totalChildren
++;
int
flex
=
_getFlex
(
child
);
int
flex
=
_getFlex
(
child
);
if
(
flex
>
0
)
{
if
(
flex
>
0
)
{
// Flexible children can only be used when the RenderFlex box's container has a finite size.
assert
(()
{
// When the container is infinite, for example if you are in a scrollable viewport, then
final
String
identity
=
_direction
==
FlexDirection
.
horizontal
?
'row'
:
'column'
;
// it wouldn't make any sense to have a flexible child.
final
String
axis
=
_direction
==
FlexDirection
.
horizontal
?
'horizontal'
:
'vertical'
;
assert
(
canFlex
&&
'See https://flutter.io/layout/#flex'
is
String
);
final
String
dimension
=
_direction
==
FlexDirection
.
horizontal
?
'width'
:
'height'
;
String
error
,
message
;
String
addendum
=
''
;
if
(
mainAxisAlignment
==
MainAxisAlignment
.
collapse
)
{
error
=
'RenderFlex children have non-zero flex but mainAxisAlignment is set to "collapse".'
;
message
=
'The MainAxisAlignment.collapse value indicates that the
$identity
is to shrink-wrap its children '
'along the
$axis
axis. Setting a flex on a child (e.g. using a Flexible) indicates that the '
'child is to expand to fill the remaining space in the
$axis
direction.'
;
}
else
if
(
mainSize
==
double
.
INFINITY
)
{
error
=
'RenderFlex children have non-zero flex but incoming
$dimension
constraints are unbounded.'
;
message
=
'When a
$identity
is in a parent that does not provide a finite
$dimension
constraint, for example '
'if it is in a
$axis
scrollable, it will try to shrink-wrap its children along the
$axis
'
'axis. Setting a flex on a child (e.g. using a Flexible) indicates that the child is to '
'expand to fill the remaining space in the
$axis
direction.'
;
StringBuffer
information
=
new
StringBuffer
();
RenderBox
node
=
this
;
switch
(
_direction
)
{
case
FlexDirection
.
horizontal
:
while
(!
node
.
constraints
.
hasBoundedWidth
&&
node
.
parent
is
RenderBox
)
node
=
node
.
parent
;
if
(!
node
.
constraints
.
hasBoundedWidth
)
node
=
null
;
break
;
case
FlexDirection
.
vertical
:
while
(!
node
.
constraints
.
hasBoundedHeight
&&
node
.
parent
is
RenderBox
)
node
=
node
.
parent
;
if
(!
node
.
constraints
.
hasBoundedHeight
)
node
=
null
;
break
;
}
if
(
node
!=
null
)
{
information
.
writeln
(
'The nearest ancestor providing an unbounded width constraint is:'
);
information
.
writeln
(
'
$node
'
);
List
<
String
>
description
=
<
String
>[];
node
.
debugFillDescription
(
description
);
for
(
String
line
in
description
)
information
.
writeln
(
'
$line
'
);
}
information
.
writeln
(
'See also: https://flutter.io/layout/'
);
addendum
=
information
.
toString
();
}
else
{
return
true
;
}
throw
new
FlutterError
(
'
$error
\n
'
'
$message
\n
'
'These two directives are mutually exclusive. If a parent is to shrink-wrap its child, the child '
'cannot simultaneously expand to fit its parent.
\n
'
'The affected RenderFlex is:
\n
'
'
$this
\n
'
'The creator information is set to:
\n
'
'
$debugCreator
\n
'
'
$addendum
'
'If this message did not help you determine the problem, consider using debugDumpRenderTree():
\n
'
' https://flutter.io/debugging/#rendering-layer
\n
'
' http://docs.flutter.io/flutter/rendering/debugDumpRenderTree.html
\n
'
'If none of the above helps enough to fix this problem, please don
\'
t hesitate to file a bug:
\n
'
' https://github.com/flutter/flutter/issues/new'
);
});
totalFlex
+=
childParentData
.
flex
;
totalFlex
+=
childParentData
.
flex
;
}
else
{
}
else
{
BoxConstraints
innerConstraints
;
BoxConstraints
innerConstraints
;
...
...
packages/flutter/lib/src/rendering/object.dart
View file @
d5deea49
...
@@ -959,8 +959,8 @@ abstract class RenderObject extends AbstractNode implements HitTestTarget {
...
@@ -959,8 +959,8 @@ abstract class RenderObject extends AbstractNode implements HitTestTarget {
/// Verify that the object's constraints are being met. Override
/// Verify that the object's constraints are being met. Override
/// this function in a subclass to verify that your state matches
/// this function in a subclass to verify that your state matches
/// the constraints object. This function is only called in checked
/// the constraints object. This function is only called in checked
/// mode
. If the constraints are not met, it should assert or throw
/// mode
and only when needsLayout is false. If the constraints are
/// an exception.
///
not met, it should assert or throw
an exception.
void
debugAssertDoesMeetConstraints
();
void
debugAssertDoesMeetConstraints
();
/// When true, debugAssertDoesMeetConstraints() is currently
/// When true, debugAssertDoesMeetConstraints() is currently
...
@@ -1111,6 +1111,7 @@ abstract class RenderObject extends AbstractNode implements HitTestTarget {
...
@@ -1111,6 +1111,7 @@ abstract class RenderObject extends AbstractNode implements HitTestTarget {
/// implemented here) to return early if the child does not need to do any
/// implemented here) to return early if the child does not need to do any
/// work to update its layout information.
/// work to update its layout information.
void
layout
(
Constraints
constraints
,
{
bool
parentUsesSize:
false
})
{
void
layout
(
Constraints
constraints
,
{
bool
parentUsesSize:
false
})
{
assert
(
constraints
!=
null
);
assert
(
constraints
.
debugAssertIsNormalized
);
assert
(
constraints
.
debugAssertIsNormalized
);
assert
(!
_debugDoingThisResize
);
assert
(!
_debugDoingThisResize
);
assert
(!
_debugDoingThisLayout
);
assert
(!
_debugDoingThisLayout
);
...
...
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