AppDelegate.swift 52 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144
  1. //
  2. // AppDelegate.swift
  3. // Nextcloud
  4. //
  5. // Created by Marino Faggiana on 04/09/14 (19/02/21 swift).
  6. // Copyright (c) 2014 Marino Faggiana. All rights reserved.
  7. //
  8. // Author Marino Faggiana <marino.faggiana@nextcloud.com>
  9. //
  10. // This program is free software: you can redistribute it and/or modify
  11. // it under the terms of the GNU General Public License as published by
  12. // the Free Software Foundation, either version 3 of the License, or
  13. // (at your option) any later version.
  14. //
  15. // This program is distributed in the hope that it will be useful,
  16. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. // GNU General Public License for more details.
  19. //
  20. // You should have received a copy of the GNU General Public License
  21. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  22. //
  23. import UIKit
  24. import BackgroundTasks
  25. import NextcloudKit
  26. import TOPasscodeViewController
  27. import LocalAuthentication
  28. import Firebase
  29. import WidgetKit
  30. import Queuer
  31. @UIApplicationMain
  32. class AppDelegate: UIResponder, UIApplicationDelegate, UNUserNotificationCenterDelegate, TOPasscodeViewControllerDelegate, NCAccountRequestDelegate, NCViewCertificateDetailsDelegate, NCUserBaseUrl {
  33. var backgroundSessionCompletionHandler: (() -> Void)?
  34. var window: UIWindow?
  35. @objc var account: String = ""
  36. @objc var urlBase: String = ""
  37. @objc var user: String = ""
  38. @objc var userId: String = ""
  39. @objc var password: String = ""
  40. var activeLogin: NCLogin?
  41. var activeLoginWeb: NCLoginWeb?
  42. var activeServerUrl: String = ""
  43. @objc var activeViewController: UIViewController?
  44. var mainTabBar: NCMainTabBar?
  45. var activeMetadata: tableMetadata?
  46. let listFilesVC = ThreadSafeDictionary<String, NCFiles>()
  47. var disableSharesView: Bool = false
  48. var documentPickerViewController: NCDocumentPickerViewController?
  49. var timerErrorNetworking: Timer?
  50. private var privacyProtectionWindow: UIWindow?
  51. var isAppRefresh: Bool = false
  52. var isAppProcessing: Bool = false
  53. var isUiTestingEnabled: Bool {
  54. return ProcessInfo.processInfo.arguments.contains("UI_TESTING")
  55. }
  56. func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
  57. if isUiTestingEnabled {
  58. deleteAllAccounts()
  59. }
  60. let utilityFileSystem = NCUtilityFileSystem()
  61. let utility = NCUtility()
  62. NCSettingsBundleHelper.checkAndExecuteSettings(delay: 0)
  63. let versionNextcloudiOS = String(format: NCBrandOptions.shared.textCopyrightNextcloudiOS, utility.getVersionApp())
  64. UserDefaults.standard.register(defaults: ["UserAgent": userAgent])
  65. if !NCKeychain().disableCrashservice, !NCBrandOptions.shared.disable_crash_service {
  66. FirebaseApp.configure()
  67. }
  68. utilityFileSystem.createDirectoryStandard()
  69. utilityFileSystem.emptyTemporaryDirectory()
  70. utilityFileSystem.clearCacheDirectory("com.limit-point.LivePhoto")
  71. // Activated singleton
  72. _ = NCActionCenter.shared
  73. _ = NCNetworking.shared
  74. NextcloudKit.shared.setup(delegate: NCNetworking.shared)
  75. NextcloudKit.shared.setup(userAgent: userAgent)
  76. startTimerErrorNetworking()
  77. var levelLog = 0
  78. NextcloudKit.shared.nkCommonInstance.pathLog = utilityFileSystem.directoryGroup
  79. if NCBrandOptions.shared.disable_log {
  80. utilityFileSystem.removeFile(atPath: NextcloudKit.shared.nkCommonInstance.filenamePathLog)
  81. utilityFileSystem.removeFile(atPath: NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true).first! + "/" + NextcloudKit.shared.nkCommonInstance.filenameLog)
  82. } else {
  83. levelLog = NCKeychain().logLevel
  84. NextcloudKit.shared.nkCommonInstance.levelLog = levelLog
  85. NextcloudKit.shared.nkCommonInstance.copyLogToDocumentDirectory = true
  86. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Start session with level \(levelLog) " + versionNextcloudiOS)
  87. }
  88. if let account = NCManageDatabase.shared.getActiveAccount() {
  89. NextcloudKit.shared.nkCommonInstance.writeLog("Account active \(account.account)")
  90. if NCKeychain().getPassword(account: account.account).isEmpty {
  91. NextcloudKit.shared.nkCommonInstance.writeLog("[ERROR] PASSWORD NOT FOUND for \(account.account)")
  92. }
  93. }
  94. if let activeAccount = NCManageDatabase.shared.getActiveAccount() {
  95. account = activeAccount.account
  96. urlBase = activeAccount.urlBase
  97. user = activeAccount.user
  98. userId = activeAccount.userId
  99. password = NCKeychain().getPassword(account: account)
  100. NextcloudKit.shared.setup(account: account, user: user, userId: userId, password: password, urlBase: urlBase)
  101. NCManageDatabase.shared.setCapabilities(account: account)
  102. NCBrandColor.shared.settingThemingColor(account: activeAccount.account)
  103. } else {
  104. NCKeychain().removeAll()
  105. if let bundleID = Bundle.main.bundleIdentifier {
  106. UserDefaults.standard.removePersistentDomain(forName: bundleID)
  107. }
  108. }
  109. NCBrandColor.shared.createUserColors()
  110. NCImageCache.shared.createImagesCache()
  111. // Push Notification & display notification
  112. application.registerForRemoteNotifications()
  113. UNUserNotificationCenter.current().delegate = self
  114. UNUserNotificationCenter.current().requestAuthorization(options: [.alert, .badge, .sound]) { _, _ in }
  115. if !utility.isSimulatorOrTestFlight() {
  116. let review = NCStoreReview()
  117. review.incrementAppRuns()
  118. review.showStoreReview()
  119. }
  120. // Background task register
  121. BGTaskScheduler.shared.register(forTaskWithIdentifier: NCGlobal.shared.refreshTask, using: nil) { task in
  122. self.handleAppRefresh(task)
  123. }
  124. BGTaskScheduler.shared.register(forTaskWithIdentifier: NCGlobal.shared.processingTask, using: nil) { task in
  125. self.handleProcessingTask(task)
  126. }
  127. // Intro
  128. if NCBrandOptions.shared.disable_intro {
  129. if account.isEmpty {
  130. openLogin(viewController: nil, selector: NCGlobal.shared.introLogin, openLoginWeb: false)
  131. }
  132. } else {
  133. if !NCKeychain().intro {
  134. if let viewController = UIStoryboard(name: "NCIntro", bundle: nil).instantiateInitialViewController() {
  135. let navigationController = NCLoginNavigationController(rootViewController: viewController)
  136. window?.rootViewController = navigationController
  137. window?.makeKeyAndVisible()
  138. }
  139. }
  140. }
  141. self.presentPasscode {
  142. self.enableTouchFaceID()
  143. }
  144. return true
  145. }
  146. // MARK: - Life Cycle
  147. // L' applicazione entrerà in attivo (sempre)
  148. func applicationDidBecomeActive(_ application: UIApplication) {
  149. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Application did become active")
  150. DispatchQueue.global().async { NCImageCache.shared.createMediaCache(account: self.account) }
  151. NCSettingsBundleHelper.setVersionAndBuildNumber()
  152. NCSettingsBundleHelper.checkAndExecuteSettings(delay: 0.5)
  153. // START OBSERVE/TIMER UPLOAD PROCESS
  154. NCNetworkingProcessUpload.shared.observeTableMetadata()
  155. NCNetworkingProcessUpload.shared.startTimer()
  156. if !NCAskAuthorization().isRequesting {
  157. hidePrivacyProtectionWindow()
  158. }
  159. NCService().startRequestServicesServer()
  160. NCAutoUpload.shared.initAutoUpload(viewController: nil) { items in
  161. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Initialize Auto upload with \(items) uploads")
  162. }
  163. NotificationCenter.default.postOnMainThread(name: NCGlobal.shared.notificationCenterApplicationDidBecomeActive)
  164. }
  165. // L' applicazione si dimetterà dallo stato di attivo
  166. func applicationWillResignActive(_ application: UIApplication) {
  167. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Application will resign active")
  168. guard !account.isEmpty else { return }
  169. // STOP OBSERVE/TIMER UPLOAD PROCESS
  170. NCNetworkingProcessUpload.shared.invalidateObserveTableMetadata()
  171. NCNetworkingProcessUpload.shared.stopTimer()
  172. if NCKeychain().privacyScreenEnabled {
  173. showPrivacyProtectionWindow()
  174. }
  175. // Reload Widget
  176. WidgetCenter.shared.reloadAllTimelines()
  177. // Clear older files
  178. let days = NCKeychain().cleanUpDay
  179. let utilityFileSystem = NCUtilityFileSystem()
  180. utilityFileSystem.cleanUp(directory: utilityFileSystem.directoryProviderStorage, days: TimeInterval(days))
  181. NotificationCenter.default.postOnMainThread(name: NCGlobal.shared.notificationCenterApplicationWillResignActive)
  182. }
  183. // L' applicazione entrerà in primo piano (dopo il background)
  184. func applicationWillEnterForeground(_ application: UIApplication) {
  185. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Application will enter in foreground")
  186. guard !account.isEmpty else { return }
  187. enableTouchFaceID()
  188. NotificationCenter.default.postOnMainThread(name: NCGlobal.shared.notificationCenterApplicationWillEnterForeground)
  189. NotificationCenter.default.postOnMainThread(name: NCGlobal.shared.notificationCenterRichdocumentGrabFocus)
  190. NotificationCenter.default.postOnMainThread(name: NCGlobal.shared.notificationCenterReloadDataSourceNetwork, second: 2)
  191. }
  192. // L' applicazione è entrata nello sfondo
  193. func applicationDidEnterBackground(_ application: UIApplication) {
  194. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Application did enter in background")
  195. guard !account.isEmpty else { return }
  196. let activeAccount = NCManageDatabase.shared.getActiveAccount()
  197. if let autoUpload = activeAccount?.autoUpload, autoUpload {
  198. NextcloudKit.shared.nkCommonInstance.writeLog("- Auto upload: true")
  199. if UIApplication.shared.backgroundRefreshStatus == .available {
  200. NextcloudKit.shared.nkCommonInstance.writeLog("- Auto upload in background: true")
  201. } else {
  202. NextcloudKit.shared.nkCommonInstance.writeLog("- Auto upload in background: false")
  203. }
  204. } else {
  205. NextcloudKit.shared.nkCommonInstance.writeLog("- Auto upload: false")
  206. }
  207. if let error = updateShareAccounts() {
  208. NextcloudKit.shared.nkCommonInstance.writeLog("[ERROR] Create share accounts \(error.localizedDescription)")
  209. }
  210. scheduleAppRefresh()
  211. scheduleAppProcessing()
  212. NCNetworking.shared.cancelDataTask()
  213. NCNetworking.shared.cancelDownloadTasks()
  214. presentPasscode { }
  215. NotificationCenter.default.postOnMainThread(name: NCGlobal.shared.notificationCenterApplicationDidEnterBackground)
  216. }
  217. // L'applicazione terminerà
  218. func applicationWillTerminate(_ application: UIApplication) {
  219. if UIApplication.shared.backgroundRefreshStatus == .available {
  220. let content = UNMutableNotificationContent()
  221. content.title = NCBrandOptions.shared.brand
  222. content.body = NSLocalizedString("_keep_running_", comment: "")
  223. let req = UNNotificationRequest(identifier: UUID().uuidString, content: content, trigger: nil)
  224. let notificationCenter = UNUserNotificationCenter.current()
  225. notificationCenter.add(req)
  226. }
  227. NextcloudKit.shared.nkCommonInstance.writeLog("bye bye")
  228. }
  229. // MARK: - Background Task
  230. /*
  231. @discussion Schedule a refresh task request to ask that the system launch your app briefly so that you can download data and keep your app's contents up-to-date. The system will fulfill this request intelligently based on system conditions and app usage.
  232. < MAX 30 seconds >
  233. */
  234. func scheduleAppRefresh() {
  235. let request = BGAppRefreshTaskRequest(identifier: NCGlobal.shared.refreshTask)
  236. request.earliestBeginDate = Date(timeIntervalSinceNow: 60) // Refresh after 60 seconds.
  237. do {
  238. try BGTaskScheduler.shared.submit(request)
  239. NextcloudKit.shared.nkCommonInstance.writeLog("- Refresh task: ok")
  240. } catch {
  241. NextcloudKit.shared.nkCommonInstance.writeLog("[ERROR] Refresh task failed to submit request: \(error)")
  242. }
  243. }
  244. /*
  245. @discussion Schedule a processing task request to ask that the system launch your app when conditions are favorable for battery life to handle deferrable, longer-running processing, such as syncing, database maintenance, or similar tasks. The system will attempt to fulfill this request to the best of its ability within the next two days as long as the user has used your app within the past week.
  246. < MAX over 1 minute >
  247. */
  248. func scheduleAppProcessing() {
  249. let request = BGProcessingTaskRequest(identifier: NCGlobal.shared.processingTask)
  250. request.earliestBeginDate = Date(timeIntervalSinceNow: 5 * 60) // Refresh after 5 minutes.
  251. request.requiresNetworkConnectivity = false
  252. request.requiresExternalPower = false
  253. do {
  254. try BGTaskScheduler.shared.submit(request)
  255. NextcloudKit.shared.nkCommonInstance.writeLog("- Processing task: ok")
  256. } catch {
  257. NextcloudKit.shared.nkCommonInstance.writeLog("[ERROR] Background Processing task failed to submit request: \(error)")
  258. }
  259. }
  260. func handleAppRefresh(_ task: BGTask) {
  261. scheduleAppRefresh()
  262. if isAppProcessing {
  263. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Processing task already in progress, abort.")
  264. task.setTaskCompleted(success: true)
  265. return
  266. }
  267. isAppRefresh = true
  268. NCAutoUpload.shared.initAutoUpload(viewController: nil) { items in
  269. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Refresh task auto upload with \(items) uploads")
  270. NCNetworkingProcessUpload.shared.start { items in
  271. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Refresh task upload process with \(items) uploads")
  272. task.setTaskCompleted(success: true)
  273. self.isAppRefresh = false
  274. }
  275. }
  276. }
  277. func handleProcessingTask(_ task: BGTask) {
  278. scheduleAppProcessing()
  279. if isAppRefresh {
  280. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Refresh task already in progress, abort.")
  281. task.setTaskCompleted(success: true)
  282. return
  283. }
  284. isAppProcessing = true
  285. NCAutoUpload.shared.initAutoUpload(viewController: nil) { items in
  286. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Processing task auto upload with \(items) uploads")
  287. NCNetworkingProcessUpload.shared.start { items in
  288. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Processing task upload process with \(items) uploads")
  289. task.setTaskCompleted(success: true)
  290. self.isAppProcessing = false
  291. }
  292. }
  293. }
  294. // MARK: - Background Networking Session
  295. func application(_ application: UIApplication, handleEventsForBackgroundURLSession identifier: String, completionHandler: @escaping () -> Void) {
  296. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Start handle Events For Background URLSession: \(identifier)")
  297. WidgetCenter.shared.reloadAllTimelines()
  298. backgroundSessionCompletionHandler = completionHandler
  299. }
  300. // MARK: - Push Notifications
  301. func userNotificationCenter(_ center: UNUserNotificationCenter, willPresent notification: UNNotification, withCompletionHandler completionHandler: @escaping (UNNotificationPresentationOptions) -> Void) {
  302. completionHandler([.list, .banner, .sound])
  303. }
  304. func userNotificationCenter(_ center: UNUserNotificationCenter, didReceive response: UNNotificationResponse, withCompletionHandler completionHandler: @escaping () -> Void) {
  305. if let pref = UserDefaults(suiteName: NCBrandOptions.shared.capabilitiesGroups),
  306. let data = pref.object(forKey: "NOTIFICATION_DATA") as? [String: AnyObject] {
  307. nextcloudPushNotificationAction(data: data)
  308. pref.set(nil, forKey: "NOTIFICATION_DATA")
  309. }
  310. completionHandler()
  311. }
  312. func application(_ application: UIApplication, didRegisterForRemoteNotificationsWithDeviceToken deviceToken: Data) {
  313. NCNetworking.shared.checkPushNotificationServerProxyCertificateUntrusted(viewController: self.window?.rootViewController) { error in
  314. if error == .success {
  315. NCPushNotification.shared().registerForRemoteNotifications(withDeviceToken: deviceToken)
  316. }
  317. }
  318. }
  319. func application(_ application: UIApplication, didReceiveRemoteNotification userInfo: [AnyHashable: Any], fetchCompletionHandler completionHandler: @escaping (UIBackgroundFetchResult) -> Void) {
  320. NCPushNotification.shared().applicationdidReceiveRemoteNotification(userInfo) { result in
  321. completionHandler(result)
  322. }
  323. }
  324. func nextcloudPushNotificationAction(data: [String: AnyObject]) {
  325. guard let data = NCApplicationHandle().nextcloudPushNotificationAction(data: data) else { return }
  326. var findAccount: Bool = false
  327. if let accountPush = data["account"] as? String {
  328. if accountPush == self.account {
  329. findAccount = true
  330. } else {
  331. let accounts = NCManageDatabase.shared.getAllAccount()
  332. for account in accounts {
  333. if account.account == accountPush {
  334. self.changeAccount(account.account, userProfile: nil)
  335. findAccount = true
  336. }
  337. }
  338. }
  339. if findAccount, let viewController = UIStoryboard(name: "NCNotification", bundle: nil).instantiateInitialViewController() as? NCNotification {
  340. DispatchQueue.main.asyncAfter(deadline: .now() + 1) {
  341. let navigationController = UINavigationController(rootViewController: viewController)
  342. navigationController.modalPresentationStyle = .fullScreen
  343. self.window?.rootViewController?.present(navigationController, animated: true)
  344. }
  345. } else if !findAccount {
  346. let message = NSLocalizedString("_the_account_", comment: "") + " " + accountPush + " " + NSLocalizedString("_does_not_exist_", comment: "")
  347. let alertController = UIAlertController(title: NSLocalizedString("_info_", comment: ""), message: message, preferredStyle: .alert)
  348. alertController.addAction(UIAlertAction(title: NSLocalizedString("_ok_", comment: ""), style: .default, handler: { _ in }))
  349. self.window?.rootViewController?.present(alertController, animated: true, completion: { })
  350. }
  351. }
  352. }
  353. // MARK: - Login & checkErrorNetworking
  354. @objc func openLogin(viewController: UIViewController?, selector: Int, openLoginWeb: Bool) {
  355. // [WEBPersonalized] [AppConfig]
  356. if NCBrandOptions.shared.use_login_web_personalized || NCBrandOptions.shared.use_AppConfig {
  357. if activeLoginWeb?.view.window == nil {
  358. activeLoginWeb = UIStoryboard(name: "NCLogin", bundle: nil).instantiateViewController(withIdentifier: "NCLoginWeb") as? NCLoginWeb
  359. activeLoginWeb?.urlBase = NCBrandOptions.shared.loginBaseUrl
  360. showLoginViewController(activeLoginWeb, contextViewController: viewController)
  361. }
  362. return
  363. }
  364. // Nextcloud standard login
  365. if selector == NCGlobal.shared.introSignup {
  366. if activeLoginWeb?.view.window == nil {
  367. activeLoginWeb = UIStoryboard(name: "NCLogin", bundle: nil).instantiateViewController(withIdentifier: "NCLoginWeb") as? NCLoginWeb
  368. if selector == NCGlobal.shared.introSignup {
  369. activeLoginWeb?.urlBase = NCBrandOptions.shared.linkloginPreferredProviders
  370. } else {
  371. activeLoginWeb?.urlBase = self.urlBase
  372. }
  373. showLoginViewController(activeLoginWeb, contextViewController: viewController)
  374. }
  375. } else if NCBrandOptions.shared.disable_intro && NCBrandOptions.shared.disable_request_login_url {
  376. if activeLoginWeb?.view.window == nil {
  377. activeLoginWeb = UIStoryboard(name: "NCLogin", bundle: nil).instantiateViewController(withIdentifier: "NCLoginWeb") as? NCLoginWeb
  378. activeLoginWeb?.urlBase = NCBrandOptions.shared.loginBaseUrl
  379. showLoginViewController(activeLoginWeb, contextViewController: viewController)
  380. }
  381. } else if openLoginWeb {
  382. // Used also for reinsert the account (change passwd)
  383. if activeLoginWeb?.view.window == nil {
  384. activeLoginWeb = UIStoryboard(name: "NCLogin", bundle: nil).instantiateViewController(withIdentifier: "NCLoginWeb") as? NCLoginWeb
  385. activeLoginWeb?.urlBase = urlBase
  386. activeLoginWeb?.user = user
  387. showLoginViewController(activeLoginWeb, contextViewController: viewController)
  388. }
  389. } else {
  390. if activeLogin?.view.window == nil {
  391. activeLogin = UIStoryboard(name: "NCLogin", bundle: nil).instantiateViewController(withIdentifier: "NCLogin") as? NCLogin
  392. showLoginViewController(activeLogin, contextViewController: viewController)
  393. }
  394. }
  395. }
  396. func showLoginViewController(_ viewController: UIViewController?, contextViewController: UIViewController?) {
  397. if contextViewController == nil {
  398. if let viewController = viewController {
  399. let navigationController = NCLoginNavigationController(rootViewController: viewController)
  400. navigationController.navigationBar.barStyle = .black
  401. navigationController.navigationBar.tintColor = NCBrandColor.shared.customerText
  402. navigationController.navigationBar.barTintColor = NCBrandColor.shared.customer
  403. navigationController.navigationBar.isTranslucent = false
  404. window?.rootViewController = navigationController
  405. window?.makeKeyAndVisible()
  406. }
  407. } else if contextViewController is UINavigationController {
  408. if let contextViewController = contextViewController, let viewController = viewController {
  409. (contextViewController as? UINavigationController)?.pushViewController(viewController, animated: true)
  410. }
  411. } else {
  412. if let viewController = viewController, let contextViewController = contextViewController {
  413. let navigationController = NCLoginNavigationController(rootViewController: viewController)
  414. navigationController.modalPresentationStyle = .fullScreen
  415. navigationController.navigationBar.barStyle = .black
  416. navigationController.navigationBar.tintColor = NCBrandColor.shared.customerText
  417. navigationController.navigationBar.barTintColor = NCBrandColor.shared.customer
  418. navigationController.navigationBar.isTranslucent = false
  419. contextViewController.present(navigationController, animated: true) { }
  420. }
  421. }
  422. }
  423. @objc func startTimerErrorNetworking() {
  424. timerErrorNetworking = Timer.scheduledTimer(timeInterval: 3, target: self, selector: #selector(checkErrorNetworking), userInfo: nil, repeats: true)
  425. }
  426. @objc private func checkErrorNetworking() {
  427. guard !account.isEmpty, NCKeychain().getPassword(account: account).isEmpty else { return }
  428. openLogin(viewController: window?.rootViewController, selector: NCGlobal.shared.introLogin, openLoginWeb: true)
  429. }
  430. func trustCertificateError(host: String) {
  431. guard let currentHost = URL(string: self.urlBase)?.host,
  432. let pushNotificationServerProxyHost = URL(string: NCBrandOptions.shared.pushNotificationServerProxy)?.host,
  433. host != pushNotificationServerProxyHost,
  434. host == currentHost
  435. else { return }
  436. let certificateHostSavedPath = NCUtilityFileSystem().directoryCertificates + "/" + host + ".der"
  437. var title = NSLocalizedString("_ssl_certificate_changed_", comment: "")
  438. if !FileManager.default.fileExists(atPath: certificateHostSavedPath) {
  439. title = NSLocalizedString("_connect_server_anyway_", comment: "")
  440. }
  441. let alertController = UIAlertController(title: title, message: NSLocalizedString("_server_is_trusted_", comment: ""), preferredStyle: .alert)
  442. alertController.addAction(UIAlertAction(title: NSLocalizedString("_yes_", comment: ""), style: .default, handler: { _ in
  443. NCNetworking.shared.writeCertificate(host: host)
  444. }))
  445. alertController.addAction(UIAlertAction(title: NSLocalizedString("_no_", comment: ""), style: .default, handler: { _ in }))
  446. alertController.addAction(UIAlertAction(title: NSLocalizedString("_certificate_details_", comment: ""), style: .default, handler: { _ in
  447. if let navigationController = UIStoryboard(name: "NCViewCertificateDetails", bundle: nil).instantiateInitialViewController() as? UINavigationController,
  448. let viewController = navigationController.topViewController as? NCViewCertificateDetails {
  449. viewController.delegate = self
  450. viewController.host = host
  451. self.window?.rootViewController?.present(navigationController, animated: true)
  452. }
  453. }))
  454. window?.rootViewController?.present(alertController, animated: true)
  455. }
  456. func viewCertificateDetailsDismiss(host: String) {
  457. trustCertificateError(host: host)
  458. }
  459. // MARK: - Account
  460. @objc func changeAccount(_ account: String, userProfile: NKUserProfile?) {
  461. NCNetworking.shared.cancelDataTask()
  462. NCNetworking.shared.cancelDownloadTasks()
  463. NCNetworking.shared.cancelUploadTasks()
  464. guard let tableAccount = NCManageDatabase.shared.setAccountActive(account) else { return }
  465. self.account = tableAccount.account
  466. self.urlBase = tableAccount.urlBase
  467. self.user = tableAccount.user
  468. self.userId = tableAccount.userId
  469. self.password = NCKeychain().getPassword(account: tableAccount.account)
  470. NextcloudKit.shared.setup(account: account, user: user, userId: userId, password: password, urlBase: urlBase)
  471. NCManageDatabase.shared.setCapabilities(account: account)
  472. if let userProfile {
  473. NCManageDatabase.shared.setAccountUserProfile(account: account, userProfile: userProfile)
  474. }
  475. NCPushNotification.shared().pushNotification()
  476. NCService().startRequestServicesServer()
  477. NCAutoUpload.shared.initAutoUpload(viewController: nil) { items in
  478. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Initialize Auto upload with \(items) uploads")
  479. }
  480. DispatchQueue.global().async {
  481. NCImageCache.shared.createMediaCache(account: self.account)
  482. NotificationCenter.default.postOnMainThread(name: NCGlobal.shared.notificationCenterChangeUser)
  483. }
  484. }
  485. @objc func deleteAccount(_ account: String, wipe: Bool) {
  486. if let account = NCManageDatabase.shared.getAccount(predicate: NSPredicate(format: "account == %@", account)) {
  487. NCPushNotification.shared().unsubscribingNextcloudServerPushNotification(account.account, urlBase: account.urlBase, user: account.user, withSubscribing: false)
  488. }
  489. let results = NCManageDatabase.shared.getTableLocalFiles(predicate: NSPredicate(format: "account == %@", account), sorted: "ocId", ascending: false)
  490. let utilityFileSystem = NCUtilityFileSystem()
  491. for result in results {
  492. utilityFileSystem.removeFile(atPath: utilityFileSystem.getDirectoryProviderStorageOcId(result.ocId))
  493. }
  494. NCManageDatabase.shared.clearDatabase(account: account, removeAccount: true)
  495. NCKeychain().clearAllKeysEndToEnd(account: account)
  496. NCKeychain().clearAllKeysPushNotification(account: account)
  497. NCKeychain().setPassword(account: account, password: nil)
  498. self.account = ""
  499. self.urlBase = ""
  500. self.user = ""
  501. self.userId = ""
  502. self.password = ""
  503. if wipe {
  504. let accounts = NCManageDatabase.shared.getAccounts()
  505. if accounts?.count ?? 0 > 0 {
  506. if let newAccount = accounts?.first {
  507. self.changeAccount(newAccount, userProfile: nil)
  508. }
  509. } else {
  510. openLogin(viewController: window?.rootViewController, selector: NCGlobal.shared.introLogin, openLoginWeb: false)
  511. }
  512. }
  513. }
  514. func deleteAllAccounts() {
  515. let accounts = NCManageDatabase.shared.getAccounts()
  516. accounts?.forEach({ account in
  517. deleteAccount(account, wipe: true)
  518. })
  519. }
  520. func updateShareAccounts() -> Error? {
  521. guard let dirGroupApps = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: NCBrandOptions.shared.capabilitiesGroupApps) else { return nil }
  522. let tableAccount = NCManageDatabase.shared.getAllAccount()
  523. var accounts = [NKShareAccounts.DataAccounts]()
  524. for account in tableAccount {
  525. let name = account.alias.isEmpty ? account.displayName : account.alias
  526. let userBaseUrl = account.user + "-" + (URL(string: account.urlBase)?.host ?? "")
  527. let avatarFileName = userBaseUrl + "-\(account.user).png"
  528. let pathAvatarFileName = NCUtilityFileSystem().directoryUserData + "/" + avatarFileName
  529. let image = UIImage(contentsOfFile: pathAvatarFileName)
  530. accounts.append(NKShareAccounts.DataAccounts(withUrl: account.urlBase, user: account.user, name: name, image: image))
  531. }
  532. return NKShareAccounts().putShareAccounts(at: dirGroupApps, app: NCGlobal.shared.appScheme, dataAccounts: accounts)
  533. }
  534. // MARK: - Account Request
  535. func accountRequestChangeAccount(account: String) {
  536. changeAccount(account, userProfile: nil)
  537. }
  538. func requestAccount() {
  539. if isPasscodePresented { return }
  540. if !NCKeychain().accountRequest { return }
  541. let accounts = NCManageDatabase.shared.getAllAccount()
  542. if accounts.count > 1 {
  543. if let vcAccountRequest = UIStoryboard(name: "NCAccountRequest", bundle: nil).instantiateInitialViewController() as? NCAccountRequest {
  544. vcAccountRequest.activeAccount = NCManageDatabase.shared.getActiveAccount()
  545. vcAccountRequest.accounts = accounts
  546. vcAccountRequest.enableTimerProgress = true
  547. vcAccountRequest.enableAddAccount = false
  548. vcAccountRequest.dismissDidEnterBackground = false
  549. vcAccountRequest.delegate = self
  550. let screenHeighMax = UIScreen.main.bounds.height - (UIScreen.main.bounds.height / 5)
  551. let numberCell = accounts.count
  552. let height = min(CGFloat(numberCell * Int(vcAccountRequest.heightCell) + 45), screenHeighMax)
  553. let popup = NCPopupViewController(contentController: vcAccountRequest, popupWidth: 300, popupHeight: height + 20)
  554. popup.backgroundAlpha = 0.8
  555. window?.rootViewController?.present(popup, animated: true)
  556. vcAccountRequest.startTimer()
  557. }
  558. }
  559. }
  560. // MARK: - Passcode
  561. var isPasscodeReset: Bool {
  562. let passcodeCounterFailReset = NCKeychain().passcodeCounterFailReset
  563. return NCKeychain().resetAppCounterFail && passcodeCounterFailReset >= NCBrandOptions.shared.resetAppPasscodeAttempts
  564. }
  565. var isPasscodeFail: Bool {
  566. let passcodeCounterFail = NCKeychain().passcodeCounterFail
  567. return passcodeCounterFail > 0 && passcodeCounterFail.isMultiple(of: 3)
  568. }
  569. var isPasscodePresented: Bool {
  570. return privacyProtectionWindow?.rootViewController?.presentedViewController is TOPasscodeViewController
  571. }
  572. func presentPasscode(completion: @escaping () -> Void) {
  573. var error: NSError?
  574. defer { self.requestAccount() }
  575. let presentedViewController = window?.rootViewController?.presentedViewController
  576. guard !account.isEmpty, NCKeychain().passcode != nil, NCKeychain().requestPasscodeAtStart, !(presentedViewController is NCLoginNavigationController) else { return }
  577. // Make sure we have a privacy window (in case it's not enabled)
  578. showPrivacyProtectionWindow()
  579. let passcodeViewController = TOPasscodeViewController(passcodeType: .sixDigits, allowCancel: false)
  580. passcodeViewController.delegate = self
  581. passcodeViewController.keypadButtonShowLettering = false
  582. if NCKeychain().touchFaceID, LAContext().canEvaluatePolicy(.deviceOwnerAuthenticationWithBiometrics, error: &error) {
  583. if error == nil {
  584. if LAContext().biometryType == .faceID {
  585. passcodeViewController.biometryType = .faceID
  586. } else if LAContext().biometryType == .touchID {
  587. passcodeViewController.biometryType = .touchID
  588. }
  589. passcodeViewController.allowBiometricValidation = true
  590. passcodeViewController.automaticallyPromptForBiometricValidation = false
  591. }
  592. }
  593. // show passcode on top of privacy window
  594. privacyProtectionWindow?.rootViewController?.present(passcodeViewController, animated: true, completion: {
  595. self.openAlert(passcodeViewController: passcodeViewController)
  596. completion()
  597. })
  598. }
  599. func enableTouchFaceID() {
  600. guard !account.isEmpty,
  601. NCKeychain().touchFaceID,
  602. NCKeychain().passcode != nil,
  603. NCKeychain().requestPasscodeAtStart,
  604. !isPasscodeFail,
  605. !isPasscodeReset,
  606. let passcodeViewController = privacyProtectionWindow?.rootViewController?.presentedViewController as? TOPasscodeViewController
  607. else { return }
  608. DispatchQueue.main.asyncAfter(deadline: .now() + 0.5) {
  609. LAContext().evaluatePolicy(.deviceOwnerAuthenticationWithBiometrics, localizedReason: NCBrandOptions.shared.brand) { success, evaluateError in
  610. if success {
  611. DispatchQueue.main.async {
  612. passcodeViewController.dismiss(animated: true) {
  613. NCKeychain().passcodeCounterFail = 0
  614. NCKeychain().passcodeCounterFailReset = 0
  615. self.hidePrivacyProtectionWindow()
  616. self.requestAccount()
  617. }
  618. }
  619. } else {
  620. if let error = evaluateError {
  621. switch error._code {
  622. case LAError.userFallback.rawValue, LAError.authenticationFailed.rawValue:
  623. if LAContext().biometryType == .faceID {
  624. NCKeychain().passcodeCounterFail = 2
  625. NCKeychain().passcodeCounterFailReset += 2
  626. } else {
  627. NCKeychain().passcodeCounterFail = 3
  628. NCKeychain().passcodeCounterFailReset += 3
  629. }
  630. self.openAlert(passcodeViewController: passcodeViewController)
  631. case LAError.biometryLockout.rawValue:
  632. LAContext().evaluatePolicy(LAPolicy.deviceOwnerAuthentication, localizedReason: NSLocalizedString("_deviceOwnerAuthentication_", comment: ""), reply: { success, _ in
  633. if success {
  634. DispatchQueue.main.async {
  635. NCKeychain().passcodeCounterFail = 0
  636. self.enableTouchFaceID()
  637. }
  638. }
  639. })
  640. case LAError.userCancel.rawValue:
  641. NCKeychain().passcodeCounterFail += 1
  642. NCKeychain().passcodeCounterFailReset += 1
  643. default:
  644. break
  645. }
  646. }
  647. }
  648. }
  649. }
  650. }
  651. func didInputCorrectPasscode(in passcodeViewController: TOPasscodeViewController) {
  652. DispatchQueue.main.async {
  653. passcodeViewController.dismiss(animated: true) {
  654. NCKeychain().passcodeCounterFail = 0
  655. NCKeychain().passcodeCounterFailReset = 0
  656. self.hidePrivacyProtectionWindow()
  657. self.requestAccount()
  658. }
  659. }
  660. }
  661. func passcodeViewController(_ passcodeViewController: TOPasscodeViewController, isCorrectCode code: String) -> Bool {
  662. if code == NCKeychain().passcode {
  663. return true
  664. } else {
  665. NCKeychain().passcodeCounterFail += 1
  666. NCKeychain().passcodeCounterFailReset += 1
  667. openAlert(passcodeViewController: passcodeViewController)
  668. return false
  669. }
  670. }
  671. func didPerformBiometricValidationRequest(in passcodeViewController: TOPasscodeViewController) {
  672. enableTouchFaceID()
  673. }
  674. func openAlert(passcodeViewController: TOPasscodeViewController) {
  675. DispatchQueue.main.asyncAfter(deadline: .now() + 0.1) {
  676. if self.isPasscodeReset {
  677. passcodeViewController.setContentHidden(true, animated: true)
  678. let alertController = UIAlertController(title: NSLocalizedString("_reset_wrong_passcode_", comment: ""), message: nil, preferredStyle: .alert)
  679. passcodeViewController.present(alertController, animated: true, completion: { })
  680. DispatchQueue.main.asyncAfter(deadline: .now() + 3) {
  681. self.resetApplication()
  682. }
  683. } else if self.isPasscodeFail {
  684. passcodeViewController.setContentHidden(true, animated: true)
  685. let alertController = UIAlertController(title: NSLocalizedString("_passcode_counter_fail_", comment: ""), message: nil, preferredStyle: .alert)
  686. passcodeViewController.present(alertController, animated: true, completion: { })
  687. var seconds = NCBrandOptions.shared.passcodeSecondsFail
  688. _ = Timer.scheduledTimer(withTimeInterval: 1.0, repeats: true) { timer in
  689. alertController.message = "\(seconds) " + NSLocalizedString("_seconds_", comment: "")
  690. seconds -= 1
  691. if seconds < 0 {
  692. timer.invalidate()
  693. alertController.dismiss(animated: true)
  694. passcodeViewController.setContentHidden(false, animated: true)
  695. NCKeychain().passcodeCounterFail = 0
  696. self.enableTouchFaceID()
  697. }
  698. }
  699. }
  700. }
  701. }
  702. // MARK: - Reset Application
  703. @objc func resetApplication() {
  704. let utilityFileSystem = NCUtilityFileSystem()
  705. NCNetworking.shared.cancelDataTask()
  706. NCNetworking.shared.cancelDownloadTasks()
  707. NCNetworking.shared.cancelUploadTasks()
  708. NCNetworking.shared.cancelUploadBackgroundTask()
  709. URLCache.shared.memoryCapacity = 0
  710. URLCache.shared.diskCapacity = 0
  711. utilityFileSystem.removeGroupDirectoryProviderStorage()
  712. utilityFileSystem.removeGroupApplicationSupport()
  713. utilityFileSystem.removeDocumentsDirectory()
  714. utilityFileSystem.removeTemporaryDirectory()
  715. NCKeychain().removeAll()
  716. exit(0)
  717. }
  718. // MARK: - Privacy Protection
  719. private func showPrivacyProtectionWindow() {
  720. guard privacyProtectionWindow == nil else {
  721. privacyProtectionWindow?.isHidden = false
  722. return
  723. }
  724. privacyProtectionWindow = UIWindow(frame: UIScreen.main.bounds)
  725. let storyboard = UIStoryboard(name: "LaunchScreen", bundle: nil)
  726. let initialViewController = storyboard.instantiateInitialViewController()
  727. self.privacyProtectionWindow?.rootViewController = initialViewController
  728. privacyProtectionWindow?.windowLevel = .alert + 1
  729. privacyProtectionWindow?.makeKeyAndVisible()
  730. }
  731. func hidePrivacyProtectionWindow() {
  732. guard !(privacyProtectionWindow?.rootViewController?.presentedViewController is TOPasscodeViewController) else { return }
  733. UIWindow.animate(withDuration: 0.25) {
  734. self.privacyProtectionWindow?.alpha = 0
  735. } completion: { _ in
  736. self.privacyProtectionWindow?.isHidden = true
  737. self.privacyProtectionWindow = nil
  738. }
  739. }
  740. // MARK: - Queue
  741. @objc func cancelAllQueue() {
  742. NCNetworking.shared.downloadQueue.cancelAll()
  743. NCNetworking.shared.downloadThumbnailQueue.cancelAll()
  744. NCNetworking.shared.downloadThumbnailActivityQueue.cancelAll()
  745. NCNetworking.shared.downloadAvatarQueue.cancelAll()
  746. NCNetworking.shared.unifiedSearchQueue.cancelAll()
  747. NCNetworking.shared.saveLivePhotoQueue.cancelAll()
  748. NCNetworking.shared.convertLivePhotoQueue.cancelAll()
  749. }
  750. // MARK: - Universal Links
  751. func application(_ application: UIApplication, continue userActivity: NSUserActivity, restorationHandler: @escaping ([UIUserActivityRestoring]?) -> Void) -> Bool {
  752. let applicationHandle = NCApplicationHandle()
  753. return applicationHandle.applicationOpenUserActivity(userActivity)
  754. }
  755. // MARK: - Scheme URL
  756. func application(_ app: UIApplication, open url: URL, options: [UIApplication.OpenURLOptionsKey: Any] = [:]) -> Bool {
  757. let scheme = url.scheme
  758. let action = url.host
  759. var fileName: String = ""
  760. var serverUrl: String = ""
  761. /*
  762. Example: nextcloud://open-action?action=create-voice-memo&&user=marinofaggiana&url=https://cloud.nextcloud.com
  763. */
  764. if !account.isEmpty && scheme == NCGlobal.shared.appScheme && action == "open-action" {
  765. if let urlComponents = URLComponents(url: url, resolvingAgainstBaseURL: false) {
  766. let queryItems = urlComponents.queryItems
  767. guard let actionScheme = queryItems?.filter({ $0.name == "action" }).first?.value,
  768. let userScheme = queryItems?.filter({ $0.name == "user" }).first?.value,
  769. let urlScheme = queryItems?.filter({ $0.name == "url" }).first?.value,
  770. let rootViewController = window?.rootViewController else { return false }
  771. if getMatchedAccount(userId: userScheme, url: urlScheme) == nil {
  772. let message = NSLocalizedString("_the_account_", comment: "") + " " + userScheme + NSLocalizedString("_of_", comment: "") + " " + urlScheme + " " + NSLocalizedString("_does_not_exist_", comment: "")
  773. let alertController = UIAlertController(title: NSLocalizedString("_info_", comment: ""), message: message, preferredStyle: .alert)
  774. alertController.addAction(UIAlertAction(title: NSLocalizedString("_ok_", comment: ""), style: .default, handler: { _ in }))
  775. window?.rootViewController?.present(alertController, animated: true, completion: { })
  776. return false
  777. }
  778. switch actionScheme {
  779. case NCGlobal.shared.actionUploadAsset:
  780. NCAskAuthorization().askAuthorizationPhotoLibrary(viewController: rootViewController) { hasPermission in
  781. if hasPermission {
  782. NCPhotosPickerViewController(viewController: rootViewController, maxSelectedAssets: 0, singleSelectedMode: false)
  783. }
  784. }
  785. case NCGlobal.shared.actionScanDocument:
  786. NCDocumentCamera.shared.openScannerDocument(viewController: rootViewController)
  787. case NCGlobal.shared.actionTextDocument:
  788. guard let navigationController = UIStoryboard(name: "NCCreateFormUploadDocuments", bundle: nil).instantiateInitialViewController(),
  789. let directEditingCreators = NCManageDatabase.shared.getDirectEditingCreators(account: account),
  790. let directEditingCreator = directEditingCreators.first(where: { $0.editor == NCGlobal.shared.editorText}),
  791. let viewController = (navigationController as? UINavigationController)?.topViewController as? NCCreateFormUploadDocuments else { return false }
  792. navigationController.modalPresentationStyle = UIModalPresentationStyle.formSheet
  793. viewController.editorId = NCGlobal.shared.editorText
  794. viewController.creatorId = directEditingCreator.identifier
  795. viewController.typeTemplate = NCGlobal.shared.templateDocument
  796. viewController.serverUrl = activeServerUrl
  797. viewController.titleForm = NSLocalizedString("_create_nextcloudtext_document_", comment: "")
  798. rootViewController.present(navigationController, animated: true, completion: nil)
  799. case NCGlobal.shared.actionVoiceMemo:
  800. NCAskAuthorization().askAuthorizationAudioRecord(viewController: rootViewController) { hasPermission in
  801. if hasPermission {
  802. let fileName = NCUtilityFileSystem().createFileNameDate(NSLocalizedString("_voice_memo_filename_", comment: ""), ext: "m4a")
  803. if let viewController = UIStoryboard(name: "NCAudioRecorderViewController", bundle: nil).instantiateInitialViewController() as? NCAudioRecorderViewController {
  804. viewController.delegate = self
  805. viewController.createRecorder(fileName: fileName)
  806. viewController.modalTransitionStyle = .crossDissolve
  807. viewController.modalPresentationStyle = UIModalPresentationStyle.overCurrentContext
  808. rootViewController.present(viewController, animated: true, completion: nil)
  809. }
  810. }
  811. }
  812. default:
  813. print("No action")
  814. }
  815. }
  816. return true
  817. }
  818. /*
  819. Example: nextcloud://open-file?path=Talk/IMG_0000123.jpg&user=marinofaggiana&link=https://cloud.nextcloud.com/f/123
  820. */
  821. else if !account.isEmpty && scheme == NCGlobal.shared.appScheme && action == "open-file" {
  822. if let urlComponents = URLComponents(url: url, resolvingAgainstBaseURL: false) {
  823. let queryItems = urlComponents.queryItems
  824. guard let userScheme = queryItems?.filter({ $0.name == "user" }).first?.value,
  825. let pathScheme = queryItems?.filter({ $0.name == "path" }).first?.value,
  826. let linkScheme = queryItems?.filter({ $0.name == "link" }).first?.value else { return false}
  827. guard let matchedAccount = getMatchedAccount(userId: userScheme, url: linkScheme) else {
  828. guard let domain = URL(string: linkScheme)?.host else { return true }
  829. fileName = (pathScheme as NSString).lastPathComponent
  830. let message = String(format: NSLocalizedString("_account_not_available_", comment: ""), userScheme, domain, fileName)
  831. let alertController = UIAlertController(title: NSLocalizedString("_info_", comment: ""), message: message, preferredStyle: .alert)
  832. alertController.addAction(UIAlertAction(title: NSLocalizedString("_ok_", comment: ""), style: .default, handler: { _ in }))
  833. window?.rootViewController?.present(alertController, animated: true, completion: { })
  834. return false
  835. }
  836. let davFiles = NextcloudKit.shared.nkCommonInstance.dav + "/files/" + self.userId
  837. if pathScheme.contains("/") {
  838. fileName = (pathScheme as NSString).lastPathComponent
  839. serverUrl = matchedAccount.urlBase + "/" + davFiles + "/" + (pathScheme as NSString).deletingLastPathComponent
  840. } else {
  841. fileName = pathScheme
  842. serverUrl = matchedAccount.urlBase + "/" + davFiles
  843. }
  844. DispatchQueue.main.asyncAfter(deadline: .now() + 0.5) {
  845. NCActionCenter.shared.openFileViewInFolder(serverUrl: serverUrl, fileNameBlink: nil, fileNameOpen: fileName)
  846. }
  847. }
  848. return true
  849. /*
  850. Example: nextcloud://open-and-switch-account?user=marinofaggiana&url=https://cloud.nextcloud.com
  851. */
  852. } else if !account.isEmpty && scheme == NCGlobal.shared.appScheme && action == "open-and-switch-account" {
  853. guard let urlComponents = URLComponents(url: url, resolvingAgainstBaseURL: false) else { return false }
  854. let queryItems = urlComponents.queryItems
  855. guard let userScheme = queryItems?.filter({ $0.name == "user" }).first?.value,
  856. let urlScheme = queryItems?.filter({ $0.name == "url" }).first?.value else { return false}
  857. // If the account doesn't exist, return false which will open the app without switching
  858. if getMatchedAccount(userId: userScheme, url: urlScheme) == nil {
  859. return false
  860. }
  861. // Otherwise open the app and switch accounts
  862. return true
  863. } else {
  864. let applicationHandle = NCApplicationHandle()
  865. let isHandled = applicationHandle.applicationOpenURL(url)
  866. if isHandled {
  867. return true
  868. } else {
  869. app.open(url)
  870. return true
  871. }
  872. }
  873. }
  874. func getMatchedAccount(userId: String, url: String) -> tableAccount? {
  875. if let activeAccount = NCManageDatabase.shared.getActiveAccount() {
  876. let urlBase = URL(string: activeAccount.urlBase)
  877. if url.contains(urlBase?.host ?? "") && userId == activeAccount.userId {
  878. return activeAccount
  879. } else {
  880. let accounts = NCManageDatabase.shared.getAllAccount()
  881. for account in accounts {
  882. let urlBase = URL(string: account.urlBase)
  883. if url.contains(urlBase?.host ?? "") && userId == account.userId {
  884. changeAccount(account.account, userProfile: nil)
  885. return account
  886. }
  887. }
  888. }
  889. }
  890. return nil
  891. }
  892. }
  893. // MARK: - NCAudioRecorder ViewController Delegate
  894. extension AppDelegate: NCAudioRecorderViewControllerDelegate {
  895. func didFinishRecording(_ viewController: NCAudioRecorderViewController, fileName: String) {
  896. guard let navigationController = UIStoryboard(name: "NCCreateFormUploadVoiceNote", bundle: nil).instantiateInitialViewController() as? UINavigationController,
  897. let viewController = navigationController.topViewController as? NCCreateFormUploadVoiceNote else { return }
  898. navigationController.modalPresentationStyle = .formSheet
  899. viewController.setup(serverUrl: activeServerUrl, fileNamePath: NSTemporaryDirectory() + fileName, fileName: fileName)
  900. window?.rootViewController?.present(navigationController, animated: true)
  901. }
  902. func didFinishWithoutRecording(_ viewController: NCAudioRecorderViewController, fileName: String) { }
  903. }
  904. extension AppDelegate: NCCreateFormUploadConflictDelegate {
  905. func dismissCreateFormUploadConflict(metadatas: [tableMetadata]?) {
  906. guard let metadatas = metadatas, !metadatas.isEmpty else { return }
  907. NCNetworkingProcessUpload.shared.createProcessUploads(metadatas: metadatas) { _ in }
  908. }
  909. }