* feat: edit dm filter

* opt: browser

* feat: sb userInfo

* mod: tvPlayUrl
This commit is contained in:
My-Responsitories
2025-11-13 09:36:50 +08:00
committed by GitHub
parent 9754b061dd
commit bca5b0419c
11 changed files with 201 additions and 73 deletions

View File

@@ -7,6 +7,7 @@ Future<void> showConfirmDialog({
dynamic content, dynamic content,
required VoidCallback onConfirm, required VoidCallback onConfirm,
}) { }) {
assert(content is String? || content is Widget);
return showDialog( return showDialog(
context: context, context: context,
builder: (context) { builder: (context) {

View File

@@ -1,26 +1,21 @@
import 'package:PiliPlus/http/api.dart'; import 'package:PiliPlus/http/api.dart';
import 'package:PiliPlus/http/init.dart'; import 'package:PiliPlus/http/init.dart';
import 'package:PiliPlus/http/loading_state.dart';
import 'package:PiliPlus/models/user/danmaku_block.dart'; import 'package:PiliPlus/models/user/danmaku_block.dart';
import 'package:PiliPlus/utils/accounts.dart'; import 'package:PiliPlus/utils/accounts.dart';
import 'package:dio/dio.dart'; import 'package:dio/dio.dart';
class DanmakuFilterHttp { class DanmakuFilterHttp {
static Future danmakuFilter() async { static Future<LoadingState<DanmakuBlockDataModel>> danmakuFilter() async {
var res = await Request().get(Api.danmakuFilter); var res = await Request().get(Api.danmakuFilter);
if (res.data['code'] == 0) { if (res.data['code'] == 0) {
return { return Success(DanmakuBlockDataModel.fromJson(res.data['data']));
'status': true,
'data': DanmakuBlockDataModel.fromJson(res.data['data']),
};
} else { } else {
return { return Error(res.data['message']);
'status': false,
'msg': res.data['message'],
};
} }
} }
static Future danmakuFilterDel({required int ids}) async { static Future<LoadingState<Null>> danmakuFilterDel({required int ids}) async {
var res = await Request().post( var res = await Request().post(
Api.danmakuFilterDel, Api.danmakuFilterDel,
data: { data: {
@@ -30,16 +25,13 @@ class DanmakuFilterHttp {
options: Options(contentType: Headers.formUrlEncodedContentType), options: Options(contentType: Headers.formUrlEncodedContentType),
); );
if (res.data['code'] == 0) { if (res.data['code'] == 0) {
return {'status': true}; return const Success(null);
} else { } else {
return { return Error(res.data['message']);
'status': false,
'msg': res.data['message'],
};
} }
} }
static Future danmakuFilterAdd({ static Future<LoadingState<SimpleRule>> danmakuFilterAdd({
required String filter, required String filter,
required int type, required int type,
}) async { }) async {
@@ -53,15 +45,9 @@ class DanmakuFilterHttp {
options: Options(contentType: Headers.formUrlEncodedContentType), options: Options(contentType: Headers.formUrlEncodedContentType),
); );
if (res.data['code'] == 0) { if (res.data['code'] == 0) {
return { return Success(SimpleRule.fromJson(res.data['data']));
'status': true,
'data': SimpleRule.fromJson(res.data['data']),
};
} else { } else {
return { return Error(res.data['message']);
'status': false,
'msg': res.data['message'],
};
} }
} }
} }

View File

@@ -1068,11 +1068,10 @@ class VideoHttp {
required int playurlType, // ugc 1, pgc 2 required int playurlType, // ugc 1, pgc 2
int? qn, int? qn,
}) async { }) async {
final accessKey = Accounts.accountMode[AccountType.video.index].accessKey; final accessKey = Accounts.get(AccountType.video).accessKey;
final params = { final params = {
'access_key': ?accessKey, 'access_key': ?accessKey,
'actionKey': 'appkey', 'actionKey': 'appkey',
'appkey': Constants.appKey,
'cid': cid, 'cid': cid,
'fourk': 1, 'fourk': 1,
'is_proj': 1, 'is_proj': 1,

View File

@@ -32,10 +32,10 @@ class DanmakuBlockController extends GetxController
Future<void> queryDanmakuFilter() async { Future<void> queryDanmakuFilter() async {
SmartDialog.showLoading(msg: '正在同步弹幕屏蔽规则……'); SmartDialog.showLoading(msg: '正在同步弹幕屏蔽规则……');
var result = await DanmakuFilterHttp.danmakuFilter(); final result = await DanmakuFilterHttp.danmakuFilter();
SmartDialog.dismiss(); SmartDialog.dismiss();
if (result['status']) { if (result.isSuccess) {
DanmakuBlockDataModel data = result['data']; final data = result.data;
rules[0].addAll(data.rule); rules[0].addAll(data.rule);
rules[1].addAll(data.rule1); rules[1].addAll(data.rule1);
rules[2].addAll(data.rule2); rules[2].addAll(data.rule2);
@@ -43,19 +43,19 @@ class DanmakuBlockController extends GetxController
SmartDialog.showToast(data.toast!); SmartDialog.showToast(data.toast!);
} }
} else { } else {
SmartDialog.showToast(result['msg']); result.toast();
} }
} }
Future<void> danmakuFilterDel(int tabIndex, int itemIndex, int id) async { Future<void> danmakuFilterDel(int tabIndex, int itemIndex, int id) async {
SmartDialog.showLoading(msg: '正在删除弹幕屏蔽规则……'); SmartDialog.showLoading(msg: '正在删除弹幕屏蔽规则……');
var result = await DanmakuFilterHttp.danmakuFilterDel(ids: id); final result = await DanmakuFilterHttp.danmakuFilterDel(ids: id);
SmartDialog.dismiss(); SmartDialog.dismiss();
if (result['status']) { if (result.isSuccess) {
rules[tabIndex].removeAt(itemIndex); rules[tabIndex].removeAt(itemIndex);
SmartDialog.showToast('删除成功'); SmartDialog.showToast('删除成功');
} else { } else {
SmartDialog.showToast(result['msg']); result.toast();
} }
} }
@@ -67,17 +67,16 @@ class DanmakuBlockController extends GetxController
filter = Crc32Xz().convert(utf8.encode(filter)).toRadixString(16); filter = Crc32Xz().convert(utf8.encode(filter)).toRadixString(16);
} }
SmartDialog.showLoading(msg: '正在添加弹幕屏蔽规则……'); SmartDialog.showLoading(msg: '正在添加弹幕屏蔽规则……');
var result = await DanmakuFilterHttp.danmakuFilterAdd( final result = await DanmakuFilterHttp.danmakuFilterAdd(
filter: filter, filter: filter,
type: type, type: type,
); );
SmartDialog.dismiss(); SmartDialog.dismiss();
if (result['status']) { if (result.isSuccess) {
SimpleRule rule = result['data']; rules[type].add(result.data);
rules[type].add(rule);
SmartDialog.showToast('添加成功'); SmartDialog.showToast('添加成功');
} else { } else {
SmartDialog.showToast(result['msg']); result.toast();
} }
} }
} }

View File

@@ -78,7 +78,7 @@ class _DanmakuBlockPageState extends State<DanmakuBlockPage> {
); );
} }
Widget tabViewBuilder(int tabIndex, List<SimpleRule> list) { Widget tabViewBuilder(final int tabIndex, List<SimpleRule> list) {
if (list.isEmpty) { if (list.isEmpty) {
return scrollErrorWidget(); return scrollErrorWidget();
} }
@@ -89,31 +89,54 @@ class _DanmakuBlockPageState extends State<DanmakuBlockPage> {
), ),
itemBuilder: (context, itemIndex) { itemBuilder: (context, itemIndex) {
final SimpleRule item = list[itemIndex]; final SimpleRule item = list[itemIndex];
final child = IconButton(
icon: const Icon(Icons.delete_outlined),
onPressed: () => showConfirmDialog(
context: context,
title: '确定删除该规则?',
onConfirm: () => _controller.danmakuFilterDel(
tabIndex,
itemIndex,
item.id,
),
),
);
return ListTile( return ListTile(
title: Text( title: Text(
item.filter, item.filter,
style: Theme.of(context).textTheme.bodyMedium, style: Theme.of(context).textTheme.bodyMedium,
), ),
trailing: IconButton( trailing: tabIndex == 2
icon: const Icon(Icons.delete_outlined), ? child
onPressed: () => showConfirmDialog( : Row(
context: context, mainAxisSize: MainAxisSize.min,
title: '确定删除该规则?', children: [
onConfirm: () => _controller.danmakuFilterDel( IconButton(
tabIndex, icon: const Icon(Icons.edit_outlined),
itemIndex, onPressed: () => _showAddDialog(
item.id, DmBlockType.values[_controller.tabController.index],
), initFilter: item.filter,
), itemIndex: itemIndex,
), itemId: item.id,
),
),
child,
],
),
); );
}, },
); );
} }
void _showAddDialog(DmBlockType type) { void _showAddDialog(
String filter = ''; DmBlockType type, {
String hintText = switch (type) { String initFilter = '',
int? itemIndex,
int? itemId,
}) {
assert((itemIndex == null) == (itemId == null));
String filter = initFilter;
final hintText = switch (type) {
DmBlockType.keyword => '输入过滤的关键词,其它类别请切换标签页后添加', DmBlockType.keyword => '输入过滤的关键词,其它类别请切换标签页后添加',
DmBlockType.regex => '输入//之间的正则表达式,无需包含头尾的"/"', DmBlockType.regex => '输入//之间的正则表达式,无需包含头尾的"/"',
DmBlockType.uid => '输入用户UID', DmBlockType.uid => '输入用户UID',
@@ -123,7 +146,7 @@ class _DanmakuBlockPageState extends State<DanmakuBlockPage> {
context: context, context: context,
builder: (context) { builder: (context) {
return AlertDialog( return AlertDialog(
title: Text('添加新的${type.label}规则'), title: Text('${itemId != null ? "编辑" : "添加新的"}${type.label}规则'),
content: Column( content: Column(
mainAxisSize: MainAxisSize.min, mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
@@ -150,15 +173,24 @@ class _DanmakuBlockPageState extends State<DanmakuBlockPage> {
), ),
TextButton( TextButton(
child: const Text('添加'), child: const Text('添加'),
onPressed: () { onPressed: () async {
if (filter.isNotEmpty) { if (filter != initFilter) {
Get.back(); Get.back();
_controller.danmakuFilterAdd( if (itemId != null) {
await _controller.danmakuFilterDel(
type.index,
itemIndex!,
itemId,
);
}
await _controller.danmakuFilterAdd(
filter: filter, filter: filter,
type: type.index, type: type.index,
); );
} else { } else {
SmartDialog.showToast('输入内容不能为空'); SmartDialog.showToast(
'输入内容${filter.isEmpty ? "不能为空" : "与上次相同"}',
);
} }
}, },
), ),

View File

@@ -78,7 +78,7 @@ class _LoginPageState extends State<LoginPage> {
if (kDebugMode || Utils.isMobile) if (kDebugMode || Utils.isMobile)
TextButton.icon( TextButton.icon(
onPressed: () => PageUtils.launchURL( onPressed: () => PageUtils.launchURL(
_loginPageCtr.codeInfo.value.data.url, 'bilibili://browser?url=${Uri.encodeComponent(_loginPageCtr.codeInfo.value.data.url)}',
mode: LaunchMode.externalNonBrowserApplication, mode: LaunchMode.externalNonBrowserApplication,
), ),
icon: const Icon(Icons.open_in_browser_outlined), icon: const Icon(Icons.open_in_browser_outlined),

View File

@@ -3,9 +3,12 @@ import 'dart:math';
import 'package:PiliPlus/common/widgets/pair.dart'; import 'package:PiliPlus/common/widgets/pair.dart';
import 'package:PiliPlus/http/constants.dart'; import 'package:PiliPlus/http/constants.dart';
import 'package:PiliPlus/http/init.dart'; import 'package:PiliPlus/http/init.dart';
import 'package:PiliPlus/http/loading_state.dart';
import 'package:PiliPlus/models/common/sponsor_block/segment_type.dart'; import 'package:PiliPlus/models/common/sponsor_block/segment_type.dart';
import 'package:PiliPlus/models/common/sponsor_block/skip_type.dart'; import 'package:PiliPlus/models/common/sponsor_block/skip_type.dart';
import 'package:PiliPlus/pages/setting/slide_color_picker.dart'; import 'package:PiliPlus/pages/setting/slide_color_picker.dart';
import 'package:PiliPlus/utils/duration_utils.dart';
import 'package:PiliPlus/utils/num_utils.dart';
import 'package:PiliPlus/utils/page_utils.dart'; import 'package:PiliPlus/utils/page_utils.dart';
import 'package:PiliPlus/utils/storage.dart'; import 'package:PiliPlus/utils/storage.dart';
import 'package:PiliPlus/utils/storage_key.dart'; import 'package:PiliPlus/utils/storage_key.dart';
@@ -35,7 +38,8 @@ class _SponsorBlockPageState extends State<SponsorBlockPage> {
bool _blockToast = Pref.blockToast; bool _blockToast = Pref.blockToast;
String _blockServer = Pref.blockServer; String _blockServer = Pref.blockServer;
bool _blockTrack = Pref.blockTrack; bool _blockTrack = Pref.blockTrack;
final Rx<bool?> _serverStatus = Rx<bool?>(null); final _serverStatus = Rxn<bool>();
final _userInfo = LoadingState<_UserInfo>.loading().obs;
Box setting = GStorage.setting; Box setting = GStorage.setting;
@@ -43,6 +47,7 @@ class _SponsorBlockPageState extends State<SponsorBlockPage> {
void initState() { void initState() {
super.initState(); super.initState();
_checkServerStatus(); _checkServerStatus();
_getUserInfo();
} }
@override @override
@@ -60,6 +65,22 @@ class _SponsorBlockPageState extends State<SponsorBlockPage> {
}); });
} }
Future<void> _getUserInfo() async {
final params = {
'userID': _userId,
'values': '["viewCount","minutesSaved","segmentCount"]',
};
final res = await Request().get(
'$_blockServer/api/userInfo',
queryParameters: params,
);
if (res.statusCode == 200) {
_userInfo.value = Success(_UserInfo.fromJson(res.data));
} else {
_userInfo.value = Error(res.data['message']);
}
}
Widget _blockLimitItem( Widget _blockLimitItem(
ThemeData theme, ThemeData theme,
TextStyle titleStyle, TextStyle titleStyle,
@@ -270,6 +291,37 @@ class _SponsorBlockPageState extends State<SponsorBlockPage> {
}, },
); );
Widget _blockUserInfo(
ThemeData theme,
TextStyle titleStyle,
TextStyle subTitleStyle,
) => Obx(
() {
return ListTile(
dense: true,
onTap: () {
_userInfo.value = LoadingState.loading();
_getUserInfo();
},
title: Text(
'您的信息',
style: titleStyle,
),
subtitle: switch (_userInfo.value) {
Loading() => const SizedBox.shrink(),
Success<_UserInfo>(:final response) => Text(
response.toString(),
style: subTitleStyle,
),
Error(:final errMsg) => Text(
errMsg ?? '服务器错误',
style: subTitleStyle.copyWith(color: theme.colorScheme.error),
),
},
);
},
);
Widget _blockServerItem( Widget _blockServerItem(
ThemeData theme, ThemeData theme,
TextStyle titleStyle, TextStyle titleStyle,
@@ -316,6 +368,8 @@ class _SponsorBlockPageState extends State<SponsorBlockPage> {
_blockServer = _textController.text; _blockServer = _textController.text;
setting.put(SettingBoxKey.blockServer, _blockServer); setting.put(SettingBoxKey.blockServer, _blockServer);
Request.accountManager.blockServer = _blockServer; Request.accountManager.blockServer = _blockServer;
_checkServerStatus();
_getUserInfo();
(context as Element).markNeedsBuild(); (context as Element).markNeedsBuild();
}, },
child: const Text('确定'), child: const Text('确定'),
@@ -461,6 +515,10 @@ class _SponsorBlockPageState extends State<SponsorBlockPage> {
SliverToBoxAdapter(child: _blockToastItem(titleStyle)), SliverToBoxAdapter(child: _blockToastItem(titleStyle)),
sliverDivider, sliverDivider,
SliverToBoxAdapter(child: _blockTrackItem(titleStyle, subTitleStyle)), SliverToBoxAdapter(child: _blockTrackItem(titleStyle, subTitleStyle)),
sliverDivider,
SliverToBoxAdapter(
child: _blockUserInfo(theme, titleStyle, subTitleStyle),
),
dividerL, dividerL,
SliverList.separated( SliverList.separated(
itemCount: _blockSettings.length, itemCount: _blockSettings.length,
@@ -599,3 +657,34 @@ class _SponsorBlockPageState extends State<SponsorBlockPage> {
); );
} }
} }
class _UserInfo {
final int viewCount;
final double minutesSaved;
final int segmentCount;
const _UserInfo({
required this.viewCount,
required this.minutesSaved,
required this.segmentCount,
});
factory _UserInfo.fromJson(Map<String, dynamic> json) => _UserInfo(
viewCount: json['viewCount'],
minutesSaved: (json['minutesSaved'] as num).toDouble(),
segmentCount: json['segmentCount'],
);
@override
String toString() {
String minutes = DurationUtils.formatTimeDuration(
Duration(minutes: minutesSaved.round()),
);
if (minutes.isEmpty) {
minutes = '0分钟';
}
return ('您提交了 ${NumUtils.formatPositiveDecimal(segmentCount)} 片段\n'
'您为大家节省了 ${NumUtils.formatPositiveDecimal(viewCount)} 片段\n'
'($minutes 的生命)');
}
}

View File

@@ -124,6 +124,7 @@ abstract class Accounts {
} }
} }
@pragma("vm:prefer-inline")
static Account get(AccountType key) { static Account get(AccountType key) {
return accountMode[key.index]; return accountMode[key.index];
} }

View File

@@ -110,6 +110,7 @@ class AccountManager extends Interceptor {
Api.ugcUrl, Api.ugcUrl,
Api.pgcUrl, Api.pgcUrl,
Api.pugvUrl, Api.pugvUrl,
Api.tvPlayUrl,
}, },
}; };

View File

@@ -1,6 +1,6 @@
import 'dart:math' show pow; import 'dart:math' show pow;
abstract class DurationUtils { abstract final class DurationUtils {
static String formatDuration(num? seconds) { static String formatDuration(num? seconds) {
if (seconds == null || seconds == 0) { if (seconds == null || seconds == 0) {
return '00:00'; return '00:00';
@@ -30,10 +30,10 @@ abstract class DurationUtils {
return duration; return duration;
} }
static String formatDurationBetween(int startMillis, int endMillis) { static String formatDurationBetween(int startMillis, int endMillis) =>
int diffMillis = endMillis - startMillis; formatTimeDuration(Duration(milliseconds: endMillis - startMillis));
final duration = Duration(milliseconds: diffMillis);
static String formatTimeDuration(Duration duration) {
final inDays = duration.inDays; final inDays = duration.inDays;
final daysLeft = inDays % 365; final daysLeft = inDays % 365;
final years = inDays ~/ 365; final years = inDays ~/ 365;
@@ -42,14 +42,14 @@ abstract class DurationUtils {
final hours = duration.inHours % 24; final hours = duration.inHours % 24;
final minutes = duration.inMinutes % 60; final minutes = duration.inMinutes % 60;
var format = ''; final format = StringBuffer();
if (years > 0) format += '$years年'; if (years > 0) format.write('$years年');
if (months > 0) format += '$months月'; if (months > 0) format.write('$months月');
if (days > 0) format += '$days天'; if (days > 0) format.write('$days天');
if (hours > 0) format += '$hours小时'; if (hours > 0) format.write('$hours小时');
if (minutes > 0) format += '$minutes分钟'; if (minutes > 0) format.write('$minutes分钟');
return format; return format.toString();
} }
} }

View File

@@ -1,7 +1,7 @@
import 'package:flutter/foundation.dart' show kDebugMode, debugPrint; import 'package:flutter/foundation.dart' show kDebugMode, debugPrint;
import 'package:get/get_utils/get_utils.dart'; import 'package:get/get_utils/get_utils.dart';
abstract class NumUtils { abstract final class NumUtils {
static final _numRegExp = RegExp(r'([\d\.]+)([千万亿])?'); static final _numRegExp = RegExp(r'([\d\.]+)([千万亿])?');
static int _getUnit(String? unit) { static int _getUnit(String? unit) {
@@ -59,4 +59,24 @@ abstract class NumUtils {
return number.toString(); return number.toString();
} }
} }
static String formatPositiveDecimal(int number) {
if (number < 1000) return number.toString();
final numStr = number.toString();
final length = numStr.length;
final sb = StringBuffer();
int firstLength = length % 3;
if (firstLength == 0) firstLength = 3;
sb.write(numStr.substring(0, firstLength));
for (int i = firstLength; i < length; i += 3) {
sb
..write(',')
..write(numStr.substring(i, i + 3));
}
return sb.toString();
}
} }