最近作一個省市區選擇的控件,產品的需求則是參考某銀行的省市區選擇的交互,是一個TabPicker的交互的控件git
自己Flutter自帶有一個CupertinoPicker能夠實現三級聯動選擇github
後面根據TabPicker交互,發現實現也不難,使用TabBar+TabBarView+ListView 就能夠實現對應的交互功能bash
主要是根據city_pickers 這個開源項目作的修改app
如下是實現的主要代碼ui
///構建底部彈出框
Widget _bottomBuild() {
if (provinceCityAreaList == null) {
return Container();
}
return Scaffold(
appBar: PreferredSize(
child: Container(
color: Colors.white,
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
mainAxisAlignment: MainAxisAlignment.start,
children: <Widget>[
_buildHeaderTitle(),
_buildHeaderTabBar(),
Container(
height: 0.5,
margin: EdgeInsets.symmetric(horizontal: 16),
color: Color(0xFFEBEBEB),
)
],
),
),
preferredSize: Size.fromHeight(84)),
body: TabBarView(
controller: _tabController,
children: _buildTabBarView(),
),
);
複製代碼
構建總體的彈框界面,使用PreferredSize來構建頂部的title和TabBar,body則是對應的TabBarView,TabBar和TabBarView則是經過同一個_tabController來聯動的this
///構建Header tabBar
List<Widget> _buildTabBarView() {
List<Widget> tabList = [];
tabList.add(_MyCityPicker(
key: Key('province'),
isShow: widget.showType.contain(ShowType.p),
height: widget.height,
controller: provinceController,
value: targetProvince.label,
itemList: provinceCityAreaList.map((v) {
return LabelSelectBean(
label: v.label, select: v.select == null ? false : v.select);
}).toList(),
changed: (index) {
_onProvinceChange(index);
Future.delayed(Duration(milliseconds: 200))
.then((value) => _tabController.animateTo(1));
},
));
if (_provinceName != null) {
tabList.add(_MyCityPicker(
key: Key('citys $targetProvince'),
// 這個屬性是爲了強制刷新
isShow: widget.showType.contain(ShowType.c),
controller: cityController,
height: widget.height,
value: targetCity == null ? null : targetCity.label,
itemList: getCityItemList(),
changed: (index) {
_onCityChange(index);
},
));
}
if (_cityName != null && widget.showType == ShowType.pca) {
tabList.add(_MyCityPicker(
key: Key('towns $targetCity'),
isShow: widget.showType.contain(ShowType.a),
controller: areaController,
value: targetArea == null ? null : targetArea.label,
height: widget.height,
itemList: getAreaItemList(),
changed: (index) {
_areaName = targetCity.children[index].label;
_onAreaChange(index);
},
));
}
return tabList;
}
複製代碼
經過添加不一樣的CityPicker,來動態的展現對應的TabBarView。spa
這裏有個坑,動態添加TabBar和TabBarView的時候須要從新賦值給_tabController,以及動態添加TabBarView,二者要長度匹配才行,主要的實如今各自的選擇回調方法code
setState(() {
_tabLabels = [targetProvince.label, "請選擇"];
_provinceName = targetProvince.label;
_tabController =
TabController(length: _tabLabels.length, vsync: this);
});
複製代碼
每次改變TabController都須要動態設置一次ip
TabBar展現主要是展現List列表便可ci
_MyCityPicker主要的builder代碼
Container(
color: Colors.white,
child: ListView.builder(
itemBuilder: (context, index) {
return Material(
color: Colors.white,
child: InkWell(
child: Container(
padding: EdgeInsets.only(left: 16, top: 8, bottom: 8),
child: Row(
children: <Widget>[
Text(
'${widget.itemList[index].label} ',
maxLines: 1,
style: TextStyle(color: Colors.black, fontSize: 16),
overflow: TextOverflow.ellipsis,
),
widget.itemList[index].select
? Icon(
Icons.check,
size: 20,
color: Color(0xFFD71718),
)
: Container()
],
)),
onTap: () {
widget.changed(index);
},
));
},
itemCount: widget.itemList.length),
);
複製代碼
就是一個簡單的ListBuilder
選擇完數據以後須要,再一次點擊的時候,須要回顯上一次的數據,邏輯也比較簡單
void _initLocation(String locationCode) {
Result dataResult = widget.initDataResult;
if (dataResult != null && dataResult.provinceName != null) {
{
CityPoint point = provinceCityAreaList.firstWhere(
(province) => dataResult.provinceName == province.label,
orElse: () => null);
if (point != null) {
targetProvince = point;
targetProvince.select = true;
_provinceName = targetProvince.label;
} else {
targetProvince = provinceCityAreaList[0];
}
}
{
CityPoint point = targetProvince.children.firstWhere(
(city) => dataResult.cityName == city.label,
orElse: () => null);
if (point != null) {
targetCity = point;
targetCity.select = true;
_cityName = targetCity.label;
} else {
targetCity = targetProvince.children[0];
}
}
{
CityPoint point = targetCity.children.firstWhere(
(area) => dataResult.areaName == area.label,
orElse: () => null);
if (point != null) {
targetArea = point;
targetArea.select = true;
_areaName = targetArea.label;
} else {
targetArea = targetCity.children[0];
}
}
} else {
targetProvince = provinceCityAreaList[0];
targetCity = targetProvince.children[0];
targetArea = targetCity.children[0];
}
}
複製代碼
須要把回顯的數據經過Result結果返回,而後遍歷對應的List數據,設置便可
主要代碼能夠經過 fluttertabpicker 查看具體的代碼邏輯