Please note, this blog entry is from a previous course. You might want to check out the current one.
Recreate the user-interface from last week’s assignment but rely entirely on the above-mentioned Core Data database to present it.
Copy the files for the second table-view controller from the previous project together with the image-view controller and the flicker cache class into the current project.
Change the table view controller to be a subclass of the core-data-table-view controller, and change the public property to be a Tag entity.
@interface FlickrPhotoTVC : CoreDataTableViewController @property (nonatomic, strong) Tag *tag;
Remove all methods for the table-view data source but the one for creating cells, and adjust it to use the Photo entity.
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { ... Photo *photo = [self.fetchedResultsController objectAtIndexPath:indexPath]; cell.textLabel.text = photo.title; cell.detailTextLabel.text = photo.subtitle; ... }
The method for the table-view delegate stays untouched as well as the segue method. Both methods use sendDataforIndexPath:toViewController: which needs to be adjusted for core data:
- (void)sendDataforIndexPath:(NSIndexPath *)indexPath toViewController:(UIViewController *)vc { if ([vc respondsToSelector:@selector(setImageURL:)]) { ... [Recent recentPhoto:photo]; [vc performSelector:@selector(setImageURL:) withObject:[NSURL URLWithString:photo.imageURL]]; [vc performSelector:@selector(setTitle:) withObject:photo.title]; } }
Note that in the method above, also the Recent entity is handled, for which we will create a new category shortly.
the fetched-results controller and the title of the view are setup when changing the tag (when segueing to the current view):
- (void)setTag:(Tag *)tag { _tag = tag; self.title = [tag.name capitalizedString]; [self setupFetchedResultsController]; } - (void)setupFetchedResultsController { if (self.tag.managedObjectContext) { NSFetchRequest *request = [NSFetchRequest fetchRequestWithEntityName:@"Photo"]; request.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"title" ascending:YES selector:@selector(localizedCaseInsensitiveCompare:)]]; request.predicate = [NSPredicate predicateWithFormat:@"%@ in tags", self.tag]; self.fetchedResultsController = [[NSFetchedResultsController alloc] initWithFetchRequest:request managedObjectContext:self.tag.managedObjectContext sectionNameKeyPath:nil cacheName:nil]; } else { self.fetchedResultsController = nil; } }
Finally create a category for the Recent entity. Check if the photo has been recently viewed. If yes, just update the entity to the current date. If not, create a new entry, check if the maximum number of entries has been reached, and delete the oldest entry if necessary.
+ (Recent *)recentPhoto:(Photo *)photo { Recent *recent = nil; NSFetchRequest *request = [NSFetchRequest fetchRequestWithEntityName:@"Recent"]; request.predicate = [NSPredicate predicateWithFormat:@"photo = %@", photo]; NSError *error = nil; NSArray *matches = [photo.managedObjectContext executeFetchRequest:request error:&error]; if (!matches || ([matches count] > 1)) { // handle error } else if (![matches count]) { recent = [NSEntityDescription insertNewObjectForEntityForName:@"Recent" inManagedObjectContext:photo.managedObjectContext]; recent.photo = photo; recent.lastViewed = [NSDate date]; request.predicate = nil; request.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"lastViewed" ascending:NO]]; matches = [photo.managedObjectContext executeFetchRequest:request error:&error]; if ([matches count] > RECENT_FLICKR_PHOTOS_NUMBER) { [photo.managedObjectContext deleteObject:[matches lastObject]]; } } else { recent = [matches lastObject]; recent.lastViewed = [NSDate date]; } return recent; }
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"