MediaViewController.swift 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. /*****************************************************************************
  2. * MediaViewController.swift
  3. * VLC for iOS
  4. *****************************************************************************
  5. * Copyright (c) 2018 VideoLAN. All rights reserved.
  6. * $Id$
  7. *
  8. * Authors: Carola Nitz <nitz.carola # gmail.com>
  9. * Mike JS. Choi <mkchoi212 # icloud.com>
  10. *
  11. * Refer to the COPYING file of the official project for license.
  12. *****************************************************************************/
  13. import Foundation
  14. @objc protocol VLCMediaViewControllerDelegate: class {
  15. func mediaViewControllerDidSelectMediaObject(_ mediaViewController: VLCMediaViewController, mediaObject: NSManagedObject)
  16. }
  17. class VLCMediaViewController: UICollectionViewController, UISearchResultsUpdating, UISearchControllerDelegate, IndicatorInfoProvider {
  18. private var services: Services
  19. private var mediaDataSourceAndDelegate: MediaDataSourceAndDelegate?
  20. private var searchController: UISearchController?
  21. private let searchDataSource = VLCLibrarySearchDisplayDataSource()
  22. private var mediaType: VLCMediaType
  23. weak var delegate: VLCMediaViewControllerDelegate?
  24. @available(iOS 11.0, *)
  25. lazy var dragAndDropManager: VLCDragAndDropManager = {
  26. let dragAndDropManager = VLCDragAndDropManager(type: mediaType)
  27. dragAndDropManager.delegate = services.mediaDataSource
  28. return dragAndDropManager
  29. }()
  30. lazy var emptyView: VLCEmptyLibraryView = {
  31. let name = String(describing: VLCEmptyLibraryView.self)
  32. let nib = Bundle.main.loadNibNamed(name, owner: self, options: nil)
  33. guard let emptyView = nib?.first as? VLCEmptyLibraryView else { fatalError("Can't find nib for \(name)") }
  34. return emptyView
  35. }()
  36. @available(*, unavailable)
  37. init() {
  38. fatalError()
  39. }
  40. init(services: Services, type: VLCMediaType) {
  41. self.services = services
  42. mediaType = type
  43. super.init(collectionViewLayout: UICollectionViewFlowLayout())
  44. NotificationCenter.default.addObserver(self, selector: #selector(themeDidChange), name: .VLCThemeDidChangeNotification, object: nil)
  45. if mediaType.category == .video {
  46. NotificationCenter.default.addObserver(self, selector: #selector(reloadData), name: .VLCAllVideosDidChangeNotification, object: nil)
  47. } else {
  48. NotificationCenter.default.addObserver(self, selector: #selector(reloadData), name: .VLCTracksDidChangeNotification, object: nil)
  49. }
  50. }
  51. override var preferredStatusBarStyle: UIStatusBarStyle {
  52. return PresentationTheme.current.colors.statusBarStyle
  53. }
  54. @objc func reloadData() {
  55. collectionView?.reloadData()
  56. updateUIForContent()
  57. }
  58. @available(*, unavailable)
  59. required public init?(coder aDecoder: NSCoder) {
  60. fatalError("init(coder: ) has not been implemented")
  61. }
  62. public override func viewDidLoad() {
  63. super.viewDidLoad()
  64. setupCollectionView()
  65. setupSearchController()
  66. _ = (MLMediaLibrary.sharedMediaLibrary() as! MLMediaLibrary).libraryDidAppear()
  67. }
  68. public override func viewWillAppear(_ animated: Bool) {
  69. super.viewWillAppear(animated)
  70. let manager = services.rendererDiscovererManager
  71. if manager.discoverers.isEmpty {
  72. // Either didn't start or stopped before
  73. manager.start()
  74. }
  75. manager.presentingViewController = self
  76. }
  77. @objc func themeDidChange() {
  78. collectionView?.backgroundColor = PresentationTheme.current.colors.background
  79. setNeedsStatusBarAppearanceUpdate()
  80. }
  81. func setupCollectionView() {
  82. mediaDataSourceAndDelegate = MediaDataSourceAndDelegate(services: services, type: mediaType)
  83. mediaDataSourceAndDelegate?.delegate = self
  84. let playlistnib = UINib(nibName: "VLCPlaylistCollectionViewCell", bundle: nil)
  85. collectionView?.register(playlistnib, forCellWithReuseIdentifier: VLCPlaylistCollectionViewCell.cellIdentifier())
  86. collectionView?.backgroundColor = PresentationTheme.current.colors.background
  87. collectionView?.alwaysBounceVertical = true
  88. collectionView?.dataSource = mediaDataSourceAndDelegate
  89. collectionView?.delegate = mediaDataSourceAndDelegate
  90. if #available(iOS 11.0, *) {
  91. collectionView?.dragDelegate = dragAndDropManager
  92. collectionView?.dropDelegate = dragAndDropManager
  93. }
  94. }
  95. public override func viewDidAppear(_ animated: Bool) {
  96. super.viewDidAppear(animated)
  97. reloadData()
  98. }
  99. func setupSearchController() {
  100. searchController = UISearchController(searchResultsController: nil)
  101. searchController?.searchResultsUpdater = self
  102. searchController?.dimsBackgroundDuringPresentation = false
  103. searchController?.delegate = self
  104. if let textfield = searchController?.searchBar.value(forKey: "searchField") as? UITextField {
  105. if let backgroundview = textfield.subviews.first {
  106. backgroundview.backgroundColor = UIColor.white
  107. backgroundview.layer.cornerRadius = 10
  108. backgroundview.clipsToBounds = true
  109. }
  110. }
  111. }
  112. func updateUIForContent() {
  113. let isEmpty = collectionView?.numberOfItems(inSection: 0) == 0
  114. if isEmpty {
  115. collectionView?.setContentOffset(.zero, animated: false)
  116. }
  117. collectionView?.backgroundView = isEmpty ? emptyView : nil
  118. if #available(iOS 11.0, *) {
  119. navigationItem.searchController = isEmpty ? nil : searchController
  120. } else {
  121. navigationItem.titleView = isEmpty ? nil : searchController?.searchBar
  122. }
  123. }
  124. // MARK: Renderer
  125. public override func viewWillTransition(to size: CGSize, with coordinator: UIViewControllerTransitionCoordinator) {
  126. collectionView?.collectionViewLayout.invalidateLayout()
  127. }
  128. // MARK: - MediaDatasourceAndDelegate
  129. override public func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
  130. if let mediaObject = services.mediaDataSource.object(at: indexPath.row, subcategory: mediaType.subcategory) as? NSManagedObject {
  131. delegate?.mediaViewControllerDidSelectMediaObject(self, mediaObject: mediaObject)
  132. }
  133. }
  134. // MARK: - Search
  135. public func updateSearchResults(for searchController: UISearchController) {
  136. searchDataSource.shouldReloadTable(forSearch: searchController.searchBar.text, searchableFiles: services.mediaDataSource.allObjects(for: mediaType.subcategory))
  137. collectionView?.reloadData()
  138. }
  139. public func didPresentSearchController(_ searchController: UISearchController) {
  140. collectionView?.dataSource = searchDataSource
  141. }
  142. public func didDismissSearchController(_ searchController: UISearchController) {
  143. collectionView?.dataSource = mediaDataSourceAndDelegate
  144. }
  145. func indicatorInfo(for pagerTabStripController: PagerTabStripViewController) -> IndicatorInfo {
  146. return services.mediaDataSource.indicatorInfo(for:mediaType.subcategory)
  147. }
  148. }