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

import 'dart:math' as math;

import 'package:flutter/widgets.dart';

import 'ink_well.dart';
import 'material.dart';

12 13 14 15 16
const Duration _kUnconfirmedRippleDuration = Duration(seconds: 1);
const Duration _kFadeInDuration = Duration(milliseconds: 75);
const Duration _kRadiusDuration = Duration(milliseconds: 225);
const Duration _kFadeOutDuration = Duration(milliseconds: 375);
const Duration _kCancelDuration = Duration(milliseconds: 75);
17

18 19
// The fade out begins 225ms after the _fadeOutController starts. See confirm().
const double _kFadeOutIntervalStart = 225.0 / 375.0;
20

21
RectCallback? _getClipCallback(RenderBox referenceBox, bool containedInkWell, RectCallback? rectCallback) {
22 23 24 25 26 27 28 29 30
  if (rectCallback != null) {
    assert(containedInkWell);
    return rectCallback;
  }
  if (containedInkWell)
    return () => Offset.zero & referenceBox.size;
  return null;
}

31
double _getTargetRadius(RenderBox referenceBox, bool containedInkWell, RectCallback? rectCallback, Offset position) {
32 33 34 35
  final Size size = rectCallback != null ? rectCallback().size : referenceBox.size;
  final double d1 = size.bottomRight(Offset.zero).distance;
  final double d2 = (size.topRight(Offset.zero) - size.bottomLeft(Offset.zero)).distance;
  return math.max(d1, d2) / 2.0;
36 37 38 39 40 41 42
}

class _InkRippleFactory extends InteractiveInkFeatureFactory {
  const _InkRippleFactory();

  @override
  InteractiveInkFeature create({
43 44 45 46 47
    required MaterialInkController controller,
    required RenderBox referenceBox,
    required Offset position,
    required Color color,
    required TextDirection textDirection,
48
    bool containedInkWell = false,
49 50 51 52 53
    RectCallback? rectCallback,
    BorderRadius? borderRadius,
    ShapeBorder? customBorder,
    double? radius,
    VoidCallback? onRemoved,
54
  }) {
55
    return InkRipple(
56 57 58 59 60 61 62
      controller: controller,
      referenceBox: referenceBox,
      position: position,
      color: color,
      containedInkWell: containedInkWell,
      rectCallback: rectCallback,
      borderRadius: borderRadius,
63
      customBorder: customBorder,
64 65
      radius: radius,
      onRemoved: onRemoved,
66
      textDirection: textDirection,
67 68 69 70 71 72 73 74 75 76 77 78 79
    );
  }
}

/// A visual reaction on a piece of [Material] to user input.
///
/// A circular ink feature whose origin starts at the input touch point and
/// whose radius expands from 60% of the final radius. The splash origin
/// animates to the center of its [referenceBox].
///
/// This object is rarely created directly. Instead of creating an ink ripple,
/// consider using an [InkResponse] or [InkWell] widget, which uses
/// gestures (such as tap and long-press) to trigger ink splashes. This class
80
/// is used when the [Theme]'s [ThemeData.splashFactory] is [InkRipple.splashFactory].
81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109
///
/// See also:
///
///  * [InkSplash], which is an ink splash feature that expands less
///    aggressively than the ripple.
///  * [InkResponse], which uses gestures to trigger ink highlights and ink
///    splashes in the parent [Material].
///  * [InkWell], which is a rectangular [InkResponse] (the most common type of
///    ink response).
///  * [Material], which is the widget on which the ink splash is painted.
///  * [InkHighlight], which is an ink feature that emphasizes a part of a
///    [Material].
class InkRipple extends InteractiveInkFeature {
  /// Begin a ripple, centered at [position] relative to [referenceBox].
  ///
  /// The [controller] argument is typically obtained via
  /// `Material.of(context)`.
  ///
  /// If [containedInkWell] is true, then the ripple will be sized to fit
  /// the well rectangle, then clipped to it when drawn. The well
  /// rectangle is the box returned by [rectCallback], if provided, or
  /// otherwise is the bounds of the [referenceBox].
  ///
  /// If [containedInkWell] is false, then [rectCallback] should be null.
  /// The ink ripple is clipped only to the edges of the [Material].
  /// This is the default.
  ///
  /// When the ripple is removed, [onRemoved] will be called.
  InkRipple({
110 111 112 113 114
    required MaterialInkController controller,
    required RenderBox referenceBox,
    required Offset position,
    required Color color,
    required TextDirection textDirection,
115
    bool containedInkWell = false,
116 117 118 119 120
    RectCallback? rectCallback,
    BorderRadius? borderRadius,
    ShapeBorder? customBorder,
    double? radius,
    VoidCallback? onRemoved,
121 122
  }) : assert(color != null),
       assert(position != null),
123
       assert(textDirection != null),
124 125
       _position = position,
       _borderRadius = borderRadius ?? BorderRadius.zero,
126
       _customBorder = customBorder,
127
       _textDirection = textDirection,
128 129
       _targetRadius = radius ?? _getTargetRadius(referenceBox, containedInkWell, rectCallback, position),
       _clipCallback = _getClipCallback(referenceBox, containedInkWell, rectCallback),
130
       super(controller: controller, referenceBox: referenceBox, color: color, onRemoved: onRemoved) {
131 132 133
    assert(_borderRadius != null);

    // Immediately begin fading-in the initial splash.
134
    _fadeInController = AnimationController(duration: _kFadeInDuration, vsync: controller.vsync)
135 136
      ..addListener(controller.markNeedsPaint)
      ..forward();
137
    _fadeIn = _fadeInController.drive(IntTween(
138 139
      begin: 0,
      end: color.alpha,
140
    ));
141 142

    // Controls the splash radius and its center. Starts upon confirm.
143
    _radiusController = AnimationController(duration: _kUnconfirmedRippleDuration, vsync: controller.vsync)
144 145
      ..addListener(controller.markNeedsPaint)
      ..forward();
Josh Soref's avatar
Josh Soref committed
146
     // Initial splash diameter is 60% of the target diameter, final
147
     // diameter is 10dps larger than the target diameter.
148 149 150 151 152
    _radius = _radiusController.drive(
      Tween<double>(
        begin: _targetRadius * 0.30,
        end: _targetRadius + 5.0,
      ).chain(_easeCurveTween),
153 154 155 156
    );

    // Controls the splash radius and its center. Starts upon confirm however its
    // Interval delays changes until the radius expansion has completed.
157
    _fadeOutController = AnimationController(duration: _kFadeOutDuration, vsync: controller.vsync)
158 159
      ..addListener(controller.markNeedsPaint)
      ..addStatusListener(_handleAlphaStatusChanged);
160 161 162 163 164
    _fadeOut = _fadeOutController.drive(
      IntTween(
        begin: color.alpha,
        end: 0,
      ).chain(_fadeOutIntervalTween),
165 166 167 168 169 170 171
    );

    controller.addInkFeature(this);
  }

  final Offset _position;
  final BorderRadius _borderRadius;
172
  final ShapeBorder? _customBorder;
173
  final double _targetRadius;
174
  final RectCallback? _clipCallback;
175
  final TextDirection _textDirection;
176

177 178
  late Animation<double> _radius;
  late AnimationController _radiusController;
179

180 181
  late Animation<int> _fadeIn;
  late AnimationController _fadeInController;
182

183 184
  late Animation<int> _fadeOut;
  late AnimationController _fadeOutController;
185

186 187
  /// Used to specify this type of ink splash for an [InkWell], [InkResponse],
  /// material [Theme], or [ButtonStyle].
188 189 190 191 192
  static const InteractiveInkFeatureFactory splashFactory = _InkRippleFactory();

  static final Animatable<double> _easeCurveTween = CurveTween(curve: Curves.ease);
  static final Animatable<double> _fadeOutIntervalTween = CurveTween(curve: const Interval(_kFadeOutIntervalStart, 1.0));

193 194 195 196 197
  @override
  void confirm() {
    _radiusController
      ..duration = _kRadiusDuration
      ..forward();
Josh Soref's avatar
Josh Soref committed
198
    // This confirm may have been preceded by a cancel.
199
    _fadeInController.forward();
200
    _fadeOutController.animateTo(1.0, duration: _kFadeOutDuration);
201 202 203 204 205
  }

  @override
  void cancel() {
    _fadeInController.stop();
206 207
    // Watch out: setting _fadeOutController's value to 1.0 will
    // trigger a call to _handleAlphaStatusChanged() which will
208
    // dispose _fadeOutController.
209 210 211 212
    final double fadeOutValue = 1.0 - _fadeInController.value;
    _fadeOutController.value = fadeOutValue;
    if (fadeOutValue < 1.0)
      _fadeOutController.animateTo(1.0, duration: _kCancelDuration);
213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230
  }

  void _handleAlphaStatusChanged(AnimationStatus status) {
    if (status == AnimationStatus.completed)
      dispose();
  }

  @override
  void dispose() {
    _radiusController.dispose();
    _fadeInController.dispose();
    _fadeOutController.dispose();
    super.dispose();
  }

  @override
  void paintFeature(Canvas canvas, Matrix4 transform) {
    final int alpha = _fadeInController.isAnimating ? _fadeIn.value : _fadeOut.value;
231
    final Paint paint = Paint()..color = color.withAlpha(alpha);
232 233 234 235 236
    // Splash moves to the center of the reference box.
    final Offset center = Offset.lerp(
      _position,
      referenceBox.size.center(Offset.zero),
      Curves.ease.transform(_radiusController.value),
237
    )!;
238 239 240 241 242 243 244 245 246 247 248
    paintInkCircle(
      canvas: canvas,
      transform: transform,
      paint: paint,
      center: center,
      textDirection: _textDirection,
      radius: _radius.value,
      customBorder: _customBorder,
      borderRadius: _borderRadius,
      clipCallback: _clipCallback,
    );
249 250
  }
}