import 'package:PiliPlus/common/constants.dart'; import 'package:PiliPlus/common/widgets/custom_sliver_persistent_header_delegate.dart'; import 'package:PiliPlus/common/widgets/loading_widget/http_error.dart'; import 'package:PiliPlus/common/widgets/loading_widget/loading_widget.dart'; import 'package:PiliPlus/common/widgets/refresh_indicator.dart'; import 'package:PiliPlus/http/loading_state.dart'; import 'package:PiliPlus/models_new/space/space_audio/item.dart'; import 'package:PiliPlus/pages/member_audio/controller.dart'; import 'package:PiliPlus/pages/member_audio/widgets/item.dart'; import 'package:PiliPlus/utils/grid.dart'; import 'package:flutter/material.dart'; import 'package:get/get.dart'; class MemberAudio extends StatefulWidget { const MemberAudio({ super.key, required this.heroTag, required this.mid, }); final String? heroTag; final int mid; @override State createState() => _MemberAudioState(); } class _MemberAudioState extends State with AutomaticKeepAliveClientMixin { late final _controller = Get.put( MemberAudioController(widget.mid), tag: widget.heroTag, ); @override Widget build(BuildContext context) { super.build(context); final colorScheme = ColorScheme.of(context); return refreshIndicator( onRefresh: _controller.onRefresh, child: CustomScrollView( physics: const AlwaysScrollableScrollPhysics(), slivers: [ SliverPadding( padding: EdgeInsets.only( bottom: MediaQuery.viewPaddingOf(context).bottom + 100, ), sliver: Obx( () => _buildBody(colorScheme, _controller.loadingState.value), ), ), ], ), ); } @override bool get wantKeepAlive => true; late final gridDelegate = SliverGridDelegateWithExtentAndRatio( mainAxisSpacing: 2, maxCrossAxisExtent: Grid.smallCardWidth * 2, childAspectRatio: StyleString.aspectRatio * 2.6, minHeight: MediaQuery.textScalerOf(context).scale(90), ); Widget _buildBody( ColorScheme colorScheme, LoadingState?> loadingState, ) { return switch (loadingState) { Loading() => linearLoading, Success(:var response) => response?.isNotEmpty == true ? SliverMainAxisGroup( slivers: [ SliverPersistentHeader( floating: true, delegate: CustomSliverPersistentHeaderDelegate( extent: 40, bgColor: colorScheme.surface, child: SizedBox( height: 40, child: Row( children: [ const SizedBox(width: 8), Padding( padding: const EdgeInsets.only(left: 6), child: Text( '共${_controller.totalSize ?? 0}首', style: const TextStyle(fontSize: 13), ), ), Container( height: 35, padding: const EdgeInsets.only(left: 6), child: TextButton.icon( onPressed: _controller.toViewPlayAll, icon: Icon( Icons.play_circle_outline_rounded, size: 16, color: colorScheme.secondary, ), label: Text( '播放全部', style: TextStyle( fontSize: 13, color: colorScheme.secondary, ), ), ), ), ], ), ), ), ), SliverGrid.builder( gridDelegate: gridDelegate, itemBuilder: (context, index) { if (index == response.length - 1) { _controller.onLoadMore(); } return MemberAudioItem( item: response[index], ); }, itemCount: response!.length, ), ], ) : HttpError(onReload: _controller.onReload), Error(:var errMsg) => HttpError( errMsg: errMsg, onReload: _controller.onReload, ), }; } }