Files
PiliPlus/lib/pages/subscription_detail/view.dart
bggRGjQaUbCoE 9ad57dccb0 opt: safearea
Signed-off-by: bggRGjQaUbCoE <githubaccount56556@proton.me>
2025-04-21 16:22:41 +08:00

226 lines
7.5 KiB
Dart

import 'package:PiliPlus/http/loading_state.dart';
import 'package:PiliPlus/models/user/sub_detail.dart';
import 'package:PiliPlus/utils/grid.dart';
import 'package:flutter/material.dart';
import 'package:get/get.dart';
import 'package:PiliPlus/common/skeleton/video_card_h.dart';
import 'package:PiliPlus/common/widgets/http_error.dart';
import 'package:PiliPlus/common/widgets/network_img_layer.dart';
import '../../models/user/sub_folder.dart';
import '../../utils/utils.dart';
import 'controller.dart';
import 'widget/sub_video_card.dart';
class SubDetailPage extends StatefulWidget {
const SubDetailPage({super.key});
@override
State<SubDetailPage> createState() => _SubDetailPageState();
}
class _SubDetailPageState extends State<SubDetailPage> {
late final SubDetailController _subDetailController = Get.put(
SubDetailController(),
tag: Utils.makeHeroTag(Get.parameters['id']),
);
final RxBool showTitle = false.obs;
@override
void initState() {
super.initState();
_subDetailController.scrollController.addListener(listener);
}
void listener() {
showTitle.value = _subDetailController.scrollController.offset > 160;
}
@override
void dispose() {
_subDetailController.scrollController.removeListener(listener);
super.dispose();
}
@override
Widget build(BuildContext context) {
return Scaffold(
body: SafeArea(
top: false,
bottom: false,
child: CustomScrollView(
controller: _subDetailController.scrollController,
physics: const AlwaysScrollableScrollPhysics(),
slivers: [
_buildAppBar,
_buildCount,
Obx(() => _buildBody(_subDetailController.loadingState.value)),
],
),
),
);
}
Widget _buildBody(LoadingState<List<SubDetailMediaItem>?> loadingState) {
return switch (loadingState) {
Loading() => SliverGrid(
gridDelegate: Grid.videoCardHDelegate(context),
delegate: SliverChildBuilderDelegate(
(context, index) => const VideoCardHSkeleton(),
childCount: 10,
),
),
Success() => loadingState.response?.isNotEmpty == true
? SliverPadding(
padding: EdgeInsets.only(
bottom: MediaQuery.paddingOf(context).bottom + 80,
),
sliver: SliverGrid(
gridDelegate: Grid.videoCardHDelegate(context),
delegate: SliverChildBuilderDelegate(
childCount: loadingState.response!.length,
(context, index) {
if (index == loadingState.response!.length - 1) {
_subDetailController.onLoadMore();
}
return SubVideoCardH(
videoItem: loadingState.response![index],
);
},
),
),
)
: HttpError(
callback: _subDetailController.onReload,
),
Error() => HttpError(
errMsg: loadingState.errMsg,
callback: _subDetailController.onReload,
),
_ => throw UnimplementedError(),
};
}
Widget get _buildCount => SliverToBoxAdapter(
child: Padding(
padding: const EdgeInsets.only(top: 12, bottom: 8, left: 14),
child: Obx(
() => Text(
'${_subDetailController.mediaCount}条视频',
style: TextStyle(
fontSize: Theme.of(context).textTheme.labelMedium!.fontSize,
color: Theme.of(context).colorScheme.outline,
letterSpacing: 1,
),
),
),
),
);
Widget get _buildAppBar => SliverAppBar(
expandedHeight: 210 - MediaQuery.paddingOf(context).top,
pinned: true,
title: Obx(
() {
return AnimatedOpacity(
opacity: showTitle.value ? 1 : 0,
curve: Curves.easeOut,
duration: const Duration(milliseconds: 500),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
_subDetailController.item.title!,
maxLines: 1,
overflow: TextOverflow.ellipsis,
style: Theme.of(context).textTheme.titleMedium,
),
Text(
'${_subDetailController.mediaCount.value}条视频',
style: Theme.of(context).textTheme.labelMedium,
)
],
),
);
},
),
flexibleSpace: FlexibleSpaceBar(
background: Container(
height: 180,
decoration: BoxDecoration(
border: Border(
bottom: BorderSide(
color: Theme.of(context).dividerColor.withOpacity(0.2),
),
),
),
padding: EdgeInsets.only(
top: kTextTabBarHeight + MediaQuery.of(context).padding.top + 15,
left: 12,
right: 12,
bottom: 12,
),
child: Row(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Hero(
tag: _subDetailController.heroTag,
child: NetworkImgLayer(
width: 180,
height: 110,
src: _subDetailController.item.cover,
),
),
const SizedBox(width: 14),
Expanded(
child: Column(
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
const SizedBox(height: 4),
Text(
_subDetailController.item.title!,
style: TextStyle(
fontSize: Theme.of(context)
.textTheme
.titleMedium!
.fontSize,
fontWeight: FontWeight.bold),
),
const SizedBox(height: 4),
GestureDetector(
onTap: () {
SubFolderItemData item = _subDetailController.item;
Get.toNamed(
'/member?mid=${item.upper!.mid}',
arguments: {
'face': item.upper!.face,
},
);
},
child: Text(
_subDetailController.item.upper!.name!,
style: TextStyle(
color: Theme.of(context).colorScheme.primary),
),
),
const SizedBox(height: 4),
Obx(
() => Text(
'${Utils.numFormat(_subDetailController.playCount.value)}次播放',
style: TextStyle(
fontSize: 12,
color: Theme.of(context).colorScheme.outline,
),
),
),
],
),
),
],
),
),
),
);
}