123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650 |
- //
- // MWPhotoBrowser.m
- // MWPhotoBrowser
- //
- // Created by Michael Waterfall on 14/10/2010.
- // Copyright 2010 d3i. All rights reserved.
- //
- #import <QuartzCore/QuartzCore.h>
- #import "MWCommon.h"
- #import "MWPhotoBrowser.h"
- #import "MWPhotoBrowserPrivate.h"
- #import "UIImage+MWPhotoBrowser.h"
- #import "VideoPlayerKit.h"
- #define PADDING 10
- static void * MWVideoPlayerObservation = &MWVideoPlayerObservation;
- @implementation MWPhotoBrowser
- #pragma mark - Init
- - (id)init {
- if ((self = [super init])) {
- [self _initialisation];
- }
- return self;
- }
- - (id)initWithDelegate:(id <MWPhotoBrowserDelegate>)delegate {
- if ((self = [self init])) {
- _delegate = delegate;
- }
- return self;
- }
- - (id)initWithPhotos:(NSArray *)photosArray {
- if ((self = [self init])) {
- _fixedPhotosArray = photosArray;
- }
- return self;
- }
- - (id)initWithCoder:(NSCoder *)decoder {
- if ((self = [super initWithCoder:decoder])) {
- [self _initialisation];
- }
- return self;
- }
- - (void)_initialisation {
-
- // Defaults
- NSNumber *isVCBasedStatusBarAppearanceNum = [[NSBundle mainBundle] objectForInfoDictionaryKey:@"UIViewControllerBasedStatusBarAppearance"];
- if (isVCBasedStatusBarAppearanceNum) {
- _isVCBasedStatusBarAppearance = isVCBasedStatusBarAppearanceNum.boolValue;
- } else {
- _isVCBasedStatusBarAppearance = YES; // default
- }
- self.hidesBottomBarWhenPushed = YES;
- _hasBelongedToViewController = NO;
- _photoCount = NSNotFound;
- _previousLayoutBounds = CGRectZero;
- _currentPageIndex = 0;
- _previousPageIndex = NSUIntegerMax;
- _currentVideoIndex = NSUIntegerMax;
- _displayActionButton = YES;
- _displayNavArrows = NO;
- _zoomPhotosToFill = YES;
- _performingLayout = NO; // Reset on view did appear
- _rotating = NO;
- _viewIsActive = NO;
- _enableGrid = YES;
- _startOnGrid = NO;
- _enableSwipeToDismiss = YES;
- _delayToHideElements = 5;
- _visiblePages = [[NSMutableSet alloc] init];
- _recycledPages = [[NSMutableSet alloc] init];
- _photos = [[NSMutableArray alloc] init];
- _thumbPhotos = [[NSMutableArray alloc] init];
- _currentGridContentOffset = CGPointMake(0, CGFLOAT_MAX);
- _didSavePreviousStateOfNavBar = NO;
- self.automaticallyAdjustsScrollViewInsets = NO;
-
- // Listen for MWPhoto notifications
- [[NSNotificationCenter defaultCenter] addObserver:self
- selector:@selector(handleMWPhotoLoadingDidEndNotification:)
- name:MWPHOTO_LOADING_DID_END_NOTIFICATION
- object:nil];
-
- }
- - (void)dealloc {
- [self clearCurrentVideo];
- _pagingScrollView.delegate = nil;
- [[NSNotificationCenter defaultCenter] removeObserver:self];
- [self releaseAllUnderlyingPhotos:NO];
- }
- - (void)releaseAllUnderlyingPhotos:(BOOL)preserveCurrent {
- // Create a copy in case this array is modified while we are looping through
- // Release photos
- NSArray *copy = [_photos copy];
- for (id p in copy) {
- if (p != [NSNull null]) {
- if (preserveCurrent && p == [self photoAtIndex:self.currentIndex]) {
- continue; // skip current
- }
- [p unloadUnderlyingImage];
- }
- }
- // Release thumbs
- copy = [_thumbPhotos copy];
- for (id p in copy) {
- if (p != [NSNull null]) {
- [p unloadUnderlyingImage];
- }
- }
- }
- - (void)didReceiveMemoryWarning {
- // Release any cached data, images, etc that aren't in use.
- [self releaseAllUnderlyingPhotos:YES];
- [_recycledPages removeAllObjects];
-
- // Releases the view if it doesn't have a superview.
- [super didReceiveMemoryWarning];
-
- }
- #pragma mark - View Loading
- // Implement viewDidLoad to do additional setup after loading the view, typically from a nib.
- - (void)viewDidLoad {
-
- // Validate grid settings
- if (_startOnGrid) _enableGrid = YES;
- if (_enableGrid) {
- _enableGrid = [_delegate respondsToSelector:@selector(photoBrowser:thumbPhotoAtIndex:)];
- }
- if (!_enableGrid) _startOnGrid = NO;
-
- // View
- self.view.backgroundColor = [UIColor blackColor];
- self.view.clipsToBounds = YES;
-
- // Setup paging scrolling view
- CGRect pagingScrollViewFrame = [self frameForPagingScrollView];
- _pagingScrollView = [[UIScrollView alloc] initWithFrame:pagingScrollViewFrame];
- _pagingScrollView.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight;
- _pagingScrollView.pagingEnabled = YES;
- _pagingScrollView.delegate = self;
- _pagingScrollView.showsHorizontalScrollIndicator = NO;
- _pagingScrollView.showsVerticalScrollIndicator = NO;
- _pagingScrollView.backgroundColor = [UIColor blackColor];
- _pagingScrollView.contentSize = [self contentSizeForPagingScrollView];
- [self.view addSubview:_pagingScrollView];
- _currentVideoPlayerViewController = nil;
- // Toolbar
- _toolbar = [[UIToolbar alloc] initWithFrame:[self frameForToolbarAtOrientation:self.interfaceOrientation]];
- _toolbar.tintColor = [UIColor whiteColor];
- _toolbar.barTintColor = nil;
- [_toolbar setBackgroundImage:nil forToolbarPosition:UIToolbarPositionAny barMetrics:UIBarMetricsDefault];
- [_toolbar setBackgroundImage:nil forToolbarPosition:UIToolbarPositionAny barMetrics:UIBarMetricsLandscapePhone];
- _toolbar.barStyle = UIBarStyleBlackTranslucent;
- _toolbar.autoresizingMask = UIViewAutoresizingFlexibleTopMargin | UIViewAutoresizingFlexibleWidth;
-
- // Toolbar Items
- if (self.displayNavArrows) {
- NSString *arrowPathFormat = @"MWPhotoBrowser.bundle/UIBarButtonItemArrow%@";
- UIImage *previousButtonImage = [UIImage imageForResourcePath:[NSString stringWithFormat:arrowPathFormat, @"Left"] ofType:@"png" inBundle:[NSBundle bundleForClass:[self class]]];
- UIImage *nextButtonImage = [UIImage imageForResourcePath:[NSString stringWithFormat:arrowPathFormat, @"Right"] ofType:@"png" inBundle:[NSBundle bundleForClass:[self class]]];
- _previousButton = [[UIBarButtonItem alloc] initWithImage:previousButtonImage style:UIBarButtonItemStylePlain target:self action:@selector(gotoPreviousPage)];
- _nextButton = [[UIBarButtonItem alloc] initWithImage:nextButtonImage style:UIBarButtonItemStylePlain target:self action:@selector(gotoNextPage)];
- }
- if (self.displayActionButton) {
- _actionButton = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemAction target:self action:@selector(actionButtonPressed:)];
- }
-
- // Update
- [self reloadData];
-
- // Swipe to dismiss
- if (_enableSwipeToDismiss) {
- UISwipeGestureRecognizer *swipeGesture = [[UISwipeGestureRecognizer alloc] initWithTarget:self action:@selector(doneButtonPressed:)];
- swipeGesture.direction = UISwipeGestureRecognizerDirectionDown | UISwipeGestureRecognizerDirectionUp;
- [self.view addGestureRecognizer:swipeGesture];
- }
-
- // Super
- [super viewDidLoad];
-
- }
- - (void)performLayout {
-
- // Setup
- _performingLayout = YES;
- NSUInteger numberOfPhotos = [self numberOfPhotos];
-
- // Setup pages
- [_visiblePages removeAllObjects];
- [_recycledPages removeAllObjects];
-
- // Navigation buttons
- if ([self.navigationController.viewControllers objectAtIndex:0] == self) {
- // We're first on stack so show done button
- _doneButton = [[UIBarButtonItem alloc] initWithTitle:NSLocalizedString(@"Done", nil) style:UIBarButtonItemStylePlain target:self action:@selector(doneButtonPressed:)];
- // Set appearance
- [_doneButton setBackgroundImage:nil forState:UIControlStateNormal barMetrics:UIBarMetricsDefault];
- [_doneButton setBackgroundImage:nil forState:UIControlStateNormal barMetrics:UIBarMetricsLandscapePhone];
- [_doneButton setBackgroundImage:nil forState:UIControlStateHighlighted barMetrics:UIBarMetricsDefault];
- [_doneButton setBackgroundImage:nil forState:UIControlStateHighlighted barMetrics:UIBarMetricsLandscapePhone];
- [_doneButton setTitleTextAttributes:[NSDictionary dictionary] forState:UIControlStateNormal];
- [_doneButton setTitleTextAttributes:[NSDictionary dictionary] forState:UIControlStateHighlighted];
- self.navigationItem.rightBarButtonItem = _doneButton;
- } else {
- // We're not first so show back button
- UIViewController *previousViewController = [self.navigationController.viewControllers objectAtIndex:self.navigationController.viewControllers.count-2];
- NSString *backButtonTitle = previousViewController.navigationItem.backBarButtonItem ? previousViewController.navigationItem.backBarButtonItem.title : previousViewController.title;
- UIBarButtonItem *newBackButton = [[UIBarButtonItem alloc] initWithTitle:backButtonTitle style:UIBarButtonItemStylePlain target:nil action:nil];
- // Appearance
- [newBackButton setBackButtonBackgroundImage:nil forState:UIControlStateNormal barMetrics:UIBarMetricsDefault];
- [newBackButton setBackButtonBackgroundImage:nil forState:UIControlStateNormal barMetrics:UIBarMetricsLandscapePhone];
- [newBackButton setBackButtonBackgroundImage:nil forState:UIControlStateHighlighted barMetrics:UIBarMetricsDefault];
- [newBackButton setBackButtonBackgroundImage:nil forState:UIControlStateHighlighted barMetrics:UIBarMetricsLandscapePhone];
- [newBackButton setTitleTextAttributes:[NSDictionary dictionary] forState:UIControlStateNormal];
- [newBackButton setTitleTextAttributes:[NSDictionary dictionary] forState:UIControlStateHighlighted];
- _previousViewControllerBackButton = previousViewController.navigationItem.backBarButtonItem; // remember previous
- previousViewController.navigationItem.backBarButtonItem = newBackButton;
- }
- // Toolbar items
- BOOL hasItems = NO;
- UIBarButtonItem *fixedSpace = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemFixedSpace target:self action:nil];
- fixedSpace.width = 32; // To balance action button
- UIBarButtonItem *flexSpace = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemFlexibleSpace target:self action:nil];
- NSMutableArray *items = [[NSMutableArray alloc] init];
- // Left button - Grid
- if (_enableGrid) {
- hasItems = YES;
- [items addObject:[[UIBarButtonItem alloc] initWithImage:[UIImage imageForResourcePath:@"MWPhotoBrowser.bundle/UIBarButtonItemGrid" ofType:@"png" inBundle:[NSBundle bundleForClass:[self class]]] style:UIBarButtonItemStylePlain target:self action:@selector(showGridAnimated)]];
- } else {
- [items addObject:fixedSpace];
- }
- // Middle - Nav
- if (_previousButton && _nextButton && numberOfPhotos > 1) {
- hasItems = YES;
- [items addObject:flexSpace];
- [items addObject:_previousButton];
- [items addObject:flexSpace];
- [items addObject:_nextButton];
- [items addObject:flexSpace];
- } else {
- [items addObject:flexSpace];
- }
- // Right - Action
- if (_actionButton && !(!hasItems && !self.navigationItem.rightBarButtonItem)) {
- [items addObject:_actionButton];
- } else {
- // We're not showing the toolbar so try and show in top right
- if (_actionButton)
- self.navigationItem.rightBarButtonItem = _actionButton;
- [items addObject:fixedSpace];
- }
- // Toolbar visibility
- [_toolbar setItems:items];
- BOOL hideToolbar = YES;
- for (UIBarButtonItem* item in _toolbar.items) {
- if (item != fixedSpace && item != flexSpace) {
- hideToolbar = NO;
- break;
- }
- }
- if (hideToolbar) {
- [_toolbar removeFromSuperview];
- } else {
- [self.view addSubview:_toolbar];
- }
-
- // Update nav
- [self updateNavigation];
-
- // Content offset
- _pagingScrollView.contentOffset = [self contentOffsetForPageAtIndex:_currentPageIndex];
- [self tilePages];
- _performingLayout = NO;
-
- }
- // Release any retained subviews of the main view.
- - (void)viewDidUnload {
- _currentPageIndex = 0;
- _pagingScrollView = nil;
- _visiblePages = nil;
- _recycledPages = nil;
- _toolbar = nil;
- _previousButton = nil;
- _nextButton = nil;
- _progressHUD = nil;
- [super viewDidUnload];
- }
- - (BOOL)presentingViewControllerPrefersStatusBarHidden {
- UIViewController *presenting = self.presentingViewController;
- if (presenting) {
- if ([presenting isKindOfClass:[UINavigationController class]]) {
- presenting = [(UINavigationController *)presenting topViewController];
- }
- } else {
- // We're in a navigation controller so get previous one!
- if (self.navigationController && self.navigationController.viewControllers.count > 1) {
- presenting = [self.navigationController.viewControllers objectAtIndex:self.navigationController.viewControllers.count-2];
- }
- }
- if (presenting) {
- return [presenting prefersStatusBarHidden];
- } else {
- return NO;
- }
- }
- #pragma mark - Appearance
- - (void)viewWillAppear:(BOOL)animated {
-
- // Super
- [super viewWillAppear:animated];
-
- // Status bar
- if (!_viewHasAppearedInitially) {
- _leaveStatusBarAlone = [self presentingViewControllerPrefersStatusBarHidden];
- // Check if status bar is hidden on first appear, and if so then ignore it
- if (CGRectEqualToRect([[UIApplication sharedApplication] statusBarFrame], CGRectZero)) {
- _leaveStatusBarAlone = YES;
- }
- }
- // Set style
- if (!_leaveStatusBarAlone && UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone) {
- _previousStatusBarStyle = [[UIApplication sharedApplication] statusBarStyle];
- [[UIApplication sharedApplication] setStatusBarStyle:UIStatusBarStyleLightContent animated:animated];
- }
-
- // Navigation bar appearance
- if (!_viewIsActive && [self.navigationController.viewControllers objectAtIndex:0] != self) {
- [self storePreviousNavBarAppearance];
- }
- [self setNavBarAppearance:animated];
-
- // Update UI
- [self hideControlsAfterDelay];
-
- // Initial appearance
- if (!_viewHasAppearedInitially) {
- if (_startOnGrid) {
- [self showGrid:NO];
- }
- }
-
- // If rotation occured while we're presenting a modal
- // and the index changed, make sure we show the right one now
- if (_currentPageIndex != _pageIndexBeforeRotation) {
- [self jumpToPageAtIndex:_pageIndexBeforeRotation animated:NO];
- }
-
- // Layout
- [self.view setNeedsLayout];
- }
- - (void)viewDidAppear:(BOOL)animated {
- [super viewDidAppear:animated];
- _viewIsActive = YES;
-
- // Autoplay if first is video
- if (!_viewHasAppearedInitially) {
- if (_autoPlayOnAppear) {
- MWPhoto *photo = [self photoAtIndex:_currentPageIndex];
- if ([photo respondsToSelector:@selector(isVideo)] && photo.isVideo) {
- [self playVideoAtIndex:_currentPageIndex];
- }
- }
- }
-
- _viewHasAppearedInitially = YES;
-
- }
- - (void)viewWillDisappear:(BOOL)animated {
-
- // Detect if rotation occurs while we're presenting a modal
- _pageIndexBeforeRotation = _currentPageIndex;
-
- // Check that we're disappearing for good
- // self.isMovingFromParentViewController just doesn't work, ever. Or self.isBeingDismissed
- if ((_doneButton && self.navigationController.isBeingDismissed) ||
- ([self.navigationController.viewControllers objectAtIndex:0] != self && ![self.navigationController.viewControllers containsObject:self])) {
- // State
- _viewIsActive = NO;
- [self clearCurrentVideo]; // Clear current playing video
-
- // Bar state / appearance
- [self restorePreviousNavBarAppearance:animated];
-
- }
-
- // Controls
- [self.navigationController.navigationBar.layer removeAllAnimations]; // Stop all animations on nav bar
- [NSObject cancelPreviousPerformRequestsWithTarget:self]; // Cancel any pending toggles from taps
- [self setControlsHidden:NO animated:NO permanent:YES];
-
- // Status bar
- if (!_leaveStatusBarAlone && UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone) {
- [[UIApplication sharedApplication] setStatusBarStyle:_previousStatusBarStyle animated:animated];
- }
-
- // Super
- [super viewWillDisappear:animated];
-
- }
- - (void)willMoveToParentViewController:(UIViewController *)parent {
- if (parent && _hasBelongedToViewController) {
- [NSException raise:@"MWPhotoBrowser Instance Reuse" format:@"MWPhotoBrowser instances cannot be reused."];
- }
- }
- - (void)didMoveToParentViewController:(UIViewController *)parent {
- if (!parent) _hasBelongedToViewController = YES;
- }
- #pragma mark - Nav Bar Appearance
- - (void)setNavBarAppearance:(BOOL)animated {
- [self.navigationController setNavigationBarHidden:NO animated:animated];
- UINavigationBar *navBar = self.navigationController.navigationBar;
- // navBar.tintColor = [UIColor whiteColor];
- // navBar.barTintColor = nil;
- navBar.shadowImage = nil;
- navBar.translucent = YES;
- navBar.barStyle = UIBarStyleBlackTranslucent;
- [navBar setBackgroundImage:nil forBarMetrics:UIBarMetricsDefault];
- [navBar setBackgroundImage:nil forBarMetrics:UIBarMetricsLandscapePhone];
- }
- - (void)storePreviousNavBarAppearance {
- _didSavePreviousStateOfNavBar = YES;
- _previousNavBarBarTintColor = self.navigationController.navigationBar.barTintColor;
- _previousNavBarTranslucent = self.navigationController.navigationBar.translucent;
- _previousNavBarTintColor = self.navigationController.navigationBar.tintColor;
- _previousNavBarHidden = self.navigationController.navigationBarHidden;
- _previousNavBarStyle = self.navigationController.navigationBar.barStyle;
- _previousNavigationBarBackgroundImageDefault = [self.navigationController.navigationBar backgroundImageForBarMetrics:UIBarMetricsDefault];
- _previousNavigationBarBackgroundImageLandscapePhone = [self.navigationController.navigationBar backgroundImageForBarMetrics:UIBarMetricsLandscapePhone];
- }
- - (void)restorePreviousNavBarAppearance:(BOOL)animated {
- if (_didSavePreviousStateOfNavBar) {
- [self.navigationController setNavigationBarHidden:_previousNavBarHidden animated:animated];
- UINavigationBar *navBar = self.navigationController.navigationBar;
- navBar.tintColor = _previousNavBarTintColor;
- navBar.translucent = _previousNavBarTranslucent;
- navBar.barTintColor = _previousNavBarBarTintColor;
- navBar.barStyle = _previousNavBarStyle;
- [navBar setBackgroundImage:_previousNavigationBarBackgroundImageDefault forBarMetrics:UIBarMetricsDefault];
- [navBar setBackgroundImage:_previousNavigationBarBackgroundImageLandscapePhone forBarMetrics:UIBarMetricsLandscapePhone];
- // Restore back button if we need to
- if (_previousViewControllerBackButton) {
- UIViewController *previousViewController = [self.navigationController topViewController]; // We've disappeared so previous is now top
- previousViewController.navigationItem.backBarButtonItem = _previousViewControllerBackButton;
- _previousViewControllerBackButton = nil;
- }
- }
- }
- #pragma mark - Layout
- - (void)viewWillLayoutSubviews {
- [super viewWillLayoutSubviews];
- [self layoutVisiblePages];
- }
- - (void)layoutVisiblePages {
-
- // Flag
- _performingLayout = YES;
-
- // Toolbar
- _toolbar.frame = [self frameForToolbarAtOrientation:self.interfaceOrientation];
-
- // Remember index
- NSUInteger indexPriorToLayout = _currentPageIndex;
-
- // Get paging scroll view frame to determine if anything needs changing
- CGRect pagingScrollViewFrame = [self frameForPagingScrollView];
-
- // Frame needs changing
- if (!_skipNextPagingScrollViewPositioning) {
- _pagingScrollView.frame = pagingScrollViewFrame;
- }
- _skipNextPagingScrollViewPositioning = NO;
-
- // Recalculate contentSize based on current orientation
- _pagingScrollView.contentSize = [self contentSizeForPagingScrollView];
-
- // Adjust frames and configuration of each visible page
- for (MWZoomingScrollView *page in _visiblePages) {
- NSUInteger index = page.index;
- page.frame = [self frameForPageAtIndex:index];
- if (page.captionView) {
- page.captionView.frame = [self frameForCaptionView:page.captionView atIndex:index];
- }
- if (page.selectedButton) {
- page.selectedButton.frame = [self frameForSelectedButton:page.selectedButton atIndex:index];
- }
- if (page.playButton) {
- page.playButton.frame = [self frameForPlayButton:page.playButton atIndex:index];
- }
-
- // Adjust scales if bounds has changed since last time
- if (!CGRectEqualToRect(_previousLayoutBounds, self.view.bounds)) {
- // Update zooms for new bounds
- [page setMaxMinZoomScalesForCurrentBounds];
- _previousLayoutBounds = self.view.bounds;
- }
- }
-
- // Adjust video loading indicator if it's visible
- [self positionVideoLoadingIndicator];
-
- // Adjust contentOffset to preserve page location based on values collected prior to location
- _pagingScrollView.contentOffset = [self contentOffsetForPageAtIndex:indexPriorToLayout];
- [self didStartViewingPageAtIndex:_currentPageIndex]; // initial
-
- // Reset
- _currentPageIndex = indexPriorToLayout;
- _performingLayout = NO;
-
- }
- #pragma mark - Rotation
- - (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)toInterfaceOrientation {
- return YES;
- }
- - (UIInterfaceOrientationMask)supportedInterfaceOrientations {
- return UIInterfaceOrientationMaskAll;
- }
- - (void)willRotateToInterfaceOrientation:(UIInterfaceOrientation)toInterfaceOrientation duration:(NSTimeInterval)duration {
-
- // Remember page index before rotation
- _pageIndexBeforeRotation = _currentPageIndex;
- _rotating = YES;
-
- // In iOS 7 the nav bar gets shown after rotation, but might as well do this for everything!
- if ([self areControlsHidden]) {
- // Force hidden
- self.navigationController.navigationBarHidden = YES;
- }
-
- }
- - (void)willAnimateRotationToInterfaceOrientation:(UIInterfaceOrientation)toInterfaceOrientation duration:(NSTimeInterval)duration {
-
- // Perform layout
- _currentPageIndex = _pageIndexBeforeRotation;
-
- // Delay control holding
- [self hideControlsAfterDelay];
-
- // Layout
- [self layoutVisiblePages];
-
- }
- - (void)didRotateFromInterfaceOrientation:(UIInterfaceOrientation)fromInterfaceOrientation {
- _rotating = NO;
- // Ensure nav bar isn't re-displayed
- if ([self areControlsHidden]) {
- self.navigationController.navigationBarHidden = NO;
- self.navigationController.navigationBar.alpha = 0;
- }
- }
- #pragma mark - Data
- - (NSUInteger)currentIndex {
- return _currentPageIndex;
- }
- - (void)reloadData {
-
- // Reset
- _photoCount = NSNotFound;
-
- // Get data
- NSUInteger numberOfPhotos = [self numberOfPhotos];
- [self releaseAllUnderlyingPhotos:YES];
- [_photos removeAllObjects];
- [_thumbPhotos removeAllObjects];
- for (int i = 0; i < numberOfPhotos; i++) {
- [_photos addObject:[NSNull null]];
- [_thumbPhotos addObject:[NSNull null]];
- }
- // Update current page index
- if (numberOfPhotos > 0) {
- _currentPageIndex = MAX(0, MIN(_currentPageIndex, numberOfPhotos - 1));
- } else {
- _currentPageIndex = 0;
- }
-
- // Update layout
- if ([self isViewLoaded]) {
- while (_pagingScrollView.subviews.count) {
- [[_pagingScrollView.subviews lastObject] removeFromSuperview];
- }
- [self performLayout];
- [self.view setNeedsLayout];
- }
-
- }
- - (NSUInteger)numberOfPhotos {
- if (_photoCount == NSNotFound) {
- if ([_delegate respondsToSelector:@selector(numberOfPhotosInPhotoBrowser:)]) {
- _photoCount = [_delegate numberOfPhotosInPhotoBrowser:self];
- } else if (_fixedPhotosArray) {
- _photoCount = _fixedPhotosArray.count;
- }
- }
- if (_photoCount == NSNotFound) _photoCount = 0;
- return _photoCount;
- }
- - (id<MWPhoto>)photoAtIndex:(NSUInteger)index {
- id <MWPhoto> photo = nil;
- if (index < _photos.count) {
- if ([_photos objectAtIndex:index] == [NSNull null]) {
- if ([_delegate respondsToSelector:@selector(photoBrowser:photoAtIndex:)]) {
- photo = [_delegate photoBrowser:self photoAtIndex:index];
- } else if (_fixedPhotosArray && index < _fixedPhotosArray.count) {
- photo = [_fixedPhotosArray objectAtIndex:index];
- }
- if (photo) [_photos replaceObjectAtIndex:index withObject:photo];
- } else {
- photo = [_photos objectAtIndex:index];
- }
- }
- return photo;
- }
- - (id<MWPhoto>)thumbPhotoAtIndex:(NSUInteger)index {
- id <MWPhoto> photo = nil;
- if (index < _thumbPhotos.count) {
- if ([_thumbPhotos objectAtIndex:index] == [NSNull null]) {
- if ([_delegate respondsToSelector:@selector(photoBrowser:thumbPhotoAtIndex:)]) {
- photo = [_delegate photoBrowser:self thumbPhotoAtIndex:index];
- }
- if (photo) [_thumbPhotos replaceObjectAtIndex:index withObject:photo];
- } else {
- photo = [_thumbPhotos objectAtIndex:index];
- }
- }
- return photo;
- }
- - (MWCaptionView *)captionViewForPhotoAtIndex:(NSUInteger)index {
- MWCaptionView *captionView = nil;
- if ([_delegate respondsToSelector:@selector(photoBrowser:captionViewForPhotoAtIndex:)]) {
- captionView = [_delegate photoBrowser:self captionViewForPhotoAtIndex:index];
- } else {
- id <MWPhoto> photo = [self photoAtIndex:index];
- if ([photo respondsToSelector:@selector(caption)]) {
- if ([photo caption]) captionView = [[MWCaptionView alloc] initWithPhoto:photo];
- }
- }
- captionView.alpha = [self areControlsHidden] ? 0 : 1; // Initial alpha
- return captionView;
- }
- - (BOOL)photoIsSelectedAtIndex:(NSUInteger)index {
- BOOL value = NO;
- if (_displaySelectionButtons) {
- if ([self.delegate respondsToSelector:@selector(photoBrowser:isPhotoSelectedAtIndex:)]) {
- value = [self.delegate photoBrowser:self isPhotoSelectedAtIndex:index];
- }
- }
- return value;
- }
- - (void)setPhotoSelected:(BOOL)selected atIndex:(NSUInteger)index {
- if (_displaySelectionButtons) {
- if ([self.delegate respondsToSelector:@selector(photoBrowser:photoAtIndex:selectedChanged:)]) {
- [self.delegate photoBrowser:self photoAtIndex:index selectedChanged:selected];
- }
- }
- }
- - (UIImage *)imageForPhoto:(id<MWPhoto>)photo {
- if (photo) {
- // Get image or obtain in background
- if ([photo underlyingImage]) {
- return [photo underlyingImage];
- } else {
- [photo loadUnderlyingImageAndNotify];
- }
- }
- return nil;
- }
- - (void)loadAdjacentPhotosIfNecessary:(id<MWPhoto>)photo {
- MWZoomingScrollView *page = [self pageDisplayingPhoto:photo];
- if (page) {
- // If page is current page then initiate loading of previous and next pages
- NSUInteger pageIndex = page.index;
- if (_currentPageIndex == pageIndex) {
- if (pageIndex > 0) {
- // Preload index - 1
- id <MWPhoto> photo = [self photoAtIndex:pageIndex-1];
- if (![photo underlyingImage]) {
- [photo loadUnderlyingImageAndNotify];
- MWLog(@"Pre-loading image at index %lu", (unsigned long)pageIndex-1);
- }
- }
- if (pageIndex < [self numberOfPhotos] - 1) {
- // Preload index + 1
- id <MWPhoto> photo = [self photoAtIndex:pageIndex+1];
- if (![photo underlyingImage]) {
- [photo loadUnderlyingImageAndNotify];
- MWLog(@"Pre-loading image at index %lu", (unsigned long)pageIndex+1);
- }
- }
- }
- }
- }
- #pragma mark - MWPhoto Loading Notification
- - (void)handleMWPhotoLoadingDidEndNotification:(NSNotification *)notification {
- id <MWPhoto> photo = [notification object];
- MWZoomingScrollView *page = [self pageDisplayingPhoto:photo];
- if (page) {
- if ([photo underlyingImage]) {
- // Successful load
- [page displayImage];
- [self loadAdjacentPhotosIfNecessary:photo];
- } else {
-
- // Failed to load
- [page displayImageFailure];
- }
- // Update nav
- [self updateNavigation];
- }
- }
- #pragma mark - Paging
- - (void)tilePages {
-
- // Calculate which pages should be visible
- // Ignore padding as paging bounces encroach on that
- // and lead to false page loads
- CGRect visibleBounds = _pagingScrollView.bounds;
- NSInteger iFirstIndex = (NSInteger)floorf((CGRectGetMinX(visibleBounds)+PADDING*2) / CGRectGetWidth(visibleBounds));
- NSInteger iLastIndex = (NSInteger)floorf((CGRectGetMaxX(visibleBounds)-PADDING*2-1) / CGRectGetWidth(visibleBounds));
- if (iFirstIndex < 0) iFirstIndex = 0;
- if (iFirstIndex > [self numberOfPhotos] - 1) iFirstIndex = [self numberOfPhotos] - 1;
- if (iLastIndex < 0) iLastIndex = 0;
- if (iLastIndex > [self numberOfPhotos] - 1) iLastIndex = [self numberOfPhotos] - 1;
-
- // Recycle no longer needed pages
- NSInteger pageIndex;
- for (MWZoomingScrollView *page in _visiblePages) {
- pageIndex = page.index;
- if (pageIndex < (NSUInteger)iFirstIndex || pageIndex > (NSUInteger)iLastIndex) {
- [_recycledPages addObject:page];
- [page.captionView removeFromSuperview];
- [page.selectedButton removeFromSuperview];
- [page.playButton removeFromSuperview];
- [page prepareForReuse];
- [page removeFromSuperview];
- MWLog(@"Removed page at index %lu", (unsigned long)pageIndex);
- }
- }
- [_visiblePages minusSet:_recycledPages];
- while (_recycledPages.count > 2) // Only keep 2 recycled pages
- [_recycledPages removeObject:[_recycledPages anyObject]];
-
- // Add missing pages
- for (NSUInteger index = (NSUInteger)iFirstIndex; index <= (NSUInteger)iLastIndex; index++) {
- if (![self isDisplayingPageForIndex:index]) {
-
- // Add new page
- MWZoomingScrollView *page = [self dequeueRecycledPage];
- if (!page) {
- page = [[MWZoomingScrollView alloc] initWithPhotoBrowser:self];
- }
- [_visiblePages addObject:page];
- [self configurePage:page forIndex:index];
- [_pagingScrollView addSubview:page];
- MWLog(@"Added page at index %lu", (unsigned long)index);
-
- // Add caption
- MWCaptionView *captionView = [self captionViewForPhotoAtIndex:index];
- if (captionView) {
- captionView.frame = [self frameForCaptionView:captionView atIndex:index];
- [_pagingScrollView addSubview:captionView];
- page.captionView = captionView;
- }
-
- // Add play button if needed
- if (page.displayingVideo) {
- UIButton *playButton = [UIButton buttonWithType:UIButtonTypeCustom];
- [playButton setImage:[UIImage imageForResourcePath:@"MWPhotoBrowser.bundle/PlayButtonOverlayLarge" ofType:@"png" inBundle:[NSBundle bundleForClass:[self class]]] forState:UIControlStateNormal];
- [playButton setImage:[UIImage imageForResourcePath:@"MWPhotoBrowser.bundle/PlayButtonOverlayLargeTap" ofType:@"png" inBundle:[NSBundle bundleForClass:[self class]]] forState:UIControlStateHighlighted];
- [playButton addTarget:self action:@selector(playButtonTapped:) forControlEvents:UIControlEventTouchUpInside];
- [playButton sizeToFit];
- playButton.frame = [self frameForPlayButton:playButton atIndex:index];
- [_pagingScrollView addSubview:playButton];
- page.playButton = playButton;
- }
-
- // Add selected button
- if (self.displaySelectionButtons) {
- UIButton *selectedButton = [UIButton buttonWithType:UIButtonTypeCustom];
- [selectedButton setImage:[UIImage imageForResourcePath:@"MWPhotoBrowser.bundle/ImageSelectedOff" ofType:@"png" inBundle:[NSBundle bundleForClass:[self class]]] forState:UIControlStateNormal];
- UIImage *selectedOnImage;
- if (self.customImageSelectedIconName) {
- selectedOnImage = [UIImage imageNamed:self.customImageSelectedIconName];
- } else {
- selectedOnImage = [UIImage imageForResourcePath:@"MWPhotoBrowser.bundle/ImageSelectedOn" ofType:@"png" inBundle:[NSBundle bundleForClass:[self class]]];
- }
- [selectedButton setImage:selectedOnImage forState:UIControlStateSelected];
- [selectedButton sizeToFit];
- selectedButton.adjustsImageWhenHighlighted = NO;
- [selectedButton addTarget:self action:@selector(selectedButtonTapped:) forControlEvents:UIControlEventTouchUpInside];
- selectedButton.frame = [self frameForSelectedButton:selectedButton atIndex:index];
- [_pagingScrollView addSubview:selectedButton];
- page.selectedButton = selectedButton;
- selectedButton.selected = [self photoIsSelectedAtIndex:index];
- }
-
- }
- }
-
- }
- - (void)updateVisiblePageStates {
- NSSet *copy = [_visiblePages copy];
- for (MWZoomingScrollView *page in copy) {
-
- // Update selection
- page.selectedButton.selected = [self photoIsSelectedAtIndex:page.index];
-
- }
- }
- - (BOOL)isDisplayingPageForIndex:(NSUInteger)index {
- for (MWZoomingScrollView *page in _visiblePages)
- if (page.index == index) return YES;
- return NO;
- }
- - (MWZoomingScrollView *)pageDisplayedAtIndex:(NSUInteger)index {
- MWZoomingScrollView *thePage = nil;
- for (MWZoomingScrollView *page in _visiblePages) {
- if (page.index == index) {
- thePage = page; break;
- }
- }
- return thePage;
- }
- - (MWZoomingScrollView *)pageDisplayingPhoto:(id<MWPhoto>)photo {
- MWZoomingScrollView *thePage = nil;
- for (MWZoomingScrollView *page in _visiblePages) {
- if (page.photo == photo) {
- thePage = page; break;
- }
- }
- return thePage;
- }
- - (void)configurePage:(MWZoomingScrollView *)page forIndex:(NSUInteger)index {
- page.frame = [self frameForPageAtIndex:index];
- page.index = index;
- id<MWPhoto> thumbPhoto = [self thumbPhotoAtIndex:index];
- [page setPhoto:[self photoAtIndex:index] thumbPhoto:thumbPhoto];
- }
- - (MWZoomingScrollView *)dequeueRecycledPage {
- MWZoomingScrollView *page = [_recycledPages anyObject];
- if (page) {
- [_recycledPages removeObject:page];
- }
- return page;
- }
- // Handle page changes
- - (void)didStartViewingPageAtIndex:(NSUInteger)index {
-
- // Handle 0 photos
- if (![self numberOfPhotos]) {
- // Show controls
- [self setControlsHidden:NO animated:YES permanent:YES];
- return;
- }
-
- // Handle video on page change
- if (!_rotating && index != _currentVideoIndex) {
- [self clearCurrentVideo];
- }
-
- // Release images further away than +/-1
- NSUInteger i;
- if (index > 0) {
- // Release anything < index - 1
- for (i = 0; i < index-1; i++) {
- id photo = [_photos objectAtIndex:i];
- if (photo != [NSNull null]) {
- [photo unloadUnderlyingImage];
- [_photos replaceObjectAtIndex:i withObject:[NSNull null]];
- MWLog(@"Released underlying image at index %lu", (unsigned long)i);
- }
- }
- }
- if (index < [self numberOfPhotos] - 1) {
- // Release anything > index + 1
- for (i = index + 2; i < _photos.count; i++) {
- id photo = [_photos objectAtIndex:i];
- if (photo != [NSNull null]) {
- [photo unloadUnderlyingImage];
- [_photos replaceObjectAtIndex:i withObject:[NSNull null]];
- MWLog(@"Released underlying image at index %lu", (unsigned long)i);
- }
- }
- }
-
- // Load adjacent images if needed and the photo is already
- // loaded. Also called after photo has been loaded in background
- id <MWPhoto> currentPhoto = [self photoAtIndex:index];
- if ([currentPhoto underlyingImage]) {
- // photo loaded so load ajacent now
- [self loadAdjacentPhotosIfNecessary:currentPhoto];
- }
-
- // Notify delegate
- if (index != _previousPageIndex) {
- if ([_delegate respondsToSelector:@selector(photoBrowser:didDisplayPhotoAtIndex:)])
- [_delegate photoBrowser:self didDisplayPhotoAtIndex:index];
- _previousPageIndex = index;
- }
-
- // Update nav
- [self updateNavigation];
-
- }
- #pragma mark - Frame Calculations
- - (CGRect)frameForPagingScrollView {
- CGRect frame = self.view.bounds;// [[UIScreen mainScreen] bounds];
- frame.origin.x -= PADDING;
- frame.size.width += (2 * PADDING);
- return CGRectIntegral(frame);
- }
- - (CGRect)frameForPageAtIndex:(NSUInteger)index {
- // We have to use our paging scroll view's bounds, not frame, to calculate the page placement. When the device is in
- // landscape orientation, the frame will still be in portrait because the pagingScrollView is the root view controller's
- // view, so its frame is in window coordinate space, which is never rotated. Its bounds, however, will be in landscape
- // because it has a rotation transform applied.
- CGRect bounds = _pagingScrollView.bounds;
- CGRect pageFrame = bounds;
- pageFrame.size.width -= (2 * PADDING);
- pageFrame.origin.x = (bounds.size.width * index) + PADDING;
- return CGRectIntegral(pageFrame);
- }
- - (CGSize)contentSizeForPagingScrollView {
- // We have to use the paging scroll view's bounds to calculate the contentSize, for the same reason outlined above.
- CGRect bounds = _pagingScrollView.bounds;
- return CGSizeMake(bounds.size.width * [self numberOfPhotos], bounds.size.height);
- }
- - (CGPoint)contentOffsetForPageAtIndex:(NSUInteger)index {
- CGFloat pageWidth = _pagingScrollView.bounds.size.width;
- CGFloat newOffset = index * pageWidth;
- return CGPointMake(newOffset, 0);
- }
- - (CGRect)frameForToolbarAtOrientation:(UIInterfaceOrientation)orientation {
- CGFloat height = 44;
- if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone &&
- UIInterfaceOrientationIsLandscape(orientation)) height = 32;
- return CGRectIntegral(CGRectMake(0, self.view.bounds.size.height - height, self.view.bounds.size.width, height));
- }
- - (CGRect)frameForCaptionView:(MWCaptionView *)captionView atIndex:(NSUInteger)index {
- CGRect pageFrame = [self frameForPageAtIndex:index];
- CGSize captionSize = [captionView sizeThatFits:CGSizeMake(pageFrame.size.width, 0)];
- CGRect captionFrame = CGRectMake(pageFrame.origin.x,
- pageFrame.size.height - captionSize.height - (_toolbar.superview?_toolbar.frame.size.height:0),
- pageFrame.size.width,
- captionSize.height);
- return CGRectIntegral(captionFrame);
- }
- - (CGRect)frameForSelectedButton:(UIButton *)selectedButton atIndex:(NSUInteger)index {
- CGRect pageFrame = [self frameForPageAtIndex:index];
- CGFloat padding = 20;
- CGFloat yOffset = 0;
- if (![self areControlsHidden]) {
- UINavigationBar *navBar = self.navigationController.navigationBar;
- yOffset = navBar.frame.origin.y + navBar.frame.size.height;
- }
- CGRect selectedButtonFrame = CGRectMake(pageFrame.origin.x + pageFrame.size.width - selectedButton.frame.size.width - padding,
- padding + yOffset,
- selectedButton.frame.size.width,
- selectedButton.frame.size.height);
- return CGRectIntegral(selectedButtonFrame);
- }
- - (CGRect)frameForPlayButton:(UIButton *)playButton atIndex:(NSUInteger)index {
- CGRect pageFrame = [self frameForPageAtIndex:index];
- return CGRectMake(floorf(CGRectGetMidX(pageFrame) - playButton.frame.size.width / 2),
- floorf(CGRectGetMidY(pageFrame) - playButton.frame.size.height / 2),
- playButton.frame.size.width,
- playButton.frame.size.height);
- }
- #pragma mark - UIScrollView Delegate
- - (void)scrollViewDidScroll:(UIScrollView *)scrollView {
-
- // Checks
- if (!_viewIsActive || _performingLayout || _rotating) return;
-
- // Tile pages
- [self tilePages];
-
- // Calculate current page
- CGRect visibleBounds = _pagingScrollView.bounds;
- NSInteger index = (NSInteger)(floorf(CGRectGetMidX(visibleBounds) / CGRectGetWidth(visibleBounds)));
- if (index < 0) index = 0;
- if (index > [self numberOfPhotos] - 1) index = [self numberOfPhotos] - 1;
- NSUInteger previousCurrentPage = _currentPageIndex;
- _currentPageIndex = index;
- if (_currentPageIndex != previousCurrentPage) {
- [self didStartViewingPageAtIndex:index];
- }
-
- }
- - (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView {
- // Hide controls when dragging begins
- [self setControlsHidden:YES animated:YES permanent:NO];
- }
- - (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView {
- // Update nav when page changes
- [self updateNavigation];
- }
- #pragma mark - Navigation
- - (void)updateNavigation {
-
- // Title
- NSUInteger numberOfPhotos = [self numberOfPhotos];
- if (_gridController) {
- if (_gridController.selectionMode) {
- self.title = NSLocalizedString(@"Select Photos", nil);
- } else {
- NSString *photosText;
- if (numberOfPhotos == 1) {
- photosText = NSLocalizedString(@"photo", @"Used in the context: '1 photo'");
- } else {
- photosText = NSLocalizedString(@"photos", @"Used in the context: '3 photos'");
- }
- self.title = [NSString stringWithFormat:@"%lu %@", (unsigned long)numberOfPhotos, photosText];
- }
- } else if (numberOfPhotos > 1) {
- if ([_delegate respondsToSelector:@selector(photoBrowser:titleForPhotoAtIndex:)]) {
- self.title = [_delegate photoBrowser:self titleForPhotoAtIndex:_currentPageIndex];
- } else {
- self.title = [NSString stringWithFormat:@"%lu %@ %lu", (unsigned long)(_currentPageIndex+1), NSLocalizedString(@"of", @"Used in the context: 'Showing 1 of 3 items'"), (unsigned long)numberOfPhotos];
- }
- } else {
- self.title = nil;
- }
-
- // Buttons
- _previousButton.enabled = (_currentPageIndex > 0);
- _nextButton.enabled = (_currentPageIndex < numberOfPhotos - 1);
-
- // Disable action button if there is no image or it's a video
- MWPhoto *photo = [self photoAtIndex:_currentPageIndex];
- if ([photo underlyingImage] == nil && !([photo respondsToSelector:@selector(isVideo)] && photo.isVideo)) {
- _actionButton.enabled = NO;
- _actionButton.tintColor = [UIColor clearColor]; // Tint to hide button
- } else {
- _actionButton.enabled = YES;
- _actionButton.tintColor = nil;
- }
-
- }
- - (void)jumpToPageAtIndex:(NSUInteger)index animated:(BOOL)animated {
-
- // Change page
- if (index < [self numberOfPhotos]) {
- CGRect pageFrame = [self frameForPageAtIndex:index];
- [_pagingScrollView setContentOffset:CGPointMake(pageFrame.origin.x - PADDING, 0) animated:animated];
- [self updateNavigation];
- }
-
- // Update timer to give more time
- [self hideControlsAfterDelay];
-
- }
- - (void)gotoPreviousPage {
- [self showPreviousPhotoAnimated:NO];
- }
- - (void)gotoNextPage {
- [self showNextPhotoAnimated:NO];
- }
- - (void)showPreviousPhotoAnimated:(BOOL)animated {
- [self jumpToPageAtIndex:_currentPageIndex-1 animated:animated];
- }
- - (void)showNextPhotoAnimated:(BOOL)animated {
- [self jumpToPageAtIndex:_currentPageIndex+1 animated:animated];
- }
- #pragma mark - Interactions
- - (void)selectedButtonTapped:(id)sender {
- UIButton *selectedButton = (UIButton *)sender;
- selectedButton.selected = !selectedButton.selected;
- NSUInteger index = NSUIntegerMax;
- for (MWZoomingScrollView *page in _visiblePages) {
- if (page.selectedButton == selectedButton) {
- index = page.index;
- break;
- }
- }
- if (index != NSUIntegerMax) {
- [self setPhotoSelected:selectedButton.selected atIndex:index];
- }
- }
- - (void)playButtonTapped:(id)sender {
- // Ignore if we're already playing a video
- if (_currentVideoIndex != NSUIntegerMax) {
- return;
- }
- NSUInteger index = [self indexForPlayButton:sender];
- if (index != NSUIntegerMax) {
- if (!_currentVideoPlayerViewController) {
- [self playVideoAtIndex:index];
- }
- }
- }
- - (NSUInteger)indexForPlayButton:(UIView *)playButton {
- NSUInteger index = NSUIntegerMax;
- for (MWZoomingScrollView *page in _visiblePages) {
- if (page.playButton == playButton) {
- index = page.index;
- break;
- }
- }
- return index;
- }
- #pragma mark - Video
- - (void)playVideoAtIndex:(NSUInteger)index {
- id photo = [self photoAtIndex:index];
- if ([photo respondsToSelector:@selector(getVideoURL:)]) {
-
- // Valid for playing
- [self clearCurrentVideo];
- _currentVideoIndex = index;
- [self setVideoLoadingIndicatorVisible:YES atPageIndex:index];
- // Get video and play
- typeof(self) __weak weakSelf = self;
- [photo getVideoURL:^(NSURL *url) {
- dispatch_async(dispatch_get_main_queue(), ^{
- // If the video is not playing anymore then bail
- typeof(self) strongSelf = weakSelf;
- if (!strongSelf) return;
- if (strongSelf->_currentVideoIndex != index || !strongSelf->_viewIsActive) {
- return;
- }
- if (url) {
- [weakSelf _playVideo:url atPhotoIndex:index];
- } else {
- [weakSelf setVideoLoadingIndicatorVisible:NO atPageIndex:index];
- }
- });
- }];
-
- }
- }
- - (void)_playVideo:(NSURL *)videoURL atPhotoIndex:(NSUInteger)index {
- [[NSNotificationCenter defaultCenter] addObserver:self
- selector:@selector(videoFinishedCallback:)
- name:AVPlayerItemDidPlayToEndTimeNotification
- object:nil];
-
- [[NSNotificationCenter defaultCenter] addObserver:self
- selector:@selector(videoFinishedCallback:)
- name:AVPlayerItemFailedToPlayToEndTimeNotification
- object:nil];
-
- [[NSNotificationCenter defaultCenter] addObserver:self
- selector:@selector(videoFinishedCallback:)
- name:@"kVideoPlayerKitPlayCanceled"
- object:nil];
-
-
-
- if (!_currentVideoPlayerViewController) {
- _currentVideoPlayerViewController = [VideoPlayerKit videoPlayerWithContainingView:self.view optionalTopView:nil hideTopViewWithControls:YES];
- _currentVideoPlayerViewController.allowPortraitFullscreen = YES;
- } else {
- [_currentVideoPlayerViewController.view removeFromSuperview];
- }
-
- [self.view addSubview:_currentVideoPlayerViewController.view];
-
- [_currentVideoPlayerViewController playVideoWithTitle:@" " URL:videoURL videoID:nil shareURL:nil isStreaming:NO playInFullScreen:YES];
- }
- - (void)videoFinishedCallback:(NSNotification*)notification {
- [[NSNotificationCenter defaultCenter] removeObserver:self name:AVPlayerItemDidPlayToEndTimeNotification object:nil];
- [[NSNotificationCenter defaultCenter] removeObserver:self name:AVPlayerItemFailedToPlayToEndTimeNotification object:nil];
- [[NSNotificationCenter defaultCenter] removeObserver:self name:@"kVideoPlayerKitPlayCanceled" object:nil];
- [self clearCurrentVideo];
- }
- - (void)clearCurrentVideo {
- _currentVideoPlayerViewController = nil;
- _currentVideoIndex = NSUIntegerMax;
- }
- - (void)setVideoLoadingIndicatorVisible:(BOOL)visible atPageIndex:(NSUInteger)pageIndex {
- }
- - (void)positionVideoLoadingIndicator {
-
- }
- #pragma mark - Grid
- - (void)showGridAnimated {
- [self showGrid:YES];
- }
- - (void)showGrid:(BOOL)animated {
- if (_gridController) return;
-
- // Clear video
- [self clearCurrentVideo];
-
- // Init grid controller
- _gridController = [[MWGridViewController alloc] init];
- _gridController.initialContentOffset = _currentGridContentOffset;
- _gridController.browser = self;
- _gridController.selectionMode = _displaySelectionButtons;
- _gridController.view.frame = self.view.bounds;
- _gridController.view.frame = CGRectOffset(_gridController.view.frame, 0, (self.startOnGrid ? -1 : 1) * self.view.bounds.size.height);
- // Stop specific layout being triggered
- _skipNextPagingScrollViewPositioning = YES;
-
- // Add as a child view controller
- [self addChildViewController:_gridController];
- [self.view addSubview:_gridController.view];
-
- // Perform any adjustments
- [_gridController.view layoutIfNeeded];
- [_gridController adjustOffsetsAsRequired];
-
- // Hide action button on nav bar if it exists
- if (self.navigationItem.rightBarButtonItem == _actionButton) {
- _gridPreviousRightNavItem = _actionButton;
- [self.navigationItem setRightBarButtonItem:nil animated:YES];
- } else {
- _gridPreviousRightNavItem = nil;
- }
-
- // Update
- [self updateNavigation];
- [self setControlsHidden:NO animated:YES permanent:YES];
-
- // Animate grid in and photo scroller out
- [_gridController willMoveToParentViewController:self];
- [UIView animateWithDuration:animated ? 0.3 : 0 animations:^(void) {
- _gridController.view.frame = self.view.bounds;
- CGRect newPagingFrame = [self frameForPagingScrollView];
- newPagingFrame = CGRectOffset(newPagingFrame, 0, (self.startOnGrid ? 1 : -1) * newPagingFrame.size.height);
- _pagingScrollView.frame = newPagingFrame;
- } completion:^(BOOL finished) {
- [_gridController didMoveToParentViewController:self];
- }];
-
- }
- - (void)hideGrid {
-
- if (!_gridController) return;
-
- // Remember previous content offset
- _currentGridContentOffset = _gridController.collectionView.contentOffset;
-
- // Restore action button if it was removed
- if (_gridPreviousRightNavItem == _actionButton && _actionButton) {
- [self.navigationItem setRightBarButtonItem:_gridPreviousRightNavItem animated:YES];
- }
-
- // Position prior to hide animation
- CGRect newPagingFrame = [self frameForPagingScrollView];
- newPagingFrame = CGRectOffset(newPagingFrame, 0, (self.startOnGrid ? 1 : -1) * newPagingFrame.size.height);
- _pagingScrollView.frame = newPagingFrame;
-
- // Remember and remove controller now so things can detect a nil grid controller
- MWGridViewController *tmpGridController = _gridController;
- _gridController = nil;
-
- // Update
- [self updateNavigation];
- [self updateVisiblePageStates];
-
- // Animate, hide grid and show paging scroll view
- [UIView animateWithDuration:0.3 animations:^{
- tmpGridController.view.frame = CGRectOffset(self.view.bounds, 0, (self.startOnGrid ? -1 : 1) * self.view.bounds.size.height);
- _pagingScrollView.frame = [self frameForPagingScrollView];
- } completion:^(BOOL finished) {
- [tmpGridController willMoveToParentViewController:nil];
- [tmpGridController.view removeFromSuperview];
- [tmpGridController removeFromParentViewController];
- [self setControlsHidden:NO animated:YES permanent:NO]; // retrigger timer
- }];
- }
- #pragma mark - Control Hiding / Showing
- // If permanent then we don't set timers to hide again
- // Fades all controls on iOS 5 & 6, and iOS 7 controls slide and fade
- - (void)setControlsHidden:(BOOL)hidden animated:(BOOL)animated permanent:(BOOL)permanent {
-
- // Force visible
- if (![self numberOfPhotos] || _gridController || _alwaysShowControls)
- hidden = NO;
-
- // Cancel any timers
- [self cancelControlHiding];
-
- // Animations & positions
- CGFloat animatonOffset = 20;
- CGFloat animationDuration = (animated ? 0.35 : 0);
-
- // Status bar
- if (!_leaveStatusBarAlone) {
- // Hide status bar
- if (!_isVCBasedStatusBarAppearance) {
-
- // Non-view controller based
- [[UIApplication sharedApplication] setStatusBarHidden:hidden withAnimation:animated ? UIStatusBarAnimationSlide : UIStatusBarAnimationNone];
-
- } else {
-
- // View controller based so animate away
- _statusBarShouldBeHidden = hidden;
- [UIView animateWithDuration:animationDuration animations:^(void) {
- [self setNeedsStatusBarAppearanceUpdate];
- } completion:^(BOOL finished) {}];
-
- }
- }
-
- // Toolbar, nav bar and captions
- // Pre-appear animation positions for sliding
- if ([self areControlsHidden] && !hidden && animated) {
-
- // Toolbar
- _toolbar.frame = CGRectOffset([self frameForToolbarAtOrientation:self.interfaceOrientation], 0, animatonOffset);
-
- // Captions
- for (MWZoomingScrollView *page in _visiblePages) {
- if (page.captionView) {
- MWCaptionView *v = page.captionView;
- // Pass any index, all we're interested in is the Y
- CGRect captionFrame = [self frameForCaptionView:v atIndex:0];
- captionFrame.origin.x = v.frame.origin.x; // Reset X
- v.frame = CGRectOffset(captionFrame, 0, animatonOffset);
- }
- }
-
- }
- [UIView animateWithDuration:animationDuration animations:^(void) {
-
- CGFloat alpha = hidden ? 0 : 1;
- // Nav bar slides up on it's own on iOS 7+
- [self.navigationController.navigationBar setAlpha:alpha];
-
- // Toolbar
- _toolbar.frame = [self frameForToolbarAtOrientation:self.interfaceOrientation];
- if (hidden) _toolbar.frame = CGRectOffset(_toolbar.frame, 0, animatonOffset);
- _toolbar.alpha = alpha;
- // Captions
- for (MWZoomingScrollView *page in _visiblePages) {
- if (page.captionView) {
- MWCaptionView *v = page.captionView;
- // Pass any index, all we're interested in is the Y
- CGRect captionFrame = [self frameForCaptionView:v atIndex:0];
- captionFrame.origin.x = v.frame.origin.x; // Reset X
- if (hidden) captionFrame = CGRectOffset(captionFrame, 0, animatonOffset);
- v.frame = captionFrame;
- v.alpha = alpha;
- }
- }
-
- // Selected buttons
- for (MWZoomingScrollView *page in _visiblePages) {
- if (page.selectedButton) {
- UIButton *v = page.selectedButton;
- CGRect newFrame = [self frameForSelectedButton:v atIndex:0];
- newFrame.origin.x = v.frame.origin.x;
- v.frame = newFrame;
- }
- }
- } completion:^(BOOL finished) {}];
-
- // Control hiding timer
- // Will cancel existing timer but only begin hiding if
- // they are visible
- if (!permanent) [self hideControlsAfterDelay];
-
- }
- - (BOOL)prefersStatusBarHidden {
- if (!_leaveStatusBarAlone) {
- return _statusBarShouldBeHidden;
- } else {
- return [self presentingViewControllerPrefersStatusBarHidden];
- }
- }
- - (UIStatusBarStyle)preferredStatusBarStyle {
- return UIStatusBarStyleLightContent;
- }
- - (UIStatusBarAnimation)preferredStatusBarUpdateAnimation {
- return UIStatusBarAnimationSlide;
- }
- - (void)cancelControlHiding {
- // If a timer exists then cancel and release
- if (_controlVisibilityTimer) {
- [_controlVisibilityTimer invalidate];
- _controlVisibilityTimer = nil;
- }
- }
- // Enable/disable control visiblity timer
- - (void)hideControlsAfterDelay {
- if (![self areControlsHidden]) {
- [self cancelControlHiding];
- _controlVisibilityTimer = [NSTimer scheduledTimerWithTimeInterval:self.delayToHideElements target:self selector:@selector(hideControls) userInfo:nil repeats:NO];
- }
- }
- - (BOOL)areControlsHidden { return (_toolbar.alpha == 0); }
- - (void)hideControls { [self setControlsHidden:YES animated:YES permanent:NO]; }
- - (void)showControls { [self setControlsHidden:NO animated:YES permanent:NO]; }
- - (void)toggleControls { [self setControlsHidden:![self areControlsHidden] animated:YES permanent:NO]; }
- #pragma mark - Properties
- - (void)setCurrentPhotoIndex:(NSUInteger)index {
- // Validate
- NSUInteger photoCount = [self numberOfPhotos];
- if (photoCount == 0) {
- index = 0;
- } else {
- if (index >= photoCount)
- index = [self numberOfPhotos]-1;
- }
- _currentPageIndex = index;
- if ([self isViewLoaded]) {
- [self jumpToPageAtIndex:index animated:NO];
- if (!_viewIsActive)
- [self tilePages]; // Force tiling if view is not visible
- }
- }
- #pragma mark - Misc
- - (void)doneButtonPressed:(id)sender {
- // Only if we're modal and there's a done button
- if (_doneButton) {
- // See if we actually just want to show/hide grid
- if (self.enableGrid) {
- if (self.startOnGrid && !_gridController) {
- [self showGrid:YES];
- return;
- } else if (!self.startOnGrid && _gridController) {
- [self hideGrid];
- return;
- }
- }
- // Dismiss view controller
- if ([_delegate respondsToSelector:@selector(photoBrowserDidFinishModalPresentation:)]) {
- // Call delegate method and let them dismiss us
- [_delegate photoBrowserDidFinishModalPresentation:self];
- } else {
- [self dismissViewControllerAnimated:YES completion:nil];
- }
- }
- }
- #pragma mark - Actions
- - (void)actionButtonPressed:(id)sender {
- // Only react when image has loaded
- id <MWPhoto> photo = [self photoAtIndex:_currentPageIndex];
-
- if (([self numberOfPhotos] > 0 && [photo underlyingImage]) || ([photo respondsToSelector:@selector(isVideo)] && photo.isVideo)) {
-
- // If they have defined a delegate method then just message them
- if ([self.delegate respondsToSelector:@selector(photoBrowser:actionButtonPressedForPhotoAtIndex:)]) {
-
- // Let delegate handle things
- [self.delegate photoBrowser:self actionButtonPressedForPhotoAtIndex:_currentPageIndex];
-
- } else {
-
- // Show activity view controller
- __block NSMutableArray *items;
- if([photo respondsToSelector:@selector(isVideo)] && photo.isVideo) {
- [photo getVideoURL:^(NSURL *url) {
- items = [NSMutableArray arrayWithObject:url];
- }];
- } else {
- items = [NSMutableArray arrayWithObject:[photo underlyingImage]];
- if (photo.caption) {
- [items addObject:photo.caption];
- }
- }
-
- self.activityViewController = [[UIActivityViewController alloc] initWithActivityItems:items applicationActivities:nil];
-
- // Show loading spinner after a couple of seconds
- double delayInSeconds = 2.0;
- dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delayInSeconds * NSEC_PER_SEC));
- dispatch_after(popTime, dispatch_get_main_queue(), ^(void){
- if (self.activityViewController) {
- [self showProgressHUDWithMessage:nil];
- }
- });
- // Show
- typeof(self) __weak weakSelf = self;
- [self.activityViewController setCompletionHandler:^(NSString *activityType, BOOL completed) {
- weakSelf.activityViewController = nil;
- [weakSelf hideControlsAfterDelay];
- [weakSelf hideProgressHUD:YES];
- }];
- // iOS 8 - Set the Anchor Point for the popover
- if (SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(@"8")) {
- self.activityViewController.popoverPresentationController.barButtonItem = _actionButton;
- }
- [self presentViewController:self.activityViewController animated:YES completion:nil];
- }
-
- // Keep controls hidden
- [self setControlsHidden:NO animated:YES permanent:YES];
- }
-
- }
- #pragma mark - Action Progress
- - (MBProgressHUD *)progressHUD {
- if (!_progressHUD) {
- _progressHUD = [[MBProgressHUD alloc] initWithView:self.view];
- _progressHUD.minSize = CGSizeMake(120, 120);
- _progressHUD.minShowTime = 1;
- [self.view addSubview:_progressHUD];
- }
- return _progressHUD;
- }
- - (void)showProgressHUDWithMessage:(NSString *)message {
- self.progressHUD.labelText = message;
- self.progressHUD.mode = MBProgressHUDModeIndeterminate;
- [self.progressHUD show:YES];
- self.navigationController.navigationBar.userInteractionEnabled = NO;
- }
- - (void)hideProgressHUD:(BOOL)animated {
- [self.progressHUD hide:animated];
- self.navigationController.navigationBar.userInteractionEnabled = YES;
- }
- - (void)showProgressHUDCompleteMessage:(NSString *)message {
- if (message) {
- if (self.progressHUD.isHidden) [self.progressHUD show:YES];
- self.progressHUD.labelText = message;
- self.progressHUD.mode = MBProgressHUDModeCustomView;
- [self.progressHUD hide:YES afterDelay:1.5];
- } else {
- [self.progressHUD hide:YES];
- }
- self.navigationController.navigationBar.userInteractionEnabled = YES;
- }
- @end
|