AlbumModel.swift 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. /*****************************************************************************
  2. * AlbumModel.swift
  3. *
  4. * Copyright © 2018 VLC authors and VideoLAN
  5. * Copyright © 2018 Videolabs
  6. *
  7. * Authors: Soomin Lee <bubu@mikan.io>
  8. *
  9. * Refer to the COPYING file of the official project for license.
  10. *****************************************************************************/
  11. class AlbumModel: AudioCollectionModel {
  12. typealias MLType = VLCMLAlbum
  13. var sortModel = SortModel([.alpha, .duration, .releaseDate, .trackNumber])
  14. var updateView: (() -> Void)?
  15. var files = [VLCMLAlbum]()
  16. var cellType: BaseCollectionViewCell.Type { return MediaCollectionViewCell.self }
  17. var medialibrary: MediaLibraryService
  18. var indicatorName: String = NSLocalizedString("ALBUMS", comment: "")
  19. required init(medialibrary: MediaLibraryService) {
  20. self.medialibrary = medialibrary
  21. medialibrary.addObserver(self)
  22. files = medialibrary.albums()
  23. }
  24. func append(_ item: VLCMLAlbum) {
  25. files.append(item)
  26. }
  27. }
  28. // MARK: - Sort
  29. extension AlbumModel {
  30. func sort(by criteria: VLCMLSortingCriteria, desc: Bool) {
  31. files = medialibrary.albums(sortingCriteria: criteria, desc: desc)
  32. sortModel.currentSort = criteria
  33. sortModel.desc = desc
  34. updateView?()
  35. }
  36. }
  37. // MARK: - Edit
  38. extension AlbumModel: EditableMLModel {
  39. func editCellType() -> BaseCollectionViewCell.Type {
  40. return MediaEditCell.self
  41. }
  42. }
  43. // MARK: - Search
  44. extension VLCMLAlbum: SearchableMLModel {
  45. func contains(_ searchString: String) -> Bool {
  46. var matches = false
  47. matches = matches || title.lowercased().contains(searchString)
  48. matches = matches || String(releaseYear()).lowercased().contains(searchString)
  49. matches = matches || shortSummary.lowercased().contains(searchString)
  50. matches = matches || albumArtist?.contains(searchString) ?? false
  51. matches = matches || tracks?.filter({ $0.contains(searchString)}).isEmpty == false
  52. return matches
  53. }
  54. }
  55. extension VLCMLAlbumTrack: SearchableMLModel {
  56. func contains(_ searchString: String) -> Bool {
  57. var matches = false
  58. matches = matches || artist?.contains(searchString) ?? false
  59. matches = matches || genre?.contains(searchString) ?? false
  60. matches = matches || album?.contains(searchString) ?? false
  61. return matches
  62. }
  63. }
  64. // MARK: - MediaLibraryObserver
  65. extension AlbumModel: MediaLibraryObserver {
  66. func medialibrary(_ medialibrary: MediaLibraryService, didAddAlbums albums: [VLCMLAlbum]) {
  67. albums.forEach({ append($0) })
  68. updateView?()
  69. }
  70. func medialibrary(_ medialibrary: MediaLibraryService,
  71. didModifyAlbumsWithIds albumsIds: [NSNumber]) {
  72. var albums = [VLCMLAlbum]()
  73. albumsIds.forEach() {
  74. guard let safeAlbum = medialibrary.medialib.album(withIdentifier: $0.int64Value)
  75. else {
  76. return
  77. }
  78. albums.append(safeAlbum)
  79. }
  80. files = swapModels(with: albums)
  81. updateView?()
  82. }
  83. func medialibrary(_ medialibrary: MediaLibraryService, didDeleteAlbumsWithIds albumsIds: [NSNumber]) {
  84. files.removeAll {
  85. albumsIds.contains(NSNumber(value: $0.identifier()))
  86. }
  87. updateView?()
  88. }
  89. }
  90. extension VLCMLAlbum: MediaCollectionModel {
  91. func sortModel() -> SortModel? {
  92. return nil
  93. }
  94. func files(with criteria: VLCMLSortingCriteria = .alpha,
  95. desc: Bool = false) -> [VLCMLMedia]? {
  96. return tracks(with: criteria, desc: desc)
  97. }
  98. func title() -> String {
  99. return title
  100. }
  101. }
  102. extension VLCMLAlbum {
  103. func numberOfTracksString() -> String {
  104. let trackCount = numberOfTracks()
  105. let tracksString = trackCount > 1 ? NSLocalizedString("TRACKS", comment: "") : NSLocalizedString("TRACK", comment: "")
  106. return String(format: tracksString, trackCount)
  107. }
  108. @objc func thumbnail() -> UIImage? {
  109. var image = UIImage(contentsOfFile: artworkMRL()?.path ?? "")
  110. if image == nil {
  111. for track in files() ?? [] where track.isThumbnailGenerated() {
  112. image = UIImage(contentsOfFile: track.thumbnail()?.path ?? "")
  113. break
  114. }
  115. }
  116. if image == nil {
  117. let isDarktheme = PresentationTheme.current == PresentationTheme.darkTheme
  118. image = isDarktheme ? UIImage(named: "album-placeholder-dark") : UIImage(named: "album-placeholder-white")
  119. }
  120. return image
  121. }
  122. func albumName() -> String {
  123. return isUnknownAlbum() ? NSLocalizedString("UNKNOWN_ALBUM", comment: "") : title
  124. }
  125. func albumArtistName() -> String {
  126. guard let artist = albumArtist else {
  127. return NSLocalizedString("UNKNOWN_ARTIST", comment: "")
  128. }
  129. return artist.artistName()
  130. }
  131. func accessibilityText(editing: Bool) -> String? {
  132. if editing {
  133. return albumName() + " " + albumArtistName() + " " + numberOfTracksString()
  134. }
  135. return albumName() + " " + albumArtistName()
  136. }
  137. }
  138. extension VLCMLAlbumTrack {
  139. func albumArtistName() -> String {
  140. guard let artist = artist else {
  141. return NSLocalizedString("UNKNOWN_ARTIST", comment: "")
  142. }
  143. return artist.artistName()
  144. }
  145. }