2020-07-31 01:59:54 +02:00
|
|
|
#include <algorithm>
|
2020-07-30 00:16:52 +02:00
|
|
|
#include <cctype>
|
2020-07-11 01:19:48 +02:00
|
|
|
#include <chrono>
|
2020-08-01 20:31:10 +02:00
|
|
|
#include <cstdint>
|
2021-02-18 21:55:29 +01:00
|
|
|
#include <cstdlib>
|
2020-07-11 01:19:48 +02:00
|
|
|
|
|
|
|
#include <QMediaPlaylist>
|
|
|
|
#include <QUrl>
|
|
|
|
|
|
|
|
#include "Cache.h"
|
2021-02-07 17:47:47 +01:00
|
|
|
#include "CallDevices.h"
|
2020-08-01 20:31:10 +02:00
|
|
|
#include "CallManager.h"
|
2020-07-11 01:19:48 +02:00
|
|
|
#include "ChatPage.h"
|
|
|
|
#include "Logging.h"
|
|
|
|
#include "MatrixClient.h"
|
2020-12-17 17:25:32 +01:00
|
|
|
#include "UserSettingsPage.h"
|
2020-10-27 17:45:28 +01:00
|
|
|
#include "Utils.h"
|
2020-07-11 01:19:48 +02:00
|
|
|
|
2020-07-26 16:59:50 +02:00
|
|
|
#include "mtx/responses/turn_server.hpp"
|
|
|
|
|
2020-07-11 01:19:48 +02:00
|
|
|
Q_DECLARE_METATYPE(std::vector<mtx::events::msg::CallCandidates::Candidate>)
|
2020-07-23 23:58:22 +02:00
|
|
|
Q_DECLARE_METATYPE(mtx::events::msg::CallCandidates::Candidate)
|
2020-07-11 01:19:48 +02:00
|
|
|
Q_DECLARE_METATYPE(mtx::responses::TurnServer)
|
|
|
|
|
|
|
|
using namespace mtx::events;
|
|
|
|
using namespace mtx::events::msg;
|
|
|
|
|
2021-02-18 21:55:29 +01:00
|
|
|
using webrtc::CallType;
|
|
|
|
|
2020-07-26 16:59:50 +02:00
|
|
|
namespace {
|
|
|
|
std::vector<std::string>
|
|
|
|
getTurnURIs(const mtx::responses::TurnServer &turnServer);
|
|
|
|
}
|
|
|
|
|
2020-10-28 21:08:17 +01:00
|
|
|
CallManager::CallManager(QObject *parent)
|
2020-10-27 17:45:28 +01:00
|
|
|
: QObject(parent)
|
2020-08-01 20:31:10 +02:00
|
|
|
, session_(WebRTCSession::instance())
|
|
|
|
, turnServerTimer_(this)
|
2020-07-11 01:19:48 +02:00
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
qRegisterMetaType<std::vector<mtx::events::msg::CallCandidates::Candidate>>();
|
|
|
|
qRegisterMetaType<mtx::events::msg::CallCandidates::Candidate>();
|
|
|
|
qRegisterMetaType<mtx::responses::TurnServer>();
|
|
|
|
|
|
|
|
connect(
|
|
|
|
&session_,
|
|
|
|
&WebRTCSession::offerCreated,
|
|
|
|
this,
|
|
|
|
[this](const std::string &sdp, const std::vector<CallCandidates::Candidate> &candidates) {
|
|
|
|
nhlog::ui()->debug("WebRTC: call id: {} - sending offer", callid_);
|
|
|
|
emit newMessage(roomid_, CallInvite{callid_, sdp, 0, timeoutms_});
|
|
|
|
emit newMessage(roomid_, CallCandidates{callid_, candidates, 0});
|
2021-01-12 21:16:59 +01:00
|
|
|
std::string callid(callid_);
|
|
|
|
QTimer::singleShot(timeoutms_, this, [this, callid]() {
|
|
|
|
if (session_.state() == webrtc::State::OFFERSENT && callid == callid_) {
|
2020-08-01 20:31:10 +02:00
|
|
|
hangUp(CallHangUp::Reason::InviteTimeOut);
|
|
|
|
emit ChatPage::instance()->showNotification(
|
|
|
|
"The remote side failed to pick up.");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
connect(
|
|
|
|
&session_,
|
|
|
|
&WebRTCSession::answerCreated,
|
|
|
|
this,
|
|
|
|
[this](const std::string &sdp, const std::vector<CallCandidates::Candidate> &candidates) {
|
|
|
|
nhlog::ui()->debug("WebRTC: call id: {} - sending answer", callid_);
|
|
|
|
emit newMessage(roomid_, CallAnswer{callid_, sdp, 0});
|
|
|
|
emit newMessage(roomid_, CallCandidates{callid_, candidates, 0});
|
|
|
|
});
|
|
|
|
|
|
|
|
connect(&session_,
|
|
|
|
&WebRTCSession::newICECandidate,
|
|
|
|
this,
|
|
|
|
[this](const CallCandidates::Candidate &candidate) {
|
|
|
|
nhlog::ui()->debug("WebRTC: call id: {} - sending ice candidate", callid_);
|
|
|
|
emit newMessage(roomid_, CallCandidates{callid_, {candidate}, 0});
|
|
|
|
});
|
|
|
|
|
|
|
|
connect(&turnServerTimer_, &QTimer::timeout, this, &CallManager::retrieveTurnServer);
|
|
|
|
|
|
|
|
connect(this,
|
|
|
|
&CallManager::turnServerRetrieved,
|
|
|
|
this,
|
|
|
|
[this](const mtx::responses::TurnServer &res) {
|
|
|
|
nhlog::net()->info("TURN server(s) retrieved from homeserver:");
|
|
|
|
nhlog::net()->info("username: {}", res.username);
|
|
|
|
nhlog::net()->info("ttl: {} seconds", res.ttl);
|
|
|
|
for (const auto &u : res.uris)
|
|
|
|
nhlog::net()->info("uri: {}", u);
|
|
|
|
|
|
|
|
// Request new credentials close to expiry
|
|
|
|
// See https://tools.ietf.org/html/draft-uberti-behave-turn-rest-00
|
|
|
|
turnURIs_ = getTurnURIs(res);
|
|
|
|
uint32_t ttl = std::max(res.ttl, UINT32_C(3600));
|
|
|
|
if (res.ttl < 3600)
|
|
|
|
nhlog::net()->warn("Setting ttl to 1 hour");
|
|
|
|
turnServerTimer_.setInterval(ttl * 1000 * 0.9);
|
|
|
|
});
|
|
|
|
|
2020-09-22 18:07:36 +02:00
|
|
|
connect(&session_, &WebRTCSession::stateChanged, this, [this](webrtc::State state) {
|
2020-08-01 20:31:10 +02:00
|
|
|
switch (state) {
|
2020-09-22 18:07:36 +02:00
|
|
|
case webrtc::State::DISCONNECTED:
|
2020-11-13 01:55:35 +01:00
|
|
|
playRingtone(QUrl("qrc:/media/media/callend.ogg"), false);
|
2020-08-01 20:31:10 +02:00
|
|
|
clear();
|
|
|
|
break;
|
2020-09-22 18:07:36 +02:00
|
|
|
case webrtc::State::ICEFAILED: {
|
2020-08-01 20:31:10 +02:00
|
|
|
QString error("Call connection failed.");
|
|
|
|
if (turnURIs_.empty())
|
|
|
|
error += " Your homeserver has no configured TURN server.";
|
|
|
|
emit ChatPage::instance()->showNotification(error);
|
|
|
|
hangUp(CallHangUp::Reason::ICEFailed);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2020-12-10 02:49:48 +01:00
|
|
|
emit newCallState();
|
2020-08-01 20:31:10 +02:00
|
|
|
});
|
|
|
|
|
2021-02-07 17:47:47 +01:00
|
|
|
connect(&CallDevices::instance(),
|
|
|
|
&CallDevices::devicesChanged,
|
|
|
|
this,
|
|
|
|
&CallManager::devicesChanged);
|
2020-12-17 17:25:32 +01:00
|
|
|
|
2020-08-01 20:31:10 +02:00
|
|
|
connect(&player_,
|
|
|
|
&QMediaPlayer::mediaStatusChanged,
|
|
|
|
this,
|
|
|
|
[this](QMediaPlayer::MediaStatus status) {
|
|
|
|
if (status == QMediaPlayer::LoadedMedia)
|
|
|
|
player_.play();
|
|
|
|
});
|
2020-11-13 01:55:35 +01:00
|
|
|
|
|
|
|
connect(&player_,
|
|
|
|
QOverload<QMediaPlayer::Error>::of(&QMediaPlayer::error),
|
|
|
|
[this](QMediaPlayer::Error error) {
|
|
|
|
stopRingtone();
|
|
|
|
switch (error) {
|
|
|
|
case QMediaPlayer::FormatError:
|
|
|
|
case QMediaPlayer::ResourceError:
|
|
|
|
nhlog::ui()->error("WebRTC: valid ringtone file not found");
|
|
|
|
break;
|
|
|
|
case QMediaPlayer::AccessDeniedError:
|
|
|
|
nhlog::ui()->error("WebRTC: access to ringtone file denied");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
nhlog::ui()->error("WebRTC: unable to play ringtone");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
});
|
2020-07-11 01:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2021-02-18 21:55:29 +01:00
|
|
|
CallManager::sendInvite(const QString &roomid, CallType callType)
|
2020-07-11 01:19:48 +02:00
|
|
|
{
|
2020-12-10 02:49:48 +01:00
|
|
|
if (isOnCall())
|
2020-08-01 20:31:10 +02:00
|
|
|
return;
|
2021-02-18 21:55:29 +01:00
|
|
|
if (callType == CallType::SCREEN && !screenShareSupported())
|
|
|
|
return;
|
2020-08-01 20:31:10 +02:00
|
|
|
|
|
|
|
auto roomInfo = cache::singleRoomInfo(roomid.toStdString());
|
|
|
|
if (roomInfo.member_count != 2) {
|
2020-10-27 18:14:06 +01:00
|
|
|
emit ChatPage::instance()->showNotification("Calls are limited to 1:1 rooms.");
|
2020-08-01 20:31:10 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string errorMessage;
|
2020-10-27 18:14:06 +01:00
|
|
|
if (!session_.havePlugins(false, &errorMessage) ||
|
2021-02-18 21:55:29 +01:00
|
|
|
((callType == CallType::VIDEO || callType == CallType::SCREEN) &&
|
|
|
|
!session_.havePlugins(true, &errorMessage))) {
|
2020-08-01 20:31:10 +02:00
|
|
|
emit ChatPage::instance()->showNotification(QString::fromStdString(errorMessage));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-02-18 21:55:29 +01:00
|
|
|
callType_ = callType;
|
|
|
|
roomid_ = roomid;
|
2020-08-01 20:31:10 +02:00
|
|
|
session_.setTurnServers(turnURIs_);
|
|
|
|
generateCallID();
|
2021-02-18 21:55:29 +01:00
|
|
|
std::string strCallType = callType_ == CallType::VOICE
|
|
|
|
? "voice"
|
|
|
|
: (callType_ == CallType::VIDEO ? "video" : "screen");
|
|
|
|
nhlog::ui()->debug("WebRTC: call id: {} - creating {} invite", callid_, strCallType);
|
2020-09-25 16:26:36 +02:00
|
|
|
std::vector<RoomMember> members(cache::getMembers(roomid.toStdString()));
|
|
|
|
const RoomMember &callee =
|
|
|
|
members.front().user_id == utils::localUser() ? members.back() : members.front();
|
2020-12-17 17:25:32 +01:00
|
|
|
callParty_ = callee.display_name.isEmpty() ? callee.user_id : callee.display_name;
|
2020-09-25 16:26:36 +02:00
|
|
|
callPartyAvatarUrl_ = QString::fromStdString(roomInfo.avatar_url);
|
2020-12-17 17:25:32 +01:00
|
|
|
emit newInviteState();
|
2020-11-13 01:55:35 +01:00
|
|
|
playRingtone(QUrl("qrc:/media/media/ringback.ogg"), true);
|
2021-02-18 21:55:29 +01:00
|
|
|
if (!session_.createOffer(callType)) {
|
2020-08-01 20:31:10 +02:00
|
|
|
emit ChatPage::instance()->showNotification("Problem setting up call.");
|
|
|
|
endCall();
|
|
|
|
}
|
2020-07-11 01:19:48 +02:00
|
|
|
}
|
|
|
|
|
2020-07-26 16:59:50 +02:00
|
|
|
namespace {
|
2020-08-01 20:31:10 +02:00
|
|
|
std::string
|
|
|
|
callHangUpReasonString(CallHangUp::Reason reason)
|
2020-07-26 16:59:50 +02:00
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
switch (reason) {
|
|
|
|
case CallHangUp::Reason::ICEFailed:
|
|
|
|
return "ICE failed";
|
|
|
|
case CallHangUp::Reason::InviteTimeOut:
|
|
|
|
return "Invite time out";
|
|
|
|
default:
|
|
|
|
return "User";
|
|
|
|
}
|
2020-07-26 16:59:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-11 01:19:48 +02:00
|
|
|
void
|
2020-07-26 00:11:11 +02:00
|
|
|
CallManager::hangUp(CallHangUp::Reason reason)
|
2020-07-11 01:19:48 +02:00
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
if (!callid_.empty()) {
|
|
|
|
nhlog::ui()->debug(
|
|
|
|
"WebRTC: call id: {} - hanging up ({})", callid_, callHangUpReasonString(reason));
|
|
|
|
emit newMessage(roomid_, CallHangUp{callid_, 0, reason});
|
|
|
|
endCall();
|
|
|
|
}
|
2020-07-11 01:19:48 +02:00
|
|
|
}
|
|
|
|
|
2020-08-01 20:31:10 +02:00
|
|
|
void
|
|
|
|
CallManager::syncEvent(const mtx::events::collections::TimelineEvents &event)
|
2020-07-11 01:19:48 +02:00
|
|
|
{
|
2020-08-14 01:03:27 +02:00
|
|
|
#ifdef GSTREAMER_AVAILABLE
|
2020-08-01 20:31:10 +02:00
|
|
|
if (handleEvent_<CallInvite>(event) || handleEvent_<CallCandidates>(event) ||
|
|
|
|
handleEvent_<CallAnswer>(event) || handleEvent_<CallHangUp>(event))
|
|
|
|
return;
|
2020-08-14 01:03:27 +02:00
|
|
|
#else
|
|
|
|
(void)event;
|
|
|
|
#endif
|
2020-07-11 01:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
bool
|
|
|
|
CallManager::handleEvent_(const mtx::events::collections::TimelineEvents &event)
|
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
if (std::holds_alternative<RoomEvent<T>>(event)) {
|
|
|
|
handleEvent(std::get<RoomEvent<T>>(event));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2020-07-11 01:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CallManager::handleEvent(const RoomEvent<CallInvite> &callInviteEvent)
|
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
const char video[] = "m=video";
|
|
|
|
const std::string &sdp = callInviteEvent.content.sdp;
|
|
|
|
bool isVideo = std::search(sdp.cbegin(),
|
|
|
|
sdp.cend(),
|
|
|
|
std::cbegin(video),
|
|
|
|
std::cend(video) - 1,
|
|
|
|
[](unsigned char c1, unsigned char c2) {
|
|
|
|
return std::tolower(c1) == std::tolower(c2);
|
|
|
|
}) != sdp.cend();
|
|
|
|
|
2020-08-11 08:51:57 +02:00
|
|
|
nhlog::ui()->debug("WebRTC: call id: {} - incoming {} CallInvite from {}",
|
2020-08-01 20:31:10 +02:00
|
|
|
callInviteEvent.content.call_id,
|
2020-08-14 09:01:56 +02:00
|
|
|
(isVideo ? "video" : "voice"),
|
2020-08-01 20:31:10 +02:00
|
|
|
callInviteEvent.sender);
|
|
|
|
|
|
|
|
if (callInviteEvent.content.call_id.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto roomInfo = cache::singleRoomInfo(callInviteEvent.room_id);
|
2020-12-10 02:49:48 +01:00
|
|
|
if (isOnCall() || roomInfo.member_count != 2) {
|
2020-08-01 20:31:10 +02:00
|
|
|
emit newMessage(QString::fromStdString(callInviteEvent.room_id),
|
|
|
|
CallHangUp{callInviteEvent.content.call_id,
|
|
|
|
0,
|
|
|
|
CallHangUp::Reason::InviteTimeOut});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-11-13 01:55:35 +01:00
|
|
|
const QString &ringtone = ChatPage::instance()->userSettings()->ringtone();
|
|
|
|
if (ringtone != "Mute")
|
|
|
|
playRingtone(ringtone == "Default" ? QUrl("qrc:/media/media/ring.ogg")
|
|
|
|
: QUrl::fromLocalFile(ringtone),
|
|
|
|
true);
|
2020-08-01 20:31:10 +02:00
|
|
|
roomid_ = QString::fromStdString(callInviteEvent.room_id);
|
|
|
|
callid_ = callInviteEvent.content.call_id;
|
|
|
|
remoteICECandidates_.clear();
|
|
|
|
|
|
|
|
std::vector<RoomMember> members(cache::getMembers(callInviteEvent.room_id));
|
|
|
|
const RoomMember &caller =
|
|
|
|
members.front().user_id == utils::localUser() ? members.back() : members.front();
|
2020-12-17 17:25:32 +01:00
|
|
|
callParty_ = caller.display_name.isEmpty() ? caller.user_id : caller.display_name;
|
2020-09-25 16:26:36 +02:00
|
|
|
callPartyAvatarUrl_ = QString::fromStdString(roomInfo.avatar_url);
|
2020-12-17 17:25:32 +01:00
|
|
|
|
|
|
|
haveCallInvite_ = true;
|
2021-02-18 21:55:29 +01:00
|
|
|
callType_ = isVideo ? CallType::VIDEO : CallType::VOICE;
|
2020-12-17 17:25:32 +01:00
|
|
|
inviteSDP_ = callInviteEvent.content.sdp;
|
|
|
|
emit newInviteState();
|
2020-07-11 01:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2020-12-17 17:25:32 +01:00
|
|
|
CallManager::acceptInvite()
|
2020-07-11 01:19:48 +02:00
|
|
|
{
|
2020-12-17 17:25:32 +01:00
|
|
|
if (!haveCallInvite_)
|
|
|
|
return;
|
|
|
|
|
2020-08-01 20:31:10 +02:00
|
|
|
stopRingtone();
|
|
|
|
std::string errorMessage;
|
2020-10-27 18:14:06 +01:00
|
|
|
if (!session_.havePlugins(false, &errorMessage) ||
|
2021-02-18 21:55:29 +01:00
|
|
|
(callType_ == CallType::VIDEO && !session_.havePlugins(true, &errorMessage))) {
|
2020-08-01 20:31:10 +02:00
|
|
|
emit ChatPage::instance()->showNotification(QString::fromStdString(errorMessage));
|
|
|
|
hangUp();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
session_.setTurnServers(turnURIs_);
|
2020-12-17 17:25:32 +01:00
|
|
|
if (!session_.acceptOffer(inviteSDP_)) {
|
2020-08-01 20:31:10 +02:00
|
|
|
emit ChatPage::instance()->showNotification("Problem setting up call.");
|
|
|
|
hangUp();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
session_.acceptICECandidates(remoteICECandidates_);
|
|
|
|
remoteICECandidates_.clear();
|
2020-12-17 17:25:32 +01:00
|
|
|
haveCallInvite_ = false;
|
|
|
|
emit newInviteState();
|
2020-07-11 01:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CallManager::handleEvent(const RoomEvent<CallCandidates> &callCandidatesEvent)
|
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
if (callCandidatesEvent.sender == utils::localUser().toStdString())
|
|
|
|
return;
|
|
|
|
|
|
|
|
nhlog::ui()->debug("WebRTC: call id: {} - incoming CallCandidates from {}",
|
|
|
|
callCandidatesEvent.content.call_id,
|
|
|
|
callCandidatesEvent.sender);
|
|
|
|
|
|
|
|
if (callid_ == callCandidatesEvent.content.call_id) {
|
2020-12-10 02:49:48 +01:00
|
|
|
if (isOnCall())
|
2020-08-01 20:31:10 +02:00
|
|
|
session_.acceptICECandidates(callCandidatesEvent.content.candidates);
|
|
|
|
else {
|
|
|
|
// CallInvite has been received and we're awaiting localUser to accept or
|
|
|
|
// reject the call
|
|
|
|
for (const auto &c : callCandidatesEvent.content.candidates)
|
|
|
|
remoteICECandidates_.push_back(c);
|
|
|
|
}
|
|
|
|
}
|
2020-07-11 01:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CallManager::handleEvent(const RoomEvent<CallAnswer> &callAnswerEvent)
|
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
nhlog::ui()->debug("WebRTC: call id: {} - incoming CallAnswer from {}",
|
|
|
|
callAnswerEvent.content.call_id,
|
|
|
|
callAnswerEvent.sender);
|
|
|
|
|
2021-01-01 14:46:08 +01:00
|
|
|
if (callAnswerEvent.sender == utils::localUser().toStdString() &&
|
2020-08-01 20:31:10 +02:00
|
|
|
callid_ == callAnswerEvent.content.call_id) {
|
2021-01-01 14:46:08 +01:00
|
|
|
if (!isOnCall()) {
|
|
|
|
emit ChatPage::instance()->showNotification(
|
|
|
|
"Call answered on another device.");
|
|
|
|
stopRingtone();
|
|
|
|
haveCallInvite_ = false;
|
|
|
|
emit newInviteState();
|
|
|
|
}
|
2020-08-01 20:31:10 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-12-10 02:49:48 +01:00
|
|
|
if (isOnCall() && callid_ == callAnswerEvent.content.call_id) {
|
2020-08-01 20:31:10 +02:00
|
|
|
stopRingtone();
|
|
|
|
if (!session_.acceptAnswer(callAnswerEvent.content.sdp)) {
|
|
|
|
emit ChatPage::instance()->showNotification("Problem setting up call.");
|
|
|
|
hangUp();
|
|
|
|
}
|
|
|
|
}
|
2020-07-11 01:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CallManager::handleEvent(const RoomEvent<CallHangUp> &callHangUpEvent)
|
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
nhlog::ui()->debug("WebRTC: call id: {} - incoming CallHangUp ({}) from {}",
|
|
|
|
callHangUpEvent.content.call_id,
|
|
|
|
callHangUpReasonString(callHangUpEvent.content.reason),
|
|
|
|
callHangUpEvent.sender);
|
|
|
|
|
2020-12-17 17:25:32 +01:00
|
|
|
if (callid_ == callHangUpEvent.content.call_id)
|
2020-08-01 20:31:10 +02:00
|
|
|
endCall();
|
2020-07-11 01:19:48 +02:00
|
|
|
}
|
|
|
|
|
2020-12-10 02:49:48 +01:00
|
|
|
void
|
|
|
|
CallManager::toggleMicMute()
|
|
|
|
{
|
|
|
|
session_.toggleMicMute();
|
|
|
|
emit micMuteChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2021-02-18 21:55:29 +01:00
|
|
|
CallManager::callsSupported()
|
2020-12-10 02:49:48 +01:00
|
|
|
{
|
|
|
|
#ifdef GSTREAMER_AVAILABLE
|
|
|
|
return true;
|
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2021-02-18 21:55:29 +01:00
|
|
|
bool
|
|
|
|
CallManager::screenShareSupported()
|
|
|
|
{
|
2021-02-20 23:33:04 +01:00
|
|
|
return std::getenv("DISPLAY") && !std::getenv("WAYLAND_DISPLAY");
|
2021-02-18 21:55:29 +01:00
|
|
|
}
|
|
|
|
|
2020-12-17 17:25:32 +01:00
|
|
|
QStringList
|
|
|
|
CallManager::devices(bool isVideo) const
|
|
|
|
{
|
|
|
|
QStringList ret;
|
|
|
|
const QString &defaultDevice = isVideo ? ChatPage::instance()->userSettings()->camera()
|
|
|
|
: ChatPage::instance()->userSettings()->microphone();
|
|
|
|
std::vector<std::string> devices =
|
2021-02-07 17:47:47 +01:00
|
|
|
CallDevices::instance().names(isVideo, defaultDevice.toStdString());
|
2020-12-17 17:25:32 +01:00
|
|
|
ret.reserve(devices.size());
|
|
|
|
std::transform(devices.cbegin(),
|
|
|
|
devices.cend(),
|
|
|
|
std::back_inserter(ret),
|
|
|
|
[](const auto &d) { return QString::fromStdString(d); });
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2020-07-11 01:19:48 +02:00
|
|
|
void
|
|
|
|
CallManager::generateCallID()
|
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
using namespace std::chrono;
|
|
|
|
uint64_t ms = duration_cast<milliseconds>(system_clock::now().time_since_epoch()).count();
|
|
|
|
callid_ = "c" + std::to_string(ms);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CallManager::clear()
|
|
|
|
{
|
|
|
|
roomid_.clear();
|
2020-12-17 17:25:32 +01:00
|
|
|
callParty_.clear();
|
2020-09-25 16:26:36 +02:00
|
|
|
callPartyAvatarUrl_.clear();
|
2020-08-01 20:31:10 +02:00
|
|
|
callid_.clear();
|
2021-02-18 21:55:29 +01:00
|
|
|
callType_ = CallType::VOICE;
|
2020-12-17 17:25:32 +01:00
|
|
|
haveCallInvite_ = false;
|
|
|
|
emit newInviteState();
|
|
|
|
inviteSDP_.clear();
|
2020-08-01 20:31:10 +02:00
|
|
|
remoteICECandidates_.clear();
|
2020-07-11 01:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CallManager::endCall()
|
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
stopRingtone();
|
|
|
|
session_.end();
|
2020-12-17 17:25:32 +01:00
|
|
|
clear();
|
2020-07-11 01:19:48 +02:00
|
|
|
}
|
|
|
|
|
2020-07-31 01:59:54 +02:00
|
|
|
void
|
|
|
|
CallManager::refreshTurnServer()
|
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
turnURIs_.clear();
|
|
|
|
turnServerTimer_.start(2000);
|
2020-07-31 01:59:54 +02:00
|
|
|
}
|
|
|
|
|
2020-07-11 01:19:48 +02:00
|
|
|
void
|
|
|
|
CallManager::retrieveTurnServer()
|
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
http::client()->get_turn_server(
|
|
|
|
[this](const mtx::responses::TurnServer &res, mtx::http::RequestErr err) {
|
|
|
|
if (err) {
|
|
|
|
turnServerTimer_.setInterval(5000);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
emit turnServerRetrieved(res);
|
|
|
|
});
|
2020-07-11 01:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2020-11-13 01:55:35 +01:00
|
|
|
CallManager::playRingtone(const QUrl &ringtone, bool repeat)
|
2020-07-26 16:59:50 +02:00
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
static QMediaPlaylist playlist;
|
|
|
|
playlist.clear();
|
|
|
|
playlist.setPlaybackMode(repeat ? QMediaPlaylist::CurrentItemInLoop
|
|
|
|
: QMediaPlaylist::CurrentItemOnce);
|
2020-11-13 01:55:35 +01:00
|
|
|
playlist.addMedia(ringtone);
|
2020-08-01 20:31:10 +02:00
|
|
|
player_.setVolume(100);
|
|
|
|
player_.setPlaylist(&playlist);
|
2020-07-26 16:59:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CallManager::stopRingtone()
|
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
player_.setPlaylist(nullptr);
|
2020-07-26 16:59:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
std::vector<std::string>
|
|
|
|
getTurnURIs(const mtx::responses::TurnServer &turnServer)
|
2020-07-11 01:19:48 +02:00
|
|
|
{
|
2020-08-01 20:31:10 +02:00
|
|
|
// gstreamer expects: turn(s)://username:password@host:port?transport=udp(tcp)
|
|
|
|
// where username and password are percent-encoded
|
|
|
|
std::vector<std::string> ret;
|
|
|
|
for (const auto &uri : turnServer.uris) {
|
|
|
|
if (auto c = uri.find(':'); c == std::string::npos) {
|
|
|
|
nhlog::ui()->error("Invalid TURN server uri: {}", uri);
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
std::string scheme = std::string(uri, 0, c);
|
|
|
|
if (scheme != "turn" && scheme != "turns") {
|
|
|
|
nhlog::ui()->error("Invalid TURN server uri: {}", uri);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString encodedUri =
|
|
|
|
QString::fromStdString(scheme) + "://" +
|
|
|
|
QUrl::toPercentEncoding(QString::fromStdString(turnServer.username)) +
|
|
|
|
":" +
|
|
|
|
QUrl::toPercentEncoding(QString::fromStdString(turnServer.password)) +
|
|
|
|
"@" + QString::fromStdString(std::string(uri, ++c));
|
|
|
|
ret.push_back(encodedUri.toStdString());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
2020-07-11 01:19:48 +02:00
|
|
|
}
|
|
|
|
}
|