基础组件库
 
 
 
 
 

395 řádky
13 KiB

  1. import 'dart:convert';
  2. import 'package:flutter/cupertino.dart';
  3. import 'package:pull_to_refresh/pull_to_refresh.dart';
  4. import 'package:tab_indicator_styler/tab_indicator_styler.dart';
  5. import 'package:zhiying_base_widget/pages/custom_page/custom_item_page.dart';
  6. import 'package:zhiying_base_widget/pages/main_page/model/background_model.dart';
  7. import 'package:zhiying_base_widget/pages/main_page/notifier/main_page_bg_notifier.dart';
  8. import 'package:zhiying_base_widget/widgets/custom/search/custom_search_widget.dart';
  9. import 'package:zhiying_base_widget/widgets/empty/empty_widget.dart';
  10. import 'package:zhiying_base_widget/widgets/others/mine_header_bg_widget.dart';
  11. import 'package:zhiying_comm/zhiying_comm.dart';
  12. import 'package:flutter/material.dart';
  13. import 'package:provider/provider.dart';
  14. import 'package:flutter_bloc/flutter_bloc.dart';
  15. import 'bloc/custom_page_bloc.dart';
  16. import 'bloc/custom_page_state.dart';
  17. import 'bloc/custom_page_event.dart';
  18. import 'bloc/custom_page_repository.dart';
  19. import 'dart:ui';
  20. import 'package:fluttertoast/fluttertoast.dart';
  21. ///
  22. /// 通用模块页面
  23. ///
  24. class CustomPage extends StatefulWidget {
  25. final Map<String, dynamic> data;
  26. CustomPage(this.data, {Key key}) : super(key: key);
  27. @override
  28. _CustomPageState createState() => _CustomPageState();
  29. }
  30. class _CustomPageState extends State<CustomPage> {
  31. @override
  32. Widget build(BuildContext context) {
  33. Logger.log("数据: "+widget?.data.toString());
  34. return MultiProvider(
  35. providers: [
  36. ChangeNotifierProvider.value(value: MainPageBgNotifier()),
  37. ],
  38. child: BlocProvider<CustomPageBloc>(
  39. create: (_) => CustomPageBloc(CustomPageRepository(data: widget?.data))..add(CustomPageInitEvent()),
  40. child: _CommonPageContainer(widget?.data),
  41. // ),
  42. ),
  43. );
  44. }
  45. }
  46. class _CommonPageContainer extends StatefulWidget {
  47. final Map<String, dynamic> data;
  48. _CommonPageContainer(this.data);
  49. @override
  50. __CommonPageContainerState createState() => __CommonPageContainerState();
  51. }
  52. class __CommonPageContainerState extends State<_CommonPageContainer> with SingleTickerProviderStateMixin {
  53. TabController _tabController;
  54. // 是否有AppBar
  55. bool _isHasAppbar = false;
  56. // 是否有TabBar
  57. bool _isHasTabBar = false;
  58. /// 刷新
  59. void _onRefreshEvent() async {
  60. BlocProvider.of<CustomPageBloc>(context).add(CustomPageRefreshEvent());
  61. }
  62. @override
  63. void dispose() {
  64. _tabController?.dispose();
  65. super.dispose();
  66. }
  67. @override
  68. Widget build(BuildContext context) {
  69. return MediaQuery.removePadding(
  70. context: context,
  71. child: BlocConsumer<CustomPageBloc, CustomPageState>(
  72. listener: (context, state) {},
  73. buildWhen: (prev, current) {
  74. if (current is CustomPageErrorState) {
  75. return false;
  76. }
  77. if (current is CustomPageRefreshSuccessState) {
  78. // _refreshController.refreshCompleted(resetFooterState: true);
  79. return false;
  80. }
  81. if (current is CustomPageRefreshErrorState) {
  82. // _refreshController.refreshFailed();
  83. return false;
  84. }
  85. return true;
  86. },
  87. builder: (context, state) {
  88. /// 有数据
  89. if (state is CustomPageLoadedState) {
  90. if (EmptyUtil.isEmpty(state.model)) return _buildEmptyWidget();
  91. return _buildMainWidget(state.model,state.backgroundModel);
  92. }
  93. /// 初始化失败
  94. if (state is CustomPageInitErrorState) {
  95. return _buildEmptyWidget();
  96. }
  97. /// 骨架图
  98. return _buildSkeletonWidget();
  99. },
  100. ),
  101. );
  102. }
  103. /// 有数据
  104. Widget _buildMainWidget(List<Map<String, dynamic>> model,BackgroundModel backgroundModel) {
  105. return Scaffold(
  106. appBar: _buildAppbar(model?.first),
  107. backgroundColor: HexColor.fromHex(backgroundModel?.bgColor??""),
  108. // floatingActionButton: _buildFloatWidget(),
  109. floatingActionButtonLocation: _CustomFloatingActionButtonLocation(FloatingActionButtonLocation.endFloat, 0, -100),
  110. body: Stack(
  111. children: <Widget>[
  112. _buildBackground(backgroundModel),
  113. Column(children: _buildFirstWidget(model)),
  114. ],
  115. ),
  116. );
  117. }
  118. /// 骨架图
  119. Widget _buildSkeletonWidget() {
  120. return Scaffold();
  121. }
  122. /// 空数据视图
  123. Widget _buildEmptyWidget() {
  124. return Scaffold(
  125. backgroundColor: HexColor.fromHex('#F9F9F9'),
  126. appBar: AppBar(
  127. brightness: Brightness.light,
  128. backgroundColor: Colors.white,
  129. leading: IconButton(
  130. icon: Icon(
  131. Icons.arrow_back_ios,
  132. size: 22,
  133. color: HexColor.fromHex('#333333'),
  134. ),
  135. onPressed: () => Navigator.maybePop(context),
  136. ),
  137. title: Text(
  138. '',
  139. style: TextStyle(color: HexColor.fromHex('#333333'), fontSize: 18, fontWeight: FontWeight.bold),
  140. ),
  141. centerTitle: true,
  142. elevation: 0,
  143. ),
  144. body: Column(
  145. crossAxisAlignment: CrossAxisAlignment.center,
  146. mainAxisAlignment: MainAxisAlignment.center,
  147. children: <Widget>[
  148. Align(
  149. alignment: Alignment.topCenter,
  150. child: EmptyWidget(
  151. tips: '网络似乎开小差了~',
  152. ),
  153. ),
  154. GestureDetector(
  155. onTap: () => _onRefreshEvent(),
  156. behavior: HitTestBehavior.opaque,
  157. child: Container(
  158. alignment: Alignment.center,
  159. decoration: BoxDecoration(borderRadius: BorderRadius.circular(10), border: Border.all(color: HexColor.fromHex('#999999'), width: 0.5), color: Colors.white),
  160. width: 80,
  161. height: 20,
  162. child: Text(
  163. '刷新一下',
  164. style: TextStyle(fontSize: 12, color: HexColor.fromHex('#333333'), fontWeight: FontWeight.bold),
  165. ),
  166. ),
  167. )
  168. ],
  169. ));
  170. }
  171. /// 数据,生成第一层widget
  172. List<Widget> _buildFirstWidget(List<Map<String, dynamic>> model) {
  173. List<Widget> result = [];
  174. // 分类导航的key ⚠️ 这里先写成Test 后续要改
  175. const String CATEGORY_KEY = 'category';
  176. // 判断是否有分类导航
  177. // 判断最后一个是否属于分类导航,如果属于,则有分类导航,如果不是,则无分类导航
  178. bool haveCategory = !EmptyUtil.isEmpty(model?.last) && model.last.containsKey('mod_name') && model.last['mod_name'] == CATEGORY_KEY;
  179. int endIndexLength = model.length;
  180. // 如果没有分类导航,则取分类导航之上的所有mod
  181. if (!haveCategory) {
  182. for (int i = 0; i < model.length; i++) {
  183. Map<String, dynamic> item = model[i];
  184. if (item['mod_name'] == CATEGORY_KEY) {
  185. endIndexLength = (i + 1);
  186. break;
  187. }
  188. }
  189. }
  190. for (int i = 0; i < endIndexLength; i++) {
  191. WidgetModel item = WidgetModel.fromJson(Map<String, dynamic>.from(model[i]));
  192. // last model
  193. if (i == endIndexLength - 1) {
  194. result.addAll(_buildTabBar(model[i], i));
  195. break;
  196. }
  197. // appBar 无需在这里添加
  198. if (item.modName.contains('appbar')) {
  199. continue;
  200. }
  201. result.addAll(WidgetFactory.create(item.modName, isSliver: false, model: model[i]));
  202. }
  203. // 没有appbar并且没有tabbar,则给第一个元素加边距
  204. if (!_isHasAppbar) {
  205. result.insert(0, SizedBox(height: MediaQueryData.fromWindow(window).padding.top, child: Container(color: HexColor.fromHex(''),),));
  206. }
  207. return result;
  208. }
  209. /// appbar
  210. Widget _buildAppbar(final Map<String, dynamic> model) {
  211. if (EmptyUtil.isEmpty(model)) return null;
  212. String mobName = model['mod_name'];
  213. if (!mobName.contains('_appbar')) return null;
  214. Map<String, dynamic> data = Map<String, dynamic>();
  215. try {
  216. data = jsonDecode(model['data']);
  217. } catch (e, s) {
  218. Logger.warn(e, s);
  219. }
  220. String parentTitle = !EmptyUtil.isEmpty(widget?.data) ? widget?.data['title'] ?? '' : '';
  221. _isHasAppbar = true;
  222. return AppBar(
  223. backgroundColor: HexColor.fromHex(null != data ? data['app_bar_bg_color'] ?? '#FFFFFF' : '#FFFFFF'),
  224. brightness: Brightness.light,
  225. leading: IconButton(
  226. icon: Icon(
  227. Icons.arrow_back_ios,
  228. size: 22,
  229. color: HexColor.fromHex('#333333'),
  230. ),
  231. onPressed: () => Navigator.maybePop(context),
  232. ),
  233. title: Text(
  234. null != data && data.containsKey('app_bar_name') ? data['app_bar_name'] != '自定义页面' ? data['app_bar_name'] : parentTitle : parentTitle,
  235. style: TextStyle(
  236. color: HexColor.fromHex(null != data ? data['app_bar_name_color'] ?? '#333333' : '#333333'),
  237. fontSize: 16,
  238. fontWeight: FontWeight.bold,
  239. ),
  240. ),
  241. centerTitle: true,
  242. elevation: 0,
  243. );
  244. }
  245. /// tabBar
  246. List<Widget> _buildTabBar(final Map<String, dynamic> model, final int index) {
  247. Map<String, dynamic> data = Map<String, dynamic>();
  248. List<Map<String, dynamic>> listStyle = [];
  249. List<Widget> result = [];
  250. try {
  251. data = jsonDecode(model['data']);
  252. listStyle = List.from(data['list_style']);
  253. } catch (e, s) {
  254. Logger.warn(e, s);
  255. }
  256. // 1、导航栏没开启的情况 传null进去进行获取没开启导航栏的widget集合
  257. if (EmptyUtil.isEmpty(listStyle)) {
  258. result.add(Expanded(
  259. child: CustomItemPage(null, 0, model['mod_id']?.toString() ?? null, model['mod_pid']?.toString() ?? null, (!_isHasAppbar && index == 0 )),
  260. ));
  261. return result;
  262. }
  263. // 2、导航栏开启的情况
  264. if (listStyle.length > 0) {
  265. // tabContorller 初始化
  266. if (null == _tabController || _tabController.length != listStyle.length) {
  267. _tabController = new TabController(length: listStyle.length, vsync: this);
  268. }
  269. result.add(Container(
  270. height: 40,
  271. width: double.infinity,
  272. color: HexColor.fromHex(data['bg_color']),
  273. child: TabBar(
  274. controller: _tabController,
  275. isScrollable: /*listStyle.length <= 5 ? false : */ true,
  276. labelColor: HexColor.fromHex(data['choose_text_color'] ?? '#FF4242'),
  277. labelStyle: TextStyle(fontSize: 14, fontWeight: FontWeight.bold),
  278. unselectedLabelColor: HexColor.fromHex(data['text_color'] ?? '#999999'),
  279. unselectedLabelStyle: TextStyle(fontSize: 14, fontWeight: FontWeight.bold),
  280. indicatorSize: TabBarIndicatorSize.label,
  281. indicator: MaterialIndicator(
  282. color: HexColor.fromHex(data['choose_color'] ?? '#FF4242'),
  283. bottomLeftRadius: 1.25,
  284. topLeftRadius: 1.25,
  285. topRightRadius: 1.25,
  286. bottomRightRadius: 1.25,
  287. height: 2.5,
  288. horizontalPadding: 5,
  289. ),
  290. tabs: listStyle.map((e) => Text(e['name'])).toList(),
  291. ),
  292. ));
  293. _isHasTabBar = true;
  294. // 最后添加TabBarView
  295. result.add(Expanded(
  296. child: TabBarView(
  297. controller: _tabController,
  298. children: _buildTabBarViewChildren(listStyle, model['mod_id']?.toString(), model['mod_pid']?.toString(), index),
  299. ),
  300. ));
  301. }
  302. return result;
  303. }
  304. /// 返回TabBarView的视图
  305. List<Widget> _buildTabBarViewChildren(final List<Map<String, dynamic>> listStyle, final String modId, final String modPid, final int index) {
  306. List<Widget> result = [];
  307. for (int i = 0; i < listStyle.length; i++) {
  308. result.add(CustomItemPage(listStyle[i], i, modId, modPid, (!_isHasAppbar && !_isHasTabBar && index == 0 )));
  309. }
  310. return result;
  311. }
  312. _buildBackground(BackgroundModel backgroundModel) {
  313. if (backgroundModel != null) {
  314. var headerBg = backgroundModel.headerBg;
  315. return Container(
  316. height: double.tryParse(headerBg?.height)?? 0,
  317. width: double.infinity,
  318. decoration: BoxDecoration(
  319. gradient: LinearGradient(
  320. begin: Alignment.topCenter,
  321. end: Alignment.bottomCenter,
  322. colors: [HexColor.fromHex(headerBg?.mainColor ?? ""), HexColor.fromHex(headerBg?.assistColor ?? ""), HexColor.fromHex(headerBg?.minorColor ?? "")])),
  323. );
  324. }else{
  325. return Container();
  326. }
  327. }
  328. // /// 悬浮按钮
  329. // Widget _buildFloatWidget() {
  330. // return Visibility(
  331. // visible: true,
  332. // child: GestureDetector(
  333. // onTap: () => _scrollTop(),
  334. // behavior: HitTestBehavior.opaque,
  335. // child: Container(
  336. // height: 30,
  337. // width: 30,
  338. // child: Icon(Icons.arrow_upward),
  339. // ),
  340. // ),
  341. // );
  342. // }
  343. }
  344. /// 回到顶部的icon
  345. class _CustomFloatingActionButtonLocation extends FloatingActionButtonLocation {
  346. FloatingActionButtonLocation location;
  347. double offsetX; // X方向的偏移量
  348. double offsetY; // Y方向的偏移量
  349. _CustomFloatingActionButtonLocation(this.location, this.offsetX, this.offsetY);
  350. @override
  351. Offset getOffset(ScaffoldPrelayoutGeometry scaffoldGeometry) {
  352. Offset offset = location.getOffset(scaffoldGeometry);
  353. return Offset(offset.dx + offsetX, offset.dy + offsetY);
  354. }
  355. }