CCDetail.m 45 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014
  1. //
  2. // CCDetail.m
  3. // Nextcloud iOS
  4. //
  5. // Created by Marino Faggiana on 16/01/15.
  6. // Copyright (c) 2017 Marino Faggiana. All rights reserved.
  7. //
  8. // Author Marino Faggiana <m.faggiana@twsweb.it>
  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 "CCDetail.h"
  24. #import "AppDelegate.h"
  25. #import "CCMain.h"
  26. #import "NCUchardet.h"
  27. #import "NCBridgeSwift.h"
  28. #import <KTVHTTPCache/KTVHTTPCache.h>
  29. #import "NCBridgeSwift.h"
  30. #define TOOLBAR_HEIGHT 49.0f
  31. #define alertRequestPasswordPDF 1
  32. @interface CCDetail () <NCTextDelegate, UIDocumentInteractionControllerDelegate>
  33. {
  34. AppDelegate *appDelegate;
  35. UIDocumentInteractionController *docController;
  36. UIBarButtonItem *buttonModifyTxt;
  37. UIBarButtonItem *buttonShare;
  38. UIBarButtonItem *buttonDelete;
  39. NSInteger indexNowVisible;
  40. NSString *fileIDNowVisible;
  41. NSMutableOrderedSet *dataSourceDirectoryID;
  42. NSString *fileNameExtension;
  43. }
  44. @end
  45. @implementation CCDetail
  46. #pragma --------------------------------------------------------------------------------------------
  47. #pragma mark ===== init =====
  48. #pragma --------------------------------------------------------------------------------------------
  49. - (id)initWithCoder:(NSCoder *)aDecoder
  50. {
  51. if (self = [super initWithCoder:aDecoder]) {
  52. appDelegate = (AppDelegate *)[[UIApplication sharedApplication] delegate];
  53. [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(triggerProgressTask:) name:@"NotificationProgressTask" object:nil];
  54. [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(changeTheming) name:@"changeTheming" object:nil];
  55. self.metadataDetail = [[tableMetadata alloc] init];
  56. self.photos = [[NSMutableArray alloc] init];
  57. self.photoDataSource = [NSMutableArray new];
  58. dataSourceDirectoryID = [[NSMutableOrderedSet alloc] init];
  59. indexNowVisible = -1;
  60. fileIDNowVisible = nil;
  61. appDelegate.activeDetail = self;
  62. }
  63. return self;
  64. }
  65. #pragma --------------------------------------------------------------------------------------------
  66. #pragma mark ===== View =====
  67. #pragma --------------------------------------------------------------------------------------------
  68. - (void)viewDidLoad
  69. {
  70. [super viewDidLoad];
  71. [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(insertGeocoderLocation:) name:@"insertGeocoderLocation" object:nil];
  72. self.imageBackground.image = [UIImage imageNamed:@"backgroundDetail"];
  73. // Change bar bottom line shadow and remove title back button <"title"
  74. self.navigationController.navigationBar.shadowImage = [CCGraphics generateSinglePixelImageWithColor:[NCBrandColor sharedInstance].brand];
  75. self.navigationController.navigationBar.topItem.title = @"";
  76. // Color Navigation Controller
  77. [appDelegate aspectNavigationControllerBar:self.navigationController.navigationBar online:[appDelegate.reachability isReachable] hidden:NO];
  78. // TabBar
  79. self.tabBarController.tabBar.hidden = YES;
  80. self.tabBarController.tabBar.translucent = YES;
  81. // Open View
  82. if ([self.metadataDetail.fileNameView length] > 0 || [self.metadataDetail.directoryID length] > 0 || [self.metadataDetail.fileID length] > 0) {
  83. // open view
  84. [self viewFile];
  85. }
  86. }
  87. - (void)viewWillAppear:(BOOL)animated
  88. {
  89. [super viewWillAppear:animated];
  90. self.tabBarController.tabBar.hidden = YES;
  91. self.tabBarController.tabBar.translucent = YES;
  92. }
  93. - (void)viewDidDisappear:(BOOL)animated
  94. {
  95. [super viewDidDisappear:animated];
  96. // If AVPlayer in play -> Stop
  97. if (appDelegate.player != nil && appDelegate.player.rate != 0) {
  98. [appDelegate.player pause];
  99. }
  100. // remove Observer AVPlayer
  101. if (self.isMediaObserver) {
  102. self.isMediaObserver = NO;
  103. @try{
  104. [[NCViewerMedia sharedInstance] removeObserver];
  105. }@catch(id anException) { }
  106. }
  107. }
  108. - (void)changeTheming
  109. {
  110. [appDelegate changeTheming:self];
  111. if (self.toolbar) {
  112. self.toolbar.barTintColor = [NCBrandColor sharedInstance].tabBar;
  113. self.toolbar.tintColor = [NCBrandColor sharedInstance].brandElement;
  114. }
  115. }
  116. - (void)changeToDisplayMode
  117. {
  118. if (_readerPDFViewController)
  119. [self.readerPDFViewController updateContentViews];
  120. }
  121. #pragma --------------------------------------------------------------------------------------------
  122. #pragma mark ===== View File =====
  123. #pragma --------------------------------------------------------------------------------------------
  124. - (void)viewFile
  125. {
  126. // Title
  127. self.navigationController.navigationBar.topItem.title = _metadataDetail.fileNameView;
  128. // verifico se esiste l'icona e se la posso creare
  129. if ([[NSFileManager defaultManager] fileExistsAtPath:[CCUtility getDirectoryProviderStorageIconFileID:self.metadataDetail.fileID fileNameView:self.metadataDetail.fileNameView]] == NO) {
  130. [CCGraphics createNewImageFrom:self.metadataDetail.fileNameView fileID:self.metadataDetail.fileID extension:[self.metadataDetail.fileNameView pathExtension] filterGrayScale:NO typeFile:self.metadataDetail.typeFile writeImage:YES];
  131. }
  132. // remove Observer AVPlayer
  133. if (self.isMediaObserver) {
  134. self.isMediaObserver = NO;
  135. [[NCViewerMedia sharedInstance] removeObserver];
  136. }
  137. // IMAGE
  138. if ([self.metadataDetail.typeFile isEqualToString: k_metadataTypeFile_image]) {
  139. self.edgesForExtendedLayout = UIRectEdgeAll;
  140. [self viewImage];
  141. }
  142. // AUDIO VIDEO
  143. if ([self.metadataDetail.typeFile isEqualToString: k_metadataTypeFile_video] || [self.metadataDetail.typeFile isEqualToString: k_metadataTypeFile_audio]) {
  144. self.edgesForExtendedLayout = UIRectEdgeAll;
  145. [self createToolbar];
  146. [[NCViewerMedia sharedInstance] viewMedia:self.metadataDetail detail:self];
  147. }
  148. // DOCUMENT
  149. if ([self.metadataDetail.typeFile isEqualToString: k_metadataTypeFile_document]) {
  150. BOOL openWithRichDocument = false;
  151. fileNameExtension = [[self.metadataDetail.fileNameView pathExtension] uppercaseString];
  152. if ([fileNameExtension isEqualToString:@"PDF"]) {
  153. self.edgesForExtendedLayout = UIRectEdgeBottom;
  154. [self createToolbar];
  155. [self viewPDF:@""];
  156. return;
  157. }
  158. // Very if mimeType is compatible with Rich Document viewer
  159. NSString *mimeType = [CCUtility getMimeType:self.metadataDetail.fileNameView];
  160. NSArray *richdocumentsMimetypes = [[NCManageDatabase sharedInstance] getRichdocumentsMimetypes];
  161. if (richdocumentsMimetypes.count > 0 & mimeType != nil && [mimeType componentsSeparatedByString:@"."].count > 2) {
  162. NSArray *mimeTypeArray = [mimeType componentsSeparatedByString:@"."];
  163. NSString *mimeType = [NSString stringWithFormat:@"%@.%@",mimeTypeArray[mimeTypeArray.count-2], mimeTypeArray[mimeTypeArray.count-1]];
  164. for (NSString *richdocumentMimetype in richdocumentsMimetypes) {
  165. if ([richdocumentMimetype containsString:mimeType]) {
  166. openWithRichDocument = true;
  167. }
  168. }
  169. }
  170. if (openWithRichDocument) {
  171. OCnetworking *ocNetworking = [[OCnetworking alloc] initWithDelegate:nil metadataNet:nil withUser:appDelegate.activeUser withUserID:appDelegate.activeUserID withPassword:appDelegate.activePassword withUrl:appDelegate.activeUrl];
  172. [ocNetworking createLinkRichdocumentsWithFileID:self.metadataDetail.fileID success:^(NSString *link) {
  173. [[NCViewerRichdocument sharedInstance] viewRichDocumentAt:link detail:self];
  174. } failure:^(NSString *message, NSInteger errorCode) {
  175. [appDelegate messageNotification:@"_error_" description:message visible:YES delay:k_dismissAfterSecond type:TWMessageBarMessageTypeError errorCode:errorCode];
  176. [self.navigationController popViewControllerAnimated:YES];
  177. }];
  178. return;
  179. }
  180. self.edgesForExtendedLayout = UIRectEdgeBottom;
  181. [self createToolbar];
  182. [[NCViewerDocumentWeb sharedInstance] viewDocumentWebAt:self.metadataDetail detail:self];
  183. }
  184. }
  185. #pragma --------------------------------------------------------------------------------------------
  186. #pragma mark ===== Toolbar =====
  187. #pragma --------------------------------------------------------------------------------------------
  188. - (void)createToolbar
  189. {
  190. CGFloat safeAreaBottom = 0;
  191. NSString *serverUrl = [[NCManageDatabase sharedInstance] getServerUrl:_metadataDetail.directoryID];
  192. if (!serverUrl)
  193. return;
  194. if (@available(iOS 11, *)) {
  195. safeAreaBottom = [UIApplication sharedApplication].delegate.window.safeAreaInsets.bottom;
  196. }
  197. self.toolbar = [[UIToolbar alloc] initWithFrame:CGRectMake(0, self.view.bounds.size.height - TOOLBAR_HEIGHT - safeAreaBottom, self.view.bounds.size.width, TOOLBAR_HEIGHT)];
  198. UIBarButtonItem *flexible = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemFlexibleSpace target:self action:nil];
  199. UIBarButtonItem *fixedSpaceMini = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemFixedSpace target:self action:nil];
  200. fixedSpaceMini.width = 25;
  201. buttonModifyTxt = [[UIBarButtonItem alloc] initWithImage:[UIImage imageNamed:@"actionSheetModify"] style:UIBarButtonItemStylePlain target:self action:@selector(modifyTxtButtonPressed:)];
  202. if (![NCBrandOptions sharedInstance].disable_openin_file) {
  203. self.buttonAction = [[UIBarButtonItem alloc] initWithImage:[UIImage imageNamed:@"openFile"] style:UIBarButtonItemStylePlain target:self action:@selector(actionButtonPressed:)];
  204. }
  205. buttonShare = [[UIBarButtonItem alloc] initWithImage:[UIImage imageNamed:@"share"] style:UIBarButtonItemStylePlain target:self action:@selector(shareButtonPressed:)];
  206. buttonDelete = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemTrash target:self action:@selector(deleteButtonPressed:)];
  207. if ([CCUtility isDocumentModifiableExtension:fileNameExtension]) {
  208. if ([CCUtility isFolderEncrypted:serverUrl account:appDelegate.activeAccount]) // E2EE
  209. [self.toolbar setItems:[NSArray arrayWithObjects: buttonModifyTxt, flexible, buttonDelete, fixedSpaceMini, self.buttonAction, nil]];
  210. else
  211. [self.toolbar setItems:[NSArray arrayWithObjects: buttonModifyTxt, flexible, buttonDelete, fixedSpaceMini, buttonShare, fixedSpaceMini, self.buttonAction, nil]];
  212. } else {
  213. if ([CCUtility isFolderEncrypted:serverUrl account:appDelegate.activeAccount]) // E2EE
  214. [self.toolbar setItems:[NSArray arrayWithObjects: flexible, buttonDelete, fixedSpaceMini, self.buttonAction, nil]];
  215. else
  216. [self.toolbar setItems:[NSArray arrayWithObjects: flexible, buttonDelete, fixedSpaceMini, buttonShare, fixedSpaceMini, self.buttonAction, nil]];
  217. }
  218. [self.toolbar setAutoresizingMask:UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleTopMargin];
  219. self.toolbar.barTintColor = [NCBrandColor sharedInstance].tabBar;
  220. self.toolbar.tintColor = [NCBrandColor sharedInstance].brandElement;
  221. [self.view addSubview:self.toolbar];
  222. }
  223. #pragma --------------------------------------------------------------------------------------------
  224. #pragma mark ===== View Image =====
  225. #pragma --------------------------------------------------------------------------------------------
  226. - (void)viewImage
  227. {
  228. self.photoBrowser = [[MWPhotoBrowser alloc] initWithDelegate:self];
  229. indexNowVisible = -1;
  230. fileIDNowVisible = nil;
  231. [self.photos removeAllObjects];
  232. [dataSourceDirectoryID removeAllObjects];
  233. // if not images, exit
  234. if ([self.photoDataSource count] == 0)
  235. return;
  236. // test
  237. NSString *serverUrl = [[NCManageDatabase sharedInstance] getServerUrl:_metadataDetail.directoryID];
  238. if (!serverUrl)
  239. return;
  240. NSUInteger index = 0;
  241. for (tableMetadata *metadata in self.photoDataSource) {
  242. // start from here ?
  243. if (self.metadataDetail.fileID && [metadata.fileID isEqualToString:self.metadataDetail.fileID])
  244. [self.photoBrowser setCurrentPhotoIndex:index];
  245. [self.photos addObject:[MWPhoto photoWithImage:nil]];
  246. // add directory
  247. [dataSourceDirectoryID addObject:metadata.directoryID];
  248. index++;
  249. }
  250. // PhotoBrowser
  251. if ([NCBrandOptions sharedInstance].disable_openin_file) {
  252. self.photoBrowser.displayActionButton = NO;
  253. } else {
  254. self.photoBrowser.displayActionButton = YES;
  255. }
  256. self.photoBrowser.displayDeleteButton = YES;
  257. if ([CCUtility isFolderEncrypted:serverUrl account:appDelegate.activeAccount]) // E2EE
  258. self.photoBrowser.displayShareButton = NO;
  259. else
  260. self.photoBrowser.displayShareButton = YES;
  261. self.photoBrowser.displayNavArrows = YES;
  262. self.photoBrowser.displaySelectionButtons = NO;
  263. self.photoBrowser.alwaysShowControls = NO;
  264. self.photoBrowser.zoomPhotosToFill = NO;
  265. self.photoBrowser.autoPlayOnAppear = NO;
  266. self.photoBrowser.delayToHideElements = 15;
  267. if ([UIDevice currentDevice].userInterfaceIdiom == UIUserInterfaceIdiomPad)
  268. self.photoBrowser.enableSwipeToDismiss = NO;
  269. if (self.traitCollection.horizontalSizeClass == UIUserInterfaceSizeClassCompact) {
  270. [self addChildViewController:self.photoBrowser];
  271. [self.view addSubview:self.photoBrowser.view];
  272. [self.photoBrowser didMoveToParentViewController:self];
  273. } else {
  274. [self.navigationController pushViewController:self.photoBrowser animated:NO];
  275. }
  276. }
  277. - (NSUInteger)numberOfPhotosInPhotoBrowser:(MWPhotoBrowser *)photoBrowser
  278. {
  279. return [self.photoDataSource count];
  280. }
  281. - (NSString *)photoBrowser:(MWPhotoBrowser *)photoBrowser titleForPhotoAtIndex:(NSUInteger)index
  282. {
  283. tableMetadata *metadata = [self.photoDataSource objectAtIndex:index];
  284. NSString *titleDir = metadata.fileNameView;
  285. self.title = titleDir;
  286. return titleDir;
  287. }
  288. - (void)photoBrowser:(MWPhotoBrowser *)photoBrowser didDisplayPhotoAtIndex:(NSUInteger)index
  289. {
  290. tableMetadata *metadata = [self.photoDataSource objectAtIndex:index];
  291. indexNowVisible = index;
  292. fileIDNowVisible = metadata.fileID;
  293. photoBrowser.toolbar.hidden = NO;
  294. // Download image ?
  295. if (metadata) {
  296. NSString *serverUrl = [[NCManageDatabase sharedInstance] getServerUrl:metadata.directoryID];
  297. NSInteger status;
  298. tableMetadata *metadataDB = [[NCManageDatabase sharedInstance] getMetadataWithPredicate:[NSPredicate predicateWithFormat:@"fileID == %@", metadata.fileID]];
  299. if (metadataDB) {
  300. status = metadataDB.status;
  301. } else {
  302. status = k_metadataStatusNormal;
  303. }
  304. if ([CCUtility fileProviderStorageExists:metadata.fileID fileNameView:metadata.fileNameView] == NO && status == k_metadataStatusNormal) {
  305. if ([[NSFileManager defaultManager] fileExistsAtPath:[CCUtility getDirectoryProviderStorageIconFileID:metadata.fileID fileNameView:metadata.fileNameView]] == NO) {
  306. [CCGraphics addImageToTitle:NSLocalizedString(@"_...loading..._", nil) colorTitle:[NCBrandColor sharedInstance].brandText imageTitle:[CCGraphics changeThemingColorImage:[UIImage imageNamed:@"load"] multiplier:2 color:[NCBrandColor sharedInstance].brandText] imageRight:NO navigationItem:self.navigationItem];
  307. CGFloat width = [[NCUtility sharedInstance] getScreenWidthForPreview];
  308. CGFloat height = [[NCUtility sharedInstance] getScreenHeightForPreview];
  309. OCnetworking *ocNetworking = [[OCnetworking alloc] initWithDelegate:nil metadataNet:nil withUser:appDelegate.activeUser withUserID:appDelegate.activeUserID withPassword:appDelegate.activePassword withUrl:appDelegate.activeUrl];
  310. [ocNetworking downloadPreviewWithMetadata:metadata serverUrl:serverUrl withWidth:width andHeight:height completion:^(NSString *message, NSInteger errorCode) {
  311. self.navigationItem.titleView = nil;
  312. self.title = metadata.fileNameView;
  313. [self.photoBrowser reloadData];
  314. }];
  315. } else {
  316. [self downloadPhotoBrowser:metadata serverUrl:serverUrl];
  317. }
  318. }
  319. }
  320. // Title
  321. if (metadata)
  322. self.title = metadata.fileNameView;
  323. }
  324. - (id <MWPhoto>)photoBrowser:(MWPhotoBrowser *)photoBrowser photoAtIndex:(NSUInteger)index
  325. {
  326. UIImage *image;
  327. tableMetadata *metadata = [self.photoDataSource objectAtIndex:index];
  328. if (index < self.photos.count) {
  329. if (metadata.fileID) {
  330. UIImage *imagePreview = [UIImage imageWithContentsOfFile:[CCUtility getDirectoryProviderStorageIconFileID:metadata.fileID fileNameView:metadata.fileNameView]];
  331. // if (!imagePreview) imagePreview = [CCGraphics changeThemingColorImage:[UIImage imageNamed:@"file_photo"] multiplier:3 color:[[NCBrandColor sharedInstance] icon]];
  332. if ([metadata.typeFile isEqualToString: k_metadataTypeFile_image]) {
  333. NSString *fileImage = [CCUtility getDirectoryProviderStorageFileID:metadata.fileID fileNameView:metadata.fileNameView];
  334. NSString *ext = [CCUtility getExtension:metadata.fileNameView];
  335. if ([ext isEqualToString:@"GIF"]) image = [UIImage animatedImageWithAnimatedGIFURL:[NSURL fileURLWithPath:fileImage]];
  336. else image = [UIImage imageWithContentsOfFile:fileImage];
  337. if (image) {
  338. MWPhoto *photo = [MWPhoto photoWithImage:image];
  339. // Location ??
  340. [self setLocationCaptionPhoto:photo fileID:metadata.fileID];
  341. [self.photos replaceObjectAtIndex:index withObject:photo];
  342. } else {
  343. if (metadata.status == k_metadataStatusDownloadError) {
  344. [self.photos replaceObjectAtIndex:index withObject:[MWPhoto photoWithImage:[UIImage imageNamed:@"filePreviewError"]]];
  345. } else {
  346. if (imagePreview)
  347. [self.photos replaceObjectAtIndex:index withObject:[MWPhoto photoWithImage:imagePreview]];
  348. }
  349. }
  350. }
  351. if ([metadata.typeFile isEqualToString: k_metadataTypeFile_video]) {
  352. if ([CCUtility fileProviderStorageExists:metadata.fileID fileNameView:metadata.fileNameView]) {
  353. NSURL *url = [NSURL fileURLWithPath:[CCUtility getDirectoryProviderStorageFileID:metadata.fileID fileNameView:metadata.fileNameView]];
  354. MWPhoto *video = [MWPhoto photoWithImage:[CCGraphics thumbnailImageForVideo:url atTime:1.0]];
  355. video.videoURL = url;
  356. [self.photos replaceObjectAtIndex:index withObject:video];
  357. } else {
  358. if (metadata.status == k_metadataStatusDownloadError) {
  359. [self.photos replaceObjectAtIndex:index withObject:[MWPhoto photoWithImage:[UIImage imageNamed:@"filePreviewError"]]];
  360. } else {
  361. if (imagePreview)
  362. [self.photos replaceObjectAtIndex:index withObject:[MWPhoto photoWithImage:imagePreview]];
  363. }
  364. }
  365. }
  366. if ([metadata.typeFile isEqualToString: k_metadataTypeFile_audio]) {
  367. if ([CCUtility fileProviderStorageExists:metadata.fileID fileNameView:metadata.fileNameView]) {
  368. MWPhoto *audio;
  369. UIImage *audioImage;
  370. NSURL *url = [NSURL fileURLWithPath:[CCUtility getDirectoryProviderStorageFileID:metadata.fileID fileNameView:metadata.fileNameView]];
  371. if ([[NSFileManager defaultManager] fileExistsAtPath:[CCUtility getDirectoryProviderStorageIconFileID:metadata.fileID fileNameView:metadata.fileNameView]]) {
  372. audioImage = [UIImage imageWithContentsOfFile:[CCUtility getDirectoryProviderStorageIconFileID:metadata.fileID fileNameView:metadata.fileNameView]];
  373. } else {
  374. audioImage = [UIImage imageNamed:@"notaMusic"]; //[CCGraphics scaleImage:[UIImage imageNamed:@"notaMusic"] toSize:CGSizeMake(200, 200) isAspectRation:YES];
  375. }
  376. audio = [MWPhoto photoWithImage:audioImage];
  377. audio.videoURL = url;
  378. [self.photos replaceObjectAtIndex:index withObject:audio];
  379. } else {
  380. if (metadata.status == k_metadataStatusDownloadError) {
  381. [self.photos replaceObjectAtIndex:index withObject:[MWPhoto photoWithImage:[UIImage imageNamed:@"filePreviewError"]]];
  382. } else {
  383. if (imagePreview)
  384. [self.photos replaceObjectAtIndex:index withObject:[MWPhoto photoWithImage:imagePreview]];
  385. }
  386. }
  387. }
  388. }
  389. // energy saving memory
  390. dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
  391. int iPrev = (int)index - 2;
  392. if (iPrev >= 0) {
  393. if ([self.photos objectAtIndex:iPrev] != nil)
  394. [self.photos replaceObjectAtIndex:iPrev withObject:[MWPhoto photoWithImage:nil]];
  395. }
  396. int iNext = (int)index + 2;
  397. if (iNext < _photos.count) {
  398. if ([self.photos objectAtIndex:iNext] != nil)
  399. [self.photos replaceObjectAtIndex:iNext withObject:[MWPhoto photoWithImage:nil]];
  400. }
  401. });
  402. return [self.photos objectAtIndex:index];
  403. }
  404. return nil;
  405. }
  406. - (void)photoBrowser:(MWPhotoBrowser *)photoBrowser actionButtonPressedForPhotoAtIndex:(NSUInteger)index
  407. {
  408. tableMetadata *metadata = [self.photoDataSource objectAtIndex:index];
  409. if (metadata == nil) return;
  410. docController = [UIDocumentInteractionController interactionControllerWithURL:[NSURL fileURLWithPath:[CCUtility getDirectoryProviderStorageFileID:metadata.fileID fileNameView:metadata.fileNameView]]];
  411. docController.delegate = self;
  412. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone)
  413. [docController presentOptionsMenuFromRect:photoBrowser.view.frame inView:photoBrowser.view animated:YES];
  414. [docController presentOptionsMenuFromBarButtonItem:photoBrowser.actionButton animated:YES];
  415. }
  416. - (void)photoBrowser:(MWPhotoBrowser *)photoBrowser shareButtonPressedForPhotoAtIndex:(NSUInteger)index
  417. {
  418. tableMetadata *metadata = [self.photoDataSource objectAtIndex:index];
  419. [appDelegate.activeMain openWindowShare:metadata];
  420. }
  421. - (void)photoBrowser:(MWPhotoBrowser *)photoBrowser deleteButtonPressedForPhotoAtIndex:(NSUInteger)index deleteButton:(UIBarButtonItem *)deleteButton
  422. {
  423. tableMetadata *metadata = [self.photoDataSource objectAtIndex:index];
  424. if (metadata == nil || [CCUtility fileProviderStorageExists:metadata.fileID fileNameView:metadata.fileNameView] == NO) {
  425. [appDelegate messageNotification:@"_info_" description:@"_file_not_found_" visible:YES delay:k_dismissAfterSecond type:TWMessageBarMessageTypeInfo errorCode:0];
  426. return;
  427. }
  428. UIAlertController *alertController = [UIAlertController alertControllerWithTitle:nil message:nil preferredStyle:UIAlertControllerStyleActionSheet];
  429. [alertController addAction: [UIAlertAction actionWithTitle:NSLocalizedString(@"_delete_", nil)
  430. style:UIAlertActionStyleDestructive
  431. handler:^(UIAlertAction *action) {
  432. [self deleteFile:metadata];
  433. }]];
  434. [alertController addAction: [UIAlertAction actionWithTitle:NSLocalizedString(@"_cancel_", nil)
  435. style:UIAlertActionStyleCancel
  436. handler:^(UIAlertAction *action) {
  437. }]];
  438. alertController.popoverPresentationController.barButtonItem = deleteButton;
  439. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad)
  440. [alertController.view layoutIfNeeded];
  441. [self.parentViewController presentViewController:alertController animated:YES completion:NULL];
  442. }
  443. - (void)photoBrowserDidFinishPresentation:(MWPhotoBrowser *)photoBrowser
  444. {
  445. [self.navigationController popViewControllerAnimated:YES];
  446. }
  447. - (void)triggerProgressTask:(NSNotification *)notification
  448. {
  449. /*
  450. NSDictionary *dict = notification.userInfo;
  451. NSString *fileID = [dict valueForKey:@"fileID"];
  452. //NSString *serverUrl = [dict valueForKey:@"serverUrl"];
  453. //long status = [[dict valueForKey:@"status"] longValue];
  454. float progress = [[dict valueForKey:@"progress"] floatValue];
  455. //long long totalBytes = [[dict valueForKey:@"totalBytes"] longLongValue];
  456. //long long totalBytesExpected = [[dict valueForKey:@"totalBytesExpected"] longLongValue];
  457. if ([fileID isEqualToString:_fileIDNowVisible])
  458. [_hud progress:progress];
  459. */
  460. }
  461. - (void)downloadPhotoBrowserSuccessFailure:(tableMetadata *)metadata selector:(NSString *)selector errorCode:(NSInteger)errorCode
  462. {
  463. // if a message for a directory of these
  464. if (![metadata.fileID isEqualToString:fileIDNowVisible])
  465. return;
  466. // Title
  467. self.navigationItem.titleView = nil;
  468. self.title = metadata.fileNameView;
  469. if (errorCode == 0) {
  470. // verifico se esiste l'icona e se la posso creare
  471. if ([[NSFileManager defaultManager] fileExistsAtPath:[CCUtility getDirectoryProviderStorageIconFileID:metadata.fileID fileNameView:metadata.fileNameView]] == NO) {
  472. [CCGraphics createNewImageFrom:metadata.fileNameView fileID:metadata.fileID extension:[metadata.fileNameView pathExtension] filterGrayScale:NO typeFile:metadata.typeFile writeImage:YES];
  473. }
  474. [self.photoBrowser reloadData];
  475. } else {
  476. [appDelegate messageNotification:@"_download_selected_files_" description:@"_error_download_photobrowser_" visible:YES delay:k_dismissAfterSecond type:TWMessageBarMessageTypeError errorCode:errorCode];
  477. [self.navigationController popViewControllerAnimated:YES];
  478. }
  479. }
  480. - (void)downloadPhotoBrowser:(tableMetadata *)metadata serverUrl:(NSString *)serverUrl
  481. {
  482. tableMetadata *metadataForDownload = [[NCManageDatabase sharedInstance] initNewMetadata:metadata];
  483. metadataForDownload.session = k_download_session;
  484. metadataForDownload.sessionError = @"";
  485. metadataForDownload.sessionSelector = selectorLoadViewImage;
  486. metadataForDownload.status = k_metadataStatusWaitDownload;
  487. // Add Metadata for Download
  488. (void)[[NCManageDatabase sharedInstance] addMetadata:metadataForDownload];
  489. [appDelegate performSelectorOnMainThread:@selector(loadAutoDownloadUpload) withObject:nil waitUntilDone:YES];
  490. [[NCMainCommon sharedInstance] reloadDatasourceWithServerUrl:serverUrl fileID:metadataForDownload.fileID action:k_action_MOD];
  491. [CCGraphics addImageToTitle:NSLocalizedString(@"_...loading..._", nil) colorTitle:[NCBrandColor sharedInstance].brandText imageTitle:[CCGraphics changeThemingColorImage:[UIImage imageNamed:@"load"] multiplier:2 color:[NCBrandColor sharedInstance].brandText] imageRight:NO navigationItem:self.navigationItem];
  492. }
  493. - (void)insertGeocoderLocation:(NSNotification *)notification
  494. {
  495. if (notification.userInfo.count == 0)
  496. return;
  497. NSString *fileID = [[notification.userInfo allKeys] objectAtIndex:0];
  498. //NSDate *date = [[notification.userInfo allValues] objectAtIndex:0];
  499. // test [Chrash V 1.14,15]
  500. if (indexNowVisible >= [self.photos count])
  501. return;
  502. if ([fileID isEqualToString:fileIDNowVisible]) {
  503. MWPhoto *photo = [self.photos objectAtIndex:indexNowVisible];
  504. [self setLocationCaptionPhoto:photo fileID:fileID];
  505. [self.photoBrowser reloadData];
  506. }
  507. }
  508. - (void)setLocationCaptionPhoto:(MWPhoto *)photo fileID:(NSString *)fileID
  509. {
  510. tableLocalFile *localFile;
  511. // read Geocoder
  512. localFile = [[NCManageDatabase sharedInstance] getTableLocalFileWithPredicate:[NSPredicate predicateWithFormat:@"fileID == %@", fileID]];
  513. if ([localFile.exifLatitude doubleValue] != 0 || [localFile.exifLongitude doubleValue] != 0) {
  514. // Fix BUG Geo latitude & longitude
  515. if ([localFile.exifLatitude doubleValue] == 9999 || [localFile.exifLongitude doubleValue] == 9999) {
  516. tableMetadata *metadata = [[NCManageDatabase sharedInstance] getMetadataWithPredicate:[NSPredicate predicateWithFormat:@"fileID == %@", fileID]];
  517. if (metadata) {
  518. [[CCExifGeo sharedInstance] setExifLocalTableEtag:metadata];
  519. }
  520. }
  521. [[CCExifGeo sharedInstance] setGeocoderEtag:fileID exifDate:localFile.exifDate latitude:localFile.exifLatitude longitude:localFile.exifLongitude];
  522. localFile = [[NCManageDatabase sharedInstance] getTableLocalFileWithPredicate:[NSPredicate predicateWithFormat:@"fileID == %@", fileID]];
  523. if ([localFile.exifLatitude floatValue] != 0 || [localFile.exifLongitude floatValue] != 0) {
  524. NSString *location = [[NCManageDatabase sharedInstance] getLocationFromGeoLatitude:localFile.exifLatitude longitude:localFile.exifLongitude];
  525. if ([localFile.exifDate isEqualToDate:[NSDate distantPast]] == NO && location) {
  526. NSString *localizedDateTime = [NSDateFormatter localizedStringFromDate:localFile.exifDate dateStyle:NSDateFormatterFullStyle timeStyle:NSDateFormatterMediumStyle];
  527. photo.caption = [NSString stringWithFormat:NSLocalizedString(@"%@\n%@", nil), localizedDateTime, location];
  528. }
  529. }
  530. }
  531. }
  532. #pragma --------------------------------------------------------------------------------------------
  533. #pragma mark ===== View PDF =====
  534. #pragma --------------------------------------------------------------------------------------------
  535. - (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex
  536. {
  537. [alertView dismissWithClickedButtonIndex:buttonIndex animated:YES];
  538. [[alertView textFieldAtIndex:0] resignFirstResponder];
  539. if (alertView.tag == alertRequestPasswordPDF) [self performSelector:@selector(viewPDF:) withObject:[alertView textFieldAtIndex:0].text afterDelay:0.3];
  540. }
  541. - (void)viewPDF:(NSString *)password
  542. {
  543. // remove cache PDF
  544. NSString *filePlistReader = [NSString stringWithFormat:@"%@/%@.plist", [CCUtility getDirectoryReaderMetadata], self.metadataDetail.fileNameView.stringByDeletingPathExtension];
  545. [CCUtility removeFileAtPath:filePlistReader];
  546. NSString *fileNamePath = [CCUtility getDirectoryProviderStorageFileID:self.metadataDetail.fileID fileNameView:self.metadataDetail.fileNameView];
  547. if ([CCUtility fileProviderStorageExists:self.metadataDetail.fileID fileNameView:self.metadataDetail.fileNameView] == NO) {
  548. // read file error
  549. UIAlertController *alertController = [UIAlertController alertControllerWithTitle:NSLocalizedString(@"_error_", nil) message:NSLocalizedString(@"_read_file_error_", nil) preferredStyle:UIAlertControllerStyleAlert];
  550. UIAlertAction *okAction = [UIAlertAction actionWithTitle:NSLocalizedString(@"_ok_", nil) style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {}];
  551. [alertController addAction:okAction];
  552. [self presentViewController:alertController animated:YES completion:nil];
  553. }
  554. CGPDFDocumentRef pdf = CGPDFDocumentCreateWithURL((CFURLRef)[NSURL fileURLWithPath:fileNamePath]);
  555. if (pdf) {
  556. // Encrypted
  557. if (CGPDFDocumentIsEncrypted(pdf) == YES) {
  558. // Try a blank password first, per Apple's Quartz PDF example
  559. if (CGPDFDocumentUnlockWithPassword(pdf, "") == YES) {
  560. // blank password
  561. [self readerPDF:fileNamePath password:@""];
  562. } else {
  563. if ([password length] == 0) {
  564. // password request
  565. UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:NSLocalizedString(@"_insert_password_pfd_",nil) message:nil delegate:self cancelButtonTitle:nil otherButtonTitles:NSLocalizedString(@"_ok_", nil), nil];
  566. [alertView setAlertViewStyle:UIAlertViewStylePlainTextInput];
  567. alertView.tag = alertRequestPasswordPDF;
  568. [alertView show];
  569. } else {
  570. const char *key = [password UTF8String];
  571. // failure
  572. if (CGPDFDocumentUnlockWithPassword(pdf, key) == NO) {
  573. UIAlertController *alertController = [UIAlertController alertControllerWithTitle:NSLocalizedString(@"_error_", nil) message:NSLocalizedString(@"_password_pdf_error_", nil) preferredStyle:UIAlertControllerStyleAlert];
  574. UIAlertAction *okAction = [UIAlertAction actionWithTitle:NSLocalizedString(@"_ok_", nil) style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {}];
  575. [alertController addAction:okAction];
  576. [self presentViewController:alertController animated:YES completion:nil];
  577. } else {
  578. // pdf with password
  579. [self readerPDF:fileNamePath password:password];
  580. }
  581. }
  582. }
  583. } else{
  584. // No password
  585. [self readerPDF:fileNamePath password:@""];
  586. }
  587. } else {
  588. // read file error
  589. UIAlertController *alertController = [UIAlertController alertControllerWithTitle:NSLocalizedString(@"_error_", nil) message:NSLocalizedString(@"_read_file_error_", nil) preferredStyle:UIAlertControllerStyleAlert];
  590. UIAlertAction *okAction = [UIAlertAction actionWithTitle:NSLocalizedString(@"_ok_", nil) style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {}];
  591. [alertController addAction:okAction];
  592. [self presentViewController:alertController animated:YES completion:nil];
  593. }
  594. }
  595. - (void)readerPDF:(NSString *)fileName password:(NSString *)password
  596. {
  597. ReaderDocument *documentPDF = [ReaderDocument withDocumentFilePath:fileName password:password];
  598. CGFloat safeAreaBottom = 0;
  599. if (@available(iOS 11, *)) {
  600. safeAreaBottom = [UIApplication sharedApplication].delegate.window.safeAreaInsets.bottom;
  601. }
  602. if (documentPDF != nil) {
  603. self.readerPDFViewController = [[ReaderViewController alloc] initWithReaderDocument:documentPDF];
  604. self.readerPDFViewController.delegate = self;
  605. self.readerPDFViewController.view.frame = CGRectMake(0, 0, self.view.bounds.size.width, self.view.bounds.size.height - TOOLBAR_HEIGHT - safeAreaBottom);
  606. [self.readerPDFViewController updateContentViews];
  607. [self addChildViewController:self.readerPDFViewController];
  608. [self.view addSubview:self.readerPDFViewController.view];
  609. [self.readerPDFViewController didMoveToParentViewController:self];
  610. } else {
  611. UIAlertController *alertController = [UIAlertController alertControllerWithTitle:NSLocalizedString(@"_error_", nil) message:NSLocalizedString(@"_read_file_error_", nil) preferredStyle:UIAlertControllerStyleAlert];
  612. UIAlertAction *okAction = [UIAlertAction actionWithTitle:NSLocalizedString(@"_ok_", nil) style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {}];
  613. [alertController addAction:okAction];
  614. [self presentViewController:alertController animated:YES completion:nil];
  615. }
  616. }
  617. - (void)handleSingleTapReader
  618. {
  619. UILayoutGuide *layoutGuide;
  620. CGFloat safeAreaTop = 0;
  621. CGFloat safeAreaBottom = 0;
  622. if (@available(iOS 11, *)) {
  623. layoutGuide = [UIApplication sharedApplication].delegate.window.safeAreaLayoutGuide;
  624. safeAreaTop = [UIApplication sharedApplication].delegate.window.safeAreaInsets.top;
  625. safeAreaBottom = [UIApplication sharedApplication].delegate.window.safeAreaInsets.bottom;
  626. }
  627. self.navigationController.navigationBarHidden = !self.navigationController.navigationBarHidden;
  628. self.toolbar.hidden = !self.toolbar.isHidden;
  629. if (self.toolbar.isHidden) {
  630. self.readerPDFViewController.view.frame = CGRectMake(0, safeAreaTop, self.view.bounds.size.width, self.view.bounds.size.height - safeAreaTop - safeAreaBottom);
  631. } else {
  632. self.readerPDFViewController.view.frame = CGRectMake(0, 0, self.view.bounds.size.width, self.view.bounds.size.height - TOOLBAR_HEIGHT - safeAreaBottom);
  633. }
  634. [self.readerPDFViewController updateContentViews];
  635. }
  636. - (void)handleSwipeUpDown
  637. {
  638. // REMOVE IT'S UNUSABLE
  639. /*
  640. self.navigationController.navigationBarHidden = false; // iOS App is unusable after swipe up or down with PDF in fullscreen #526
  641. [self removeAllView];
  642. [self.navigationController popViewControllerAnimated:YES];
  643. */
  644. }
  645. #pragma --------------------------------------------------------------------------------------------
  646. #pragma mark ===== Delete =====
  647. #pragma --------------------------------------------------------------------------------------------
  648. - (void)deleteFile:(tableMetadata *)metadata
  649. {
  650. NSString *serverUrl = [[NCManageDatabase sharedInstance] getServerUrl:metadata.directoryID];
  651. tableDirectory *tableDirectory = [[NCManageDatabase sharedInstance] getTableDirectoryWithPredicate:[NSPredicate predicateWithFormat:@"account == %@ AND e2eEncrypted == 1 AND serverUrl == %@", appDelegate.activeAccount, serverUrl]];
  652. [[NCMainCommon sharedInstance ] deleteFileWithMetadatas:[[NSArray alloc] initWithObjects:metadata, nil] e2ee:tableDirectory.e2eEncrypted serverUrl:serverUrl folderFileID:tableDirectory.fileID completion:^(NSInteger errorCode, NSString *message) {
  653. if (errorCode == 0) {
  654. // reload data source
  655. [[NCMainCommon sharedInstance] reloadDatasourceWithServerUrl:serverUrl fileID:metadata.fileID action:k_action_DEL];
  656. // Not image
  657. if ([self.metadataDetail.typeFile isEqualToString: k_metadataTypeFile_image] == NO) {
  658. // exit
  659. [self.navigationController popViewControllerAnimated:YES];
  660. } else {
  661. for (NSUInteger index=0; index < [self.photoDataSource count] && _photoBrowser; index++ ) {
  662. tableMetadata *metadataTemp = [self.photoDataSource objectAtIndex:index];
  663. if ([metadata isInvalidated] || [metadataTemp.fileID isEqualToString:metadata.fileID]) {
  664. [self.photoDataSource removeObjectAtIndex:index];
  665. [self.photos removeObjectAtIndex:index];
  666. [self.photoBrowser reloadData];
  667. // exit
  668. if ([self.photoDataSource count] == 0) {
  669. [self.navigationController popViewControllerAnimated:YES];
  670. }
  671. }
  672. }
  673. }
  674. } else {
  675. NSLog(@"[LOG] DeleteFileOrFolder failure error %d, %@", (int)errorCode, message);
  676. }
  677. }];
  678. }
  679. #pragma --------------------------------------------------------------------------------------------
  680. #pragma mark ===== ButtonPressed =====
  681. #pragma --------------------------------------------------------------------------------------------
  682. - (void)dismissTextView
  683. {
  684. if (self.webView) {
  685. NSString *fileNamePath = [NSTemporaryDirectory() stringByAppendingString:self.metadataDetail.fileNameView];
  686. [[NSFileManager defaultManager] removeItemAtPath:fileNamePath error:nil];
  687. [[NSFileManager defaultManager] linkItemAtPath:[CCUtility getDirectoryProviderStorageFileID:self.metadataDetail.fileID fileNameView:self.metadataDetail.fileNameView] toPath:fileNamePath error:nil];
  688. [self.webView reload];
  689. }
  690. }
  691. - (void)modifyTxtButtonPressed:(UIBarButtonItem *)sender
  692. {
  693. tableMetadata *metadata = [[NCManageDatabase sharedInstance] getMetadataWithPredicate:[NSPredicate predicateWithFormat:@"fileID == %@", self.metadataDetail.fileID]];
  694. if (metadata) {
  695. UINavigationController* navigationController = [[UIStoryboard storyboardWithName:@"NCText" bundle:nil] instantiateViewControllerWithIdentifier:@"NCText"];
  696. NCText *viewController = (NCText *)navigationController.topViewController;
  697. viewController.metadata = metadata;
  698. viewController.delegate = self;
  699. navigationController.modalPresentationStyle = UIModalPresentationPageSheet;
  700. navigationController.modalTransitionStyle = UIModalTransitionStyleCrossDissolve;
  701. [self presentViewController:navigationController animated:YES completion:nil];
  702. }
  703. }
  704. - (void)actionButtonPressed:(UIBarButtonItem *)sender
  705. {
  706. if ([self.metadataDetail.fileNameView length] == 0) return;
  707. NSString *filePath = [CCUtility getDirectoryProviderStorageFileID:self.metadataDetail.fileID fileNameView:self.metadataDetail.fileNameView];
  708. docController = [UIDocumentInteractionController interactionControllerWithURL:[NSURL fileURLWithPath:filePath]];
  709. docController.delegate = self;
  710. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone)
  711. [docController presentOptionsMenuFromRect:self.view.frame inView:self.view animated:YES];
  712. [docController presentOptionsMenuFromBarButtonItem:sender animated:YES];
  713. }
  714. - (void)shareButtonPressed:(UIBarButtonItem *)sender
  715. {
  716. [appDelegate.activeMain openWindowShare:self.metadataDetail];
  717. }
  718. - (void)deleteButtonPressed:(UIBarButtonItem *)sender
  719. {
  720. if ([self.metadataDetail.fileNameView length] == 0) return;
  721. UIAlertController *alertController = [UIAlertController alertControllerWithTitle:nil message:nil preferredStyle:UIAlertControllerStyleActionSheet];
  722. [alertController addAction: [UIAlertAction actionWithTitle:NSLocalizedString(@"_delete_", nil)
  723. style:UIAlertActionStyleDestructive
  724. handler:^(UIAlertAction *action) {
  725. [self deleteFile:self.metadataDetail];
  726. }]];
  727. [alertController addAction: [UIAlertAction actionWithTitle:NSLocalizedString(@"_cancel_", nil)
  728. style:UIAlertActionStyleCancel
  729. handler:^(UIAlertAction *action) {
  730. [alertController dismissViewControllerAnimated:YES completion:nil];
  731. }]];
  732. alertController.popoverPresentationController.barButtonItem = buttonDelete;
  733. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad)
  734. [alertController.view layoutIfNeeded];
  735. [self presentViewController:alertController animated:YES completion:NULL];
  736. }
  737. @end