| /* |
| * Copyright (C) 2020 by Savoir-faire Linux |
| * Author: Edric Ladent Milaret <edric.ladent-milaret@savoirfairelinux.com> |
| * Author: Andreas Traczyk <andreas.traczyk@savoirfairelinux.com> |
| * Author: Mingrui Zhang <mingrui.zhang@savoirfairelinux.com> |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; either version 3 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program. If not, see <http://www.gnu.org/licenses/>. |
| */ |
| |
| #include "contactadapter.h" |
| |
| #include "lrcinstance.h" |
| |
| ContactAdapter::ContactAdapter(QObject *parent) |
| : QmlAdapterBase(parent) |
| { |
| selectableProxyModel_.reset(new SelectableProxyModel(smartListModel_.get())); |
| } |
| |
| ContactAdapter::~ContactAdapter() {} |
| |
| QVariant |
| ContactAdapter::getContactSelectableModel(int type) |
| { |
| /* |
| * Called from qml every time contact picker refreshes. |
| */ |
| listModeltype_ = Utils::toEnum<SmartListModel::Type>(type); |
| smartListModel_.reset(new SmartListModel(LRCInstance::getCurrAccId(), |
| this, |
| listModeltype_, |
| LRCInstance::getCurrentConvUid())); |
| selectableProxyModel_->setSourceModel(smartListModel_.get()); |
| |
| /* |
| * Adjust filter. |
| */ |
| switch (listModeltype_) { |
| case SmartListModel::Type::CONFERENCE: |
| selectableProxyModel_->setPredicate([this](const QModelIndex &index, const QRegExp &) { |
| return index.data(SmartListModel::Presence).toBool(); |
| }); |
| break; |
| case SmartListModel::Type::TRANSFER: |
| selectableProxyModel_->setPredicate([this](const QModelIndex &index, const QRegExp ®exp) { |
| /* |
| * Regex to remove current callee. |
| */ |
| QRegExp matchExcept = QRegExp(QString("\\b(?!" + calleeDisplayName_ + "\\b)\\w+")); |
| bool match = false; |
| bool match_non_self = matchExcept.indexIn( |
| index.data(SmartListModel::Role::DisplayName).toString()) |
| != -1; |
| if (match_non_self) { |
| match = regexp.indexIn(index.data(SmartListModel::Role::DisplayName).toString()) |
| != -1; |
| } |
| return match && !index.parent().isValid(); |
| }); |
| break; |
| default: |
| break; |
| } |
| selectableProxyModel_->invalidate(); |
| |
| return QVariant::fromValue(selectableProxyModel_.get()); |
| } |
| |
| void |
| ContactAdapter::setSearchFilter(const QString &filter) |
| { |
| if (listModeltype_ == SmartListModel::Type::CONFERENCE) { |
| smartListModel_->setConferenceableFilter(filter); |
| } |
| selectableProxyModel_->setFilterRegExp( |
| QRegExp(filter, Qt::CaseInsensitive, QRegExp::FixedString)); |
| } |
| |
| void |
| ContactAdapter::contactSelected(int index) |
| { |
| auto contactIndex = selectableProxyModel_->index(index, 0); |
| auto* callModel = LRCInstance::getCurrentCallModel(); |
| auto* convModel = LRCInstance::getCurrentConversationModel(); |
| const auto conversation = convModel->getConversationForUID(LRCInstance::getCurrentConvUid()); |
| |
| if (contactIndex.isValid()) { |
| switch (listModeltype_) { |
| case SmartListModel::Type::CONFERENCE: { |
| /* |
| * Conference. |
| */ |
| const auto sectionName = contactIndex.data(SmartListModel::Role::SectionName).value<QString>(); |
| if (!sectionName.isEmpty()) { |
| smartListModel_->toggleSection(sectionName); |
| return; |
| } |
| |
| const auto convUid = contactIndex.data(SmartListModel::Role::UID).value<QString>(); |
| const auto accId = contactIndex.data(SmartListModel::Role::AccountId).value<QString>(); |
| const auto callId = LRCInstance::getCallIdForConversationUid(convUid, accId); |
| |
| if (!callId.isEmpty()) { |
| if (conversation.uid.isEmpty()) { |
| return; |
| } |
| auto thisCallId = conversation.confId.isEmpty() ? conversation.callId |
| : conversation.confId; |
| |
| callModel->joinCalls(thisCallId, callId); |
| } else { |
| const auto contactUri = contactIndex.data(SmartListModel::Role::URI).value<QString>(); |
| auto call = LRCInstance::getCallInfoForConversation(conversation); |
| if (!call) { |
| return; |
| } |
| callModel->callAndAddParticipant(contactUri, call->id, call->isAudioOnly); |
| } |
| } break; |
| case SmartListModel::Type::TRANSFER: { |
| /* |
| * SIP Transfer. |
| */ |
| const auto contactUri = contactIndex.data(SmartListModel::Role::URI).value<QString>(); |
| |
| if (conversation.uid.isEmpty()) { |
| return; |
| } |
| const auto callId = conversation.confId.isEmpty() ? conversation.callId : conversation.confId; |
| |
| QString destCallId; |
| |
| try { |
| /* |
| * Check if the call exist - (check non-finished calls). |
| */ |
| const auto callInfo = callModel->getCallFromURI(contactUri, true); |
| destCallId = callInfo.id; |
| } catch (std::exception &e) { |
| qDebug().noquote() << e.what(); |
| destCallId = ""; |
| } |
| /* |
| * If no second call -> blind transfer. |
| * If there is a second call -> attended transfer. |
| */ |
| if (destCallId.size() == 0) { |
| callModel->transfer(callId, "sip:" + contactUri); |
| callModel->hangUp(callId); |
| } else { |
| callModel->transferToCall(callId, destCallId); |
| callModel->hangUp(callId); |
| callModel->hangUp(destCallId); |
| } |
| } break; |
| default: |
| break; |
| } |
| } |
| } |
| |
| void |
| ContactAdapter::setCalleeDisplayName(const QString &name) |
| { |
| calleeDisplayName_ = name; |
| } |
| |
| void |
| ContactAdapter::initQmlObject() |
| {} |