VLCMediaLibraryManager.swift 12 KB

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