flutter_logo.dart 18 KB
Newer Older
1 2 3 4 5 6
// Copyright 2016 The Chromium Authors. All rights reserved.
// 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 'dart:typed_data';
7
import 'dart:ui' as ui show Gradient, TextBox, lerpDouble;
8

9
import 'package:flutter/foundation.dart';
10

11
import 'alignment.dart';
12
import 'basic_types.dart';
13
import 'box_fit.dart';
14
import 'decoration.dart';
15
import 'edge_insets.dart';
16
import 'image_provider.dart';
17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38
import 'text_painter.dart';
import 'text_span.dart';
import 'text_style.dart';

/// Possible ways to draw Flutter's logo.
enum FlutterLogoStyle {
  /// Show only Flutter's logo, not the "Flutter" label.
  ///
  /// This is the default behavior for [FlutterLogoDecoration] objects.
  markOnly,

  /// Show Flutter's logo on the left, and the "Flutter" label to its right.
  horizontal,

  /// Show Flutter's logo above the "Flutter" label.
  stacked,
}

/// An immutable description of how to paint Flutter's logo.
class FlutterLogoDecoration extends Decoration {
  /// Creates a decoration that knows how to paint Flutter's logo.
  ///
39 40 41
  /// The [lightColor] and [darkColor] are used to fill the logo. The [style]
  /// controls whether and where to draw the "Flutter" label. If one is shown,
  /// the [textColor] controls the color of the label.
42
  ///
43 44
  /// The [lightColor], [darkColor], [textColor], [style], and [margin]
  /// arguments must not be null.
45
  const FlutterLogoDecoration({
46 47 48 49 50
    this.lightColor = const Color(0xFF42A5F5), // Colors.blue[400]
    this.darkColor = const Color(0xFF0D47A1), // Colors.blue[900]
    this.textColor = const Color(0xFF616161),
    this.style = FlutterLogoStyle.markOnly,
    this.margin = EdgeInsets.zero,
51 52 53 54 55
  }) : assert(lightColor != null),
       assert(darkColor != null),
       assert(textColor != null),
       assert(style != null),
       assert(margin != null),
56
       _position = identical(style, FlutterLogoStyle.markOnly) ? 0.0 : identical(style, FlutterLogoStyle.horizontal) ? 1.0 : -1.0,
57 58 59
       // (see https://github.com/dart-lang/sdk/issues/26980 for details about that ignore statement)
       _opacity = 1.0;

60
  const FlutterLogoDecoration._(this.lightColor, this.darkColor, this.textColor, this.style, this.margin, this._position, this._opacity);
61

62
  /// The lighter of the two colors used to paint the logo.
63
  ///
64
  /// If possible, the default should be used. It corresponds to the 400 and 900
65
  /// values of [material.Colors.blue] from the Material library.
66
  ///
67
  /// If for some reason that color scheme is impractical, the same entries from
68 69 70
  /// [material.Colors.amber], [material.Colors.red], or
  /// [material.Colors.indigo] colors can be used. These are Flutter's secondary
  /// colors.
71 72
  ///
  /// In extreme cases where none of those four color schemes will work,
73 74
  /// [material.Colors.pink], [material.Colors.purple], or
  /// [material.Colors.cyan] can be used. These are Flutter's tertiary colors.
75 76 77 78 79 80
  final Color lightColor;

  /// The darker of the two colors used to paint the logo.
  ///
  /// See [lightColor] for more information about selecting the logo's colors.
  final Color darkColor;
81

82 83 84 85 86 87
  /// The color used to paint the "Flutter" text on the logo, if [style] is
  /// [FlutterLogoStyle.horizontal] or [FlutterLogoStyle.stacked]. The
  /// appropriate color is `const Color(0xFF616161)` (a medium gray), against a
  /// white background.
  final Color textColor;

88 89 90 91 92 93
  /// Whether and where to draw the "Flutter" text. By default, only the logo
  /// itself is drawn.
  // This property isn't actually used when painting. It's only really used to
  // set the internal _position property.
  final FlutterLogoStyle style;

94 95 96
  /// How far to inset the logo from the edge of the container.
  final EdgeInsets margin;

97 98 99 100 101 102 103 104
  // The following are set when lerping, to represent states that can't be
  // represented by the constructor.
  final double _position; // -1.0 for stacked, 1.0 for horizontal, 0.0 for no logo
  final double _opacity; // 0.0 .. 1.0

  bool get _inTransition => _opacity != 1.0 || (_position != -1.0 && _position != 0.0 && _position != 1.0);

  @override
105
  bool debugAssertIsValid() {
106 107
    assert(lightColor != null
        && darkColor != null
108
        && textColor != null
109
        && style != null
110
        && margin != null
111 112 113 114
        && _position != null
        && _position.isFinite
        && _opacity != null
        && _opacity >= 0.0
115
        && _opacity <= 1.0);
116 117 118 119 120 121 122 123 124 125
    return true;
  }

  @override
  bool get isComplex => !_inTransition;

  /// Linearly interpolate between two Flutter logo descriptions.
  ///
  /// Interpolates both the color and the style in a continuous fashion.
  ///
126 127
  /// If both values are null, this returns null. Otherwise, it returns a
  /// non-null value. If one of the values is null, then the result is obtained
128
  /// by scaling the other value's opacity and [margin].
129
  ///
130
  /// {@macro dart.ui.shadow.lerp}
131 132 133 134
  ///
  /// See also:
  ///
  ///  * [Decoration.lerp], which interpolates between arbitrary decorations.
135
  static FlutterLogoDecoration lerp(FlutterLogoDecoration a, FlutterLogoDecoration b, double t) {
136
    assert(t != null);
137 138
    assert(a == null || a.debugAssertIsValid());
    assert(b == null || b.debugAssertIsValid());
139 140 141
    if (a == null && b == null)
      return null;
    if (a == null) {
142
      return FlutterLogoDecoration._(
143 144
        b.lightColor,
        b.darkColor,
145
        b.textColor,
146
        b.style,
147
        b.margin * t,
148 149 150 151 152
        b._position,
        b._opacity * t.clamp(0.0, 1.0),
      );
    }
    if (b == null) {
153
      return FlutterLogoDecoration._(
154 155
        a.lightColor,
        a.darkColor,
156
        a.textColor,
157
        a.style,
158
        a.margin * t,
159 160 161 162
        a._position,
        a._opacity * (1.0 - t).clamp(0.0, 1.0),
      );
    }
163 164 165 166
    if (t == 0.0)
      return a;
    if (t == 1.0)
      return b;
167
    return FlutterLogoDecoration._(
168 169
      Color.lerp(a.lightColor, b.lightColor, t),
      Color.lerp(a.darkColor, b.darkColor, t),
170
      Color.lerp(a.textColor, b.textColor, t),
171
      t < 0.5 ? a.style : b.style,
172
      EdgeInsets.lerp(a.margin, b.margin, t),
173 174 175 176 177 178 179
      a._position + (b._position - a._position) * t,
      (a._opacity + (b._opacity - a._opacity) * t).clamp(0.0, 1.0),
    );
  }

  @override
  FlutterLogoDecoration lerpFrom(Decoration a, double t) {
180
    assert(debugAssertIsValid());
181 182 183 184 185
    if (a == null || a is FlutterLogoDecoration) {
      assert(a == null || a.debugAssertIsValid());
      return FlutterLogoDecoration.lerp(a, this, t);
    }
    return super.lerpFrom(a, t);
186 187 188 189
  }

  @override
  FlutterLogoDecoration lerpTo(Decoration b, double t) {
190
    assert(debugAssertIsValid());
191 192 193 194 195
    if (b == null || b is FlutterLogoDecoration) {
      assert(b == null || b.debugAssertIsValid());
      return FlutterLogoDecoration.lerp(this, b, t);
    }
    return super.lerpTo(b, t);
196 197 198 199
  }

  @override
  // TODO(ianh): better hit testing
200
  bool hitTest(Size size, Offset position, { TextDirection textDirection }) => true;
201 202

  @override
203
  BoxPainter createBoxPainter([ VoidCallback onChanged ]) {
204
    assert(debugAssertIsValid());
205
    return _FlutterLogoPainter(this);
206 207 208 209
  }

  @override
  bool operator ==(dynamic other) {
210
    assert(debugAssertIsValid());
211 212 213 214 215
    if (identical(this, other))
      return true;
    if (other is! FlutterLogoDecoration)
      return false;
    final FlutterLogoDecoration typedOther = other;
216 217
    return lightColor == typedOther.lightColor
        && darkColor == typedOther.darkColor
218
        && textColor == typedOther.textColor
219 220 221 222 223 224
        && _position == typedOther._position
        && _opacity == typedOther._opacity;
  }

  @override
  int get hashCode {
225
    assert(debugAssertIsValid());
226
    return hashValues(
227 228
      lightColor,
      darkColor,
229
      textColor,
230
      _position,
231
      _opacity,
232 233 234 235
    );
  }

  @override
236 237
  void debugFillProperties(DiagnosticPropertiesBuilder properties) {
    super.debugFillProperties(properties);
238 239
    properties.add(DiagnosticsNode.message('$lightColor/$darkColor on $textColor'));
    properties.add(EnumProperty<FlutterLogoStyle>('style', style));
240
    if (_inTransition)
241
      properties.add(DiagnosticsNode.message('transition ${debugFormatDouble(_position)}:${debugFormatDouble(_opacity)}'));
242 243 244 245 246 247
  }
}


/// An object that paints a [BoxDecoration] into a canvas.
class _FlutterLogoPainter extends BoxPainter {
248
  _FlutterLogoPainter(this._config)
249 250 251
      : assert(_config != null),
        assert(_config.debugAssertIsValid()),
        super(null) {
252 253 254 255 256 257 258 259 260 261 262
    _prepareText();
  }

  final FlutterLogoDecoration _config;

  // these are configured assuming a font size of 100.0.
  TextPainter _textPainter;
  Rect _textBoundingRect;

  void _prepareText() {
    const String kLabel = 'Flutter';
263 264
    _textPainter = TextPainter(
      text: TextSpan(
265
        text: kLabel,
266
        style: TextStyle(
267
          color: _config.textColor,
268 269 270
          fontFamily: 'Roboto',
          fontSize: 100.0 * 350.0 / 247.0, // 247 is the height of the F when the fontSize is 350, assuming device pixel ratio 1.0
          fontWeight: FontWeight.w300,
Ian Hickson's avatar
Ian Hickson committed
271 272 273 274
          textBaseline: TextBaseline.alphabetic,
        ),
      ),
      textDirection: TextDirection.ltr,
275 276
    );
    _textPainter.layout();
277
    final ui.TextBox textSize = _textPainter.getBoxesForSelection(const TextSelection(baseOffset: 0, extentOffset: kLabel.length)).single;
278
    _textBoundingRect = Rect.fromLTRB(textSize.left, textSize.top, textSize.right, textSize.bottom);
279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297
  }

  // This class contains a lot of magic numbers. They were derived from the
  // values in the SVG files exported from the original artwork source.

  void _paintLogo(Canvas canvas, Rect rect) {
    // Our points are in a coordinate space that's 166 pixels wide and 202 pixels high.
    // First, transform the rectangle so that our coordinate space is a square 202 pixels
    // to a side, with the top left at the origin.
    canvas.save();
    canvas.translate(rect.left, rect.top);
    canvas.scale(rect.width / 202.0, rect.height / 202.0);
    // Next, offset it some more so that the 166 horizontal pixels are centered
    // in that square (as opposed to being on the left side of it). This means
    // that if we draw in the rectangle from 0,0 to 166,202, we are drawing in
    // the center of the given rect.
    canvas.translate((202.0 - 166.0) / 2.0, 0.0);

    // Set up the styles.
298
    final Paint lightPaint = Paint()
299
      ..color = _config.lightColor.withOpacity(0.8);
300
    final Paint mediumPaint = Paint()
301
      ..color = _config.lightColor;
302
    final Paint darkPaint = Paint()
303
      ..color = _config.darkColor;
304

305
    final ui.Gradient triangleGradient = ui.Gradient.linear(
306 307
      const Offset(87.2623 + 37.9092, 28.8384 + 123.4389),
      const Offset(42.9205 + 37.9092, 35.0952 + 123.4389),
308 309 310 311 312 313 314 315 316 317 318
      <Color>[
        const Color(0xBFFFFFFF),
        const Color(0xBFFCFCFC),
        const Color(0xBFF4F4F4),
        const Color(0xBFE5E5E5),
        const Color(0xBFD1D1D1),
        const Color(0xBFB6B6B6),
        const Color(0xBF959595),
        const Color(0xBF6E6E6E),
        const Color(0xBF616161),
      ],
319
      <double>[ 0.2690, 0.4093, 0.4972, 0.5708, 0.6364, 0.6968, 0.7533, 0.8058, 0.8219 ],
320
    );
321
    final Paint trianglePaint = Paint()
322
      ..shader = triangleGradient
323
      ..blendMode = BlendMode.multiply;
324

325
    final ui.Gradient rectangleGradient = ui.Gradient.linear(
326 327
      const Offset(62.3643 + 37.9092, 40.135 + 123.4389),
      const Offset(54.0376 + 37.9092, 31.8083 + 123.4389),
328 329 330 331 332 333 334 335 336 337 338 339 340
      <Color>[
        const Color(0x80FFFFFF),
        const Color(0x80FCFCFC),
        const Color(0x80F4F4F4),
        const Color(0x80E5E5E5),
        const Color(0x80D1D1D1),
        const Color(0x80B6B6B6),
        const Color(0x80959595),
        const Color(0x806E6E6E),
        const Color(0x80616161),
      ],
      <double>[ 0.4588, 0.5509, 0.6087, 0.6570, 0.7001, 0.7397, 0.7768, 0.8113, 0.8219 ],
    );
341
    final Paint rectanglePaint = Paint()
342
      ..shader = rectangleGradient
343
      ..blendMode = BlendMode.multiply;
344 345

    // Draw the basic shape.
346
    final Path topBeam = Path()
347 348 349 350 351 352
      ..moveTo(37.7, 128.9)
      ..lineTo(9.8, 101.0)
      ..lineTo(100.4, 10.4)
      ..lineTo(156.2, 10.4);
    canvas.drawPath(topBeam, lightPaint);

353
    final Path middleBeam = Path()
354 355 356 357 358 359
      ..moveTo(156.2, 94.0)
      ..lineTo(100.4, 94.0)
      ..lineTo(79.5, 114.9)
      ..lineTo(107.4, 142.8);
    canvas.drawPath(middleBeam, lightPaint);

360
    final Path bottomBeam = Path()
361 362 363 364 365 366 367 368
      ..moveTo(79.5, 170.7)
      ..lineTo(100.4, 191.6)
      ..lineTo(156.2, 191.6)
      ..lineTo(156.2, 191.6)
      ..lineTo(107.4, 142.8);
    canvas.drawPath(bottomBeam, darkPaint);

    canvas.save();
369
    canvas.transform(Float64List.fromList(const <double>[
370 371 372 373 374 375
      // careful, this is in _column_-major order
      0.7071, -0.7071, 0.0, 0.0,
      0.7071, 0.7071, 0.0, 0.0,
      0.0, 0.0, 1.0, 0.0,
      -77.697, 98.057, 0.0, 1.0,
    ]));
Dan Field's avatar
Dan Field committed
376
    canvas.drawRect(const Rect.fromLTWH(59.8, 123.1, 39.4, 39.4), mediumPaint);
377 378 379
    canvas.restore();

    // The two gradients.
380
    final Path triangle = Path()
381 382 383 384 385
      ..moveTo(79.5, 170.7)
      ..lineTo(120.9, 156.4)
      ..lineTo(107.4, 142.8);
    canvas.drawPath(triangle, trianglePaint);

386
    final Path rectangle = Path()
387 388 389 390 391 392 393 394 395 396 397 398
      ..moveTo(107.4, 142.8)
      ..lineTo(79.5, 170.7)
      ..lineTo(86.1, 177.3)
      ..lineTo(114.0, 149.4);
    canvas.drawPath(rectangle, rectanglePaint);

    canvas.restore();
  }

  @override
  void paint(Canvas canvas, Offset offset, ImageConfiguration configuration) {
    offset += _config.margin.topLeft;
399
    final Size canvasSize = _config.margin.deflateSize(configuration.size);
400 401
    if (canvasSize.isEmpty)
      return;
402 403 404 405 406 407 408 409 410 411 412
    Size logoSize;
    if (_config._position > 0.0) {
      // horizontal style
      logoSize = const Size(820.0, 232.0);
    } else if (_config._position < 0.0) {
      // stacked style
      logoSize = const Size(252.0, 306.0);
    } else {
      // only the mark
      logoSize = const Size(202.0, 202.0);
    }
413
    final FittedSizes fittedSize = applyBoxFit(BoxFit.contain, logoSize, canvasSize);
414
    assert(fittedSize.source == logoSize);
415
    final Rect rect = Alignment.center.inscribe(fittedSize.destination, offset & canvasSize);
416
    final double centerSquareHeight = canvasSize.shortestSide;
417
    final Rect centerSquare = Rect.fromLTWH(
418 419 420
      offset.dx + (canvasSize.width - centerSquareHeight) / 2.0,
      offset.dy + (canvasSize.height - centerSquareHeight) / 2.0,
      centerSquareHeight,
421
      centerSquareHeight,
422 423 424 425 426
    );

    Rect logoTargetSquare;
    if (_config._position > 0.0) {
      // horizontal style
427
      logoTargetSquare = Rect.fromLTWH(rect.left, rect.top, rect.height, rect.height);
428 429 430
    } else if (_config._position < 0.0) {
      // stacked style
      final double logoHeight = rect.height * 191.0 / 306.0;
431
      logoTargetSquare = Rect.fromLTWH(
432 433 434
        rect.left + (rect.width - logoHeight) / 2.0,
        rect.top,
        logoHeight,
435
        logoHeight,
436 437 438 439 440 441 442 443 444 445
      );
    } else {
      // only the mark
      logoTargetSquare = centerSquare;
    }
    final Rect logoSquare = Rect.lerp(centerSquare, logoTargetSquare, _config._position.abs());

    if (_config._opacity < 1.0) {
      canvas.saveLayer(
        offset & canvasSize,
446 447
        Paint()
          ..colorFilter = ColorFilter.mode(
448
            const Color(0xFFFFFFFF).withOpacity(_config._opacity),
449
            BlendMode.modulate,
450
          ),
451 452 453 454 455 456 457 458 459 460 461 462
      );
    }
    if (_config._position != 0.0) {
      if (_config._position > 0.0) {
        // horizontal style
        final double fontSize = 2.0 / 3.0 * logoSquare.height * (1 - (10.4 * 2.0) / 202.0);
        final double scale = fontSize / 100.0;
        final double finalLeftTextPosition = // position of text in rest position
          (256.4 / 820.0) * rect.width - // 256.4 is the distance from the left edge to the left of the F when the whole logo is 820.0 wide
          (32.0 / 350.0) * fontSize; // 32 is the distance from the text bounding box edge to the left edge of the F when the font size is 350
        final double initialLeftTextPosition = // position of text when just starting the animation
          rect.width / 2.0 - _textBoundingRect.width * scale;
463
        final Offset textOffset = Offset(
464
          rect.left + ui.lerpDouble(initialLeftTextPosition, finalLeftTextPosition, _config._position),
465
          rect.top + (rect.height - _textBoundingRect.height * scale) / 2.0,
466 467 468
        );
        canvas.save();
        if (_config._position < 1.0) {
469
          final Offset center = logoSquare.center;
470
          final Path path = Path()
471 472 473
            ..moveTo(center.dx, center.dy)
            ..lineTo(center.dx + rect.width, center.dy - rect.width)
            ..lineTo(center.dx + rect.width, center.dy + rect.width)
474 475 476 477 478 479 480 481 482 483 484 485
            ..close();
          canvas.clipPath(path);
        }
        canvas.translate(textOffset.dx, textOffset.dy);
        canvas.scale(scale, scale);
        _textPainter.paint(canvas, Offset.zero);
        canvas.restore();
      } else if (_config._position < 0.0) {
        // stacked style
        final double fontSize = 0.35 * logoTargetSquare.height * (1 - (10.4 * 2.0) / 202.0);
        final double scale = fontSize / 100.0;
        if (_config._position > -1.0) {
486
          // This limits what the drawRect call below is going to blend with.
487
          canvas.saveLayer(_textBoundingRect, Paint());
488 489 490 491
        } else {
          canvas.save();
        }
        canvas.translate(
492
          logoTargetSquare.center.dx - (_textBoundingRect.width * scale / 2.0),
493
          logoTargetSquare.bottom,
494 495 496 497
        );
        canvas.scale(scale, scale);
        _textPainter.paint(canvas, Offset.zero);
        if (_config._position > -1.0) {
498
          canvas.drawRect(_textBoundingRect.inflate(_textBoundingRect.width * 0.5), Paint()
499
            ..blendMode = BlendMode.modulate
500 501 502
            ..shader = ui.Gradient.linear(
              Offset(_textBoundingRect.width * -0.5, 0.0),
              Offset(_textBoundingRect.width * 1.5, 0.0),
503 504
              <Color>[const Color(0xFFFFFFFF), const Color(0xFFFFFFFF), const Color(0x00FFFFFF), const Color(0x00FFFFFF)],
              <double>[ 0.0, math.max(0.0, _config._position.abs() - 0.1), math.min(_config._position.abs() + 0.1, 1.0), 1.0 ],
505
            ),
506 507 508 509 510 511 512 513 514 515
          );
        }
        canvas.restore();
      }
    }
    _paintLogo(canvas, logoSquare);
    if (_config._opacity < 1.0)
      canvas.restore();
  }
}