forked from mirrors/linphone-iphone
714 lines
24 KiB
C++
714 lines
24 KiB
C++
/*
|
|
* client-group-chat-room.cpp
|
|
* Copyright (C) 2010-2018 Belledonne Communications SARL
|
|
*
|
|
* 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 2
|
|
* 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, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "linphone/utils/utils.h"
|
|
|
|
#include "address/address-p.h"
|
|
#include "basic-to-client-group-chat-room.h"
|
|
#include "c-wrapper/c-wrapper.h"
|
|
#include "client-group-chat-room-p.h"
|
|
#include "conference/handlers/remote-conference-event-handler.h"
|
|
#include "conference/participant-p.h"
|
|
#include "conference/remote-conference-p.h"
|
|
#include "conference/session/call-session-p.h"
|
|
#include "content/content-disposition.h"
|
|
#include "content/content-type.h"
|
|
#include "core/core-p.h"
|
|
#include "logger/logger.h"
|
|
#include "sal/refer-op.h"
|
|
|
|
// =============================================================================
|
|
|
|
using namespace std;
|
|
|
|
LINPHONE_BEGIN_NAMESPACE
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
list<IdentityAddress> ClientGroupChatRoomPrivate::cleanAddressesList (const list<IdentityAddress> &addresses) const {
|
|
L_Q();
|
|
list<IdentityAddress> cleanedList(addresses);
|
|
cleanedList.sort();
|
|
cleanedList.unique();
|
|
for (auto it = cleanedList.begin(); it != cleanedList.end();) {
|
|
if (q->findParticipant(*it) || (q->getMe()->getAddress() == *it))
|
|
it = cleanedList.erase(it);
|
|
else
|
|
it++;
|
|
}
|
|
return cleanedList;
|
|
}
|
|
|
|
shared_ptr<CallSession> ClientGroupChatRoomPrivate::createSession () {
|
|
L_Q();
|
|
L_Q_T(RemoteConference, qConference);
|
|
|
|
CallSessionParams csp;
|
|
csp.addCustomHeader("Require", "recipient-list-invite");
|
|
csp.addCustomContactParameter("text");
|
|
if (capabilities & ClientGroupChatRoom::Capabilities::OneToOne)
|
|
csp.addCustomHeader("One-To-One-Chat-Room", "true");
|
|
|
|
ParticipantPrivate *dFocus = qConference->getPrivate()->focus->getPrivate();
|
|
shared_ptr<CallSession> session = dFocus->createSession(*q, &csp, false, callSessionListener);
|
|
Address myCleanedAddress(q->getMe()->getAddress());
|
|
myCleanedAddress.removeUriParam("gr"); // Remove gr parameter for INVITE.
|
|
session->configure(LinphoneCallOutgoing, nullptr, nullptr, myCleanedAddress, dFocus->getDevices().front()->getAddress());
|
|
session->initiateOutgoing();
|
|
session->getPrivate()->createOp();
|
|
return session;
|
|
}
|
|
|
|
void ClientGroupChatRoomPrivate::notifyReceived (const string &body) {
|
|
L_Q_T(RemoteConference, qConference);
|
|
qConference->getPrivate()->eventHandler->notifyReceived(body);
|
|
}
|
|
|
|
void ClientGroupChatRoomPrivate::multipartNotifyReceived (const string &body) {
|
|
L_Q_T(RemoteConference, qConference);
|
|
qConference->getPrivate()->eventHandler->multipartNotifyReceived(body);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
void ClientGroupChatRoomPrivate::setCallSessionListener (CallSessionListener *listener) {
|
|
L_Q();
|
|
L_Q_T(RemoteConference, qConference);
|
|
callSessionListener = listener;
|
|
shared_ptr<CallSession> session = qConference->getPrivate()->focus->getPrivate()->getSession();
|
|
if (session)
|
|
session->getPrivate()->setCallSessionListener(listener);
|
|
for (const auto &participant : q->getParticipants()) {
|
|
session = participant->getPrivate()->getSession();
|
|
if (session)
|
|
session->getPrivate()->setCallSessionListener(listener);
|
|
}
|
|
}
|
|
|
|
void ClientGroupChatRoomPrivate::confirmJoining (SalCallOp *op) {
|
|
L_Q();
|
|
L_Q_T(RemoteConference, qConference);
|
|
|
|
auto focus = qConference->getPrivate()->focus;
|
|
bool previousSession = (focus->getPrivate()->getSession() != nullptr);
|
|
auto session = focus->getPrivate()->createSession(*q, nullptr, false, this);
|
|
session->configure(LinphoneCallIncoming, nullptr, op, Address(op->get_from()), Address(op->get_to()));
|
|
session->startIncomingNotification(false);
|
|
|
|
if (!previousSession) {
|
|
setState(ClientGroupChatRoom::State::CreationPending);
|
|
// Handle participants addition
|
|
list<IdentityAddress> identAddresses = ClientGroupChatRoom::parseResourceLists(op->get_remote_body());
|
|
for (const auto &addr : identAddresses) {
|
|
auto participant = q->findParticipant(addr);
|
|
if (!participant) {
|
|
participant = make_shared<Participant>(addr);
|
|
qConference->getPrivate()->participants.push_back(participant);
|
|
}
|
|
}
|
|
}
|
|
acceptSession(session);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
void ClientGroupChatRoomPrivate::onChatRoomInsertRequested (const shared_ptr<AbstractChatRoom> &chatRoom) {
|
|
L_Q();
|
|
q->getCore()->getPrivate()->insertChatRoom(chatRoom);
|
|
}
|
|
|
|
void ClientGroupChatRoomPrivate::onChatRoomInsertInDatabaseRequested (const shared_ptr<AbstractChatRoom> &chatRoom) {
|
|
L_Q();
|
|
q->getCore()->getPrivate()->insertChatRoomWithDb(chatRoom);
|
|
}
|
|
|
|
void ClientGroupChatRoomPrivate::onChatRoomDeleteRequested (const shared_ptr<AbstractChatRoom> &chatRoom) {
|
|
L_Q();
|
|
q->getCore()->deleteChatRoom(q->getSharedFromThis());
|
|
setState(ClientGroupChatRoom::State::Deleted);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
void ClientGroupChatRoomPrivate::onCallSessionSetReleased (const shared_ptr<CallSession> &session) {
|
|
L_Q_T(RemoteConference, qConference);
|
|
|
|
ParticipantPrivate *participantPrivate = qConference->getPrivate()->focus->getPrivate();
|
|
if (session == participantPrivate->getSession())
|
|
participantPrivate->removeSession();
|
|
}
|
|
|
|
void ClientGroupChatRoomPrivate::onCallSessionStateChanged (
|
|
const shared_ptr<CallSession> &session,
|
|
CallSession::State newState,
|
|
const string &message
|
|
) {
|
|
L_Q();
|
|
L_Q_T(RemoteConference, qConference);
|
|
|
|
if (newState == CallSession::State::Connected) {
|
|
if (q->getState() == ChatRoom::State::CreationPending) {
|
|
onChatRoomCreated(*session->getRemoteContactAddress());
|
|
} else if (q->getState() == ChatRoom::State::TerminationPending)
|
|
qConference->getPrivate()->focus->getPrivate()->getSession()->terminate();
|
|
} else if (newState == CallSession::State::End) {
|
|
q->onConferenceTerminated(q->getConferenceAddress());
|
|
} else if (newState == CallSession::State::Released) {
|
|
if (q->getState() == ChatRoom::State::TerminationPending) {
|
|
if (session->getReason() == LinphoneReasonNone) {
|
|
// Everything is fine, the chat room has been left on the server side
|
|
q->onConferenceTerminated(q->getConferenceAddress());
|
|
} else {
|
|
// Go to state TerminationFailed and then back to Created since it has not been terminated
|
|
setState(ChatRoom::State::TerminationFailed);
|
|
setState(ChatRoom::State::Created);
|
|
}
|
|
}
|
|
} else if (newState == CallSession::State::Error) {
|
|
if (q->getState() == ChatRoom::State::CreationPending)
|
|
setState(ChatRoom::State::CreationFailed);
|
|
else if (q->getState() == ChatRoom::State::TerminationPending) {
|
|
if (session->getReason() == LinphoneReasonNotFound) {
|
|
// Somehow the chat room is no longer know on the server, so terminate it
|
|
q->onConferenceTerminated(q->getConferenceAddress());
|
|
} else {
|
|
// Go to state TerminationFailed and then back to Created since it has not been terminated
|
|
setState(ChatRoom::State::TerminationFailed);
|
|
setState(ChatRoom::State::Created);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void ClientGroupChatRoomPrivate::onChatRoomCreated (const Address &remoteContact) {
|
|
L_Q();
|
|
L_Q_T(RemoteConference, qConference);
|
|
|
|
IdentityAddress addr(remoteContact);
|
|
q->onConferenceCreated(addr);
|
|
if (remoteContact.hasParam("isfocus")) {
|
|
bgTask.start(q->getCore(), 32); // It will be stopped when receiving the first notify
|
|
qConference->getPrivate()->eventHandler->subscribe(q->getChatRoomId());
|
|
}
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
void ClientGroupChatRoomPrivate::acceptSession (const shared_ptr<CallSession> &session) {
|
|
if (session->getState() == CallSession::State::UpdatedByRemote)
|
|
session->acceptUpdate();
|
|
else
|
|
session->accept();
|
|
}
|
|
|
|
// =============================================================================
|
|
|
|
ClientGroupChatRoom::ClientGroupChatRoom (
|
|
const shared_ptr<Core> &core,
|
|
const string &uri,
|
|
const IdentityAddress &me,
|
|
const string &subject,
|
|
const Content &content
|
|
) : ChatRoom(*new ClientGroupChatRoomPrivate, core, ChatRoomId(IdentityAddress(), me)),
|
|
RemoteConference(core, me, nullptr) {
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
IdentityAddress focusAddr(uri);
|
|
dConference->focus = make_shared<Participant>(focusAddr);
|
|
dConference->focus->getPrivate()->addDevice(focusAddr);
|
|
RemoteConference::setSubject(subject);
|
|
list<IdentityAddress> identAddresses = Conference::parseResourceLists(content);
|
|
for (const auto &addr : identAddresses)
|
|
dConference->participants.push_back(make_shared<Participant>(addr));
|
|
}
|
|
|
|
ClientGroupChatRoom::ClientGroupChatRoom (
|
|
const shared_ptr<Core> &core,
|
|
const ChatRoomId &chatRoomId,
|
|
shared_ptr<Participant> &me,
|
|
AbstractChatRoom::CapabilitiesMask capabilities,
|
|
const string &subject,
|
|
list<shared_ptr<Participant>> &&participants,
|
|
unsigned int lastNotifyId
|
|
) : ChatRoom(*new ClientGroupChatRoomPrivate, core, chatRoomId),
|
|
RemoteConference(core, me->getAddress(), nullptr) {
|
|
L_D();
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
d->capabilities |= capabilities & ClientGroupChatRoom::Capabilities::OneToOne;
|
|
const IdentityAddress &peerAddress = chatRoomId.getPeerAddress();
|
|
dConference->focus = make_shared<Participant>(peerAddress);
|
|
dConference->focus->getPrivate()->addDevice(peerAddress);
|
|
dConference->conferenceAddress = peerAddress;
|
|
dConference->subject = subject;
|
|
dConference->participants = move(participants);
|
|
|
|
getMe()->getPrivate()->setAdmin(me->isAdmin());
|
|
|
|
dConference->eventHandler->setLastNotify(lastNotifyId);
|
|
dConference->eventHandler->subscribe(getChatRoomId());
|
|
}
|
|
|
|
ClientGroupChatRoom::~ClientGroupChatRoom () {
|
|
L_D();
|
|
d->setCallSessionListener(nullptr);
|
|
}
|
|
|
|
shared_ptr<Core> ClientGroupChatRoom::getCore () const {
|
|
return ChatRoom::getCore();
|
|
}
|
|
|
|
void ClientGroupChatRoom::allowCpim (bool value) {
|
|
|
|
}
|
|
|
|
void ClientGroupChatRoom::allowMultipart (bool value) {
|
|
|
|
}
|
|
|
|
bool ClientGroupChatRoom::canHandleCpim () const {
|
|
return true;
|
|
}
|
|
|
|
bool ClientGroupChatRoom::canHandleMultipart () const {
|
|
return true;
|
|
}
|
|
|
|
ClientGroupChatRoom::CapabilitiesMask ClientGroupChatRoom::getCapabilities () const {
|
|
L_D();
|
|
return d->capabilities;
|
|
}
|
|
|
|
bool ClientGroupChatRoom::hasBeenLeft () const {
|
|
return (getState() != State::Created);
|
|
}
|
|
|
|
bool ClientGroupChatRoom::canHandleParticipants () const {
|
|
return RemoteConference::canHandleParticipants();
|
|
}
|
|
|
|
const IdentityAddress &ClientGroupChatRoom::getConferenceAddress () const {
|
|
return RemoteConference::getConferenceAddress();
|
|
}
|
|
|
|
void ClientGroupChatRoom::deleteFromDb () {
|
|
L_D();
|
|
if (!hasBeenLeft()) {
|
|
d->deletionOnTerminationEnabled = true;
|
|
leave();
|
|
return;
|
|
}
|
|
d->chatRoomListener->onChatRoomDeleteRequested(getSharedFromThis());
|
|
}
|
|
|
|
void ClientGroupChatRoom::addParticipant (const IdentityAddress &addr, const CallSessionParams *params, bool hasMedia) {
|
|
L_D();
|
|
|
|
if ((getState() != ChatRoom::State::Instantiated) && (getState() != ChatRoom::State::Created)) {
|
|
lError() << "Cannot add participants to the ClientGroupChatRoom in a state other than Instantiated or Created";
|
|
return;
|
|
}
|
|
|
|
if ((getState() == ChatRoom::State::Created) && (d->capabilities & ClientGroupChatRoom::Capabilities::OneToOne)) {
|
|
lError() << "Cannot add more participants to a OneToOne ClientGroupChatRoom";
|
|
return;
|
|
}
|
|
|
|
LinphoneCore *cCore = getCore()->getCCore();
|
|
if (getState() == ChatRoom::State::Instantiated) {
|
|
list<IdentityAddress> addressesList;
|
|
addressesList.push_back(addr);
|
|
Content content;
|
|
content.setBody(getResourceLists(addressesList));
|
|
content.setContentType(ContentType::ResourceLists);
|
|
content.setContentDisposition(ContentDisposition::RecipientList);
|
|
|
|
auto session = d->createSession();
|
|
session->startInvite(nullptr, getSubject(), &content);
|
|
d->setState(ChatRoom::State::CreationPending);
|
|
} else {
|
|
SalReferOp *referOp = new SalReferOp(cCore->sal);
|
|
LinphoneAddress *lAddr = linphone_address_new(getConferenceAddress().asString().c_str());
|
|
linphone_configure_op(cCore, referOp, lAddr, nullptr, true);
|
|
linphone_address_unref(lAddr);
|
|
Address referToAddr = addr;
|
|
referToAddr.setParam("text");
|
|
referOp->send_refer(referToAddr.getPrivate()->getInternalAddress());
|
|
referOp->unref();
|
|
}
|
|
}
|
|
|
|
void ClientGroupChatRoom::addParticipants (
|
|
const list<IdentityAddress> &addresses,
|
|
const CallSessionParams *params,
|
|
bool hasMedia
|
|
) {
|
|
L_D();
|
|
|
|
list<IdentityAddress> addressesList = d->cleanAddressesList(addresses);
|
|
if (addressesList.empty())
|
|
return;
|
|
|
|
if ((getState() == ChatRoom::State::Instantiated)
|
|
&& (addressesList.size() == 1)
|
|
&& (linphone_config_get_bool(linphone_core_get_config(L_GET_C_BACK_PTR(getCore())),
|
|
"misc", "one_to_one_chat_room_enabled", TRUE))
|
|
) {
|
|
d->capabilities |= ClientGroupChatRoom::Capabilities::OneToOne;
|
|
}
|
|
|
|
if (getState() == ChatRoom::State::Instantiated) {
|
|
Content content;
|
|
content.setBody(getResourceLists(addressesList));
|
|
content.setContentType(ContentType::ResourceLists);
|
|
content.setContentDisposition(ContentDisposition::RecipientList);
|
|
if (linphone_core_content_encoding_supported(getCore()->getCCore(), "deflate"))
|
|
content.setContentEncoding("deflate");
|
|
|
|
auto session = d->createSession();
|
|
session->startInvite(nullptr, getSubject(), &content);
|
|
d->setState(ChatRoom::State::CreationPending);
|
|
} else {
|
|
for (const auto &addr : addresses)
|
|
addParticipant(addr, params, hasMedia);
|
|
}
|
|
}
|
|
|
|
void ClientGroupChatRoom::removeParticipant (const shared_ptr<Participant> &participant) {
|
|
LinphoneCore *cCore = getCore()->getCCore();
|
|
|
|
SalReferOp *referOp = new SalReferOp(cCore->sal);
|
|
LinphoneAddress *lAddr = linphone_address_new(getConferenceAddress().asString().c_str());
|
|
linphone_configure_op(cCore, referOp, lAddr, nullptr, false);
|
|
linphone_address_unref(lAddr);
|
|
Address referToAddr = participant->getAddress();
|
|
referToAddr.setParam("text");
|
|
referToAddr.setUriParam("method", "BYE");
|
|
referOp->send_refer(referToAddr.getPrivate()->getInternalAddress());
|
|
referOp->unref();
|
|
}
|
|
|
|
void ClientGroupChatRoom::removeParticipants (const list<shared_ptr<Participant>> &participants) {
|
|
RemoteConference::removeParticipants(participants);
|
|
}
|
|
|
|
shared_ptr<Participant> ClientGroupChatRoom::findParticipant (const IdentityAddress &addr) const {
|
|
return RemoteConference::findParticipant(addr);
|
|
}
|
|
|
|
shared_ptr<Participant> ClientGroupChatRoom::getMe () const {
|
|
return RemoteConference::getMe();
|
|
}
|
|
|
|
int ClientGroupChatRoom::getParticipantCount () const {
|
|
return RemoteConference::getParticipantCount();
|
|
}
|
|
|
|
const list<shared_ptr<Participant>> &ClientGroupChatRoom::getParticipants () const {
|
|
return RemoteConference::getParticipants();
|
|
}
|
|
|
|
void ClientGroupChatRoom::setParticipantAdminStatus (const shared_ptr<Participant> &participant, bool isAdmin) {
|
|
if (isAdmin == participant->isAdmin())
|
|
return;
|
|
|
|
if (!getMe()->isAdmin()) {
|
|
lError() << "Cannot change the participant admin status because I am not admin";
|
|
return;
|
|
}
|
|
|
|
LinphoneCore *cCore = getCore()->getCCore();
|
|
|
|
SalReferOp *referOp = new SalReferOp(cCore->sal);
|
|
LinphoneAddress *lAddr = linphone_address_new(getConferenceAddress().asString().c_str());
|
|
linphone_configure_op(cCore, referOp, lAddr, nullptr, false);
|
|
linphone_address_unref(lAddr);
|
|
Address referToAddr = participant->getAddress();
|
|
referToAddr.setParam("text");
|
|
referToAddr.setParam("admin", Utils::toString(isAdmin));
|
|
referOp->send_refer(referToAddr.getPrivate()->getInternalAddress());
|
|
referOp->unref();
|
|
}
|
|
|
|
const string &ClientGroupChatRoom::getSubject () const {
|
|
return RemoteConference::getSubject();
|
|
}
|
|
|
|
void ClientGroupChatRoom::setSubject (const string &subject) {
|
|
L_D();
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
if (getState() != ChatRoom::State::Created) {
|
|
lError() << "Cannot change the ClientGroupChatRoom subject in a state other than Created";
|
|
return;
|
|
}
|
|
|
|
if (!getMe()->isAdmin()) {
|
|
lError() << "Cannot change the ClientGroupChatRoom subject because I am not admin";
|
|
return;
|
|
}
|
|
|
|
shared_ptr<CallSession> session = dConference->focus->getPrivate()->getSession();
|
|
if (session)
|
|
session->update(nullptr, subject);
|
|
else {
|
|
session = d->createSession();
|
|
session->startInvite(nullptr, subject, nullptr);
|
|
}
|
|
}
|
|
|
|
void ClientGroupChatRoom::join () {
|
|
L_D();
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
shared_ptr<CallSession> session = dConference->focus->getPrivate()->getSession();
|
|
if (!session && ((getState() == ChatRoom::State::Instantiated) || (getState() == ChatRoom::State::Terminated))) {
|
|
session = d->createSession();
|
|
}
|
|
if (session) {
|
|
if (getState() != ChatRoom::State::TerminationPending)
|
|
session->startInvite(nullptr, "", nullptr);
|
|
if (getState() != ChatRoom::State::Created)
|
|
d->setState(ChatRoom::State::CreationPending);
|
|
}
|
|
}
|
|
|
|
void ClientGroupChatRoom::leave () {
|
|
L_D();
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
dConference->eventHandler->unsubscribe();
|
|
|
|
shared_ptr<CallSession> session = dConference->focus->getPrivate()->getSession();
|
|
if (session)
|
|
session->terminate();
|
|
else {
|
|
session = d->createSession();
|
|
session->startInvite(nullptr, "", nullptr);
|
|
}
|
|
|
|
d->setState(ChatRoom::State::TerminationPending);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
void ClientGroupChatRoom::onConferenceCreated (const IdentityAddress &addr) {
|
|
L_D();
|
|
L_D_T(RemoteConference, dConference);
|
|
dConference->conferenceAddress = addr;
|
|
dConference->focus->getPrivate()->setAddress(addr);
|
|
dConference->focus->getPrivate()->clearDevices();
|
|
dConference->focus->getPrivate()->addDevice(addr);
|
|
d->chatRoomId = ChatRoomId(addr, d->chatRoomId.getLocalAddress());
|
|
d->chatRoomListener->onChatRoomInsertRequested(getSharedFromThis());
|
|
d->setState(ChatRoom::State::Created);
|
|
}
|
|
|
|
void ClientGroupChatRoom::onConferenceKeywordsChanged (const vector<string> &keywords) {
|
|
L_D();
|
|
d->capabilities |= ClientGroupChatRoom::Capabilities::OneToOne;
|
|
}
|
|
|
|
void ClientGroupChatRoom::onConferenceTerminated (const IdentityAddress &addr) {
|
|
L_D();
|
|
L_D_T(RemoteConference, dConference);
|
|
dConference->eventHandler->unsubscribe();
|
|
dConference->eventHandler->resetLastNotify();
|
|
d->setState(ChatRoom::State::Terminated);
|
|
d->addEvent(make_shared<ConferenceEvent>(
|
|
EventLog::Type::ConferenceTerminated,
|
|
time(nullptr),
|
|
d->chatRoomId
|
|
));
|
|
if (d->deletionOnTerminationEnabled) {
|
|
d->deletionOnTerminationEnabled = false;
|
|
d->chatRoomListener->onChatRoomDeleteRequested(getSharedFromThis());
|
|
}
|
|
}
|
|
|
|
void ClientGroupChatRoom::onFirstNotifyReceived (const IdentityAddress &addr) {
|
|
L_D();
|
|
|
|
bool performMigration = false;
|
|
shared_ptr<AbstractChatRoom> chatRoom;
|
|
if (getParticipantCount() == 1) {
|
|
ChatRoomId id(getParticipants().front()->getAddress(), getMe()->getAddress());
|
|
chatRoom = getCore()->findChatRoom(id);
|
|
if (chatRoom && (chatRoom->getCapabilities() & ChatRoom::Capabilities::Basic))
|
|
performMigration = true;
|
|
}
|
|
|
|
if (performMigration)
|
|
BasicToClientGroupChatRoom::migrate(getSharedFromThis(), chatRoom);
|
|
else
|
|
d->chatRoomListener->onChatRoomInsertInDatabaseRequested(getSharedFromThis());
|
|
|
|
LinphoneChatRoom *cr = d->getCChatRoom();
|
|
_linphone_chat_room_notify_all_information_received(cr);
|
|
|
|
d->bgTask.stop();
|
|
|
|
// TODO: Bug. Event is inserted many times.
|
|
// Avoid this in the future. Deal with signals/slots system.
|
|
#if 0
|
|
d->addEvent(make_shared<ConferenceEvent>(
|
|
EventLog::Type::ConferenceCreated,
|
|
time(nullptr),
|
|
d->chatRoomId
|
|
));
|
|
#endif
|
|
}
|
|
|
|
void ClientGroupChatRoom::onParticipantAdded (const shared_ptr<ConferenceParticipantEvent> &event, bool isFullState) {
|
|
L_D();
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
const IdentityAddress &addr = event->getParticipantAddress();
|
|
if (isMe(addr))
|
|
return;
|
|
|
|
shared_ptr<Participant> participant = findParticipant(addr);
|
|
if (participant) {
|
|
lWarning() << "Participant " << participant << " added but already in the list of participants!";
|
|
return;
|
|
}
|
|
|
|
participant = make_shared<Participant>(addr);
|
|
dConference->participants.push_back(participant);
|
|
|
|
if (isFullState)
|
|
return;
|
|
|
|
d->addEvent(event);
|
|
|
|
LinphoneChatRoom *cr = d->getCChatRoom();
|
|
_linphone_chat_room_notify_participant_added(cr, L_GET_C_BACK_PTR(event));
|
|
}
|
|
|
|
void ClientGroupChatRoom::onParticipantRemoved (const shared_ptr<ConferenceParticipantEvent> &event, bool isFullState) {
|
|
(void)isFullState;
|
|
|
|
L_D();
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
const IdentityAddress &addr = event->getParticipantAddress();
|
|
shared_ptr<Participant> participant = findParticipant(addr);
|
|
if (!participant) {
|
|
lWarning() << "Participant " << addr.asString() << " removed but not in the list of participants!";
|
|
return;
|
|
}
|
|
|
|
dConference->participants.remove(participant);
|
|
d->addEvent(event);
|
|
|
|
LinphoneChatRoom *cr = d->getCChatRoom();
|
|
_linphone_chat_room_notify_participant_removed(cr, L_GET_C_BACK_PTR(event));
|
|
}
|
|
|
|
void ClientGroupChatRoom::onParticipantSetAdmin (const shared_ptr<ConferenceParticipantEvent> &event, bool isFullState) {
|
|
L_D();
|
|
|
|
const IdentityAddress &addr = event->getParticipantAddress();
|
|
shared_ptr<Participant> participant;
|
|
if (isMe(addr))
|
|
participant = getMe();
|
|
else
|
|
participant = findParticipant(addr);
|
|
if (!participant) {
|
|
lWarning() << "Participant " << addr.asString() << " admin status has been changed but is not in the list of participants!";
|
|
return;
|
|
}
|
|
|
|
bool isAdmin = event->getType() == EventLog::Type::ConferenceParticipantSetAdmin;
|
|
if (participant->isAdmin() == isAdmin)
|
|
return; // No change in the local admin status, do not notify
|
|
participant->getPrivate()->setAdmin(isAdmin);
|
|
|
|
if (isFullState)
|
|
return;
|
|
|
|
d->addEvent(event);
|
|
|
|
LinphoneChatRoom *cr = d->getCChatRoom();
|
|
_linphone_chat_room_notify_participant_admin_status_changed(cr, L_GET_C_BACK_PTR(event));
|
|
}
|
|
|
|
void ClientGroupChatRoom::onSubjectChanged (const shared_ptr<ConferenceSubjectEvent> &event, bool isFullState) {
|
|
L_D();
|
|
|
|
if (getSubject() == event->getSubject())
|
|
return; // No change in the local subject, do not notify
|
|
RemoteConference::setSubject(event->getSubject());
|
|
|
|
if (isFullState)
|
|
return;
|
|
|
|
d->addEvent(event);
|
|
|
|
LinphoneChatRoom *cr = d->getCChatRoom();
|
|
_linphone_chat_room_notify_subject_changed(cr, L_GET_C_BACK_PTR(event));
|
|
}
|
|
|
|
void ClientGroupChatRoom::onParticipantDeviceAdded (const shared_ptr<ConferenceParticipantDeviceEvent> &event, bool isFullState) {
|
|
L_D();
|
|
|
|
const IdentityAddress &addr = event->getParticipantAddress();
|
|
shared_ptr<Participant> participant;
|
|
if (isMe(addr))
|
|
participant = getMe();
|
|
else
|
|
participant = findParticipant(addr);
|
|
if (!participant) {
|
|
lWarning() << "Participant " << addr.asString() << " added a device but is not in the list of participants!";
|
|
return;
|
|
}
|
|
participant->getPrivate()->addDevice(event->getDeviceAddress());
|
|
|
|
if (isFullState)
|
|
return;
|
|
|
|
d->addEvent(event);
|
|
|
|
LinphoneChatRoom *cr = d->getCChatRoom();
|
|
_linphone_chat_room_notify_participant_device_added(cr, L_GET_C_BACK_PTR(event));
|
|
}
|
|
|
|
void ClientGroupChatRoom::onParticipantDeviceRemoved (const shared_ptr<ConferenceParticipantDeviceEvent> &event, bool isFullState) {
|
|
L_D();
|
|
|
|
(void)isFullState;
|
|
|
|
const IdentityAddress &addr = event->getParticipantAddress();
|
|
shared_ptr<Participant> participant;
|
|
if (isMe(addr))
|
|
participant = getMe();
|
|
else
|
|
participant = findParticipant(addr);
|
|
if (!participant) {
|
|
lWarning() << "Participant " << addr.asString() << " removed a device but is not in the list of participants!";
|
|
return;
|
|
}
|
|
participant->getPrivate()->removeDevice(event->getDeviceAddress());
|
|
d->addEvent(event);
|
|
|
|
LinphoneChatRoom *cr = d->getCChatRoom();
|
|
_linphone_chat_room_notify_participant_device_removed(cr, L_GET_C_BACK_PTR(event));
|
|
}
|
|
|
|
LINPHONE_END_NAMESPACE
|