fix account list singleton fix magic search list thread connection fix allow calling a connected account fix magic search flags fix crash on close settings destruction magic search thread meeeting fix time zone rename settingscore remove settings from notifier fix zrtp appearance received call remove deprecated function; TODO : send invitations when sdk updated
226 lines
7.3 KiB
C++
226 lines
7.3 KiB
C++
/*
|
|
* Copyright (c) 2010-2024 Belledonne Communications SARL.
|
|
*
|
|
* This file is part of linphone-desktop
|
|
* (see https://www.linphone.org).
|
|
*
|
|
* 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 "CallList.hpp"
|
|
#include "CallCore.hpp"
|
|
#include "CallGui.hpp"
|
|
#include "core/App.hpp"
|
|
#include <QSharedPointer>
|
|
#include <linphone++/linphone.hh>
|
|
|
|
// =============================================================================
|
|
|
|
DEFINE_ABSTRACT_OBJECT(CallList)
|
|
|
|
QSharedPointer<CallList> CallList::create() {
|
|
auto model = QSharedPointer<CallList>(new CallList(), &QObject::deleteLater);
|
|
model->moveToThread(App::getInstance()->thread());
|
|
model->setSelf(model);
|
|
return model;
|
|
}
|
|
|
|
QSharedPointer<CallCore> CallList::createCallCore(const std::shared_ptr<linphone::Call> &call) {
|
|
auto callCore = CallCore::create(call);
|
|
connect(callCore.get(), &CallCore::stateChanged, this, &CallList::onStateChanged);
|
|
return callCore;
|
|
}
|
|
|
|
CallList::CallList(QObject *parent) : ListProxy(parent) {
|
|
mustBeInMainThread(getClassName());
|
|
App::getInstance()->mEngine->setObjectOwnership(this, QQmlEngine::CppOwnership);
|
|
}
|
|
|
|
CallList::~CallList() {
|
|
mustBeInMainThread("~" + getClassName());
|
|
mModelConnection = nullptr;
|
|
}
|
|
|
|
void CallList::setSelf(QSharedPointer<CallList> me) {
|
|
mModelConnection = QSharedPointer<SafeConnection<CallList, CoreModel>>(
|
|
new SafeConnection<CallList, CoreModel>(me, CoreModel::getInstance()), &QObject::deleteLater);
|
|
|
|
mModelConnection->makeConnectToCore(&CallList::lUpdate, [this]() {
|
|
mModelConnection->invokeToModel([this]() {
|
|
// Avoid copy to lambdas
|
|
QList<QSharedPointer<CallCore>> *calls = new QList<QSharedPointer<CallCore>>();
|
|
mustBeInLinphoneThread(getClassName());
|
|
auto linphoneCalls = CoreModel::getInstance()->getCore()->getCalls();
|
|
auto currentCall = CoreModel::getInstance()->getCore()->getCurrentCall();
|
|
QSharedPointer<CallCore> currentCallCore;
|
|
for (auto it : linphoneCalls) {
|
|
auto model = createCallCore(it);
|
|
if (it == currentCall) currentCallCore = model;
|
|
calls->push_back(model);
|
|
}
|
|
mModelConnection->invokeToCore([this, calls, currentCallCore]() {
|
|
mustBeInMainThread(getClassName());
|
|
resetData();
|
|
add(*calls);
|
|
setHaveCall(calls->size() > 0);
|
|
setCurrentCall(currentCallCore);
|
|
delete calls;
|
|
});
|
|
});
|
|
});
|
|
mModelConnection->makeConnectToCore(&CallList::lMergeAll, [this]() {
|
|
mModelConnection->invokeToModel([this]() {
|
|
auto core = CoreModel::getInstance()->getCore();
|
|
auto currentCalls = CoreModel::getInstance()->getCore()->getCalls();
|
|
std::shared_ptr<linphone::Conference> conference = nullptr;
|
|
|
|
// Search a managable conference from calls
|
|
for (auto call : currentCalls) {
|
|
auto dbConference = call->getConference();
|
|
if (dbConference && dbConference->getMe()->isAdmin()) {
|
|
conference = dbConference;
|
|
break;
|
|
}
|
|
}
|
|
|
|
auto currentCall = CoreModel::getInstance()->getCore()->getCurrentCall();
|
|
bool enablingVideo = false;
|
|
if (currentCall) enablingVideo = currentCall->getCurrentParams()->videoEnabled();
|
|
if (!conference) {
|
|
auto parameters = core->createConferenceParams(conference);
|
|
auto audioVideoConfFactoryUri =
|
|
core->getDefaultAccount()->getParams()->getAudioVideoConferenceFactoryAddress();
|
|
if (audioVideoConfFactoryUri) {
|
|
parameters->setConferenceFactoryAddress(audioVideoConfFactoryUri);
|
|
parameters->setSubject("Meeting");
|
|
} else {
|
|
parameters->setSubject("Local meeting");
|
|
}
|
|
parameters->enableVideo(enablingVideo);
|
|
conference = core->createConferenceWithParams(parameters);
|
|
}
|
|
|
|
std::list<std::shared_ptr<linphone::Address>> allLinphoneAddresses;
|
|
std::list<std::shared_ptr<linphone::Address>> newCalls;
|
|
std::list<std::shared_ptr<linphone::Call>> runningCallsToAdd;
|
|
|
|
for (auto call : currentCalls) {
|
|
if (!call->getConference()) {
|
|
runningCallsToAdd.push_back(call);
|
|
}
|
|
}
|
|
|
|
// 1) Add running calls
|
|
if (runningCallsToAdd.size() > 0) {
|
|
conference->addParticipants(runningCallsToAdd);
|
|
}
|
|
|
|
// emit lUpdate();
|
|
});
|
|
});
|
|
|
|
mModelConnection->makeConnectToModel(&CoreModel::firstCallStarted,
|
|
[this]() { mModelConnection->invokeToCore([this]() { setHaveCall(true); }); });
|
|
mModelConnection->makeConnectToModel(&CoreModel::lastCallEnded, [this]() {
|
|
mModelConnection->invokeToCore([this]() {
|
|
setHaveCall(false);
|
|
setCurrentCall(nullptr);
|
|
});
|
|
});
|
|
mModelConnection->makeConnectToModel(&CoreModel::callCreated, [this](const std::shared_ptr<linphone::Call> &call) {
|
|
auto model = createCallCore(call);
|
|
mModelConnection->invokeToCore([this, model]() {
|
|
// We set the current here and not on firstCallStarted event because we don't want to add unicity check
|
|
// while keeping the same model between list and current call.
|
|
if (mList.size() == 0) setCurrentCall(model);
|
|
add(model);
|
|
});
|
|
});
|
|
lUpdate();
|
|
}
|
|
|
|
QSharedPointer<CallCore> CallList::getCurrentCallCore() const {
|
|
return mCurrentCall;
|
|
}
|
|
|
|
CallGui *CallList::getCurrentCall() const {
|
|
auto call = getCurrentCallCore();
|
|
if (call) return new CallGui(call);
|
|
else return nullptr;
|
|
}
|
|
|
|
void CallList::setCurrentCall(QSharedPointer<CallCore> call) {
|
|
if (mCurrentCall != call) {
|
|
mCurrentCall = call;
|
|
emit currentCallChanged();
|
|
}
|
|
}
|
|
|
|
bool CallList::getHaveCall() const {
|
|
return mHaveCall;
|
|
}
|
|
|
|
void CallList::setHaveCall(bool haveCall) {
|
|
if (mHaveCall != haveCall) {
|
|
mHaveCall = haveCall;
|
|
emit haveCallChanged();
|
|
}
|
|
}
|
|
|
|
QSharedPointer<CallCore> CallList::getNextCall() const {
|
|
QSharedPointer<CallCore> call;
|
|
auto currentCall = getCurrentCallCore();
|
|
for (auto it = mList.rbegin(); !call && it != mList.rend(); ++it) {
|
|
if (*it != currentCall) {
|
|
call = it->objectCast<CallCore>();
|
|
}
|
|
}
|
|
|
|
return call;
|
|
}
|
|
|
|
void CallList::onStateChanged() {
|
|
auto call = dynamic_cast<CallCore *>(sender());
|
|
switch (call->getState()) {
|
|
case LinphoneEnums::CallState::StreamsRunning:
|
|
case LinphoneEnums::CallState::Resuming: {
|
|
auto sharedCall = get(call);
|
|
setCurrentCall(sharedCall.objectCast<CallCore>());
|
|
break;
|
|
}
|
|
case LinphoneEnums::CallState::Released: {
|
|
auto sharedCall = get(call);
|
|
auto currentCall = getCurrentCallCore();
|
|
// Update current call
|
|
if (sharedCall == currentCall) {
|
|
// Unpause the next call. The current call will change on resume.
|
|
// Assumption: All calls that are not the current are paused.
|
|
auto nextCall = getNextCall();
|
|
if (nextCall) nextCall->lSetPaused(false);
|
|
}
|
|
sharedCall->disconnect(this);
|
|
remove(sharedCall);
|
|
break;
|
|
}
|
|
default: {
|
|
}
|
|
}
|
|
}
|
|
|
|
QVariant CallList::data(const QModelIndex &index, int role) const {
|
|
int row = index.row();
|
|
if (!index.isValid() || row < 0 || row >= mList.count()) return QVariant();
|
|
if (role == Qt::DisplayRole) return QVariant::fromValue(new CallGui(mList[row].objectCast<CallCore>()));
|
|
return QVariant();
|
|
}
|