/* * Copyright (c) 2013 Tobias Markmann * Licensed under the simplified BSD license. * See Documentation/Licenses/BSD-simplified.txt for more information. */ /* * Copyright (c) 2014 Kevin Smith and Remko Tronçon * Licensed under the GNU General Public License v3. * See Documentation/Licenses/GPLv3.txt for more information. */ #include #include #include #include #include #include #include #include #include #include #include namespace lambda = boost::lambda; namespace Swift { ContactSuggester::ContactSuggester() { } ContactSuggester::~ContactSuggester() { } void ContactSuggester::addContactProvider(ContactProvider* provider) { contactProviders_.push_back(provider); } bool ContactSuggester::matchContact(const std::string& search, const Contact::ref& c) { return fuzzyMatch(c->name, search) || fuzzyMatch(c->jid.toString(), search); } std::vector ContactSuggester::getSuggestions(const std::string& search) const { std::vector results; foreach(ContactProvider* provider, contactProviders_) { append(results, provider->getContacts()); } std::sort(results.begin(), results.end(), ContactSuggester::contactLexicographicalSortPredicate); results.erase(std::unique(results.begin(), results.end(), ContactSuggester::contactEqualityPredicate), results.end()); results.erase(std::remove_if(results.begin(), results.end(), !lambda::bind(&ContactSuggester::matchContact, search, lambda::_1)), results.end()); std::sort(results.begin(), results.end(), ContactSuggester::contactSmartSortPredicate); return results; } bool ContactSuggester::fuzzyMatch(std::string text, std::string match) { for (std::string::iterator currentQueryChar = match.begin(); currentQueryChar != match.end(); currentQueryChar++) { //size_t result = text.find(*currentQueryChar); std::string::iterator result = boost::algorithm::ifind_first(text, std::string(currentQueryChar, currentQueryChar+1)).begin(); if (result == text.end()) { return false; } text.erase(result); } return true; } bool ContactSuggester::contactLexicographicalSortPredicate(const Contact::ref& a, const Contact::ref& b) { return a->jid < b->jid; } bool ContactSuggester::contactEqualityPredicate(const Contact::ref& a, const Contact::ref& b) { return a->jid == b->jid; } bool ContactSuggester::contactSmartSortPredicate(const Contact::ref& a, const Contact::ref& b) { if (a->statusType == b->statusType) { return a->name.compare(b->name) < 0; } else { return a->statusType < b->statusType; } } }