import 'dart:async'; 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/pages/home/index.dart'; import 'package:PiliPalaX/pages/main/index.dart'; import '../../utils/grid.dart'; import 'controller.dart'; import 'widgets/live_item.dart'; class LivePage extends StatefulWidget { const LivePage({super.key}); @override State createState() => _LivePageState(); } class _LivePageState extends State with AutomaticKeepAliveClientMixin { final LiveController _liveController = Get.put(LiveController()); late Future _futureBuilderFuture; @override bool get wantKeepAlive => true; @override void initState() { super.initState(); _futureBuilderFuture = _liveController.queryLiveList('init'); StreamController mainStream = Get.find().bottomBarStream; StreamController searchBarStream = Get.find().searchBarStream; _liveController.scrollController.addListener( () { if (_liveController.scrollController.position.pixels >= _liveController.scrollController.position.maxScrollExtent - 200) { EasyThrottle.throttle('liveList', const Duration(milliseconds: 200), () { _liveController.onLoad(); }); } final ScrollDirection direction = _liveController.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() { _liveController.scrollController.removeListener(() {}); _liveController.scrollController.dispose(); super.dispose(); } @override Widget build(BuildContext context) { super.build(context); return Container( clipBehavior: Clip.hardEdge, margin: const EdgeInsets.only( left: StyleString.cardSpace, right: StyleString.cardSpace), decoration: const BoxDecoration( borderRadius: BorderRadius.all(StyleString.imgRadius), ), child: RefreshIndicator( onRefresh: () async { return await _liveController.onRefresh(); }, child: CustomScrollView( physics: const AlwaysScrollableScrollPhysics(), controller: _liveController.scrollController, slivers: [ SliverPadding( // 单列布局 EdgeInsets.zero padding: const EdgeInsets.fromLTRB(0, StyleString.cardSpace, 0, 0), sliver: FutureBuilder( future: _futureBuilderFuture, builder: (context, snapshot) { if (snapshot.connectionState == ConnectionState.done) { if (snapshot.data == null) { return const SliverToBoxAdapter(child: SizedBox()); } Map data = snapshot.data as Map; if (data['status']) { return SliverLayoutBuilder( builder: (context, boxConstraints) { return Obx(() => contentGrid( _liveController, _liveController.liveList)); }); } else { return HttpError( errMsg: data['msg'], fn: () { setState(() { _futureBuilderFuture = _liveController.queryLiveList('init'); }); }, ); } } else { return contentGrid(_liveController, []); } }, ), ), ], ), ), ); } void _removePopupDialog() { _liveController.popupDialog.last?.remove(); _liveController.popupDialog.removeLast(); } OverlayEntry _createPopupDialog(liveItem) { return OverlayEntry( builder: (context) => AnimatedDialog( closeFn: _removePopupDialog, videoItem: liveItem, ), ); } Widget contentGrid(ctr, liveList) { return SliverGrid( gridDelegate: SliverGridDelegateWithExtentAndRatio( mainAxisSpacing: StyleString.cardSpace, crossAxisSpacing: StyleString.cardSpace, maxCrossAxisExtent: Grid.maxRowWidth, childAspectRatio: StyleString.aspectRatio, mainAxisExtent: MediaQuery.textScalerOf(context).scale(80), ), delegate: SliverChildBuilderDelegate( (BuildContext context, int index) { return liveList!.isNotEmpty ? LiveCardV( liveItem: liveList[index], longPress: () { _liveController.popupDialog .add(_createPopupDialog(liveList[index])); Overlay.of(context) .insert(_liveController.popupDialog.last!); }, longPressEnd: _removePopupDialog, ) : const VideoCardVSkeleton(); }, childCount: liveList!.isNotEmpty ? liveList!.length : 10, ), ); } }