MediaLibraryService.swift 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523
  1. /*****************************************************************************
  2. * MediaLibraryService.swift
  3. * VLC for iOS
  4. *****************************************************************************
  5. * Copyright © 2018 VideoLAN. All rights reserved.
  6. * Copyright © 2018 Videolabs
  7. *
  8. * Authors: Soomin Lee <bubu # mikan.io>
  9. *
  10. * Refer to the COPYING file of the official project for license.
  11. *****************************************************************************/
  12. // MARK: - Notification names
  13. extension Notification.Name {
  14. static let VLCNewFileAddedNotification = Notification.Name("NewFileAddedNotification")
  15. }
  16. // For objc
  17. extension NSNotification {
  18. @objc static let VLCNewFileAddedNotification = Notification.Name.VLCNewFileAddedNotification
  19. }
  20. // MARK: -
  21. @objc protocol MediaLibraryObserver: class {
  22. // Video
  23. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  24. didModifyVideo video: [VLCMLMedia])
  25. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  26. didDeleteMediaWithIds ids: [NSNumber])
  27. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  28. didAddVideos videos: [VLCMLMedia])
  29. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  30. didAddShowEpisodes showEpisodes: [VLCMLMedia])
  31. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  32. thumbnailReady media: VLCMLMedia)
  33. // Audio
  34. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  35. didAddAudios audios: [VLCMLMedia])
  36. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  37. didAddArtists artists: [VLCMLArtist])
  38. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  39. didDeleteArtistsWithIds artistsIds: [NSNumber])
  40. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  41. didAddAlbums albums: [VLCMLAlbum])
  42. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  43. didDeleteAlbumsWithIds albumsIds: [NSNumber])
  44. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  45. didAddAlbumTracks albumTracks: [VLCMLMedia])
  46. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  47. didAddGenres genres: [VLCMLGenre])
  48. // Playlist
  49. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  50. didAddPlaylists playlists: [VLCMLPlaylist])
  51. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  52. didModifyPlaylists playlists: [VLCMLPlaylist])
  53. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  54. didDeletePlaylistsWithIds playlistsIds: [NSNumber])
  55. }
  56. // MARK: -
  57. protocol MediaLibraryMigrationDelegate: class {
  58. func medialibraryDidStartMigration(_ medialibrary: MediaLibraryService)
  59. func medialibraryDidFinishMigration(_ medialibrary: MediaLibraryService)
  60. func medialibraryDidStopMigration(_ medialibrary: MediaLibraryService)
  61. }
  62. // MARK: -
  63. class MediaLibraryService: NSObject {
  64. private static let databaseName: String = "medialibrary.db"
  65. private static let migrationKey: String = "MigratedToVLCMediaLibraryKit"
  66. private var didMigrate = UserDefaults.standard.bool(forKey: MediaLibraryService.migrationKey)
  67. private var didFinishDiscovery = false
  68. // Using ObjectIdentifier to avoid duplication and facilitate
  69. // identification of observing object
  70. private var observers = [ObjectIdentifier: Observer]()
  71. private var medialib: VLCMediaLibrary!
  72. weak var migrationDelegate: MediaLibraryMigrationDelegate?
  73. override init() {
  74. super.init()
  75. medialib = VLCMediaLibrary()
  76. medialib.delegate = self
  77. setupMediaLibrary()
  78. NotificationCenter.default.addObserver(self, selector: #selector(reload),
  79. name: .VLCNewFileAddedNotification, object: nil)
  80. }
  81. }
  82. // MARK: - Private initializers
  83. private extension MediaLibraryService {
  84. private func setupMediaDiscovery(at path: String) {
  85. let mediaFileDiscoverer = VLCMediaFileDiscoverer.sharedInstance()
  86. mediaFileDiscoverer?.directoryPath = path
  87. mediaFileDiscoverer?.addObserver(self)
  88. mediaFileDiscoverer?.startDiscovering()
  89. }
  90. private func setupMediaLibrary() {
  91. guard let documentPath = NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true).first,
  92. let libraryPath = NSSearchPathForDirectoriesInDomains(.libraryDirectory, .userDomainMask, true).first else {
  93. preconditionFailure("MediaLibraryService: Unable to init medialibrary.")
  94. }
  95. setupMediaDiscovery(at: documentPath)
  96. let databasePath = libraryPath + "/MediaLibrary/" + MediaLibraryService.databaseName
  97. let thumbnailPath = libraryPath + "/MediaLibrary/Thumbnails"
  98. do {
  99. try FileManager.default.createDirectory(atPath: thumbnailPath,
  100. withIntermediateDirectories: true)
  101. } catch let error as NSError {
  102. assertionFailure("Failed to create directory: \(error.localizedDescription)")
  103. }
  104. let medialibraryStatus = medialib.setupMediaLibrary(databasePath: databasePath,
  105. thumbnailPath: thumbnailPath)
  106. switch medialibraryStatus {
  107. case .success:
  108. guard medialib.start() else {
  109. assertionFailure("MediaLibraryService: Medialibrary failed to start.")
  110. return
  111. }
  112. medialib.reload()
  113. medialib.discover(onEntryPoint: "file://" + documentPath)
  114. case .alreadyInitialized:
  115. assertionFailure("MediaLibraryService: Medialibrary already initialized.")
  116. case .failed:
  117. preconditionFailure("MediaLibraryService: Failed to setup medialibrary.")
  118. case .dbReset:
  119. // should still start and discover but warn the user that the db has been wipped
  120. assertionFailure("MediaLibraryService: The database was resetted, please re-configure.")
  121. }
  122. }
  123. }
  124. // MARK: - Migration
  125. private extension MediaLibraryService {
  126. func startMigrationIfNeeded() {
  127. guard !didMigrate else {
  128. return
  129. }
  130. migrationDelegate?.medialibraryDidStartMigration(self)
  131. guard migrateToNewMediaLibrary() else {
  132. migrationDelegate?.medialibraryDidStopMigration(self)
  133. return
  134. }
  135. migrationDelegate?.medialibraryDidFinishMigration(self)
  136. }
  137. func migrateMedia(_ oldMedialibrary: MLMediaLibrary) -> Bool {
  138. guard let allFiles = MLFile.allFiles() as? [MLFile] else {
  139. assertionFailure("MediaLibraryService: Migration: Unable to retreive all files")
  140. return false
  141. }
  142. for media in allFiles {
  143. if let newMedia = fetchMedia(with: media.url) {
  144. newMedia.updateTitle(media.title)
  145. newMedia.setPlayCount(media.playCount.uint32Value)
  146. newMedia.setMetadataOf(.progress, intValue: media.lastPosition.int64Value)
  147. newMedia.setMetadataOf(.seen, intValue: media.unread.int64Value)
  148. // Only delete files that are not in playlist
  149. if media.labels.isEmpty {
  150. oldMedialibrary.remove(media)
  151. }
  152. }
  153. }
  154. oldMedialibrary.save()
  155. return true
  156. }
  157. // This private method migrates old playlist and removes file and playlist
  158. // from the old medialibrary.
  159. // Note: This removes **only** files that are in a playlist
  160. func migratePlaylists(_ oldMedialibrary: MLMediaLibrary) -> Bool {
  161. guard let allLabels = MLLabel.allLabels() as? [MLLabel] else {
  162. assertionFailure("MediaLibraryService: Migration: Unable to retreive all labels")
  163. return false
  164. }
  165. for label in allLabels {
  166. guard let newPlaylist = createPlaylist(with: label.name) else {
  167. assertionFailure("MediaLibraryService: Migration: Unable to create playlist.")
  168. continue
  169. }
  170. guard let files = label.files as? Set<MLFile> else {
  171. assertionFailure("MediaLibraryService: Migration: Unable to retreive files from label")
  172. oldMedialibrary.remove(label)
  173. continue
  174. }
  175. for file in files {
  176. if let newMedia = fetchMedia(with: file.url) {
  177. if newPlaylist.appendMedia(withIdentifier: newMedia.identifier()) {
  178. oldMedialibrary.remove(file)
  179. }
  180. }
  181. }
  182. oldMedialibrary.remove(label)
  183. }
  184. oldMedialibrary.save()
  185. return true
  186. }
  187. func migrateToNewMediaLibrary() -> Bool {
  188. guard let oldMedialibrary = MLMediaLibrary.sharedMediaLibrary() as? MLMediaLibrary else {
  189. assertionFailure("MediaLibraryService: Migration: Unable to retreive old medialibrary")
  190. return false
  191. }
  192. if migrateMedia(oldMedialibrary) && migratePlaylists(oldMedialibrary) {
  193. UserDefaults.standard.set(true, forKey: MediaLibraryService.migrationKey)
  194. didMigrate = true
  195. return true
  196. }
  197. return false
  198. }
  199. }
  200. // MARK: - Observer
  201. private extension MediaLibraryService {
  202. struct Observer {
  203. weak var observer: MediaLibraryObserver?
  204. }
  205. }
  206. extension MediaLibraryService {
  207. func addObserver(_ observer: MediaLibraryObserver) {
  208. let identifier = ObjectIdentifier(observer)
  209. observers[identifier] = Observer(observer: observer)
  210. }
  211. func removeObserver(_ observer: MediaLibraryObserver) {
  212. let identifier = ObjectIdentifier(observer)
  213. observers.removeValue(forKey: identifier)
  214. }
  215. }
  216. // MARK: - Helpers
  217. @objc extension MediaLibraryService {
  218. @objc func reload() {
  219. medialib.reload()
  220. }
  221. /// Returns number of *ALL* files(audio and video) present in the medialibrary database
  222. func numberOfFiles() -> Int {
  223. return (medialib.audioFiles()?.count ?? 0) + (medialib.videoFiles()?.count ?? 0)
  224. }
  225. /// Returns *ALL* file found for a specified VLCMLMediaType
  226. ///
  227. /// - Parameter type: Type of the media
  228. /// - Returns: Array of VLCMLMedia
  229. func media(ofType type: VLCMLMediaType,
  230. sortingCriteria sort: VLCMLSortingCriteria = .filename,
  231. desc: Bool = false) -> [VLCMLMedia] {
  232. return type == .video ? medialib.videoFiles(with: sort, desc: desc) ?? []
  233. : medialib.audioFiles(with: sort, desc: desc) ?? []
  234. }
  235. @objc func fetchMedia(with mrl: URL?) -> VLCMLMedia? {
  236. guard let mrl = mrl else {
  237. assertionFailure("MedialibraryService: no mrl")
  238. return nil
  239. }
  240. return medialib.media(withMrl: mrl)
  241. }
  242. @objc func media(for identifier: VLCMLIdentifier) -> VLCMLMedia? {
  243. return medialib.media(withIdentifier: identifier)
  244. }
  245. func savePlaybackState(from player: VLCPlaybackController) {
  246. let media = player.currentlyPlayingMedia
  247. guard let mlMedia = fetchMedia(with: media.url.absoluteURL) else {
  248. // we opened a url and not a local file
  249. return
  250. }
  251. mlMedia.isNew = false
  252. mlMedia.progress = player.playbackPosition
  253. mlMedia.audioTrackIndex = Int64(player.indexOfCurrentAudioTrack)
  254. mlMedia.subtitleTrackIndex = Int64(player.indexOfCurrentSubtitleTrack)
  255. mlMedia.chapterIndex = Int64(player.indexOfCurrentChapter)
  256. mlMedia.titleIndex = Int64(player.indexOfCurrentTitle)
  257. //create a new thumbnail
  258. }
  259. }
  260. // MARK: - Audio methods
  261. @objc extension MediaLibraryService {
  262. func artists(sortingCriteria sort: VLCMLSortingCriteria = .artist, desc: Bool = false) -> [VLCMLArtist] {
  263. return medialib.artists(with: sort, desc: desc, all: true) ?? []
  264. }
  265. func albums(sortingCriteria sort: VLCMLSortingCriteria = .album, desc: Bool = false) -> [VLCMLAlbum] {
  266. return medialib.albums(with: sort, desc: desc) ?? []
  267. }
  268. }
  269. // MARK: - Video methods
  270. extension MediaLibraryService {
  271. func requestThumbnail(for media: [VLCMLMedia]) {
  272. media.forEach() {
  273. guard !$0.isThumbnailGenerated() else { return }
  274. if !medialib.requestThumbnail(for: $0) {
  275. assertionFailure("MediaLibraryService: Failed to generate thumbnail for: \($0.identifier())")
  276. }
  277. }
  278. }
  279. }
  280. // MARK: - Playlist methods
  281. @objc extension MediaLibraryService {
  282. func createPlaylist(with name: String) -> VLCMLPlaylist? {
  283. return medialib.createPlaylist(withName: name)
  284. }
  285. func deletePlaylist(with identifier: VLCMLIdentifier) -> Bool {
  286. return medialib.deletePlaylist(withIdentifier: identifier)
  287. }
  288. func playlists(sortingCriteria sort: VLCMLSortingCriteria = .default, desc: Bool = false) -> [VLCMLPlaylist] {
  289. return medialib.playlists(with: sort, desc: desc) ?? []
  290. }
  291. }
  292. // MARK: - Genre methods
  293. extension MediaLibraryService {
  294. func genres(sortingCriteria sort: VLCMLSortingCriteria = .default, desc: Bool = false) -> [VLCMLGenre] {
  295. return medialib.genres(with: sort, desc: desc) ?? []
  296. }
  297. }
  298. // MARK: - VLCMediaFileDiscovererDelegate
  299. extension MediaLibraryService: VLCMediaFileDiscovererDelegate {
  300. func mediaFileAdded(_ filePath: String!, loading isLoading: Bool) {
  301. guard !isLoading else {
  302. return
  303. }
  304. /* exclude media files from backup (QA1719) */
  305. var excludeURL = URL(fileURLWithPath: filePath)
  306. var resourceValue = URLResourceValues()
  307. resourceValue.isExcludedFromBackup = true
  308. do {
  309. try excludeURL.setResourceValues(resourceValue)
  310. } catch let error {
  311. assertionFailure("MediaLibraryService: VLCMediaFileDiscovererDelegate: \(error.localizedDescription)")
  312. }
  313. reload()
  314. }
  315. func mediaFileDeleted(_ filePath: String!) {
  316. reload()
  317. }
  318. }
  319. // MARK: - VLCMediaLibraryDelegate - Media
  320. extension MediaLibraryService: VLCMediaLibraryDelegate {
  321. func medialibrary(_ medialibrary: VLCMediaLibrary, didAddMedia media: [VLCMLMedia]) {
  322. media.forEach { $0.updateCoreSpotlightEntry() }
  323. let videos = media.filter {( $0.type() == .video )}
  324. let audio = media.filter {( $0.type() == .audio )}
  325. // thumbnails only for videos
  326. requestThumbnail(for: videos)
  327. for observer in observers {
  328. observer.value.observer?.medialibrary?(self, didAddVideos: videos)
  329. observer.value.observer?.medialibrary?(self, didAddAudios: audio)
  330. }
  331. }
  332. func medialibrary(_ medialibrary: VLCMediaLibrary, didModifyMedia media: [VLCMLMedia]) {
  333. media.forEach { $0.updateCoreSpotlightEntry() }
  334. let showEpisodes = media.filter {( $0.subtype() == .showEpisode )}
  335. let albumTrack = media.filter {( $0.subtype() == .albumTrack )}
  336. for observer in observers {
  337. observer.value.observer?.medialibrary?(self, didAddShowEpisodes: showEpisodes)
  338. observer.value.observer?.medialibrary?(self, didAddAlbumTracks: albumTrack)
  339. }
  340. }
  341. func medialibrary(_ medialibrary: VLCMediaLibrary, didDeleteMediaWithIds mediaIds: [NSNumber]) {
  342. var stringIds = [String]()
  343. mediaIds.forEach { stringIds.append("\($0)") }
  344. CSSearchableIndex.default().deleteSearchableItems(withIdentifiers: stringIds, completionHandler: nil)
  345. for observer in observers {
  346. observer.value.observer?.medialibrary?(self, didDeleteMediaWithIds: mediaIds)
  347. }
  348. }
  349. func medialibrary(_ medialibrary: VLCMediaLibrary, thumbnailReadyFor media: VLCMLMedia, withSuccess success: Bool) {
  350. for observer in observers {
  351. observer.value.observer?.medialibrary?(self, thumbnailReady: media)
  352. }
  353. }
  354. }
  355. // MARK: - VLCMediaLibraryDelegate - Artists
  356. extension MediaLibraryService {
  357. func medialibrary(_ medialibrary: VLCMediaLibrary, didAdd artists: [VLCMLArtist]) {
  358. for observer in observers {
  359. observer.value.observer?.medialibrary?(self, didAddArtists: artists)
  360. }
  361. }
  362. func medialibrary(_ medialibrary: VLCMediaLibrary, didDeleteArtistsWithIds artistsIds: [NSNumber]) {
  363. for observer in observers {
  364. observer.value.observer?.medialibrary?(self, didDeleteArtistsWithIds: artistsIds)
  365. }
  366. }
  367. }
  368. // MARK: - VLCMediaLibraryDelegate - Albums
  369. extension MediaLibraryService {
  370. func medialibrary(_ medialibrary: VLCMediaLibrary, didAdd albums: [VLCMLAlbum]) {
  371. for observer in observers {
  372. observer.value.observer?.medialibrary?(self, didAddAlbums: albums)
  373. }
  374. }
  375. func medialibrary(_ medialibrary: VLCMediaLibrary, didDeleteAlbumsWithIds albumsIds: [NSNumber]) {
  376. for observer in observers {
  377. observer.value.observer?.medialibrary?(self, didDeleteAlbumsWithIds: albumsIds)
  378. }
  379. }
  380. }
  381. // MARK: - VLCMediaLibraryDelegate - Playlists
  382. extension MediaLibraryService {
  383. func medialibrary(_ medialibrary: VLCMediaLibrary, didAdd playlists: [VLCMLPlaylist]) {
  384. for observer in observers {
  385. observer.value.observer?.medialibrary?(self, didAddPlaylists: playlists)
  386. }
  387. }
  388. func medialibrary(_ medialibrary: VLCMediaLibrary, didModifyPlaylists playlists: [VLCMLPlaylist]) {
  389. for observer in observers {
  390. observer.value.observer?.medialibrary?(self, didModifyPlaylists: playlists)
  391. }
  392. }
  393. func medialibrary(_ medialibrary: VLCMediaLibrary, didDeletePlaylistsWithIds playlistsIds: [NSNumber]) {
  394. for observer in observers {
  395. observer.value.observer?.medialibrary?(self, didDeletePlaylistsWithIds: playlistsIds)
  396. }
  397. }
  398. }
  399. // MARK: - VLCMediaLibraryDelegate - Discovery
  400. extension MediaLibraryService {
  401. func medialibrary(_ medialibrary: VLCMediaLibrary, didStartDiscovery entryPoint: String) {
  402. }
  403. func medialibrary(_ medialibrary: VLCMediaLibrary, didCompleteDiscovery entryPoint: String) {
  404. didFinishDiscovery = true
  405. }
  406. func medialibrary(_ medialibrary: VLCMediaLibrary, didProgressDiscovery entryPoint: String) {
  407. }
  408. func medialibrary(_ medialibrary: VLCMediaLibrary, didUpdateParsingStatsWithPercent percent: UInt32) {
  409. if didFinishDiscovery && percent == 100 {
  410. startMigrationIfNeeded()
  411. }
  412. }
  413. }