Commit f0f99a45 authored by Matt Perry's avatar Matt Perry Committed by GitHub

Pesto logo now smoothly animates between image and image+text states. (#5890)

Fixes https://github.com/flutter/flutter/issues/5684
parent 18212382
...@@ -2,9 +2,8 @@ ...@@ -2,9 +2,8 @@
// 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 'dart:math';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter/rendering.dart';
class PestoDemo extends StatelessWidget { class PestoDemo extends StatelessWidget {
PestoDemo({ Key key }) : super(key: key); PestoDemo({ Key key }) : super(key: key);
...@@ -51,7 +50,8 @@ class PestoStyle extends TextStyle { ...@@ -51,7 +50,8 @@ class PestoStyle extends TextStyle {
double fontSize: 12.0, double fontSize: 12.0,
FontWeight fontWeight, FontWeight fontWeight,
Color color: Colors.black87, Color color: Colors.black87,
double height double letterSpacing,
double height,
}) : super( }) : super(
inherit: false, inherit: false,
color: color, color: color,
...@@ -59,7 +59,8 @@ class PestoStyle extends TextStyle { ...@@ -59,7 +59,8 @@ class PestoStyle extends TextStyle {
fontSize: fontSize, fontSize: fontSize,
fontWeight: fontWeight, fontWeight: fontWeight,
textBaseline: TextBaseline.alphabetic, textBaseline: TextBaseline.alphabetic,
height: height letterSpacing: letterSpacing,
height: height,
); );
} }
...@@ -118,17 +119,17 @@ class _RecipeGridPageState extends State<RecipeGridPage> { ...@@ -118,17 +119,17 @@ class _RecipeGridPageState extends State<RecipeGridPage> {
builder: (BuildContext context, BoxConstraints constraints) { builder: (BuildContext context, BoxConstraints constraints) {
final Size size = constraints.biggest; final Size size = constraints.biggest;
final double appBarHeight = size.height - statusBarHeight; final double appBarHeight = size.height - statusBarHeight;
final String logo = appBarHeight >= 70.0 ? _kMediumLogoImage : _kSmallLogoImage; final double extraPadding = new Tween<double>(begin: 10.0, end: 24.0).lerp(
// Extra padding. Calculated to give about 16px on the bottom for the ((appBarHeight - kToolBarHeight) / _kAppBarHeight).clamp(0.0, 1.0)
// `small` logo at its native size, and 30px for the `medium`. );
final double extraPadding = min(0.19 * appBarHeight + 5.4, 40.0); final double logoHeight = appBarHeight - 1.5 * extraPadding;
return new Padding( return new Padding(
padding: new EdgeInsets.only( padding: new EdgeInsets.only(
top: statusBarHeight + 0.5 * extraPadding, top: statusBarHeight + 0.5 * extraPadding,
bottom: extraPadding bottom: extraPadding
), ),
child: new Center( child: new Center(
child: new Image.asset(logo, fit: ImageFit.scaleDown) child: new PestoLogo(showText: appBarHeight >= 70.0, height: logoHeight)
) )
); );
} }
...@@ -176,6 +177,94 @@ class _RecipeGridPageState extends State<RecipeGridPage> { ...@@ -176,6 +177,94 @@ class _RecipeGridPageState extends State<RecipeGridPage> {
} }
} }
class PestoLogo extends StatefulWidget {
PestoLogo({this.showText, this.height});
final bool showText;
final double height;
@override
_PestoLogoState createState() => new _PestoLogoState();
}
class _PestoLogoState extends State<PestoLogo> {
// Native sizes for logo and its image/text components.
static const double kLogoHeight = 162.0;
static const double kLogoWidth = 220.0;
static const double kImageHeight = 108.0;
static const double kTextHeight = 48.0;
final TextStyle titleStyle = const PestoStyle(fontSize: kTextHeight, fontWeight: FontWeight.w900, color: Colors.white, letterSpacing: 3.0);
final Rect textRect = new Rect.fromLTWH(0.0, kImageHeight, kLogoWidth, kTextHeight);
AnimationController _controller;
Animation<double> _textOpacity;
Animation<Rect> _imageRect;
@override
void initState() {
super.initState();
_controller = new AnimationController(duration: const Duration(milliseconds: 200))
..value = config.showText ? 1.0 : 0.0;
_textOpacity = new CurvedAnimation(parent: _controller.view, curve: const Interval(0.3, 1.0, curve: Curves.easeInOut));
_imageRect = new RectTween(
begin: new Rect.fromLTWH(0.0, 0.0, kLogoWidth, kLogoHeight),
end: new Rect.fromLTWH(0.0, 0.0, kLogoWidth, kImageHeight)
).animate(
new CurvedAnimation(parent: _controller.view, curve: const Interval(0.0, 0.9, curve: Curves.easeInOut))
);
}
@override
void dispose() {
_controller.dispose();
super.dispose();
}
@override
void didUpdateConfig(PestoLogo oldConfig) {
if (config.showText != oldConfig.showText) {
if (config.showText)
_controller.forward();
else
_controller.reverse();
}
}
@override
Widget build(BuildContext context) {
return new Transform(
transform: new Matrix4.identity()
..scale(config.height / kLogoHeight),
alignment: FractionalOffset.topCenter,
child: new SizedBox(
width: kLogoWidth,
child: new Stack(
overflow: Overflow.visible,
children: <Widget>[
new AnimatedBuilder(
animation: _imageRect,
builder: (BuildContext context, Widget child) {
return new Positioned.fromRect(
rect: _imageRect.value,
child: child
);
},
child: new Image.asset(_kSmallLogoImage, fit: ImageFit.contain)
),
new Positioned.fromRect(
rect: textRect,
child: new FadeTransition(
opacity: _textOpacity,
child: new Text('PESTO', style: titleStyle, textAlign: TextAlign.center),
)
)
]
)
)
);
}
}
// A card with the recipe's image, author, and title. // A card with the recipe's image, author, and title.
class RecipeCard extends StatelessWidget { class RecipeCard extends StatelessWidget {
final TextStyle titleStyle = const PestoStyle(fontSize: 24.0, fontWeight: FontWeight.w600); final TextStyle titleStyle = const PestoStyle(fontSize: 24.0, fontWeight: FontWeight.w600);
......
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