mirror of
https://github.com/bggRGjQaUbCoE/PiliPlus.git
synced 2026-05-31 08:08:19 +08:00
tweaks (#1738)
* feat: edit dm filter * opt: browser * feat: sb userInfo * mod: tvPlayUrl
This commit is contained in:
committed by
GitHub
parent
9754b061dd
commit
bca5b0419c
@@ -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) {
|
||||||
|
|||||||
@@ -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'],
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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,
|
||||||
|
|||||||
@@ -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();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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,12 +89,7 @@ class _DanmakuBlockPageState extends State<DanmakuBlockPage> {
|
|||||||
),
|
),
|
||||||
itemBuilder: (context, itemIndex) {
|
itemBuilder: (context, itemIndex) {
|
||||||
final SimpleRule item = list[itemIndex];
|
final SimpleRule item = list[itemIndex];
|
||||||
return ListTile(
|
final child = IconButton(
|
||||||
title: Text(
|
|
||||||
item.filter,
|
|
||||||
style: Theme.of(context).textTheme.bodyMedium,
|
|
||||||
),
|
|
||||||
trailing: IconButton(
|
|
||||||
icon: const Icon(Icons.delete_outlined),
|
icon: const Icon(Icons.delete_outlined),
|
||||||
onPressed: () => showConfirmDialog(
|
onPressed: () => showConfirmDialog(
|
||||||
context: context,
|
context: context,
|
||||||
@@ -105,15 +100,43 @@ class _DanmakuBlockPageState extends State<DanmakuBlockPage> {
|
|||||||
item.id,
|
item.id,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
);
|
||||||
|
return ListTile(
|
||||||
|
title: Text(
|
||||||
|
item.filter,
|
||||||
|
style: Theme.of(context).textTheme.bodyMedium,
|
||||||
|
),
|
||||||
|
trailing: tabIndex == 2
|
||||||
|
? child
|
||||||
|
: Row(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
IconButton(
|
||||||
|
icon: const Icon(Icons.edit_outlined),
|
||||||
|
onPressed: () => _showAddDialog(
|
||||||
|
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 ? "不能为空" : "与上次相同"}',
|
||||||
|
);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -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),
|
||||||
|
|||||||
@@ -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 的生命)');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -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];
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -110,6 +110,7 @@ class AccountManager extends Interceptor {
|
|||||||
Api.ugcUrl,
|
Api.ugcUrl,
|
||||||
Api.pgcUrl,
|
Api.pgcUrl,
|
||||||
Api.pugvUrl,
|
Api.pugvUrl,
|
||||||
|
Api.tvPlayUrl,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -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();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user