Пользовательские основные данные 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;
}
Ответы
Ответ 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];
}