2022-07-16 21:46:49 -04:00
|
|
|
//
|
|
|
|
// LocalLibraryExtensions.swift
|
|
|
|
// App
|
|
|
|
//
|
|
|
|
// Created by Ron Heft on 7/16/22.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
|
|
|
|
extension LocalLibraryItem {
|
2022-08-10 17:09:49 -04:00
|
|
|
convenience init(_ item: LibraryItem, localUrl: String, server: ServerConnectionConfig, files: [LocalFile], coverPath: String?) {
|
2022-07-16 21:46:49 -04:00
|
|
|
self.init()
|
2022-08-10 17:09:49 -04:00
|
|
|
|
2022-08-08 19:25:59 -04:00
|
|
|
self.contentUrl = localUrl
|
2022-07-16 21:46:49 -04:00
|
|
|
self.mediaType = item.mediaType
|
2022-08-10 17:09:49 -04:00
|
|
|
self.localFiles.append(objectsIn: files)
|
2022-08-10 22:17:12 -04:00
|
|
|
self._coverContentUrl = coverPath
|
2022-07-16 21:46:49 -04:00
|
|
|
self.libraryItemId = item.id
|
|
|
|
self.serverConnectionConfigId = server.id
|
|
|
|
self.serverAddress = server.address
|
|
|
|
self.serverUserId = server.userId
|
2022-08-10 22:17:12 -04:00
|
|
|
|
|
|
|
// Link the audio tracks and files
|
2022-08-11 12:30:45 -04:00
|
|
|
linkLocalFiles(files, fromMedia: item.media)
|
2022-07-16 21:46:49 -04:00
|
|
|
}
|
2022-07-31 13:33:36 -04:00
|
|
|
|
2022-08-11 12:30:45 -04:00
|
|
|
mutating func addFiles(_ files: [LocalFile], item: LibraryItem) throws {
|
|
|
|
guard self.isPodcast else { throw LibraryItemDownloadError.podcastOnlySupported }
|
|
|
|
self.localFiles.append(contentsOf: files.filter({ $0.isAudioFile() }))
|
|
|
|
linkLocalFiles(self.localFiles, fromMedia: item.media)
|
2022-08-10 17:09:49 -04:00
|
|
|
}
|
|
|
|
|
2022-08-11 12:30:45 -04:00
|
|
|
mutating private func linkLocalFiles(_ files: [LocalFile], fromMedia: MediaType) {
|
|
|
|
var fromMedia = fromMedia
|
|
|
|
let fileMap = files.map { ($0.filename ?? "", $0.id) }
|
|
|
|
let fileIdByFilename = Dictionary(fileMap, uniquingKeysWith: { (_, last) in last })
|
2022-08-11 12:32:58 -04:00
|
|
|
if ( self.isBook ) {
|
2022-08-11 12:30:45 -04:00
|
|
|
if let tracks = fromMedia.tracks {
|
2022-08-10 22:17:12 -04:00
|
|
|
for i in tracks.indices {
|
2022-08-11 12:30:45 -04:00
|
|
|
_ = fromMedia.tracks?[i].setLocalInfo(filenameIdMap: fileIdByFilename, serverIndex: i)
|
2022-08-10 22:17:12 -04:00
|
|
|
}
|
|
|
|
}
|
2022-08-11 12:32:58 -04:00
|
|
|
} else if ( self.isPodcast ) {
|
2022-08-11 12:30:45 -04:00
|
|
|
if let episodes = fromMedia.episodes {
|
|
|
|
fromMedia.episodes = episodes.compactMap { episode in
|
|
|
|
// Filter out episodes not downloaded
|
|
|
|
var episode = episode
|
|
|
|
let episodeIsDownloaded = episode.audioTrack?.setLocalInfo(filenameIdMap: fileIdByFilename, serverIndex: 0) ?? false
|
|
|
|
return episodeIsDownloaded ? episode : nil
|
2022-08-10 22:17:12 -04:00
|
|
|
}
|
2022-08-10 17:09:49 -04:00
|
|
|
}
|
|
|
|
}
|
2022-08-11 12:30:45 -04:00
|
|
|
self.media = fromMedia
|
2022-08-10 17:09:49 -04:00
|
|
|
}
|
|
|
|
|
2022-07-31 13:33:36 -04:00
|
|
|
func getDuration() -> Double {
|
|
|
|
var total = 0.0
|
2022-08-10 17:09:49 -04:00
|
|
|
self.media?.tracks.enumerated().forEach { _, track in total += track.duration }
|
2022-07-31 13:33:36 -04:00
|
|
|
return total
|
|
|
|
}
|
|
|
|
|
2022-08-11 18:29:55 -04:00
|
|
|
func getPodcastEpisode(episodeId: String?) -> PodcastEpisode? {
|
|
|
|
guard self.isPodcast else { return nil }
|
|
|
|
guard let episodes = self.media?.episodes else { return nil }
|
|
|
|
return episodes.first(where: { $0.id == episodeId })
|
|
|
|
}
|
|
|
|
|
2022-08-10 22:17:12 -04:00
|
|
|
func getPlaybackSession(episode: PodcastEpisode?) -> PlaybackSession {
|
2022-07-31 13:33:36 -04:00
|
|
|
let localEpisodeId = episode?.id
|
|
|
|
let sessionId = "play_local_\(UUID().uuidString)"
|
|
|
|
|
|
|
|
// Get current progress from local media
|
|
|
|
let mediaProgressId = (localEpisodeId != nil) ? "\(self.id)-\(localEpisodeId!)" : self.id
|
|
|
|
let mediaProgress = Database.shared.getLocalMediaProgress(localMediaProgressId: mediaProgressId)
|
|
|
|
|
|
|
|
let mediaMetadata = self.media?.metadata
|
2022-08-11 18:29:55 -04:00
|
|
|
let chapters = self.media?.chapters
|
|
|
|
var audioTracks = self.media?.tracks
|
|
|
|
let authorName = mediaMetadata?.authorDisplayName
|
2022-07-31 13:33:36 -04:00
|
|
|
|
2022-08-11 18:29:55 -04:00
|
|
|
if let episode = episode, let track = episode.audioTrack {
|
|
|
|
audioTracks = [track]
|
2022-07-31 13:33:36 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
let dateNow = Date().timeIntervalSince1970
|
|
|
|
return PlaybackSession(
|
|
|
|
id: sessionId,
|
|
|
|
userId: self.serverUserId,
|
|
|
|
libraryItemId: self.libraryItemId,
|
|
|
|
episodeId: episode?.serverEpisodeId,
|
|
|
|
mediaType: self.mediaType,
|
2022-08-11 18:29:55 -04:00
|
|
|
chapters: chapters ?? [],
|
2022-07-31 13:33:36 -04:00
|
|
|
displayTitle: mediaMetadata?.title,
|
|
|
|
displayAuthor: authorName,
|
2022-08-11 18:29:55 -04:00
|
|
|
coverPath: self.coverContentUrl,
|
2022-07-31 13:33:36 -04:00
|
|
|
duration: self.getDuration(),
|
2022-08-11 18:29:55 -04:00
|
|
|
playMethod: PlayMethod.local.rawValue,
|
2022-07-31 13:33:36 -04:00
|
|
|
startedAt: dateNow,
|
|
|
|
updatedAt: 0,
|
|
|
|
timeListening: 0.0,
|
2022-08-11 18:29:55 -04:00
|
|
|
audioTracks: audioTracks ?? [],
|
2022-07-31 13:33:36 -04:00
|
|
|
currentTime: mediaProgress?.currentTime ?? 0.0,
|
|
|
|
libraryItem: nil,
|
2022-08-11 18:34:59 -04:00
|
|
|
localLibraryItem: self,
|
2022-07-31 13:33:36 -04:00
|
|
|
serverConnectionConfigId: self.serverConnectionConfigId,
|
|
|
|
serverAddress: self.serverAddress
|
|
|
|
)
|
|
|
|
}
|
2022-07-16 21:46:49 -04:00
|
|
|
}
|
2022-07-17 21:50:19 -04:00
|
|
|
|
|
|
|
extension LocalFile {
|
2022-08-10 17:09:49 -04:00
|
|
|
convenience init(_ libraryItemId: String, _ filename: String, _ mimeType: String, _ localUrl: String, fileSize: Int) {
|
2022-07-17 21:50:19 -04:00
|
|
|
self.init()
|
2022-08-10 17:09:49 -04:00
|
|
|
|
2022-07-30 17:21:50 -04:00
|
|
|
self.id = "\(libraryItemId)_\(filename.toBase64())"
|
2022-07-17 21:50:19 -04:00
|
|
|
self.filename = filename
|
2022-08-07 10:27:08 -04:00
|
|
|
self.mimeType = mimeType
|
2022-08-10 22:17:12 -04:00
|
|
|
self._contentUrl = localUrl
|
2022-08-08 19:25:59 -04:00
|
|
|
self.size = fileSize
|
2022-07-30 16:22:41 -04:00
|
|
|
}
|
|
|
|
|
2022-08-10 17:09:49 -04:00
|
|
|
var absolutePath: String {
|
|
|
|
return AbsDownloader.downloadsDirectory.appendingPathComponent(self.contentUrl).absoluteString
|
|
|
|
}
|
|
|
|
|
2022-07-30 16:22:41 -04:00
|
|
|
func isAudioFile() -> Bool {
|
|
|
|
switch self.mimeType {
|
2022-07-30 18:25:20 -04:00
|
|
|
case "application/octet-stream",
|
|
|
|
"video/mp4":
|
|
|
|
return true
|
|
|
|
default:
|
|
|
|
return self.mimeType?.starts(with: "audio") ?? false
|
2022-07-30 16:22:41 -04:00
|
|
|
}
|
2022-07-17 21:50:19 -04:00
|
|
|
}
|
|
|
|
}
|
2022-08-07 10:27:08 -04:00
|
|
|
|
|
|
|
extension LocalMediaProgress {
|
2022-08-14 17:48:31 -04:00
|
|
|
convenience init(localLibraryItem: LocalLibraryItem, episode: PodcastEpisode?) {
|
2022-08-10 17:09:49 -04:00
|
|
|
self.init()
|
|
|
|
|
2022-08-07 10:27:08 -04:00
|
|
|
self.id = localLibraryItem.id
|
|
|
|
self.localLibraryItemId = localLibraryItem.id
|
|
|
|
self.libraryItemId = localLibraryItem.libraryItemId
|
|
|
|
|
|
|
|
self.serverAddress = localLibraryItem.serverAddress
|
|
|
|
self.serverUserId = localLibraryItem.serverUserId
|
|
|
|
self.serverConnectionConfigId = localLibraryItem.serverConnectionConfigId
|
|
|
|
|
2022-08-12 21:58:54 -04:00
|
|
|
self.duration = localLibraryItem.getDuration()
|
|
|
|
self.progress = 0.0
|
|
|
|
self.currentTime = 0.0
|
|
|
|
self.isFinished = false
|
|
|
|
self.lastUpdate = Int(Date().timeIntervalSince1970)
|
|
|
|
self.startedAt = 0
|
|
|
|
self.finishedAt = nil
|
|
|
|
|
|
|
|
if let episode = episode {
|
|
|
|
self.id += "-\(episode.id)"
|
|
|
|
self.episodeId = episode.id
|
|
|
|
self.duration = episode.duration ?? 0.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
init(localLibraryItem: LocalLibraryItem, episode: PodcastEpisode?, progress: MediaProgress) {
|
|
|
|
self.init(localLibraryItem: localLibraryItem, episode: episode)
|
2022-08-07 10:27:08 -04:00
|
|
|
self.duration = progress.duration
|
2022-08-11 16:26:32 -04:00
|
|
|
self.progress = progress.progress
|
2022-08-07 10:27:08 -04:00
|
|
|
self.currentTime = progress.currentTime
|
2022-08-12 21:58:54 -04:00
|
|
|
self.isFinished = progress.isFinished
|
2022-08-07 10:27:08 -04:00
|
|
|
self.lastUpdate = progress.lastUpdate
|
|
|
|
self.startedAt = progress.startedAt
|
|
|
|
self.finishedAt = progress.finishedAt
|
|
|
|
}
|
2022-08-11 20:38:51 -04:00
|
|
|
|
|
|
|
mutating func updateIsFinished(_ finished: Bool) {
|
|
|
|
if self.isFinished != finished {
|
|
|
|
self.progress = finished ? 1.0 : 0.0
|
|
|
|
}
|
2022-08-12 21:58:54 -04:00
|
|
|
|
|
|
|
if self.startedAt == 0 && finished {
|
|
|
|
self.startedAt = Int(Date().timeIntervalSince1970)
|
|
|
|
}
|
2022-08-11 20:38:51 -04:00
|
|
|
|
|
|
|
self.isFinished = finished
|
|
|
|
self.lastUpdate = Int(Date().timeIntervalSince1970)
|
|
|
|
self.finishedAt = finished ? lastUpdate : nil
|
|
|
|
}
|
|
|
|
|
|
|
|
mutating func updateFromPlaybackSession(_ playbackSession: PlaybackSession) {
|
|
|
|
self.currentTime = playbackSession.currentTime
|
|
|
|
self.progress = playbackSession.progress
|
|
|
|
self.lastUpdate = Int(Date().timeIntervalSince1970)
|
|
|
|
self.isFinished = playbackSession.progress >= 100.0
|
|
|
|
self.finishedAt = self.isFinished ? self.lastUpdate : nil
|
|
|
|
}
|
|
|
|
|
|
|
|
mutating func updateFromServerMediaProgress(_ serverMediaProgress: MediaProgress) {
|
|
|
|
self.isFinished = serverMediaProgress.isFinished
|
|
|
|
self.progress = serverMediaProgress.progress
|
|
|
|
self.currentTime = serverMediaProgress.currentTime
|
|
|
|
self.duration = serverMediaProgress.duration
|
|
|
|
self.lastUpdate = serverMediaProgress.lastUpdate
|
|
|
|
self.finishedAt = serverMediaProgress.finishedAt
|
|
|
|
self.startedAt = serverMediaProgress.startedAt
|
|
|
|
}
|
2022-08-12 23:11:09 -04:00
|
|
|
|
|
|
|
static func fetchOrCreateLocalMediaProgress(localMediaProgressId: String?, localLibraryItemId: String?, localEpisodeId: String?) -> LocalMediaProgress? {
|
|
|
|
if let localMediaProgressId = localMediaProgressId {
|
|
|
|
return Database.shared.getLocalMediaProgress(localMediaProgressId: localMediaProgressId)
|
|
|
|
} else if let localLibraryItemId = localLibraryItemId {
|
|
|
|
guard let localLibraryItem = Database.shared.getLocalLibraryItem(localLibraryItemId: localLibraryItemId) else { return nil }
|
|
|
|
let episode = localLibraryItem.getPodcastEpisode(episodeId: localEpisodeId)
|
|
|
|
return LocalMediaProgress(localLibraryItem: localLibraryItem, episode: episode)
|
|
|
|
} else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
2022-08-07 10:27:08 -04:00
|
|
|
}
|