Flutter Hero動(dòng)畫案例

Hero 指的是可以在路由(頁面)之間“飛行”的 widget,簡單來說 Hero 動(dòng)畫就是在路由切換時(shí),有一個(gè)共享的 widget 可以在新舊路由間切換笑旺。由于共享的 widget 在新舊路由頁面上的位置量承、外觀可能有所差異,所以在路由切換時(shí)會(huì)從舊路逐漸過渡到新路由中的指定位置鞋诗,這樣就會(huì)產(chǎn)生一個(gè) Hero 動(dòng)畫膀捷。

實(shí)現(xiàn)一個(gè)簡單的 Hero 動(dòng)畫

 Container(
                      height: 100,
                      alignment: Alignment.center,
                      child: InkWell(
                        child: Hero(
                            tag: Tag,
                            child: ClipOval(
                              child: Image.asset(
                                "assets/datas/night.jpg",
                                width: 80,
                                height: 80,
                                fit: BoxFit.cover,
                              ),
                            )),
                        onTap: () {
                      Navigator.of(context).push(
                          new MaterialPageRoute(builder: (BuildContext context) {
                           return new HeroAnimationPage1();
                       }));
                        },
                      ),
                    ),
  • HeroAnimationPage1
class HeroAnimationPage1 extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
        appBar: AppBar(
          title: Text("Hero1"),
        ),
        body: Center(
          child: Container(
            margin: EdgeInsets.all(20),
            child: Hero(
              tag: Tag, //唯一標(biāo)記,前后兩個(gè)路由頁Hero的tag必須相同
              child: ClipOval(
                child: Image.asset(
                  "assets/datas/night.jpg",
                  width: 120,
                  height: 120,
                  fit: BoxFit.cover,
                ),
              ),
            ),
          ),
        ));
  }
}

效果圖


可以看到削彬,實(shí)現(xiàn) Hero 動(dòng)畫只需要用 Hero 組件將要共享的 widget 包裝起來全庸,并提供一個(gè)相同的 tag 即可,中間的過渡幀都是 Flutter Framework 自動(dòng)完成的融痛。必須要注意壶笼, 前后路由頁的共享 Hero 的 tag 必須是相同的,F(xiàn)lutter Framework 內(nèi)部正是通過 tag 來確定新舊路由頁 widget 的對(duì)應(yīng)關(guān)系的雁刷。

但我們用的是 MaterialPageRoute 這個(gè)系統(tǒng)給我們提供好的路由覆劈,這個(gè)路由能讓我們?cè)?Android 或者 Ios 上呈現(xiàn)相應(yīng)的頁面跳轉(zhuǎn)效果,但在這里和 Hero 合起來看有點(diǎn)雜亂,別扭墩崩,我們稍微改一下:

   Container(
                      height: 100,
                      alignment: Alignment.center,
                      child: InkWell(
                        child: Hero(
                            tag: Tag,
                            child: ClipOval(
                              child: Image.asset(
                                "assets/datas/night.jpg",
                                width: 80,
                                height: 80,
                                fit: BoxFit.cover,
                              ),
                            )),
                        onTap: () {
//                    Navigator.of(context).push(
//                        new MaterialPageRoute(builder: (BuildContext context) {
//                          return new HeroAnimationPage1();
//                        }));

                          Navigator.push(context, PageRouteBuilder(pageBuilder:
                              (BuildContext context, Animation animation,
                                  Animation secondaryAnimation) {
                            return new FadeTransition(
                              opacity: animation,
                              child: Scaffold(
                                appBar: AppBar(
                                  title: Text("Fade"),
                                ),
                                body: HeroAnimationRouteWithFade(),
                              ),
                            );
                          }));
                        },
                      ),
                    ),
  • HeroAnimationRouteWithFade
class HeroAnimationRouteWithFade extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Center(
      child: Container(
        margin: EdgeInsets.all(20),
        child: Hero(
          tag: "tag", //唯一標(biāo)記氓英,前后兩個(gè)路由頁Hero的tag必須相同
          child: ClipOval(
            child: Image.asset(
              "assets/datas/night.jpg",
              width: 120,
              height: 120,
              fit: BoxFit.cover,
            ),
          ),
        ),
      ),
    );
  }
}

效果圖

通過 PageRouteBuilder 自定義自己的路由器,動(dòng)畫看起來明顯干凈了很多鹦筹。

不過細(xì)心的我發(fā)現(xiàn)铝阐,在運(yùn)動(dòng)(飛翔)過程中,圖片在慢慢變大的同時(shí)好像也稍微有點(diǎn)變形铐拐。這其實(shí)我也不知道為啥徘键,后來我從 Flutter 中文官網(wǎng)看到 MaterialRectCenterArcTween 這個(gè)類。這個(gè)類居然能讓控件在運(yùn)動(dòng)過程中圓角不變形遍蟋。

我們先回頭看看 Hero 的構(gòu)造函數(shù):

  const Hero({
    Key key,
    @required this.tag,
    this.createRectTween,
    this.flightShuttleBuilder,
    this.placeholderBuilder,
    this.transitionOnUserGestures = false,
    @required this.child,
  }) : assert(tag != null),
       assert(transitionOnUserGestures != null),
       assert(child != null),
       super(key: key);
  • tag:[必須]用于關(guān)聯(lián)兩個(gè) Hero 動(dòng)畫的標(biāo)識(shí),前后兩個(gè)路由頁 Hero 的 tag 必須相同.

  • createRectTween:[可選]定義目標(biāo) Hero 的邊界吹害,在從起始位置到目的位置的運(yùn)動(dòng)過程中該如何變化。

  • child:[必須]定義動(dòng)畫所呈現(xiàn)的 widget虚青。

MaterialRectCenterArcTween 這個(gè)類正好能給我們返回 createRectTween 參數(shù)所需的實(shí)例它呀。

我們?cè)賮砜匆粋€(gè)稍微高大上的 Hero 動(dòng)畫,其實(shí)我覺得沒啥改動(dòng)棒厘,只是界面做的稍微復(fù)雜一點(diǎn)纵穿,添加了 createRectTween。

實(shí)現(xiàn)一個(gè)復(fù)雜的 Hero 動(dòng)畫

  • 這里提一下 timeDilation = 3; 這是讓過渡動(dòng)畫時(shí)間稍微慢一點(diǎn)奢人,默認(rèn)為 1.
timeDilation =3;//動(dòng)畫過渡時(shí)間

其實(shí)第二個(gè)案例就只是添加了 createRectTween谓媒,你可以試著注釋 createRectTween;或者案例 2 中直接注釋 Hero 控件何乎;看看它們的動(dòng)畫效果是啥句惯,想來你印象會(huì)更深。

有興趣的小伙伴關(guān)注公眾號(hào)支救,和我一起學(xué)習(xí)吧抢野,本文代碼在文末。


全部代碼

import 'package:flutter/material.dart';
import 'package:flutter_travel/widgets/item_widget.dart';

import 'package:flutter/scheduler.dart' show timeDilation;
import 'dart:math' as math;

const Tag = "tag"; ////唯一標(biāo)記搂妻,前后兩個(gè)路由頁Hero的tag必須相同

class HeroPage extends StatefulWidget {
  @override
  _HeroPageState createState() => _HeroPageState();
}

class _HeroPageState extends State<HeroPage> {
  @override
  void initState() {
    // TODO: implement initState
    super.initState();
  }

  @override
  Widget build(BuildContext context) {
    timeDilation = 3; //動(dòng)畫過渡時(shí)間
    return Scaffold(
      appBar: AppBar(
        title: Text("Hero"),
      ),
      body: Container(
        margin: EdgeInsets.only(top: 30, right: 5, left: 5),
        child: Column(
          children: <Widget>[
            Expanded(
                flex: 1,
                child: Column(
                  mainAxisAlignment: MainAxisAlignment.center,
                  crossAxisAlignment: CrossAxisAlignment.center,
                  children: <Widget>[
                    Container(
                      height: 100,
                      alignment: Alignment.center,
                      child: InkWell(
                        child: Hero(
                            tag: Tag,
                            child: ClipOval(
                              child: Image.asset(
                                "assets/datas/night.jpg",
                                width: 80,
                                height: 80,
                                fit: BoxFit.cover,
                              ),
                            )),
                        onTap: () {
//                    Navigator.of(context).push(
//                        new MaterialPageRoute(builder: (BuildContext context) {
//                          return new HeroAnimationPage1();
//                        }));

                          Navigator.push(context, PageRouteBuilder(pageBuilder:
                              (BuildContext context, Animation animation,
                                  Animation secondaryAnimation) {
                            return new FadeTransition(
                              opacity: animation,
                              child: Scaffold(
                                appBar: AppBar(
                                  title: Text("Fade"),
                                ),
                                body: HeroAnimationRouteWithFade(),
                              ),
                            );
                          }));
                        },
                      ),
                    ),
                    Container(
                      height: 40,
                      width: double.infinity,
                      alignment: Alignment.center,
                      margin: EdgeInsets.only(top: 30),
                      child: Text(
                        "下面的案例僅僅只是添加了RectTween",
                        style: TextStyle(fontWeight: FontWeight.bold),
                      ),
                    ),
                  ],
                )),
            Expanded(
                flex: 2,
                child: Container(
                  alignment: Alignment.bottomCenter,
                  margin: EdgeInsets.only(bottom: 100),
                  child: _buildHero(context, 'assets/datas/empty.png', '空空如也'),
                ))
          ],
        ),
      ),
    );
  }

  @override
  void dispose() {
    // TODO: implement dispose
    super.dispose();
  }
}

class HeroAnimationPage1 extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
        appBar: AppBar(
          title: Text("Hero1"),
        ),
        body: Center(
          child: Container(
            margin: EdgeInsets.all(20),
            child: Hero(
              tag: Tag, //唯一標(biāo)記蒙保,前后兩個(gè)路由頁Hero的tag必須相同
              child: ClipOval(
                child: Image.asset(
                  "assets/datas/night.jpg",
                  width: 120,
                  height: 120,
                  fit: BoxFit.cover,
                ),
              ),
            ),
          ),
        ));
  }
}

class HeroAnimationRouteWithFade extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Center(
      child: Container(
        margin: EdgeInsets.all(20),
        child: Hero(
          tag: "tag", //唯一標(biāo)記,前后兩個(gè)路由頁Hero的tag必須相同
          child: ClipOval(
            child: Image.asset(
              "assets/datas/night.jpg",
              width: 120,
              height: 120,
              fit: BoxFit.cover,
            ),
          ),
        ),
      ),
    );
  }
}

///////////////////////////////////
//華麗的分割線
///////////////////////////////////

const double kMinRadius = 40.0;
const double kMaxRadius = 150.0;

class Photo extends StatelessWidget {
  Photo({Key key, this.photo, this.color, this.onTap}) : super(key: key);

  final String photo;
  final Color color;
  final VoidCallback onTap;

  Widget build(BuildContext context) {
    return Material(
      color: Colors.grey.withOpacity(0.25),
      child: InkWell(
        onTap: onTap,
        child: Image.asset(
          photo,
          color: Colors.green.withOpacity(.8),
          fit: BoxFit.contain,
        ),
      ),
    );
  }
}

RectTween _createRectTween(Rect begin, Rect end) {
  print("begin=${begin}\t end=${end}");
  return MaterialRectCenterArcTween(begin: begin, end: end);
}

Widget _buildHero(BuildContext context, String imageName, String description) {
  return Container(
    width: kMinRadius * 2.0,
    height: kMinRadius * 2.0,
    child: Hero(
      createRectTween: _createRectTween,
      tag: imageName,
      child: RadialExpansion(
        maxRadius: kMaxRadius,
        child: Photo(
          photo: imageName,
          onTap: () {
            Navigator.of(context).push(
              PageRouteBuilder<void>(
                pageBuilder: (BuildContext context, Animation<double> animation,
                    Animation<double> secondaryAnimation) {
                  return FadeTransition(
                      opacity: animation,
                      child: _showDetailPage(context, imageName, description));
                },
              ),
            );
          },
        ),
      ),
    ),
  );
}

Widget _showDetailPage(
    BuildContext context, String imageName, String description) {
  return Container(
    color: Theme.of(context).canvasColor,
    child: Center(
      child: Card(
        elevation: 8.0,
        child: Column(
          mainAxisSize: MainAxisSize.min,
          children: [
            SizedBox(
              width: kMaxRadius * 2.0,
              height: kMaxRadius * 2.0,
              child: Hero(
                createRectTween: _createRectTween,
                tag: imageName,
                child: RadialExpansion(
                  maxRadius: kMaxRadius,
                  child: Photo(
                    photo: imageName,
                    onTap: () {
                      Navigator.of(context).pop();
                    },
                  ),
                ),
              ),
            ),
            Text(
              description,
              style: TextStyle(
                fontWeight: FontWeight.bold,
              ),
              textScaleFactor: 3.0,
            ),
            const SizedBox(height: 16.0),
          ],
        ),
      ),
    ),
  );
}

class RadialExpansion extends StatelessWidget {
  RadialExpansion({
    Key key,
    this.maxRadius,
    this.child,
  })  : clipRectSize = 2.0 * (maxRadius / math.sqrt2),
        super(key: key);

  final double maxRadius;
  final clipRectSize;
  final Widget child;

  @override
  Widget build(BuildContext context) {
    return ClipOval(
      child: Center(
        child: SizedBox(
          width: clipRectSize,
          height: clipRectSize,
          child: ClipRect(
            child: child,
          ),
        ),
      ),
    );
  }
}

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末欲主,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子逝嚎,更是在濱河造成了極大的恐慌扁瓢,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,126評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件补君,死亡現(xiàn)場(chǎng)離奇詭異引几,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門伟桅,熙熙樓的掌柜王于貴愁眉苦臉地迎上來敞掘,“玉大人,你說我怎么就攤上這事楣铁【裂悖” “怎么了?”我有些...
    開封第一講書人閱讀 152,445評(píng)論 0 341
  • 文/不壞的土叔 我叫張陵盖腕,是天一觀的道長赫冬。 經(jīng)常有香客問我,道長溃列,這世上最難降的妖魔是什么劲厌? 我笑而不...
    開封第一講書人閱讀 55,185評(píng)論 1 278
  • 正文 為了忘掉前任,我火速辦了婚禮听隐,結(jié)果婚禮上补鼻,老公的妹妹穿的比我還像新娘。我一直安慰自己雅任,他們只是感情好风范,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,178評(píng)論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著椿访,像睡著了一般乌企。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上成玫,一...
    開封第一講書人閱讀 48,970評(píng)論 1 284
  • 那天加酵,我揣著相機(jī)與錄音,去河邊找鬼哭当。 笑死猪腕,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的钦勘。 我是一名探鬼主播陋葡,決...
    沈念sama閱讀 38,276評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼彻采!你這毒婦竟也來了腐缤?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,927評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤肛响,失蹤者是張志新(化名)和其女友劉穎岭粤,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體特笋,經(jīng)...
    沈念sama閱讀 43,400評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡剃浇,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,883評(píng)論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片虎囚。...
    茶點(diǎn)故事閱讀 37,997評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡角塑,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出淘讥,到底是詐尸還是另有隱情圃伶,我是刑警寧澤,帶...
    沈念sama閱讀 33,646評(píng)論 4 322
  • 正文 年R本政府宣布适揉,位于F島的核電站留攒,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏嫉嘀。R本人自食惡果不足惜炼邀,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,213評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望剪侮。 院中可真熱鬧拭宁,春花似錦、人聲如沸瓣俯。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽彩匕。三九已至腔剂,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間驼仪,已是汗流浹背掸犬。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評(píng)論 1 260
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留绪爸,地道東北人湾碎。 一個(gè)月前我還...
    沈念sama閱讀 45,423評(píng)論 2 352
  • 正文 我出身青樓,卻偏偏與公主長得像奠货,于是被迫代替她去往敵國和親介褥。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,722評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容