2

単語のインクリメンタル検索を実装したいと思います。以下のように実装しましたが、filterContentForSearchText と shouldReloadTableForSearchString メソッドが呼び出されません。なんで?

WordsIndexViewController.h

#import <UIKit/UIKit.h>

@interface WordsIndexViewController : UIViewController <UITableViewDataSource, UITableViewDelegate, UISearchDisplayDelegate, UISearchBarDelegate>

@end

WordsIndexViewController.m

#import "WordsIndexViewController.h"
#import "WordsShowViewController.h"
#import "Word.h"

@interface WordsIndexViewController ()

@property (strong, nonatomic) UITableView *tableView;

@property (strong, nonatomic) NSArray *words;
@property (strong, nonatomic) NSMutableArray *searchedWords;

@end

@implementation WordsIndexViewController

- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
{
    self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
    if (self) {
        // Custom initialization
    }
    return self;
}

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view.

    _words = [[app models] objectForKey:@"words"];
    _searchedWords = [NSMutableArray arrayWithArray:@[]];

    UITableView *tableView = [[UITableView alloc] initWithFrame:CGRectMake(0, 0, 320, 480) style:UITableViewStylePlain];
    tableView.delegate = self;
    tableView.dataSource = self;
    _tableView = tableView;

    UISearchBar *searchBar = [[UISearchBar alloc] initWithFrame:CGRectMake(0, 0, self.view.bounds.size.width, 44)];
    searchBar.delegate = self; // needed?
    [searchBar sizeToFit]; // needed?

    UISearchDisplayController *searchDisplayController = [[UISearchDisplayController alloc] initWithSearchBar:searchBar contentsController:self];
    searchDisplayController.delegate = self;
    searchDisplayController.searchResultsDelegate = self;
    searchDisplayController.searchResultsDataSource = self;
    _tableView.tableHeaderView = searchBar;

    [self.view addSubview:_tableView];
}

- (void)didReceiveMemoryWarning
{
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

#pragma mark - related to UITableView
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
    NSLog(@"START numberOfRowsInSection");

    NSArray *words = nil;
    if (tableView == self.searchDisplayController.searchResultsTableView) {
        words = _searchedWords;
    } else {
        words = _words;
    }

    return [words count];
}
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    NSLog(@"START cellForRowAtIndexPath");

    NSArray *words = nil;
    if (tableView == self.searchDisplayController.searchResultsTableView) {
        words = _searchedWords;
    } else {
        words = _words;
    }

    UITableViewCell *cell = [[UITableViewCell alloc] init];
    Word *word = words[indexPath.row];
    cell.textLabel.text = word.spelling;
    cell.accessoryType = UITableViewCellAccessoryDisclosureIndicator;

    return cell;
}
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
    NSArray *words = nil;
    if (tableView == self.searchDisplayController.searchResultsTableView) {
        words = _searchedWords;
    } else {
        words = _words;
    }

    WordsShowViewController *controller = [[WordsShowViewController alloc] initWithNibName:@"ViewController" bundle:nil];
    controller.word = (Word *)words[indexPath.row];
    [self.navigationController pushViewController:controller animated:YES];
}

#pragma mark - related to Search Bar
- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchString:(NSString *)searchString {
    NSLog(@"Query: %@", searchString); // NOT CALLED...

    [self filterContentForSearchText:searchString
        scope:[
            [self.searchDisplayController.searchBar scopeButtonTitles]
            objectAtIndex:[self.searchDisplayController.searchBar selectedScopeButtonIndex]
        ]
    ];

    return YES;
}
- (void)filterContentForSearchText:(NSString*)searchString scope:(NSString*)scope {
    NSLog(@"START filterContentForSearchText"); // NOT CALLED...

    [_searchedWords removeAllObjects];

    for(Word *word in _words) {
        NSRange range = [word.spelling rangeOfString:searchString options:NSCaseInsensitiveSearch];
        if(range.length > 0) {
            [_searchedWords addObject:word];
        }
    }
}

@end

Word.h

#import <Foundation/Foundation.h>

@interface Word : NSObject

@property (strong, nonatomic) NSString *identifier;
@property (strong, nonatomic) NSString *spelling;

- (id)initWith:(NSString *)spelling;

@end

Word.m

#import "Word.h"

@implementation Word

- (id)initWith:(NSString *)spelling {
    self = [super init];
    if (!self) {
        return nil;
    }

    CFUUIDRef uuid = CFUUIDCreate(NULL);
    _identifier = (NSString *)CFBridgingRelease(CFUUIDCreateString(NULL, uuid));
    CFRelease(uuid);

    _spelling = spelling;

    return self;
}

@end

よろしく、

4

2 に答える 2

1

最後に、私は自分で質問を解決しました。

それを解決するために、textDidChange で shouldReloadTableForSearchString を呼び出すと、インクリメンタル検索が機能しました。以下の差分:

WordsIndexViewController.m

@@ -16,6 +16,7 @@
      @property (strong, nonatomic) NSArray *words;
 @property (strong, nonatomic) NSMutableArray *searchedWords;
+@property (strong, nonatomic) UISearchDisplayController *displayController;

 @end

@@ -52,6 +53,7 @@
     searchDisplayController.searchResultsDelegate = self;
     searchDisplayController.searchResultsDataSource = self;
     _tableView.tableHeaderView = searchBar;
+    _displayController = searchDisplayController;

     [self.view addSubview:_tableView];
 }
@@ -132,5 +134,9 @@
         }
     }
 }
+- (void) searchBar:(UISearchBar *)searchBar textDidChange:(NSString *) searchText {
+    [self searchDisplayController:_displayController shouldReloadTableForSearchString:searchText];
+}

 @end

ありがとう、

于 2013-05-27T22:39:08.303 に答える
1

ヘッダー ファイルで UISearchDisplayController の iVar を必ず作成してください。

以下を使用して UISearchDisplayController を作成する場合:

UISearchDisplayController* searchDisplayController = [[UISearchDisplayController alloc] initWithSearchBar:searchField contentsController:self];

ARCによってリリースされ、デリゲートメソッドは呼び出されず、self.searchDisplayController(UIViewControllerのプロパティ)を呼び出すと、 nil.

したがって、修正は次のとおりです。ヘッダー(.h)ファイルで:

@interface MenuViewController : UIViewController <UITableViewDataSource, UITableViewDelegate, UISearchBarDelegate, UISearchDisplayDelegate> {
        UISearchDisplayController* searchDisplayController;
        UISearchBar *searchField;
        UITableView* tableView;
        NSArray* searchResults;
}

および実装 (.m) ファイルで:

searchField = [[UISearchBar alloc] initWithFrame:CGRectMake(0, 0, self.view.frame.size.width, 49)];
searchField.delegate = self;

searchDisplayController = [[UISearchDisplayController alloc] initWithSearchBar:searchField contentsController:self];
searchDisplayController.delegate = self;
searchDisplayController.searchResultsDataSource = self;
searchDisplayController.searchResultsDelegate = self;

tableView.tableHeaderView = searchField;
tableView.contentOffset = CGPointMake(0, searchField.frame.size.height);

そのように実装すると、コードの残りの部分でself.searchDisplayControllerとの両方を呼び出すことができます。searchDisplayController

于 2013-06-09T14:36:12.317 に答える