mirror of
https://github.com/cranci1/Sora.git
synced 2026-04-21 00:22:12 +00:00
* Aniskip logic and basic buttons * good fuckin enough for now * im callin good enough * bug fix * its something * hallelujah * Update SearchView.swift * made subs go up the progress bar if it is showing --------- Co-authored-by: ibro <54913038+xibrox@users.noreply.github.com> Co-authored-by: Seiike <122684677+Seeike@users.noreply.github.com>
This commit is contained in:
parent
0ad4659d2c
commit
5d076e0cf7
7 changed files with 435 additions and 80 deletions
|
|
@ -101,4 +101,51 @@ class AniListMutation {
|
||||||
|
|
||||||
task.resume()
|
task.resume()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func fetchMalID(animeId: Int, completion: @escaping (Result<Int, Error>) -> Void) {
|
||||||
|
let query = """
|
||||||
|
query ($id: Int) {
|
||||||
|
Media(id: $id) {
|
||||||
|
idMal
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
let variables: [String: Any] = ["id": animeId]
|
||||||
|
let requestBody: [String: Any] = [
|
||||||
|
"query": query,
|
||||||
|
"variables": variables
|
||||||
|
]
|
||||||
|
guard let jsonData = try? JSONSerialization.data(withJSONObject: requestBody, options: []) else {
|
||||||
|
completion(.failure(NSError(domain: "", code: -1,
|
||||||
|
userInfo: [NSLocalizedDescriptionKey: "Failed to serialize GraphQL request"])))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var request = URLRequest(url: apiURL)
|
||||||
|
request.httpMethod = "POST"
|
||||||
|
request.setValue("application/json", forHTTPHeaderField: "Content-Type")
|
||||||
|
// no auth required for read
|
||||||
|
request.httpBody = jsonData
|
||||||
|
|
||||||
|
URLSession.shared.dataTask(with: request) { data, resp, error in
|
||||||
|
if let e = error {
|
||||||
|
return completion(.failure(e))
|
||||||
|
}
|
||||||
|
guard let data = data,
|
||||||
|
let json = try? JSONDecoder().decode(AniListMediaResponse.self, from: data),
|
||||||
|
let mal = json.data.Media?.idMal else {
|
||||||
|
return completion(.failure(NSError(domain: "", code: -1,
|
||||||
|
userInfo: [NSLocalizedDescriptionKey: "Failed to decode AniList response or idMal missing"])))
|
||||||
|
}
|
||||||
|
completion(.success(mal))
|
||||||
|
}.resume()
|
||||||
|
}
|
||||||
|
|
||||||
|
private struct AniListMediaResponse: Decodable {
|
||||||
|
struct DataField: Decodable {
|
||||||
|
struct Media: Decodable { let idMal: Int? }
|
||||||
|
let Media: Media?
|
||||||
|
}
|
||||||
|
let data: DataField
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -44,3 +44,30 @@ class VolumeViewModel: ObservableObject {
|
||||||
@Published var value: Double = 0.0
|
@Published var value: Double = 0.0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class SliderViewModel: ObservableObject {
|
||||||
|
@Published var sliderValue: Double = 0.0
|
||||||
|
@Published var introSegments: [ClosedRange<Double>] = []
|
||||||
|
@Published var outroSegments: [ClosedRange<Double>] = []
|
||||||
|
}
|
||||||
|
|
||||||
|
struct AniListMediaResponse: Decodable {
|
||||||
|
struct DataField: Decodable {
|
||||||
|
struct Media: Decodable { let idMal: Int? }
|
||||||
|
let Media: Media?
|
||||||
|
}
|
||||||
|
let data: DataField
|
||||||
|
}
|
||||||
|
|
||||||
|
struct AniSkipResponse: Decodable {
|
||||||
|
struct Result: Decodable {
|
||||||
|
struct Interval: Decodable {
|
||||||
|
let startTime: Double
|
||||||
|
let endTime: Double
|
||||||
|
}
|
||||||
|
let interval: Interval
|
||||||
|
let skipType: String
|
||||||
|
}
|
||||||
|
let found: Bool
|
||||||
|
let results: [Result]
|
||||||
|
let statusCode: Int
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -18,6 +18,10 @@ struct MusicProgressSlider<T: BinaryFloatingPoint>: View {
|
||||||
let emptyColor: Color
|
let emptyColor: Color
|
||||||
let height: CGFloat
|
let height: CGFloat
|
||||||
let onEditingChanged: (Bool) -> Void
|
let onEditingChanged: (Bool) -> Void
|
||||||
|
let introSegments: [ClosedRange<T>] // Changed
|
||||||
|
let outroSegments: [ClosedRange<T>] // Changed
|
||||||
|
let introColor: Color
|
||||||
|
let outroColor: Color
|
||||||
|
|
||||||
@State private var localRealProgress: T = 0
|
@State private var localRealProgress: T = 0
|
||||||
@State private var localTempProgress: T = 0
|
@State private var localTempProgress: T = 0
|
||||||
|
|
@ -26,10 +30,38 @@ struct MusicProgressSlider<T: BinaryFloatingPoint>: View {
|
||||||
var body: some View {
|
var body: some View {
|
||||||
GeometryReader { bounds in
|
GeometryReader { bounds in
|
||||||
ZStack {
|
ZStack {
|
||||||
VStack (spacing: 8) {
|
VStack(spacing: 8) {
|
||||||
ZStack(alignment: .center) {
|
ZStack(alignment: .center) {
|
||||||
Capsule()
|
ZStack(alignment: .center) {
|
||||||
.fill(emptyColor)
|
// Intro Segments
|
||||||
|
ForEach(introSegments, id: \.self) { segment in
|
||||||
|
HStack(spacing: 0) {
|
||||||
|
Spacer()
|
||||||
|
.frame(width: bounds.size.width * CGFloat(segment.lowerBound))
|
||||||
|
Rectangle()
|
||||||
|
.fill(introColor.opacity(0.5))
|
||||||
|
.frame(width: bounds.size.width * CGFloat(segment.upperBound - segment.lowerBound))
|
||||||
|
Spacer()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Outro Segments
|
||||||
|
ForEach(outroSegments, id: \.self) { segment in
|
||||||
|
HStack(spacing: 0) {
|
||||||
|
Spacer()
|
||||||
|
.frame(width: bounds.size.width * CGFloat(segment.lowerBound))
|
||||||
|
Rectangle()
|
||||||
|
.fill(outroColor.opacity(0.5))
|
||||||
|
.frame(width: bounds.size.width * CGFloat(segment.upperBound - segment.lowerBound))
|
||||||
|
Spacer()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Rest of the existing code...
|
||||||
|
Capsule()
|
||||||
|
.fill(emptyColor)
|
||||||
|
}
|
||||||
|
|
||||||
Capsule()
|
Capsule()
|
||||||
.fill(isActive ? activeFillColor : fillColor)
|
.fill(isActive ? activeFillColor : fillColor)
|
||||||
.mask({
|
.mask({
|
||||||
|
|
|
||||||
|
|
@ -11,13 +11,6 @@ import AVKit
|
||||||
import SwiftUI
|
import SwiftUI
|
||||||
import AVFoundation
|
import AVFoundation
|
||||||
import MediaPlayer
|
import MediaPlayer
|
||||||
|
|
||||||
// MARK: - SliderViewModel
|
|
||||||
|
|
||||||
class SliderViewModel: ObservableObject {
|
|
||||||
@Published var sliderValue: Double = 0.0
|
|
||||||
}
|
|
||||||
|
|
||||||
// MARK: - CustomMediaPlayerViewController
|
// MARK: - CustomMediaPlayerViewController
|
||||||
|
|
||||||
class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDelegate {
|
class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDelegate {
|
||||||
|
|
@ -72,7 +65,7 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
var landscapeButtonHiddenConstraints: [NSLayoutConstraint] = []
|
var landscapeButtonHiddenConstraints: [NSLayoutConstraint] = []
|
||||||
var currentMarqueeConstraints: [NSLayoutConstraint] = []
|
var currentMarqueeConstraints: [NSLayoutConstraint] = []
|
||||||
private var currentMenuButtonTrailing: NSLayoutConstraint!
|
private var currentMenuButtonTrailing: NSLayoutConstraint!
|
||||||
|
|
||||||
|
|
||||||
var subtitleForegroundColor: String = "white"
|
var subtitleForegroundColor: String = "white"
|
||||||
var subtitleBackgroundEnabled: Bool = true
|
var subtitleBackgroundEnabled: Bool = true
|
||||||
|
|
@ -115,13 +108,22 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
var watchNextButtonControlsConstraints: [NSLayoutConstraint] = []
|
var watchNextButtonControlsConstraints: [NSLayoutConstraint] = []
|
||||||
var isControlsVisible = false
|
var isControlsVisible = false
|
||||||
|
|
||||||
var subtitleBottomConstraint: NSLayoutConstraint?
|
private var subtitleBottomToSliderConstraint: NSLayoutConstraint?
|
||||||
|
private var subtitleBottomToSafeAreaConstraint: NSLayoutConstraint?
|
||||||
var subtitleBottomPadding: CGFloat = 10.0 {
|
var subtitleBottomPadding: CGFloat = 10.0 {
|
||||||
didSet {
|
didSet {
|
||||||
updateSubtitleLabelConstraints()
|
updateSubtitleLabelConstraints()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private var malID: Int?
|
||||||
|
private var skipIntervals: (op: CMTimeRange?, ed: CMTimeRange?) = (nil, nil)
|
||||||
|
|
||||||
|
private var skipIntroButton: UIButton!
|
||||||
|
private var skipOutroButton: UIButton!
|
||||||
|
private let skipButtonBaseAlpha: CGFloat = 0.9
|
||||||
|
@Published var segments: [ClosedRange<Double>] = []
|
||||||
|
|
||||||
private var playerItemKVOContext = 0
|
private var playerItemKVOContext = 0
|
||||||
private var loadedTimeRangesObservation: NSKeyValueObservation?
|
private var loadedTimeRangesObservation: NSKeyValueObservation?
|
||||||
private var playerTimeControlStatusObserver: NSKeyValueObservation?
|
private var playerTimeControlStatusObserver: NSKeyValueObservation?
|
||||||
|
|
@ -131,7 +133,7 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
private var dimButtonToSlider: NSLayoutConstraint!
|
private var dimButtonToSlider: NSLayoutConstraint!
|
||||||
private var dimButtonToRight: NSLayoutConstraint!
|
private var dimButtonToRight: NSLayoutConstraint!
|
||||||
private var dimButtonTimer: Timer?
|
private var dimButtonTimer: Timer?
|
||||||
|
|
||||||
private lazy var controlsToHide: [UIView] = [
|
private lazy var controlsToHide: [UIView] = [
|
||||||
dismissButton,
|
dismissButton,
|
||||||
playPauseButton,
|
playPauseButton,
|
||||||
|
|
@ -146,7 +148,7 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
watchNextButton,
|
watchNextButton,
|
||||||
volumeSliderHostingView!
|
volumeSliderHostingView!
|
||||||
]
|
]
|
||||||
|
|
||||||
private var originalHiddenStates: [UIView: Bool] = [:]
|
private var originalHiddenStates: [UIView: Bool] = [:]
|
||||||
|
|
||||||
private var volumeObserver: NSKeyValueObservation?
|
private var volumeObserver: NSKeyValueObservation?
|
||||||
|
|
@ -224,9 +226,23 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
setupMenuButton()
|
setupMenuButton()
|
||||||
setupMarqueeLabel()
|
setupMarqueeLabel()
|
||||||
setupSkip85Button()
|
setupSkip85Button()
|
||||||
|
setupSkipButtons()
|
||||||
addTimeObserver()
|
addTimeObserver()
|
||||||
startUpdateTimer()
|
startUpdateTimer()
|
||||||
setupAudioSession()
|
setupAudioSession()
|
||||||
|
updateSkipButtonsVisibility()
|
||||||
|
|
||||||
|
|
||||||
|
AniListMutation().fetchMalID(animeId: aniListID) { [weak self] result in
|
||||||
|
switch result {
|
||||||
|
case .success(let mal):
|
||||||
|
self?.malID = mal
|
||||||
|
self?.fetchSkipTimes(type: "op")
|
||||||
|
self?.fetchSkipTimes(type: "ed")
|
||||||
|
case .failure(let error):
|
||||||
|
Logger.shared.log("⚠️ Unable to fetch MAL ID: \(error)",type:"Error")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
controlsToHide.forEach { originalHiddenStates[$0] = $0.isHidden }
|
controlsToHide.forEach { originalHiddenStates[$0] = $0.isHidden }
|
||||||
|
|
||||||
|
|
@ -361,10 +377,10 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
// 1) reveal the quality button
|
// 1) reveal the quality button
|
||||||
self.qualityButton.isHidden = false
|
self.qualityButton.isHidden = false
|
||||||
self.qualityButton.menu = self.qualitySelectionMenu()
|
self.qualityButton.menu = self.qualitySelectionMenu()
|
||||||
|
|
||||||
// 2) update the trailing constraint for the menuButton
|
// 2) update the trailing constraint for the menuButton
|
||||||
self.updateMenuButtonConstraints()
|
self.updateMenuButtonConstraints()
|
||||||
|
|
||||||
// 3) animate the shift
|
// 3) animate the shift
|
||||||
UIView.animate(withDuration: 0.25, delay: 0, options: .curveEaseInOut) {
|
UIView.animate(withDuration: 0.25, delay: 0, options: .curveEaseInOut) {
|
||||||
self.view.layoutIfNeeded()
|
self.view.layoutIfNeeded()
|
||||||
|
|
@ -453,7 +469,7 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
playPauseTap.delaysTouchesBegan = false
|
playPauseTap.delaysTouchesBegan = false
|
||||||
playPauseTap.delegate = self
|
playPauseTap.delegate = self
|
||||||
playPauseButton.addGestureRecognizer(playPauseTap)
|
playPauseButton.addGestureRecognizer(playPauseTap)
|
||||||
|
|
||||||
|
|
||||||
playPauseButton.addGestureRecognizer(playPauseTap)
|
playPauseButton.addGestureRecognizer(playPauseTap)
|
||||||
controlsContainerView.addSubview(playPauseButton)
|
controlsContainerView.addSubview(playPauseButton)
|
||||||
|
|
@ -514,7 +530,11 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
|
introSegments: sliderViewModel.introSegments, // Added
|
||||||
|
outroSegments: sliderViewModel.outroSegments, // Added
|
||||||
|
introColor: .yellow, // Add your colors here
|
||||||
|
outroColor: .yellow // Or use settings.accentColor
|
||||||
)
|
)
|
||||||
|
|
||||||
sliderHostingController = UIHostingController(rootView: sliderView)
|
sliderHostingController = UIHostingController(rootView: sliderView)
|
||||||
|
|
@ -669,15 +689,24 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
view.addSubview(subtitleLabel)
|
view.addSubview(subtitleLabel)
|
||||||
subtitleLabel.translatesAutoresizingMaskIntoConstraints = false
|
subtitleLabel.translatesAutoresizingMaskIntoConstraints = false
|
||||||
|
|
||||||
subtitleBottomConstraint = subtitleLabel.bottomAnchor.constraint(equalTo: view.safeAreaLayoutGuide.bottomAnchor, constant: -subtitleBottomPadding)
|
subtitleBottomToSliderConstraint = subtitleLabel.bottomAnchor.constraint(
|
||||||
|
equalTo: sliderHostingController!.view.topAnchor,
|
||||||
|
constant: -20
|
||||||
|
)
|
||||||
|
|
||||||
|
subtitleBottomToSafeAreaConstraint = subtitleLabel.bottomAnchor.constraint(
|
||||||
|
equalTo: view.safeAreaLayoutGuide.bottomAnchor,
|
||||||
|
constant: -subtitleBottomPadding
|
||||||
|
)
|
||||||
|
|
||||||
NSLayoutConstraint.activate([
|
NSLayoutConstraint.activate([
|
||||||
subtitleLabel.centerXAnchor.constraint(equalTo: view.centerXAnchor),
|
subtitleLabel.centerXAnchor.constraint(equalTo: view.centerXAnchor),
|
||||||
subtitleBottomConstraint!,
|
|
||||||
subtitleLabel.leadingAnchor.constraint(greaterThanOrEqualTo: view.leadingAnchor, constant: 36),
|
subtitleLabel.leadingAnchor.constraint(greaterThanOrEqualTo: view.leadingAnchor, constant: 36),
|
||||||
subtitleLabel.trailingAnchor.constraint(lessThanOrEqualTo: view.trailingAnchor, constant: -36)
|
subtitleLabel.trailingAnchor.constraint(lessThanOrEqualTo: view.trailingAnchor, constant: -36)
|
||||||
])
|
])
|
||||||
|
|
||||||
|
subtitleBottomToSafeAreaConstraint?.isActive = true
|
||||||
|
|
||||||
topSubtitleLabel = UILabel()
|
topSubtitleLabel = UILabel()
|
||||||
topSubtitleLabel.textAlignment = .center
|
topSubtitleLabel.textAlignment = .center
|
||||||
topSubtitleLabel.numberOfLines = 0
|
topSubtitleLabel.numberOfLines = 0
|
||||||
|
|
@ -697,7 +726,12 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
}
|
}
|
||||||
|
|
||||||
func updateSubtitleLabelConstraints() {
|
func updateSubtitleLabelConstraints() {
|
||||||
subtitleBottomConstraint?.constant = -subtitleBottomPadding
|
if isControlsVisible {
|
||||||
|
subtitleBottomToSliderConstraint?.constant = -20
|
||||||
|
} else {
|
||||||
|
subtitleBottomToSafeAreaConstraint?.constant = -subtitleBottomPadding
|
||||||
|
}
|
||||||
|
|
||||||
view.setNeedsLayout()
|
view.setNeedsLayout()
|
||||||
UIView.animate(withDuration: 0.2) {
|
UIView.animate(withDuration: 0.2) {
|
||||||
self.view.layoutIfNeeded()
|
self.view.layoutIfNeeded()
|
||||||
|
|
@ -809,6 +843,181 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
])
|
])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private func updateSkipButtonsVisibility() {
|
||||||
|
let t = currentTimeVal
|
||||||
|
let controlsShowing = isControlsVisible // true ⇒ main UI is on‑screen
|
||||||
|
|
||||||
|
func handle(_ button: UIButton, range: CMTimeRange?) {
|
||||||
|
guard let r = range else { button.isHidden = true; return }
|
||||||
|
|
||||||
|
let inInterval = t >= r.start.seconds && t <= r.end.seconds
|
||||||
|
let target = controlsShowing ? 0.0 : skipButtonBaseAlpha
|
||||||
|
|
||||||
|
if inInterval {
|
||||||
|
if button.isHidden {
|
||||||
|
button.alpha = 0
|
||||||
|
}
|
||||||
|
button.isHidden = false
|
||||||
|
|
||||||
|
UIView.animate(withDuration: 0.25) {
|
||||||
|
button.alpha = target
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
guard !button.isHidden else { return }
|
||||||
|
UIView.animate(withDuration: 0.15, animations: {
|
||||||
|
button.alpha = 0
|
||||||
|
}) { _ in
|
||||||
|
button.isHidden = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
handle(skipIntroButton, range: skipIntervals.op)
|
||||||
|
handle(skipOutroButton, range: skipIntervals.ed)
|
||||||
|
}
|
||||||
|
|
||||||
|
private func updateSegments() {
|
||||||
|
sliderViewModel.introSegments.removeAll()
|
||||||
|
sliderViewModel.outroSegments.removeAll()
|
||||||
|
|
||||||
|
if let op = skipIntervals.op {
|
||||||
|
let start = max(0, op.start.seconds / duration)
|
||||||
|
let end = min(1, op.end.seconds / duration)
|
||||||
|
sliderViewModel.introSegments.append(start...end)
|
||||||
|
}
|
||||||
|
|
||||||
|
if let ed = skipIntervals.ed {
|
||||||
|
let start = max(0, ed.start.seconds / duration)
|
||||||
|
let end = min(1, ed.end.seconds / duration)
|
||||||
|
sliderViewModel.outroSegments.append(start...end)
|
||||||
|
}
|
||||||
|
// Force SwiftUI to update
|
||||||
|
DispatchQueue.main.async {
|
||||||
|
self.sliderHostingController?.rootView = MusicProgressSlider(
|
||||||
|
value: Binding(
|
||||||
|
get: { max(0, min(self.sliderViewModel.sliderValue, self.duration)) }, // Remove extra ')'
|
||||||
|
set: { self.sliderViewModel.sliderValue = max(0, min($0, self.duration)) } // Remove extra ')'
|
||||||
|
),
|
||||||
|
inRange: 0...(self.duration > 0 ? self.duration : 1.0),
|
||||||
|
activeFillColor: .white,
|
||||||
|
fillColor: .white.opacity(0.6),
|
||||||
|
textColor: .white.opacity(0.7),
|
||||||
|
emptyColor: .white.opacity(0.3),
|
||||||
|
height: 33,
|
||||||
|
onEditingChanged: { editing in
|
||||||
|
if !editing {
|
||||||
|
let targetTime = CMTime(
|
||||||
|
seconds: self.sliderViewModel.sliderValue,
|
||||||
|
preferredTimescale: 600
|
||||||
|
)
|
||||||
|
self.player.seek(to: targetTime)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
introSegments: self.sliderViewModel.introSegments,
|
||||||
|
outroSegments: self.sliderViewModel.outroSegments,
|
||||||
|
introColor: .yellow, // Match your color choices
|
||||||
|
outroColor: .yellow
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func fetchSkipTimes(type: String) {
|
||||||
|
guard let mal = malID else { return }
|
||||||
|
let url = URL(string: "https://api.aniskip.com/v2/skip-times/\(mal)/\(episodeNumber)?types=\(type)&episodeLength=0")!
|
||||||
|
URLSession.shared.dataTask(with: url) { data, _, _ in
|
||||||
|
guard let d = data,
|
||||||
|
let resp = try? JSONDecoder().decode(AniSkipResponse.self, from: d),
|
||||||
|
resp.found,
|
||||||
|
let interval = resp.results.first?.interval else { return }
|
||||||
|
|
||||||
|
let range = CMTimeRange(
|
||||||
|
start: CMTime(seconds: interval.startTime, preferredTimescale: 600),
|
||||||
|
end: CMTime(seconds: interval.endTime, preferredTimescale: 600)
|
||||||
|
)
|
||||||
|
DispatchQueue.main.async {
|
||||||
|
if type == "op" {
|
||||||
|
self.skipIntervals.op = range
|
||||||
|
} else {
|
||||||
|
self.skipIntervals.ed = range
|
||||||
|
}
|
||||||
|
// Update segments only if duration is available
|
||||||
|
if self.duration > 0 {
|
||||||
|
self.updateSegments()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}.resume()
|
||||||
|
}
|
||||||
|
|
||||||
|
private func setupSkipButtons() {
|
||||||
|
let introConfig = UIImage.SymbolConfiguration(pointSize: 14, weight: .bold)
|
||||||
|
let introImage = UIImage(systemName: "forward.frame", withConfiguration: introConfig)
|
||||||
|
|
||||||
|
skipIntroButton = UIButton(type: .system)
|
||||||
|
skipIntroButton.setImage(introImage, for: .normal)
|
||||||
|
skipIntroButton.setTitle(" Skip Intro", for: .normal)
|
||||||
|
skipIntroButton.titleLabel?.font = UIFont.systemFont(ofSize: 14, weight: .bold)
|
||||||
|
|
||||||
|
// match skip85Button styling:
|
||||||
|
skipIntroButton.backgroundColor = UIColor(red: 51/255, green: 51/255, blue: 51/255, alpha: 0.8)
|
||||||
|
skipIntroButton.tintColor = .white
|
||||||
|
skipIntroButton.setTitleColor(.white, for: .normal)
|
||||||
|
skipIntroButton.layer.cornerRadius = 15
|
||||||
|
skipIntroButton.alpha = skipButtonBaseAlpha
|
||||||
|
skipIntroButton.contentEdgeInsets = UIEdgeInsets(top: 6, left: 10, bottom: 6, right: 10)
|
||||||
|
skipIntroButton.layer.shadowColor = UIColor.black.cgColor
|
||||||
|
skipIntroButton.layer.shadowOffset = CGSize(width: 0, height: 2)
|
||||||
|
skipIntroButton.layer.shadowOpacity = 0.6
|
||||||
|
skipIntroButton.layer.shadowRadius = 4
|
||||||
|
skipIntroButton.layer.masksToBounds = false
|
||||||
|
|
||||||
|
skipIntroButton.addTarget(self, action: #selector(skipIntro), for: .touchUpInside)
|
||||||
|
view.addSubview(skipIntroButton)
|
||||||
|
skipIntroButton.translatesAutoresizingMaskIntoConstraints = false
|
||||||
|
|
||||||
|
NSLayoutConstraint.activate([
|
||||||
|
skipIntroButton.leadingAnchor.constraint(
|
||||||
|
equalTo: sliderHostingController!.view.leadingAnchor),
|
||||||
|
skipIntroButton.bottomAnchor.constraint(
|
||||||
|
equalTo: sliderHostingController!.view.topAnchor, constant: -5)
|
||||||
|
])
|
||||||
|
|
||||||
|
// MARK: – Skip Outro Button
|
||||||
|
let outroConfig = UIImage.SymbolConfiguration(pointSize: 14, weight: .bold)
|
||||||
|
let outroImage = UIImage(systemName: "forward.frame", withConfiguration: outroConfig)
|
||||||
|
|
||||||
|
skipOutroButton = UIButton(type: .system)
|
||||||
|
skipOutroButton.setImage(outroImage, for: .normal)
|
||||||
|
skipOutroButton.setTitle(" Skip Outro", for: .normal)
|
||||||
|
skipOutroButton.titleLabel?.font = UIFont.systemFont(ofSize: 14, weight: .bold)
|
||||||
|
|
||||||
|
// same styling as above
|
||||||
|
skipOutroButton.backgroundColor = skipIntroButton.backgroundColor
|
||||||
|
skipOutroButton.tintColor = skipIntroButton.tintColor
|
||||||
|
skipOutroButton.setTitleColor(.white, for: .normal)
|
||||||
|
skipOutroButton.layer.cornerRadius = skipIntroButton.layer.cornerRadius
|
||||||
|
skipOutroButton.alpha = skipIntroButton.alpha
|
||||||
|
skipOutroButton.contentEdgeInsets = skipIntroButton.contentEdgeInsets
|
||||||
|
skipOutroButton.layer.shadowColor = skipIntroButton.layer.shadowColor
|
||||||
|
skipOutroButton.layer.shadowOffset = skipIntroButton.layer.shadowOffset
|
||||||
|
skipOutroButton.layer.shadowOpacity = skipIntroButton.layer.shadowOpacity
|
||||||
|
skipOutroButton.layer.shadowRadius = skipIntroButton.layer.shadowRadius
|
||||||
|
skipOutroButton.layer.masksToBounds = false
|
||||||
|
|
||||||
|
skipOutroButton.addTarget(self, action: #selector(skipOutro), for: .touchUpInside)
|
||||||
|
view.addSubview(skipOutroButton)
|
||||||
|
skipOutroButton.translatesAutoresizingMaskIntoConstraints = false
|
||||||
|
|
||||||
|
NSLayoutConstraint.activate([
|
||||||
|
skipOutroButton.leadingAnchor.constraint(
|
||||||
|
equalTo: sliderHostingController!.view.leadingAnchor),
|
||||||
|
skipOutroButton.bottomAnchor.constraint(
|
||||||
|
equalTo: sliderHostingController!.view.topAnchor, constant: -5)
|
||||||
|
])
|
||||||
|
|
||||||
|
view.bringSubviewToFront(skipOutroButton)
|
||||||
|
}
|
||||||
|
|
||||||
private func setupDimButton() {
|
private func setupDimButton() {
|
||||||
let cfg = UIImage.SymbolConfiguration(pointSize: 24, weight: .regular)
|
let cfg = UIImage.SymbolConfiguration(pointSize: 24, weight: .regular)
|
||||||
dimButton = UIButton(type: .system)
|
dimButton = UIButton(type: .system)
|
||||||
|
|
@ -823,22 +1032,22 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
dimButton.layer.shadowOpacity = 0.6
|
dimButton.layer.shadowOpacity = 0.6
|
||||||
dimButton.layer.shadowRadius = 4
|
dimButton.layer.shadowRadius = 4
|
||||||
dimButton.layer.masksToBounds = false
|
dimButton.layer.masksToBounds = false
|
||||||
|
|
||||||
NSLayoutConstraint.activate([
|
NSLayoutConstraint.activate([
|
||||||
dimButton.centerYAnchor.constraint(equalTo: dismissButton.centerYAnchor),
|
dimButton.centerYAnchor.constraint(equalTo: dismissButton.centerYAnchor),
|
||||||
dimButton.widthAnchor.constraint(equalToConstant: 24),
|
dimButton.widthAnchor.constraint(equalToConstant: 24),
|
||||||
dimButton.heightAnchor.constraint(equalToConstant: 24),
|
dimButton.heightAnchor.constraint(equalToConstant: 24),
|
||||||
])
|
])
|
||||||
|
|
||||||
dimButtonToSlider = dimButton.trailingAnchor.constraint(
|
dimButtonToSlider = dimButton.trailingAnchor.constraint(
|
||||||
equalTo: volumeSliderHostingView!.leadingAnchor,
|
equalTo: volumeSliderHostingView!.leadingAnchor,
|
||||||
constant: -8
|
constant: -8
|
||||||
)
|
)
|
||||||
dimButtonToRight = dimButton.trailingAnchor.constraint(
|
dimButtonToRight = dimButton.trailingAnchor.constraint(
|
||||||
equalTo: controlsContainerView.trailingAnchor,
|
equalTo: controlsContainerView.trailingAnchor,
|
||||||
constant: -16
|
constant: -16
|
||||||
)
|
)
|
||||||
|
|
||||||
dimButtonToSlider.isActive = true
|
dimButtonToSlider.isActive = true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -848,13 +1057,13 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
|
|
||||||
let leftSpacing: CGFloat = 2
|
let leftSpacing: CGFloat = 2
|
||||||
let rightSpacing: CGFloat = 6
|
let rightSpacing: CGFloat = 6
|
||||||
let trailingAnchor: NSLayoutXAxisAnchor = (volumeSliderHostingView?.isHidden == false)
|
let trailingAnchor: NSLayoutXAxisAnchor = dimButton.leadingAnchor
|
||||||
? volumeSliderHostingView!.leadingAnchor
|
|
||||||
: view.safeAreaLayoutGuide.trailingAnchor
|
|
||||||
|
|
||||||
currentMarqueeConstraints = [
|
currentMarqueeConstraints = [
|
||||||
marqueeLabel.leadingAnchor.constraint(equalTo: dismissButton.trailingAnchor, constant: leftSpacing),
|
marqueeLabel.leadingAnchor.constraint(
|
||||||
marqueeLabel.trailingAnchor.constraint(equalTo: trailingAnchor, constant: -rightSpacing - 10),
|
equalTo: dismissButton.trailingAnchor, constant: leftSpacing),
|
||||||
|
marqueeLabel.trailingAnchor.constraint(
|
||||||
|
equalTo: trailingAnchor, constant: -rightSpacing - 10),
|
||||||
marqueeLabel.centerYAnchor.constraint(equalTo: dismissButton.centerYAnchor)
|
marqueeLabel.centerYAnchor.constraint(equalTo: dismissButton.centerYAnchor)
|
||||||
]
|
]
|
||||||
NSLayoutConstraint.activate(currentMarqueeConstraints)
|
NSLayoutConstraint.activate(currentMarqueeConstraints)
|
||||||
|
|
@ -891,7 +1100,7 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
menuButton.widthAnchor.constraint(equalToConstant: 40),
|
menuButton.widthAnchor.constraint(equalToConstant: 40),
|
||||||
menuButton.heightAnchor.constraint(equalToConstant: 40),
|
menuButton.heightAnchor.constraint(equalToConstant: 40),
|
||||||
])
|
])
|
||||||
|
|
||||||
currentMenuButtonTrailing = menuButton.trailingAnchor.constraint(equalTo: qualityButton.leadingAnchor, constant: -6)
|
currentMenuButtonTrailing = menuButton.trailingAnchor.constraint(equalTo: qualityButton.leadingAnchor, constant: -6)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -1018,21 +1227,21 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
subtitleLabel.font = UIFont.systemFont(ofSize: CGFloat(subtitleFontSize))
|
subtitleLabel.font = UIFont.systemFont(ofSize: CGFloat(subtitleFontSize))
|
||||||
subtitleLabel.textColor = subtitleUIColor()
|
subtitleLabel.textColor = subtitleUIColor()
|
||||||
subtitleLabel.backgroundColor = subtitleBackgroundEnabled
|
subtitleLabel.backgroundColor = subtitleBackgroundEnabled
|
||||||
? UIColor.black.withAlphaComponent(0.6)
|
? UIColor.black.withAlphaComponent(0.6)
|
||||||
: .clear
|
: .clear
|
||||||
subtitleLabel.layer.cornerRadius = 5
|
subtitleLabel.layer.cornerRadius = 5
|
||||||
subtitleLabel.clipsToBounds = true
|
subtitleLabel.clipsToBounds = true
|
||||||
subtitleLabel.layer.shadowColor = UIColor.black.cgColor
|
subtitleLabel.layer.shadowColor = UIColor.black.cgColor
|
||||||
subtitleLabel.layer.shadowRadius = CGFloat(subtitleShadowRadius)
|
subtitleLabel.layer.shadowRadius = CGFloat(subtitleShadowRadius)
|
||||||
subtitleLabel.layer.shadowOpacity = 1.0
|
subtitleLabel.layer.shadowOpacity = 1.0
|
||||||
subtitleLabel.layer.shadowOffset = .zero
|
subtitleLabel.layer.shadowOffset = .zero
|
||||||
|
|
||||||
// only style it if it’s been created already
|
// only style it if it’s been created already
|
||||||
topSubtitleLabel?.font = UIFont.systemFont(ofSize: CGFloat(subtitleFontSize))
|
topSubtitleLabel?.font = UIFont.systemFont(ofSize: CGFloat(subtitleFontSize))
|
||||||
topSubtitleLabel?.textColor = subtitleUIColor()
|
topSubtitleLabel?.textColor = subtitleUIColor()
|
||||||
topSubtitleLabel?.backgroundColor = subtitleBackgroundEnabled
|
topSubtitleLabel?.backgroundColor = subtitleBackgroundEnabled
|
||||||
? UIColor.black.withAlphaComponent(0.6)
|
? UIColor.black.withAlphaComponent(0.6)
|
||||||
: .clear
|
: .clear
|
||||||
topSubtitleLabel?.layer.cornerRadius = 5
|
topSubtitleLabel?.layer.cornerRadius = 5
|
||||||
topSubtitleLabel?.clipsToBounds = true
|
topSubtitleLabel?.clipsToBounds = true
|
||||||
topSubtitleLabel?.layer.shadowColor = UIColor.black.cgColor
|
topSubtitleLabel?.layer.shadowColor = UIColor.black.cgColor
|
||||||
|
|
@ -1067,6 +1276,7 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
|
|
||||||
self.currentTimeVal = time.seconds
|
self.currentTimeVal = time.seconds
|
||||||
self.duration = currentDuration
|
self.duration = currentDuration
|
||||||
|
self.updateSegments()
|
||||||
|
|
||||||
if !self.isSliderEditing {
|
if !self.isSliderEditing {
|
||||||
self.sliderViewModel.sliderValue = max(0, min(self.currentTimeVal, self.duration))
|
self.sliderViewModel.sliderValue = max(0, min(self.currentTimeVal, self.duration))
|
||||||
|
|
@ -1098,6 +1308,9 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
self.topSubtitleLabel.isHidden = true
|
self.topSubtitleLabel.isHidden = true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let current = self.currentTimeVal
|
||||||
|
|
||||||
|
|
||||||
DispatchQueue.main.async {
|
DispatchQueue.main.async {
|
||||||
if let currentItem = self.player.currentItem, currentItem.duration.seconds > 0 {
|
if let currentItem = self.player.currentItem, currentItem.duration.seconds > 0 {
|
||||||
let progress = min(max(self.currentTimeVal / self.duration, 0), 1.0)
|
let progress = min(max(self.currentTimeVal / self.duration, 0), 1.0)
|
||||||
|
|
@ -1149,12 +1362,31 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
)
|
)
|
||||||
self.player.seek(to: targetTime)
|
self.player.seek(to: targetTime)
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
|
introSegments: self.sliderViewModel.introSegments,
|
||||||
|
outroSegments: self.sliderViewModel.outroSegments,
|
||||||
|
introColor: .yellow, // Match your color choices
|
||||||
|
outroColor: .yellow
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@objc private func skipIntro() {
|
||||||
|
if let range = skipIntervals.op {
|
||||||
|
player.seek(to: range.end)
|
||||||
|
// optionally hide button immediately:
|
||||||
|
skipIntroButton.isHidden = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@objc private func skipOutro() {
|
||||||
|
if let range = skipIntervals.ed {
|
||||||
|
player.seek(to: range.end)
|
||||||
|
skipOutroButton.isHidden = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
func startUpdateTimer() {
|
func startUpdateTimer() {
|
||||||
updateTimer = Timer.scheduledTimer(withTimeInterval: 1, repeats: true) { [weak self] _ in
|
updateTimer = Timer.scheduledTimer(withTimeInterval: 1, repeats: true) { [weak self] _ in
|
||||||
|
|
@ -1164,22 +1396,21 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
}
|
}
|
||||||
|
|
||||||
func updateMenuButtonConstraints() {
|
func updateMenuButtonConstraints() {
|
||||||
// tear down last one
|
// tear down last one
|
||||||
currentMenuButtonTrailing.isActive = false
|
currentMenuButtonTrailing.isActive = false
|
||||||
|
|
||||||
// pick the “next” visible control
|
// pick the “next” visible control
|
||||||
let anchor: NSLayoutXAxisAnchor
|
let anchor: NSLayoutXAxisAnchor
|
||||||
if !qualityButton.isHidden {
|
if !qualityButton.isHidden {
|
||||||
anchor = qualityButton.leadingAnchor
|
anchor = qualityButton.leadingAnchor
|
||||||
} else if !speedButton.isHidden {
|
} else if !speedButton.isHidden {
|
||||||
anchor = speedButton.leadingAnchor
|
anchor = speedButton.leadingAnchor
|
||||||
} else {
|
} else {
|
||||||
anchor = controlsContainerView.trailingAnchor
|
anchor = controlsContainerView.trailingAnchor
|
||||||
}
|
}
|
||||||
|
|
||||||
// rebuild & activate
|
currentMenuButtonTrailing = menuButton.trailingAnchor.constraint(equalTo: anchor, constant: -6)
|
||||||
currentMenuButtonTrailing = menuButton.trailingAnchor.constraint(equalTo: anchor, constant: -6)
|
currentMenuButtonTrailing.isActive = true
|
||||||
currentMenuButtonTrailing.isActive = true
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@objc func toggleControls() {
|
@objc func toggleControls() {
|
||||||
|
|
@ -1199,7 +1430,13 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
let a: CGFloat = self.isControlsVisible ? 1 : 0
|
let a: CGFloat = self.isControlsVisible ? 1 : 0
|
||||||
self.controlsContainerView.alpha = a
|
self.controlsContainerView.alpha = a
|
||||||
self.skip85Button.alpha = a
|
self.skip85Button.alpha = a
|
||||||
|
|
||||||
|
self.subtitleBottomToSafeAreaConstraint?.isActive = !self.isControlsVisible
|
||||||
|
self.subtitleBottomToSliderConstraint?.isActive = self.isControlsVisible
|
||||||
|
|
||||||
|
self.view.layoutIfNeeded()
|
||||||
}
|
}
|
||||||
|
self.updateSkipButtonsVisibility()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -1234,7 +1471,7 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
player.seek(to: CMTime(seconds: currentTimeVal, preferredTimescale: 600)) { [weak self] finished in
|
player.seek(to: CMTime(seconds: currentTimeVal, preferredTimescale: 600)) { [weak self] finished in
|
||||||
guard self != nil else { return }
|
guard self != nil else { return }
|
||||||
}
|
}
|
||||||
animateButtonRotation(backwardButton, clockwise: false)
|
animateButtonRotation(backwardButton, clockwise: false)
|
||||||
}
|
}
|
||||||
|
|
||||||
@objc func seekForward() {
|
@objc func seekForward() {
|
||||||
|
|
@ -1310,21 +1547,21 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
@objc private func dimTapped() {
|
@objc private func dimTapped() {
|
||||||
isDimmed.toggle()
|
isDimmed.toggle()
|
||||||
dimButtonTimer?.invalidate()
|
dimButtonTimer?.invalidate()
|
||||||
|
|
||||||
// animate black overlay
|
// animate black overlay
|
||||||
UIView.animate(withDuration: 0.25) {
|
UIView.animate(withDuration: 0.25) {
|
||||||
self.blackCoverView.alpha = self.isDimmed ? 1.0 : 0.4
|
self.blackCoverView.alpha = self.isDimmed ? 1.0 : 0.4
|
||||||
}
|
}
|
||||||
|
|
||||||
// fade controls instead of hiding
|
// fade controls instead of hiding
|
||||||
UIView.animate(withDuration: 0.25) {
|
UIView.animate(withDuration: 0.25) {
|
||||||
for view in self.controlsToHide {
|
for view in self.controlsToHide {
|
||||||
view.alpha = self.isDimmed ? 0 : 1
|
view.alpha = self.isDimmed ? 0 : 1
|
||||||
}
|
}
|
||||||
// keep the dim button visible/in front
|
// keep the dim button visible/in front
|
||||||
self.dimButton.alpha = self.isDimmed ? 0 : 1
|
self.dimButton.alpha = self.isDimmed ? 0 : 1
|
||||||
}
|
}
|
||||||
|
|
||||||
// swap your trailing constraints on the dim‑button
|
// swap your trailing constraints on the dim‑button
|
||||||
dimButtonToSlider.isActive = !isDimmed
|
dimButtonToSlider.isActive = !isDimmed
|
||||||
dimButtonToRight.isActive = isDimmed
|
dimButtonToRight.isActive = isDimmed
|
||||||
|
|
@ -1385,24 +1622,24 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
button.superview?.layoutIfNeeded()
|
button.superview?.layoutIfNeeded()
|
||||||
|
|
||||||
button.layer.shouldRasterize = true
|
button.layer.shouldRasterize = true
|
||||||
button.layer.rasterizationScale = UIScreen.main.scale
|
button.layer.rasterizationScale = UIScreen.main.scale
|
||||||
button.layer.allowsEdgeAntialiasing = true
|
button.layer.allowsEdgeAntialiasing = true
|
||||||
|
|
||||||
let rotation = CABasicAnimation(keyPath: "transform.rotation.z")
|
let rotation = CABasicAnimation(keyPath: "transform.rotation.z")
|
||||||
rotation.fromValue = 0
|
rotation.fromValue = 0
|
||||||
rotation.toValue = CGFloat.pi * 2 * (clockwise ? 1 : -1)
|
rotation.toValue = CGFloat.pi * 2 * (clockwise ? 1 : -1)
|
||||||
rotation.duration = 0.43
|
rotation.duration = 0.43
|
||||||
rotation.timingFunction = CAMediaTimingFunction(name: .linear)
|
rotation.timingFunction = CAMediaTimingFunction(name: .linear)
|
||||||
|
|
||||||
button.layer.add(rotation, forKey: "rotate360")
|
button.layer.add(rotation, forKey: "rotate360")
|
||||||
|
|
||||||
DispatchQueue.main.asyncAfter(deadline: .now() + rotation.duration) {
|
DispatchQueue.main.asyncAfter(deadline: .now() + rotation.duration) {
|
||||||
button.layer.shouldRasterize = false
|
button.layer.shouldRasterize = false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private func parseM3U8(url: URL, completion: @escaping () -> Void) {
|
private func parseM3U8(url: URL, completion: @escaping () -> Void) {
|
||||||
var request = URLRequest(url: url)
|
var request = URLRequest(url: url)
|
||||||
|
|
@ -1890,7 +2127,7 @@ class CustomMediaPlayerViewController: UIViewController, UIGestureRecognizerDele
|
||||||
height: 10,
|
height: 10,
|
||||||
onEditingChanged: { _ in }
|
onEditingChanged: { _ in }
|
||||||
)
|
)
|
||||||
.shadow(color: Color.black.opacity(0.6), radius: 4, x: 0, y: 2)
|
.shadow(color: Color.black.opacity(0.6), radius: 4, x: 0, y: 2)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -165,11 +165,10 @@ struct EpisodeCell: View {
|
||||||
}
|
}
|
||||||
|
|
||||||
DispatchQueue.main.async {
|
DispatchQueue.main.async {
|
||||||
// Always stop loading
|
|
||||||
self.isLoading = false
|
self.isLoading = false
|
||||||
// Only display metadata if enabled
|
if UserDefaults.standard.object(forKey: "fetchEpisodeMetadata") == nil
|
||||||
if UserDefaults.standard.bool(forKey: "fetchEpisodeMetadata") {
|
|| UserDefaults.standard.bool(forKey: "fetchEpisodeMetadata") {
|
||||||
self.episodeTitle = title["en"] ?? ""
|
self.episodeTitle = title["en"] ?? ""
|
||||||
self.episodeImageUrl = image
|
self.episodeImageUrl = image
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -15,6 +15,7 @@ struct SearchItem: Identifiable {
|
||||||
let href: String
|
let href: String
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
struct SearchView: View {
|
struct SearchView: View {
|
||||||
@AppStorage("selectedModuleId") private var selectedModuleId: String?
|
@AppStorage("selectedModuleId") private var selectedModuleId: String?
|
||||||
@AppStorage("mediaColumnsPortrait") private var mediaColumnsPortrait: Int = 2
|
@AppStorage("mediaColumnsPortrait") private var mediaColumnsPortrait: Int = 2
|
||||||
|
|
|
||||||
|
|
@ -17,7 +17,10 @@ struct SettingsViewPlayer: View {
|
||||||
@AppStorage("holdForPauseEnabled") private var holdForPauseEnabled = false
|
@AppStorage("holdForPauseEnabled") private var holdForPauseEnabled = false
|
||||||
@AppStorage("skip85Visible") private var skip85Visible: Bool = true
|
@AppStorage("skip85Visible") private var skip85Visible: Bool = true
|
||||||
@AppStorage("doubleTapSeekEnabled") private var doubleTapSeekEnabled: Bool = false
|
@AppStorage("doubleTapSeekEnabled") private var doubleTapSeekEnabled: Bool = false
|
||||||
|
@AppStorage("skipIntroOutroVisible") private var skipIntroOutroVisible: Bool = true
|
||||||
|
|
||||||
|
// @AppStorage("introColor") private var introColor: Color = .yellow
|
||||||
|
//@AppStorage("outroColor") private var outroColor: Color = .yellow
|
||||||
|
|
||||||
private let mediaPlayers = ["Default", "VLC", "OutPlayer", "Infuse", "nPlayer", "Sora"]
|
private let mediaPlayers = ["Default", "VLC", "OutPlayer", "Infuse", "nPlayer", "Sora"]
|
||||||
|
|
||||||
|
|
@ -61,6 +64,12 @@ struct SettingsViewPlayer: View {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Section(header: Text("Progress bar Marker Colors")) {
|
||||||
|
// ColorPicker("Intro Color", selection: $introColor)
|
||||||
|
//ColorPicker("Outro Color", selection: $outroColor)
|
||||||
|
}
|
||||||
|
|
||||||
Section(header: Text("Skip Settings"), footer : Text("Double tapping the screen on it's sides will skip with the short tap setting.")) {
|
Section(header: Text("Skip Settings"), footer : Text("Double tapping the screen on it's sides will skip with the short tap setting.")) {
|
||||||
HStack {
|
HStack {
|
||||||
Text("Tap Skip:")
|
Text("Tap Skip:")
|
||||||
|
|
@ -79,6 +88,9 @@ struct SettingsViewPlayer: View {
|
||||||
|
|
||||||
Toggle("Show Skip 85s Button", isOn: $skip85Visible)
|
Toggle("Show Skip 85s Button", isOn: $skip85Visible)
|
||||||
.tint(.accentColor)
|
.tint(.accentColor)
|
||||||
|
|
||||||
|
Toggle("Show Skip Intro / Outro Buttons", isOn: $skipIntroOutroVisible)
|
||||||
|
.tint(.accentColor)
|
||||||
}
|
}
|
||||||
SubtitleSettingsSection()
|
SubtitleSettingsSection()
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue