UISearchDisplayController без результатов tableView?

Обычно UISearchDisplayController, когда он активирован, затемняет tableView и фокусирует панель поиска. Как только вы вводите текст в searchBar, он создает searchResultsTableView, который отображается между searchBar и клавиатурой. Делегат searchDisplayController вызывается, когда этот второй UITableView загружен / показан / скрыт / выгружен. Обычно он показывает результаты поиска в реальном времени или записи автозаполнения при наборе текста.

В моем приложении я хочу выполнить поиск в веб-сервисе и не хочу вызывать веб-сервис для каждого письма, которое вводит пользователь. Поэтому я хочу полностью отключить searchResultsTableView и оставить затемненный черный наложение, пока он вводит текст. Затем я запускаю поиск (с загрузочным экраном), как только он нажимает кнопку поиска.

Простой возврат нулевых строк для searchResultsTableView выглядит не очень хорошо, поскольку он отображает пустой searchResultsTableView с сообщением "нет результатов". Я пытался скрыть таблицу, когда она появляется (searchDisplayController:didLoadSearchResultsTableView:), который работает, но затемненное затемненное наложение также скрыто, так что базовый tableView снова полностью виден.

Любые идеи, кроме воссоздания функциональности UISearchDisplayController с нуля?

10 ответов

Решение

Вот небольшой трюк, который я только что выяснил, а также вы должны вернуть 0 результатов при редактировании строки поиска

- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchString:(NSString *)searchString
{
    savedSearchTerm = searchString;

    [controller.searchResultsTableView setBackgroundColor:[UIColor colorWithWhite:0.0 alpha:0.8]];
    [controller.searchResultsTableView setRowHeight:800];
    [controller.searchResultsTableView setScrollEnabled:NO];
    return NO;
}

- (void)searchDisplayController:(UISearchDisplayController *)controller didHideSearchResultsTableView:(UITableView *)tableView
{
    // undo the changes above to prevent artefacts reported below by mclin
}

я думаю, ты поймешь, что делать дальше

Вы пробовали это:

[NSObject cancelPreviousPerformRequestsWithTarget:self selector:@selector(_lookup:) object:nil];
[self performSelector:@selector(_lookup:) withObject:txt afterDelay:0.20];

Таким образом, если пользователь вводит другой символ в течение 1/5сек, вы делаете только один веб-вызов.

У меня была та же проблема, что и у вас, я справился с ней: а) установив альфа searchResultsTableView в 0 при начале поиска, а затем б) добавив / удалив оверлей в представлении viewController. Работает как шарм для меня.

@interface MyViewController()
//...
@property(nonatomic, retain) UIView *overlayView;
//...
@end

@implementation MyViewController
@synthesize overlayView = _overlayView;

//...

- (void)viewDidLoad
{
    //...

    //define your overlayView
    _overlayView = [[UIView alloc] initWithFrame:CGRectMake(0, 44, 320, 480)];
    _overlayView.backgroundColor = [UIColor colorWithRed:0 green:0 blue:0 alpha:0.8];
}

//hide the searchResultsTableView
- (void)searchDisplayControllerWillBeginSearch:(UISearchDisplayController *)controller
{
    self.searchDisplayController.searchResultsTableView.alpha = 0.0f;
}

//when ending the search, hide the overlayView
- (void) searchDisplayControllerWillEndSearch:(UISearchDisplayController *)controller
{
    [_overlayView removeFromSuperview];
}

//depending on what the user has inputed, add or remove the overlayView to the view of the current viewController 
- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchString:(NSString *)searchString
{   

    if ([searchString length]>0) 
    {
        [self.view addSubview:_overlayView];
    }
    else
    {
        [_overlayView removeFromSuperview];
    }

    return NO;
}

@end

Ничто из вышеперечисленного, похоже, не сработало в конце, поэтому я придумал следующее (вы должны вызвать removeTableHeader, когда будете готовы отобразить свои результаты):

- (BOOL)searchBarShouldBeginEditing:(UISearchBar *)searchBar {
    [self setTableHeader];
}

- (void)searchBar:(UISearchBar *)searchBar textDidChange:(NSString *)searchText {
    [self setTableHeader];
}

- (void)setTableHeader {
    UIView *headerView = [[UIView alloc] initWithFrame:self.searchDisplayController.searchResultsTableView.frame];
    headerView.backgroundColor = [UIColor colorWithWhite:0 alpha:0.8];

    [self.searchDisplayController.searchResultsTableView setBackgroundColor:[UIColor clearColor]];
    [self.searchDisplayController.searchResultsTableView setScrollEnabled:NO];
    [self.searchDisplayController.searchResultsTableView setTableHeaderView:headerView];

    [headerView release];
}

- (void)removeTableHeader {
    [self.searchDisplayController.searchResultsTableView setBackgroundColor:[UIColor whiteColor]];
    [self.searchDisplayController.searchResultsTableView setScrollEnabled:YES];
    [self.searchDisplayController.searchResultsTableView setTableHeaderView:nil];
}

Очевидно, что это делает таблицу прозрачной, добавляет черный / полупрозрачный заголовок таблицы того же размера, что и таблица, и отключает прокрутку таблицы, чтобы вы не могли подняться выше или выше заголовка. В качестве бонуса вы можете добавить что-нибудь в заголовок ("пожалуйста, подождите..." или индикатор активности).

Как насчет просто сделать это так просто:

- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchString:(NSString *)searchString
{
self.searchDisplayController.searchResultsTableView.hidden=YES;
return YES;
}

Работает нормально для меня..

Этого должно быть достаточно для реализации следующего метода в вашем UISearchDisplayDelegate (который обычно является вашим пользовательским подклассом UITableViewController)

- (BOOL) searchDisplayController: (UISearchDisplayController *) controller shouldReloadTableForSearchString: (NSString *) searchString
{
    [self startMyCustomWebserviceSearchAsBackgroundProcessForString: searchString]; //starts new NSThread
    return NO; 
}

ты пробовал это?

На основе кода пользователя 182820 ниже моя версия. Я скрываю табличное представление UISearchDisplayController. Когда в поле поиска вводится символ, я помещаю "затемненный вид", чтобы он выглядел так, как будто "затемненный вид" UISearchDisplayController никогда не исчезал, а затем удаляю его по окончании поиска. Если вы введете несколько символов и нажмете "Отмена", табличное представление на короткое время станет белым, и я не знаю, как обойти это.

- (void)viewDidLoad {
    ...
    tableViewMask=[UIView new];
    tableViewMask.backgroundColor = [UIColor blackColor];
    tableViewMask.alpha = 0.8;
}

- (void)searchDisplayControllerDidBeginSearch:(UISearchDisplayController *)controller{
    tableViewMask.frame=CGRectMake(self.tableView.frame.origin.x, self.tableView.frame.origin.y+controller.searchBar.frame.size.height, self.tableView.frame.size.width, self.tableView.frame.size.height-controller.searchBar.frame.size.height);
    controller.searchResultsTableView.hidden=YES;
}

- (void)searchDisplayControllerWillEndSearch:(UISearchDisplayController *)controller{
    [tableViewMask removeFromSuperview];
}

- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchString:(NSString *)searchString{
    if (searchString.length==0)
        [tableViewMask removeFromSuperview];
    else 
        [self.tableView addSubview:tableViewMask];
    [searchText autorelease];
    searchText=[searchString retain];
    return NO;
}   

Я предпочитаю лучший способ, так как есть ошибка с "лучшим ответом" - разделитель и "Нет результатов" будут показаны при прокрутке таблицы черного фона.

- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchString:(NSString *)searchString {

    controller.searchResultsTableView.backgroundColor = [UIColor blackColor];
    controller.searchResultsTableView.alpha = 0.8;
    controller.searchResultsTableView.separatorStyle = UITableViewCellSeparatorStyleNone;

    for(UIView *subview in tableView.subviews) {
        if([subview isKindOfClass:UILabel.class]) {
            subview.hidden = YES;
        }
    }

    return NO;
}

- (void) searchBarSearchButtonClicked:(UISearchBar *)searchBar {

    self.searchDisplayController.searchResultsTableView.backgroundColor = [UIColor whiteColor];
    self.searchDisplayController.searchResultsTableView.alpha = 1;
    self.searchDisplayController.searchResultsTableView.separatorStyle = UITableViewCellSeparatorStyleSingleLine;

    for(UIView *subview in tableView.subviews) {
        if([subview isKindOfClass:UILabel.class]) {
            subview.hidden = NO;
        }
    }

    // search results and reload data ....
}

Все существующие ответы слишком сложны. Вы можете уйти, просто сразу скрыв представление таблицы результатов.

-(void)searchDisplayController:(UISearchDisplayController *)controller didShowSearchResultsTableView:(UITableView *)tableView
{
    tableView.hidden = YES;
}

Я думаю, что нашел лучшую реализацию для этой проблемы. Все предыдущие ответы правильно показывают тусклое представление, идентичное тому, как выглядит UITableView перед поиском, но в каждом решении отсутствует функция касания области для отмены поиска.

По этой причине я думаю, что этот код работает лучше.

Прежде всего, создайте BOOL, например searchButtonTapped, чтобы указать, была ли нажата кнопка поиска. По умолчанию это НЕТ.

Затем:

- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchString:(NSString *)searchString {
if (!searchButtonTapped) {        
    // To prevent results from being shown and to show an identical look to how the tableview looks before a search
    [controller.searchResultsTableView setBackgroundColor:[UIColor clearColor]];
    [controller.searchResultsTableView setRowHeight:160];
    self.searchDisplayController.searchResultsTableView.scrollEnabled = NO;
} else {
    // Restore original settings
    [controller.searchResultsTableView setBackgroundColor:[UIColor whiteColor]];
    [controller.searchResultsTableView setRowHeight:44];
    self.searchDisplayController.searchResultsTableView.scrollEnabled = YES;
}

return YES;
}

Это должно быть ясно сейчас, основываясь на других ответах. Не забудьте также восстановить исходные настройки, когда пользователь нажимает кнопку "Поиск".

Кроме того, в методе cellForIndexPath добавьте:

cell.selectionStyle = UITableViewCellSelectionStyleNone;
cell.contentView.backgroundColor = [UIColor colorWithWhite:0.0 alpha:0.8];

Для создания того же затемненного вида, который отображается перед вводом текста. Убедитесь, что вы применили эти свойства к правой ячейке, то есть проверьте, какой UITableView активен, и что пользователь не нажал кнопку "Поиск".

Затем, что особенно важно, в didSelectRowAtIndexPath:

if (tableView == self.searchDisplayController.searchResultsTableView) {
    if (searchButtonTapped) {
           // Code for when the user select a row after actually having performed a search
    {
else 
    [self.searchDisplayController setActive:NO animated:YES];

Теперь пользователь может коснуться затемненной области, что не приведет к видимому выделению UITableViewCell, а вместо этого отменит поиск.

Другие вопросы по тегам