Flutter 105: 圖解自定義 ACEPageMenu 滑動菜單 (一)

??????小菜嘗試做一個類似 BottomSheet 的滑動 Menu合冀,不局限于底部,可以從屏幕四周滑出项贺;因涉及內(nèi)容較多君躺,小菜計劃拆分開來總結和完善,先介紹大體結構开缎,之后再詳細學習棕叫;


??????小菜自定義的 ACEPageMenu 滑動菜單在繪制及動畫主要涉及兩方面,小菜簡單介紹奕删;

AnimatedBuilder

??????小菜需要 Menu 從屏幕四周滑動出來俺泣,此時一定需要 Animation 動畫,而對于動畫抄伍,小菜嘗試用 AnimatedBuilder 來處理赤兴,雖然需要設置 AnimatedController 等钩杰,但對于動畫的處理相對靈活;

1. AnimationController

??????首先需要設置一個 Animation 控制器熟掂,在指定的 Duration 時長內(nèi),屏幕繪制過程中扎拣,會線性的生成 0.0-1.0 的數(shù)值用來控制動畫的開始與結束以及設置動畫的監(jiān)聽赴肚;通過 vsync 防止在屏幕外的 Animation 消耗不必要資源;

??????使用 AnimationController 時需要注意在 initState() 生命周期中進行初始化和在 dispose() 結束生命周期時進行銷毀二蓝;同時可以通過 addStatusListener() 對動畫過程進行監(jiān)聽誉券;
??????a. AnimationStatus.forward 為動畫開始時的回調(diào)監(jiān)聽,與 AnimationController.forward() 對應刊愚;
??????b. AnimationStatus.completed 為動畫執(zhí)行結束時的回調(diào)監(jiān)聽踊跟;
??????c. AnimationStatus.reverse 為動畫反向執(zhí)行時的回調(diào)監(jiān)聽,與 AnimationController.reverse() 對應鸥诽;
??????d. AnimationStatus.dismissed 為動畫反向執(zhí)行結束時的回調(diào)監(jiān)聽琴锭;

@override
void initState() {
  super.initState();
  _controller = AnimationController(
      duration: const Duration(milliseconds: 600), vsync: this);
  _controller.addStatusListener((status) {
    switch(status){
      case AnimationStatus.dismissed:
        print("Current status is dismissed !");
        break;
      case AnimationStatus.forward:
        print("Current status is forward !");
        break;
      case AnimationStatus.reverse:
        print("Current status is reverse !");
        break;
      case AnimationStatus.completed:
        print("Current status is completed !");
        break;
    }
  });
}

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

2. AnimatedBuilder

??????AnimationController 之后需要設置具體 Menu Widget 所在的 AnimatedBuilder 動畫構造器晰甚;在其中設置平移動畫,并與 AnimationController 控制器進行關聯(lián)决帖;具體的動畫相關的會在之后的博客中繼續(xù)詳細學習厕九;

return AnimatedBuilder(
    animation: _controller,
    child: Container(
        color: Color(0xF3242424),
        height: 200.0,
        width: ScreenUtils.getScreenWidth()),
    builder: (BuildContext context, Widget child) {
      return Transform.translate(offset: Offset(0, _controller.value * 50), child: child);
    });

SingleChildLayoutDelegate

??????動畫的處理基本搞定,重要的是如何讓 Widget 從屏幕四周外部開始平移地回,此時小菜嘗試用 SingleChildLayoutDelegate 來處理扁远;

??????SingleChildLayoutDelegate 是用于計算帶有單個子對象的渲染對象的布局的委托,其本身是一個抽象類刻像,需要自己實現(xiàn)對應的 Delegate 委托畅买;小菜自定義一個 ACEMenuDelegate,主要實現(xiàn)兩個方法细睡,分別為:確定要應用于子項的約束的 getConstraintsForChild() 和確定子項位置的 getPositionForChild()谷羞;

??????當提供對應的實例時,應調(diào)用 shouldRelayout()溜徙,判斷實例是否實際代表其他信息湃缎;具體的應用小菜會在之后的博客中進一步學習;

class ACEMenuDelegate extends SingleChildLayoutDelegate {
  final MenuType _menuType;
  final double _controllerValue;

  ACEMenuDelegate(this._menuType, this._controllerValue);

  @override
  BoxConstraints getConstraintsForChild(BoxConstraints constraints) {
    return BoxConstraints(
        minWidth: (_menuType == MenuType.MENU_LEFT ||
                _menuType == MenuType.MENU_RIGHT)
            ? 0
            : constraints.maxWidth,
        maxWidth: (_menuType == MenuType.MENU_LEFT ||
                _menuType == MenuType.MENU_RIGHT)
            ? ScreenUtils.getScreenWidth() * 0.75
            : constraints.maxWidth,
        minHeight: 0.0,
        maxHeight: (_menuType == MenuType.MENU_LEFT ||
                _menuType == MenuType.MENU_RIGHT)
            ? constraints.maxHeight
            : constraints.maxHeight * 0.45);
  }

  @override
  Offset getPositionForChild(Size size, Size childSize) {
    double _offsetX = Offset.zero.dx, _offsetY = Offset.zero.dy;
    switch (_menuType) {
      case MenuType.MENU_TOP:
        _offsetY = -childSize.height * (1 - _controllerValue);
        break;
      case MenuType.MENU_BOTTOM:
        _offsetY = size.height - childSize.height * _controllerValue;
        break;
      case MenuType.MENU_LEFT:
        _offsetX = -childSize.width * (1 - _controllerValue);
        break;
      case MenuType.MENU_RIGHT:
        _offsetX = size.width - childSize.width * _controllerValue;
        break;
    }
    return Offset(_offsetX, _offsetY);
  }

  @override
  bool shouldRelayout(ACEMenuDelegate oldDelegate) {
    return _controllerValue != oldDelegate._controllerValue;
  }
}

??????ACEPageMenu 源碼


??????小菜今天只是大概介紹一下功能實現(xiàn)蠢壹,對于細節(jié)部分以及手勢操作正在進一步完善嗓违,對于動畫和委托的學習會在之后進一步學習;如有錯誤图贸,請多多指導蹂季!

來源: 阿策小和尚

?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市疏日,隨后出現(xiàn)的幾起案子偿洁,更是在濱河造成了極大的恐慌,老刑警劉巖沟优,帶你破解...
    沈念sama閱讀 218,386評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件父能,死亡現(xiàn)場離奇詭異,居然都是意外死亡净神,警方通過查閱死者的電腦和手機何吝,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,142評論 3 394
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來鹃唯,“玉大人爱榕,你說我怎么就攤上這事∑禄牛” “怎么了黔酥?”我有些...
    開封第一講書人閱讀 164,704評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我跪者,道長棵帽,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,702評論 1 294
  • 正文 為了忘掉前任渣玲,我火速辦了婚禮逗概,結果婚禮上,老公的妹妹穿的比我還像新娘忘衍。我一直安慰自己逾苫,他們只是感情好,可當我...
    茶點故事閱讀 67,716評論 6 392
  • 文/花漫 我一把揭開白布枚钓。 她就那樣靜靜地躺著铅搓,像睡著了一般。 火紅的嫁衣襯著肌膚如雪搀捷。 梳的紋絲不亂的頭發(fā)上星掰,一...
    開封第一講書人閱讀 51,573評論 1 305
  • 那天,我揣著相機與錄音嫩舟,去河邊找鬼氢烘。 笑死,一個胖子當著我的面吹牛至壤,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播枢纠,決...
    沈念sama閱讀 40,314評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼像街,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了晋渺?” 一聲冷哼從身側響起镰绎,我...
    開封第一講書人閱讀 39,230評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎木西,沒想到半個月后畴栖,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,680評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡八千,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,873評論 3 336
  • 正文 我和宋清朗相戀三年吗讶,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片恋捆。...
    茶點故事閱讀 39,991評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡照皆,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出沸停,到底是詐尸還是另有隱情膜毁,我是刑警寧澤,帶...
    沈念sama閱讀 35,706評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站瘟滨,受9級特大地震影響候醒,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜杂瘸,卻給世界環(huán)境...
    茶點故事閱讀 41,329評論 3 330
  • 文/蒙蒙 一倒淫、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧胧沫,春花似錦昌简、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,910評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至南蹂,卻和暖如春犬金,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背六剥。 一陣腳步聲響...
    開封第一講書人閱讀 33,038評論 1 270
  • 我被黑心中介騙來泰國打工晚顷, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人疗疟。 一個月前我還...
    沈念sama閱讀 48,158評論 3 370
  • 正文 我出身青樓该默,卻偏偏與公主長得像,于是被迫代替她去往敵國和親策彤。 傳聞我的和親對象是個殘疾皇子栓袖,可洞房花燭夜當晚...
    茶點故事閱讀 44,941評論 2 355

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