Изменить NSFetchedResultsController при изменении SegmentedControl - PullRequest
5 голосов
/ 16 апреля 2011

У меня есть UITableView, который заполняется с использованием базовых данных NSFetchedResultsController. Теперь я добавил UISegmentedControl к представлению, и когда вы меняете текущий сегмент, я бы хотел, чтобы содержимое таблицы изменилось.

Я читал в Интернете, что было бы разумно использовать два разных NSFetchedResultsControllers, так как тогда я мог бы извлечь выгоду из встроенного кэширования. Единственная проблема в том, что я не могу найти пример кода для этого и не знаю, с чего начать.

Может ли кто-нибудь объяснить, с чего начать создание второго NSFetchedResultsController и изменение источников просмотра таблицы на основе сегментированного элемента управления?

Просмотреть код заголовка:

#import <CoreData/CoreData.h>

@interface DomainViewController : UIViewController <NSFetchedResultsControllerDelegate, UITableViewDataSource, UITableViewDelegate> {

    UITableView *domainView;
    UISegmentedControl *segmentedControl;
    NSString *domain;

}

@property (nonatomic, retain) NSFetchedResultsController *fetchedResultsController;
@property (nonatomic, retain) NSManagedObjectContext *managedObjectContext;
@property (nonatomic, retain) NSString *domain;

@property (nonatomic, retain) IBOutlet UITableView *domainView;
@property (nonatomic, retain) IBOutlet UISegmentedControl *segmentedControl;

- (IBAction)segmentedControlIndexChanged;

Просмотр кода реализации:

@interface DomainViewController ()
- (void)configureCell:(UITableViewCell *)cell atIndexPath:(NSIndexPath *)indexPath;
@end

@implementation DomainViewController

@synthesize fetchedResultsController = __fetchedResultsController;
@synthesize managedObjectContext = __managedObjectContext;
@synthesize domain;
@synthesize domainView;
@synthesize segmentedControl;

- (void)viewDidLoad
{
    [super viewDidLoad];

    if (self.managedObjectContext == nil) 
    { 
        self.managedObjectContext = [(GARankingsAppDelegate *)[[UIApplication sharedApplication] delegate] managedObjectContext]; 
    }

}

- (IBAction)segmentedControlIndexChanged
{
    switch(self.segmentedControl.selectedSegmentIndex){
        case 0:
            break;
        case 1:
            break;
        default:
            break;
    }
}

- (void)configureCell:(UITableViewCell *)cell atIndexPath:(NSIndexPath *)indexPath
{
    NSManagedObject *managedObject = [self.fetchedResultsController objectAtIndexPath:indexPath];
    cell.textLabel.text = [[managedObject valueForKey:@"Keyphrase"] description];
}

#pragma mark - Table view data source

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{
    return [[self.fetchedResultsController sections] count];
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
    id <NSFetchedResultsSectionInfo> sectionInfo = [[self.fetchedResultsController sections] objectAtIndex:section];
    return [sectionInfo numberOfObjects];
}

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    static NSString *CellIdentifier = @"Cell";

    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
    if (cell == nil) {
        cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier] autorelease];
    }

    // Configure the cell.
    [self configureCell:cell atIndexPath:indexPath];
    return cell;
}

- (NSFetchedResultsController *)fetchedResultsController
{
    if (__fetchedResultsController != nil)
    {
        return __fetchedResultsController;
    }

    /*
     Set up the fetched results controller.
     */
    // Create the fetch request for the entity.
    NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
    // Edit the entity name as appropriate.
    NSEntityDescription *entity = [NSEntityDescription entityForName:@"Result" inManagedObjectContext:self.managedObjectContext];
    [fetchRequest setEntity:entity];

    // Set the batch size to a suitable number.
    [fetchRequest setFetchBatchSize:20];

    // Edit the sort key as appropriate.
    NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"Keyphrase" ascending:NO];
    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;

    [aFetchedResultsController release];
    [fetchRequest release];
    [sortDescriptor release];
    [sortDescriptors release];

    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();
    }

    return __fetchedResultsController;
}

Любая помощь очень ценится. Благодаря.

ОБНОВЛЕНИЕ: обновленный код

Посмотреть код заголовка:

@property (nonatomic, retain) NSFetchedResultsController *currentFetchedResultsController;
@property (nonatomic, retain) NSFetchedResultsController *competitorFetchedResultsController;
@property (nonatomic, retain) NSFetchedResultsController *keyphraseFetchedResultsController;

Просмотр кода реализации:

@synthesize currentFetchedResultsController = __fetchedResultsController;
@synthesize competitorFetchedResultsController;
@synthesize keyphraseFetchedResultsController;

- (NSFetchedResultsController *)fetchedResultsController
{
    if (__fetchedResultsController != nil)
    {
        return __fetchedResultsController;
    }

    /*
     Set up the fetched results controller.
     */
    // Create the fetch request for the entity.
    NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
    // Edit the entity name as appropriate.
    NSEntityDescription *entity = [NSEntityDescription entityForName:@"Result" inManagedObjectContext:self.managedObjectContext];
    [fetchRequest setEntity:entity];

    // Set the batch size to a suitable number.
    [fetchRequest setFetchBatchSize:20];

    // Edit the sort key as appropriate.
    NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"Keyphrase" ascending:NO];
    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;
    self.currentFetchedResultsController = aFetchedResultsController;

    [aFetchedResultsController release];
    [fetchRequest release];
    [sortDescriptor release];
    [sortDescriptors release];

    NSError *error = nil;
    //if (![self.fetchedResultsController performFetch:&error])
    if (![self.currentFetchedResultsController 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();
    }

    return __fetchedResultsController;
} 

1 Ответ

8 голосов
/ 16 апреля 2011

Вы можете добавить другую переменную экземпляра, которая представляет текущий выбранный NSFetchedResultsController. И когда UISegmentedControl изменится, обновите этот ivar.

это может быть действие, которое вызывается событием изменения значения сегмента

- (IBAction *)segmentChanged:(UISegmentedControl *)sender {
    if ([sender selectedSegmentIndex] == 0) {
        self.currentFetchedResultsController = self.nsfrc1;
        [self.tableView reloadData];
    }
    else if ([sender selectedSegmentIndex] == 1) {
        self.currentFetchedResultsController = self.nsfrc2;     
        [self.tableView reloadData];
    }
}

один метод UITableViewDataSource в качестве примера:

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {
    return [[self.currentFetchedResultsController sections] count];
}

и вы должны убедиться, что только текущий nsfrc запускает обновление таблицы в методах NSFetchedResultsControllerDelegate. Таким образом, вы также должны изменить их все.

- (void)controllerWillChangeContent:(NSFetchedResultsController *)controller {
    if (controller == self.currentFetchedResultsController) {
        [self.tableView beginUpdates];
    } 
}

РЕДАКТИРОВАТЬ: Нет, вы делаете это неправильно. currentFetchedResultsController - это просто ivar без ленивого загрузчика. Это просто указатель на используемый контроллер.

Но два других fetchedResultsControllers должны иметь такой ленивый загрузчик загрузки.

- (NSFetchedResultsController *)competitorFetchedResultsController {
    if (!myCompetitorFetchedResultsController) {
        // create competitorFetchedResultsController
    }
    return myCompetitorFetchedResultsController;
}

- (NSFetchedResultsController *)keyphraseFetchedResultsController {
    if (!myKeyphraseFetchedResultsController) {
        // create keyphraseFetchedResultsController
    }
    return myKeyphraseFetchedResultsController;
}

и затем переключитесь с помощью:

self.currentFetchedResultsController = self.keyphraseFetchedResultsController;

или

self.currentFetchedResultsController = self.competitorFetchedResultsController;
...