0

テーブルビューの検索バーに少し問題があります

配列tableViewArrayでテーブルビューを作成しました。
tableViewArray は、[テキスト、距離] で構成される別の配列の多くの行で構成されます。すべてが正常に機能しています。
ここで、(tableViewArray の「テキスト」オブジェクトからの) 文字列の新しい配列に基づいて検索する searchBar と searchdisplaycontroller を追加しました。検索はテキストに対してのみ利用可能であるべきだと考え、検索方法はその上に実装されています。

検索結果を取得すると、見栄えがよく、検索により期待される行が返されます。問題は、検索 tableViews サブタイトルにあります。tableViewArray の行 1、2、3 の距離を示しています。

検索テーブルビューの行に表示されるテキストへの距離をマッピングする必要があります。[テキスト距離] で構成される検索結果に対して、新しいテーブル ビュー配列を作成する必要があると思います。テキストは検索結果のものなので問題ありませんが、新しい距離を古い距離にマッピングするにはどうすればよいですか???

検索デリゲートで使用している検索方法は次のとおりです。

searchResults = [[NSArray alloc]init];
NSPredicate *resultPredicate = [NSPredicate predicateWithFormat:@"SELF contains[cd] %@",
                            searchText];
searchResults = [searchItems  filteredArrayUsingPredicate:resultPredicate];

誰かが助けてくれることを願っています:) よろしくお願いします!

元のコード:

- (NSInteger)tableView:(UITableView *)theTableView numberOfRowsInSection:(NSInteger)section
{

    if (theTableView == self.searchDisplayController.searchResultsTableView) {
        return [searchResults count];
    } else {
        return [tableViewArray count];
    }
}  

- (UITableViewCell *)tableView:(UITableView *)theTableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    static NSString *CellIdentifier = @"Cell";

    UITableViewCell *cell = [[UITableViewCell alloc] initWithStyle:     UITableViewCellStyleSubtitle
                                  reuseIdentifier: CellIdentifier] ;

     if (theTableView == self.searchDisplayController.searchResultsTableView) {

    } else {
         cell = [theTableView dequeueReusableCellWithIdentifier:CellIdentifier     forIndexPath:indexPath];
}

    // Configure the cell...
    // cell.textLabel.numberOfLines = 0;
   // cell.textLabel.lineBreakMode = UILineBreakModeWordWrap;

    if (theTableView == self.searchDisplayController.searchResultsTableView) {
        cell.textLabel.text = [searchResults objectAtIndex:indexPath.row];
    } else {
        cell.textLabel.text = [[[tableViewArray objectAtIndex:indexPath.row]     objectAtIndex:0] subtitle];
}

    cell.detailTextLabel.textColor = [UIColor redColor];




float blabla=  [[[tableViewArray objectAtIndex:indexPath.row] objectAtIndex: 1] doubleValue];
if (blabla < 1000) {
         cell.detailTextLabel.text = [NSString stringWithFormat:@"%.2f m",blabla];
} else {
         cell.detailTextLabel.text = [NSString stringWithFormat:@"%.2f km",blabla/1000];
}





NSString *text = [[[tableViewArray objectAtIndex:indexPath.row] objectAtIndex:0] subtitle];
NSRange q8RangeValue = [text rangeOfString:@"Q8" options:NSCaseInsensitiveSearch];
NSRange okRangeValue = [text rangeOfString:@"OK" options:NSCaseInsensitiveSearch];

if (q8RangeValue.length >0  ) {
        cell.imageView.image = [UIImage imageNamed:@"q8.png"];
} else if (okRangeValue.length >0 ) {
        cell.imageView.image = [UIImage imageNamed:@"OK logo.png"];

} else {
        cell.imageView.image = nil;
}

return cell;
}

そして、検索用の配列を作成する場所:

- (void)locationManager:(CLLocationManager *)manager didUpdateToLocation:(CLLocation *)newLocation fromLocation:(CLLocation *)oldLocation {

NSLog(@"didUpdateToLocation Location calculator distance array for the tableView");

NSMutableArray * distancesInReverseOrder = [[NSMutableArray alloc] init];



for (int i = 0; i<allAnnotations.count; i++) {

NSIndexPath *indexPath = [NSIndexPath indexPathForRow:i inSection:0] ;
CLLocationCoordinate2D annotationCoord = [[allAnnotations objectAtIndex:indexPath.row] coordinate];


CLLocation *location = [[CLLocation alloc] initWithLatitude:annotationCoord.latitude  longitude:annotationCoord.longitude];

distanceToMe = [newLocation distanceFromLocation:location];

[distancesInReverseOrder insertObject:[NSNumber numberWithFloat: distanceToMe] atIndex:0];

}


distances = [[NSMutableArray alloc] initWithArray:[[distancesInReverseOrder reverseObjectEnumerator] allObjects]];

// Assuming you have your points on the map in an NSArray called
// allAnnotations and your distances in distances, create a
// new mutable array to hold both
tableViewArray = [[NSMutableArray alloc] init];

// Iterate over all of the points, and add a new element to the mutable
// array which is a new array containing a point and its distance
for (int i = 0; i < allAnnotations.count; i++) {
    NSArray *newItem = [NSArray arrayWithObjects: [allAnnotations objectAtIndex: i], [distances objectAtIndex: i], nil];
    [tableViewArray addObject: newItem];
}

// Now, sort the new array based upon the distance in the second element
// of each array (ie, the distance).
[tableViewArray sortUsingComparator: ^(id obj1, id obj2) {
    NSNumber *dist1 = [obj1 objectAtIndex:1];
    NSNumber *dist2 = [obj2 objectAtIndex:1];

    return [dist1 compare:dist2];
}];

searchResults = [NSMutableArray arrayWithCapacity:[tableViewArray count]];




searchItems = [[NSMutableArray alloc] init];

for (int i = 0; i < allAnnotations.count; i++) {
    NSArray *newItem = [NSArray arrayWithObjects: [[[tableViewArray objectAtIndex:i] objectAtIndex:0] subtitle], @"bla", nil];
    [searchItems addObject: newItem];
}

/*
for (int i=0; i<tableViewArray.count; i++) {
    [searchItems insertObject:[[[tableViewArray objectAtIndex:i] objectAtIndex:0] subtitle] atIndex:0];
}
*/

NSLog(@"searchitems count is %i", searchItems.count);

[tableView reloadData];

[locationManager stopUpdatingLocation];

}

    - (void)filterContentForSearchText:(NSString*)searchText scope:(NSString*)scope
{
 /*
// Update the filtered array based on the search text and scope.
// Remove all objects from the filtered search array
[searchResults removeAllObjects];
// Filter the array using NSPredicate
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF.name contains[c] %@",searchText];

// searchResults = [NSMutableArray arrayWithArray:[[tableViewArray objectAtIndex:1]filteredArrayUsingPredicate:predicate]];

*/

/*

searchResults = [[NSArray alloc]init];

NSPredicate *resultPredicate = [NSPredicate
                                predicateWithFormat:@"SELF contains[cd] %@",
                                searchText];

    searchResults =[searchItems filteredArrayUsingPredicate:resultPredicate];


*/

// Create index set of all objects in textArray that contain searchText:
NSIndexSet *set = [searchItems indexesOfObjectsPassingTest:
                   ^BOOL(NSString *text, NSUInteger idx, BOOL *stop) {
                       NSRange range = [text rangeOfString:searchText options:(NSCaseInsensitiveSearch|NSDiacriticInsensitiveSearch)];
                       return (range.location != NSNotFound);
                   }];

// Filter textArray:
filteredTextArray = [searchItems objectsAtIndexes:set];
// Filter distanceArray:
filteredDistanceArray = [distances objectsAtIndexes:set];

NSLog(@"filtered text array is %@", filteredTextArray);
NSLog(@"filtered distance array is %@",filteredDistanceArray);
4

1 に答える 1

2

私があなたの問題を正しく理解していれば、テーブル ビューのデータ ソースとして使用される 2 つの別個の配列があり、それらtextArrayを と と呼びましょうdistanceArray

textArray検索文字列に従って をフィルタリングすると、 の「対応する」フィルタリングが必要になりますdistanceArray

filteredArrayUsingPredicateこれを行う 1 つの方法は、をに置き換えることです。これは、indexesOfObjectsPassingTest両方の配列に適用できる一致するインデックスのセットを返すためです。

// Create index set of all objects in textArray that contain searchText:
NSIndexSet *set = [textArray indexesOfObjectsPassingTest:
    ^BOOL(NSString *text, NSUInteger idx, BOOL *stop) {
        NSRange range = [text rangeOfString:searchText options:(NSCaseInsensitiveSearch|NSDiacriticInsensitiveSearch)];
        return (range.location != NSNotFound);
    }];

// Filter textArray:
filteredTextArray = [textArray objectsAtIndexes:set];
// Filter distanceArray:
filteredDistanceArray = [distanceArray objectsAtIndexes:set];

検索テーブル ビューのデータ ソースとしてfilteredTextArrayとを使用できるようになりました。filteredDistanceArray

または、配列内の各オブジェクトが、たとえば 1 行のテキストと距離の両方を含む辞書である場合、単一の配列をデータ ソースとして使用できます。

更新:私が理解しているように、あなたの各アイテムはtableViewArray2つのアイテム(テキスト用と距離用)の配列です。

この場合、tableViewArray直接フィルタリングすることをお勧めします。

NSIndexSet *set = [tableViewArray indexesOfObjectsPassingTest:
                   ^BOOL(NSArray *item, NSUInteger idx, BOOL *stop) {
                       NSString *subtitle = [[item objectAtIndex:0] subtitle];
                       NSRange range = [subtitle rangeOfString:searchText options:(NSCaseInsensitiveSearch|NSDiacriticInsensitiveSearch)];
                       return (range.location != NSNotFound);
                   }];

searchResults = [tableViewArray objectsAtIndexes:set];

searchResultsこれでフィルター処理された配列になり、各アイテムは のアイテムと同じ構造を持ちますtableViewArray

これによりcellForRowAtIndexPath、たとえば、

if (theTableView == self.searchDisplayController.searchResultsTableView) {
    cell.textLabel.text = [[[searchResults objectAtIndex:indexPath.row] objectAtIndex:0] subtitle];
     distance = [[[searchResults objectAtIndex:indexPath.row] objectAtIndex: 1] doubleValue];
} else {
    cell.textLabel.text = [[[tableViewArray objectAtIndex:indexPath.row] objectAtIndex:0] subtitle];
    distance = [[[tableViewArray objectAtIndex:indexPath.row] objectAtIndex: 1] doubleValue];
}
于 2012-09-19T13:06:11.243 に答える