2021-12-26 21:14:46 +00:00
|
|
|
import CoreData
|
2022-05-29 15:50:54 +00:00
|
|
|
import CoreMedia
|
2021-12-26 21:14:46 +00:00
|
|
|
import Defaults
|
|
|
|
import Foundation
|
|
|
|
|
|
|
|
@objc(Watch)
|
|
|
|
final class Watch: NSManagedObject, Identifiable {
|
|
|
|
@Default(.watchedThreshold) private var watchedThreshold
|
2022-05-29 15:50:54 +00:00
|
|
|
@Default(.saveHistory) private var saveHistory
|
2021-12-26 21:14:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
extension Watch {
|
|
|
|
@nonobjc class func fetchRequest() -> NSFetchRequest<Watch> {
|
|
|
|
NSFetchRequest<Watch>(entityName: "Watch")
|
|
|
|
}
|
|
|
|
|
2023-04-22 15:59:28 +00:00
|
|
|
@nonobjc class func markAsWatched(videoID: String, account: Account, duration: Double, watchedAt: Date? = nil, context: NSManagedObjectContext) {
|
2022-06-30 10:10:43 +00:00
|
|
|
let watchFetchRequest = Watch.fetchRequest()
|
|
|
|
watchFetchRequest.predicate = NSPredicate(format: "videoID = %@", videoID as String)
|
|
|
|
|
|
|
|
let results = try? context.fetch(watchFetchRequest)
|
|
|
|
|
|
|
|
context.perform {
|
|
|
|
let watch: Watch?
|
|
|
|
|
|
|
|
if results?.isEmpty ?? true {
|
|
|
|
watch = Watch(context: context)
|
|
|
|
watch?.videoID = videoID
|
2022-12-09 00:15:19 +00:00
|
|
|
watch?.appName = account.app?.rawValue
|
|
|
|
watch?.instanceURL = account.url
|
2022-06-30 10:10:43 +00:00
|
|
|
} else {
|
|
|
|
watch = results?.first
|
|
|
|
}
|
|
|
|
|
2022-09-28 14:27:01 +00:00
|
|
|
guard let watch else { return }
|
2022-06-30 10:10:43 +00:00
|
|
|
|
|
|
|
watch.videoDuration = duration
|
|
|
|
watch.stoppedAt = duration
|
2023-04-22 15:59:28 +00:00
|
|
|
watch.watchedAt = watchedAt ?? .init()
|
2022-06-30 10:10:43 +00:00
|
|
|
|
|
|
|
try? context.save()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
@NSManaged var videoID: String
|
|
|
|
@NSManaged var videoDuration: Double
|
|
|
|
|
|
|
|
@NSManaged var watchedAt: Date?
|
|
|
|
@NSManaged var stoppedAt: Double
|
|
|
|
|
2022-12-09 00:15:19 +00:00
|
|
|
@NSManaged var appName: String?
|
|
|
|
@NSManaged var instanceURL: URL?
|
|
|
|
|
|
|
|
var app: VideosApp! {
|
|
|
|
guard let appName else { return nil }
|
|
|
|
return .init(rawValue: appName)
|
|
|
|
}
|
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
var progress: Double {
|
|
|
|
guard videoDuration.isFinite, !videoDuration.isZero else {
|
2022-12-15 21:39:42 +00:00
|
|
|
return 100
|
2021-12-26 21:14:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let progress = (stoppedAt / videoDuration) * 100
|
2022-01-02 19:39:19 +00:00
|
|
|
|
|
|
|
if progress >= Double(watchedThreshold) {
|
|
|
|
return 100
|
|
|
|
}
|
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
return min(max(progress, 0), 100)
|
|
|
|
}
|
|
|
|
|
|
|
|
var finished: Bool {
|
2022-12-12 23:38:26 +00:00
|
|
|
guard videoDuration.isFinite, !videoDuration.isZero else {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return progress >= Double(watchedThreshold)
|
2021-12-26 21:14:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var watchedAtString: String? {
|
2022-09-28 14:27:01 +00:00
|
|
|
guard let watchedAt else {
|
2021-12-26 21:14:46 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
let formatter = RelativeDateTimeFormatter()
|
|
|
|
formatter.unitsStyle = .full
|
|
|
|
return formatter.localizedString(for: watchedAt, relativeTo: Date())
|
|
|
|
}
|
2022-05-29 15:50:54 +00:00
|
|
|
|
|
|
|
var timeToRestart: CMTime? {
|
|
|
|
finished ? nil : saveHistory ? .secondsInDefaultTimescale(stoppedAt) : nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var video: Video {
|
2022-12-09 19:33:01 +00:00
|
|
|
let url = URL(string: videoID)
|
|
|
|
|
2022-12-10 00:23:13 +00:00
|
|
|
if !Video.VideoID.isValid(videoID) {
|
2022-12-09 19:33:01 +00:00
|
|
|
if let url {
|
|
|
|
return .local(url)
|
|
|
|
}
|
2022-11-10 17:11:28 +00:00
|
|
|
}
|
|
|
|
|
2022-12-16 20:01:01 +00:00
|
|
|
return Video(app: app ?? AccountsModel.shared.current?.app ?? .local, instanceURL: instanceURL, videoID: videoID)
|
2022-05-29 15:50:54 +00:00
|
|
|
}
|
2021-12-26 21:14:46 +00:00
|
|
|
}
|