import 'dart:async'; import 'package:PiliPalaX/http/loading_state.dart'; import 'package:easy_debounce/easy_throttle.dart'; import 'package:flutter/material.dart'; import 'package:flutter/rendering.dart'; import 'package:get/get.dart'; import 'package:PiliPalaX/common/constants.dart'; import 'package:PiliPalaX/common/skeleton/video_card_v.dart'; import 'package:PiliPalaX/common/widgets/animated_dialog.dart'; import 'package:PiliPalaX/common/widgets/http_error.dart'; import 'package:PiliPalaX/common/widgets/video_card_v.dart'; import 'package:PiliPalaX/pages/home/index.dart'; import 'package:PiliPalaX/pages/main/index.dart'; import '../../utils/grid.dart'; import 'controller.dart'; class RcmdPage extends StatefulWidget { const RcmdPage({super.key}); @override State createState() => _RcmdPageState(); } class _RcmdPageState extends State with AutomaticKeepAliveClientMixin { final RcmdController _rcmdController = Get.put(RcmdController()); @override bool get wantKeepAlive => true; @override void initState() { super.initState(); StreamController mainStream = Get.find().bottomBarStream; StreamController searchBarStream = Get.find().searchBarStream; _rcmdController.scrollController.addListener( () { if (_rcmdController.scrollController.position.pixels >= _rcmdController.scrollController.position.maxScrollExtent - 200) { EasyThrottle.throttle( 'my-throttler', const Duration(milliseconds: 200), () { _rcmdController.onLoadMore(); }); } final ScrollDirection direction = _rcmdController.scrollController.position.userScrollDirection; if (direction == ScrollDirection.forward) { mainStream.add(true); searchBarStream.add(true); } else if (direction == ScrollDirection.reverse) { mainStream.add(false); searchBarStream.add(false); } }, ); } @override void dispose() { _rcmdController.scrollController.removeListener(() {}); super.dispose(); } @override Widget build(BuildContext context) { super.build(context); return Container( clipBehavior: Clip.hardEdge, margin: const EdgeInsets.only( left: StyleString.safeSpace, right: StyleString.safeSpace), decoration: const BoxDecoration( borderRadius: BorderRadius.all(StyleString.imgRadius), ), child: RefreshIndicator( onRefresh: () async { await _rcmdController.onRefresh(); }, child: CustomScrollView( controller: _rcmdController.scrollController, physics: const AlwaysScrollableScrollPhysics(), slivers: [ SliverPadding( padding: const EdgeInsets.only(top: StyleString.cardSpace), sliver: Obx( () => _rcmdController.loadingState.value is Loading || _rcmdController.loadingState.value is Success ? contentGrid(_rcmdController.loadingState.value) : HttpError( errMsg: _rcmdController.loadingState.value is Error ? (_rcmdController.loadingState.value as Error) .errMsg : '没有相关数据', fn: () { _rcmdController.loadingState.value = LoadingState.loading(); _rcmdController.onRefresh(); }), ), ), ], ), ), ); } void _removePopupDialog() { _rcmdController.popupDialog.last?.remove(); _rcmdController.popupDialog.removeLast(); } OverlayEntry _createPopupDialog(videoItem) { return OverlayEntry( builder: (context) => AnimatedDialog( closeFn: _removePopupDialog, videoItem: videoItem, ), ); } Widget contentGrid(LoadingState loadingState) { return SliverGrid( gridDelegate: SliverGridDelegateWithExtentAndRatio( // 行间距 mainAxisSpacing: StyleString.cardSpace, // 列间距 crossAxisSpacing: StyleString.cardSpace, // 最大宽度 maxCrossAxisExtent: Grid.maxRowWidth, childAspectRatio: StyleString.aspectRatio, mainAxisExtent: MediaQuery.textScalerOf(context).scale(90), ), delegate: SliverChildBuilderDelegate( (BuildContext context, int index) { return loadingState is Success ? VideoCardV( videoItem: loadingState.response[index], longPress: () { _rcmdController.popupDialog .add(_createPopupDialog(loadingState.response[index])); Overlay.of(context) .insert(_rcmdController.popupDialog.last!); }, longPressEnd: _removePopupDialog, ) : const VideoCardVSkeleton(); }, childCount: loadingState is Success ? loadingState.response.length : 10, ), ); } }