CTAssetsPickerController.m 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621
  1. /*
  2. MIT License (MIT)
  3. Copyright (c) 2015 Clement CN Tsang
  4. Permission is hereby granted, free of charge, to any person obtaining a copy
  5. of this software and associated documentation files (the "Software"), to deal
  6. in the Software without restriction, including without limitation the rights
  7. to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  8. copies of the Software, and to permit persons to whom the Software is
  9. furnished to do so, subject to the following conditions:
  10. The above copyright notice and this permission notice shall be included in
  11. all copies or substantial portions of the Software.
  12. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  13. IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  14. FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  15. AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  16. LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  17. OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  18. THE SOFTWARE.
  19. */
  20. #import "CTAssetsPickerDefines.h"
  21. #import "CTAssetsPickerController.h"
  22. #import "CTAssetsPickerAccessDeniedView.h"
  23. #import "CTAssetsPickerNoAssetsView.h"
  24. #import "CTAssetCollectionViewController.h"
  25. #import "CTAssetsGridViewController.h"
  26. #import "CTAssetScrollView.h"
  27. #import "CTAssetsPageViewController.h"
  28. #import "CTAssetsViewControllerTransition.h"
  29. #import "NSBundle+CTAssetsPickerController.h"
  30. #import "UIImage+CTAssetsPickerController.h"
  31. #import "NSNumberFormatter+CTAssetsPickerController.h"
  32. #import "CTAssetsNavigationController.h"
  33. NSString * const CTAssetsPickerSelectedAssetsDidChangeNotification = @"CTAssetsPickerSelectedAssetsDidChangeNotification";
  34. NSString * const CTAssetsPickerDidSelectAssetNotification = @"CTAssetsPickerDidSelectAssetNotification";
  35. NSString * const CTAssetsPickerDidDeselectAssetNotification = @"CTAssetsPickerDidDeselectAssetNotification";
  36. @interface CTAssetsPickerController ()
  37. <PHPhotoLibraryChangeObserver, UISplitViewControllerDelegate, UINavigationControllerDelegate>
  38. @property (nonatomic, assign) BOOL shouldCollapseDetailViewController;
  39. @property (nonatomic, assign) CGSize assetCollectionThumbnailSize;
  40. @property (nonatomic, assign) CGSize assetThumbnailSize;
  41. @property (nonatomic, strong) PHImageRequestOptions *thumbnailRequestOptions;
  42. @end
  43. @implementation CTAssetsPickerController
  44. - (instancetype)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
  45. {
  46. if (self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil])
  47. {
  48. _shouldCollapseDetailViewController = YES;
  49. _assetCollectionThumbnailSize = CTAssetCollectionThumbnailSize;
  50. _assetCollectionFetchOptions = nil;
  51. _assetsFetchOptions = nil;
  52. _selectedAssets = [NSMutableArray new];
  53. _showsCancelButton = YES;
  54. _showsEmptyAlbums = YES;
  55. _showsNumberOfAssets = YES;
  56. _alwaysEnableDoneButton = NO;
  57. _showsSelectionIndex = NO;
  58. _defaultAssetCollection = PHAssetCollectionSubtypeAny;
  59. [self initAssetCollectionSubtypes];
  60. [self initThumbnailRequestOptions];
  61. self.preferredContentSize = CTAssetsPickerPopoverContentSize;
  62. }
  63. return self;
  64. }
  65. - (void)viewDidLoad
  66. {
  67. [super viewDidLoad];
  68. [self setupViews];
  69. [self setupEmptyViewController];
  70. [self checkAuthorizationStatus];
  71. [self addKeyValueObserver];
  72. [self registerChangeObserver];
  73. }
  74. - (void)dealloc
  75. {
  76. [self removeKeyValueObserver];
  77. [self unregisterChangeObserver];
  78. }
  79. - (UIViewController *)childViewControllerForStatusBarStyle
  80. {
  81. return self.childSplitViewController.viewControllers.firstObject;
  82. }
  83. - (UIViewController *)childViewControllerForStatusBarHidden
  84. {
  85. UIViewController *vc = self.childSplitViewController.viewControllers.lastObject;
  86. if ([vc isMemberOfClass:[UINavigationController class]])
  87. return ((UINavigationController *)vc).topViewController;
  88. else
  89. return nil;
  90. }
  91. #pragma mark - Init properties
  92. - (void)initAssetCollectionSubtypes
  93. {
  94. _assetCollectionSubtypes =
  95. @[@(PHAssetCollectionSubtypeSmartAlbumUserLibrary),
  96. @(PHAssetCollectionSubtypeAlbumMyPhotoStream),
  97. @(PHAssetCollectionSubtypeSmartAlbumRecentlyAdded),
  98. @(PHAssetCollectionSubtypeSmartAlbumFavorites),
  99. @(PHAssetCollectionSubtypeSmartAlbumPanoramas),
  100. @(PHAssetCollectionSubtypeSmartAlbumVideos),
  101. @(PHAssetCollectionSubtypeSmartAlbumSlomoVideos),
  102. @(PHAssetCollectionSubtypeSmartAlbumTimelapses),
  103. @(PHAssetCollectionSubtypeSmartAlbumBursts),
  104. @(PHAssetCollectionSubtypeSmartAlbumAllHidden),
  105. @(PHAssetCollectionSubtypeSmartAlbumGeneric),
  106. @(PHAssetCollectionSubtypeAlbumRegular),
  107. @(PHAssetCollectionSubtypeAlbumSyncedAlbum),
  108. @(PHAssetCollectionSubtypeAlbumSyncedEvent),
  109. @(PHAssetCollectionSubtypeAlbumSyncedFaces),
  110. @(PHAssetCollectionSubtypeAlbumImported),
  111. @(PHAssetCollectionSubtypeAlbumCloudShared)];
  112. // Add iOS 9's new albums
  113. if ([[PHAsset new] respondsToSelector:@selector(sourceType)])
  114. {
  115. NSMutableArray *subtypes = [NSMutableArray arrayWithArray:self.assetCollectionSubtypes];
  116. [subtypes insertObject:@(PHAssetCollectionSubtypeSmartAlbumSelfPortraits) atIndex:4];
  117. [subtypes insertObject:@(PHAssetCollectionSubtypeSmartAlbumScreenshots) atIndex:10];
  118. self.assetCollectionSubtypes = [NSArray arrayWithArray:subtypes];
  119. }
  120. }
  121. - (void)initThumbnailRequestOptions
  122. {
  123. PHImageRequestOptions *options = [[PHImageRequestOptions alloc] init];
  124. options.resizeMode = PHImageRequestOptionsResizeModeFast;
  125. options.deliveryMode = PHImageRequestOptionsDeliveryModeHighQualityFormat;
  126. _thumbnailRequestOptions = options;
  127. }
  128. #pragma mark - Check authorization status
  129. - (void)checkAuthorizationStatus
  130. {
  131. PHAuthorizationStatus status = [PHPhotoLibrary authorizationStatus];
  132. switch (status)
  133. {
  134. case PHAuthorizationStatusNotDetermined:
  135. [self requestAuthorizationStatus];
  136. break;
  137. case PHAuthorizationStatusRestricted:
  138. case PHAuthorizationStatusDenied:
  139. {
  140. [self showAccessDenied];
  141. break;
  142. }
  143. case PHAuthorizationStatusAuthorized:
  144. default:
  145. {
  146. [self checkAssetsCount];
  147. break;
  148. }
  149. }
  150. }
  151. #pragma mark - Request authorization status
  152. - (void)requestAuthorizationStatus
  153. {
  154. [PHPhotoLibrary requestAuthorization:^(PHAuthorizationStatus status){
  155. switch (status) {
  156. case PHAuthorizationStatusAuthorized:
  157. {
  158. dispatch_async(dispatch_get_main_queue(), ^{
  159. [self checkAssetsCount];
  160. });
  161. break;
  162. }
  163. default:
  164. {
  165. dispatch_async(dispatch_get_main_queue(), ^{
  166. [self showAccessDenied];
  167. });
  168. break;
  169. }
  170. }
  171. }];
  172. }
  173. #pragma mark - Check assets count
  174. - (void)checkAssetsCount
  175. {
  176. PHFetchResult *fetchResult = [PHAsset fetchAssetsWithOptions:self.assetsFetchOptions];
  177. if (fetchResult.count > 0) {
  178. [self showAssetCollectionViewController];
  179. } else {
  180. [self showNoAssets];
  181. }
  182. }
  183. #pragma mark - Setup views
  184. - (void)setupViews
  185. {
  186. self.view.backgroundColor = [UIColor whiteColor];
  187. }
  188. #pragma mark - Setup view controllers
  189. - (void)setupEmptyViewController
  190. {
  191. UINavigationController *nav = [self emptyNavigationController];
  192. [self setupChildViewController:nav];
  193. }
  194. - (void)setupSplitViewController
  195. {
  196. CTAssetCollectionViewController *vc = [CTAssetCollectionViewController new];
  197. CTAssetsNavigationController *master = [[CTAssetsNavigationController alloc] initWithRootViewController:vc];
  198. UINavigationController *detail = [self emptyNavigationController];
  199. UISplitViewController *svc = [UISplitViewController new];
  200. svc.delegate = self;
  201. svc.viewControllers = @[master, detail];
  202. svc.presentsWithGesture = NO;
  203. svc.preferredDisplayMode = UISplitViewControllerDisplayModeAllVisible;
  204. [self addChildViewController:svc];
  205. svc.view.frame = CGRectMake(0, 0, self.view.frame.size.width, self.view.frame.size.height);
  206. [self.view addSubview:svc.view];
  207. [svc didMoveToParentViewController:self];
  208. [vc reloadUserInterface];
  209. }
  210. - (void)setupChildViewController:(UIViewController *)vc
  211. {
  212. [self addChildViewController:vc];
  213. vc.view.frame = CGRectMake(0, 0, self.view.frame.size.width, self.view.frame.size.height);
  214. [self.view addSubview:vc.view];
  215. [vc didMoveToParentViewController:self];
  216. }
  217. - (void)removeChildViewController
  218. {
  219. UIViewController *vc = self.childViewControllers.firstObject;
  220. [vc willMoveToParentViewController:nil];
  221. [vc.view removeFromSuperview];
  222. [vc removeFromParentViewController];
  223. }
  224. #pragma mark - Setup view controllers
  225. - (UINavigationController *)emptyNavigationController
  226. {
  227. UIViewController *vc = [self emptyViewController];
  228. return [[UINavigationController alloc] initWithRootViewController:vc];
  229. }
  230. - (UIViewController *)emptyViewController
  231. {
  232. UIViewController *vc = [UIViewController new];
  233. vc.view.backgroundColor = [UIColor whiteColor];
  234. vc.navigationItem.hidesBackButton = YES;
  235. return vc;
  236. }
  237. #pragma mark - Show asset collection view controller
  238. - (void)showAssetCollectionViewController
  239. {
  240. [self removeChildViewController];
  241. [self setupSplitViewController];
  242. }
  243. #pragma mark - Show auxiliary view
  244. - (void)showAuxiliaryView:(UIView *)view
  245. {
  246. [self removeChildViewController];
  247. UIViewController *vc = [self emptyViewController];
  248. UINavigationController *nav = [[UINavigationController alloc] initWithRootViewController:vc];
  249. [vc.view addSubview:view];
  250. [view setNeedsUpdateConstraints];
  251. [view updateConstraintsIfNeeded];
  252. [self setupButtonInViewController:vc];
  253. [self setupChildViewController:nav];
  254. }
  255. #pragma mark - Access denied
  256. - (void)showAccessDenied
  257. {
  258. [self showAuxiliaryView:[CTAssetsPickerAccessDeniedView new]];
  259. }
  260. #pragma mark - No Assets
  261. - (void)showNoAssets
  262. {
  263. [self showAuxiliaryView:[CTAssetsPickerNoAssetsView new]];
  264. }
  265. #pragma mark - Cancel button
  266. - (void)setupButtonInViewController:(UIViewController *)viewController
  267. {
  268. if (self.showsCancelButton)
  269. {
  270. viewController.navigationItem.leftBarButtonItem =
  271. [[UIBarButtonItem alloc] initWithTitle:CTAssetsPickerLocalizedString(@"Cancel", nil)
  272. style:UIBarButtonItemStylePlain
  273. target:self
  274. action:@selector(dismiss:)];
  275. }
  276. }
  277. #pragma mark - Key-Value observer
  278. - (void)addKeyValueObserver
  279. {
  280. [self addObserver:self
  281. forKeyPath:@"selectedAssets"
  282. options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld
  283. context:nil];
  284. }
  285. - (void)removeKeyValueObserver
  286. {
  287. @try {
  288. [self removeObserver:self forKeyPath:@"selectedAssets"];
  289. }
  290. @catch (NSException *exception) {
  291. // do nothing
  292. }
  293. }
  294. #pragma mark - Key-Value changed
  295. - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
  296. {
  297. if ([keyPath isEqual:@"selectedAssets"])
  298. {
  299. [self toggleDoneButton];
  300. [self postSelectedAssetsDidChangeNotification:[object valueForKey:keyPath]];
  301. }
  302. }
  303. #pragma mark - Photo library change observer
  304. - (void)registerChangeObserver
  305. {
  306. [[PHPhotoLibrary sharedPhotoLibrary] registerChangeObserver:self];
  307. }
  308. - (void)unregisterChangeObserver
  309. {
  310. [[PHPhotoLibrary sharedPhotoLibrary] unregisterChangeObserver:self];
  311. }
  312. #pragma mark - Photo library changed
  313. - (void)photoLibraryDidChange:(PHChange *)changeInstance
  314. {
  315. // Call might come on any background queue. Re-dispatch to the main queue to handle it.
  316. dispatch_async(dispatch_get_main_queue(), ^{
  317. NSMutableArray *deselectAssets = [NSMutableArray new];
  318. for (PHAsset *asset in self.selectedAssets)
  319. {
  320. PHObjectChangeDetails *changeDetails = [changeInstance changeDetailsForObject:asset];
  321. if (changeDetails.objectWasDeleted)
  322. [deselectAssets addObject:asset];
  323. }
  324. // Deselect asset if it was deleted from library
  325. for (PHAsset *asset in deselectAssets)
  326. [self deselectAsset:asset];
  327. });
  328. }
  329. #pragma mark - Toggle button
  330. - (void)toggleDoneButton
  331. {
  332. UIViewController *vc = self.childSplitViewController.viewControllers.firstObject;
  333. if ([vc isMemberOfClass:[UINavigationController class]])
  334. {
  335. BOOL enabled = (self.alwaysEnableDoneButton) ? YES : (self.selectedAssets.count > 0);
  336. for (UIViewController *viewController in ((UINavigationController *)vc).viewControllers)
  337. viewController.navigationItem.rightBarButtonItem.enabled = enabled;
  338. }
  339. }
  340. #pragma mark - Post notifications
  341. - (void)postSelectedAssetsDidChangeNotification:(id)sender
  342. {
  343. [[NSNotificationCenter defaultCenter] postNotificationName:CTAssetsPickerSelectedAssetsDidChangeNotification
  344. object:sender];
  345. }
  346. - (void)postDidSelectAssetNotification:(id)sender
  347. {
  348. [[NSNotificationCenter defaultCenter] postNotificationName:CTAssetsPickerDidSelectAssetNotification
  349. object:sender];
  350. }
  351. - (void)postDidDeselectAssetNotification:(id)sender
  352. {
  353. [[NSNotificationCenter defaultCenter] postNotificationName:CTAssetsPickerDidDeselectAssetNotification
  354. object:sender];
  355. }
  356. #pragma mark - Accessors
  357. - (UISplitViewController *)childSplitViewController
  358. {
  359. return (UISplitViewController *)self.childViewControllers.firstObject;
  360. }
  361. #pragma mark - Indexed accessors
  362. - (NSUInteger)countOfSelectedAssets
  363. {
  364. return self.selectedAssets.count;
  365. }
  366. - (instancetype)objectInSelectedAssetsAtIndex:(NSUInteger)index
  367. {
  368. return self.selectedAssets[index];
  369. }
  370. - (void)insertObject:(id)object inSelectedAssetsAtIndex:(NSUInteger)index
  371. {
  372. [self.selectedAssets insertObject:object atIndex:index];
  373. }
  374. - (void)removeObjectFromSelectedAssetsAtIndex:(NSUInteger)index
  375. {
  376. [self.selectedAssets removeObjectAtIndex:index];
  377. }
  378. - (void)replaceObjectInSelectedAssetsAtIndex:(NSUInteger)index withObject:(PHAsset *)object
  379. {
  380. self.selectedAssets[index] = object;
  381. }
  382. #pragma mark - De/Select asset
  383. - (void)selectAsset:(PHAsset *)asset
  384. {
  385. [self insertObject:asset inSelectedAssetsAtIndex:self.countOfSelectedAssets];
  386. [self postDidSelectAssetNotification:asset];
  387. }
  388. - (void)deselectAsset:(PHAsset *)asset
  389. {
  390. [self removeObjectFromSelectedAssetsAtIndex:[self.selectedAssets indexOfObject:asset]];
  391. [self postDidDeselectAssetNotification:asset];
  392. }
  393. #pragma mark - Selected assets string
  394. - (NSPredicate *)predicateOfMediaType:(PHAssetMediaType)type
  395. {
  396. return [NSPredicate predicateWithBlock:^BOOL(PHAsset *asset, NSDictionary *bindings) {
  397. return (asset.mediaType == type);
  398. }];
  399. }
  400. - (NSString *)selectedAssetsString
  401. {
  402. if (self.selectedAssets.count == 0)
  403. return nil;
  404. NSPredicate *photoPredicate = [self predicateOfMediaType:PHAssetMediaTypeImage];
  405. NSPredicate *videoPredicate = [self predicateOfMediaType:PHAssetMediaTypeVideo];
  406. BOOL photoSelected = ([self.selectedAssets filteredArrayUsingPredicate:photoPredicate].count > 0);
  407. BOOL videoSelected = ([self.selectedAssets filteredArrayUsingPredicate:videoPredicate].count > 0);
  408. NSString *format;
  409. if (photoSelected && videoSelected)
  410. format = CTAssetsPickerLocalizedString(@"%@ Items Selected", nil);
  411. else if (photoSelected)
  412. format = (self.selectedAssets.count > 1) ?
  413. CTAssetsPickerLocalizedString(@"%@ Photos Selected", nil) :
  414. CTAssetsPickerLocalizedString(@"%@ Photo Selected", nil);
  415. else if (videoSelected)
  416. format = (self.selectedAssets.count > 1) ?
  417. CTAssetsPickerLocalizedString(@"%@ Videos Selected", nil) :
  418. CTAssetsPickerLocalizedString(@"%@ Video Selected", nil);
  419. NSNumberFormatter *nf = [NSNumberFormatter new];
  420. return [NSString stringWithFormat:format, [nf ctassetsPickerStringFromAssetsCount:self.selectedAssets.count]];
  421. }
  422. #pragma mark - Image target size
  423. - (CGSize)imageSizeForContainerSize:(CGSize)size
  424. {
  425. CGFloat scale = UIScreen.mainScreen.scale;
  426. return CGSizeMake(size.width * scale, size.height * scale);
  427. }
  428. #pragma mark - Split view controller delegate
  429. - (BOOL)splitViewController:(UISplitViewController *)splitViewController collapseSecondaryViewController:(UIViewController *)secondaryViewController ontoPrimaryViewController:(UIViewController *)primaryViewController
  430. {
  431. return self.shouldCollapseDetailViewController;
  432. }
  433. #pragma mark - Navigation controller delegate
  434. - (id<UIViewControllerAnimatedTransitioning>)navigationController:(UINavigationController *)navigationController
  435. animationControllerForOperation:(UINavigationControllerOperation)operation
  436. fromViewController:(UIViewController *)fromVC
  437. toViewController:(UIViewController *)toVC
  438. {
  439. if ((operation == UINavigationControllerOperationPush && [toVC isKindOfClass:[CTAssetsPageViewController class]]) ||
  440. (operation == UINavigationControllerOperationPop && [fromVC isKindOfClass:[CTAssetsPageViewController class]]))
  441. {
  442. CTAssetsViewControllerTransition *transition = [[CTAssetsViewControllerTransition alloc] init];
  443. transition.operation = operation;
  444. return transition;
  445. }
  446. else
  447. {
  448. return nil;
  449. }
  450. }
  451. #pragma mark - Actions
  452. - (void)dismiss:(id)sender
  453. {
  454. if ([self.delegate respondsToSelector:@selector(assetsPickerControllerDidCancel:)])
  455. [self.delegate assetsPickerControllerDidCancel:self];
  456. else
  457. [self.presentingViewController dismissViewControllerAnimated:YES completion:nil];
  458. }
  459. - (void)finishPickingAssets:(id)sender
  460. {
  461. if ([self.delegate respondsToSelector:@selector(assetsPickerController:didFinishPickingAssets:)])
  462. [self.delegate assetsPickerController:self didFinishPickingAssets:self.selectedAssets];
  463. }
  464. @end