AppDelegate.m 56 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180
  1. //
  2. // AppDelegate.m
  3. // Nextcloud
  4. //
  5. // Created by Marino Faggiana on 04/09/14.
  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 "AppDelegate.h"
  24. #import "CCGraphics.h"
  25. #import "NCBridgeSwift.h"
  26. #import "NCAutoUpload.h"
  27. #import "NCPushNotificationEncryption.h"
  28. #import <QuartzCore/QuartzCore.h>
  29. @import Sentry;
  30. @class NCViewerRichdocument;
  31. @interface AppDelegate() <TOPasscodeViewControllerDelegate>
  32. @end
  33. @implementation AppDelegate
  34. + (void)initialize
  35. {
  36. [[NSUserDefaults standardUserDefaults] registerDefaults:@{@"UserAgent": [CCUtility getUserAgent]}];
  37. }
  38. - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
  39. {
  40. BOOL isSimulatorOrTestFlight = [[NCUtility shared] isSimulatorOrTestFlight];
  41. if (isSimulatorOrTestFlight) {
  42. NCBrandOptions.sharedInstance.disable_crash_service = false;
  43. }
  44. if (![CCUtility getDisableCrashservice] && NCBrandOptions.sharedInstance.disable_crash_service == false) {
  45. [SentrySDK startWithOptions: @{
  46. @"dsn": @"https://42eaf570ec2646b1a564a4c4bfc8c279@o394108.ingest.sentry.io/5243836",
  47. @"debug": @(YES),
  48. @"enableAutoSessionTracking": @(YES)
  49. /* PRIVACY : https://github.com/getsentry/sentry-cocoa
  50. By default, we don’t apply the user identification provided to the SDK via the API. Instead, we use
  51. the installation ID generated with the first use of the application. The ID doesn’t contain any
  52. private or public data of your users or any public or shared data of their device.
  53. */
  54. }];
  55. }
  56. [CCUtility createDirectoryStandard];
  57. [CCUtility emptyTemporaryDirectory];
  58. // Networking
  59. [[NCCommunicationCommon shared] setupWithDelegate:[NCNetworking shared]];
  60. [[NCCommunicationCommon shared] setupWithUserAgent:[CCUtility getUserAgent]];
  61. NSInteger logLevel = [CCUtility getLogLevel];
  62. [[NCCommunicationCommon shared] setFileLogWithLevel:logLevel];
  63. NSString *versionApp = [NSString stringWithFormat:@"%@.%@", [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleShortVersionString"], [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleVersion"]];
  64. NSString *versionNextcloudiOS = [NSString stringWithFormat:[NCBrandOptions sharedInstance].textCopyrightNextcloudiOS, versionApp];
  65. if (isSimulatorOrTestFlight) {
  66. [[NCCommunicationCommon shared] writeLog:[NSString stringWithFormat:@"Start session with level %lu %@ (Simulator / TestFlight)", (unsigned long)logLevel, versionNextcloudiOS]];
  67. } else {
  68. [[NCCommunicationCommon shared] writeLog:[NSString stringWithFormat:@"Start session with level %lu %@", (unsigned long)logLevel, versionNextcloudiOS]];
  69. }
  70. //
  71. [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(initializeMain:) name:k_notificationCenter_initializeMain object:nil];
  72. // Set account, if no exists clear all
  73. tableAccount *tableAccount = [[NCManageDatabase sharedInstance] getAccountActive];
  74. if (tableAccount == nil) {
  75. // remove all the keys Chain
  76. [CCUtility deleteAllChainStore];
  77. // remove all the App group key
  78. [[NSUserDefaults standardUserDefaults] removePersistentDomainForName:[[NSBundle mainBundle] bundleIdentifier]];
  79. } else {
  80. // FIX 3.0.5 lost urlbase
  81. if (tableAccount.urlBase.length == 0) {
  82. NSString *user = [tableAccount.user stringByAppendingString:@" "];
  83. NSString *urlBase = [tableAccount.account stringByReplacingOccurrencesOfString:user withString:@""];
  84. tableAccount.urlBase = urlBase;
  85. [[NCManageDatabase sharedInstance] updateAccount:tableAccount];
  86. tableAccount = [[NCManageDatabase sharedInstance] getAccountActive];
  87. }
  88. [self settingAccount:tableAccount.account urlBase:tableAccount.urlBase user:tableAccount.user userID:tableAccount.userID password:[CCUtility getPassword:tableAccount.account]];
  89. }
  90. // UserDefaults
  91. self.ncUserDefaults = [[NSUserDefaults alloc] initWithSuiteName:[NCBrandOptions sharedInstance].capabilitiesGroups];
  92. // Background Fetch
  93. [application setMinimumBackgroundFetchInterval:UIApplicationBackgroundFetchIntervalMinimum];
  94. self.listProgressMetadata = [NSMutableDictionary new];
  95. self.listFilesVC = [NSMutableDictionary new];
  96. self.listMainVC = [NSMutableDictionary new];
  97. self.listFavoriteVC = [NSMutableDictionary new];
  98. self.listOfflineVC = [NSMutableDictionary new];
  99. // Push Notification
  100. [application registerForRemoteNotifications];
  101. // Display notification
  102. [UNUserNotificationCenter currentNotificationCenter].delegate = self;
  103. UNAuthorizationOptions authOptions = UNAuthorizationOptionAlert | UNAuthorizationOptionSound | UNAuthorizationOptionBadge;
  104. [[UNUserNotificationCenter currentNotificationCenter] requestAuthorizationWithOptions:authOptions completionHandler:^(BOOL granted, NSError * _Nullable error) { }];
  105. //AV Session
  106. [[AVAudioSession sharedInstance] setCategory: AVAudioSessionCategoryPlayback error:nil];
  107. [[UIApplication sharedApplication] beginReceivingRemoteControlEvents];
  108. // ProgressView Detail
  109. self.progressViewDetail = [[UIProgressView alloc] initWithProgressViewStyle: UIProgressViewStyleBar];
  110. // Quick Actions
  111. if([[UIApplicationShortcutItem class] respondsToSelector:@selector(new)]) {
  112. [self configDynamicShortcutItems];
  113. UIApplicationShortcutItem *shortcutItem = [launchOptions objectForKeyedSubscript:UIApplicationLaunchOptionsShortcutItemKey];
  114. if (shortcutItem)
  115. [self handleShortCutItem:shortcutItem];
  116. }
  117. // Start Timer
  118. self.timerUpdateApplicationIconBadgeNumber = [NSTimer scheduledTimerWithTimeInterval:k_timerUpdateApplicationIconBadgeNumber target:self selector:@selector(updateApplicationIconBadgeNumber) userInfo:nil repeats:YES];
  119. [self startTimerErrorNetworking];
  120. // Store review
  121. if ([[NCUtility shared] isSimulatorOrTestFlight] == false) {
  122. NCStoreReview *review = [NCStoreReview new];
  123. [review incrementAppRuns];
  124. [review showStoreReview];
  125. }
  126. // Detect Dark mode
  127. if (@available(iOS 13.0, *)) {
  128. if ([CCUtility getDarkModeDetect]) {
  129. if ([[UITraitCollection currentTraitCollection] userInterfaceStyle] == UIUserInterfaceStyleDark) {
  130. [CCUtility setDarkMode:YES];
  131. } else {
  132. [CCUtility setDarkMode:NO];
  133. }
  134. }
  135. }
  136. // TabBar Controller
  137. //[self createTabBarController];
  138. if ([NCBrandOptions sharedInstance].disable_intro) {
  139. [CCUtility setIntro:YES];
  140. if (self.account.length == 0) {
  141. [self openLoginView:nil selector:k_intro_login openLoginWeb:false];
  142. }
  143. } else {
  144. if ([CCUtility getIntro] == NO) {
  145. UIViewController *introViewController = [[UIStoryboard storyboardWithName:@"NCIntro" bundle:[NSBundle mainBundle]] instantiateInitialViewController];
  146. UINavigationController *navController = [[UINavigationController alloc] initWithRootViewController: introViewController];
  147. self.window.rootViewController = navController;
  148. [self.window makeKeyAndVisible];
  149. }
  150. }
  151. // init home
  152. [[NSNotificationCenter defaultCenter] postNotificationOnMainThreadName:k_notificationCenter_initializeMain object:nil userInfo:nil];
  153. // Passcode
  154. dispatch_async(dispatch_get_main_queue(), ^{
  155. [self passcodeWithAutomaticallyPromptForBiometricValidation:true];
  156. });
  157. // Auto upload
  158. self.networkingAutoUpload = [NCNetworkingAutoUpload new];
  159. [[NCCommunicationCommon shared] writeLog:@"Application did finish launching"];
  160. return YES;
  161. }
  162. //
  163. // L' applicazione si dimetterà dallo stato di attivo
  164. //
  165. - (void)applicationWillResignActive:(UIApplication *)application
  166. {
  167. // Test Maintenance
  168. if (self.account.length == 0 || self.maintenanceMode)
  169. return;
  170. // Dismiss FileViewInFolder
  171. if (self.activeFileViewInFolder != nil ) {
  172. [self.activeFileViewInFolder dismissViewControllerAnimated:false completion:^{
  173. self.activeFileViewInFolder = nil;
  174. }];
  175. }
  176. [self updateApplicationIconBadgeNumber];
  177. }
  178. //
  179. // L' applicazione entrerà in primo piano (attivo solo dopo il background)
  180. //
  181. - (void)applicationWillEnterForeground:(UIApplication *)application
  182. {
  183. if (self.account.length == 0 || self.maintenanceMode) { return; }
  184. [[NSNotificationCenter defaultCenter] postNotificationOnMainThreadName:k_notificationCenter_applicationWillEnterForeground object:nil];
  185. // Request Passcode
  186. [self passcodeWithAutomaticallyPromptForBiometricValidation:true];
  187. // Initialize Auto upload
  188. [[NCAutoUpload sharedInstance] initStateAutoUpload];
  189. // Read active directory
  190. [self.activeFiles reloadDataSourceNetworkWithForced:true];
  191. // Required unsubscribing / subscribing
  192. [self pushNotification];
  193. // RichDocument
  194. [[NSNotificationCenter defaultCenter] postNotificationOnMainThreadName:k_notificationCenter_richdocumentGrabFocus object:nil];
  195. // Request Service Server Nextcloud
  196. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, 1 * NSEC_PER_SEC), dispatch_get_main_queue(), ^(void) {
  197. [[NCService shared] startRequestServicesServer];
  198. });
  199. [[NCCommunicationCommon shared] writeLog:@"The application Will enter in foreground"];
  200. }
  201. //
  202. // L' applicazione entrerà in primo piano (attivo sempre)
  203. //
  204. - (void)applicationDidBecomeActive:(UIApplication *)application
  205. {
  206. if (self.account.length == 0 || self.maintenanceMode) { return; }
  207. // Brand
  208. #if defined(HC)
  209. tableAccount *account = [[NCManageDatabase sharedInstance] getAccountActive];
  210. if (account.hcIsTrial == true || account.hcTrialExpired == true || account.hcNextGroupExpirationGroupExpired == true) {
  211. HCTrial *vc = [[UIStoryboard storyboardWithName:@"HCTrial" bundle:nil] instantiateInitialViewController];
  212. vc.account = account;
  213. [self.window.rootViewController presentViewController:vc animated:YES completion:nil];
  214. }
  215. #endif
  216. [[NCNetworking shared] verifyUploadZombie];
  217. [[NCCommunicationCommon shared] writeLog:@"The application did become active"];
  218. }
  219. //
  220. // L' applicazione è entrata nello sfondo
  221. //
  222. - (void)applicationDidEnterBackground:(UIApplication *)application
  223. {
  224. if (self.account.length == 0 || self.maintenanceMode) { return; }
  225. [[NSNotificationCenter defaultCenter] postNotificationOnMainThreadName:k_notificationCenter_applicationDidEnterBackground object:nil];
  226. [self passcodeWithAutomaticallyPromptForBiometricValidation:false];
  227. }
  228. //
  229. // L'applicazione terminerà
  230. //
  231. - (void)applicationWillTerminate:(UIApplication *)application
  232. {
  233. [[NCCommunicationCommon shared] writeLog:@"bye bye"];
  234. }
  235. // NotificationCenter
  236. - (void)initializeMain:(NSNotification *)notification
  237. {
  238. if (self.account.length == 0) return;
  239. // Clear error certificate
  240. [CCUtility setCertificateError:self.account error:NO];
  241. // Setting Theming
  242. [self settingThemingColorBrand];
  243. // If AVPlayer in play -> Stop
  244. if (self.player != nil && self.player.rate != 0) {
  245. [self.player pause];
  246. }
  247. // close detail
  248. [[NSNotificationCenter defaultCenter] postNotificationOnMainThreadName:k_notificationCenter_menuDetailClose object:nil];
  249. // Not Photos Video in library ? then align and Init Auto Upload
  250. NSArray *recordsPhotoLibrary = [[NCManageDatabase sharedInstance] getPhotoLibraryWithPredicate:[NSPredicate predicateWithFormat:@"account == %@", self.account]];
  251. if ([recordsPhotoLibrary count] == 0) {
  252. [[NCAutoUpload sharedInstance] alignPhotoLibrary];
  253. }
  254. // Start Auto Upload
  255. [[NCAutoUpload sharedInstance] initStateAutoUpload];
  256. // Start services
  257. [[NCCommunicationCommon shared] writeLog:@"Request Service Server Nextcloud"];
  258. [[NCService shared] startRequestServicesServer];
  259. // Registeration push notification
  260. [self pushNotification];
  261. // Registeration domain File Provider
  262. FileProviderDomain *fileProviderDomain = [FileProviderDomain new];
  263. [fileProviderDomain removeAllDomains];
  264. //[fileProviderDomain registerDomains];
  265. [[NCCommunicationCommon shared] writeLog:@"initialize Main"];
  266. }
  267. #pragma --------------------------------------------------------------------------------------------
  268. #pragma mark ===== Login / checkErrorNetworking =====
  269. #pragma --------------------------------------------------------------------------------------------
  270. - (void)checkErrorNetworking
  271. {
  272. // test
  273. if (self.account.length == 0 || self.maintenanceMode)
  274. return;
  275. // check unauthorized server (401)
  276. if ([CCUtility getPassword:self.account].length == 0) {
  277. [self openLoginView:self.window.rootViewController selector:k_intro_login openLoginWeb:true];
  278. }
  279. // check certificate untrusted (-1202)
  280. if ([CCUtility getCertificateError:self.account]) {
  281. UIAlertController *alertController = [UIAlertController alertControllerWithTitle:NSLocalizedString(@"_ssl_certificate_untrusted_", nil) message:NSLocalizedString(@"_connect_server_anyway_", nil) preferredStyle:UIAlertControllerStyleAlert];
  282. [alertController addAction: [UIAlertAction actionWithTitle:NSLocalizedString(@"_yes_", nil) style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {
  283. [[NCNetworking shared] wrtiteCertificateWithDirectoryCertificate:[CCUtility getDirectoryCerificates]];
  284. [self startTimerErrorNetworking];
  285. }]];
  286. [alertController addAction: [UIAlertAction actionWithTitle:NSLocalizedString(@"_no_", nil) style:UIAlertActionStyleCancel handler:^(UIAlertAction *action) {
  287. [self startTimerErrorNetworking];
  288. }]];
  289. [self.window.rootViewController presentViewController:alertController animated:YES completion:^{
  290. // Stop timer error network
  291. [self.timerErrorNetworking invalidate];
  292. }];
  293. }
  294. }
  295. - (void)openLoginView:(UIViewController *)viewController selector:(NSInteger)selector openLoginWeb:(BOOL)openLoginWeb
  296. {
  297. // use appConfig [MDM]
  298. if ([NCBrandOptions sharedInstance].use_configuration) {
  299. if (!(_appConfigView.isViewLoaded && _appConfigView.view.window)) {
  300. self.appConfigView = [[UIStoryboard storyboardWithName:@"CCLogin" bundle:nil] instantiateViewControllerWithIdentifier:@"NCAppConfigView"];
  301. [self showLoginViewController:self.appConfigView forContext:viewController];
  302. }
  303. return;
  304. }
  305. // only for personalized LoginWeb [customer]
  306. if ([NCBrandOptions sharedInstance].use_login_web_personalized) {
  307. if (!(_activeLoginWeb.isViewLoaded && _activeLoginWeb.view.window)) {
  308. self.activeLoginWeb = [[UIStoryboard storyboardWithName:@"CCLogin" bundle:nil] instantiateViewControllerWithIdentifier:@"NCLoginWeb"];
  309. self.activeLoginWeb.urlBase = [[NCBrandOptions sharedInstance] loginBaseUrl];
  310. [self showLoginViewController:self.activeLoginWeb forContext:viewController];
  311. }
  312. return;
  313. }
  314. // normal login
  315. if (selector == k_intro_signup) {
  316. if (!(_activeLoginWeb.isViewLoaded && _activeLoginWeb.view.window)) {
  317. self.activeLoginWeb = [[UIStoryboard storyboardWithName:@"CCLogin" bundle:nil] instantiateViewControllerWithIdentifier:@"NCLoginWeb"];
  318. if (selector == k_intro_signup) {
  319. self.activeLoginWeb.urlBase = [[NCBrandOptions sharedInstance] linkloginPreferredProviders];
  320. } else {
  321. self.activeLoginWeb.urlBase = self.urlBase;
  322. }
  323. [self showLoginViewController:self.activeLoginWeb forContext:viewController];
  324. }
  325. } else if ([NCBrandOptions sharedInstance].disable_intro && [NCBrandOptions sharedInstance].disable_request_login_url) {
  326. self.activeLoginWeb = [[UIStoryboard storyboardWithName:@"CCLogin" bundle:nil] instantiateViewControllerWithIdentifier:@"NCLoginWeb"];
  327. self.activeLoginWeb.urlBase = [[NCBrandOptions sharedInstance] loginBaseUrl];
  328. [self showLoginViewController:self.activeLoginWeb forContext:viewController];
  329. } else if (openLoginWeb) {
  330. if (!(_activeLoginWeb.isViewLoaded && _activeLoginWeb.view.window)) {
  331. self.activeLoginWeb = [[UIStoryboard storyboardWithName:@"CCLogin" bundle:nil] instantiateViewControllerWithIdentifier:@"NCLoginWeb"];
  332. self.activeLoginWeb.urlBase = self.urlBase;
  333. [self showLoginViewController:self.activeLoginWeb forContext:viewController];
  334. }
  335. } else {
  336. if (!(_activeLogin.isViewLoaded && _activeLogin.view.window)) {
  337. _activeLogin = [[UIStoryboard storyboardWithName:@"CCLogin" bundle:nil] instantiateViewControllerWithIdentifier:@"CCLoginNextcloud"];
  338. [self showLoginViewController:_activeLogin forContext:viewController];
  339. }
  340. }
  341. }
  342. -(void)showLoginViewController:(UIViewController *)viewController forContext:(UIViewController *)contextViewController
  343. {
  344. if (contextViewController == NULL) {
  345. UINavigationController *navController = [[UINavigationController alloc] initWithRootViewController:viewController];
  346. navController.navigationBar.barStyle = UIBarStyleBlack;
  347. navController.navigationBar.tintColor = NCBrandColor.sharedInstance.customerText;
  348. navController.navigationBar.barTintColor = NCBrandColor.sharedInstance.customer;
  349. [navController.navigationBar setTranslucent:false];
  350. self.window.rootViewController = navController;
  351. [self.window makeKeyAndVisible];
  352. } else if ([contextViewController isKindOfClass:[UINavigationController class]]) {
  353. UINavigationController *navController = ((UINavigationController *)contextViewController);
  354. [navController pushViewController:viewController animated:true];
  355. } else {
  356. UINavigationController *navController = [[UINavigationController alloc] initWithRootViewController:viewController];
  357. navController.modalPresentationStyle = UIModalPresentationFullScreen;
  358. navController.navigationBar.barStyle = UIBarStyleBlack;
  359. navController.navigationBar.tintColor = NCBrandColor.sharedInstance.customerText;
  360. navController.navigationBar.barTintColor = NCBrandColor.sharedInstance.customer;
  361. [navController.navigationBar setTranslucent:false];
  362. [contextViewController presentViewController:navController animated:true completion:nil];
  363. }
  364. }
  365. - (void)startTimerErrorNetworking
  366. {
  367. self.timerErrorNetworking = [NSTimer scheduledTimerWithTimeInterval:k_timerErrorNetworking target:self selector:@selector(checkErrorNetworking) userInfo:nil repeats:YES];
  368. }
  369. #pragma --------------------------------------------------------------------------------------------
  370. #pragma mark ===== Account & Communication =====
  371. #pragma --------------------------------------------------------------------------------------------
  372. - (void)settingAccount:(NSString *)account urlBase:(NSString *)urlBase user:(NSString *)user userID:(NSString *)userID password:(NSString *)password
  373. {
  374. self.account = account;
  375. self.urlBase = urlBase;
  376. self.user = user;
  377. self.userID = userID;
  378. self.password = password;
  379. (void)[NCNetworkingNotificationCenter shared];
  380. [[NCCommunicationCommon shared] setupWithAccount:account user:user userId:userID password:password urlBase:urlBase];
  381. [self settingSetupCommunication:account];
  382. }
  383. - (void)deleteAccount:(NSString *)account wipe:(BOOL)wipe
  384. {
  385. // Push Notification
  386. tableAccount *accountPN = [[NCManageDatabase sharedInstance] getAccountWithPredicate:[NSPredicate predicateWithFormat:@"account == %@", account]];
  387. [self unsubscribingNextcloudServerPushNotification:accountPN.account urlBase:accountPN.urlBase user:accountPN.user withSubscribing:false];
  388. [self settingAccount:nil urlBase:nil user:nil userID:nil password:nil];
  389. /* DELETE ALL FILES LOCAL FS */
  390. NSArray *results = [[NCManageDatabase sharedInstance] getTableLocalFilesWithPredicate:[NSPredicate predicateWithFormat:@"account == %@", account] sorted:@"ocId" ascending:NO];
  391. for (tableLocalFile *result in results) {
  392. [CCUtility removeFileAtPath:[CCUtility getDirectoryProviderStorageOcId:result.ocId]];
  393. }
  394. // Clear database
  395. [[NCManageDatabase sharedInstance] clearDatabaseWithAccount:account removeAccount:true];
  396. [CCUtility clearAllKeysEndToEnd:account];
  397. [CCUtility clearAllKeysPushNotification:account];
  398. [CCUtility setCertificateError:account error:false];
  399. [CCUtility setPassword:account password:nil];
  400. if (wipe) {
  401. NSArray *listAccount = [[NCManageDatabase sharedInstance] getAccounts];
  402. if ([listAccount count] > 0) {
  403. NSString *newAccount = listAccount[0];
  404. tableAccount *tableAccount = [[NCManageDatabase sharedInstance] setAccountActive:newAccount];
  405. [self settingAccount:newAccount urlBase:tableAccount.urlBase user:tableAccount.user userID:tableAccount.userID password:[CCUtility getPassword:tableAccount.account]];
  406. [[NSNotificationCenter defaultCenter] postNotificationOnMainThreadName:k_notificationCenter_initializeMain object:nil userInfo:nil];
  407. } else {
  408. [self openLoginView:self.window.rootViewController selector:k_intro_login openLoginWeb:false];
  409. }
  410. }
  411. }
  412. - (void)settingSetupCommunication:(NSString *)account
  413. {
  414. NSInteger serverVersionMajor = [[NCManageDatabase sharedInstance] getCapabilitiesServerIntWithAccount:account elements:NCElementsJSON.shared.capabilitiesVersionMajor];
  415. if (serverVersionMajor > 0) {
  416. [[NCCommunicationCommon shared] setupWithNextcloudVersion:serverVersionMajor];
  417. }
  418. [[NCCommunicationCommon shared] setupWithWebDav:[[NCUtility shared] getWebDAVWithAccount:account]];
  419. [[NCCommunicationCommon shared] setupWithDav:[[NCUtility shared] getDAV]];
  420. }
  421. #pragma --------------------------------------------------------------------------------------------
  422. #pragma mark ===== Push Notifications =====
  423. #pragma --------------------------------------------------------------------------------------------
  424. - (void)pushNotification
  425. {
  426. // test
  427. if (self.account.length == 0 || self.maintenanceMode || self.pushKitToken.length == 0)
  428. return;
  429. for (tableAccount *result in [[NCManageDatabase sharedInstance] getAllAccount]) {
  430. NSString *token = [CCUtility getPushNotificationToken:result.account];
  431. if (![token isEqualToString:self.pushKitToken]) {
  432. if (token != nil) {
  433. // unsubscribing + subscribing
  434. [self unsubscribingNextcloudServerPushNotification:result.account urlBase:result.urlBase user:result.user withSubscribing:true];
  435. } else {
  436. [self subscribingNextcloudServerPushNotification:result.account urlBase:result.urlBase user:result.user];
  437. }
  438. }
  439. }
  440. }
  441. - (void)subscribingNextcloudServerPushNotification:(NSString *)account urlBase:(NSString *)urlBase user:(NSString *)user
  442. {
  443. // test
  444. if (self.account.length == 0 || self.maintenanceMode || self.pushKitToken.length == 0)
  445. return;
  446. [[NCPushNotificationEncryption sharedInstance] generatePushNotificationsKeyPair:account];
  447. NSString *pushTokenHash = [[NCEndToEndEncryption sharedManager] createSHA512:self.pushKitToken];
  448. NSData *pushPublicKey = [CCUtility getPushNotificationPublicKey:account];
  449. NSString *pushDevicePublicKey = [[NSString alloc] initWithData:pushPublicKey encoding:NSUTF8StringEncoding];
  450. NSString *proxyServerPath = [NCBrandOptions sharedInstance].pushNotificationServerProxy;
  451. [[NCCommunication shared] subscribingPushNotificationWithServerUrl:urlBase account:account user:user password:[CCUtility getPassword:account] pushTokenHash:pushTokenHash devicePublicKey:pushDevicePublicKey proxyServerUrl:proxyServerPath customUserAgent:nil addCustomHeaders:nil completionHandler:^(NSString *account, NSString *deviceIdentifier, NSString *signature, NSString *publicKey, NSInteger errorCode, NSString *errorDescription) {
  452. if (errorCode == 0) {
  453. NSString *userAgent = [NSString stringWithFormat:@"%@ (Strict VoIP)", [CCUtility getUserAgent]];
  454. [[NCCommunication shared] subscribingPushProxyWithProxyServerUrl:proxyServerPath pushToken:self.pushKitToken deviceIdentifier:deviceIdentifier signature:signature publicKey:publicKey userAgent:userAgent completionHandler:^(NSInteger errorCode, NSString *errorDescription) {
  455. if (errorCode == 0) {
  456. [[NCCommunicationCommon shared] writeLog:@"Subscribed to Push Notification server & proxy successfully"];
  457. [CCUtility setPushNotificationToken:account token:self.pushKitToken];
  458. [CCUtility setPushNotificationDeviceIdentifier:account deviceIdentifier:deviceIdentifier];
  459. [CCUtility setPushNotificationDeviceIdentifierSignature:account deviceIdentifierSignature:signature];
  460. [CCUtility setPushNotificationSubscribingPublicKey:account publicKey:publicKey];
  461. }
  462. }];
  463. }
  464. }];
  465. }
  466. - (void)unsubscribingNextcloudServerPushNotification:(NSString *)account urlBase:(NSString *)urlBase user:(NSString *)user withSubscribing:(BOOL)subscribing
  467. {
  468. // test
  469. if (self.account.length == 0 || self.maintenanceMode)
  470. return;
  471. NSString *deviceIdentifier = [CCUtility getPushNotificationDeviceIdentifier:account];
  472. NSString *signature = [CCUtility getPushNotificationDeviceIdentifierSignature:account];
  473. NSString *publicKey = [CCUtility getPushNotificationSubscribingPublicKey:account];
  474. [[NCCommunication shared] unsubscribingPushNotificationWithServerUrl:urlBase account:account user:user password:[CCUtility getPassword:account] customUserAgent:nil addCustomHeaders:nil completionHandler:^(NSString *account, NSInteger errorCode, NSString *errorDescription) {
  475. if (errorCode == 0) {
  476. NSString *userAgent = [NSString stringWithFormat:@"%@ (Strict VoIP)", [CCUtility getUserAgent]];
  477. NSString *proxyServerPath = [NCBrandOptions sharedInstance].pushNotificationServerProxy;
  478. [[NCCommunication shared] unsubscribingPushProxyWithProxyServerUrl:proxyServerPath deviceIdentifier:deviceIdentifier signature:signature publicKey:publicKey userAgent:userAgent completionHandler:^(NSInteger errorCode, NSString *errorDescription) {
  479. if (errorCode == 0) {
  480. [[NCCommunicationCommon shared] writeLog:@"Unsubscribed to Push Notification server & proxy successfully."];
  481. [CCUtility setPushNotificationPublicKey:account data:nil];
  482. [CCUtility setPushNotificationSubscribingPublicKey:account publicKey:nil];
  483. [CCUtility setPushNotificationPrivateKey:account data:nil];
  484. [CCUtility setPushNotificationToken:account token:nil];
  485. [CCUtility setPushNotificationDeviceIdentifier:account deviceIdentifier:nil];
  486. [CCUtility setPushNotificationDeviceIdentifierSignature:account deviceIdentifierSignature:nil];
  487. if (self.pushKitToken != nil && subscribing) {
  488. [self subscribingNextcloudServerPushNotification:account urlBase:urlBase user:user];
  489. }
  490. }
  491. }];
  492. }
  493. }];
  494. }
  495. -(void)userNotificationCenter:(UNUserNotificationCenter *)center willPresentNotification:(UNNotification *)notification withCompletionHandler:(void (^)(UNNotificationPresentationOptions options))completionHandler
  496. {
  497. //Called when a notification is delivered to a foreground app.
  498. completionHandler(UNNotificationPresentationOptionAlert);
  499. }
  500. -(void)userNotificationCenter:(UNUserNotificationCenter *)center didReceiveNotificationResponse:(nonnull UNNotificationResponse *)response withCompletionHandler:(nonnull void (^)(void))completionHandler
  501. {
  502. completionHandler();
  503. }
  504. - (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken
  505. {
  506. self.pushKitToken = [self stringWithDeviceToken:deviceToken];
  507. [self pushNotification];
  508. }
  509. - (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler
  510. {
  511. NSString *message = [userInfo objectForKey:@"subject"];
  512. if (message) {
  513. NSArray *results = [[NCManageDatabase sharedInstance] getAllAccount];
  514. for (tableAccount *result in results) {
  515. if ([CCUtility getPushNotificationPrivateKey:result.account]) {
  516. NSData *decryptionKey = [CCUtility getPushNotificationPrivateKey:result.account];
  517. NSString *decryptedMessage = [[NCPushNotificationEncryption sharedInstance] decryptPushNotification:message withDevicePrivateKey:decryptionKey];
  518. if (decryptedMessage) {
  519. NSData *data = [decryptedMessage dataUsingEncoding:NSUTF8StringEncoding];
  520. NSDictionary *json = [NSJSONSerialization JSONObjectWithData:data options:0 error:nil];
  521. NSInteger nid = [[json objectForKey:@"nid"] integerValue];
  522. BOOL delete = [[json objectForKey:@"delete"] boolValue];
  523. BOOL deleteAll = [[json objectForKey:@"delete-all"] boolValue];
  524. if (delete) {
  525. [self removeNotificationWithNotificationId:nid usingDecryptionKey:decryptionKey];
  526. } else if (deleteAll) {
  527. [self cleanAllNotifications];
  528. }
  529. }
  530. }
  531. }
  532. }
  533. completionHandler(UIBackgroundFetchResultNoData);
  534. }
  535. - (void)cleanAllNotifications
  536. {
  537. [[UNUserNotificationCenter currentNotificationCenter] removeAllDeliveredNotifications];
  538. }
  539. - (void)removeNotificationWithNotificationId:(NSInteger)notificationId usingDecryptionKey:(NSData *)key
  540. {
  541. // Check in pending notifications
  542. [[UNUserNotificationCenter currentNotificationCenter] getPendingNotificationRequestsWithCompletionHandler:^(NSArray<UNNotificationRequest *> * _Nonnull requests) {
  543. for (UNNotificationRequest *notificationRequest in requests) {
  544. NSString *message = [notificationRequest.content.userInfo objectForKey:@"subject"];
  545. NSString *decryptedMessage = [[NCPushNotificationEncryption sharedInstance] decryptPushNotification:message withDevicePrivateKey:key];
  546. if (decryptedMessage) {
  547. NSData *data = [decryptedMessage dataUsingEncoding:NSUTF8StringEncoding];
  548. NSDictionary *json = [NSJSONSerialization JSONObjectWithData:data options:0 error:nil];
  549. NSInteger nid = [[json objectForKey:@"nid"] integerValue];
  550. if (nid == notificationId) {
  551. [[UNUserNotificationCenter currentNotificationCenter] removePendingNotificationRequestsWithIdentifiers:@[notificationRequest.identifier]];
  552. }
  553. }
  554. }
  555. }];
  556. // Check in delivered notifications
  557. [[UNUserNotificationCenter currentNotificationCenter] getDeliveredNotificationsWithCompletionHandler:^(NSArray<UNNotification *> * _Nonnull notifications) {
  558. for (UNNotification *notification in notifications) {
  559. NSString *message = [notification.request.content.userInfo objectForKey:@"subject"];
  560. NSString *decryptedMessage = [[NCPushNotificationEncryption sharedInstance] decryptPushNotification:message withDevicePrivateKey:key];
  561. if (decryptedMessage) {
  562. NSData *data = [decryptedMessage dataUsingEncoding:NSUTF8StringEncoding];
  563. NSDictionary *json = [NSJSONSerialization JSONObjectWithData:data options:0 error:nil];
  564. NSInteger nid = [[json objectForKey:@"nid"] integerValue];
  565. if (nid == notificationId) {
  566. [[UNUserNotificationCenter currentNotificationCenter] removeDeliveredNotificationsWithIdentifiers:@[notification.request.identifier]];
  567. }
  568. }
  569. }
  570. }];
  571. }
  572. - (NSString *)stringWithDeviceToken:(NSData *)deviceToken
  573. {
  574. const char *data = [deviceToken bytes];
  575. NSMutableString *token = [NSMutableString string];
  576. for (NSUInteger i = 0; i < [deviceToken length]; i++) {
  577. [token appendFormat:@"%02.2hhX", data[i]];
  578. }
  579. return [token copy];
  580. }
  581. #pragma --------------------------------------------------------------------------------------------
  582. #pragma mark ===== Quick Actions - ShotcutItem =====
  583. #pragma --------------------------------------------------------------------------------------------
  584. - (void)configDynamicShortcutItems
  585. {
  586. NSString *bundleId = [NSBundle mainBundle].bundleIdentifier;
  587. UIApplicationShortcutIcon *shortcutMediaIcon = [UIApplicationShortcutIcon iconWithTemplateImageName:@"media"];
  588. UIApplicationShortcutItem *shortcutMedia = [[UIApplicationShortcutItem alloc] initWithType:[NSString stringWithFormat:@"%@.media", bundleId] localizedTitle:NSLocalizedString(@"_media_", nil) localizedSubtitle:nil icon:shortcutMediaIcon userInfo:nil];
  589. // add the array to our app
  590. if (shortcutMedia)
  591. [UIApplication sharedApplication].shortcutItems = @[shortcutMedia];
  592. }
  593. - (void)application:(UIApplication *)application performActionForShortcutItem:(UIApplicationShortcutItem *)shortcutItem completionHandler:(void (^)(BOOL))completionHandler
  594. {
  595. BOOL handledShortCutItem = [self handleShortCutItem:shortcutItem];
  596. completionHandler(handledShortCutItem);
  597. }
  598. - (BOOL)handleShortCutItem:(UIApplicationShortcutItem *)shortcutItem
  599. {
  600. BOOL handled = NO;
  601. NSString *bundleId = [NSBundle mainBundle].bundleIdentifier;
  602. NSString *shortcutMedia = [NSString stringWithFormat:@"%@.media", bundleId];
  603. if ([shortcutItem.type isEqualToString:shortcutMedia] && self.account) {
  604. dispatch_async(dispatch_get_main_queue(), ^{
  605. UITabBarController *tabBarController = (UITabBarController *)self.window.rootViewController;
  606. if ([tabBarController isKindOfClass:[UITabBarController class]]) {
  607. }
  608. UISplitViewController *splitViewController = (UISplitViewController *)self.window.rootViewController;
  609. if ([splitViewController isKindOfClass:[UISplitViewController class]]) {
  610. UINavigationController *navigationControllerMaster = (UINavigationController *)splitViewController.viewControllers.firstObject;
  611. if ([navigationControllerMaster isKindOfClass:[UINavigationController class]]) {
  612. UITabBarController *tabBarController = (UITabBarController *)navigationControllerMaster.topViewController;
  613. if ([tabBarController isKindOfClass:[UITabBarController class]]) {
  614. if (splitViewController.isCollapsed) {
  615. [navigationControllerMaster popToRootViewControllerAnimated:false];
  616. UINavigationController *navigationControllerMaster = (UINavigationController *)splitViewController.viewControllers.firstObject;
  617. if ([navigationControllerMaster isKindOfClass:[UINavigationController class]]) {
  618. UITabBarController *tabBarController = (UITabBarController *)navigationControllerMaster.topViewController;
  619. if ([tabBarController isKindOfClass:[UITabBarController class]]) {
  620. [tabBarController setSelectedIndex: k_tabBarApplicationIndexMedia];
  621. }
  622. }
  623. } else {
  624. if ([tabBarController isKindOfClass:[UITabBarController class]]) {
  625. [tabBarController setSelectedIndex: k_tabBarApplicationIndexMedia];
  626. }
  627. }
  628. }
  629. }
  630. }
  631. });
  632. handled = YES;
  633. }
  634. return handled;
  635. }
  636. #pragma --------------------------------------------------------------------------------------------
  637. #pragma mark ===== ApplicationIconBadgeNumber =====
  638. #pragma --------------------------------------------------------------------------------------------
  639. - (void)updateApplicationIconBadgeNumber
  640. {
  641. if (self.account.length == 0 || self.maintenanceMode) return;
  642. NSInteger counterDownload = [[NCOperationQueue shared] downloadCount];
  643. NSInteger counterUpload = [[NCManageDatabase sharedInstance] getMetadatasWithPredicate:[NSPredicate predicateWithFormat:@"status == %d OR status == %d OR status == %d", k_metadataStatusWaitUpload, k_metadataStatusInUpload, k_metadataStatusUploading]].count;
  644. NSInteger total = counterDownload + counterUpload;
  645. [UIApplication sharedApplication].applicationIconBadgeNumber = total;
  646. UITabBarController *tabBarController = (UITabBarController *)self.window.rootViewController;
  647. if ([tabBarController isKindOfClass:[UITabBarController class]]) {
  648. UITabBarItem *tabBarItem = [tabBarController.tabBar.items objectAtIndex:0];
  649. if (total > 0) {
  650. [tabBarItem setBadgeValue:[NSString stringWithFormat:@"%li", (unsigned long)total]];
  651. } else {
  652. [tabBarItem setBadgeValue:nil];
  653. }
  654. }
  655. }
  656. #pragma --------------------------------------------------------------------------------------------
  657. #pragma mark ===== Theming Color =====
  658. #pragma --------------------------------------------------------------------------------------------
  659. - (void)settingThemingColorBrand
  660. {
  661. if (self.account.length == 0 || self.maintenanceMode)
  662. return;
  663. if ([NCBrandOptions sharedInstance].use_themingColor) {
  664. NSString *themingColor = [[NCManageDatabase sharedInstance] getCapabilitiesServerStringWithAccount:self.account elements:NCElementsJSON.shared.capabilitiesThemingColor];
  665. NSString *themingColorElement = [[NCManageDatabase sharedInstance] getCapabilitiesServerStringWithAccount:self.account elements:NCElementsJSON.shared.capabilitiesThemingColorElement];
  666. NSString *themingColorText = [[NCManageDatabase sharedInstance] getCapabilitiesServerStringWithAccount:self.account elements:NCElementsJSON.shared.capabilitiesThemingColorText];
  667. [CCGraphics settingThemingColor:themingColor themingColorElement:themingColorElement themingColorText:themingColorText];
  668. BOOL isTooLight = NCBrandColor.sharedInstance.brandElement.isTooLight;
  669. BOOL isTooDark = NCBrandColor.sharedInstance.brandElement.isTooDark;
  670. if (isTooLight) {
  671. NCBrandColor.sharedInstance.brandElement = [NCBrandColor.sharedInstance.brandElement darkerBy:10];
  672. } else if (isTooDark) {
  673. NCBrandColor.sharedInstance.brandElement = [NCBrandColor.sharedInstance.brandElement lighterBy:40];
  674. }
  675. } else {
  676. BOOL isTooLight = NCBrandColor.sharedInstance.customer.isTooLight;
  677. BOOL isTooDark = NCBrandColor.sharedInstance.customer.isTooDark;
  678. if (isTooLight) {
  679. NCBrandColor.sharedInstance.brandElement = [NCBrandColor.sharedInstance.customer darkerBy:10];
  680. } else if (isTooDark) {
  681. NCBrandColor.sharedInstance.brandElement = [NCBrandColor.sharedInstance.customer lighterBy:40];
  682. } else {
  683. NCBrandColor.sharedInstance.brandElement = NCBrandColor.sharedInstance.customer;
  684. }
  685. NCBrandColor.sharedInstance.brand = NCBrandColor.sharedInstance.customer;
  686. NCBrandColor.sharedInstance.brandText = NCBrandColor.sharedInstance.customerText;
  687. }
  688. [NCBrandColor.sharedInstance setDarkMode];
  689. dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
  690. [[NCCollectionCommon shared] createImagesThemingColor];
  691. [[NSNotificationCenter defaultCenter] postNotificationOnMainThreadName:k_notificationCenter_changeTheming object:nil];
  692. });
  693. }
  694. - (void)changeTheming:(UIViewController *)viewController tableView:(UITableView *)tableView collectionView:(UICollectionView *)collectionView form:(BOOL)form
  695. {
  696. [NCBrandColor.sharedInstance setDarkMode];
  697. [self.window setTintColor:NCBrandColor.sharedInstance.textView];
  698. //Tab bar
  699. UITabBarController *tabBarController = (UITabBarController *)self.window.rootViewController;
  700. if ([tabBarController isKindOfClass:[UITabBarController class]]) {
  701. tabBarController.tabBar.barTintColor = NCBrandColor.sharedInstance.backgroundView;
  702. tabBarController.tabBar.backgroundColor = NCBrandColor.sharedInstance.tabBar;
  703. tabBarController.tabBar.tintColor = NCBrandColor.sharedInstance.brandElement;
  704. [tabBarController.tabBar viewWithTag:99].backgroundColor = NCBrandColor.sharedInstance.brandElement;
  705. }
  706. // Nav bar
  707. [self configureNavBarForViewController:viewController];
  708. // View
  709. if (form) viewController.view.backgroundColor = NCBrandColor.sharedInstance.backgroundForm;
  710. else viewController.view.backgroundColor = NCBrandColor.sharedInstance.backgroundView;
  711. // TableView
  712. if (tableView) {
  713. if (form) tableView.backgroundColor = NCBrandColor.sharedInstance.backgroundForm;
  714. else tableView.backgroundColor = NCBrandColor.sharedInstance.backgroundView;
  715. tableView.separatorColor = NCBrandColor.sharedInstance.separator;
  716. [tableView reloadData];
  717. }
  718. // CollectionView
  719. if (collectionView) {
  720. if (form) collectionView.backgroundColor = NCBrandColor.sharedInstance.backgroundForm;
  721. else collectionView.backgroundColor = NCBrandColor.sharedInstance.backgroundView;
  722. [collectionView reloadData];
  723. }
  724. }
  725. #pragma --------------------------------------------------------------------------------------------
  726. #pragma mark ===== Fetch =====
  727. #pragma --------------------------------------------------------------------------------------------
  728. - (void)application:(UIApplication *)application performFetchWithCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler
  729. {
  730. // Test Maintenance
  731. if (self.account.length == 0 || self.maintenanceMode) {
  732. completionHandler(UIBackgroundFetchResultNoData);
  733. return;
  734. }
  735. [[NCCommunicationCommon shared] writeLog:@"Start perform Fetch With Completion Handler"];
  736. // Verify new photo
  737. [[NCAutoUpload sharedInstance] initStateAutoUpload];
  738. // after 20 sec
  739. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, 20 * NSEC_PER_SEC), dispatch_get_main_queue(), ^{
  740. [[NCCommunicationCommon shared] writeLog:@"End 20 sec. perform Fetch With Completion Handler"];
  741. completionHandler(UIBackgroundFetchResultNoData);
  742. });
  743. }
  744. #pragma --------------------------------------------------------------------------------------------
  745. #pragma mark ===== Operation Networking & Session =====
  746. #pragma --------------------------------------------------------------------------------------------
  747. //
  748. // Method called by the system when all the background task has end
  749. //
  750. - (void)application:(UIApplication *)application handleEventsForBackgroundURLSession:(NSString *)identifier completionHandler:(void (^)(void))completionHandler
  751. {
  752. [[NCCommunicationCommon shared] writeLog:[NSString stringWithFormat:@"Start handle Events For Background URLSession: %@", identifier]];
  753. [self updateApplicationIconBadgeNumber];
  754. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, 20 * NSEC_PER_SEC), dispatch_get_main_queue(), ^{
  755. self.backgroundSessionCompletionHandler = completionHandler;
  756. void (^completionHandler)() = self.backgroundSessionCompletionHandler;
  757. self.backgroundSessionCompletionHandler = nil;
  758. completionHandler();
  759. });
  760. }
  761. #pragma --------------------------------------------------------------------------------------------
  762. #pragma mark ===== OpenURL =====
  763. #pragma --------------------------------------------------------------------------------------------
  764. // Method called from iOS system to send a file from other app.
  765. - (BOOL)application:(UIApplication *)application openURL:(NSURL *)url options:(NSDictionary<NSString *,id> *)options
  766. {
  767. if (self.account.length == 0 || self.maintenanceMode)
  768. return YES;
  769. NSString *scheme = url.scheme;
  770. NSString *fileName;
  771. NSString *serverUrl;
  772. if ([scheme isEqualToString:@"nextcloud"]) {
  773. NSString *action = url.host;
  774. if ([action isEqualToString:@"open-file"]) {
  775. NSURLComponents *urlComponents = [NSURLComponents componentsWithURL:url resolvingAgainstBaseURL:NO];
  776. NSArray *queryItems = urlComponents.queryItems;
  777. NSString *user = [CCUtility valueForKey:@"user" fromQueryItems:queryItems];
  778. NSString *path = [CCUtility valueForKey:@"path" fromQueryItems:queryItems];
  779. NSString *link = [CCUtility valueForKey:@"link" fromQueryItems:queryItems];
  780. tableAccount *matchedAccount = nil;
  781. // verify parameter
  782. if (user.length == 0 || path.length == 0 || [[NSURL URLWithString:link] host].length == 0) {
  783. UIAlertController *alertController = [UIAlertController alertControllerWithTitle:NSLocalizedString(@"_error_", nil) message:NSLocalizedString(@"_error_parameter_schema_", nil) preferredStyle:UIAlertControllerStyleAlert];
  784. UIAlertAction *okAction = [UIAlertAction actionWithTitle:NSLocalizedString(@"_ok_", nil) style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {}];
  785. [alertController addAction:okAction];
  786. [self.window.rootViewController presentViewController:alertController animated:YES completion:nil];
  787. } else {
  788. tableAccount *account = [[NCManageDatabase sharedInstance] getAccountActive];
  789. if (account) {
  790. NSURL *accountURL = [NSURL URLWithString:account.urlBase];
  791. NSString *accountUser = account.user;
  792. if ([link containsString:accountURL.host] && [user isEqualToString:accountUser]) {
  793. matchedAccount = account;
  794. } else {
  795. NSArray *accounts = [[NCManageDatabase sharedInstance] getAllAccount];
  796. for (tableAccount *account in accounts) {
  797. NSURL *accountURL = [NSURL URLWithString:account.urlBase];
  798. NSString *accountUser = account.user;
  799. if ([link containsString:accountURL.host] && [user isEqualToString:accountUser]) {
  800. matchedAccount = [[NCManageDatabase sharedInstance] setAccountActive:account.account];
  801. [self settingAccount:matchedAccount.account urlBase:matchedAccount.urlBase user:matchedAccount.user userID:matchedAccount.userID password:[CCUtility getPassword:matchedAccount.account]];
  802. [[NSNotificationCenter defaultCenter] postNotificationOnMainThreadName:k_notificationCenter_initializeMain object:nil userInfo:nil];
  803. }
  804. }
  805. }
  806. if (matchedAccount) {
  807. NSString *webDAV = [[NCUtility shared] getWebDAVWithAccount:self.account];
  808. if ([path containsString:@"/"]) {
  809. fileName = [path lastPathComponent];
  810. serverUrl = [NSString stringWithFormat:@"%@/%@/%@", matchedAccount.urlBase, webDAV, [path stringByDeletingLastPathComponent]];
  811. } else {
  812. fileName = path;
  813. serverUrl = [NSString stringWithFormat:@"%@/%@", matchedAccount.urlBase, webDAV];
  814. }
  815. [[NCCollectionCommon shared] openFileViewInFolderWithServerUrl:serverUrl fileName:fileName];
  816. } else {
  817. NSString *domain = [[NSURL URLWithString:link] host];
  818. NSString *fileName = [path lastPathComponent];
  819. NSString *message = [NSString stringWithFormat:NSLocalizedString(@"_account_not_available_", nil), user, domain, fileName];
  820. UIAlertController *alertController = [UIAlertController alertControllerWithTitle:NSLocalizedString(@"_info_", nil) message:message preferredStyle:UIAlertControllerStyleAlert];
  821. UIAlertAction *okAction = [UIAlertAction actionWithTitle:NSLocalizedString(@"_ok_", nil) style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {}];
  822. [alertController addAction:okAction];
  823. [self.window.rootViewController presentViewController:alertController animated:YES completion:nil];
  824. }
  825. }
  826. }
  827. }
  828. return YES;
  829. }
  830. NSError *error;
  831. NSLog(@"[LOG] the path is: %@", url.path);
  832. NSArray *splitedUrl = [url.path componentsSeparatedByString:@"/"];
  833. self.fileNameUpload = [NSString stringWithFormat:@"%@",[splitedUrl objectAtIndex:([splitedUrl count]-1)]];
  834. if (self.account) {
  835. [[NSFileManager defaultManager]removeItemAtPath:[NSTemporaryDirectory() stringByAppendingString:self.fileNameUpload] error:nil];
  836. [[NSFileManager defaultManager]moveItemAtPath:url.path toPath:[NSTemporaryDirectory() stringByAppendingString:self.fileNameUpload] error:&error];
  837. if (error == nil) {
  838. UISplitViewController *splitViewController = (UISplitViewController *)self.window.rootViewController;
  839. if ([splitViewController isKindOfClass:[UISplitViewController class]]) {
  840. UINavigationController *navigationControllerMaster = (UINavigationController *)splitViewController.viewControllers.firstObject;
  841. if ([navigationControllerMaster isKindOfClass:[UINavigationController class]]) {
  842. UIViewController *uploadNavigationViewController = [[UIStoryboard storyboardWithName:@"CCUploadFromOtherUpp" bundle:nil] instantiateViewControllerWithIdentifier:@"CCUploadNavigationViewController"];
  843. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, 1 * NSEC_PER_SEC), dispatch_get_main_queue(), ^{
  844. [navigationControllerMaster presentViewController:uploadNavigationViewController animated:YES completion:nil];
  845. });
  846. }
  847. }
  848. }
  849. }
  850. return YES;
  851. }
  852. #pragma --------------------------------------------------------------------------------------------
  853. #pragma mark ===== Passcode + Delegate =====
  854. #pragma --------------------------------------------------------------------------------------------
  855. - (void)passcodeWithAutomaticallyPromptForBiometricValidation:(BOOL)automaticallyPromptForBiometricValidation
  856. {
  857. LAContext *laContext = [LAContext new];
  858. NSError *error;
  859. BOOL isBiometryAvailable = false;
  860. if ([[CCUtility getPasscode] length] == 0 || [self.account length] == 0 || [CCUtility getNotPasscodeAtStart]) return;
  861. if (self.passcodeViewController == nil) {
  862. self.passcodeViewController = [[TOPasscodeViewController alloc] initWithStyle:TOPasscodeViewStyleTranslucentLight passcodeType:TOPasscodeTypeSixDigits];
  863. if (@available(iOS 13.0, *)) {
  864. if ([[UITraitCollection currentTraitCollection] userInterfaceStyle] == UIUserInterfaceStyleDark) {
  865. self.passcodeViewController.style = TOPasscodeViewStyleTranslucentDark;
  866. }
  867. }
  868. self.passcodeViewController.delegate = self;
  869. self.passcodeViewController.keypadButtonShowLettering = false;
  870. if (CCUtility.getEnableTouchFaceID && [laContext canEvaluatePolicy:LAPolicyDeviceOwnerAuthenticationWithBiometrics error:&error]) {
  871. if (error == NULL) {
  872. if (laContext.biometryType == LABiometryTypeFaceID) {
  873. self.passcodeViewController.biometryType = TOPasscodeBiometryTypeFaceID;
  874. self.passcodeViewController.allowBiometricValidation = true;
  875. isBiometryAvailable = true;
  876. } else if (laContext.biometryType == LABiometryTypeTouchID) {
  877. self.passcodeViewController.biometryType = TOPasscodeBiometryTypeTouchID;
  878. self.passcodeViewController.allowBiometricValidation = true;
  879. isBiometryAvailable = true;
  880. } else {
  881. isBiometryAvailable = false;
  882. NSLog(@"No Biometric support");
  883. }
  884. }
  885. }
  886. [self.window.rootViewController presentViewController:self.passcodeViewController animated:YES completion:^{
  887. [self enableTouchFaceID:automaticallyPromptForBiometricValidation];
  888. }];
  889. } else {
  890. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, 0.5 * NSEC_PER_SEC), dispatch_get_main_queue(), ^(void) {
  891. [self enableTouchFaceID:automaticallyPromptForBiometricValidation];
  892. });
  893. }
  894. }
  895. - (void)didInputCorrectPasscodeInPasscodeViewController:(TOPasscodeViewController *)passcodeViewController
  896. {
  897. [passcodeViewController dismissViewControllerAnimated:YES completion:^{
  898. self.passcodeViewController = nil;
  899. }];
  900. }
  901. - (BOOL)passcodeViewController:(TOPasscodeViewController *)passcodeViewController isCorrectCode:(NSString *)code
  902. {
  903. return [code isEqualToString:[CCUtility getPasscode]];
  904. }
  905. - (void)didPerformBiometricValidationRequestInPasscodeViewController:(TOPasscodeViewController *)passcodeViewController
  906. {
  907. [[LAContext new] evaluatePolicy:LAPolicyDeviceOwnerAuthenticationWithBiometrics localizedReason:[[NCBrandOptions sharedInstance] brand] reply:^(BOOL success, NSError * _Nullable error) {
  908. if (success) {
  909. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, 0.5 * NSEC_PER_SEC), dispatch_get_main_queue(), ^(void) {
  910. [passcodeViewController dismissViewControllerAnimated:YES completion:^{
  911. self.passcodeViewController = nil;
  912. }];
  913. });
  914. }
  915. }];
  916. }
  917. - (void)enableTouchFaceID:(BOOL)automaticallyPromptForBiometricValidation
  918. {
  919. if (CCUtility.getEnableTouchFaceID && automaticallyPromptForBiometricValidation && self.passcodeViewController.view.window) {
  920. [[LAContext new] evaluatePolicy:LAPolicyDeviceOwnerAuthenticationWithBiometrics localizedReason:[[NCBrandOptions sharedInstance] brand] reply:^(BOOL success, NSError * _Nullable error) {
  921. if (success) {
  922. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, 0.5 * NSEC_PER_SEC), dispatch_get_main_queue(), ^(void) {
  923. [self.passcodeViewController dismissViewControllerAnimated:YES completion:^{
  924. self.passcodeViewController = nil;
  925. }];
  926. });
  927. }
  928. }];
  929. }
  930. }
  931. #pragma --------------------------------------------------------------------------------------------
  932. #pragma mark ===== Maintenance Mode =====
  933. #pragma --------------------------------------------------------------------------------------------
  934. - (void)maintenanceMode:(BOOL)mode
  935. {
  936. self.maintenanceMode = mode;
  937. }
  938. @end