Unverified Commit 365f577c authored by Greg Spencer's avatar Greg Spencer Committed by GitHub

Normalize assert checking of clipBehavior (#38568)

I noticed that we were pretty inconsistent with the way that we checked the value of clipBehavior in the framework, so I normalized the usages and updated docs where necessary.

This is a breaking change if you used to pass null explicitly to FlatButton, OutlineButton or RaisedButton constructors, expecting to get Clip.none. It will now assert if you do that. Existing implementations that pass null implicitly by not specifying clipBehavior won't need to change their call sites. It always implicitly defaulted to Clip.none before, and it will continue to do that, it's only places where it was explicitly set to null in order to get the implicit default that it will fail.
parent 53168db9
...@@ -42,7 +42,7 @@ import 'theme.dart'; ...@@ -42,7 +42,7 @@ import 'theme.dart';
class BottomAppBar extends StatefulWidget { class BottomAppBar extends StatefulWidget {
/// Creates a bottom application bar. /// Creates a bottom application bar.
/// ///
/// The [clipBehavior] argument must not be null. /// The [clipBehavior] argument defaults to [Clip.none] and must not be null.
/// Additionally, [elevation] must be non-negative. /// Additionally, [elevation] must be non-negative.
/// ///
/// If [color], [elevation], or [shape] are null, their [BottomAppBarTheme] values will be used. /// If [color], [elevation], or [shape] are null, their [BottomAppBarTheme] values will be used.
...@@ -56,9 +56,9 @@ class BottomAppBar extends StatefulWidget { ...@@ -56,9 +56,9 @@ class BottomAppBar extends StatefulWidget {
this.clipBehavior = Clip.none, this.clipBehavior = Clip.none,
this.notchMargin = 4.0, this.notchMargin = 4.0,
this.child, this.child,
}) : assert(clipBehavior != null), }) : assert(elevation == null || elevation >= 0.0),
assert(elevation == null || elevation >= 0.0),
assert(notchMargin != null), assert(notchMargin != null),
assert(clipBehavior != null),
super(key: key); super(key: key);
/// The widget below this widget in the tree. /// The widget below this widget in the tree.
...@@ -92,6 +92,8 @@ class BottomAppBar extends StatefulWidget { ...@@ -92,6 +92,8 @@ class BottomAppBar extends StatefulWidget {
final NotchedShape shape; final NotchedShape shape;
/// {@macro flutter.widgets.Clip} /// {@macro flutter.widgets.Clip}
///
/// Defaults to [Clip.none], and must not be null.
final Clip clipBehavior; final Clip clipBehavior;
/// The margin between the [FloatingActionButton] and the [BottomAppBar]'s /// The margin between the [FloatingActionButton] and the [BottomAppBar]'s
......
...@@ -241,6 +241,8 @@ class RawMaterialButton extends StatefulWidget { ...@@ -241,6 +241,8 @@ class RawMaterialButton extends StatefulWidget {
final bool autofocus; final bool autofocus;
/// {@macro flutter.widgets.Clip} /// {@macro flutter.widgets.Clip}
///
/// Defaults to [Clip.none], and must not be null.
final Clip clipBehavior; final Clip clipBehavior;
@override @override
......
...@@ -145,6 +145,7 @@ class Card extends StatelessWidget { ...@@ -145,6 +145,7 @@ class Card extends StatelessWidget {
final bool borderOnForeground; final bool borderOnForeground;
/// {@macro flutter.widgets.Clip} /// {@macro flutter.widgets.Clip}
///
/// If this property is null then [ThemeData.cardTheme.clipBehavior] is used. /// If this property is null then [ThemeData.cardTheme.clipBehavior] is used.
/// If that's null then the behavior will be [Clip.none]. /// If that's null then the behavior will be [Clip.none].
final Clip clipBehavior; final Clip clipBehavior;
...@@ -177,7 +178,6 @@ class Card extends StatelessWidget { ...@@ -177,7 +178,6 @@ class Card extends StatelessWidget {
final Widget child; final Widget child;
static const double _defaultElevation = 1.0; static const double _defaultElevation = 1.0;
static const Clip _defaultClipBehavior = Clip.none;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
...@@ -195,7 +195,7 @@ class Card extends StatelessWidget { ...@@ -195,7 +195,7 @@ class Card extends StatelessWidget {
borderRadius: BorderRadius.all(Radius.circular(4.0)), borderRadius: BorderRadius.all(Radius.circular(4.0)),
), ),
borderOnForeground: borderOnForeground, borderOnForeground: borderOnForeground,
clipBehavior: clipBehavior ?? cardTheme.clipBehavior ?? _defaultClipBehavior, clipBehavior: clipBehavior ?? cardTheme.clipBehavior ?? Clip.none,
child: Semantics( child: Semantics(
explicitChildNodes: !semanticContainer, explicitChildNodes: !semanticContainer,
child: child, child: child,
......
...@@ -96,6 +96,8 @@ abstract class ChipAttributes { ...@@ -96,6 +96,8 @@ abstract class ChipAttributes {
ShapeBorder get shape; ShapeBorder get shape;
/// {@macro flutter.widgets.Clip} /// {@macro flutter.widgets.Clip}
///
/// Defaults to [Clip.none], and must not be null.
Clip get clipBehavior; Clip get clipBehavior;
/// {@macro flutter.widgets.Focus.focusNode} /// {@macro flutter.widgets.Focus.focusNode}
...@@ -1227,6 +1229,7 @@ class ActionChip extends StatelessWidget implements ChipAttributes, TappableChip ...@@ -1227,6 +1229,7 @@ class ActionChip extends StatelessWidget implements ChipAttributes, TappableChip
this.elevation, this.elevation,
this.shadowColor, this.shadowColor,
}) : assert(label != null), }) : assert(label != null),
assert(clipBehavior != null),
assert(autofocus != null), assert(autofocus != null),
assert( assert(
onPressed != null, onPressed != null,
......
...@@ -115,12 +115,13 @@ class FlatButton extends MaterialButton { ...@@ -115,12 +115,13 @@ class FlatButton extends MaterialButton {
Brightness colorBrightness, Brightness colorBrightness,
EdgeInsetsGeometry padding, EdgeInsetsGeometry padding,
ShapeBorder shape, ShapeBorder shape,
Clip clipBehavior, Clip clipBehavior = Clip.none,
FocusNode focusNode, FocusNode focusNode,
bool autofocus = false, bool autofocus = false,
MaterialTapTargetSize materialTapTargetSize, MaterialTapTargetSize materialTapTargetSize,
@required Widget child, @required Widget child,
}) : assert(autofocus != null), }) : assert(clipBehavior != null),
assert(autofocus != null),
super( super(
key: key, key: key,
onPressed: onPressed, onPressed: onPressed,
...@@ -196,7 +197,7 @@ class FlatButton extends MaterialButton { ...@@ -196,7 +197,7 @@ class FlatButton extends MaterialButton {
padding: buttonTheme.getPadding(this), padding: buttonTheme.getPadding(this),
constraints: buttonTheme.getConstraints(this), constraints: buttonTheme.getConstraints(this),
shape: buttonTheme.getShape(this), shape: buttonTheme.getShape(this),
clipBehavior: clipBehavior ?? Clip.none, clipBehavior: clipBehavior,
focusNode: focusNode, focusNode: focusNode,
autofocus: autofocus, autofocus: autofocus,
materialTapTargetSize: buttonTheme.getMaterialTapTargetSize(this), materialTapTargetSize: buttonTheme.getMaterialTapTargetSize(this),
...@@ -227,7 +228,7 @@ class _FlatButtonWithIcon extends FlatButton with MaterialButtonWithIconMixin { ...@@ -227,7 +228,7 @@ class _FlatButtonWithIcon extends FlatButton with MaterialButtonWithIconMixin {
Brightness colorBrightness, Brightness colorBrightness,
EdgeInsetsGeometry padding, EdgeInsetsGeometry padding,
ShapeBorder shape, ShapeBorder shape,
Clip clipBehavior, Clip clipBehavior = Clip.none,
FocusNode focusNode, FocusNode focusNode,
bool autofocus = false, bool autofocus = false,
MaterialTapTargetSize materialTapTargetSize, MaterialTapTargetSize materialTapTargetSize,
...@@ -235,6 +236,7 @@ class _FlatButtonWithIcon extends FlatButton with MaterialButtonWithIconMixin { ...@@ -235,6 +236,7 @@ class _FlatButtonWithIcon extends FlatButton with MaterialButtonWithIconMixin {
@required Widget label, @required Widget label,
}) : assert(icon != null), }) : assert(icon != null),
assert(label != null), assert(label != null),
assert(clipBehavior != null),
assert(autofocus != null), assert(autofocus != null),
super( super(
key: key, key: key,
......
...@@ -122,7 +122,7 @@ class _DefaultHeroTag { ...@@ -122,7 +122,7 @@ class _DefaultHeroTag {
class FloatingActionButton extends StatelessWidget { class FloatingActionButton extends StatelessWidget {
/// Creates a circular floating action button. /// Creates a circular floating action button.
/// ///
/// The [mini] and [clipBehavior] arguments must be non-null. Additionally, /// The [mini] and [clipBehavior] arguments must not be null. Additionally,
/// [elevation], [highlightElevation], and [disabledElevation] (if specified) /// [elevation], [highlightElevation], and [disabledElevation] (if specified)
/// must be non-negative. /// must be non-negative.
const FloatingActionButton({ const FloatingActionButton({
...@@ -154,6 +154,7 @@ class FloatingActionButton extends StatelessWidget { ...@@ -154,6 +154,7 @@ class FloatingActionButton extends StatelessWidget {
assert(highlightElevation == null || highlightElevation >= 0.0), assert(highlightElevation == null || highlightElevation >= 0.0),
assert(disabledElevation == null || disabledElevation >= 0.0), assert(disabledElevation == null || disabledElevation >= 0.0),
assert(mini != null), assert(mini != null),
assert(clipBehavior != null),
assert(isExtended != null), assert(isExtended != null),
assert(autofocus != null), assert(autofocus != null),
_sizeConstraints = mini ? _kMiniSizeConstraints : _kSizeConstraints, _sizeConstraints = mini ? _kMiniSizeConstraints : _kSizeConstraints,
...@@ -162,7 +163,7 @@ class FloatingActionButton extends StatelessWidget { ...@@ -162,7 +163,7 @@ class FloatingActionButton extends StatelessWidget {
/// Creates a wider [StadiumBorder]-shaped floating action button with /// Creates a wider [StadiumBorder]-shaped floating action button with
/// an optional [icon] and a [label]. /// an optional [icon] and a [label].
/// ///
/// The [label], [autofocus], and [clipBehavior] arguments must non-null. /// The [label], [autofocus], and [clipBehavior] arguments must not be null.
/// Additionally, [elevation], [highlightElevation], and [disabledElevation] /// Additionally, [elevation], [highlightElevation], and [disabledElevation]
/// (if specified) must be non-negative. /// (if specified) must be non-negative.
FloatingActionButton.extended({ FloatingActionButton.extended({
...@@ -194,6 +195,7 @@ class FloatingActionButton extends StatelessWidget { ...@@ -194,6 +195,7 @@ class FloatingActionButton extends StatelessWidget {
assert(highlightElevation == null || highlightElevation >= 0.0), assert(highlightElevation == null || highlightElevation >= 0.0),
assert(disabledElevation == null || disabledElevation >= 0.0), assert(disabledElevation == null || disabledElevation >= 0.0),
assert(isExtended != null), assert(isExtended != null),
assert(clipBehavior != null),
assert(autofocus != null), assert(autofocus != null),
_sizeConstraints = _kExtendedSizeConstraints, _sizeConstraints = _kExtendedSizeConstraints,
mini = false, mini = false,
...@@ -363,6 +365,8 @@ class FloatingActionButton extends StatelessWidget { ...@@ -363,6 +365,8 @@ class FloatingActionButton extends StatelessWidget {
final ShapeBorder shape; final ShapeBorder shape;
/// {@macro flutter.widgets.Clip} /// {@macro flutter.widgets.Clip}
///
/// Defaults to [Clip.none], and must not be null.
final Clip clipBehavior; final Clip clipBehavior;
/// True if this is an "extended" floating action button. /// True if this is an "extended" floating action button.
...@@ -463,7 +467,7 @@ class FloatingActionButton extends StatelessWidget { ...@@ -463,7 +467,7 @@ class FloatingActionButton extends StatelessWidget {
splashColor: splashColor, splashColor: splashColor,
textStyle: textStyle, textStyle: textStyle,
shape: shape, shape: shape,
clipBehavior: clipBehavior ?? Clip.none, clipBehavior: clipBehavior,
focusNode: focusNode, focusNode: focusNode,
autofocus: autofocus, autofocus: autofocus,
child: child, child: child,
......
...@@ -157,9 +157,9 @@ abstract class MaterialInkController { ...@@ -157,9 +157,9 @@ abstract class MaterialInkController {
class Material extends StatefulWidget { class Material extends StatefulWidget {
/// Creates a piece of material. /// Creates a piece of material.
/// ///
/// The [type], [elevation], [shadowColor], [borderOnForeground] and /// The [type], [elevation], [shadowColor], [borderOnForeground],
/// [animationDuration] arguments must not be null. Additionally, [elevation] /// [clipBehavior], and [animationDuration] arguments must not be null.
/// must be non-negative. /// Additionally, [elevation] must be non-negative.
/// ///
/// If a [shape] is specified, then the [borderRadius] property must be /// If a [shape] is specified, then the [borderRadius] property must be
/// null and the [type] property must not be [MaterialType.circle]. If the /// null and the [type] property must not be [MaterialType.circle]. If the
...@@ -185,8 +185,8 @@ class Material extends StatefulWidget { ...@@ -185,8 +185,8 @@ class Material extends StatefulWidget {
assert(!(shape != null && borderRadius != null)), assert(!(shape != null && borderRadius != null)),
assert(animationDuration != null), assert(animationDuration != null),
assert(!(identical(type, MaterialType.circle) && (borderRadius != null || shape != null))), assert(!(identical(type, MaterialType.circle) && (borderRadius != null || shape != null))),
assert(clipBehavior != null),
assert(borderOnForeground != null), assert(borderOnForeground != null),
assert(clipBehavior != null),
super(key: key); super(key: key);
/// The widget below this widget in the tree. /// The widget below this widget in the tree.
...@@ -264,6 +264,8 @@ class Material extends StatefulWidget { ...@@ -264,6 +264,8 @@ class Material extends StatefulWidget {
/// See the enum [Clip] for details of all possible options and their common /// See the enum [Clip] for details of all possible options and their common
/// use cases. /// use cases.
/// {@endtemplate} /// {@endtemplate}
///
/// Defaults to [Clip.none], and must not be null.
final Clip clipBehavior; final Clip clipBehavior;
/// Defines the duration of animated changes for [shape], [elevation], /// Defines the duration of animated changes for [shape], [elevation],
...@@ -670,6 +672,11 @@ class ShapeBorderTween extends Tween<ShapeBorder> { ...@@ -670,6 +672,11 @@ class ShapeBorderTween extends Tween<ShapeBorder> {
/// ///
/// Animates [elevation], [shadowColor], and [shape]. /// Animates [elevation], [shadowColor], and [shape].
class _MaterialInterior extends ImplicitlyAnimatedWidget { class _MaterialInterior extends ImplicitlyAnimatedWidget {
/// Creates a const instance of [_MaterialInterior].
///
/// The [child], [shape], [clipBehavior], [color], and [shadowColor] arguments
/// must not be null. The [elevation] must be specified and greater than or
/// equal to zero.
const _MaterialInterior({ const _MaterialInterior({
Key key, Key key,
@required this.child, @required this.child,
...@@ -707,6 +714,8 @@ class _MaterialInterior extends ImplicitlyAnimatedWidget { ...@@ -707,6 +714,8 @@ class _MaterialInterior extends ImplicitlyAnimatedWidget {
final bool borderOnForeground; final bool borderOnForeground;
/// {@macro flutter.widgets.Clip} /// {@macro flutter.widgets.Clip}
///
/// Defaults to [Clip.none], and must not be null.
final Clip clipBehavior; final Clip clipBehavior;
/// The target z-coordinate at which to place this physical object relative /// The target z-coordinate at which to place this physical object relative
......
...@@ -77,7 +77,8 @@ class MaterialButton extends StatelessWidget { ...@@ -77,7 +77,8 @@ class MaterialButton extends StatelessWidget {
this.minWidth, this.minWidth,
this.height, this.height,
this.child, this.child,
}) : assert(autofocus != null), }) : assert(clipBehavior != null),
assert(autofocus != null),
assert(elevation == null || elevation >= 0.0), assert(elevation == null || elevation >= 0.0),
assert(focusElevation == null || focusElevation >= 0.0), assert(focusElevation == null || focusElevation >= 0.0),
assert(hoverElevation == null || hoverElevation >= 0.0), assert(hoverElevation == null || hoverElevation >= 0.0),
...@@ -306,6 +307,8 @@ class MaterialButton extends StatelessWidget { ...@@ -306,6 +307,8 @@ class MaterialButton extends StatelessWidget {
final ShapeBorder shape; final ShapeBorder shape;
/// {@macro flutter.widgets.Clip} /// {@macro flutter.widgets.Clip}
///
/// Defaults to [Clip.none], and must not be null.
final Clip clipBehavior; final Clip clipBehavior;
/// {@macro flutter.widgets.Focus.focusNode} /// {@macro flutter.widgets.Focus.focusNode}
...@@ -362,7 +365,7 @@ class MaterialButton extends StatelessWidget { ...@@ -362,7 +365,7 @@ class MaterialButton extends StatelessWidget {
minHeight: height, minHeight: height,
), ),
shape: buttonTheme.getShape(this), shape: buttonTheme.getShape(this),
clipBehavior: clipBehavior ?? Clip.none, clipBehavior: clipBehavior,
focusNode: focusNode, focusNode: focusNode,
animationDuration: buttonTheme.getAnimationDuration(this), animationDuration: buttonTheme.getAnimationDuration(this),
child: child, child: child,
......
...@@ -75,11 +75,12 @@ class OutlineButton extends MaterialButton { ...@@ -75,11 +75,12 @@ class OutlineButton extends MaterialButton {
this.highlightedBorderColor, this.highlightedBorderColor,
EdgeInsetsGeometry padding, EdgeInsetsGeometry padding,
ShapeBorder shape, ShapeBorder shape,
Clip clipBehavior, Clip clipBehavior = Clip.none,
FocusNode focusNode, FocusNode focusNode,
bool autofocus = false, bool autofocus = false,
Widget child, Widget child,
}) : assert(highlightElevation == null || highlightElevation >= 0.0), }) : assert(highlightElevation == null || highlightElevation >= 0.0),
assert(clipBehavior != null),
assert(autofocus != null), assert(autofocus != null),
super( super(
key: key, key: key,
...@@ -220,12 +221,13 @@ class _OutlineButtonWithIcon extends OutlineButton with MaterialButtonWithIconMi ...@@ -220,12 +221,13 @@ class _OutlineButtonWithIcon extends OutlineButton with MaterialButtonWithIconMi
BorderSide borderSide, BorderSide borderSide,
EdgeInsetsGeometry padding, EdgeInsetsGeometry padding,
ShapeBorder shape, ShapeBorder shape,
Clip clipBehavior, Clip clipBehavior = Clip.none,
FocusNode focusNode, FocusNode focusNode,
bool autofocus = false, bool autofocus = false,
@required Widget icon, @required Widget icon,
@required Widget label, @required Widget label,
}) : assert(highlightElevation == null || highlightElevation >= 0.0), }) : assert(highlightElevation == null || highlightElevation >= 0.0),
assert(clipBehavior != null),
assert(autofocus != null), assert(autofocus != null),
assert(icon != null), assert(icon != null),
assert(label != null), assert(label != null),
...@@ -279,12 +281,13 @@ class _OutlineButton extends StatefulWidget { ...@@ -279,12 +281,13 @@ class _OutlineButton extends StatefulWidget {
@required this.highlightedBorderColor, @required this.highlightedBorderColor,
this.padding, this.padding,
this.shape, this.shape,
this.clipBehavior, this.clipBehavior = Clip.none,
this.focusNode, this.focusNode,
this.autofocus = false, this.autofocus = false,
this.child, this.child,
}) : assert(highlightElevation != null && highlightElevation >= 0.0), }) : assert(highlightElevation != null && highlightElevation >= 0.0),
assert(highlightedBorderColor != null), assert(highlightedBorderColor != null),
assert(clipBehavior != null),
assert(autofocus != null), assert(autofocus != null),
super(key: key); super(key: key);
......
...@@ -126,7 +126,7 @@ class RaisedButton extends MaterialButton { ...@@ -126,7 +126,7 @@ class RaisedButton extends MaterialButton {
double disabledElevation, double disabledElevation,
EdgeInsetsGeometry padding, EdgeInsetsGeometry padding,
ShapeBorder shape, ShapeBorder shape,
Clip clipBehavior, Clip clipBehavior = Clip.none,
FocusNode focusNode, FocusNode focusNode,
bool autofocus = false, bool autofocus = false,
MaterialTapTargetSize materialTapTargetSize, MaterialTapTargetSize materialTapTargetSize,
...@@ -138,6 +138,7 @@ class RaisedButton extends MaterialButton { ...@@ -138,6 +138,7 @@ class RaisedButton extends MaterialButton {
assert(hoverElevation == null || hoverElevation >= 0.0), assert(hoverElevation == null || hoverElevation >= 0.0),
assert(highlightElevation == null || highlightElevation >= 0.0), assert(highlightElevation == null || highlightElevation >= 0.0),
assert(disabledElevation == null || disabledElevation >= 0.0), assert(disabledElevation == null || disabledElevation >= 0.0),
assert(clipBehavior != null),
super( super(
key: key, key: key,
onPressed: onPressed, onPressed: onPressed,
...@@ -209,7 +210,7 @@ class RaisedButton extends MaterialButton { ...@@ -209,7 +210,7 @@ class RaisedButton extends MaterialButton {
return RawMaterialButton( return RawMaterialButton(
onPressed: onPressed, onPressed: onPressed,
onHighlightChanged: onHighlightChanged, onHighlightChanged: onHighlightChanged,
clipBehavior: clipBehavior ?? Clip.none, clipBehavior: clipBehavior,
fillColor: buttonTheme.getFillColor(this), fillColor: buttonTheme.getFillColor(this),
textStyle: theme.textTheme.button.copyWith(color: buttonTheme.getTextColor(this)), textStyle: theme.textTheme.button.copyWith(color: buttonTheme.getTextColor(this)),
focusColor: buttonTheme.getFocusColor(this), focusColor: buttonTheme.getFocusColor(this),
...@@ -276,6 +277,7 @@ class _RaisedButtonWithIcon extends RaisedButton with MaterialButtonWithIconMixi ...@@ -276,6 +277,7 @@ class _RaisedButtonWithIcon extends RaisedButton with MaterialButtonWithIconMixi
}) : assert(elevation == null || elevation >= 0.0), }) : assert(elevation == null || elevation >= 0.0),
assert(highlightElevation == null || highlightElevation >= 0.0), assert(highlightElevation == null || highlightElevation >= 0.0),
assert(disabledElevation == null || disabledElevation >= 0.0), assert(disabledElevation == null || disabledElevation >= 0.0),
assert(clipBehavior != null),
assert(icon != null), assert(icon != null),
assert(label != null), assert(label != null),
assert(autofocus != null), assert(autofocus != null),
......
...@@ -1058,8 +1058,10 @@ class OffsetLayer extends ContainerLayer { ...@@ -1058,8 +1058,10 @@ class OffsetLayer extends ContainerLayer {
class ClipRectLayer extends ContainerLayer { class ClipRectLayer extends ContainerLayer {
/// Creates a layer with a rectangular clip. /// Creates a layer with a rectangular clip.
/// ///
/// The [clipRect] and [clipBehavior] properties must be non-null before the /// The [clipRect] argument must not be null before the compositing phase of
/// compositing phase of the pipeline. /// the pipeline.
///
/// The [clipBehavior] argument must not be null, and must not be [Clip.none].
ClipRectLayer({ ClipRectLayer({
Rect clipRect, Rect clipRect,
Clip clipBehavior = Clip.hardEdge, Clip clipBehavior = Clip.hardEdge,
...@@ -1082,10 +1084,12 @@ class ClipRectLayer extends ContainerLayer { ...@@ -1082,10 +1084,12 @@ class ClipRectLayer extends ContainerLayer {
} }
/// {@template flutter.clipper.clipBehavior} /// {@template flutter.clipper.clipBehavior}
/// Controls how to clip (defaults to [Clip.hardEdge]). /// Controls how to clip.
/// ///
/// [Clip.none] is not allowed here. /// Must not be set to null or [Clip.none].
/// {@endtemplate} /// {@endtemplate}
///
/// Defaults to [Clip.hardEdge].
Clip get clipBehavior => _clipBehavior; Clip get clipBehavior => _clipBehavior;
Clip _clipBehavior; Clip _clipBehavior;
set clipBehavior(Clip value) { set clipBehavior(Clip value) {
...@@ -1170,6 +1174,8 @@ class ClipRRectLayer extends ContainerLayer { ...@@ -1170,6 +1174,8 @@ class ClipRRectLayer extends ContainerLayer {
} }
/// {@macro flutter.clipper.clipBehavior} /// {@macro flutter.clipper.clipBehavior}
///
/// Defaults to [Clip.antiAlias].
Clip get clipBehavior => _clipBehavior; Clip get clipBehavior => _clipBehavior;
Clip _clipBehavior; Clip _clipBehavior;
set clipBehavior(Clip value) { set clipBehavior(Clip value) {
...@@ -1254,6 +1260,8 @@ class ClipPathLayer extends ContainerLayer { ...@@ -1254,6 +1260,8 @@ class ClipPathLayer extends ContainerLayer {
} }
/// {@macro flutter.clipper.clipBehavior} /// {@macro flutter.clipper.clipBehavior}
///
/// Defaults to [Clip.antiAlias].
Clip get clipBehavior => _clipBehavior; Clip get clipBehavior => _clipBehavior;
Clip _clipBehavior; Clip _clipBehavior;
set clipBehavior(Clip value) { set clipBehavior(Clip value) {
......
...@@ -1285,12 +1285,14 @@ class RenderClipRect extends _RenderCustomClip<Rect> { ...@@ -1285,12 +1285,14 @@ class RenderClipRect extends _RenderCustomClip<Rect> {
/// If [clipper] is null, the clip will match the layout size and position of /// If [clipper] is null, the clip will match the layout size and position of
/// the child. /// the child.
/// ///
/// The [clipBehavior] cannot be [Clip.none]. /// The [clipBehavior] must not be null or [Clip.none].
RenderClipRect({ RenderClipRect({
RenderBox child, RenderBox child,
CustomClipper<Rect> clipper, CustomClipper<Rect> clipper,
Clip clipBehavior = Clip.antiAlias, Clip clipBehavior = Clip.antiAlias,
}) : super(child: child, clipper: clipper, clipBehavior: clipBehavior); }) : assert(clipBehavior != null),
assert(clipBehavior != Clip.none),
super(child: child, clipper: clipper, clipBehavior: clipBehavior);
@override @override
Rect get _defaultClip => Offset.zero & size; Rect get _defaultClip => Offset.zero & size;
...@@ -1342,13 +1344,14 @@ class RenderClipRRect extends _RenderCustomClip<RRect> { ...@@ -1342,13 +1344,14 @@ class RenderClipRRect extends _RenderCustomClip<RRect> {
/// ///
/// If [clipper] is non-null, then [borderRadius] is ignored. /// If [clipper] is non-null, then [borderRadius] is ignored.
/// ///
/// The [clipBehavior] cannot be [Clip.none]. /// The [clipBehavior] argument must not be null or [Clip.none].
RenderClipRRect({ RenderClipRRect({
RenderBox child, RenderBox child,
BorderRadius borderRadius = BorderRadius.zero, BorderRadius borderRadius = BorderRadius.zero,
CustomClipper<RRect> clipper, CustomClipper<RRect> clipper,
Clip clipBehavior = Clip.antiAlias, Clip clipBehavior = Clip.antiAlias,
}) : assert(clipBehavior != Clip.none), }) : assert(clipBehavior != null),
assert(clipBehavior != Clip.none),
_borderRadius = borderRadius, _borderRadius = borderRadius,
super(child: child, clipper: clipper, clipBehavior: clipBehavior) { super(child: child, clipper: clipper, clipBehavior: clipBehavior) {
assert(_borderRadius != null || clipper != null); assert(_borderRadius != null || clipper != null);
...@@ -1418,12 +1421,13 @@ class RenderClipOval extends _RenderCustomClip<Rect> { ...@@ -1418,12 +1421,13 @@ class RenderClipOval extends _RenderCustomClip<Rect> {
/// If [clipper] is null, the oval will be inscribed into the layout size and /// If [clipper] is null, the oval will be inscribed into the layout size and
/// position of the child. /// position of the child.
/// ///
/// The [clipBehavior] cannot be [Clip.none]. /// The [clipBehavior] argument must not be null or [Clip.none].
RenderClipOval({ RenderClipOval({
RenderBox child, RenderBox child,
CustomClipper<Rect> clipper, CustomClipper<Rect> clipper,
Clip clipBehavior = Clip.antiAlias, Clip clipBehavior = Clip.antiAlias,
}) : assert(clipBehavior != Clip.none), }) : assert(clipBehavior != null),
assert(clipBehavior != Clip.none),
super(child: child, clipper: clipper, clipBehavior: clipBehavior); super(child: child, clipper: clipper, clipBehavior: clipBehavior);
Rect _cachedRect; Rect _cachedRect;
...@@ -1496,12 +1500,13 @@ class RenderClipPath extends _RenderCustomClip<Path> { ...@@ -1496,12 +1500,13 @@ class RenderClipPath extends _RenderCustomClip<Path> {
/// consider using a [RenderClipRect], which can achieve the same effect more /// consider using a [RenderClipRect], which can achieve the same effect more
/// efficiently. /// efficiently.
/// ///
/// The [clipBehavior] cannot be [Clip.none]. /// The [clipBehavior] argument must not be null or [Clip.none].
RenderClipPath({ RenderClipPath({
RenderBox child, RenderBox child,
CustomClipper<Path> clipper, CustomClipper<Path> clipper,
Clip clipBehavior = Clip.antiAlias, Clip clipBehavior = Clip.antiAlias,
}) : assert(clipBehavior != Clip.none), }) : assert(clipBehavior != null),
assert(clipBehavior != Clip.none),
super(child: child, clipper: clipper, clipBehavior: clipBehavior); super(child: child, clipper: clipper, clipBehavior: clipBehavior);
@override @override
...@@ -1632,8 +1637,9 @@ class RenderPhysicalModel extends _RenderPhysicalModelBase<RRect> { ...@@ -1632,8 +1637,9 @@ class RenderPhysicalModel extends _RenderPhysicalModelBase<RRect> {
/// ///
/// The [color] is required. /// The [color] is required.
/// ///
/// The [shape], [elevation], [color], and [shadowColor] must not be null. /// The [shape], [elevation], [color], [clipBehavior], and [shadowColor]
/// Additionally, the [elevation] must be non-negative. /// arguments must not be null. Additionally, the [elevation] must be
/// non-negative.
RenderPhysicalModel({ RenderPhysicalModel({
RenderBox child, RenderBox child,
BoxShape shape = BoxShape.rectangle, BoxShape shape = BoxShape.rectangle,
......
...@@ -582,19 +582,29 @@ class ClipRect extends SingleChildRenderObjectWidget { ...@@ -582,19 +582,29 @@ class ClipRect extends SingleChildRenderObjectWidget {
/// ///
/// If [clipper] is null, the clip will match the layout size and position of /// If [clipper] is null, the clip will match the layout size and position of
/// the child. /// the child.
const ClipRect({ Key key, this.clipper, this.clipBehavior = Clip.hardEdge, Widget child }) : super(key: key, child: child); ///
/// The [clipBehavior] argument must not be null or [Clip.none].
const ClipRect({ Key key, this.clipper, this.clipBehavior = Clip.hardEdge, Widget child })
: assert(clipBehavior != null),
super(key: key, child: child);
/// If non-null, determines which clip to use. /// If non-null, determines which clip to use.
final CustomClipper<Rect> clipper; final CustomClipper<Rect> clipper;
/// {@macro flutter.clipper.clipBehavior} /// {@macro flutter.clipper.clipBehavior}
///
/// Defaults to [Clip.hardEdge].
final Clip clipBehavior; final Clip clipBehavior;
@override @override
RenderClipRect createRenderObject(BuildContext context) => RenderClipRect(clipper: clipper, clipBehavior: clipBehavior); RenderClipRect createRenderObject(BuildContext context) {
assert(clipBehavior != Clip.none);
return RenderClipRect(clipper: clipper, clipBehavior: clipBehavior);
}
@override @override
void updateRenderObject(BuildContext context, RenderClipRect renderObject) { void updateRenderObject(BuildContext context, RenderClipRect renderObject) {
assert(clipBehavior != Clip.none);
renderObject renderObject
..clipper = clipper ..clipper = clipper
..clipBehavior = clipBehavior; ..clipBehavior = clipBehavior;
...@@ -633,6 +643,8 @@ class ClipRRect extends SingleChildRenderObjectWidget { ...@@ -633,6 +643,8 @@ class ClipRRect extends SingleChildRenderObjectWidget {
/// right-angled corners. /// right-angled corners.
/// ///
/// If [clipper] is non-null, then [borderRadius] is ignored. /// If [clipper] is non-null, then [borderRadius] is ignored.
///
/// The [clipBehavior] argument must not be null or [Clip.none].
const ClipRRect({ const ClipRRect({
Key key, Key key,
this.borderRadius, this.borderRadius,
...@@ -655,13 +667,19 @@ class ClipRRect extends SingleChildRenderObjectWidget { ...@@ -655,13 +667,19 @@ class ClipRRect extends SingleChildRenderObjectWidget {
final CustomClipper<RRect> clipper; final CustomClipper<RRect> clipper;
/// {@macro flutter.clipper.clipBehavior} /// {@macro flutter.clipper.clipBehavior}
///
/// Defaults to [Clip.antiAlias].
final Clip clipBehavior; final Clip clipBehavior;
@override @override
RenderClipRRect createRenderObject(BuildContext context) => RenderClipRRect(borderRadius: borderRadius, clipper: clipper, clipBehavior: clipBehavior); RenderClipRRect createRenderObject(BuildContext context) {
assert(clipBehavior != Clip.none);
return RenderClipRRect(borderRadius: borderRadius, clipper: clipper, clipBehavior: clipBehavior);
}
@override @override
void updateRenderObject(BuildContext context, RenderClipRRect renderObject) { void updateRenderObject(BuildContext context, RenderClipRRect renderObject) {
assert(clipBehavior != Clip.none);
renderObject renderObject
..borderRadius = borderRadius ..borderRadius = borderRadius
..clipBehavior = clipBehavior ..clipBehavior = clipBehavior
...@@ -693,7 +711,11 @@ class ClipOval extends SingleChildRenderObjectWidget { ...@@ -693,7 +711,11 @@ class ClipOval extends SingleChildRenderObjectWidget {
/// ///
/// If [clipper] is null, the oval will be inscribed into the layout size and /// If [clipper] is null, the oval will be inscribed into the layout size and
/// position of the child. /// position of the child.
const ClipOval({ Key key, this.clipper, this.clipBehavior = Clip.antiAlias, Widget child }) : super(key: key, child: child); ///
/// The [clipBehavior] argument must not be null or [Clip.none].
const ClipOval({Key key, this.clipper, this.clipBehavior = Clip.antiAlias, Widget child})
: assert(clipBehavior != null),
super(key: key, child: child);
/// If non-null, determines which clip to use. /// If non-null, determines which clip to use.
/// ///
...@@ -707,13 +729,19 @@ class ClipOval extends SingleChildRenderObjectWidget { ...@@ -707,13 +729,19 @@ class ClipOval extends SingleChildRenderObjectWidget {
final CustomClipper<Rect> clipper; final CustomClipper<Rect> clipper;
/// {@macro flutter.clipper.clipBehavior} /// {@macro flutter.clipper.clipBehavior}
///
/// Defaults to [Clip.antiAlias].
final Clip clipBehavior; final Clip clipBehavior;
@override @override
RenderClipOval createRenderObject(BuildContext context) => RenderClipOval(clipper: clipper, clipBehavior: clipBehavior); RenderClipOval createRenderObject(BuildContext context) {
assert(clipBehavior != Clip.none);
return RenderClipOval(clipper: clipper, clipBehavior: clipBehavior);
}
@override @override
void updateRenderObject(BuildContext context, RenderClipOval renderObject) { void updateRenderObject(BuildContext context, RenderClipOval renderObject) {
assert(clipBehavior != Clip.none);
renderObject renderObject
..clipper = clipper ..clipper = clipper
..clipBehavior = clipBehavior; ..clipBehavior = clipBehavior;
...@@ -754,12 +782,15 @@ class ClipPath extends SingleChildRenderObjectWidget { ...@@ -754,12 +782,15 @@ class ClipPath extends SingleChildRenderObjectWidget {
/// size and location of the child. However, rather than use this default, /// size and location of the child. However, rather than use this default,
/// consider using a [ClipRect], which can achieve the same effect more /// consider using a [ClipRect], which can achieve the same effect more
/// efficiently. /// efficiently.
///
/// The [clipBehavior] argument must not be null or [Clip.none].
const ClipPath({ const ClipPath({
Key key, Key key,
this.clipper, this.clipper,
this.clipBehavior = Clip.antiAlias, this.clipBehavior = Clip.antiAlias,
Widget child, Widget child,
}) : super(key: key, child: child); }) : assert(clipBehavior != null),
super(key: key, child: child);
/// Creates a shape clip. /// Creates a shape clip.
/// ///
...@@ -771,6 +802,8 @@ class ClipPath extends SingleChildRenderObjectWidget { ...@@ -771,6 +802,8 @@ class ClipPath extends SingleChildRenderObjectWidget {
Clip clipBehavior = Clip.antiAlias, Clip clipBehavior = Clip.antiAlias,
Widget child, Widget child,
}) { }) {
assert(clipBehavior != null);
assert(clipBehavior != Clip.none);
assert(shape != null); assert(shape != null);
return Builder( return Builder(
key: key, key: key,
...@@ -795,13 +828,19 @@ class ClipPath extends SingleChildRenderObjectWidget { ...@@ -795,13 +828,19 @@ class ClipPath extends SingleChildRenderObjectWidget {
final CustomClipper<Path> clipper; final CustomClipper<Path> clipper;
/// {@macro flutter.clipper.clipBehavior} /// {@macro flutter.clipper.clipBehavior}
///
/// Defaults to [Clip.antiAlias].
final Clip clipBehavior; final Clip clipBehavior;
@override @override
RenderClipPath createRenderObject(BuildContext context) => RenderClipPath(clipper: clipper, clipBehavior: clipBehavior); RenderClipPath createRenderObject(BuildContext context) {
assert(clipBehavior != Clip.none);
return RenderClipPath(clipper: clipper, clipBehavior: clipBehavior);
}
@override @override
void updateRenderObject(BuildContext context, RenderClipPath renderObject) { void updateRenderObject(BuildContext context, RenderClipPath renderObject) {
assert(clipBehavior != Clip.none);
renderObject renderObject
..clipper = clipper ..clipper = clipper
..clipBehavior = clipBehavior; ..clipBehavior = clipBehavior;
...@@ -838,8 +877,8 @@ class PhysicalModel extends SingleChildRenderObjectWidget { ...@@ -838,8 +877,8 @@ class PhysicalModel extends SingleChildRenderObjectWidget {
/// ///
/// The [color] is required; physical things have a color. /// The [color] is required; physical things have a color.
/// ///
/// The [shape], [elevation], [color], and [shadowColor] must not be null. /// The [shape], [elevation], [color], [clipBehavior], and [shadowColor] must
/// Additionally, the [elevation] must be non-negative. /// not be null. Additionally, the [elevation] must be non-negative.
const PhysicalModel({ const PhysicalModel({
Key key, Key key,
this.shape = BoxShape.rectangle, this.shape = BoxShape.rectangle,
...@@ -853,12 +892,15 @@ class PhysicalModel extends SingleChildRenderObjectWidget { ...@@ -853,12 +892,15 @@ class PhysicalModel extends SingleChildRenderObjectWidget {
assert(elevation != null && elevation >= 0.0), assert(elevation != null && elevation >= 0.0),
assert(color != null), assert(color != null),
assert(shadowColor != null), assert(shadowColor != null),
assert(clipBehavior != null),
super(key: key, child: child); super(key: key, child: child);
/// The type of shape. /// The type of shape.
final BoxShape shape; final BoxShape shape;
/// {@macro flutter.widgets.Clip} /// {@macro flutter.widgets.Clip}
///
/// Defaults to [Clip.none].
final Clip clipBehavior; final Clip clipBehavior;
/// The border radius of the rounded corners. /// The border radius of the rounded corners.
...@@ -931,8 +973,8 @@ class PhysicalShape extends SingleChildRenderObjectWidget { ...@@ -931,8 +973,8 @@ class PhysicalShape extends SingleChildRenderObjectWidget {
/// ///
/// The [color] is required; physical things have a color. /// The [color] is required; physical things have a color.
/// ///
/// The [clipper], [elevation], [color], and [shadowColor] must not be null. /// The [clipper], [elevation], [color], [clipBehavior], and [shadowColor]
/// Additionally, the [elevation] must be non-negative. /// must not be null. Additionally, the [elevation] must be non-negative.
const PhysicalShape({ const PhysicalShape({
Key key, Key key,
@required this.clipper, @required this.clipper,
...@@ -956,6 +998,8 @@ class PhysicalShape extends SingleChildRenderObjectWidget { ...@@ -956,6 +998,8 @@ class PhysicalShape extends SingleChildRenderObjectWidget {
final CustomClipper<Path> clipper; final CustomClipper<Path> clipper;
/// {@macro flutter.widgets.Clip} /// {@macro flutter.widgets.Clip}
///
/// Defaults to [Clip.none].
final Clip clipBehavior; final Clip clipBehavior;
/// The z-coordinate relative to the parent at which to place this physical /// The z-coordinate relative to the parent at which to place this physical
......
...@@ -1450,9 +1450,9 @@ class _AnimatedDefaultTextStyleState extends AnimatedWidgetBaseState<AnimatedDef ...@@ -1450,9 +1450,9 @@ class _AnimatedDefaultTextStyleState extends AnimatedWidgetBaseState<AnimatedDef
class AnimatedPhysicalModel extends ImplicitlyAnimatedWidget { class AnimatedPhysicalModel extends ImplicitlyAnimatedWidget {
/// Creates a widget that animates the properties of a [PhysicalModel]. /// Creates a widget that animates the properties of a [PhysicalModel].
/// ///
/// The [child], [shape], [borderRadius], [elevation], [color], [shadowColor], [curve], and /// The [child], [shape], [borderRadius], [elevation], [color], [shadowColor],
/// [duration] arguments must not be null. Additionally, [elevation] must be /// [curve], [clipBehavior], and [duration] arguments must not be null.
/// non-negative. /// Additionally, [elevation] must be non-negative.
/// ///
/// Animating [color] is optional and is controlled by the [animateColor] flag. /// Animating [color] is optional and is controlled by the [animateColor] flag.
/// ///
...@@ -1492,6 +1492,8 @@ class AnimatedPhysicalModel extends ImplicitlyAnimatedWidget { ...@@ -1492,6 +1492,8 @@ class AnimatedPhysicalModel extends ImplicitlyAnimatedWidget {
final BoxShape shape; final BoxShape shape;
/// {@macro flutter.widgets.Clip} /// {@macro flutter.widgets.Clip}
///
/// Defaults to [Clip.none].
final Clip clipBehavior; final Clip clipBehavior;
/// The target border radius of the rounded corners for a rectangle shape. /// The target border radius of the rounded corners for a rectangle shape.
......
...@@ -2,6 +2,7 @@ ...@@ -2,6 +2,7 @@
// Use of this source code is governed by a BSD-style license that can be // Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file. // found in the LICENSE file.
import 'package:flutter/painting.dart';
import 'package:flutter/rendering.dart'; import 'package:flutter/rendering.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter/widgets.dart'; import 'package:flutter/widgets.dart';
...@@ -171,4 +172,19 @@ void main() { ...@@ -171,4 +172,19 @@ void main() {
await tester.pumpWidget(const Card(clipBehavior: Clip.antiAlias)); await tester.pumpWidget(const Card(clipBehavior: Clip.antiAlias));
expect(tester.widget<Material>(find.byType(Material)).clipBehavior, Clip.antiAlias); expect(tester.widget<Material>(find.byType(Material)).clipBehavior, Clip.antiAlias);
}); });
testWidgets('Card clipBehavior property defers to theme when null', (WidgetTester tester) async {
await tester.pumpWidget(Builder(builder: (BuildContext context) {
final ThemeData themeData = Theme.of(context);
return Theme(
data: themeData.copyWith(
cardTheme: themeData.cardTheme.copyWith(
clipBehavior: Clip.antiAliasWithSaveLayer,
),
),
child: const Card(clipBehavior: null),
);
}));
expect(tester.widget<Material>(find.byType(Material)).clipBehavior, Clip.antiAliasWithSaveLayer);
});
} }
...@@ -300,6 +300,22 @@ void main() { ...@@ -300,6 +300,22 @@ void main() {
expect(find.byKey(materialKey), hasNoImmediateClip); expect(find.byKey(materialKey), hasNoImmediateClip);
}); });
testWidgets('Null clipBehavior asserts', (WidgetTester tester) async {
final GlobalKey materialKey = GlobalKey();
Future<void> doPump() async {
await tester.pumpWidget(
Material(
key: materialKey,
type: MaterialType.transparency,
child: const SizedBox(width: 100.0, height: 100.0),
clipBehavior: null,
)
);
}
expect(() async => doPump(), throwsAssertionError);
});
testWidgets('clips to bounding rect by default given Clip.antiAlias', (WidgetTester tester) async { testWidgets('clips to bounding rect by default given Clip.antiAlias', (WidgetTester tester) async {
final GlobalKey materialKey = GlobalKey(); final GlobalKey materialKey = GlobalKey();
await tester.pumpWidget( await tester.pumpWidget(
......
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