Как стать автором
Обновить

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

Время на прочтение 2 мин
Количество просмотров 9.9K


Недавно появилась задача — встроить в приложение динамический поиск информации, то есть выдача должна генерироваться параллельно вводу пользователем символов. Вся информация хранится в 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:.

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



Задача получилась достаточно тривиальной, но, надеюсь, полученный опыт пригодится кому-либо из хабровчан.
Теги:
Хабы:
+6
Комментарии 7
Комментарии Комментарии 7

Публикации

Истории

Работа

Swift разработчик
38 вакансий
iOS разработчик
23 вакансии

Ближайшие события

Московский туристический хакатон
Дата 23 марта – 7 апреля
Место
Москва Онлайн
Геймтон «DatsEdenSpace» от DatsTeam
Дата 5 – 6 апреля
Время 17:00 – 20:00
Место
Онлайн