NSFetchedResultsController с поиском
-
07-07-2019 - |
Вопрос
Какова наилучшая практика для фильтрации NSFetchedResultsController
данные?нужно ли мне повторно инициализировать его каждый раз, когда меняется текст на панели поиска?
Я использую UISearchDisplayControllers
и я внедряю:
- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchString:(NSString *)searchString;
Спасибо.
Решение
Чем код ответа Парня отличается от самого вопроса?Насколько я могу догадаться, метод filterContentForSearchText:scope вызывается методами shouldReload?
В любом случае, вот несколько похожих кодов, которые я добавил в пример CoreDataBooks, чтобы включить поиск.Добавьте контроллер отображения поиска в IB для примера CoreDataBooks.Затем я добавил код в RootViewController.m следующим образом:
- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchString:(NSString *)searchString {
NSInteger searchOption = controller.searchBar.selectedScopeButtonIndex;
return [self searchDisplayController:controller shouldReloadTableForSearchString:searchString searchScope:searchOption];
}
- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchScope:(NSInteger)searchOption {
NSString* searchString = controller.searchBar.text;
return [self searchDisplayController:controller shouldReloadTableForSearchString:searchString searchScope:searchOption];
}
- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchString:(NSString*)searchString searchScope:(NSInteger)searchOption {
NSPredicate *predicate = nil;
if ([searchString length])
if (searchOption == 0) // full text, in my implementation. Other scope button titles are "Author", "Title"
predicate = [NSPredicate predicateWithFormat:@"title contains[cd] %@ OR author contains[cd] %@", searchString, searchString];
else
// docs say keys are case insensitive, but apparently not so.
predicate = [NSPredicate predicateWithFormat:@"%K contains[cd] %@", [[controller.searchBar.scopeButtonTitles objectAtIndex:searchOption] lowercaseString], searchString];
[fetchedResultsController.fetchRequest setPredicate:predicate];
NSError *error = nil;
if (![[self fetchedResultsController] performFetch:&error]) {
NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
abort();
}
return YES;
}
PS.Чтобы ответить Vivas, используя UISearchDisplayController, он автоматически создает новое табличное представление для наложения отфильтрованного списка.Вы можете проверить, какой TableView используется, как показано в документах, но при простейшей настройке это работает просто потому, что fetchedResultsController либо показывает отфильтрованную версию в табличном представлении поиска, либо показывает все данные в вашем табличном представлении.
Другие советы
По-видимому, это лучший способ:
- (void)filterContentForSearchText:(NSString*)searchText scope:(NSString*)scope
{
self.savedSearchTerm = searchText;
freshData = NO;
if (searchText !=nil)
{
NSPredicate *predicate =[NSPredicate predicateWithFormat:@"name contains[cd] %@", searchText];
[fetchedResultsController.fetchRequest setPredicate:predicate];
}
else
{
NSPredicate *predicate =[NSPredicate predicateWithFormat:@"All"];
[fetchedResultsController.fetchRequest setPredicate:predicate];
}
NSError *error = nil;
if (![[self fetchedResultsController] performFetch:&error]) {
// Handle error
NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
exit(-1); // Fail
}
[self.tableView reloadData];
// [searchBar resignFirstResponder];
// [_shadeView setAlpha:0.0f];
}