mod: dynamic panel (#738)

This commit is contained in:
My-Responsitories
2025-04-23 16:01:09 +08:00
committed by GitHub
parent 7bb0307e6a
commit e0fe16fd14
6 changed files with 286 additions and 276 deletions

View File

@@ -775,4 +775,6 @@ class Api {
'${HttpString.appBaseUrl}/x/v2/search/recommend'; '${HttpString.appBaseUrl}/x/v2/search/recommend';
static const String articleInfo = '/x/article/viewinfo'; static const String articleInfo = '/x/article/viewinfo';
static const String dynamicReport = '/x/dynamic/feed/dynamic_report/add';
} }

View File

@@ -666,4 +666,28 @@ class UserHttp {
return {'status': false, 'msg': res.data['message']}; return {'status': false, 'msg': res.data['message']};
} }
} }
static Future<Map> dynamicReport({
required dynamic mid,
required dynamic dynId,
required int reasonType,
String? reasonDesc,
}) async {
final res = await Request().post(
Api.dynamicReport,
queryParameters: {
'csrf': Accounts.main.csrf,
},
data: {
"accused_uid": mid,
"dynamic_id": dynId,
"reason_type": reasonType,
"reason_desc": reasonType == 0 ? reasonDesc : null,
},
options: Options(
contentType: Headers.formUrlEncodedContentType,
),
);
return res.data as Map;
}
} }

View File

@@ -3,13 +3,11 @@ import 'dart:math';
import 'package:PiliPlus/common/widgets/avatar.dart'; import 'package:PiliPlus/common/widgets/avatar.dart';
import 'package:PiliPlus/common/widgets/report.dart'; import 'package:PiliPlus/common/widgets/report.dart';
import 'package:PiliPlus/common/widgets/save_panel.dart'; import 'package:PiliPlus/common/widgets/save_panel.dart';
import 'package:PiliPlus/http/index.dart';
import 'package:PiliPlus/http/video.dart'; import 'package:PiliPlus/http/video.dart';
import 'package:PiliPlus/utils/extension.dart'; import 'package:PiliPlus/utils/extension.dart';
import 'package:PiliPlus/utils/request_utils.dart'; import 'package:PiliPlus/utils/request_utils.dart';
import 'package:PiliPlus/utils/storage.dart'; import 'package:PiliPlus/utils/storage.dart';
import 'package:cached_network_image/cached_network_image.dart'; import 'package:cached_network_image/cached_network_image.dart';
import 'package:dio/dio.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart'; import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
@@ -213,254 +211,235 @@ class AuthorPanel extends StatelessWidget {
style: ButtonStyle( style: ButtonStyle(
padding: WidgetStateProperty.all(EdgeInsets.zero), padding: WidgetStateProperty.all(EdgeInsets.zero),
), ),
onPressed: () { onPressed: () => morePanel(context),
showModalBottomSheet(
context: context,
useSafeArea: true,
isScrollControlled: true,
constraints: BoxConstraints(
maxWidth: min(640, min(Get.width, Get.height)),
),
builder: (context) {
return morePanel(context);
},
);
},
icon: const Icon(Icons.more_vert_outlined, size: 18), icon: const Icon(Icons.more_vert_outlined, size: 18),
), ),
); );
Widget morePanel(context) { void morePanel(BuildContext context) {
String? bvid; String? bvid;
try { getBvid(String? type, dynamic major) => switch (type) {
bvid = switch (item.type) { 'DYNAMIC_TYPE_AV' => major?.archive?.bvid,
'DYNAMIC_TYPE_AV' => item.modules.moduleDynamic.major.archive.bvid, 'DYNAMIC_TYPE_UGC_SEASON' => major?.ugcSeason?.bvid,
'DYNAMIC_TYPE_UGC_SEASON' =>
item.modules.moduleDynamic.major.ugcSeason.bvid,
_ => null,
};
} catch (_) {}
if (bvid == null && item.orig != null) {
try {
bvid = switch (item.orig.type) {
'DYNAMIC_TYPE_AV' =>
item.orig.modules.moduleDynamic.major.archive.bvid,
'DYNAMIC_TYPE_UGC_SEASON' =>
item.orig.modules.moduleDynamic.major.ugcSeason.bvid,
_ => null, _ => null,
}; };
} catch (_) {} bvid = getBvid(item.type, item.modules?.moduleDynamic?.major);
if (bvid == null && item.orig != null) {
bvid = getBvid(item.type, item.orig?.modules?.moduleDynamic?.major);
} }
return Padding(
padding: EdgeInsets.only(bottom: MediaQuery.of(context).padding.bottom), showModalBottomSheet(
child: Column( context: context,
mainAxisSize: MainAxisSize.min, useSafeArea: true,
children: [ isScrollControlled: true,
InkWell( constraints: BoxConstraints(
onTap: Get.back, maxWidth: min(640, min(Get.width, Get.height)),
borderRadius: const BorderRadius.only( ),
topLeft: Radius.circular(28), builder: (context) {
topRight: Radius.circular(28), return Padding(
), padding:
child: Container( EdgeInsets.only(bottom: MediaQuery.of(context).padding.bottom),
height: 35, child: Column(
padding: const EdgeInsets.only(bottom: 2), mainAxisSize: MainAxisSize.min,
child: Center( children: [
InkWell(
onTap: Get.back,
borderRadius: const BorderRadius.only(
topLeft: Radius.circular(28),
topRight: Radius.circular(28),
),
child: Container( child: Container(
width: 32, height: 35,
height: 3, padding: const EdgeInsets.only(bottom: 2),
decoration: BoxDecoration( child: Center(
color: Theme.of(context).colorScheme.outline, child: Container(
borderRadius: const BorderRadius.all(Radius.circular(3))), width: 32,
height: 3,
decoration: BoxDecoration(
color: Theme.of(context).colorScheme.outline,
borderRadius:
const BorderRadius.all(Radius.circular(3))),
),
),
), ),
), ),
), if (bvid != null)
), ListTile(
if (bvid != null) onTap: () async {
ListTile( Get.back();
onTap: () async { try {
Get.back(); var res = await UserHttp.toViewLater(bvid: bvid);
try { SmartDialog.showToast(res['msg']);
var res = await UserHttp.toViewLater(bvid: bvid); } catch (err) {
SmartDialog.showToast(res['msg']); SmartDialog.showToast('出错了:${err.toString()}');
} catch (err) { }
SmartDialog.showToast('出错了:${err.toString()}'); },
} minLeadingWidth: 0,
}, leading: const Icon(Icons.watch_later_outlined, size: 19),
minLeadingWidth: 0, title: Text(
leading: const Icon(Icons.watch_later_outlined, size: 19), '稍后再看',
title: Text( style: Theme.of(context).textTheme.titleSmall,
'稍后再看', ),
style: Theme.of(context).textTheme.titleSmall, ),
ListTile(
title: Text(
'分享动态',
style: Theme.of(context).textTheme.titleSmall,
),
leading: const Icon(Icons.share_outlined, size: 19),
onTap: () {
Get.back();
Utils.shareText(
'${HttpString.dynamicShareBaseUrl}/${item.idStr}');
},
minLeadingWidth: 0,
), ),
),
ListTile(
title: Text(
'分享动态',
style: Theme.of(context).textTheme.titleSmall,
),
leading: const Icon(Icons.share_outlined, size: 19),
onTap: () {
Get.back();
Utils.shareText(
'${HttpString.dynamicShareBaseUrl}/${item.idStr}');
},
minLeadingWidth: 0,
),
ListTile(
onTap: () {
Get.back();
SavePanel.toSavePanel(item: item);
},
minLeadingWidth: 0,
leading: const Icon(Icons.save_alt, size: 19),
title: Text('保存动态', style: Theme.of(context).textTheme.titleSmall!),
),
ListTile(
title: Text(
'临时屏蔽:${item.modules.moduleAuthor.name}',
style: Theme.of(context).textTheme.titleSmall,
),
leading: const Icon(Icons.visibility_off_outlined, size: 19),
onTap: () {
Get.back();
DynamicsController dynamicsController =
Get.find<DynamicsController>();
dynamicsController.tempBannedList
.add(item.modules.moduleAuthor.mid);
SmartDialog.showToast(
'已临时屏蔽${item.modules.moduleAuthor.name}(${item.modules.moduleAuthor.mid}),重启恢复');
},
minLeadingWidth: 0,
),
if (item.modules.moduleAuthor.mid == Accounts.main.mid) ...[
ListTile(
onTap: () {
Get.back();
RequestUtils.checkCreatedDyn(id: item.idStr, isManual: true);
},
minLeadingWidth: 0,
leading: Stack(
alignment: Alignment.center,
children: [
const Icon(Icons.shield_outlined, size: 19),
const Icon(Icons.published_with_changes_sharp, size: 12),
],
),
title:
Text('检查动态', style: Theme.of(context).textTheme.titleSmall!),
),
if (onSetTop != null)
ListTile( ListTile(
onTap: () { onTap: () {
Get.back(); Get.back();
onSetTop!(item.modules?.moduleTag?.text != null, item.idStr); SavePanel.toSavePanel(item: item);
}, },
minLeadingWidth: 0, minLeadingWidth: 0,
leading: const Icon(Icons.vertical_align_top, size: 19), leading: const Icon(Icons.save_alt, size: 19),
title: Text( title: Text('保存动态',
'${item.modules?.moduleTag?.text != null ? '取消' : ''}置顶',
style: Theme.of(context).textTheme.titleSmall!), style: Theme.of(context).textTheme.titleSmall!),
), ),
if (onRemove != null)
ListTile( ListTile(
title: Text(
'临时屏蔽:${item.modules?.moduleAuthor?.name}',
style: Theme.of(context).textTheme.titleSmall,
),
leading: const Icon(Icons.visibility_off_outlined, size: 19),
onTap: () { onTap: () {
Get.back(); Get.back();
showDialog( Get.find<DynamicsController>()
context: context, .tempBannedList
builder: (context) => AlertDialog( .add(item.modules!.moduleAuthor!.mid!);
title: const Text('确定删除该动态?'), SmartDialog.showToast(
actions: [ '已临时屏蔽${item.modules?.moduleAuthor?.name}(${item.modules!.moduleAuthor!.mid}),重启恢复');
TextButton(
onPressed: Get.back,
child: Text(
'取消',
style: TextStyle(
color: Theme.of(context).colorScheme.outline,
),
),
),
TextButton(
onPressed: () {
Get.back();
onRemove?.call(item.idStr);
},
child: const Text('确定'),
),
],
),
);
}, },
minLeadingWidth: 0, minLeadingWidth: 0,
leading: Icon(Icons.delete_outline,
color: Theme.of(context).colorScheme.error, size: 19),
title: Text('删除',
style: Theme.of(context)
.textTheme
.titleSmall!
.copyWith(color: Theme.of(context).colorScheme.error)),
), ),
], if (item.modules?.moduleAuthor?.mid == Accounts.main.mid) ...[
if (Accounts.main.isLogin) ListTile(
ListTile( onTap: () {
title: Text( Get.back();
'举报', RequestUtils.checkCreatedDyn(
style: Theme.of(context).textTheme.titleSmall!.copyWith( id: item.idStr, isManual: true);
color: Theme.of(context).colorScheme.error,
),
),
leading: Icon(
Icons.error_outline_outlined,
size: 19,
color: Theme.of(context).colorScheme.error,
),
onTap: () {
Get.back();
autoWrapReportDialog(
context,
ReportOptions.dynamicReport,
(reasonType, reasonDesc, banUid) async {
if (banUid) {
VideoHttp.relationMod(
mid: item.modules.moduleAuthor.mid,
act: 5,
reSrc: 11,
);
}
final res = await Request().post(
'/x/dynamic/feed/dynamic_report/add',
queryParameters: {
'csrf': Accounts.main.csrf,
},
data: {
"accused_uid": item.modules.moduleAuthor.mid,
"dynamic_id": item.idStr,
"reason_type": reasonType,
"reason_desc": reasonType == 0 ? reasonDesc : null,
},
options: Options(
contentType: Headers.formUrlEncodedContentType,
),
);
return res.data as Map;
}, },
); minLeadingWidth: 0,
}, leading: const Stack(
minLeadingWidth: 0, alignment: Alignment.center,
), children: [
const Divider(thickness: 0.1, height: 1), Icon(Icons.shield_outlined, size: 19),
ListTile( Icon(Icons.published_with_changes_sharp, size: 12),
onTap: Get.back, ],
minLeadingWidth: 0, ),
dense: true, title: Text('检查动态',
title: Text( style: Theme.of(context).textTheme.titleSmall!),
'取消', ),
style: TextStyle(color: Theme.of(context).colorScheme.outline), if (onSetTop != null)
textAlign: TextAlign.center, ListTile(
), onTap: () {
Get.back();
onSetTop!(
item.modules?.moduleTag?.text != null, item.idStr);
},
minLeadingWidth: 0,
leading: const Icon(Icons.vertical_align_top, size: 19),
title: Text(
'${item.modules?.moduleTag?.text != null ? '取消' : ''}置顶',
style: Theme.of(context).textTheme.titleSmall!),
),
if (onRemove != null)
ListTile(
onTap: () {
Get.back();
showDialog(
context: context,
builder: (context) => AlertDialog(
title: const Text('确定删除该动态?'),
actions: [
TextButton(
onPressed: Get.back,
child: Text(
'取消',
style: TextStyle(
color: Theme.of(context).colorScheme.outline,
),
),
),
TextButton(
onPressed: () {
Get.back();
onRemove?.call(item.idStr);
},
child: const Text('确定'),
),
],
),
);
},
minLeadingWidth: 0,
leading: Icon(Icons.delete_outline,
color: Theme.of(context).colorScheme.error, size: 19),
title: Text('删除',
style: Theme.of(context).textTheme.titleSmall!.copyWith(
color: Theme.of(context).colorScheme.error)),
),
],
if (Accounts.main.isLogin)
ListTile(
title: Text(
'举报',
style: Theme.of(context).textTheme.titleSmall!.copyWith(
color: Theme.of(context).colorScheme.error,
),
),
leading: Icon(
Icons.error_outline_outlined,
size: 19,
color: Theme.of(context).colorScheme.error,
),
onTap: () {
Get.back();
autoWrapReportDialog(
context,
ReportOptions.dynamicReport,
(reasonType, reasonDesc, banUid) {
if (banUid) {
VideoHttp.relationMod(
mid: item.modules!.moduleAuthor!.mid!,
act: 5,
reSrc: 11,
);
}
return UserHttp.dynamicReport(
mid: item.modules!.moduleAuthor!.mid,
dynId: item.idStr,
reasonType: reasonType,
);
},
);
},
minLeadingWidth: 0,
),
const Divider(thickness: 0.1, height: 1),
ListTile(
onTap: Get.back,
minLeadingWidth: 0,
dense: true,
title: Text(
'取消',
style:
TextStyle(color: Theme.of(context).colorScheme.outline),
textAlign: TextAlign.center,
),
),
],
), ),
], );
), },
); );
} }
} }

View File

@@ -1,4 +1,5 @@
import 'package:PiliPlus/common/widgets/image_save.dart'; import 'package:PiliPlus/common/widgets/image_save.dart';
import 'package:PiliPlus/models/dynamics/result.dart';
import 'package:PiliPlus/utils/extension.dart'; import 'package:PiliPlus/utils/extension.dart';
import 'package:PiliPlus/utils/page_utils.dart'; import 'package:PiliPlus/utils/page_utils.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
@@ -9,7 +10,7 @@ import 'content_panel.dart';
import 'forward_panel.dart'; import 'forward_panel.dart';
class DynamicPanel extends StatelessWidget { class DynamicPanel extends StatelessWidget {
final dynamic item; final DynamicItemModel item;
final String? source; final String? source;
final Function? onRemove; final Function? onRemove;
final Function(List<String>, int)? callback; final Function(List<String>, int)? callback;
@@ -28,6 +29,13 @@ class DynamicPanel extends StatelessWidget {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final authorWidget = AuthorPanel(
item: item,
source: source,
onRemove: onRemove,
isSave: isSave,
onSetTop: onSetTop,
);
return Container( return Container(
decoration: isSave || decoration: isSave ||
(source == 'detail' && (source == 'detail' &&
@@ -46,7 +54,7 @@ class DynamicPanel extends StatelessWidget {
color: Colors.transparent, color: Colors.transparent,
child: InkWell( child: InkWell(
onTap: source == 'detail' && onTap: source == 'detail' &&
[ const {
'DYNAMIC_TYPE_AV', 'DYNAMIC_TYPE_AV',
'DYNAMIC_TYPE_UGC_SEASON', 'DYNAMIC_TYPE_UGC_SEASON',
'DYNAMIC_TYPE_PGC_UNION', 'DYNAMIC_TYPE_PGC_UNION',
@@ -54,59 +62,20 @@ class DynamicPanel extends StatelessWidget {
'DYNAMIC_TYPE_LIVE', 'DYNAMIC_TYPE_LIVE',
'DYNAMIC_TYPE_LIVE_RCMD', 'DYNAMIC_TYPE_LIVE_RCMD',
'DYNAMIC_TYPE_MEDIALIST', 'DYNAMIC_TYPE_MEDIALIST',
].contains(item.type).not }.contains(item.type).not
? null ? null
: () => PageUtils.pushDynDetail(item, 1), : () => PageUtils.pushDynDetail(item, 1),
onLongPress: () { onLongPress: () => _imageSaveDialog(context, authorWidget.morePanel),
if (item.type == 'DYNAMIC_TYPE_AV') {
imageSaveDialog(
context: context,
title: item.modules.moduleDynamic.major.archive.title,
cover: item.modules.moduleDynamic.major.archive.cover,
);
} else if (item.type == 'DYNAMIC_TYPE_UGC_SEASON') {
imageSaveDialog(
context: context,
title: item.modules.moduleDynamic.major.ugcSeason.title,
cover: item.modules.moduleDynamic.major.ugcSeason.cover,
);
} else if (item.type == 'DYNAMIC_TYPE_PGC' ||
item.type == 'DYNAMIC_TYPE_PGC_UNION') {
imageSaveDialog(
context: context,
title: item.modules.moduleDynamic.major.pgc.title,
cover: item.modules.moduleDynamic.major.pgc.cover,
);
} else if (item.type == 'DYNAMIC_TYPE_LIVE_RCMD') {
imageSaveDialog(
context: context,
title: item.modules.moduleDynamic.major.liveRcmd.title,
cover: item.modules.moduleDynamic.major.liveRcmd.cover,
);
} else if (item.type == 'DYNAMIC_TYPE_LIVE') {
imageSaveDialog(
context: context,
title: item.modules.moduleDynamic.major.live.title,
cover: item.modules.moduleDynamic.major.live.cover,
);
}
},
child: Column( child: Column(
mainAxisSize: MainAxisSize.min, mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
Padding( Padding(
padding: const EdgeInsets.fromLTRB(12, 12, 12, 6), padding: const EdgeInsets.fromLTRB(12, 12, 12, 6),
child: AuthorPanel( child: authorWidget,
item: item,
source: source,
onRemove: onRemove,
isSave: isSave,
onSetTop: onSetTop,
),
), ),
if (item!.modules!.moduleDynamic!.desc != null || if (item.modules!.moduleDynamic!.desc != null ||
item!.modules!.moduleDynamic!.major != null) item.modules!.moduleDynamic!.major != null)
content(isSave, context, item, source, callback), content(isSave, context, item, source, callback),
forWard(isSave, item, context, source, callback), forWard(isSave, item, context, source, callback),
const SizedBox(height: 2), const SizedBox(height: 2),
@@ -118,4 +87,43 @@ class DynamicPanel extends StatelessWidget {
), ),
); );
} }
void _imageSaveDialog(
BuildContext context,
Function(BuildContext) morePanel,
) {
late String? title;
late String? cover;
late final major = item.modules?.moduleDynamic?.major;
switch (item.type) {
case 'DYNAMIC_TYPE_AV':
title = major?.archive?.title;
cover = major?.archive?.cover;
break;
case 'DYNAMIC_TYPE_UGC_SEASON':
title = major?.ugcSeason?.title;
cover = major?.ugcSeason?.cover;
break;
case 'DYNAMIC_TYPE_PGC' || 'DYNAMIC_TYPE_PGC_UNION':
title = major?.pgc?.title;
cover = major?.pgc?.cover;
break;
case 'DYNAMIC_TYPE_LIVE_RCMD':
title = major?.liveRcmd?.title;
cover = major?.liveRcmd?.cover;
break;
case 'DYNAMIC_TYPE_LIVE':
title = major?.live?.title;
cover = major?.live?.cover;
break;
default:
morePanel(context);
return;
}
imageSaveDialog(
context: context,
title: title,
cover: cover,
);
}
} }

View File

@@ -125,14 +125,11 @@ class _MemberDynamicsPageState extends State<MemberDynamicsPage>
? LastChildLayoutType.foot ? LastChildLayoutType.foot
: LastChildLayoutType.none; : LastChildLayoutType.none;
}, },
children: (loadingState.response as List) children: loadingState.response!
.map( .map((item) => DynamicPanel(
(item) => DynamicPanel(
item: item, item: item,
onRemove: _memberDynamicController.onRemove, onRemove: _memberDynamicController.onRemove,
onSetTop: _memberDynamicController.onSetTop, onSetTop: _memberDynamicController.onSetTop))
),
)
.toList(), .toList(),
) )
: SliverCrossAxisGroup( : SliverCrossAxisGroup(

View File

@@ -111,7 +111,7 @@ class _SearchDynamicState extends State<SearchDynamic>
? LastChildLayoutType.foot ? LastChildLayoutType.foot
: LastChildLayoutType.none; : LastChildLayoutType.none;
}, },
children: (loadingState.response as List) children: loadingState.response!
.map((item) => DynamicPanel(item: item)) .map((item) => DynamicPanel(item: item))
.toList(), .toList(),
) )