mirror of
https://github.com/HChaZZY/PiliPlus.git
synced 2025-12-06 09:13:48 +08:00
178 lines
5.4 KiB
Dart
178 lines
5.4 KiB
Dart
import 'package:PiliPlus/common/skeleton/msg_feed_top.dart';
|
||
import 'package:PiliPlus/common/widgets/dialog/dialog.dart';
|
||
import 'package:PiliPlus/common/widgets/image/network_img_layer.dart';
|
||
import 'package:PiliPlus/common/widgets/loading_widget/http_error.dart';
|
||
import 'package:PiliPlus/common/widgets/refresh_indicator.dart';
|
||
import 'package:PiliPlus/common/widgets/view_sliver_safe_area.dart';
|
||
import 'package:PiliPlus/http/loading_state.dart';
|
||
import 'package:PiliPlus/models/common/image_type.dart';
|
||
import 'package:PiliPlus/models_new/fans/list.dart';
|
||
import 'package:PiliPlus/pages/fan/controller.dart';
|
||
import 'package:PiliPlus/pages/share/view.dart' show UserModel;
|
||
import 'package:PiliPlus/services/account_service.dart';
|
||
import 'package:PiliPlus/utils/grid.dart';
|
||
import 'package:PiliPlus/utils/utils.dart';
|
||
import 'package:flutter/material.dart';
|
||
import 'package:get/get.dart';
|
||
|
||
class FansPage extends StatefulWidget {
|
||
const FansPage({
|
||
super.key,
|
||
this.mid,
|
||
this.onSelect,
|
||
});
|
||
|
||
final int? mid;
|
||
final ValueChanged<UserModel>? onSelect;
|
||
|
||
@override
|
||
State<FansPage> createState() => _FansPageState();
|
||
}
|
||
|
||
class _FansPageState extends State<FansPage> {
|
||
late int mid;
|
||
String? name;
|
||
late bool isOwner;
|
||
late FansController _fansController;
|
||
|
||
@override
|
||
void initState() {
|
||
super.initState();
|
||
AccountService accountService = Get.find<AccountService>();
|
||
late final mid = Get.parameters['mid'];
|
||
this.mid =
|
||
widget.mid ?? (mid != null ? int.parse(mid) : accountService.mid);
|
||
isOwner = this.mid == accountService.mid;
|
||
name = Get.parameters['name'] ?? accountService.name.value;
|
||
_fansController = Get.put(
|
||
FansController(this.mid),
|
||
tag: Utils.makeHeroTag(this.mid),
|
||
);
|
||
}
|
||
|
||
@override
|
||
Widget build(BuildContext context) {
|
||
final theme = Theme.of(context).colorScheme;
|
||
return Scaffold(
|
||
resizeToAvoidBottomInset: false,
|
||
appBar: widget.mid != null
|
||
? null
|
||
: AppBar(title: Text(isOwner ? '我的粉丝' : '$name的粉丝')),
|
||
body: refreshIndicator(
|
||
onRefresh: _fansController.onRefresh,
|
||
child: CustomScrollView(
|
||
physics: const AlwaysScrollableScrollPhysics(),
|
||
controller: _fansController.scrollController,
|
||
slivers: [
|
||
ViewSliverSafeArea(
|
||
sliver: Obx(
|
||
() => _buildBody(theme, _fansController.loadingState.value),
|
||
),
|
||
),
|
||
],
|
||
),
|
||
),
|
||
);
|
||
}
|
||
|
||
late final gridDelegate = SliverGridDelegateWithMaxCrossAxisExtent(
|
||
maxCrossAxisExtent: Grid.smallCardWidth * 2,
|
||
mainAxisExtent: 66,
|
||
);
|
||
|
||
Widget _buildBody(
|
||
ColorScheme theme,
|
||
LoadingState<List<FansItemModel>?> loadingState,
|
||
) {
|
||
return switch (loadingState) {
|
||
Loading() => SliverGrid.builder(
|
||
gridDelegate: gridDelegate,
|
||
itemBuilder: (context, index) => const MsgFeedTopSkeleton(),
|
||
itemCount: 16,
|
||
),
|
||
Success(:var response) =>
|
||
response?.isNotEmpty == true
|
||
? SliverGrid.builder(
|
||
gridDelegate: gridDelegate,
|
||
itemBuilder: (context, index) {
|
||
if (index == response.length - 1) {
|
||
_fansController.onLoadMore();
|
||
}
|
||
return _buildItem(theme, index, response[index]);
|
||
},
|
||
itemCount: response!.length,
|
||
)
|
||
: HttpError(onReload: _fansController.onReload),
|
||
Error(:var errMsg) => HttpError(
|
||
errMsg: errMsg,
|
||
onReload: _fansController.onReload,
|
||
),
|
||
};
|
||
}
|
||
|
||
Widget _buildItem(ColorScheme theme, int index, FansItemModel item) {
|
||
final isSelect = widget.onSelect != null;
|
||
void onRemove() => showConfirmDialog(
|
||
context: context,
|
||
title: '确定移除 ${item.uname} ?',
|
||
onConfirm: () => _fansController.onRemoveFan(index, item.mid!),
|
||
);
|
||
|
||
final flag = !isSelect && isOwner;
|
||
return SizedBox(
|
||
height: 66,
|
||
child: InkWell(
|
||
onTap: () {
|
||
if (widget.onSelect != null) {
|
||
widget.onSelect!(
|
||
UserModel(
|
||
mid: item.mid!,
|
||
name: item.uname!,
|
||
avatar: item.face!,
|
||
),
|
||
);
|
||
return;
|
||
}
|
||
Get.toNamed('/member?mid=${item.mid}');
|
||
},
|
||
onLongPress: flag ? onRemove : null,
|
||
onSecondaryTap: flag && !Utils.isMobile ? onRemove : null,
|
||
child: Padding(
|
||
padding: const EdgeInsets.symmetric(
|
||
horizontal: 12,
|
||
vertical: 10,
|
||
),
|
||
child: Row(
|
||
spacing: 10,
|
||
children: [
|
||
NetworkImgLayer(
|
||
width: 45,
|
||
height: 45,
|
||
type: ImageType.avatar,
|
||
src: item.face,
|
||
),
|
||
Column(
|
||
spacing: 3,
|
||
crossAxisAlignment: CrossAxisAlignment.start,
|
||
children: [
|
||
Text(
|
||
item.uname!,
|
||
style: const TextStyle(fontSize: 14),
|
||
),
|
||
if (item.sign != null)
|
||
Text(
|
||
item.sign!,
|
||
maxLines: 1,
|
||
overflow: TextOverflow.ellipsis,
|
||
style: TextStyle(fontSize: 13, color: theme.outline),
|
||
),
|
||
],
|
||
),
|
||
],
|
||
),
|
||
),
|
||
),
|
||
);
|
||
}
|
||
}
|