Flutter動(dòng)畫心動(dòng)的感覺

Flutter動(dòng)畫心動(dòng)的感覺

為了追求更好的用戶體驗(yàn)椰棘,有時(shí)候我們需要一個(gè)類似心跳一樣跳動(dòng)著的控件來吸引用戶的注意力裹匙,這是一個(gè)小小的優(yōu)化需求,但是在 Flutter 里動(dòng)畫兩件套就像裹腳布一樣臭長,所以需要像封裝一個(gè) AnimatedWidget其兴,解放生產(chǎn)力。

實(shí)現(xiàn)動(dòng)畫

混入 SingleTickerProviderStateMixin

當(dāng)創(chuàng)建一個(gè) AnimationController 時(shí)夸政,需要傳遞一個(gè)vsync參數(shù)元旬,存在vsync時(shí)會防止動(dòng)畫的UI不在當(dāng)前屏幕時(shí)消耗不必要的資源。 通過混入 SingleTickerProviderStateMixin 守问。

class _MyHomePageState extends State<MyHomePage>  with SingleTickerProviderStateMixin{}

創(chuàng)建動(dòng)畫

創(chuàng)建一個(gè)間隔將近一秒鐘的動(dòng)畫控制器:

  late final AnimationController animController;

  @override
  void initState() {
    super.initState();
    animController = AnimationController(
      duration: const Duration(milliseconds: 800),
      vsync: this,
    );
    }

心跳動(dòng)畫是從小變大匀归,再變小,所以需要一個(gè)值大小變化的動(dòng)畫:

  late final Animation<double> animation;

  @override
  void initState() {
    super.initState();
    animController = AnimationController(
      duration: const Duration(milliseconds: 800),
      vsync: this,
    );
     animation = Tween<double>(
      begin: 0.9,
      end: 1.05,
    );
    }

心跳是不間斷的耗帕,所以需要監(jiān)聽動(dòng)畫完成時(shí)恢復(fù)動(dòng)畫穆端,再繼續(xù)開始動(dòng)畫:

    animation = Tween<double>(
      begin: 0.9,
      end: 1.05,
    ).animate(animController)
      ..addListener(() {
        setState(() {});
      })
      ..addStatusListener((status) {
        if (status == AnimationStatus.completed) {
          animController.reverse();
        } else if (status == AnimationStatus.dismissed) {
          animController.forward();
        }
      });

使用縮放控件:

Transform.scale(
                scale: animation.value,
                child: const FlutterLogo(
                  size: 80,
                ),
              ),
Simulator Screen Recording - iPhone 13 - 2021-12-07 at 17.30.50.gif

為了跳動(dòng)效果,突出跳動(dòng)動(dòng)畫仿便,把縮回去的時(shí)間改短:

   animController = AnimationController(
      reverseDuration: const Duration(milliseconds: 700),
      duration: const Duration(milliseconds: 800),
      vsync: this,
    );

最后別忘了釋放資源:

  @override
  void dispose() {
    animController.dispose();
    super.dispose();
  }

抽離成小組件

為了每次用到類似的動(dòng)畫只需引入即可体啰,需要分離動(dòng)畫和顯示的組件。新建一個(gè)BounceWidget嗽仪,包含動(dòng)畫荒勇,然后可以傳入U(xiǎn)I組件:

class BounceWidget extends StatefulWidget {
  final Widget child;

  const BounceWidget({
    Key? key,
    required this.child,
  }) : super(key: key);

  @override
  State<BounceWidget> createState() => _BounceWidgetState();
}

繼續(xù)實(shí)現(xiàn)動(dòng)畫:

class _BounceWidgetState extends State<BounceWidget>
    with SingleTickerProviderStateMixin {
  late Animation<double> animation;
  late AnimationController animController;

  @override
  void initState() {
    super.initState();
    animController = AnimationController(
      reverseDuration: const Duration(milliseconds: 700),
      duration: const Duration(milliseconds: 800),
      vsync: this,
    );
    animation = Tween<double>(
      begin: 0.9,
      end: 1.05,
    ).animate(animController)
      ..addListener(() {
        setState(() {});
      })
      ..addStatusListener((status) {
        if (status == AnimationStatus.completed) {
          animController.reverse();
        } else if (status == AnimationStatus.dismissed) {
          animController.forward();
        }
      });
    animController.forward();
  }

  @override
  Widget build(BuildContext context) {
    return Transform.scale(
      scale: animation.value,
      child: widget.child,
    );
  }

  @override
  void dispose() {
    animController.dispose();
    super.dispose();
  }
}

去引入動(dòng)畫:

  Center(
              child: BounceWidget(
                child: FlutterLogo(
                  size: 80,
                ),
              ),

完整代碼:

void main() {
  runApp(const MyApp());
}

class MyApp extends StatelessWidget {
  const MyApp({Key? key}) : super(key: key);

  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: 'Flutter Demo',
      theme: ThemeData(
        primarySwatch: Colors.blue,
      ),
      home: const MyHomePage(title: 'Flutter Demo Home Page'),
    );
  }
}

class MyHomePage extends StatefulWidget {
  const MyHomePage({Key? key, required this.title}) : super(key: key);

  final String title;

  @override
  State<MyHomePage> createState() => _MyHomePageState();
}

class _MyHomePageState extends State<MyHomePage> {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text(widget.title),
      ),
      body: Padding(
        padding: const EdgeInsets.only(top: 80, left: 16),
        child: Column(
          crossAxisAlignment: CrossAxisAlignment.start,
          children: const <Widget>[
            Text(
              "心動(dòng)的",
              style: TextStyle(
                fontSize: 28,
                color: Colors.black,
              ),
            ),
            Text(
              "感覺",
              style: TextStyle(
                fontSize: 48,
                color: Colors.black,
              ),
            ),
            Center(
              child: BounceWidget(
                child: FlutterLogo(
                  size: 80,
                ),
              ),
            ),
          ],
        ),
      ),
    );
  }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市钦幔,隨后出現(xiàn)的幾起案子枕屉,更是在濱河造成了極大的恐慌,老刑警劉巖鲤氢,帶你破解...
    沈念sama閱讀 216,402評論 6 499
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件搀擂,死亡現(xiàn)場離奇詭異,居然都是意外死亡卷玉,警方通過查閱死者的電腦和手機(jī)哨颂,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,377評論 3 392
  • 文/潘曉璐 我一進(jìn)店門樱衷,熙熙樓的掌柜王于貴愁眉苦臉地迎上來绍撞,“玉大人贯被,你說我怎么就攤上這事≡茸啵” “怎么了?”我有些...
    開封第一講書人閱讀 162,483評論 0 353
  • 文/不壞的土叔 我叫張陵塑陵,是天一觀的道長哎迄。 經(jīng)常有香客問我,道長斤蔓,這世上最難降的妖魔是什么植酥? 我笑而不...
    開封第一講書人閱讀 58,165評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮弦牡,結(jié)果婚禮上友驮,老公的妹妹穿的比我還像新娘。我一直安慰自己驾锰,他們只是感情好卸留,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,176評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著椭豫,像睡著了一般耻瑟。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上捻悯,一...
    開封第一講書人閱讀 51,146評論 1 297
  • 那天匆赃,我揣著相機(jī)與錄音,去河邊找鬼今缚。 笑死算柳,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的姓言。 我是一名探鬼主播瞬项,決...
    沈念sama閱讀 40,032評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼何荚!你這毒婦竟也來了囱淋?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,896評論 0 274
  • 序言:老撾萬榮一對情侶失蹤餐塘,失蹤者是張志新(化名)和其女友劉穎妥衣,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體戒傻,經(jīng)...
    沈念sama閱讀 45,311評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡税手,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,536評論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了需纳。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片芦倒。...
    茶點(diǎn)故事閱讀 39,696評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖不翩,靈堂內(nèi)的尸體忽然破棺而出兵扬,到底是詐尸還是另有隱情麻裳,我是刑警寧澤,帶...
    沈念sama閱讀 35,413評論 5 343
  • 正文 年R本政府宣布器钟,位于F島的核電站津坑,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏傲霸。R本人自食惡果不足惜国瓮,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,008評論 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望狞谱。 院中可真熱鬧,春花似錦禁漓、人聲如沸跟衅。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽伶跷。三九已至,卻和暖如春秘狞,著一層夾襖步出監(jiān)牢的瞬間叭莫,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,815評論 1 269
  • 我被黑心中介騙來泰國打工烁试, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留雇初,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,698評論 2 368
  • 正文 我出身青樓减响,卻偏偏與公主長得像靖诗,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子支示,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,592評論 2 353

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