/* |
File: RecipeListTableViewController.m |
Abstract: Table view controller to manage an editable table view that displays a list of recipes. |
Recipes are displayed in a custom table view cell. |
|
Version: 1.5 |
|
Disclaimer: IMPORTANT: This Apple software is supplied to you by Apple |
Inc. ("Apple") in consideration of your agreement to the following |
terms, and your use, installation, modification or redistribution of |
this Apple software constitutes acceptance of these terms. If you do |
not agree with these terms, please do not use, install, modify or |
redistribute this Apple software. |
|
In consideration of your agreement to abide by the following terms, and |
subject to these terms, Apple grants you a personal, non-exclusive |
license, under Apple's copyrights in this original Apple software (the |
"Apple Software"), to use, reproduce, modify and redistribute the Apple |
Software, with or without modifications, in source and/or binary forms; |
provided that if you redistribute the Apple Software in its entirety and |
without modifications, you must retain this notice and the following |
text and disclaimers in all such redistributions of the Apple Software. |
Neither the name, trademarks, service marks or logos of Apple Inc. may |
be used to endorse or promote products derived from the Apple Software |
without specific prior written permission from Apple. Except as |
expressly stated in this notice, no other rights or licenses, express or |
implied, are granted by Apple herein, including but not limited to any |
patent rights that may be infringed by your derivative works or by other |
works in which the Apple Software may be incorporated. |
|
The Apple Software is provided by Apple on an "AS IS" basis. APPLE |
MAKES NO WARRANTIES, EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION |
THE IMPLIED WARRANTIES OF NON-INFRINGEMENT, MERCHANTABILITY AND FITNESS |
FOR A PARTICULAR PURPOSE, REGARDING THE APPLE SOFTWARE OR ITS USE AND |
OPERATION ALONE OR IN COMBINATION WITH YOUR PRODUCTS. |
|
IN NO EVENT SHALL APPLE BE LIABLE FOR ANY SPECIAL, INDIRECT, INCIDENTAL |
OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
INTERRUPTION) ARISING IN ANY WAY OUT OF THE USE, REPRODUCTION, |
MODIFICATION AND/OR DISTRIBUTION OF THE APPLE SOFTWARE, HOWEVER CAUSED |
AND WHETHER UNDER THEORY OF CONTRACT, TORT (INCLUDING NEGLIGENCE), |
STRICT LIABILITY OR OTHERWISE, EVEN IF APPLE HAS BEEN ADVISED OF THE |
POSSIBILITY OF SUCH DAMAGE. |
|
Copyright (C) 2014 Apple Inc. All Rights Reserved. |
|
*/ |
|
#import "RecipeListTableViewController.h" |
#import "RecipeDetailViewController.h" |
#import "Recipe.h" |
#import "RecipeTableViewCell.h" |
|
@interface RecipeListTableViewController () |
@property (nonatomic, strong) NSFetchedResultsController *fetchedResultsController; |
@end |
|
@implementation RecipeListTableViewController |
|
// segue ID when "+" button is tapped |
static NSString *kShowRecipeSegueID = @"showRecipe"; |
|
// segue ID when "Add Ingredient" cell is tapped |
static NSString *kAddRecipeSegueID = @"addRecipe"; |
|
|
- (void)viewDidLoad { |
|
[super viewDidLoad]; |
|
// add the table's edit button to the left side of the nav bar |
self.navigationItem.leftBarButtonItem = self.editButtonItem; |
|
// Set the table view's row height |
self.tableView.rowHeight = 44.0; |
|
NSError *error = nil; |
if (![[self fetchedResultsController] performFetch:&error]) { |
/* |
Replace this implementation with code to handle the error appropriately. |
|
abort() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development. If it is not possible to recover from the error, display an alert panel that instructs the user to quit the application by pressing the Home button. |
*/ |
NSLog(@"Unresolved error %@, %@", error, [error userInfo]); |
abort(); |
} |
} |
|
|
#pragma mark - Recipe support |
|
- (void)recipeAddViewController:(RecipeAddViewController *)recipeAddViewController didAddRecipe:(Recipe *)recipe { |
|
if (recipe) { |
// show the recipe in the RecipeDetailViewController |
[self performSegueWithIdentifier:kShowRecipeSegueID sender:recipe]; |
} |
|
// dismiss the RecipeAddViewController |
[self dismissViewControllerAnimated:YES completion:nil]; |
} |
|
|
#pragma mark - UITableViewDataSource |
|
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView { |
|
NSInteger count = [self.fetchedResultsController sections].count; |
|
if (count == 0) { |
count = 1; |
} |
|
return count; |
} |
|
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section { |
|
NSInteger numberOfRows = 0; |
|
if ([self.fetchedResultsController sections].count > 0) { |
id <NSFetchedResultsSectionInfo> sectionInfo = [[self.fetchedResultsController sections] objectAtIndex:section]; |
numberOfRows = [sectionInfo numberOfObjects]; |
} |
|
return numberOfRows; |
} |
|
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { |
|
// dequeue a RecipeTableViewCell, then set its recipe to the recipe for the current row |
RecipeTableViewCell *recipeCell = |
(RecipeTableViewCell *)[tableView dequeueReusableCellWithIdentifier:@"MyIdentifier" forIndexPath:indexPath]; |
[self configureCell:recipeCell atIndexPath:indexPath]; |
|
return recipeCell; |
} |
|
- (void)configureCell:(RecipeTableViewCell *)cell atIndexPath:(NSIndexPath *)indexPath { |
|
Recipe *recipe = (Recipe *)[self.fetchedResultsController objectAtIndexPath:indexPath]; |
cell.recipe = recipe; |
} |
|
|
#pragma mark - UITableViewDelegate |
|
- (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender { |
|
if ([segue.identifier isEqualToString:kShowRecipeSegueID]) { |
// show a recipe |
// |
RecipeDetailViewController *detailViewController = (RecipeDetailViewController *)segue.destinationViewController; |
|
Recipe *recipe = nil; |
if ([sender isKindOfClass:[Recipe class]]) { |
// the sender is the actual recipe send from "didAddRecipe" delegate (user created a new recipe) |
recipe = (Recipe *)sender; |
} |
else { |
// the sender is ourselves (user tapped an existing recipe) |
NSIndexPath *indexPath = [self.tableView indexPathForSelectedRow]; |
recipe = (Recipe *)[self.fetchedResultsController objectAtIndexPath:indexPath]; |
} |
detailViewController.recipe = recipe; |
} |
else if ([segue.identifier isEqualToString:kAddRecipeSegueID]) { |
// add a recipe |
// |
Recipe *newRecipe = [NSEntityDescription insertNewObjectForEntityForName:@"Recipe" |
inManagedObjectContext:self.managedObjectContext]; |
|
UINavigationController *navController = segue.destinationViewController; |
RecipeAddViewController *addController = (RecipeAddViewController *)navController.topViewController; |
addController.delegate = self; // do didAddRecipe delegate method is called when cancel or save are tapped |
addController.recipe = newRecipe; |
} |
} |
|
// Override to support editing the table view. |
- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath { |
|
if (editingStyle == UITableViewCellEditingStyleDelete) { |
// Delete the managed object for the given index path |
NSManagedObjectContext *context = [self.fetchedResultsController managedObjectContext]; |
[context deleteObject:[self.fetchedResultsController objectAtIndexPath:indexPath]]; |
|
// Save the context. |
NSError *error; |
if (![context save:&error]) { |
/* |
Replace this implementation with code to handle the error appropriately. |
|
abort() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development. If it is not possible to recover from the error, display an alert panel that instructs the user to quit the application by pressing the Home button. |
*/ |
NSLog(@"Unresolved error %@, %@", error, [error userInfo]); |
abort(); |
} |
} |
} |
|
|
#pragma mark - Fetched results controller |
|
- (NSFetchedResultsController *)fetchedResultsController { |
|
// Set up the fetched results controller if needed. |
if (_fetchedResultsController == nil) { |
// Create the fetch request for the entity. |
NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init]; |
// Edit the entity name as appropriate. |
NSEntityDescription *entity = [NSEntityDescription entityForName:@"Recipe" inManagedObjectContext:self.managedObjectContext]; |
[fetchRequest setEntity:entity]; |
|
// Edit the sort key as appropriate. |
NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"name" ascending:YES]; |
NSArray *sortDescriptors = [[NSArray alloc] initWithObjects:sortDescriptor, nil]; |
|
[fetchRequest setSortDescriptors:sortDescriptors]; |
|
// Edit the section name key path and cache name if appropriate. |
// nil for section name key path means "no sections". |
NSFetchedResultsController *aFetchedResultsController = [[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest managedObjectContext:self.managedObjectContext sectionNameKeyPath:nil cacheName:@"Root"]; |
aFetchedResultsController.delegate = self; |
self.fetchedResultsController = aFetchedResultsController; |
} |
|
return _fetchedResultsController; |
} |
|
/** |
Delegate methods of NSFetchedResultsController to respond to additions, removals and so on. |
*/ |
- (void)controllerWillChangeContent:(NSFetchedResultsController *)controller { |
|
// The fetch controller is about to start sending change notifications, so prepare the table view for updates. |
[self.tableView beginUpdates]; |
} |
|
- (void)controller:(NSFetchedResultsController *)controller didChangeObject:(id)anObject atIndexPath:(NSIndexPath *)indexPath forChangeType:(NSFetchedResultsChangeType)type newIndexPath:(NSIndexPath *)newIndexPath { |
|
UITableView *tableView = self.tableView; |
|
switch(type) { |
case NSFetchedResultsChangeInsert: |
[tableView insertRowsAtIndexPaths:[NSArray arrayWithObject:newIndexPath] withRowAnimation:UITableViewRowAnimationFade]; |
break; |
|
case NSFetchedResultsChangeDelete: |
[tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade]; |
break; |
|
case NSFetchedResultsChangeUpdate: |
[self configureCell:(RecipeTableViewCell *)[tableView cellForRowAtIndexPath:indexPath] atIndexPath:indexPath]; |
break; |
|
case NSFetchedResultsChangeMove: |
[tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade]; |
[tableView insertRowsAtIndexPaths:[NSArray arrayWithObject:newIndexPath] withRowAnimation:UITableViewRowAnimationFade]; |
break; |
} |
} |
|
- (void)controller:(NSFetchedResultsController *)controller didChangeSection:(id <NSFetchedResultsSectionInfo>)sectionInfo atIndex:(NSUInteger)sectionIndex forChangeType:(NSFetchedResultsChangeType)type { |
|
switch(type) { |
case NSFetchedResultsChangeInsert: |
[self.tableView insertSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade]; |
break; |
|
case NSFetchedResultsChangeDelete: |
[self.tableView deleteSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade]; |
break; |
} |
} |
|
- (void)controllerDidChangeContent:(NSFetchedResultsController *)controller { |
|
// The fetch controller has sent all current change notifications, |
// so tell the table view to process all updates. |
[self.tableView endUpdates]; |
} |
|
@end |