Please note, this blog entry is from a previous course. You might want to check out the current one.
When the user chooses a place in a table view, you must query Flickr again to get an array of 50 photos from that place and display them in a list. The URL provided by FlickrFetcher’s URLforPhotosInPlace:maxResults: method will get this from Flickr.
Create a new class derived from the table-view-controller class to be used to show a list of photos with a public property to hold an array of photos:
@property (nonatomic, strong) NSArray *photos; // of Flickr photos NSDicitionary
Everytime the photos are set, reload the table:
- (void)setPhotos:(NSArray *)photos { _photos = photos; [self.tableView reloadData]; }
The number of rows equals the number of photos:
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section { return [self.photos count]; }
The cells are populated with data from the photos for now (we will change that in the following task):
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { static NSString *CellIdentifier = @"Photo Cell"; UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier forIndexPath:indexPath]; NSDictionary *photo = self.photos[indexPath.row]; cell.textLabel.text = [photo valueForKeyPath:FLICKR_PHOTO_TITLE]; cell.detailTextLabel.text = [photo valueForKeyPath:FLICKR_PHOTO_DESCRIPTION]; return cell; }
From this new class derive a new class with another public property to store the current place:
@property (nonatomic, strong) NSDictionary *place;
When the view did load or then the refresh action is triggered, download the available photos for the current place using another helper method:
#define MAX_PHOTO_RESULTS 50 - (IBAction)fetchPhotos { [self.refreshControl beginRefreshing]; [self.tableView setContentOffset:CGPointMake(0, -self.refreshControl.frame.size.height) animated:YES]; [FlickrHelper loadPhotosInPlace:self.place maxResults:MAX_PHOTO_RESULTS onCompletion:^(NSArray *photos, NSError *error) { if (!error) { self.photos = photos; [self.refreshControl endRefreshing]; } else { NSLog(@"Error loading Photos of %@: %@", self.place, error); } }]; } - (void)viewDidLoad { [super viewDidLoad]; [self fetchPhotos]; }
The helper method is similar to the one used to load places:
+ (void)loadPhotosInPlace:(NSDictionary *)place maxResults:(NSUInteger)results onCompletion:(void (^)(NSArray *photos, NSError *error))completionHandler { NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration ephemeralSessionConfiguration]]; NSURLSessionDownloadTask *task = [session downloadTaskWithURL:[FlickrHelper URLforPhotosInPlace:[place valueForKeyPath:FLICKR_PLACE_ID] maxResults:results] completionHandler:^(NSURL *location, NSURLResponse *response, NSError *error) { NSArray *photos; if (!error) { photos = [[NSJSONSerialization JSONObjectWithData:[NSData dataWithContentsOfURL:location] options:0 error:&error] valueForKeyPath:FLICKR_RESULTS_PHOTOS]; } dispatch_async(dispatch_get_main_queue(), ^{ completionHandler(photos, error); }); }]; [task resume]; }
The current place and the title of the view are set when selecting a cell in the places table:
- (void)preparePhotosTVC:(PlacePhotosTVC *)tvc forPlace:(NSDictionary *)place { tvc.place = place; tvc.title = [FlickrHelper titleOfPlace:place]; } - (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender { NSIndexPath *indexPath = [self.tableView indexPathForCell:sender]; if ([segue.identifier isEqualToString:@"Show Place"] && indexPath) { [self preparePhotosTVC:segue.destinationViewController forPlace:self.placesByCountry[self.countries[indexPath.section]][indexPath.row]]; } }
Finally add a new table view controllor in story board and link it to the just created class above. Don’t forget to set the cell identifier. Create a push segue from the cell of the first table view controller to this new controller. Don’t for set its identifier, too:
The complete code is available on github.
data:image/s3,"s3://crabby-images/0463e/0463e68a5d77bbec630deb1cc37d4d49d1933e77" alt="Share on Facebook Facebook"
data:image/s3,"s3://crabby-images/93960/93960774b56bccf62f288efe326475d585239d8b" alt="Share on Twitter twitter"
data:image/s3,"s3://crabby-images/0c920/0c920341ab12cbcb5a3b1d2fb88f22374f2db577" alt="Share on Reddit reddit"
data:image/s3,"s3://crabby-images/d89b8/d89b80500f1626baf2ffa1e4369c261e01677d36" alt="Pin it with Pinterest pinterest"
data:image/s3,"s3://crabby-images/e9270/e92707689a9d88fc53f343370506e7f2aaedb7d1" alt="Share on Linkedin linkedin"
data:image/s3,"s3://crabby-images/f159f/f159fb333858fef1148f2a558ee7fa6a6881800a" alt="Share on tumblr tumblr"
data:image/s3,"s3://crabby-images/c7ab2/c7ab2f6c43fffe17aac7f185fd64ee95565487b9" alt="Share by email mail"
data:image/s3,"s3://crabby-images/6d85a/6d85a390df03e0c75624ac0579180ad35a62eeac" alt="Share on Facebook Facebook"
data:image/s3,"s3://crabby-images/e5f0c/e5f0cd4639d7b3d76b66c05981935f6c2bb06c71" alt="Share on Twitter twitter"
data:image/s3,"s3://crabby-images/40b77/40b7752481674a140035def4b6339fa4accc753f" alt="Share on Reddit reddit"
data:image/s3,"s3://crabby-images/1393f/1393f0291ffd80beac5e1f70b986671c6635d8b7" alt="Pin it with Pinterest pinterest"
data:image/s3,"s3://crabby-images/aa22d/aa22de91a0cd30772c623a5c7979ee81fe426013" alt="Share on Linkedin linkedin"
data:image/s3,"s3://crabby-images/5dc32/5dc32b39b30592e11870503bb4e8e715d50b21a0" alt="Share on tumblr tumblr"
data:image/s3,"s3://crabby-images/56655/56655b42634351b14f9d7507dacc4e33f64b90af" alt="Share by email mail"
Trying to take course on my own and can’t get URL for PhotosInPlace to work. Very much a newbie.
Flickr returns fail message of: “Method “flickr.photos.searcapih” not found”.
Is query still working?
Never mind. Figured it out had inadvertently modified URL string in FlickrFetcher.m
… which, theoretically, you should not touch 😉