Pull to refresh

Динамический поиск строки в iOS

Reading time2 min
Views10K


Недавно появилась задача — встроить в приложение динамический поиск информации, то есть выдача должна генерироваться параллельно вводу пользователем символов. Вся информация хранится в SQLite базе данных. Все бы ничего, но в каждом слове выдачи искомые символы должны подсвечиваться.
Было принято волевое решение не использовать внешние библиотеки, а попробовать обойтись своими силами.

Начнем с внешней формы.



В качестве поисковой строки используется UISearchBar. Результаты выводятся в обычный UITableView (в данном случае, с ячейками, кроме текста включающими в себя и картинку).
Обработчиком для события изменения текста в строке является:

- (void)searchBar:(UISearchBar *)searchBar textDidChange:(NSString *)searchText
{
    _data = [SBHotspotData findHotspot:searchText];
    [self.searchResultsTableView reloadData];
}

где _data — NSArray, хранящий результаты выдачи.

На этом этапе возникла проблема — во всех названиях объектов, по которым проводится поиск, есть как заглавные буквы, так и строчные, которые SQLite воспринимает по-разному, а функции lower() и upper(), в отличии от SQL, не поддерживаются. Путем наименьшего сопротивления решили добавить в базу данных отдельный столбец, хранящий в себе все те же заголовки, но уже в нижнем регистре, тем более, что в данном случае увеличение объема базы было практически незаметно. Метод поиска (для работы с БД используется библиотека FMDatabase):

+ (NSArray *) findHotspot: (NSString *)partOfTitle
{
    SBHotspotData *item;
    NSMutableArray *result = [NSMutableArray array];
    
    NSString *path = [self getDatabasePath];
    FMDatabase *database;
    database = [FMDatabase databaseWithPath:path];
    [database open];
    
    NSString *query = [NSString stringWithFormat:@"select * from hotspots where TitleLow like '%%%@%%'", [partOfTitle lowercaseString]];
    FMResultSet *results = [database executeQuery:query];
    
    while([results next]) {
        item = [[SBHotspotData alloc ]init];
        item.hotspotIdentity = [results stringForColumn:@"Identity"];
        item.hotspotTitle = [results stringForColumn:@"Title"];
        item.hotspotDescription = [results stringForColumn:@"Description"];
        [result addObject:item];
    }
        
    [database close];

    return [result sortedArrayUsingSelector:@selector(compare:)];
}

- (NSComparisonResult)compare:(SBHotspotData *)otherObject {
    return [self.hotspotTitle compare:otherObject.hotspotTitle];
}

При формировании запроса используется %%%@%%, где знаками процента обозначается, что перед и после строки может идти любое количество символов. Метод возвращает массив объектов из базы данных, отсортированный по алфавиту.
Самая интересная часть задания — подсветка введенных символов. Реализуется она следующим образом:

NSMutableAttributedString *attributedString =
[[NSMutableAttributedString alloc] initWithString:item.hotspotTitle];

NSRange range = [item.hotspotTitle rangeOfString:_searchHotspotBar.text];
[attributedString addAttribute:NSBackgroundColorAttributeName
                   value:[UIColor colorWithRed:216/255.0f green:87/255.0f blue:23/255.0f alpha:1.0f]
                   range:range];

cell.cellLabel.attributedText = attributedString;

Для подсветки определенных символов используется NSMutableAttributedString и его метод addAttribute:value:range:.

Ну и результат работы программы:



Задача получилась достаточно тривиальной, но, надеюсь, полученный опыт пригодится кому-либо из хабровчан.
Tags:
Hubs:
Total votes 14: ↑10 and ↓4+6
Comments7

Articles