Commit 05b99bf1 authored by Ronan Abhamon's avatar Ronan Abhamon

feat(app): smart search can display contacts & unregistered contacts (unstable)

parent bed7cddf
...@@ -68,7 +68,8 @@ set(SOURCES ...@@ -68,7 +68,8 @@ set(SOURCES
src/components/settings/SettingsModel.cpp src/components/settings/SettingsModel.cpp
src/components/sip-addresses/SipAddressesModel.cpp src/components/sip-addresses/SipAddressesModel.cpp
src/components/sip-addresses/UnregisteredSipAddressesModel.cpp src/components/sip-addresses/UnregisteredSipAddressesModel.cpp
src/components/sip-addresses/UnregisteredSipAddressesProxyModel.hpp src/components/sip-addresses/UnregisteredSipAddressesProxyModel.cpp
src/components/smart-search-bar/SmartSearchBarModel.cpp
src/components/timeline/TimelineModel.cpp src/components/timeline/TimelineModel.cpp
src/main.cpp src/main.cpp
) )
...@@ -94,6 +95,7 @@ set(HEADERS ...@@ -94,6 +95,7 @@ set(HEADERS
src/components/sip-addresses/SipAddressesModel.hpp src/components/sip-addresses/SipAddressesModel.hpp
src/components/sip-addresses/UnregisteredSipAddressesModel.hpp src/components/sip-addresses/UnregisteredSipAddressesModel.hpp
src/components/sip-addresses/UnregisteredSipAddressesProxyModel.hpp src/components/sip-addresses/UnregisteredSipAddressesProxyModel.hpp
src/components/smart-search-bar/SmartSearchBarModel.hpp
src/components/timeline/TimelineModel.hpp src/components/timeline/TimelineModel.hpp
src/utils.hpp src/utils.hpp
) )
......
...@@ -6,14 +6,13 @@ ...@@ -6,14 +6,13 @@
#include "../components/camera/Camera.hpp" #include "../components/camera/Camera.hpp"
#include "../components/chat/ChatProxyModel.hpp" #include "../components/chat/ChatProxyModel.hpp"
#include "../components/contacts/ContactsListModel.hpp"
#include "../components/contacts/ContactsListProxyModel.hpp" #include "../components/contacts/ContactsListProxyModel.hpp"
#include "../components/core/CoreManager.hpp" #include "../components/core/CoreManager.hpp"
#include "../components/notifier/Notifier.hpp" #include "../components/notifier/Notifier.hpp"
#include "../components/settings/AccountSettingsModel.hpp" #include "../components/settings/AccountSettingsModel.hpp"
#include "../components/sip-addresses/SipAddressesModel.hpp"
#include "../components/sip-addresses/UnregisteredSipAddressesProxyModel.hpp" #include "../components/sip-addresses/UnregisteredSipAddressesProxyModel.hpp"
#include "../components/timeline/TimelineModel.hpp" #include "../components/timeline/TimelineModel.hpp"
#include "../components/smart-search-bar/SmartSearchBarModel.hpp"
#include "App.hpp" #include "App.hpp"
...@@ -140,6 +139,12 @@ void App::registerTypes () { ...@@ -140,6 +139,12 @@ void App::registerTypes () {
} }
); );
qmlRegisterSingletonType<SmartSearchBarModel>(
"Linphone", 1, 0, "SmartSearchBarModel",
[](QQmlEngine *, QJSEngine *) -> QObject *{
return new SmartSearchBarModel();
}
);
qmlRegisterType<Camera>("Linphone", 1, 0, "Camera"); qmlRegisterType<Camera>("Linphone", 1, 0, "Camera");
qmlRegisterType<ContactsListProxyModel>("Linphone", 1, 0, "ContactsListProxyModel"); qmlRegisterType<ContactsListProxyModel>("Linphone", 1, 0, "ContactsListProxyModel");
qmlRegisterType<ChatModel>("Linphone", 1, 0, "ChatModel"); qmlRegisterType<ChatModel>("Linphone", 1, 0, "ChatModel");
......
...@@ -35,7 +35,6 @@ ContactsListProxyModel::ContactsListProxyModel (QObject *parent) : QSortFilterPr ...@@ -35,7 +35,6 @@ ContactsListProxyModel::ContactsListProxyModel (QObject *parent) : QSortFilterPr
m_list = CoreManager::getInstance()->getContactsListModel(); m_list = CoreManager::getInstance()->getContactsListModel();
setSourceModel(m_list); setSourceModel(m_list);
setFilterCaseSensitivity(Qt::CaseInsensitive);
setDynamicSortFilter(false); setDynamicSortFilter(false);
for (const ContactModel *contact : m_list->m_list) for (const ContactModel *contact : m_list->m_list)
...@@ -88,12 +87,8 @@ float ContactsListProxyModel::computeStringWeight (const QString &string, float ...@@ -88,12 +87,8 @@ float ContactsListProxyModel::computeStringWeight (const QString &string, float
if ((offset = tmp_offset) == 0) break; if ((offset = tmp_offset) == 0) break;
} }
// No weight.
if (offset == -1)
return 0;
// Weight & offset.
switch (offset) { switch (offset) {
case -1: return 0;
case 0: return percentage * FACTOR_POS_0; case 0: return percentage * FACTOR_POS_0;
case 1: return percentage * FACTOR_POS_1; case 1: return percentage * FACTOR_POS_1;
case 2: return percentage * FACTOR_POS_2; case 2: return percentage * FACTOR_POS_2;
......
...@@ -25,7 +25,7 @@ public: ...@@ -25,7 +25,7 @@ public:
public slots: public slots:
void setFilter (const QString &pattern) { void setFilter (const QString &pattern) {
setFilterFixedString(pattern); setFilterFixedString(pattern);
invalidateFilter(); invalidate();
} }
protected: protected:
......
...@@ -10,5 +10,5 @@ UnregisteredSipAddressesModel::UnregisteredSipAddressesModel (QObject *parent) : ...@@ -10,5 +10,5 @@ UnregisteredSipAddressesModel::UnregisteredSipAddressesModel (QObject *parent) :
bool UnregisteredSipAddressesModel::filterAcceptsRow (int source_row, const QModelIndex &source_parent) const { bool UnregisteredSipAddressesModel::filterAcceptsRow (int source_row, const QModelIndex &source_parent) const {
QModelIndex index = sourceModel()->index(source_row, 0, source_parent); QModelIndex index = sourceModel()->index(source_row, 0, source_parent);
return index.data().toMap().contains("contact"); return !index.data().toMap().contains("contact");
} }
#include "../core/CoreManager.hpp"
#include "UnregisteredSipAddressesProxyModel.hpp" #include "UnregisteredSipAddressesProxyModel.hpp"
#define WEIGHT_POS_0 5
#define WEIGHT_POS_1 4
#define WEIGHT_POS_2 3
#define WEIGHT_POS_3 2
#define WEIGHT_POS_OTHER 1
// ============================================================================= // =============================================================================
const QRegExp UnregisteredSipAddressesProxyModel::m_search_separators("^[^_.-;@ ][_.-;@ ]");
// -----------------------------------------------------------------------------
UnregisteredSipAddressesProxyModel::UnregisteredSipAddressesProxyModel (QObject *parent) :
QSortFilterProxyModel(parent) {
setSourceModel(CoreManager::getInstance()->getUnregisteredSipAddressesModel());
setDynamicSortFilter(false);
sort(0);
}
bool UnregisteredSipAddressesProxyModel::filterAcceptsRow (int source_row, const QModelIndex &source_parent) const {
QModelIndex index = sourceModel()->index(source_row, 0, source_parent);
return computeStringWeight(index.data().toMap()["sipAddress"].toString()) > 0;
}
bool UnregisteredSipAddressesProxyModel::lessThan (const QModelIndex &left, const QModelIndex &right) const {
return computeStringWeight(
sourceModel()->data(left).toMap()["sipAddress"].toString()
) > computeStringWeight(
sourceModel()->data(right).toMap()["sipAddress"].toString()
);
}
int UnregisteredSipAddressesProxyModel::computeStringWeight (const QString &string) const {
int index = -1;
int offset = -1;
while ((index = filterRegExp().indexIn(string, index + 1)) != -1) {
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 WEIGHT_POS_0;
case 1: return WEIGHT_POS_1;
case 2: return WEIGHT_POS_2;
case 3: return WEIGHT_POS_3;
default: break;
}
return WEIGHT_POS_OTHER;
}
...@@ -7,6 +7,25 @@ ...@@ -7,6 +7,25 @@
class UnregisteredSipAddressesProxyModel : public QSortFilterProxyModel { class UnregisteredSipAddressesProxyModel : public QSortFilterProxyModel {
Q_OBJECT; Q_OBJECT;
public:
UnregisteredSipAddressesProxyModel (QObject *parent = Q_NULLPTR);
~UnregisteredSipAddressesProxyModel () = default;
public slots:
void setFilter (const QString &pattern) {
setFilterFixedString(pattern);
invalidate();
}
protected:
bool filterAcceptsRow (int source_row, const QModelIndex &source_parent) const override;
bool lessThan (const QModelIndex &left, const QModelIndex &right) const override;
private:
int computeStringWeight (const QString &string) const;
static const QRegExp m_search_separators;
}; };
#endif // UNREGISTERED_SIP_ADDRESSES_PROXY_MODEL_H_ #endif // UNREGISTERED_SIP_ADDRESSES_PROXY_MODEL_H_
#include "SmartSearchBarModel.hpp"
// =============================================================================
int SmartSearchBarModel::rowCount (const QModelIndex &) const {
return m_contacts.rowCount() + m_sip_addresses.rowCount();
}
QHash<int, QByteArray> SmartSearchBarModel::roleNames () const {
QHash<int, QByteArray> roles;
roles[Qt::DisplayRole] = "$entry";
return roles;
}
QVariant SmartSearchBarModel::data (const QModelIndex &index, int role) const {
int row = index.row();
int n_contacts = m_contacts.rowCount();
int n_sip_addresses = m_sip_addresses.rowCount();
if (row < 0 || row >= n_contacts + n_sip_addresses)
return QVariant();
if (role == Qt::DisplayRole) {
if (row < n_contacts)
return QVariant::fromValue(m_contacts.data(m_contacts.index(row, 0), role));
return QVariant::fromValue(m_sip_addresses.data(m_sip_addresses.index(row - n_contacts, 0), role));
}
return QVariant();
}
#ifndef SMART_SEARCH_BAR_MODEL_H_
#define SMART_SEARCH_BAR_MODEL_H_
#include <QAbstractListModel>
#include "../contacts/ContactsListProxyModel.hpp"
#include "../sip-addresses/UnregisteredSipAddressesProxyModel.hpp"
// =============================================================================
class SmartSearchBarModel : public QAbstractListModel {
Q_OBJECT;
public:
SmartSearchBarModel (QObject *parent = Q_NULLPTR) : QAbstractListModel(parent) {}
~SmartSearchBarModel () = default;
int rowCount (const QModelIndex &index = QModelIndex()) const override;
QHash<int, QByteArray> roleNames () const override;
QVariant data (const QModelIndex &index, int role) const override;
private:
ContactsListProxyModel m_contacts;
UnregisteredSipAddressesProxyModel m_sip_addresses;
};
#endif // SMART_SEARCH_BAR_MODEL_H_
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
#include <QString> #include <QString>
// =================================================================== // =============================================================================
namespace Utils { namespace Utils {
inline QString linphoneStringToQString (const std::string &string) { inline QString linphoneStringToQString (const std::string &string) {
......
import Utils 1.0 import Utils 1.0
// =================================================================== // =============================================================================
// Menu which supports `ListView`. // Menu which supports `ListView`.
// =================================================================== // =============================================================================
AbstractDropDownMenu { AbstractDropDownMenu {
// Can be computed, but for performance usage, it must be given // Can be computed, but for performance usage, it must be given
...@@ -14,10 +14,7 @@ AbstractDropDownMenu { ...@@ -14,10 +14,7 @@ AbstractDropDownMenu {
var list = _content[0] var list = _content[0]
Utils.assert(list != null, 'No list found.') Utils.assert(list != null, 'No list found.')
Utils.assert( Utils.assert(Utils.qmlTypeof(list, 'QQuickListView'), 'No list view parameter.')
Utils.qmlTypeof(list, 'QQuickListView'),
'No list view parameter.'
)
var height = list.count * entryHeight var height = list.count * entryHeight
......
...@@ -4,10 +4,10 @@ import Common 1.0 ...@@ -4,10 +4,10 @@ import Common 1.0
import Common.Styles 1.0 import Common.Styles 1.0
import Utils 1.0 import Utils 1.0
// =================================================================== // =============================================================================
// A reusable search input which display a entries model in a menu. // A reusable search input which display a entries model in a menu.
// Each entry can be filtered with the search input. // Each entry can be filtered with the search input.
// =================================================================== // =============================================================================
Item { Item {
id: searchBox id: searchBox
...@@ -16,8 +16,7 @@ Item { ...@@ -16,8 +16,7 @@ Item {
property alias entryHeight: menu.entryHeight property alias entryHeight: menu.entryHeight
property alias maxMenuHeight: menu.maxMenuHeight property alias maxMenuHeight: menu.maxMenuHeight
// This property must implement `setFilterFixedString` and/or // This property must implement `setFilter` function.
// `invalidate` functions.
property alias model: list.model property alias model: list.model
property alias placeholderText: searchField.placeholderText property alias placeholderText: searchField.placeholderText
...@@ -27,7 +26,7 @@ Item { ...@@ -27,7 +26,7 @@ Item {
signal menuClosed signal menuClosed
signal menuOpened signal menuOpened
// ----------------------------------------------------------------- // ---------------------------------------------------------------------------
function hideMenu () { function hideMenu () {
if (!_isOpen) { if (!_isOpen) {
...@@ -47,17 +46,14 @@ Item { ...@@ -47,17 +46,14 @@ Item {
function _filter (text) { function _filter (text) {
Utils.assert( Utils.assert(
model.setFilterFixedString != null, model.setFilter != null,
'`model.setFilterFixedString` must be defined.' '`model.setFilter` must be defined.'
) )
model.setFilterFixedString(text) model.setFilter(text)
if (model.invalidate) {
model.invalidate()
}
} }
// ----------------------------------------------------------------- // ---------------------------------------------------------------------------
implicitHeight: searchField.height implicitHeight: searchField.height
...@@ -112,7 +108,7 @@ Item { ...@@ -112,7 +108,7 @@ Item {
} }
} }
// ----------------------------------------------------------------- // ---------------------------------------------------------------------------
states: State { states: State {
name: 'opened' name: 'opened'
......
import QtQuick 2.7
import QtQuick.Layouts 1.3
import Common 1.0 import Common 1.0
import Linphone 1.0 import Linphone 1.0
import LinphoneUtils 1.0
// =================================================================== // =============================================================================
SearchBox { SearchBox {
id: searchBox id: searchBox
delegate: Contact { delegate: Rectangle {
// contact: $contact id: searchBoxEntry
width: parent.width width: parent.width
height: searchBox.entryHeight
color: '#FFFFFF'
Rectangle {
id: indicator
anchors.left: parent.left
color: 'transparent'
height: parent.height
width: 5
}
MouseArea {
id: mouseArea
anchors.fill: parent
hoverEnabled: true
RowLayout {
anchors {
fill: parent
leftMargin: 22
rightMargin: 10
}
spacing: 15
// ---------------------------------------------------------------------
// Contact or address info
// ---------------------------------------------------------------------
Avatar {
id: avatar
Layout.preferredHeight: 30
Layout.preferredWidth: 30
image: $entry.vcard && $entry.vcard.avatar
presenceLevel: $entry.presenceLevel != null ? $entry.presenceLevel : -1
username: LinphoneUtils.getContactUsername($entry.sipAddress || $entry)
}
Text {
Layout.fillWidth: true
color: '#4B5964'
elide: Text.ElideRight
font {
bold: true
pointSize: 9
}
text: $entry.vcard ? $entry.vcard.username : $entry.sipAddress
}
// ---------------------------------------------------------------------
// Actions
// ---------------------------------------------------------------------
ActionBar {
iconSize: 36
actions: [
ActionButton { ActionButton {
icon: 'call' icon: 'video_call'
onClicked: CallsWindow.show() onClicked: CallsWindow.show()
}, }
ActionButton { ActionButton {
icon: 'video_call' icon: 'call'
onClicked: CallsWindow.show() onClicked: CallsWindow.show()
} }
]
ActionButton {
icon: 'chat'
onClicked: window.setView('Conversation', {
sipAddress: $entry.sipAddress || $entry.vcard.sipAddresses[0] // FIXME: Display menu if many addresses.
})
}
}
}
}
// -------------------------------------------------------------------------
states: State {
when: mouseArea.containsMouse
PropertyChanges {
color: '#D0D8DE'
target: searchBoxEntry
}
PropertyChanges {
color: '#FF5E00'
target: indicator
}
}
} }
} }
...@@ -89,8 +89,7 @@ ApplicationWindow { ...@@ -89,8 +89,7 @@ ApplicationWindow {
maxMenuHeight: MainWindowStyle.searchBox.maxHeight maxMenuHeight: MainWindowStyle.searchBox.maxHeight
placeholderText: qsTr('mainSearchBarPlaceholder') placeholderText: qsTr('mainSearchBarPlaceholder')
contactsModel: ContactsListProxyModel {} model: SmartSearchBarModel
othersSipAddresses: UnregisteredSipAddressesProxyModel {}
} }
} }
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment