yattee/Model/Player/PlayerSponsorBlock.swift

106 lines
3.0 KiB
Swift
Raw Normal View History

2021-10-23 16:49:45 +00:00
import CoreMedia
2022-06-14 21:20:19 +00:00
import AVFAudio
2021-10-23 16:49:45 +00:00
import Defaults
import Foundation
extension PlayerModel {
func handleSegments(at time: CMTime) {
if let segment = lastSkipped {
if time > .secondsInDefaultTimescale(segment.end + 10) {
2021-10-23 16:49:45 +00:00
resetLastSegment()
}
}
guard let firstSegment = sponsorBlock.segments.first(where: { $0.timeInSegment(time) }) else {
return
}
// find last segment in case they are 2 sec or less after each other
// to avoid multiple skips in a row
var nextSegments = [firstSegment]
while let segment = sponsorBlock.segments.first(where: {
2022-06-14 21:20:19 +00:00
!nextSegments.contains($0) &&
$0.timeInSegment(.secondsInDefaultTimescale(nextSegments.last!.end + 2))
2021-10-23 16:49:45 +00:00
}) {
nextSegments.append(segment)
}
2022-06-14 21:20:19 +00:00
if let segmentToSkip = nextSegments.last, shouldSkip(segmentToSkip, at: time) {
2021-10-23 16:49:45 +00:00
skip(segmentToSkip, at: time)
}
}
private func skip(_ segment: Segment, at time: CMTime) {
2022-06-14 21:20:19 +00:00
if let duration = playerItemDuration, segment.endTime.seconds >= duration.seconds - 3 {
logger.error("segment end time is: \(segment.end) when player item duration is: \(duration.seconds)")
DispatchQueue.main.async { [weak self] in
guard let self = self else {
return
}
self.prepareCurrentItemForHistory(finished: true)
self.pause()
if self.queue.isEmpty {
#if !os(macOS)
try? AVAudioSession.sharedInstance().setActive(false)
#endif
self.resetQueue()
self.hide()
} else {
self.advanceToNextItem()
}
}
2021-10-23 16:49:45 +00:00
return
}
2022-02-16 20:23:11 +00:00
backend.seek(to: segment.endTime)
DispatchQueue.main.async { [weak self] in
self?.lastSkipped = segment
self?.segmentRestorationTime = time
}
2021-10-24 09:16:04 +00:00
logger.info("SponsorBlock skipping to: \(segment.end)")
2021-10-23 16:49:45 +00:00
}
private func shouldSkip(_ segment: Segment, at time: CMTime) -> Bool {
guard isPlaying,
!restoredSegments.contains(segment),
Defaults[.sponsorBlockCategories].contains(segment.category),
2022-06-10 11:26:24 +00:00
segment.end > 4
2021-10-23 16:49:45 +00:00
else {
return false
}
return time.seconds - segment.start < 2 && segment.end - time.seconds > 2
}
func restoreLastSkippedSegment() {
guard let segment = lastSkipped,
let time = segmentRestorationTime
else {
return
}
restoredSegments.append(segment)
2022-02-16 20:23:11 +00:00
backend.seek(to: time)
2021-10-23 16:49:45 +00:00
resetLastSegment()
}
private func resetLastSegment() {
2022-02-16 20:23:11 +00:00
DispatchQueue.main.async { [weak self] in
self?.lastSkipped = nil
self?.segmentRestorationTime = nil
}
2021-10-23 16:49:45 +00:00
}
func resetSegments() {
resetLastSegment()
restoredSegments = []
}
}