import 'dart:async'; import 'package:PiliPlus/common/widgets/flutter/refresh_indicator.dart'; import 'package:PiliPlus/common/widgets/loading_widget/http_error.dart'; import 'package:PiliPlus/http/loading_state.dart'; import 'package:PiliPlus/models/common/dynamic/dynamics_type.dart'; import 'package:PiliPlus/models/dynamics/result.dart'; import 'package:PiliPlus/pages/dynamics/controller.dart'; import 'package:PiliPlus/pages/dynamics/widgets/dynamic_panel.dart'; import 'package:PiliPlus/pages/dynamics_tab/controller.dart'; import 'package:PiliPlus/utils/extension/get_ext.dart'; import 'package:PiliPlus/utils/global_data.dart'; import 'package:PiliPlus/utils/waterfall.dart'; import 'package:flutter/material.dart'; import 'package:get/get.dart'; import 'package:waterfall_flow/waterfall_flow.dart' hide SliverWaterfallFlowDelegateWithMaxCrossAxisExtent; class DynamicsTabPage extends StatefulWidget { const DynamicsTabPage({super.key, required this.dynamicsType}); final DynamicsTabType dynamicsType; @override State createState() => _DynamicsTabPageState(); } class _DynamicsTabPageState extends State with AutomaticKeepAliveClientMixin, DynMixin { StreamSubscription? _listener; DynamicsController dynamicsController = Get.putOrFind(DynamicsController.new); late final DynamicsTabController controller; @override bool get wantKeepAlive => true; @override void initState() { controller = Get.putOrFind( () => DynamicsTabController(dynamicsType: widget.dynamicsType) ..mid = dynamicsController.mid.value, tag: widget.dynamicsType.name, ); super.initState(); if (widget.dynamicsType == DynamicsTabType.up) { _listener = dynamicsController.mid.listen((mid) { if (mid != -1) { controller ..mid = mid ..onReload(); } }); } } @override void dispose() { _listener?.cancel(); dynamicsController.mid.close(); super.dispose(); } @override Widget build(BuildContext context) { super.build(context); return refreshIndicator( onRefresh: () { dynamicsController.queryFollowUp(); return controller.onRefresh(); }, child: CustomScrollView( physics: const AlwaysScrollableScrollPhysics(), controller: controller.scrollController, slivers: [ SliverPadding( padding: const EdgeInsets.only(bottom: 100), sliver: buildPage( Obx(() => _buildBody(controller.loadingState.value)), ), ), ], ), ); } Widget _buildBody(LoadingState?> loadingState) { return switch (loadingState) { Loading() => dynSkeleton, Success(:final response) => response != null && response.isNotEmpty ? GlobalData().dynamicsWaterfallFlow ? SliverWaterfallFlow( gridDelegate: dynGridDelegate, delegate: SliverChildBuilderDelegate( (_, index) { if (index == response.length - 1) { controller.onLoadMore(); } final item = response[index]; return DynamicPanel( item: item, onRemove: (idStr) => controller.onRemove(index, idStr), onBlock: () => controller.onBlock(index), onUnfold: () => controller.onUnfold(item, index), ); }, childCount: response.length, ), ) : SliverList.builder( itemBuilder: (context, index) { if (index == response.length - 1) { controller.onLoadMore(); } final item = response[index]; return DynamicPanel( item: item, onRemove: (idStr) => controller.onRemove(index, idStr), onBlock: () => controller.onBlock(index), onUnfold: () => controller.onUnfold(item, index), ); }, itemCount: response.length, ) : HttpError(onReload: controller.onReload), Error(:final errMsg) => HttpError( errMsg: errMsg, onReload: controller.onReload, ), }; } }