shape_decoration.dart 13.3 KB
Newer Older
Ian Hickson's avatar
Ian Hickson committed
1
// Copyright 2014 The Flutter Authors. All rights reserved.
2 3 4
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

5

6 7 8 9 10 11 12 13
import 'package:flutter/foundation.dart';

import 'basic_types.dart';
import 'borders.dart';
import 'box_border.dart';
import 'box_decoration.dart';
import 'box_shadow.dart';
import 'circle_border.dart';
14
import 'colors.dart';
15
import 'decoration.dart';
16
import 'decoration_image.dart';
17 18
import 'edge_insets.dart';
import 'gradient.dart';
19
import 'image_provider.dart';
20 21 22 23 24 25 26 27
import 'rounded_rectangle_border.dart';

/// An immutable description of how to paint an arbitrary shape.
///
/// The [ShapeDecoration] class provides a way to draw a [ShapeBorder],
/// optionally filling it with a color or a gradient, optionally painting an
/// image into it, and optionally casting a shadow.
///
28
/// {@tool snippet}
29 30 31 32 33 34
///
/// The following example uses the [Container] widget from the widgets layer to
/// draw a white rectangle with a 24-pixel multicolor outline, with the text
/// "RGB" inside it:
///
/// ```dart
35 36
/// Container(
///   decoration: ShapeDecoration(
37
///     color: Colors.white,
38
///     shape: Border.all(
39 40
///       color: Colors.red,
///       width: 8.0,
41
///     ) + Border.all(
42 43
///       color: Colors.green,
///       width: 8.0,
44
///     ) + Border.all(
45 46 47 48 49 50 51
///       color: Colors.blue,
///       width: 8.0,
///     ),
///   ),
///   child: const Text('RGB', textAlign: TextAlign.center),
/// )
/// ```
52
/// {@end-tool}
53 54 55 56 57
///
/// See also:
///
///  * [DecoratedBox] and [Container], widgets that can be configured with
///    [ShapeDecoration] objects.
58
///  * [BoxDecoration], a similar [Decoration] that is optimized for rectangles
59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78
///    specifically.
///  * [ShapeBorder], the base class for the objects that are used in the
///    [shape] property.
class ShapeDecoration extends Decoration {
  /// Creates a shape decoration.
  ///
  /// * If [color] is null, this decoration does not paint a background color.
  /// * If [gradient] is null, this decoration does not paint gradients.
  /// * If [image] is null, this decoration does not paint a background image.
  /// * If [shadows] is null, this decoration does not paint a shadow.
  ///
  /// The [color] and [gradient] properties are mutually exclusive, one (or
  /// both) of them must be null.
  ///
  /// The [shape] must not be null.
  const ShapeDecoration({
    this.color,
    this.image,
    this.gradient,
    this.shadows,
79
    required this.shape,
80 81 82 83 84 85 86 87 88 89 90 91 92 93
  }) : assert(!(color != null && gradient != null)),
       assert(shape != null);

  /// Creates a shape decoration configured to match a [BoxDecoration].
  ///
  /// The [BoxDecoration] class is more efficient for shapes that it can
  /// describe than the [ShapeDecoration] class is for those same shapes,
  /// because [ShapeDecoration] has to be more general as it can support any
  /// shape. However, having a [ShapeDecoration] is sometimes necessary, for
  /// example when calling [ShapeDecoration.lerp] to transition between
  /// different shapes (e.g. from a [CircleBorder] to a
  /// [RoundedRectangleBorder]; the [BoxDecoration] class cannot animate the
  /// transition from a [BoxShape.circle] to [BoxShape.rectangle]).
  factory ShapeDecoration.fromBoxDecoration(BoxDecoration source) {
94
    final ShapeBorder shape;
95 96 97 98
    assert(source.shape != null);
    switch (source.shape) {
      case BoxShape.circle:
        if (source.border != null) {
99 100
          assert(source.border!.isUniform);
          shape = CircleBorder(side: source.border!.top);
101 102 103 104 105 106
        } else {
          shape = const CircleBorder();
        }
        break;
      case BoxShape.rectangle:
        if (source.borderRadius != null) {
107
          assert(source.border == null || source.border!.isUniform);
108
          shape = RoundedRectangleBorder(
109
            side: source.border?.top ?? BorderSide.none,
110
            borderRadius: source.borderRadius!,
111 112 113 114 115 116
          );
        } else {
          shape = source.border ?? const Border();
        }
        break;
    }
117
    return ShapeDecoration(
118 119 120 121 122 123 124 125
      color: source.color,
      image: source.image,
      gradient: source.gradient,
      shadows: source.boxShadow,
      shape: shape,
    );
  }

126 127 128 129 130
  @override
  Path getClipPath(Rect rect, TextDirection textDirection) {
    return shape.getOuterPath(rect, textDirection: textDirection);
  }

131 132 133 134 135
  /// The color to fill in the background of the shape.
  ///
  /// The color is under the [image].
  ///
  /// If a [gradient] is specified, [color] must be null.
136
  final Color? color;
137 138 139 140 141 142

  /// A gradient to use when filling the shape.
  ///
  /// The gradient is under the [image].
  ///
  /// If a [color] is specified, [gradient] must be null.
143
  final Gradient? gradient;
144 145 146 147

  /// An image to paint inside the shape (clipped to its outline).
  ///
  /// The image is drawn over the [color] or [gradient].
148
  final DecorationImage? image;
149

Ian Hickson's avatar
Ian Hickson committed
150 151 152 153 154 155 156
  /// A list of shadows cast by the [shape].
  ///
  /// See also:
  ///
  ///  * [kElevationToShadow], for some predefined shadows used in Material
  ///    Design.
  ///  * [PhysicalModel], a widget for showing shadows.
157
  final List<BoxShadow>? shadows;
158 159 160 161 162 163 164 165 166

  /// The shape to fill the [color], [gradient], and [image] into and to cast as
  /// the [shadows].
  ///
  /// Shapes can be stacked (using the `+` operator). The color, gradient, and
  /// image are drawn into the inner-most shape specified.
  ///
  /// The [shape] property specifies the outline (border) of the decoration. The
  /// shape must not be null.
167 168 169 170 171 172 173 174 175 176 177 178 179 180
  ///
  /// ## Directionality-dependent shapes
  ///
  /// Some [ShapeBorder] subclasses are sensitive to the [TextDirection]. The
  /// direction that is provided to the border (e.g. for its [ShapeBorder.paint]
  /// method) is the one specified in the [ImageConfiguration]
  /// ([ImageConfiguration.textDirection]) provided to the [BoxPainter] (via its
  /// [BoxPainter.paint method). The [BoxPainter] is obtained when
  /// [createBoxPainter] is called.
  ///
  /// When a [ShapeDecoration] is used with a [Container] widget or a
  /// [DecoratedBox] widget (which is what [Container] uses), the
  /// [TextDirection] specified in the [ImageConfiguration] is obtained from the
  /// ambient [Directionality], using [createLocalImageConfiguration].
181 182 183 184 185 186
  final ShapeBorder shape;

  /// The inset space occupied by the [shape]'s border.
  ///
  /// This value may be misleading. See the discussion at [ShapeBorder.dimensions].
  @override
187
  EdgeInsetsGeometry get padding => shape.dimensions;
188 189 190 191 192

  @override
  bool get isComplex => shadows != null;

  @override
193
  ShapeDecoration? lerpFrom(Decoration? a, double t) {
194
    if (a is BoxDecoration) {
195
      return ShapeDecoration.lerp(ShapeDecoration.fromBoxDecoration(a), this, t);
196
    } else if (a == null || a is ShapeDecoration) {
197
      return ShapeDecoration.lerp(a as ShapeDecoration?, this, t);
198
    }
199
    return super.lerpFrom(a, t) as ShapeDecoration?;
200 201 202
  }

  @override
203
  ShapeDecoration? lerpTo(Decoration? b, double t) {
204
    if (b is BoxDecoration) {
205
      return ShapeDecoration.lerp(this, ShapeDecoration.fromBoxDecoration(b), t);
206
    } else if (b == null || b is ShapeDecoration) {
207
      return ShapeDecoration.lerp(this, b as ShapeDecoration?, t);
208
    }
209
    return super.lerpTo(b, t) as ShapeDecoration?;
210 211 212 213 214 215 216
  }

  /// Linearly interpolate between two shapes.
  ///
  /// Interpolates each parameter of the decoration separately.
  ///
  /// If both values are null, this returns null. Otherwise, it returns a
217
  /// non-null value, with null arguments treated like a [ShapeDecoration] whose
218 219 220
  /// fields are all null (including the [shape], which cannot normally be
  /// null).
  ///
221
  /// {@macro dart.ui.shadow.lerp}
222
  ///
223 224 225 226 227 228 229
  /// See also:
  ///
  ///  * [Decoration.lerp], which can interpolate between any two types of
  ///    [Decoration]s, not just [ShapeDecoration]s.
  ///  * [lerpFrom] and [lerpTo], which are used to implement [Decoration.lerp]
  ///    and which use [ShapeDecoration.lerp] when interpolating two
  ///    [ShapeDecoration]s or a [ShapeDecoration] to or from null.
230
  static ShapeDecoration? lerp(ShapeDecoration? a, ShapeDecoration? b, double t) {
231
    assert(t != null);
232 233 234 235 236 237 238 239
    if (a == null && b == null)
      return null;
    if (a != null && b != null) {
      if (t == 0.0)
        return a;
      if (t == 1.0)
        return b;
    }
240
    return ShapeDecoration(
241 242
      color: Color.lerp(a?.color, b?.color, t),
      gradient: Gradient.lerp(a?.gradient, b?.gradient, t),
243
      image: t < 0.5 ? a!.image : b!.image, // TODO(ianh): cross-fade the image
244
      shadows: BoxShadow.lerpList(a?.shadows, b?.shadows, t),
245
      shape: ShapeBorder.lerp(a?.shape, b?.shape, t)!,
246 247 248 249
    );
  }

  @override
250
  bool operator ==(Object other) {
251 252
    if (identical(this, other))
      return true;
253
    if (other.runtimeType != runtimeType)
254
      return false;
255 256 257 258
    return other is ShapeDecoration
        && other.color == color
        && other.gradient == gradient
        && other.image == image
259
        && listEquals<BoxShadow>(other.shadows, shadows)
260
        && other.shape == shape;
261 262 263 264 265 266 267 268 269
  }

  @override
  int get hashCode {
    return hashValues(
      color,
      gradient,
      image,
      shape,
270
      hashList(shadows),
271 272 273 274 275 276 277
    );
  }

  @override
  void debugFillProperties(DiagnosticPropertiesBuilder properties) {
    super.debugFillProperties(properties);
    properties.defaultDiagnosticsTreeStyle = DiagnosticsTreeStyle.whitespace;
278
    properties.add(ColorProperty('color', color, defaultValue: null));
279 280 281 282
    properties.add(DiagnosticsProperty<Gradient>('gradient', gradient, defaultValue: null));
    properties.add(DiagnosticsProperty<DecorationImage>('image', image, defaultValue: null));
    properties.add(IterableProperty<BoxShadow>('shadows', shadows, defaultValue: null, style: DiagnosticsTreeStyle.whitespace));
    properties.add(DiagnosticsProperty<ShapeBorder>('shape', shape));
283 284 285
  }

  @override
286
  bool hitTest(Size size, Offset position, { TextDirection? textDirection }) {
287 288 289 290
    return shape.getOuterPath(Offset.zero & size, textDirection: textDirection).contains(position);
  }

  @override
291
  _ShapeDecorationPainter createBoxPainter([ VoidCallback? onChanged ]) {
292
    assert(onChanged != null || image == null);
293
    return _ShapeDecorationPainter(this, onChanged!);
294 295 296 297 298
  }
}

/// An object that paints a [ShapeDecoration] into a canvas.
class _ShapeDecorationPainter extends BoxPainter {
299
  _ShapeDecorationPainter(this._decoration, VoidCallback onChanged)
300
    : assert(_decoration != null),
301
      super(onChanged);
302 303 304

  final ShapeDecoration _decoration;

305 306 307 308 309 310 311 312
  Rect? _lastRect;
  TextDirection? _lastTextDirection;
  late Path _outerPath;
  Path? _innerPath;
  Paint? _interiorPaint;
  int? _shadowCount;
  late List<Path> _shadowPaths;
  late List<Paint> _shadowPaints;
313

314 315 316 317
  @override
  VoidCallback get onChanged => super.onChanged!;

  void _precache(Rect rect, TextDirection? textDirection) {
318
    assert(rect != null);
319
    if (rect == _lastRect && textDirection == _lastTextDirection)
320
      return;
321

322 323 324 325 326
    // We reach here in two cases:
    //  - the very first time we paint, in which case everything except _decoration is null
    //  - subsequent times, if the rect has changed, in which case we only need to update
    //    the features that depend on the actual rect.
    if (_interiorPaint == null && (_decoration.color != null || _decoration.gradient != null)) {
327
      _interiorPaint = Paint();
328
      if (_decoration.color != null)
329
        _interiorPaint!.color = _decoration.color!;
330 331
    }
    if (_decoration.gradient != null)
332
      _interiorPaint!.shader = _decoration.gradient!.createShader(rect);
333 334
    if (_decoration.shadows != null) {
      if (_shadowCount == null) {
335 336 337 338
        _shadowCount = _decoration.shadows!.length;
        _shadowPaints = <Paint>[
          ..._decoration.shadows!.map((BoxShadow shadow) => shadow.toPaint()),
        ];
339
      }
340 341 342 343 344
      _shadowPaths = <Path>[
        ..._decoration.shadows!.map((BoxShadow shadow) {
          return _decoration.shape.getOuterPath(rect.shift(shadow.offset).inflate(shadow.spreadRadius), textDirection: textDirection);
        }),
      ];
345 346
    }
    if (_interiorPaint != null || _shadowCount != null)
347
      _outerPath = _decoration.shape.getOuterPath(rect, textDirection: textDirection);
348
    if (_decoration.image != null)
349
      _innerPath = _decoration.shape.getInnerPath(rect, textDirection: textDirection);
350

351
    _lastRect = rect;
352
    _lastTextDirection = textDirection;
353 354 355 356
  }

  void _paintShadows(Canvas canvas) {
    if (_shadowCount != null) {
357
      for (int index = 0; index < _shadowCount!; index += 1)
358 359 360 361 362 363
        canvas.drawPath(_shadowPaths[index], _shadowPaints[index]);
    }
  }

  void _paintInterior(Canvas canvas) {
    if (_interiorPaint != null)
364
      canvas.drawPath(_outerPath, _interiorPaint!);
365 366
  }

367
  DecorationImagePainter? _imagePainter;
368
  void _paintImage(Canvas canvas, ImageConfiguration configuration) {
369
    if (_decoration.image == null)
370
      return;
371 372
    _imagePainter ??= _decoration.image!.createPainter(onChanged);
    _imagePainter!.paint(canvas, _lastRect!, _innerPath, configuration);
373 374 375 376
  }

  @override
  void dispose() {
377
    _imagePainter?.dispose();
378 379 380 381 382 383 384
    super.dispose();
  }

  @override
  void paint(Canvas canvas, Offset offset, ImageConfiguration configuration) {
    assert(configuration != null);
    assert(configuration.size != null);
385 386
    final Rect rect = offset & configuration.size!;
    final TextDirection? textDirection = configuration.textDirection;
387
    _precache(rect, textDirection);
388 389 390
    _paintShadows(canvas);
    _paintInterior(canvas);
    _paintImage(canvas, configuration);
391
    _decoration.shape.paint(canvas, rect, textDirection: textDirection);
392 393
  }
}