notification_view_model.dart 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. import 'package:cpt_main/modules/notification/notification_group_data.dart';
  2. import 'package:cs_resources/generated/l10n.dart';
  3. import 'package:domain/entity/notification_page_entity.dart';
  4. import 'package:domain/repository/main_repository.dart';
  5. import 'package:plugin_platform/engine/dialog/dialog_engine.dart';
  6. import 'package:plugin_platform/engine/notify/notify_engine.dart';
  7. import 'package:plugin_platform/engine/toast/toast_engine.dart';
  8. import 'package:plugin_platform/http/dio/dio_cancelable_mixin.dart';
  9. import 'package:riverpod_annotation/riverpod_annotation.dart';
  10. import 'package:router/componentRouter/component_service_manager.dart';
  11. import 'package:shared/utils/date_time_utils.dart';
  12. import 'package:shared/utils/log_utils.dart';
  13. import 'package:widgets/dialog/app_default_dialog.dart';
  14. import 'package:widgets/load_state_layout.dart';
  15. import 'package:widgets/widget_export.dart';
  16. import 'notification_state.dart';
  17. part 'notification_view_model.g.dart';
  18. @riverpod
  19. class NotificationViewModel extends _$NotificationViewModel with DioCancelableMixin {
  20. late final MainRepository _mainRepository;
  21. @override
  22. NotificationState build() {
  23. _mainRepository = ref.read(mainRepositoryProvider);
  24. return NotificationState(datas: []);
  25. }
  26. var _curPage = 1; //请求参数当前的页面
  27. var _needShowPlaceholder = true; //是否展示LoadingView
  28. // Refresh 控制器
  29. final EasyRefreshController refreshController = EasyRefreshController(
  30. controlFinishRefresh: true, //允许刷新
  31. controlFinishLoad: true, //允许加载
  32. );
  33. //刷新页面状态
  34. void changeLoadingState(LoadState loadState, String? errorMsg) {
  35. state = state.copyWith(loadingState: loadState, errorMessage: errorMsg);
  36. }
  37. // Refresh 刷新事件
  38. Future onRefresh() async {
  39. _curPage = 1;
  40. fetchList();
  41. }
  42. // Refresh 加载事件
  43. Future loadMore() async {
  44. _curPage++;
  45. fetchList();
  46. }
  47. // 重试请求
  48. Future retryRequest() async {
  49. _curPage = 1;
  50. _needShowPlaceholder = true;
  51. fetchList();
  52. }
  53. /// 获取服务器数据
  54. Future fetchList() async {
  55. if (_needShowPlaceholder) {
  56. changeLoadingState(LoadState.State_Loading, null);
  57. }
  58. // 获取列表
  59. var listResult = await _mainRepository.fetchNotificationList(
  60. curPage: _curPage,
  61. cancelToken: cancelToken,
  62. );
  63. // 处理数据
  64. if (listResult.isSuccess) {
  65. handleList(listResult.data?.list);
  66. } else {
  67. changeLoadingState(LoadState.State_Error, listResult.errorMsg);
  68. }
  69. // 最后赋值
  70. _needShowPlaceholder = false;
  71. }
  72. // 处理数据与展示的逻辑
  73. void handleList(List<NotificationPageList>? list) {
  74. if (list != null && list.isNotEmpty) {
  75. //有数据,处理转换数据
  76. List<NotificationGroupData> newDatas;
  77. //先从当的数据中找到 groupId 比对获取到的list 是否有匹配的 groupId
  78. if (_curPage == 1) {
  79. newDatas = []; // 重新加载数据
  80. } else {
  81. newDatas = List.from(state.datas); // 获取当前状态的数据的副本去添加数据
  82. }
  83. for (var item in list) {
  84. String? createAt = item.createdAt!.split(" at ")[0];
  85. //处理 groupId 的显示字段 21 May 2025
  86. DateTime? dateTime = DateTimeUtils.parseCustomDate(createAt.trim(), format: 'dd MMM yyyy');
  87. String itemGroupId;
  88. if (dateTime != null) {
  89. if (DateTimeUtils.isTodayByDateTime(dateTime)) {
  90. itemGroupId = S.current.today;
  91. } else {
  92. itemGroupId = "${DateTimeUtils.getWeekday(dateTime, languageCode: 'en')} ${DateTimeUtils.formatDate(dateTime, format: 'dd MMM yyyy')}";
  93. }
  94. } else {
  95. itemGroupId = createAt;
  96. }
  97. // 尝试在当前分组中找到匹配的 createdAt
  98. var curGroup = newDatas.firstWhere((element) => element.groupId == itemGroupId, orElse: () => NotificationGroupData(null, []));
  99. // 如果有匹配的 createdAt,则添加进去
  100. if (curGroup.groupId != null) {
  101. curGroup.groupDatas!.add(item); // 添加通知到现有组
  102. } else {
  103. // 如果没有匹配的 createdAt,则创建新的分组
  104. newDatas.add(NotificationGroupData(itemGroupId, [item])); // 创建新组
  105. }
  106. }
  107. if (_curPage == 1) {
  108. //刷新的方式
  109. state = state.copyWith(datas: newDatas);
  110. refreshController.finishRefresh();
  111. //更新展示的状态
  112. changeLoadingState(LoadState.State_Success, null);
  113. } else {
  114. //加载更多
  115. state.datas.addAll(newDatas);
  116. refreshController.finishLoad();
  117. }
  118. } else {
  119. if (_curPage == 1) {
  120. //展示无数据的布局
  121. state = state.copyWith(datas: []);
  122. changeLoadingState(LoadState.State_Empty, null);
  123. refreshController.finishRefresh();
  124. } else {
  125. //展示加载完成,没有更多数据了
  126. refreshController.finishLoad(IndicatorResult.noMore);
  127. }
  128. }
  129. }
  130. /// 点击标记全部
  131. void markAll() {
  132. DialogEngine.show(
  133. widget: AppDefaultDialog(
  134. message: S.current.read_all_msg,
  135. confirmAction: () {
  136. _requestReadAll();
  137. },
  138. ));
  139. }
  140. ///设置单个已读
  141. void setReadStatus(NotificationPageList item) async {
  142. // 创建新的数据列表
  143. List<NotificationGroupData> newDatas = List.from(state.datas); // 复制一份当前数据
  144. // 找到对应的组并更新状态
  145. for (var group in newDatas) {
  146. for (var notification in group.groupDatas!) {
  147. if (notification.id == item.id) {
  148. // 假设 item 有一个唯一的 id 字段
  149. notification.read = true; // 更新状态为已读
  150. break; // 找到并更新后可以退出循环
  151. }
  152. }
  153. }
  154. // 更新状态,通知 UI 刷新
  155. state = state.copyWith(datas: newDatas);
  156. //转转详情
  157. _godoDetailPage(item);
  158. // 发送请求设置服务器已读
  159. final result = await _mainRepository.setNotificationRead(id: item.id);
  160. if (result.isSuccess) {
  161. Log.d("设置 ${item.id} 的消息已读成功!");
  162. } else {
  163. ToastEngine.show(result.errorMsg ?? "UnKnow Error");
  164. }
  165. }
  166. ///跳转到详情页
  167. void _godoDetailPage(NotificationPageList item) {
  168. NotificationPageListData? data = item.data;
  169. if (data == null) return;
  170. // 1、加入unit成功 ApprovedJoinUnitNotification
  171. // 2、加入unit失败 RejectedJoinUnitNotification
  172. // 3、online form批准了 ApprovedApplyOnlineFormNotification
  173. // 4、online form拒绝了 RejectedApplyOnlineFormNotification
  174. // 5、facility 预定成功了(不需要,因为支付了就是预定成功了)
  175. // 6、facility 预定失败了(长时间未支付,被系统取消了) FacilityBookingNotPaidCancelNotification
  176. // 7、每月物业费账单生成了
  177. // 8、每月停车费账单生成了
  178. // 9、服务订单长时间未支付 PaidServiceOrderNotPaidCancelNotification
  179. String? type = data.type;
  180. if (type == 'ApprovedApplyOnlineFormNotification' || type == 'RejectedApplyOnlineFormNotification') {
  181. //去 Form 详情
  182. ComponentServiceManager().formService.startFormDetailPage(data.estateOnlineFormId ?? "", data.id ?? "", data.onlineFormTypeId ?? "");
  183. }
  184. }
  185. /// 请求设置全部已读
  186. void _requestReadAll() async {
  187. // 发送请求设置服务器已读
  188. final result = await _mainRepository.setNotificationRead();
  189. if (result.isSuccess) {
  190. Log.d("设置全部的消息已读成功!");
  191. NotifyEngine.showSuccess(S.current.successful);
  192. // 创建新的数据列表
  193. List<NotificationGroupData> newDatas = List.from(state.datas); // 复制一份当前数据
  194. // 找到对应的组并更新状态
  195. for (var group in newDatas) {
  196. for (var notification in group.groupDatas!) {
  197. notification.read = true; // 更新状态为已读
  198. }
  199. }
  200. // 更新状态,通知 UI 刷新
  201. state = state.copyWith(datas: newDatas);
  202. } else {
  203. ToastEngine.show(result.errorMsg ?? "UnKnow Error");
  204. }
  205. }
  206. }