opt: segment (#1831)

* opt: segment

* revert: block config
This commit is contained in:
My-Responsitories
2026-02-10 16:13:38 +08:00
committed by GitHub
parent 0460030a2b
commit ed2bd069ee
6 changed files with 134 additions and 93 deletions

View File

@@ -1,19 +1,54 @@
// ignore_for_file: non_constant_identifier_names
import 'package:PiliPlus/common/widgets/pair.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_new/sponsor_block/segment_item.dart';
import 'package:PiliPlus/pages/sponsor_block/block_mixin.dart';
import 'package:PiliPlus/utils/storage_pref.dart';
class SegmentModel {
class SegmentModel implements Comparable<SegmentModel> {
SegmentModel({
required this.UUID,
required this.uuid,
required this.segmentType,
required this.segment,
required this.skipType,
});
String UUID;
SegmentType segmentType;
Pair<int, int> segment;
final String uuid;
final SegmentType segmentType;
final (int, int) segment;
final SkipType skipType;
bool hasSkipped = false;
factory SegmentModel.fromItemModel(
SegmentItemModel model,
BlockConfigMixin? config,
) {
final segmentType = SegmentType.values.byName(model.category);
final segment = (model.segment[0], model.segment[1]);
SkipType skipType;
late bool hasSkipped = false;
if (config != null) {
skipType = config.blockSettings[segmentType.index].second;
if (skipType != SkipType.showOnly) {
if (segment.isEq || segment.length < config.blockLimit) {
skipType = SkipType.showOnly;
}
}
} else {
skipType = Pref.pgcSkipType;
}
return SegmentModel(
uuid: model.uuid,
segmentType: segmentType,
segment: segment,
skipType: skipType,
);
}
@override
int compareTo(SegmentModel other) => segment.$1.compareTo(other.segment.$1);
}
extension IntRecordExt on (int, int) {
bool get isEq => $1 == $2;
int get length => $2 - $1;
bool contains(num other) => $1 <= other && other < $2;
}

View File

@@ -1,6 +1,6 @@
import 'dart:async' show StreamSubscription, Timer;
import 'dart:math' as math;
import 'package:PiliPlus/common/widgets/pair.dart';
import 'package:PiliPlus/common/widgets/progress_bar/segment_progress_bar.dart';
import 'package:PiliPlus/http/loading_state.dart';
import 'package:PiliPlus/http/sponsor_block.dart';
@@ -21,15 +21,17 @@ import 'package:media_kit/media_kit.dart';
mixin BlockConfigMixin {
late final pgcSkipType = Pref.pgcSkipType;
late final enablePgcSkip = pgcSkipType != SkipType.disable;
late final bool enableSponsorBlock = Pref.enableSponsorBlock;
late final bool enableBlock = enableSponsorBlock || enablePgcSkip;
late final double blockLimit = Pref.blockLimit;
late final enableSponsorBlock = Pref.enableSponsorBlock;
late final enableBlock = enableSponsorBlock || enablePgcSkip;
late final blockColor = Pref.blockColor;
late final blockLimit = Pref.blockLimit;
late final blockSettings = Pref.blockSettings;
late final List<Color> blockColor = Pref.blockColor;
late final Set<String> enableList = blockSettings
late final enableList = blockSettings
.where((item) => item.second != SkipType.disable)
.map((item) => item.first.name)
.toSet();
Color _getColor(SegmentType segment) => blockColor[segment.index];
}
mixin BlockMixin on GetxController {
@@ -37,14 +39,12 @@ mixin BlockMixin on GetxController {
BlockConfigMixin get blockConfig;
StreamSubscription<Duration>? _blockListener;
StreamSubscription<Duration>? get blockListener => _blockListener;
Color _getBlockColor(SegmentType segment) =>
blockConfig.blockColor[segment.index];
late final List<SegmentModel> _segmentList = <SegmentModel>[];
late final RxList<Segment> segmentProgressList = <Segment>[].obs;
Timer? _skipTimer;
late final listKey = GlobalKey<AnimatedListState>();
late final List listData = [];
late final List<Object> listData = [];
RxString? get videoLabel => null;
Player? get player;
@@ -89,8 +89,7 @@ mixin BlockMixin on GetxController {
// debugPrint(
// '${position.inSeconds},,${item.segment.first},,${item.segment.second},,${item.skipType.name},,${item.hasSkipped}');
// }
if (msPos <= item.segment.first &&
item.segment.first <= msPos + 1000) {
if (msPos <= item.segment.$1 && item.segment.$1 <= msPos + 1000) {
switch (item.skipType) {
case SkipType.alwaysSkip:
onSkip(item, isSeek: false);
@@ -118,7 +117,7 @@ mixin BlockMixin on GetxController {
Future<void> handleSBData(List<SegmentItemModel> list) async {
if (list.isNotEmpty) {
try {
Future? future;
Future<void>? future;
final duration = list.first.videoDuration ?? timeLength!;
// segmentList
_segmentList.addAll(
@@ -130,41 +129,19 @@ mixin BlockMixin on GetxController {
)
.map(
(item) {
final segmentType = SegmentType.values.byName(item.category);
if (item.segment[0] == 0 && item.segment[1] == 0) {
videoLabel?.value +=
'${videoLabel!.value.isNotEmpty ? '/' : ''}${segmentType.title}';
}
SkipType skipType;
if (isBlock) {
skipType =
blockConfig.blockSettings[segmentType.index].second;
if (skipType != SkipType.showOnly) {
if (item.segment[1] == item.segment[0] ||
item.segment[1] - item.segment[0] <
blockConfig.blockLimit) {
skipType = SkipType.showOnly;
}
}
} else {
skipType = blockConfig.pgcSkipType;
}
final segmentModel = SegmentModel(
UUID: item.uuid,
segmentType: segmentType,
segment: Pair(
first: item.segment[0],
second: item.segment[1],
),
skipType: skipType,
final segmentModel = SegmentModel.fromItemModel(
item,
isBlock ? blockConfig : null,
);
if (segmentModel.segment == const (0, 0)) {
videoLabel?.value +=
'${videoLabel!.value.isNotEmpty ? '/' : ''}${segmentModel.segmentType.title}';
}
if (_blockListener == null && autoPlay && player != null) {
final currPos = currPosInMilliseconds;
if (currPos >= segmentModel.segment.first &&
currPos < segmentModel.segment.second) {
if (segmentModel.segment.contains(currPos)) {
_lastBlockPos = currPos;
switch (segmentModel.skipType) {
@@ -182,7 +159,7 @@ mixin BlockMixin on GetxController {
return true;
}
return false;
});
}, orElse: () => false);
}
break;
case SkipType.skipManually:
@@ -202,12 +179,12 @@ mixin BlockMixin on GetxController {
// _segmentProgressList
segmentProgressList.addAll(
_segmentList.map((e) {
double start = (e.segment.first / duration).clamp(0.0, 1.0);
double end = (e.segment.second / duration).clamp(0.0, 1.0);
double start = (e.segment.$1 / duration).clamp(0.0, 1.0);
double end = (e.segment.$2 / duration).clamp(0.0, 1.0);
return Segment(
start: start,
end: end,
color: _getBlockColor(e.segmentType),
color: blockConfig._getColor(e.segmentType),
);
}),
);
@@ -222,7 +199,7 @@ mixin BlockMixin on GetxController {
}
}
void onAddItem(dynamic item) {
void onAddItem(Object item) {
if (listData.contains(item)) return;
listData.insert(0, item);
listKey.currentState?.insertItem(0);
@@ -233,7 +210,7 @@ mixin BlockMixin on GetxController {
});
}
void onRemoveItem(int index, item) {
void onRemoveItem(int index, Object item) {
EasyThrottle.throttle(
'onRemoveItem',
const Duration(milliseconds: 500),
@@ -252,7 +229,7 @@ mixin BlockMixin on GetxController {
);
}
Widget buildItem(dynamic item, Animation<double> animation) =>
Widget buildItem(Object item, Animation<double> animation) =>
throw UnimplementedError();
void _stopSkipTimer() {
@@ -264,6 +241,15 @@ mixin BlockMixin on GetxController {
Future<void>? seekTo(Duration duration, {required bool isSeek});
void _skipToast(SegmentModel item) {
if (autoPlay && Pref.blockToast) {
_showBlockToast('已跳过${item.segmentType.shortTitle}片段');
}
if (isBlock && Pref.blockTrack) {
SponsorBlock.viewedVideoSponsorTime(item.uuid);
}
}
Future<void> onSkip(
SegmentModel item, {
bool isSkip = true,
@@ -271,16 +257,11 @@ mixin BlockMixin on GetxController {
}) async {
try {
await seekTo(
Duration(milliseconds: item.segment.second),
Duration(milliseconds: item.segment.$2),
isSeek: isSeek,
);
if (isSkip) {
if (autoPlay && Pref.blockToast) {
_showBlockToast('已跳过${item.segmentType.shortTitle}片段');
}
if (isBlock && Pref.blockTrack) {
SponsorBlock.viewedVideoSponsorTime(item.UUID);
}
_skipToast(item);
} else {
_showBlockToast('已跳至${item.segmentType.shortTitle}');
}
@@ -316,7 +297,7 @@ mixin BlockMixin on GetxController {
title: const Text('赞成票', style: TextStyle(fontSize: 14)),
onTap: () {
Get.back();
_doVote(segment.UUID, 1);
_doVote(segment.uuid, 1);
},
),
ListTile(
@@ -324,7 +305,7 @@ mixin BlockMixin on GetxController {
title: const Text('反对票', style: TextStyle(fontSize: 14)),
onTap: () {
Get.back();
_doVote(segment.UUID, 0);
_doVote(segment.uuid, 0);
},
),
ListTile(
@@ -363,7 +344,7 @@ mixin BlockMixin on GetxController {
onTap: () {
Get.back();
SponsorBlock.voteOnSponsorTime(
uuid: segment.UUID,
uuid: segment.uuid,
category: item,
).then((i) {
SmartDialog.showToast(
@@ -381,7 +362,7 @@ mixin BlockMixin on GetxController {
width: 10,
decoration: BoxDecoration(
shape: BoxShape.circle,
color: _getBlockColor(item),
color: blockConfig._getColor(item),
),
),
style: const TextStyle(fontSize: 14, height: 1),
@@ -431,7 +412,7 @@ mixin BlockMixin on GetxController {
width: 10,
decoration: BoxDecoration(
shape: BoxShape.circle,
color: _getBlockColor(item.segmentType),
color: blockConfig._getColor(item.segmentType),
),
),
style: const TextStyle(fontSize: 14, height: 1),
@@ -445,7 +426,7 @@ mixin BlockMixin on GetxController {
),
contentPadding: const EdgeInsets.only(left: 16, right: 8),
subtitle: Text(
'${DurationUtils.formatDuration(item.segment.first / 1000)}${DurationUtils.formatDuration(item.segment.second / 1000)}',
'${DurationUtils.formatDuration(item.segment.$1 / 1000)}${DurationUtils.formatDuration(item.segment.$2 / 1000)}',
style: const TextStyle(fontSize: 13),
),
trailing: Row(
@@ -455,7 +436,7 @@ mixin BlockMixin on GetxController {
item.skipType.label,
style: const TextStyle(fontSize: 13),
),
if (item.segment.second != 0)
if (item.segment.$2 != 0)
SizedBox(
width: 36,
height: 36,
@@ -514,6 +495,30 @@ mixin BlockMixin on GetxController {
segmentProgressList.clear();
}
Duration? getFirstSegment([int pos = 0]) {
for (var i in _segmentList..sort()) {
final (start, end) = i.segment;
if (start == end) {
continue;
} else if (start - pos < 100) {
if (switch (i.skipType) {
.alwaysSkip => true,
.skipOnce => !i.hasSkipped,
_ => false,
}) {
_skipToast(i);
pos = math.max(pos, i.segment.$2);
}
} else {
break;
}
}
if (pos != 0) {
return Duration(milliseconds: pos);
}
return null;
}
@override
void onClose() {
_stopSkipTimer();

View File

@@ -31,7 +31,7 @@ class _SponsorBlockPageState extends State<SponsorBlockPage> {
final _url = 'https://github.com/hanydd/BilibiliSponsorBlock';
final _textController = TextEditingController();
double _blockLimit = Pref.blockLimit;
final List<Pair<SegmentType, SkipType>> _blockSettings = Pref.blockSettings;
final _blockSettings = Pref.blockSettings;
final List<Color> _blockColor = Pref.blockColor;
String _userId = Pref.blockUserID;
bool _blockToast = Pref.blockToast;

View File

@@ -508,7 +508,7 @@ class VideoDetailController extends GetxController
@override
Widget buildItem(dynamic item, Animation<double> animation) {
Widget buildItem(Object item, Animation<double> animation) {
final theme = Get.theme;
return Align(
alignment: Alignment.centerLeft,
@@ -536,7 +536,7 @@ class VideoDetailController extends GetxController
fontSize: 14,
text: item is SegmentModel
? '跳过: ${item.segmentType.shortTitle}'
: '上次看到第${item + 1}P点击跳转',
: '上次看到第${(item as int) + 1}P点击跳转',
onTap: (_) {
if (item is int) {
try {
@@ -679,6 +679,10 @@ class VideoDetailController extends GetxController
Volume? volume,
}) async {
final onlyPlayAudio = plPlayerController.onlyPlayAudio.value;
Duration? seek = seekToTime ?? defaultST ?? playedTime;
if (seek == null || seek == Duration.zero) {
seek = getFirstSegment();
}
await plPlayerController.setDataSource(
DataSource(
videoSource: isFileSource
@@ -695,7 +699,7 @@ class VideoDetailController extends GetxController
'referer': HttpString.baseUrl,
},
),
seekTo: seekToTime ?? defaultST ?? playedTime,
seekTo: seek,
duration:
duration ??
(data.timeLength == null
@@ -803,16 +807,11 @@ class VideoDetailController extends GetxController
volume = data.volume;
final progress = args['progress'];
final progress = args.remove('progress');
if (progress != null) {
this.defaultST = Duration(milliseconds: progress);
args['progress'] = null;
} else {
this.defaultST =
defaultST ??
(data.lastPlayTime == null
? Duration.zero
: Duration(milliseconds: data.lastPlayTime!));
} else if (defaultST == null && data.lastPlayTime != null) {
this.defaultST = Duration(milliseconds: data.lastPlayTime!);
}
if (!isUgc && !fromReset && plPlayerController.enablePgcSkip) {

View File

@@ -1,3 +1,4 @@
import 'dart:async';
import 'dart:math';
import 'package:PiliPlus/common/widgets/button/icon_button.dart';
@@ -473,14 +474,14 @@ class _PostPanelState extends State<PostPanel>
await videoCtr.play();
}
final delay = start - seek;
if (delay > 0) {
await Future.delayed(Duration(milliseconds: delay));
}
videoCtr.seek(
Duration(
milliseconds: (item.segment.second * 1000).round(),
),
Future<void> seekTo() => videoCtr.seek(
Duration(milliseconds: (item.segment.second * 1000).round()),
);
if (delay > 0) {
Timer(Duration(milliseconds: delay), seekTo);
} else {
seekTo();
}
}
},
),

View File

@@ -39,6 +39,7 @@ import 'package:PiliPlus/utils/storage_key.dart';
import 'package:PiliPlus/utils/utils.dart';
import 'package:crypto/crypto.dart';
import 'package:flex_seed_scheme/flex_seed_scheme.dart' show FlexSchemeVariant;
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:get/get.dart';
import 'package:hive/hive.dart';
@@ -324,7 +325,7 @@ abstract final class Pref {
);
static bool get blockTrack =>
_setting.get(SettingBoxKey.blockTrack, defaultValue: true);
_setting.get(SettingBoxKey.blockTrack, defaultValue: !kDebugMode);
static bool get checkDynamic =>
_setting.get(SettingBoxKey.checkDynamic, defaultValue: true);