advplyr.audiobookshelf-app/ios/App/Shared/models/LocalLibraryExtensions.swift

156 lines
5.7 KiB
Swift
Raw Normal View History

2022-07-16 21:46:49 -04:00
//
// LocalLibraryExtensions.swift
// App
//
// Created by Ron Heft on 7/16/22.
//
import Foundation
extension LocalLibraryItem {
init(_ item: LibraryItem, localUrl: String, server: ServerConnectionConfig, files: [LocalFile], coverPath: String?) {
2022-07-16 21:46:49 -04:00
self.init()
self.id = "local_\(item.id)"
2022-08-10 22:17:12 -04:00
self._contentUrl = localUrl
2022-07-16 21:46:49 -04:00
self.mediaType = item.mediaType
self.localFiles = 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
linkLocalFiles(files, fromMedia: item.media)
}
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)
}
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 ) {
if let tracks = fromMedia.tracks {
2022-08-10 22:17:12 -04:00
for i in tracks.indices {
_ = 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 ) {
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
}
}
}
self.media = fromMedia
2022-07-16 21:46:49 -04:00
}
2022-07-31 13:33:36 -04:00
func getDuration() -> Double {
var total = 0.0
self.media?.tracks?.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,
2022-08-11 18:29:55 -04:00
episodeId: episode?.serverEpisodeId,
2022-07-31 13:33:36 -04:00
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 {
init(_ libraryItemId: String, _ filename: String, _ mimeType: String, _ localUrl: String, fileSize: Int) {
2022-07-17 21:50:19 -04:00
self.init()
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
self.size = fileSize
}
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-17 21:50:19 -04:00
}
}
2022-08-07 10:27:08 -04:00
extension LocalMediaProgress {
2022-08-10 22:17:12 -04:00
init(localLibraryItem: LocalLibraryItem, episode: PodcastEpisode?, progress: MediaProgress) {
2022-08-07 10:27:08 -04:00
self.id = localLibraryItem.id
self.localLibraryItemId = localLibraryItem.id
self.libraryItemId = localLibraryItem.libraryItemId
if let episode = episode {
self.id += "-\(episode.id)"
self.episodeId = episode.id
}
self.serverAddress = localLibraryItem.serverAddress
self.serverUserId = localLibraryItem.serverUserId
self.serverConnectionConfigId = localLibraryItem.serverConnectionConfigId
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
self.isFinished = false
self.lastUpdate = progress.lastUpdate
self.startedAt = progress.startedAt
self.finishedAt = progress.finishedAt
}
}