Пользовательские основные данные SectionNameKeyPath - программирование
Подтвердить что ты не робот

Пользовательские основные данные SectionNameKeyPath

Я новичок в основных данных и пытаюсь понять, как создать пользовательский sectionNameKeyPath в моем NSFetchedResultsController. У меня есть управляемый объект с атрибутом acctPeriod. Это NSString. Я хочу создавать разделы на основе первых 4 символов этого поля. Первые 4 символа представляют год отчетного периода и не нуждаются в сохранении.

Я прошел через этот сайт и видел сообщения о переходных атрибутах, но я не могу заставить их работать. В основном я хочу это, а затем назначить periodYear для моего sectionNameKeyPath.

@dynamic periodYear;

-(NSString *)periodYear
{
    return [self.acctPeriod substringToIndex:4];
}

Любая помощь будет оценена.

** UPDATE: Используя ответ Мартина Р., я смог заставить его работать так, как ожидалось.

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

NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];

// Edit the entity name as appropriate.
NSEntityDescription *entity = [NSEntityDescription entityForName:@"Billing" 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:@"acctPeriod" ascending:NO];
NSArray *sortDescriptors = @[sortDescriptor];

//Predicate
NSPredicate *pred = [NSPredicate predicateWithFormat:@"clients = %@", self.client];
NSLog(@"%@",pred);

//[fetchRequest setResultType:NSDictionaryResultType];
//[fetchRequest setReturnsDistinctResults:YES];

[fetchRequest setPredicate:pred];
[fetchRequest setSortDescriptors:sortDescriptors];

NSFetchedResultsController *aFetchedResultsController = [[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest managedObjectContext:self.managedObjectContext sectionNameKeyPath:@"periodYear" cacheName:nil];
aFetchedResultsController.delegate = self;
self.fetchedResultsController = aFetchedResultsController;

NSError *error = nil;
if (![self.fetchedResultsController performFetch:&error])
{
    NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
    abort();
}

return _fetchedResultsController;  
}
4b9b3361

Ответ 1

Должно работать следующее: Внедрить метод periodYear (который будет использоваться как "путь ключа имени раздела" ) в расширении класса подкласса управляемого объекта:

@interface Event (AdditionalMethods)
- (NSString *)periodYear;
@end

@implementation Event (AdditionalMethods)
- (NSString *)periodYear {
    return [self.acctPeriod substringToIndex:4];
}
@end

Убедитесь, что acctPeriod используется как первый (или единственный) дескриптор сортировки для запроса выборки:

NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"acctPeriod" ascending:YES];
NSArray *sortDescriptors = @[sortDescriptor];
[fetchRequest setSortDescriptors:sortDescriptors];

Используйте periodYear как sectionNameKeyPath для выбранного контроллера результатов:

NSFetchedResultsController *_fetchedResultsController = [[NSFetchedResultsController alloc]
                  initWithFetchRequest:fetchRequest 
                   managedObjectContext:self.managedObjectContext 
                     sectionNameKeyPath:@"periodYear"
                              cacheName:nil];
_fetchedResultsController.delegate = self;
self.fetchedResultsController = _fetchedResultsController;

И, наконец, добавьте метод titleForHeaderInSection по умолчанию:

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

В качестве альтернативы вы можете определить periodYear как переходный атрибут управляемого объекта. Он также не будет храниться в базе данных в этом случае, но может быть реализован таким образом, что значение вычисляется по требованию и кэшируется.

Проект DateSectionTitles из библиотеки разработчиков Apple демонстрирует, как это работает.

Ответ 2

Я рекомендую не использовать свойство transient как sectionNameKeyPath, так как это приведет к сбою всех объектов, полученных с помощью запроса на выборку, чтобы свойство могло использоваться как путь раздела.
Лучше определить постоянное свойство year и использовать его как ваш sectionNameKeyPath.
настройте FRC sectionNameKeyPath на year следующим образом:

[[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest
                                    managedObjectContext:self.managedObjectContext
                                      sectionNameKeyPath:@"year"
                                               cacheName:nil/*your chioce*/];

чтобы отобразить название раздела в качестве заголовка в таблице, выполните:

- (NSString *)tableView:(UITableView *)tableView titleForHeaderInSection:(NSInteger)section
{
    id<NSFetchedResultsSectionInfo> sec = [self.fetchedResultsController sections][section];
    return [sec name];
}