load_state_layout.dart 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. import 'package:flutter/material.dart';
  2. import 'package:get/get.dart';
  3. import 'package:cs_resources/constants/color_constants.dart';
  4. import 'my_load_image.dart';
  5. import 'my_text_view.dart';
  6. ///四种视图状态
  7. enum LoadState { State_Success, State_Error, State_Loading, State_Empty }
  8. ///根据不同状态来展示不同的视图
  9. class LoadStateLayout extends StatefulWidget {
  10. final LoadState state; //页面状态
  11. final Widget? successWidget; //成功视图
  12. final List<SliverMultiBoxAdaptorWidget>? successSliverWidget; //成功的滚动视图
  13. final VoidCallback? errorRetry; //错误事件处理
  14. String? errorMessage;
  15. LoadStateLayout({
  16. Key? key,
  17. this.state = LoadState.State_Loading, //默认为加载状态
  18. this.successWidget, //成功的布局 (二选一)
  19. this.successSliverWidget, //成功的滚动布局(二选一)
  20. this.errorMessage, //错误的信息展示
  21. this.errorRetry, //错误重试的事件
  22. }) : super(key: key);
  23. @override
  24. _LoadStateLayoutState createState() => _LoadStateLayoutState();
  25. }
  26. class _LoadStateLayoutState extends State<LoadStateLayout> {
  27. @override
  28. Widget build(BuildContext context) {
  29. if (widget.successSliverWidget != null) {
  30. //如果有 successSliverWidget 就使用 Slivers 的方式布局
  31. return CustomScrollView(
  32. slivers: _buildSlivers(),
  33. );
  34. } else {
  35. //如果没有有 successSliverWidget 就使用默认布局的方式布局
  36. return SizedBox(
  37. width: double.infinity,
  38. height: double.infinity,
  39. child: _buildWidget,
  40. );
  41. }
  42. }
  43. //Slivers的布局
  44. List<Widget> _buildSlivers() {
  45. return _buildListWidget;
  46. }
  47. ///根据不同状态来显示不同的视图 (默认布局)
  48. Widget get _buildWidget {
  49. switch (widget.state) {
  50. case LoadState.State_Success:
  51. return widget.successWidget ?? const SizedBox();
  52. case LoadState.State_Error:
  53. return _errorView;
  54. case LoadState.State_Loading:
  55. return _loadingView;
  56. case LoadState.State_Empty:
  57. return _emptyView;
  58. default:
  59. return _loadingView;
  60. }
  61. }
  62. ///根据不同状态来显示不同的视图 (CustomScrollView)
  63. List<Widget> get _buildListWidget {
  64. switch (widget.state) {
  65. case LoadState.State_Success:
  66. return widget.successSliverWidget != null
  67. ? widget.successSliverWidget!
  68. : widget.successWidget != null
  69. ? [widget.successWidget!]
  70. : [const SizedBox()];
  71. case LoadState.State_Error:
  72. return [widget.successSliverWidget != null ? _warpStateLayout(_errorView) : _errorView];
  73. case LoadState.State_Loading:
  74. return [widget.successSliverWidget != null ? _warpStateLayout(_loadingView) : _loadingView];
  75. case LoadState.State_Empty:
  76. return [widget.successSliverWidget != null ? _warpStateLayout(_emptyView) : _emptyView];
  77. default:
  78. return [widget.successSliverWidget != null ? _warpStateLayout(_loadingView) : _loadingView];
  79. }
  80. }
  81. //如果父布局是 CustomScrollView 则使用 SliverFillViewport 包裹状态布局
  82. Widget _warpStateLayout(Widget widget) {
  83. return SliverFillViewport(
  84. delegate: SliverChildBuilderDelegate(
  85. (context, index) {
  86. return widget;
  87. },
  88. childCount: 1,
  89. ),
  90. );
  91. }
  92. // ===================================== 真正的状态布局 ↓ =====================================
  93. ///加载中视图
  94. Widget get _loadingView {
  95. return Container(
  96. width: double.infinity,
  97. height: double.infinity,
  98. alignment: Alignment.center,
  99. child: Column(
  100. mainAxisSize: MainAxisSize.max,
  101. mainAxisAlignment: MainAxisAlignment.center,
  102. crossAxisAlignment: CrossAxisAlignment.center,
  103. children: [
  104. Get.isDarkMode
  105. ? const CircularProgressIndicator(
  106. strokeWidth: 3,
  107. valueColor: AlwaysStoppedAnimation(Colors.white),
  108. )
  109. : CircularProgressIndicator(
  110. strokeWidth: 3,
  111. valueColor: AlwaysStoppedAnimation(ColorConstants.appBlue),
  112. ),
  113. MyTextView('加载中...'.tr, marginTop: 15, fontSize: 15.5)
  114. ],
  115. ),
  116. );
  117. }
  118. ///错误视图
  119. Widget get _errorView {
  120. return Container(
  121. width: double.infinity,
  122. height: double.infinity,
  123. alignment: Alignment.center,
  124. padding: const EdgeInsets.only(bottom: 10),
  125. child: GestureDetector(
  126. onTap: widget.errorRetry,
  127. child: Column(
  128. mainAxisSize: MainAxisSize.max,
  129. crossAxisAlignment: CrossAxisAlignment.center,
  130. mainAxisAlignment: MainAxisAlignment.center,
  131. children: <Widget>[
  132. const MyAssetImage('page_load_error.png', width: 180, height: 180, fit: BoxFit.contain),
  133. MyTextView(widget.errorMessage ?? '加载数据错误,请重试'.tr, marginTop: 10, fontSize: 15.5),
  134. ],
  135. )));
  136. }
  137. ///数据为空的视图
  138. Widget get _emptyView {
  139. return Container(
  140. width: double.infinity,
  141. height: double.infinity,
  142. alignment: Alignment.center,
  143. padding: const EdgeInsets.only(bottom: 10),
  144. child: Column(
  145. mainAxisSize: MainAxisSize.max,
  146. crossAxisAlignment: CrossAxisAlignment.center,
  147. mainAxisAlignment: MainAxisAlignment.center,
  148. children: <Widget>[
  149. const MyAssetImage('page_no_data.png', width: 180, height: 180, fit: BoxFit.contain),
  150. MyTextView('暂无数据'.tr, marginTop: 10, fontSize: 15.5),
  151. ],
  152. ),
  153. );
  154. }
  155. }