NCDeepLinkHandler.swift 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. //
  2. // DeepLinkHandler.swift
  3. // Nextcloud
  4. //
  5. // Created by Amrut Waghmare on 29/05/24.
  6. // Copyright © 2024 Marino Faggiana. All rights reserved.
  7. //
  8. // This program is free software: you can redistribute it and/or modify
  9. // it under the terms of the GNU General Public License as published by
  10. // the Free Software Foundation, either version 3 of the License, or
  11. // (at your option) any later version.
  12. //
  13. // This program is distributed in the hope that it will be useful,
  14. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. // GNU General Public License for more details.
  17. //
  18. // You should have received a copy of the GNU General Public License
  19. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  20. //
  21. import Foundation
  22. import UIKit
  23. import SwiftUI
  24. enum DeepLink: String {
  25. case openFiles // nextcloud://openFiles
  26. case openFavorites // nextcloud://openFavorites
  27. case openMedia // nextcloud://openMedia
  28. case openShared // nextcloud://openShared
  29. case openOffline // nextcloud://openOffline
  30. case openNotifications // nextcloud://openNotifications
  31. case openDeleted // nextcloud://openDeleted
  32. case openSettings // nextcloud://openSettings
  33. case openAutoUpload // nextcloud://openAutoUpload
  34. case openUrl // nextcloud://openUrl?url=https://nextcloud.com
  35. case createNew // nextcloud://createNew
  36. case checkAppUpdate // nextcloud://checkAppUpdate
  37. }
  38. enum ControllerConstants {
  39. static let filesIndex = 0
  40. static let favouriteIndex = 1
  41. static let mediaIndex = 3
  42. static let moreIndex = 4
  43. static let notification = "NCNotification"
  44. static let shares = "segueShares"
  45. static let offline = "segueOffline"
  46. static let delete = "segueTrash"
  47. }
  48. class NCDeepLinkHandler {
  49. func parseDeepLink(_ url: URL, controller: NCMainTabBarController) {
  50. guard let action = url.host, let deepLink = DeepLink(rawValue: action) else { return }
  51. let params = getQueryParamsFromUrl(url: url)
  52. handleDeepLink(deepLink, controller: controller, params: params)
  53. }
  54. func getQueryParamsFromUrl(url: URL) -> [String: Any]? {
  55. guard
  56. let components = URLComponents(url: url, resolvingAgainstBaseURL: true),
  57. let queryItems = components.queryItems else { return nil }
  58. return queryItems.reduce(into: [String: Any]()) { result, item in
  59. result[item.name] = item.value
  60. }
  61. }
  62. func handleDeepLink(_ deepLink: DeepLink, controller: NCMainTabBarController, params: [String: Any]? = nil) {
  63. switch deepLink {
  64. case .openFiles:
  65. navigateTo(index: ControllerConstants.filesIndex, controller: controller)
  66. case .openFavorites:
  67. navigateTo(index: ControllerConstants.favouriteIndex, controller: controller)
  68. case .openMedia:
  69. navigateTo(index: ControllerConstants.mediaIndex, controller: controller)
  70. case .openShared:
  71. navigateToMore(withSegue: ControllerConstants.shares, controller: controller)
  72. case .openOffline:
  73. navigateToMore(withSegue: ControllerConstants.offline, controller: controller)
  74. case .openNotifications:
  75. navigateToNotification(controller: controller)
  76. case .openDeleted:
  77. navigateToMore(withSegue: ControllerConstants.delete, controller: controller)
  78. case .openSettings:
  79. navigateToSettings(controller: controller)
  80. case .openAutoUpload:
  81. navigateToAutoUpload(controller: controller)
  82. case .openUrl:
  83. openUrl(params: params)
  84. case .createNew:
  85. navigateToCreateNew(controller: controller)
  86. case .checkAppUpdate:
  87. navigateAppUpdate()
  88. }
  89. }
  90. private func navigateTo(index: Int, controller: NCMainTabBarController) {
  91. controller.selectedIndex = index
  92. }
  93. private func navigateToNotification(controller: NCMainTabBarController) {
  94. controller.selectedIndex = ControllerConstants.filesIndex
  95. guard let navigationController = controller.viewControllers?[controller.selectedIndex] as? UINavigationController,
  96. let viewController = UIStoryboard(name: ControllerConstants.notification, bundle: nil).instantiateInitialViewController() as? NCNotification else { return }
  97. navigationController.pushViewController(viewController, animated: true)
  98. }
  99. private func navigateToCreateNew(controller: NCMainTabBarController) {
  100. guard let appDelegate = UIApplication.shared.delegate as? AppDelegate else { return }
  101. controller.selectedIndex = ControllerConstants.filesIndex
  102. DispatchQueue.main.asyncAfter(deadline: .now() + 4) {
  103. appDelegate.toggleMenu(controller: controller)
  104. }
  105. }
  106. private func navigateToMore(withSegue segue: String, controller: NCMainTabBarController) {
  107. controller.selectedIndex = ControllerConstants.moreIndex
  108. guard let navigationController = controller.viewControllers?[controller.selectedIndex] as? UINavigationController else { return }
  109. navigationController.viewControllers = navigationController.viewControllers.filter({$0.isKind(of: NCMore.self)})
  110. navigationController.performSegue(withIdentifier: segue, sender: self)
  111. }
  112. private func navigateToSettings(controller: NCMainTabBarController) {
  113. controller.selectedIndex = ControllerConstants.moreIndex
  114. guard let navigationController = controller.viewControllers?[controller.selectedIndex] as? UINavigationController else { return }
  115. let settingsView = NCSettingsView(model: NCSettingsModel(controller: controller))
  116. let settingsController = UIHostingController(rootView: settingsView)
  117. navigationController.pushViewController(settingsController, animated: true)
  118. }
  119. private func navigateToAutoUpload(controller: NCMainTabBarController) {
  120. controller.selectedIndex = ControllerConstants.moreIndex
  121. guard let navigationController = controller.viewControllers?[controller.selectedIndex] as? UINavigationController else { return }
  122. let autoUploadView = NCAutoUploadView(model: NCAutoUploadModel(controller: controller))
  123. let autoUploadController = UIHostingController(rootView: autoUploadView)
  124. navigationController.pushViewController(autoUploadController, animated: true)
  125. }
  126. private func navigateAppUpdate() {
  127. guard let url = URL(string: NCBrandOptions.shared.appStoreUrl) else { return }
  128. handleUrl(url: url)
  129. }
  130. private func openUrl(params: [String: Any]?) {
  131. guard let urlString = params?["url"] as? String, let url = URL(string: urlString) else { return }
  132. handleUrl(url: url)
  133. }
  134. private func handleUrl(url: URL) {
  135. if UIApplication.shared.canOpenURL(url) {
  136. UIApplication.shared.open(url, options: [:], completionHandler: nil)
  137. }
  138. }
  139. }