VLCAppDelegate.m 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481
  1. /*****************************************************************************
  2. * VLCAppDelegate.m
  3. * VLC for iOS
  4. *****************************************************************************
  5. * Copyright (c) 2013-2014 VideoLAN. All rights reserved.
  6. * $Id$
  7. *
  8. * Authors: Felix Paul Kühne <fkuehne # videolan.org>
  9. * Gleb Pinigin <gpinigin # gmail.com>
  10. * Jean-Romain Prévost <jr # 3on.fr>
  11. * Luis Fernandes <zipleen # gmail.com>
  12. * Carola Nitz <nitz.carola # googlemail.com>
  13. *
  14. * Refer to the COPYING file of the official project for license.
  15. *****************************************************************************/
  16. #import "VLCAppDelegate.h"
  17. #import "VLCMediaFileDiscoverer.h"
  18. #import "NSString+SupportedMedia.h"
  19. #import "UIDevice+SpeedCategory.h"
  20. #import "VLCPlaylistViewController.h"
  21. #import "VLCMovieViewController.h"
  22. #import "PAPasscodeViewController.h"
  23. #import "UINavigationController+Theme.h"
  24. #import "VLCHTTPUploaderController.h"
  25. #import "VLCMenuTableViewController.h"
  26. #import "BWQuincyManager.h"
  27. #import "VLCAlertView.h"
  28. @interface VLCAppDelegate () <PAPasscodeViewControllerDelegate, VLCMediaFileDiscovererDelegate, BWQuincyManagerDelegate> {
  29. PAPasscodeViewController *_passcodeLockController;
  30. VLCDropboxTableViewController *_dropboxTableViewController;
  31. VLCGoogleDriveTableViewController *_googleDriveTableViewController;
  32. VLCDownloadViewController *_downloadViewController;
  33. int _idleCounter;
  34. int _networkActivityCounter;
  35. VLCMovieViewController *_movieViewController;
  36. BOOL _passcodeValidated;
  37. }
  38. @end
  39. @implementation VLCAppDelegate
  40. + (void)initialize
  41. {
  42. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  43. NSNumber *skipLoopFilterDefaultValue;
  44. int deviceSpeedCategory = [[UIDevice currentDevice] speedCategory];
  45. if (deviceSpeedCategory < 3)
  46. skipLoopFilterDefaultValue = kVLCSettingSkipLoopFilterNonKey;
  47. else
  48. skipLoopFilterDefaultValue = kVLCSettingSkipLoopFilterNonRef;
  49. NSDictionary *appDefaults = @{kVLCSettingPasscodeKey : @"", kVLCSettingPasscodeOnKey : @(NO), kVLCSettingContinueAudioInBackgroundKey : @(YES), kVLCSettingStretchAudio : @(NO), kVLCSettingTextEncoding : kVLCSettingTextEncodingDefaultValue, kVLCSettingSkipLoopFilter : skipLoopFilterDefaultValue, kVLCSettingSubtitlesFont : kVLCSettingSubtitlesFontDefaultValue, kVLCSettingSubtitlesFontColor : kVLCSettingSubtitlesFontColorDefaultValue, kVLCSettingSubtitlesFontSize : kVLCSettingSubtitlesFontSizeDefaultValue, kVLCSettingSubtitlesBoldFont: kVLCSettingSubtitlesBoldFontDefaulValue, kVLCSettingDeinterlace : kVLCSettingDeinterlaceDefaultValue, kVLCSettingNetworkCaching : kVLCSettingNetworkCachingDefaultValue, kVLCSettingPlaybackGestures : [NSNumber numberWithBool:YES], kVLCSettingFTPTextEncoding : kVLCSettingFTPTextEncodingDefaultValue };
  50. [defaults registerDefaults:appDefaults];
  51. }
  52. - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
  53. {
  54. if (SYSTEM_RUNS_IOS7_OR_LATER) {
  55. // Change the keyboard for UISearchBar
  56. [[UITextField appearance] setKeyboardAppearance:UIKeyboardAppearanceDark];
  57. // For the cursor
  58. [[UITextField appearance] setTintColor:[UIColor VLCOrangeTintColor]];
  59. // Don't override the 'Cancel' button color in the search bar with the previous UITextField call. Use the default blue color
  60. [[UIBarButtonItem appearanceWhenContainedIn:[UISearchBar class], nil] setTitleTextAttributes:@{NSForegroundColorAttributeName: [UIColor colorWithRed:0.0 green:122.0/255.0 blue:1.0 alpha:1.0]} forState:UIControlStateNormal];
  61. // For the edit selection indicators
  62. [[UITableView appearance] setTintColor:[UIColor VLCOrangeTintColor]];
  63. }
  64. [[UISwitch appearance] setOnTintColor:[UIColor VLCOrangeTintColor]];
  65. BWQuincyManager *quincyManager = [BWQuincyManager sharedQuincyManager];
  66. [quincyManager setSubmissionURL:@"http://crash.videolan.org/crash_v200.php"];
  67. [quincyManager setDelegate:self];
  68. [quincyManager setShowAlwaysButton:YES];
  69. [quincyManager startManager];
  70. /* clean caches on launch (since those are used for wifi upload only) */
  71. [self cleanCache];
  72. // Init the HTTP Server
  73. self.uploadController = [[VLCHTTPUploaderController alloc] init];
  74. // enable crash preventer
  75. [[MLMediaLibrary sharedMediaLibrary] applicationWillStart];
  76. self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
  77. _playlistViewController = [[VLCPlaylistViewController alloc] init];
  78. UINavigationController *navCon = [[UINavigationController alloc] initWithRootViewController:_playlistViewController];
  79. [navCon loadTheme];
  80. _revealController = [[GHRevealViewController alloc] initWithNibName:nil bundle:nil];
  81. _revealController.wantsFullScreenLayout = YES;
  82. _menuViewController = [[VLCMenuTableViewController alloc] initWithNibName:nil bundle:nil];
  83. _revealController.sidebarViewController = _menuViewController;
  84. _revealController.contentViewController = navCon;
  85. self.window.rootViewController = self.revealController;
  86. // necessary to avoid navbar blinking in VLCOpenNetworkStreamViewController & VLCDownloadViewController
  87. _revealController.contentViewController.view.backgroundColor = [UIColor VLCDarkBackgroundColor];
  88. [self.window makeKeyAndVisible];
  89. VLCMediaFileDiscoverer *discoverer = [VLCMediaFileDiscoverer sharedInstance];
  90. [discoverer addObserver:self];
  91. [discoverer startDiscovering:[self directoryPath]];
  92. [self validatePasscode];
  93. return YES;
  94. }
  95. - (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation
  96. {
  97. if ([[DBSession sharedSession] handleOpenURL:url]) {
  98. [self.dropboxTableViewController updateViewAfterSessionChange];
  99. return YES;
  100. }
  101. if (_playlistViewController && url != nil) {
  102. APLog(@"%@ requested %@ to be opened", sourceApplication, url);
  103. if (url.isFileURL) {
  104. NSArray *searchPaths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
  105. NSString *directoryPath = searchPaths[0];
  106. NSURL *destinationURL = [NSURL fileURLWithPath:[NSString stringWithFormat:@"%@/%@", directoryPath, url.lastPathComponent]];
  107. NSError *theError;
  108. [[NSFileManager defaultManager] moveItemAtURL:url toURL:destinationURL error:&theError];
  109. if (theError.code != noErr)
  110. APLog(@"saving the file failed (%li): %@", (long)theError.code, theError.localizedDescription);
  111. [self updateMediaList];
  112. } else if ([url.scheme isEqualToString:@"vlc-x-callback"] || [url.host isEqualToString:@"x-callback-url"]) {
  113. // URL confirmes to the x-callback-url specification
  114. // vlc-x-callback://x-callback-url/action?param=value&x-success=callback
  115. APLog(@"x-callback-url with host '%@' path '%@' parameters '%@'", url.host, url.path, url.query);
  116. NSString *action = [url.path stringByReplacingOccurrencesOfString:@"/" withString:@""];
  117. NSURL *movieURL = nil;
  118. NSURL *successCallback = nil;
  119. for (NSString *entry in [url.query componentsSeparatedByString:@"&"]) {
  120. NSArray *keyvalue = [entry componentsSeparatedByString:@"="];
  121. if (keyvalue.count < 2) continue;
  122. NSString *key = keyvalue[0];
  123. NSString *value = [keyvalue[1] stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
  124. if ([key isEqualToString:@"url"]) {
  125. movieURL = [NSURL URLWithString:value];
  126. }
  127. else if ([key isEqualToString:@"x-success"]) {
  128. successCallback = [NSURL URLWithString:value];
  129. }
  130. }
  131. if ([action isEqualToString:@"stream"] && movieURL) {
  132. [self openMovieFromURL:movieURL successCallback:successCallback];
  133. }
  134. } else {
  135. NSString *receivedUrl = [url absoluteString];
  136. if ([receivedUrl length] > 6) {
  137. NSString *verifyVlcUrl = [receivedUrl substringToIndex:6];
  138. if ([verifyVlcUrl isEqualToString:@"vlc://"]) {
  139. NSString *parsedString = [receivedUrl substringFromIndex:6];
  140. NSUInteger location = [parsedString rangeOfString:@"//"].location;
  141. /* Safari & al mangle vlc://http:// so fix this */
  142. if (location != NSNotFound && [parsedString characterAtIndex:location - 1] != 0x3a) { // :
  143. parsedString = [NSString stringWithFormat:@"%@://%@", [parsedString substringToIndex:location], [parsedString substringFromIndex:location+2]];
  144. } else {
  145. parsedString = [receivedUrl substringFromIndex:6];
  146. if (![parsedString hasPrefix:@"http://"] && ![parsedString hasPrefix:@"https://"] && ![parsedString hasPrefix:@"ftp://"]) {
  147. parsedString = [@"http://" stringByAppendingString:[receivedUrl substringFromIndex:6]];
  148. }
  149. }
  150. url = [NSURL URLWithString:parsedString];
  151. }
  152. }
  153. [self.menuViewController selectRowAtIndexPath:[NSIndexPath indexPathForRow:0 inSection:0] animated:NO scrollPosition:UITableViewScrollPositionNone];
  154. NSString *scheme = url.scheme;
  155. if ([scheme isEqualToString:@"http"] || [scheme isEqualToString:@"https"] || [scheme isEqualToString:@"ftp"]) {
  156. VLCAlertView *alert = [[VLCAlertView alloc] initWithTitle:NSLocalizedString(@"OPEN_STREAM_OR_DOWNLOAD", nil) message:url.absoluteString cancelButtonTitle:NSLocalizedString(@"BUTTON_DOWNLOAD", nil) otherButtonTitles:@[NSLocalizedString(@"BUTTON_PLAY", nil)]];
  157. alert.completion = ^(BOOL cancelled, NSInteger buttonIndex) {
  158. if (cancelled)
  159. [[self downloadViewController] addURLToDownloadList:url fileNameOfMedia:nil];
  160. else
  161. [self openMovieFromURL:url];
  162. };
  163. [alert show];
  164. } else
  165. [self openMovieFromURL:url];
  166. }
  167. return YES;
  168. }
  169. return NO;
  170. }
  171. - (void)applicationWillEnterForeground:(UIApplication *)application
  172. {
  173. [[MLMediaLibrary sharedMediaLibrary] applicationWillStart];
  174. }
  175. - (void)applicationWillResignActive:(UIApplication *)application
  176. {
  177. _passcodeValidated = NO;
  178. [self validatePasscode];
  179. [[MLMediaLibrary sharedMediaLibrary] applicationWillExit];
  180. }
  181. - (void)applicationDidBecomeActive:(UIApplication *)application
  182. {
  183. [[MLMediaLibrary sharedMediaLibrary] updateMediaDatabase];
  184. [self updateMediaList];
  185. }
  186. - (void)applicationWillTerminate:(UIApplication *)application
  187. {
  188. _passcodeValidated = NO;
  189. [[NSUserDefaults standardUserDefaults] synchronize];
  190. }
  191. #pragma mark - properties
  192. - (VLCDropboxTableViewController *)dropboxTableViewController
  193. {
  194. if (_dropboxTableViewController == nil)
  195. _dropboxTableViewController = [[VLCDropboxTableViewController alloc] initWithNibName:@"VLCCloudStorageTableViewController" bundle:nil];
  196. return _dropboxTableViewController;
  197. }
  198. - (VLCGoogleDriveTableViewController *)googleDriveTableViewController
  199. {
  200. if (_googleDriveTableViewController == nil)
  201. _googleDriveTableViewController = [[VLCGoogleDriveTableViewController alloc] initWithNibName:@"VLCCloudStorageTableViewController" bundle:nil];
  202. return _googleDriveTableViewController;
  203. }
  204. - (VLCDownloadViewController *)downloadViewController
  205. {
  206. if (_downloadViewController == nil) {
  207. if (SYSTEM_RUNS_IOS7_OR_LATER)
  208. _downloadViewController = [[VLCDownloadViewController alloc] initWithNibName:@"VLCFutureDownloadViewController" bundle:nil];
  209. else
  210. _downloadViewController = [[VLCDownloadViewController alloc] initWithNibName:@"VLCDownloadViewController" bundle:nil];
  211. }
  212. return _downloadViewController;
  213. }
  214. #pragma mark - media discovering
  215. - (void)mediaFileAdded:(NSString *)fileName loading:(BOOL)isLoading
  216. {
  217. if (!isLoading) {
  218. MLMediaLibrary *sharedLibrary = [MLMediaLibrary sharedMediaLibrary];
  219. [sharedLibrary addFilePaths:@[fileName]];
  220. /* exclude media files from backup (QA1719) */
  221. NSURL *excludeURL = [NSURL fileURLWithPath:fileName];
  222. [excludeURL setResourceValue:@YES forKey:NSURLIsExcludedFromBackupKey error:nil];
  223. // TODO Should we update media db after adding new files?
  224. [sharedLibrary updateMediaDatabase];
  225. [_playlistViewController updateViewContents];
  226. }
  227. }
  228. - (void)mediaFileDeleted:(NSString *)name
  229. {
  230. [[MLMediaLibrary sharedMediaLibrary] updateMediaDatabase];
  231. [_playlistViewController updateViewContents];
  232. }
  233. - (void)cleanCache
  234. {
  235. if ([self haveNetworkActivity])
  236. return;
  237. NSArray *searchPaths = NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES);
  238. NSString* uploadDirPath = [searchPaths[0] stringByAppendingPathComponent:@"Upload"];
  239. NSFileManager *fileManager = [NSFileManager defaultManager];
  240. if ([fileManager fileExistsAtPath:uploadDirPath])
  241. [fileManager removeItemAtPath:uploadDirPath error:nil];
  242. }
  243. #pragma mark - media list methods
  244. - (NSString *)directoryPath
  245. {
  246. NSArray *searchPaths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
  247. NSString *directoryPath = searchPaths[0];
  248. return directoryPath;
  249. }
  250. - (void)updateMediaList
  251. {
  252. NSString *directoryPath = [self directoryPath];
  253. NSMutableArray *foundFiles = [NSMutableArray arrayWithObject:@""];
  254. NSMutableArray *filePaths = [NSMutableArray array];
  255. NSURL *fileURL;
  256. while (foundFiles.count) {
  257. NSString *fileName = foundFiles.firstObject;
  258. NSString *filePath = [directoryPath stringByAppendingPathComponent:fileName];
  259. [foundFiles removeObject:fileName];
  260. if ([fileName isSupportedMediaFormat] || [fileName isSupportedAudioMediaFormat]) {
  261. [filePaths addObject:filePath];
  262. /* exclude media files from backup (QA1719) */
  263. fileURL = [NSURL fileURLWithPath:filePath];
  264. [fileURL setResourceValue:@YES forKey:NSURLIsExcludedFromBackupKey error:nil];
  265. } else {
  266. BOOL isDirectory = NO;
  267. BOOL exists = [[NSFileManager defaultManager] fileExistsAtPath:filePath isDirectory:&isDirectory];
  268. // add folders
  269. if (exists && isDirectory) {
  270. NSArray *files = [[NSFileManager defaultManager] contentsOfDirectoryAtPath:filePath error:nil];
  271. for (NSString* file in files) {
  272. NSString *fullFilePath = [directoryPath stringByAppendingPathComponent:file];
  273. isDirectory = NO;
  274. exists = [[NSFileManager defaultManager] fileExistsAtPath:fullFilePath isDirectory:&isDirectory];
  275. //only add folders or files in folders
  276. if ((exists && isDirectory) || ![filePath.lastPathComponent isEqualToString:@"Documents"]) {
  277. NSString *folderpath = [filePath stringByReplacingOccurrencesOfString:directoryPath withString:@""];
  278. if (![folderpath isEqualToString:@""]) {
  279. folderpath = [folderpath stringByAppendingString:@"/"];
  280. }
  281. NSString *path = [folderpath stringByAppendingString:file];
  282. [foundFiles addObject:path];
  283. }
  284. }
  285. }
  286. }
  287. }
  288. [[MLMediaLibrary sharedMediaLibrary] addFilePaths:filePaths];
  289. [_playlistViewController updateViewContents];
  290. }
  291. #pragma mark - pass code validation
  292. - (void)validatePasscode
  293. {
  294. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  295. NSString *passcode = [defaults objectForKey:kVLCSettingPasscodeKey];
  296. if ([passcode isEqualToString:@""] || ![[defaults objectForKey:kVLCSettingPasscodeOnKey] boolValue]) {
  297. _passcodeValidated = YES;
  298. return;
  299. }
  300. if (!_passcodeValidated) {
  301. _passcodeLockController = [[PAPasscodeViewController alloc] initForAction:PasscodeActionEnter];
  302. _passcodeLockController.delegate = self;
  303. _passcodeLockController.passcode = passcode;
  304. if (self.window.rootViewController.presentedViewController)
  305. [self.window.rootViewController dismissViewControllerAnimated:NO completion:nil];
  306. UINavigationController *navCon = [[UINavigationController alloc] initWithRootViewController:_passcodeLockController];
  307. navCon.modalPresentationStyle = UIModalPresentationFullScreen;
  308. [self.window.rootViewController presentViewController:navCon animated:NO completion:nil];
  309. }
  310. }
  311. - (void)PAPasscodeViewControllerDidEnterPasscode:(PAPasscodeViewController *)controller
  312. {
  313. [self.window.rootViewController dismissViewControllerAnimated:YES completion:nil];
  314. }
  315. - (void)PAPasscodeViewController:(PAPasscodeViewController *)controller didFailToEnterPasscode:(NSInteger)attempts
  316. {
  317. // FIXME: handle countless failed passcode attempts
  318. }
  319. #pragma mark - idle timer preventer
  320. - (void)disableIdleTimer
  321. {
  322. _idleCounter++;
  323. if ([UIApplication sharedApplication].idleTimerDisabled == NO)
  324. [UIApplication sharedApplication].idleTimerDisabled = YES;
  325. }
  326. - (void)activateIdleTimer
  327. {
  328. _idleCounter--;
  329. if (_idleCounter < 1)
  330. [UIApplication sharedApplication].idleTimerDisabled = NO;
  331. }
  332. - (void)networkActivityStarted
  333. {
  334. _networkActivityCounter++;
  335. if ([UIApplication sharedApplication].networkActivityIndicatorVisible == NO)
  336. [UIApplication sharedApplication].networkActivityIndicatorVisible = YES;
  337. }
  338. - (BOOL)haveNetworkActivity
  339. {
  340. return _networkActivityCounter >= 1;
  341. }
  342. - (void)networkActivityStopped
  343. {
  344. _networkActivityCounter--;
  345. if (_networkActivityCounter < 1)
  346. [UIApplication sharedApplication].networkActivityIndicatorVisible = NO;
  347. }
  348. #pragma mark - playback view handling
  349. - (void)openMediaFromManagedObject:(NSManagedObject *)mediaObject
  350. {
  351. if (!_movieViewController)
  352. _movieViewController = [[VLCMovieViewController alloc] initWithNibName:nil bundle:nil];
  353. if ([mediaObject isKindOfClass:[MLFile class]])
  354. _movieViewController.fileFromMediaLibrary = (MLFile *)mediaObject;
  355. else if ([mediaObject isKindOfClass:[MLAlbumTrack class]])
  356. _movieViewController.fileFromMediaLibrary = [(MLAlbumTrack*)mediaObject files].anyObject;
  357. else if ([mediaObject isKindOfClass:[MLShowEpisode class]])
  358. _movieViewController.fileFromMediaLibrary = [(MLShowEpisode*)mediaObject files].anyObject;
  359. [(MLFile *)_movieViewController.fileFromMediaLibrary setUnread:@(NO)];
  360. UINavigationController *navCon = [[UINavigationController alloc] initWithRootViewController:_movieViewController];
  361. navCon.modalPresentationStyle = UIModalPresentationFullScreen;
  362. [self.window.rootViewController presentViewController:navCon animated:YES completion:nil];
  363. }
  364. - (void)openMovieFromURL:(NSURL *)url
  365. successCallback:(NSURL *)successCallback
  366. {
  367. if (!_movieViewController)
  368. _movieViewController = [[VLCMovieViewController alloc] initWithNibName:nil bundle:nil];
  369. _movieViewController.url = url;
  370. _movieViewController.successCallback = successCallback;
  371. UINavigationController *navCon = [[UINavigationController alloc] initWithRootViewController:_movieViewController];
  372. navCon.modalPresentationStyle = UIModalPresentationFullScreen;
  373. [self.window.rootViewController presentViewController:navCon animated:YES completion:nil];
  374. }
  375. - (void)openMovieFromURL:(NSURL *)url
  376. {
  377. [self openMovieFromURL:url successCallback:nil];
  378. }
  379. - (void)openMediaList:(VLCMediaList *)list atIndex:(int)index
  380. {
  381. if (!_movieViewController)
  382. _movieViewController = [[VLCMovieViewController alloc] initWithNibName:nil bundle:nil];
  383. _movieViewController.mediaList = list;
  384. _movieViewController.itemInMediaListToBePlayedFirst = index;
  385. _movieViewController.pathToExternalSubtitlesFile = nil;
  386. UINavigationController *navCon = [[UINavigationController alloc] initWithRootViewController:_movieViewController];
  387. navCon.modalPresentationStyle = UIModalPresentationFullScreen;
  388. [self.window.rootViewController presentViewController:navCon animated:YES completion:nil];
  389. }
  390. - (void)openMovieWithExternalSubtitleFromURL:(NSURL *)url externalSubURL:(NSString *)SubtitlePath
  391. {
  392. if (!_movieViewController)
  393. _movieViewController = [[VLCMovieViewController alloc] initWithNibName:nil bundle:nil];
  394. _movieViewController.url = url;
  395. _movieViewController.pathToExternalSubtitlesFile = SubtitlePath;
  396. UINavigationController *navCon = [[UINavigationController alloc] initWithRootViewController:_movieViewController];
  397. navCon.modalPresentationStyle = UIModalPresentationFullScreen;
  398. [self.window.rootViewController presentViewController:navCon animated:YES completion:nil];
  399. }
  400. @end