??????小菜嘗試做一個類似 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é)部分以及手勢操作正在進一步完善嗓违,對于動畫和委托的學習會在之后進一步學習;如有錯誤图贸,請多多指導蹂季!
來源: 阿策小和尚