VLCMediaLibraryManager.swift 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374
  1. /*****************************************************************************
  2. * VLCMediaLibraryManager.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. @objc protocol MediaLibraryObserver: class {
  13. @objc optional func medialibrary(_ medialibrary: VLCMediaLibraryManager,
  14. didUpdateVideo video: [VLCMLMedia])
  15. @objc optional func medialibrary(_ medialibrary: VLCMediaLibraryManager,
  16. didAddVideo video: [VLCMLMedia])
  17. @objc optional func medialibrary(_ medialibrary: VLCMediaLibraryManager,
  18. didAddShowEpisode showEpisode: [VLCMLMedia])
  19. // Audio
  20. @objc optional func medialibrary(_ medialibrary: VLCMediaLibraryManager,
  21. didAddAudio audio: [VLCMLMedia])
  22. @objc optional func medialibrary(_ medialibrary: VLCMediaLibraryManager,
  23. didAddAlbumTrack audio: [VLCMLMedia])
  24. @objc optional func medialibrary(_ medialibrary: VLCMediaLibraryManager,
  25. didAddArtist artist: [VLCMLArtist])
  26. @objc optional func medialibrary(_ medialibrary: VLCMediaLibraryManager,
  27. didAddAlbum album: [VLCMLAlbum])
  28. @objc optional func medialibrary(_ medialibrary: VLCMediaLibraryManager,
  29. didAddGenre genre: [VLCMLGenre])
  30. }
  31. class VLCMediaLibraryManager: NSObject {
  32. private static let databaseName: String = "medialibrary.db"
  33. private var databasePath: String!
  34. private var thumbnailPath: String!
  35. // Using ObjectIdentifier to avoid duplication and facilitate
  36. // identification of observing object
  37. private var observers = [ObjectIdentifier: Observer]()
  38. private lazy var medialib: VLCMediaLibrary = {
  39. let medialibrary = VLCMediaLibrary()
  40. medialibrary.delegate = self
  41. return medialibrary
  42. }()
  43. override init() {
  44. super.init()
  45. setupMediaLibrary()
  46. }
  47. // MARK: Private
  48. private func setupMediaLibrary() {
  49. guard let documentPath = NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true).first,
  50. let dbPath = NSSearchPathForDirectoriesInDomains(.libraryDirectory, .userDomainMask, true).first else {
  51. preconditionFailure("VLCMediaLibraryManager: Unable to init medialibrary.")
  52. }
  53. medialibrary.setVerbosity(.info)
  54. databasePath = dbPath + "/" + VLCMediaLibraryManager.databaseName
  55. thumbnailPath = documentPath
  56. let medialibraryStatus = medialib.setupMediaLibrary(databasePath: databasePath,
  57. thumbnailPath: thumbnailPath)
  58. switch medialibraryStatus {
  59. case .success:
  60. guard medialib.start() else {
  61. assertionFailure("VLCMediaLibraryManager: Medialibrary failed to start.")
  62. return
  63. }
  64. medialib.reload()
  65. medialib.discover(onEntryPoint: "file://" + documentPath)
  66. break
  67. case .alreadyInitialized:
  68. assertionFailure("VLCMediaLibraryManager: Medialibrary already initialized.")
  69. break
  70. case .failed:
  71. preconditionFailure("VLCMediaLibraryManager: Failed to setup medialibrary.")
  72. break
  73. case .dbReset:
  74. // should still start and discover but warn the user that the db has been wipped
  75. assertionFailure("VLCMediaLibraryManager: The database was resetted, please re-configure.")
  76. break
  77. }
  78. }
  79. // MARK: Internal
  80. /// Returns number of *ALL* files(audio and video) present in the medialibrary database
  81. func numberOfFiles() -> Int {
  82. var media = medialib.audioFiles(with: .filename, desc: false)
  83. media += medialib.videoFiles(with: .filename, desc: false)
  84. return media.count
  85. }
  86. /// Returns *ALL* file found for a specified VLCMLMediaType
  87. ///
  88. /// - Parameter type: Type of the media
  89. /// - Returns: Array of VLCMLMedia
  90. func media(ofType type: VLCMLMediaType) -> [VLCMLMedia] {
  91. return type == .video ? medialib.videoFiles(with: .filename, desc: false) : medialib.audioFiles(with: .filename, desc: false)
  92. }
  93. func addMedia(withMrl mrl: URL) {
  94. medialib.addMedia(withMrl: mrl)
  95. }
  96. func genre(sortingCriteria sort: VLCMLSortingCriteria = .default, desc: Bool = false) -> [VLCMLGenre] {
  97. return medialib.genres(with: sort, desc: desc)
  98. }
  99. }
  100. // MARK: - Observer
  101. private extension VLCMediaLibraryManager {
  102. struct Observer {
  103. weak var observer: MediaLibraryObserver?
  104. init(_ observer: MediaLibraryObserver) {
  105. self.observer = observer
  106. }
  107. }
  108. }
  109. extension VLCMediaLibraryManager {
  110. func addObserver(_ observer: MediaLibraryObserver) {
  111. let identifier = ObjectIdentifier(observer)
  112. observers[identifier] = Observer(observer)
  113. }
  114. func removeObserver(_ observer: MediaLibraryObserver) {
  115. let identifier = ObjectIdentifier(observer)
  116. observers.removeValue(forKey: identifier)
  117. }
  118. }
  119. // MARK: MediaDataSource - Audio methods
  120. extension VLCMediaLibraryManager {
  121. private func getAllAudio() {
  122. // foundAudio = medialibrary.media(ofType: .audio)
  123. // artistsFromAudio()
  124. // albumsFromAudio()
  125. // audioPlaylistsFromAudio()
  126. // genresFromAudio()
  127. }
  128. func getArtists() -> [VLCMLArtist] {
  129. return medialib.artists(with: .artist, desc: false, all: true)
  130. }
  131. func getAlbums() -> [VLCMLAlbum] {
  132. return medialib.albums(with: .album, desc: false)
  133. }
  134. private func getAudioPlaylists() {
  135. // let labels = MLLabel.allLabels() as! [MLLabel]
  136. // audioPlaylist = labels.filter {
  137. // let audioFiles = $0.files.filter {
  138. // if let file = $0 as? MLFile {
  139. // return file.isSupportedAudioFile()
  140. // }
  141. // return false
  142. // }
  143. // return !audioFiles.isEmpty
  144. // }
  145. }
  146. private func genresFromAudio() {
  147. // let albumtracks = MLAlbumTrack.allTracks() as! [MLAlbumTrack]
  148. // let tracksWithArtist = albumtracks.filter { $0.genre != nil && $0.genre != "" }
  149. // genres = tracksWithArtist.map { $0.genre }
  150. }
  151. }
  152. // MARK: MediaDataSource - Video methods
  153. extension VLCMediaLibraryManager {
  154. private func getAllVideos() {
  155. // moviesFromVideos()
  156. // episodesFromVideos()
  157. // videoPlaylistsFromVideos()
  158. }
  159. private func getMovies() {
  160. // movies = foundVideos.filter { $0.subtype() == .movie }
  161. }
  162. private func getShowEpisodes() {
  163. // episodes = foundVideos.filter { $0.subtype() == .showEpisode }
  164. }
  165. private func getVideoPlaylists() {
  166. // let labels = MLLabel.allLabels() as! [MLLabel]
  167. // audioPlaylist = labels.filter {
  168. // let audioFiles = $0.files.filter {
  169. // if let file = $0 as? MLFile {
  170. // return file.isShowEpisode() || file.isMovie() || file.isClip()
  171. // }
  172. // return false
  173. // }
  174. // return !audioFiles.isEmpty
  175. // }
  176. }
  177. }
  178. // MARK: VLCMediaLibraryDelegate
  179. extension VLCMediaLibraryManager: VLCMediaLibraryDelegate {
  180. func medialibrary(_ medialibrary: VLCMediaLibrary, didAddMedia media: [VLCMLMedia]) {
  181. let video = media.filter {( $0.type() == .video )}
  182. let audio = media.filter {( $0.type() == .audio )}
  183. let showEpisode = media.filter {( $0.subtype() == .showEpisode )}
  184. let albumTrack = media.filter {( $0.subtype() == .albumTrack )}
  185. for observer in observers {
  186. observer.value.observer?.medialibrary?(self, didAddVideo: video)
  187. observer.value.observer?.medialibrary?(self, didAddAudio: audio)
  188. observer.value.observer?.medialibrary?(self, didAddShowEpisode: showEpisode)
  189. observer.value.observer?.medialibrary?(self, didAddAlbumTrack: albumTrack)
  190. }
  191. }
  192. func medialibrary(_ medialibrary: VLCMediaLibrary, didAdd artists: [VLCMLArtist]) {
  193. print("VLCMediaLibraryDelegate: Did add artists: \(artists), with count: \(artists.count)")
  194. for observer in observers {
  195. observer.value.observer?.medialibrary?(self, didAddArtist: artists)
  196. }
  197. }
  198. func medialibrary(_ medialibrary: VLCMediaLibrary, didAdd albums: [VLCMLAlbum]) {
  199. print("VLCMediaLibraryDelegate: Did add albums: \(albums), with count: \(albums.count)")
  200. for observer in observers {
  201. observer.value.observer?.medialibrary?(self, didAddAlbum: albums)
  202. }
  203. }
  204. func medialibrary(_ medialibrary: VLCMediaLibrary, didStartDiscovery entryPoint: String) {
  205. }
  206. func medialibrary(_ medialibrary: VLCMediaLibrary, didCompleteDiscovery entryPoint: String) {
  207. }
  208. func medialibrary(_ medialibrary: VLCMediaLibrary, didProgressDiscovery entryPoint: String) {
  209. }
  210. func medialibrary(_ medialibrary: VLCMediaLibrary, didUpdateParsingStatsWithPercent percent: UInt32) {
  211. }
  212. }
  213. // MARK: Future MediaDataSource extension
  214. // Todo: implement the remove
  215. // - (void)removeMediaObjectFromFolder:(NSManagedObject *)managedObject
  216. // {
  217. // NSAssert(([managedObject isKindOfClass:[MLFile class]] && ((MLFile *)managedObject).labels.count > 0), @"All media in a folder should be of type MLFile and it should be in a folder");
  218. //
  219. // if (![managedObject isKindOfClass:[MLFile class]]) return;
  220. //
  221. // MLFile *mediaFile = (MLFile *)managedObject;
  222. // [self rearrangeFolderTrackNumbersForRemovedItem:mediaFile];
  223. // mediaFile.labels = nil;
  224. // mediaFile.folderTrackNumber = nil;
  225. // }
  226. //
  227. // - (void)removeMediaObject:(NSManagedObject *)managedObject
  228. // {
  229. // if ([managedObject isKindOfClass:[MLAlbum class]]) {
  230. // MLAlbum *album = (MLAlbum *)managedObject;
  231. // NSSet *iterAlbumTrack = [NSSet setWithSet:album.tracks];
  232. //
  233. // for (MLAlbumTrack *track in iterAlbumTrack) {
  234. // NSSet *iterFiles = [NSSet setWithSet:track.files];
  235. //
  236. // for (MLFile *file in iterFiles)
  237. // [self _deleteMediaObject:file];
  238. // }
  239. // [[MLMediaLibrary sharedMediaLibrary] removeObject: album];
  240. // // delete all episodes from a show
  241. // } else if ([managedObject isKindOfClass:[MLShow class]]) {
  242. // MLShow *show = (MLShow *)managedObject;
  243. // NSSet *iterShowEpisodes = [NSSet setWithSet:show.episodes];
  244. //
  245. // for (MLShowEpisode *episode in iterShowEpisodes) {
  246. // NSSet *iterFiles = [NSSet setWithSet:episode.files];
  247. //
  248. // for (MLFile *file in iterFiles)
  249. // [self _deleteMediaObject:file];
  250. // }
  251. // [[MLMediaLibrary sharedMediaLibrary] removeObject: show];
  252. // // delete all files from an episode
  253. // } else if ([managedObject isKindOfClass:[MLShowEpisode class]]) {
  254. // MLShowEpisode *episode = (MLShowEpisode *)managedObject;
  255. // NSSet *iterFiles = [NSSet setWithSet:episode.files];
  256. //
  257. // for (MLFile *file in iterFiles)
  258. // [self _deleteMediaObject:file];
  259. // // delete all files from a track
  260. // [[MLMediaLibrary sharedMediaLibrary] removeObject: episode];
  261. // } else if ([managedObject isKindOfClass:[MLAlbumTrack class]]) {
  262. // MLAlbumTrack *track = (MLAlbumTrack *)managedObject;
  263. // NSSet *iterFiles = [NSSet setWithSet:track.files];
  264. //
  265. // for (MLFile *file in iterFiles)
  266. // [self _deleteMediaObject:file];
  267. // } else if ([managedObject isKindOfClass:[MLLabel class]]) {
  268. // MLLabel *folder = (MLLabel *)managedObject;
  269. // NSSet *iterFiles = [NSSet setWithSet:folder.files];
  270. // [folder removeFiles:folder.files];
  271. // for (MLFile *file in iterFiles)
  272. // [self _deleteMediaObject:file];
  273. // [[MLMediaLibrary sharedMediaLibrary] removeObject:folder];
  274. // }
  275. // else
  276. // [self _deleteMediaObject:(MLFile *)managedObject];
  277. // }
  278. //
  279. // - (void)_deleteMediaObject:(MLFile *)mediaObject
  280. // {
  281. // [self rearrangeFolderTrackNumbersForRemovedItem:mediaObject];
  282. //
  283. // /* stop playback if needed */
  284. // VLCPlaybackController *vpc = [VLCPlaybackController sharedInstance];
  285. // VLCMedia *media = [vpc currentlyPlayingMedia];
  286. // MLFile *currentlyPlayingFile = [MLFile fileForURL:media.url].firstObject;
  287. // if (currentlyPlayingFile && currentlyPlayingFile == mediaObject) {
  288. // [vpc stopPlayback];
  289. // }
  290. //
  291. // NSFileManager *fileManager = [NSFileManager defaultManager];
  292. // NSString *folderLocation = [[mediaObject.url path] stringByDeletingLastPathComponent];
  293. // NSArray *allfiles = [fileManager contentsOfDirectoryAtPath:folderLocation error:nil];
  294. // NSString *fileName = [mediaObject.path.lastPathComponent stringByDeletingPathExtension];
  295. // if (!fileName)
  296. // return;
  297. // NSIndexSet *indexSet = [allfiles indexesOfObjectsPassingTest:^BOOL(id obj, NSUInteger idx, BOOL *stop) {
  298. // return ([obj rangeOfString:fileName].location != NSNotFound);
  299. // }];
  300. // NSUInteger count = indexSet.count;
  301. // NSString *additionalFilePath;
  302. // NSUInteger currentIndex = [indexSet firstIndex];
  303. // for (unsigned int x = 0; x < count; x++) {
  304. // additionalFilePath = allfiles[currentIndex];
  305. // if ([additionalFilePath isSupportedSubtitleFormat])
  306. // [fileManager removeItemAtPath:[folderLocation stringByAppendingPathComponent:additionalFilePath] error:nil];
  307. // currentIndex = [indexSet indexGreaterThanIndex:currentIndex];
  308. // }
  309. // [fileManager removeItemAtURL:mediaObject.url error:nil];
  310. // }
  311. //
  312. // - (void)rearrangeFolderTrackNumbersForRemovedItem:(MLFile *) mediaObject
  313. // {
  314. // MLLabel *label = [mediaObject.labels anyObject];
  315. // NSSet *allFiles = label.files;
  316. // for (MLFile *file in allFiles) {
  317. // if (file.folderTrackNumber > mediaObject.folderTrackNumber) {
  318. // int value = [file.folderTrackNumber intValue];
  319. // file.folderTrackNumber = [NSNumber numberWithInt:value - 1];
  320. // }
  321. // }
  322. // }
  323. // @end