42

通常,UISearchDisplayController 在激活时会使 tableView 变暗并聚焦 searchBar。一旦您在 searchBar 中输入文本,它就会创建一个 searchResultsTableView,显示在 searchBar 和键盘之间。当加载/显示/隐藏/卸载第二个 UITableView 时,将调用 searchDisplayController 的委托。通常它会在输入时显示实时搜索结果或自动完成条目。

在我的应用程序中,我想搜索一个网络服务,我不想为用户输入的每个字母调用网络服务。因此,我想完全禁用 searchResultsTableView 并在他输入文本时保持变暗的黑色覆盖。一旦他点击搜索按钮,我就会触发搜索(带有加载屏幕)。

只为 searchResultsTableView 返回零行看起来不太好,因为它显示一个空的 searchResultsTableView 并带有“无结果”消息。我试图在表格出现时隐藏它searchDisplayController:didLoadSearchResultsTableView:

除了从头开始重新创建 UISearchDisplayController 功能之外还有什么想法吗?

4

10 回答 10

38

这是我刚刚想出的一个小技巧,而且您在编辑搜索字符串时必须返回 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
}

我想你会弄清楚下一步该怎么做

于 2009-10-02T00:59:33.507 回答
19

Have you tried this:

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

This way, if the user types another char within 1/5sec, you only make one web call.

于 2009-09-10T16:51:14.087 回答
9

和你有同样的问题,我通过a)在开始搜索时将searchResultsTableView的alpha设置为0,然后b)将overlayView添加/删除到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
于 2012-01-30T15:28:48.627 回答
9

最后似乎没有任何效果,所以我想出了以下内容(当你准备好显示结果时,你必须调用 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];
}

显然,它使表格透明,添加与表格大小相同的黑色/半透明表格标题,并禁用表格上的滚动,因此您无法超过或超过标题。作为奖励,您可以在标题视图中添加一些内容(“请稍候...”或活动指示器)。

于 2011-09-03T17:00:01.657 回答
2

在您的 UISearchDisplayDelegate 中实现以下方法就足够了(通常是您的自定义 UITableViewController 子类)

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

你试过这个吗?

于 2009-08-01T13:25:58.947 回答
2

像这样简单地做这件事怎么样:

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

对我来说很好..

于 2012-11-04T20:37:22.260 回答
2

下面基于 user182820 的代码是我的版本。我隐藏了 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;
}   
于 2010-03-28T19:54:13.160 回答
1

我找到了一个更好的方法,因为“最佳答案”存在一个错误 - 滚动黑色背景表时将显示分隔符和“无结果”。

- (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 ....
}
于 2010-08-17T08:49:43.437 回答
1

所有现有的答案都过于复杂。您只需立即隐藏结果表视图即可逃脱。

-(void)searchDisplayController:(UISearchDisplayController *)controller didShowSearchResultsTableView:(UITableView *)tableView
{
    tableView.hidden = YES;
}
于 2014-06-24T17:15:17.913 回答
0

我想我为这个问题找到了更好的实现。所有先前的答案都正确显示了一个与搜索前 UITableView 相同的暗色视图,但是每个解决方案都缺少点击该区域以取消搜索的功能。

出于这个原因,我认为这段代码效果更好。

首先,创建一个诸如searchButtonTapped 之类的BOOL 来指示搜索按钮是否被点击。默认为否。

然后:

- (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 的可见选择,而是取消搜索。

于 2011-11-01T20:22:33.773 回答