Commit d5deea49 authored by Ian Hickson's avatar Ian Hickson

Merge pull request #2942 from Hixie/layout-asserts

More elaborate exceptions.
parents e1ac331b aedf41bf
...@@ -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,89 +656,106 @@ abstract class RenderBox extends RenderObject { ...@@ -628,89 +656,106 @@ abstract class RenderBox extends RenderObject {
@override @override
void debugAssertDoesMeetConstraints() { void debugAssertDoesMeetConstraints() {
assert(constraints != null); assert(constraints != null);
assert(_size != null); assert(() {
// verify that the size is not infinite if (!hasSize) {
if (_size.isInfinite) { assert(!needsLayout); // this is called in the size= setter during layout, but in that case we have a size
StringBuffer information = new StringBuffer(); String contract;
if (!constraints.hasBoundedWidth) { if (sizedByParent)
RenderBox node = this; contract = 'Because this RenderBox has sizedByParent set to true, it must set its size in performResize().\n';
while (!node.constraints.hasBoundedWidth && node.parent is RenderBox) else
node = node.parent; contract = 'Because this RenderBox has sizedByParent set to false, it must set its size in performLayout().\n';
information.writeln('The nearest ancestor providing an unbounded width constraint is:'); throw new FlutterError(
information.writeln(' $node'); 'RenderBox did not set its size during layout.\n'
List<String> description = <String>[]; '$contract'
node.debugFillDescription(description); 'It appears that this did not happen; layout completed, but the size property is still null.\n'
for (String line in description) 'The RenderBox in question is:\n'
information.writeln(' $line'); ' $this'
);
} }
if (!constraints.hasBoundedHeight) { // verify that the size is not infinite
RenderBox node = this; if (_size.isInfinite) {
while (!node.constraints.hasBoundedHeight && node.parent is RenderBox) StringBuffer information = new StringBuffer();
node = node.parent; if (!constraints.hasBoundedWidth) {
information.writeln('The nearest ancestor providing an unbounded height constraint is:'); RenderBox node = this;
information.writeln(' $node'); while (!node.constraints.hasBoundedWidth && node.parent is RenderBox)
List<String> description = <String>[]; node = node.parent;
node.debugFillDescription(description); information.writeln('The nearest ancestor providing an unbounded width constraint is:');
for (String line in description) information.writeln(' $node');
information.writeln(' $line'); List<String> description = <String>[];
node.debugFillDescription(description);
for (String line in description)
information.writeln(' $line');
}
if (!constraints.hasBoundedHeight) {
RenderBox node = this;
while (!node.constraints.hasBoundedHeight && node.parent is RenderBox)
node = node.parent;
information.writeln('The nearest ancestor providing an unbounded height constraint is:');
information.writeln(' $node');
List<String> description = <String>[];
node.debugFillDescription(description);
for (String line in description)
information.writeln(' $line');
}
throw new FlutterError(
'$runtimeType object was given an infinite size during layout.\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 '
'that allows its children to pick their own size.\n'
'$information'
'See https://flutter.io/layout/ for more information.'
);
} }
throw new FlutterError( // verify that the size is within the constraints
'$runtimeType object was given an infinite size during layout.\n' if (!constraints.isSatisfiedBy(_size)) {
'This probably means that it is a render object that tries to be\n' throw new FlutterError(
'as big as possible, but it was put inside another render object\n' '$runtimeType does not meet its constraints.\n'
'that allows its children to pick their own size.\n' 'Constraints: $constraints\n'
'$information' 'Size: $_size\n'
'See https://flutter.io/layout/ for more information.' 'If you are not writing your own RenderBox subclass, then this is not '
); 'your fault. Contact support: https://github.com/flutter/flutter/issues/new'
} );
// verify that the size is within the constraints }
if (!constraints.isSatisfiedBy(_size)) { // verify that the intrinsics are also within the constraints
throw new FlutterError( assert(!RenderObject.debugCheckingIntrinsics);
'$runtimeType does not meet its constraints.\n' RenderObject.debugCheckingIntrinsics = true;
'Constraints: $constraints\n' double intrinsic;
'Size: $_size\n' StringBuffer failures = new StringBuffer();
'If you are not writing your own RenderBox subclass, then this is not\n' int failureCount = 0;
'your fault. Contact support: https://github.com/flutter/flutter/issues/new' intrinsic = getMinIntrinsicWidth(constraints);
); if (intrinsic != constraints.constrainWidth(intrinsic)) {
} failures.writeln(' * getMinIntrinsicWidth() -- returned: w=$intrinsic');
// verify that the intrinsics are also within the constraints failureCount += 1;
assert(!RenderObject.debugCheckingIntrinsics); }
RenderObject.debugCheckingIntrinsics = true; intrinsic = getMaxIntrinsicWidth(constraints);
double intrinsic; if (intrinsic != constraints.constrainWidth(intrinsic)) {
StringBuffer failures = new StringBuffer(); failures.writeln(' * getMaxIntrinsicWidth() -- returned: w=$intrinsic');
int failureCount = 0; failureCount += 1;
intrinsic = getMinIntrinsicWidth(constraints); }
if (intrinsic != constraints.constrainWidth(intrinsic)) { intrinsic = getMinIntrinsicHeight(constraints);
failures.writeln(' * getMinIntrinsicWidth() -- returned: w=$intrinsic'); if (intrinsic != constraints.constrainHeight(intrinsic)) {
failureCount += 1; failures.writeln(' * getMinIntrinsicHeight() -- returned: h=$intrinsic');
} failureCount += 1;
intrinsic = getMaxIntrinsicWidth(constraints); }
if (intrinsic != constraints.constrainWidth(intrinsic)) { intrinsic = getMaxIntrinsicHeight(constraints);
failures.writeln(' * getMaxIntrinsicWidth() -- returned: w=$intrinsic'); if (intrinsic != constraints.constrainHeight(intrinsic)) {
failureCount += 1; failures.writeln(' * getMaxIntrinsicHeight() -- returned: h=$intrinsic');
} failureCount += 1;
intrinsic = getMinIntrinsicHeight(constraints); }
if (intrinsic != constraints.constrainHeight(intrinsic)) { RenderObject.debugCheckingIntrinsics = false;
failures.writeln(' * getMinIntrinsicHeight() -- returned: h=$intrinsic'); if (failures.isNotEmpty) {
failureCount += 1; assert(failureCount > 0);
} throw new FlutterError(
intrinsic = getMaxIntrinsicHeight(constraints); 'The intrinsic dimension methods of the $runtimeType class returned values that violate the given constraints.\n'
if (intrinsic != constraints.constrainHeight(intrinsic)) { 'The constraints were: $constraints\n'
failures.writeln(' * getMaxIntrinsicHeight() -- returned: h=$intrinsic'); 'The following method${failureCount > 1 ? "s" : ""} returned values outside of those constraints:\n'
failureCount += 1; '$failures'
} 'If you are not writing your own RenderBox subclass, then this is not\n'
RenderObject.debugCheckingIntrinsics = false; 'your fault. Contact support: https://github.com/flutter/flutter/issues/new'
if (failures.isNotEmpty) { );
assert(failureCount > 0); }
throw new FlutterError( return true;
'The intrinsic dimension methods of the $runtimeType class returned values that violate the given constraints.\n' });
'The constraints were: $constraints\n'
'The following method${failureCount > 1 ? "s" : ""} returned values outside of those constraints:\n'
'$failures'
'If you are not writing your own RenderBox subclass, then this is not\n'
'your fault. Contact support: https://github.com/flutter/flutter/issues/new'
);
}
} }
@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)) {
......
...@@ -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;
......
...@@ -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);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment