#include #include #include "../../utils.hpp" #include "../core/CoreManager.hpp" #include "ContactsListProxyModel.hpp" #define USERNAME_WEIGHT 50.f #define SIP_ADDRESSES_WEIGHT 50.f #define FACTOR_POS_0 1.0f #define FACTOR_POS_1 0.9f #define FACTOR_POS_2 0.8f #define FACTOR_POS_3 0.7f #define FACTOR_POS_OTHER 0.6f using namespace std; // ============================================================================= // Notes: // // - First `^` is necessary to search two words with one separator // between them like `Claire Manning`. // // - [^_.-;@ ] is used to search patterns which starts with // a separator like ` word`. // // - [_.-;@ ] is the main pattern (a separator). const QRegExp ContactsListProxyModel::m_search_separators("^[^_.-;@ ][_.-;@ ]"); // ----------------------------------------------------------------------------- ContactsListProxyModel::ContactsListProxyModel (QObject *parent) : QSortFilterProxyModel(parent) { setSourceModel(CoreManager::getInstance()->getContactsListModel()); sort(0); } // ----------------------------------------------------------------------------- void ContactsListProxyModel::setFilter (const QString &pattern) { m_filter = pattern; invalidate(); } // ----------------------------------------------------------------------------- bool ContactsListProxyModel::filterAcceptsRow ( int source_row, const QModelIndex &source_parent ) const { const QModelIndex &index = sourceModel()->index(source_row, 0, source_parent); const ContactModel *contact = index.data().value(); m_weights[contact] = static_cast(round(computeContactWeight(*contact))); return m_weights[contact] > 0 && ( !m_use_connected_filter || contact->getPresenceLevel() != Presence::PresenceLevel::White ); } bool ContactsListProxyModel::lessThan (const QModelIndex &left, const QModelIndex &right) const { const ContactModel *contact_a = sourceModel()->data(left).value(); const ContactModel *contact_b = sourceModel()->data(right).value(); unsigned int weight_a = m_weights[contact_a]; unsigned int weight_b = m_weights[contact_b]; // Sort by weight and name. return weight_a > weight_b || ( weight_a == weight_b && contact_a->m_linphone_friend->getName() <= contact_b->m_linphone_friend->getName() ); } // ----------------------------------------------------------------------------- float ContactsListProxyModel::computeStringWeight (const QString &string, float percentage) const { int index = -1; int offset = -1; // Search pattern. while ((index = string.indexOf(m_filter, index + 1, Qt::CaseInsensitive)) != -1) { // Search n chars between one separator and index. int tmp_offset = index - string.lastIndexOf(m_search_separators, index) - 1; if ((tmp_offset != -1 && tmp_offset < offset) || offset == -1) if ((offset = tmp_offset) == 0) break; } switch (offset) { case -1: return 0; case 0: return percentage * FACTOR_POS_0; case 1: return percentage * FACTOR_POS_1; case 2: return percentage * FACTOR_POS_2; case 3: return percentage * FACTOR_POS_3; default: break; } return percentage * FACTOR_POS_OTHER; } float ContactsListProxyModel::computeContactWeight (const ContactModel &contact) const { float weight = computeStringWeight(contact.getVcardModel()->getUsername(), USERNAME_WEIGHT); // Get all contact's addresses. const list > addresses = contact.m_linphone_friend->getAddresses(); float size = static_cast(addresses.size()); for (auto it = addresses.cbegin(); it != addresses.cend(); ++it) weight += computeStringWeight( ::Utils::linphoneStringToQString((*it)->asStringUriOnly()), SIP_ADDRESSES_WEIGHT / size ); return weight; } // ----------------------------------------------------------------------------- void ContactsListProxyModel::setConnectedFilter (bool use_connected_filter) { if (use_connected_filter != m_use_connected_filter) { m_use_connected_filter = use_connected_filter; invalidateFilter(); } }