2017-04-19 18:06:01 +00:00
|
|
|
/*
|
|
|
|
This file is part of Telegram Desktop,
|
2018-01-03 10:23:14 +00:00
|
|
|
the official desktop application for the Telegram messaging service.
|
2017-04-19 18:06:01 +00:00
|
|
|
|
2018-01-03 10:23:14 +00:00
|
|
|
For license and copyright information please follow this link:
|
|
|
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
2017-04-19 18:06:01 +00:00
|
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
|
2017-11-30 17:33:27 +00:00
|
|
|
#include "base/weak_ptr.h"
|
2017-04-19 20:25:48 +00:00
|
|
|
#include "base/timer.h"
|
2018-03-27 12:16:00 +00:00
|
|
|
#include "base/bytes.h"
|
2017-04-24 12:16:38 +00:00
|
|
|
#include "mtproto/sender.h"
|
2019-11-18 09:28:14 +00:00
|
|
|
#include "mtproto/mtproto_auth_key.h"
|
2017-04-19 18:06:01 +00:00
|
|
|
|
2017-05-03 13:43:01 +00:00
|
|
|
namespace Media {
|
|
|
|
namespace Audio {
|
|
|
|
class Track;
|
|
|
|
} // namespace Audio
|
|
|
|
} // namespace Media
|
|
|
|
|
2020-07-09 17:38:26 +00:00
|
|
|
namespace tgcalls {
|
|
|
|
class Instance;
|
2020-07-16 16:23:55 +00:00
|
|
|
class VideoCaptureInterface;
|
2020-07-09 17:38:26 +00:00
|
|
|
enum class State;
|
2020-07-14 17:45:34 +00:00
|
|
|
enum class VideoState;
|
2020-08-05 12:11:18 +00:00
|
|
|
enum class AudioState;
|
2020-07-09 17:38:26 +00:00
|
|
|
} // namespace tgcalls
|
2017-04-19 18:06:01 +00:00
|
|
|
|
2020-08-11 09:59:48 +00:00
|
|
|
namespace Webrtc {
|
2020-08-05 12:11:18 +00:00
|
|
|
enum class VideoState;
|
2020-07-17 18:59:38 +00:00
|
|
|
class VideoTrack;
|
2020-08-11 09:59:48 +00:00
|
|
|
} // namespace Webrtc
|
2020-07-17 18:59:38 +00:00
|
|
|
|
2017-04-19 18:06:01 +00:00
|
|
|
namespace Calls {
|
|
|
|
|
|
|
|
struct DhConfig {
|
|
|
|
int32 version = 0;
|
|
|
|
int32 g = 0;
|
2018-03-27 12:16:00 +00:00
|
|
|
bytes::vector p;
|
2017-04-19 18:06:01 +00:00
|
|
|
};
|
|
|
|
|
2020-08-18 14:00:33 +00:00
|
|
|
enum class ErrorType {
|
|
|
|
NoCamera,
|
|
|
|
NoMicrophone,
|
|
|
|
NotStartedCall,
|
|
|
|
NotVideoCall,
|
|
|
|
Unknown,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Error {
|
|
|
|
ErrorType type = ErrorType::Unknown;
|
|
|
|
QString details;
|
|
|
|
};
|
|
|
|
|
2021-04-22 16:05:17 +00:00
|
|
|
enum class CallType {
|
|
|
|
Incoming,
|
|
|
|
Outgoing,
|
|
|
|
};
|
|
|
|
|
2019-11-27 08:02:56 +00:00
|
|
|
class Call : public base::has_weak_ptr {
|
2017-04-19 18:06:01 +00:00
|
|
|
public:
|
|
|
|
class Delegate {
|
|
|
|
public:
|
|
|
|
virtual DhConfig getDhConfig() const = 0;
|
2017-08-17 08:31:24 +00:00
|
|
|
virtual void callFinished(not_null<Call*> call) = 0;
|
|
|
|
virtual void callFailed(not_null<Call*> call) = 0;
|
|
|
|
virtual void callRedial(not_null<Call*> call) = 0;
|
2017-04-19 18:06:01 +00:00
|
|
|
|
2020-12-18 08:44:19 +00:00
|
|
|
enum class CallSound {
|
2017-05-03 13:43:01 +00:00
|
|
|
Connecting,
|
|
|
|
Busy,
|
|
|
|
Ended,
|
|
|
|
};
|
2020-12-18 08:44:19 +00:00
|
|
|
virtual void callPlaySound(CallSound sound) = 0;
|
2020-11-30 20:56:43 +00:00
|
|
|
virtual void callRequestPermissionsOrFail(
|
|
|
|
Fn<void()> onSuccess,
|
|
|
|
bool video) = 0;
|
2020-12-18 08:44:19 +00:00
|
|
|
|
2021-08-09 12:05:58 +00:00
|
|
|
virtual auto callGetVideoCapture(
|
|
|
|
const QString &deviceId,
|
|
|
|
bool isScreenCapture)
|
|
|
|
-> std::shared_ptr<tgcalls::VideoCaptureInterface> = 0;
|
2017-05-03 13:43:01 +00:00
|
|
|
|
2020-05-19 06:33:57 +00:00
|
|
|
virtual ~Delegate() = default;
|
2018-09-18 10:43:14 +00:00
|
|
|
|
2017-04-19 18:06:01 +00:00
|
|
|
};
|
|
|
|
|
2017-05-07 19:09:20 +00:00
|
|
|
static constexpr auto kSoundSampleMs = 100;
|
2017-04-19 18:06:01 +00:00
|
|
|
|
2021-04-22 16:05:17 +00:00
|
|
|
using Type = CallType;
|
|
|
|
Call(
|
|
|
|
not_null<Delegate*> delegate,
|
|
|
|
not_null<UserData*> user,
|
|
|
|
Type type,
|
|
|
|
bool video);
|
2017-04-19 20:25:48 +00:00
|
|
|
|
2020-06-25 17:57:36 +00:00
|
|
|
[[nodiscard]] Type type() const {
|
2017-04-19 20:25:48 +00:00
|
|
|
return _type;
|
|
|
|
}
|
2020-06-25 17:57:36 +00:00
|
|
|
[[nodiscard]] not_null<UserData*> user() const {
|
2017-04-19 20:25:48 +00:00
|
|
|
return _user;
|
|
|
|
}
|
2021-11-01 00:56:30 +00:00
|
|
|
[[nodiscard]] CallId id() const {
|
2021-10-27 07:27:04 +00:00
|
|
|
return _id;
|
|
|
|
}
|
2020-06-25 17:57:36 +00:00
|
|
|
[[nodiscard]] bool isIncomingWaiting() const;
|
2017-04-19 18:06:01 +00:00
|
|
|
|
2018-03-27 12:16:00 +00:00
|
|
|
void start(bytes::const_span random);
|
2017-04-19 18:06:01 +00:00
|
|
|
bool handleUpdate(const MTPPhoneCall &call);
|
2020-05-19 06:33:57 +00:00
|
|
|
bool handleSignalingData(const MTPDupdatePhoneCallSignalingData &data);
|
2017-04-19 18:06:01 +00:00
|
|
|
|
2017-04-19 20:25:48 +00:00
|
|
|
enum State {
|
2017-04-25 20:36:04 +00:00
|
|
|
Starting,
|
2017-04-19 20:25:48 +00:00
|
|
|
WaitingInit,
|
|
|
|
WaitingInitAck,
|
|
|
|
Established,
|
2017-05-09 12:06:21 +00:00
|
|
|
FailedHangingUp,
|
2017-04-19 20:25:48 +00:00
|
|
|
Failed,
|
|
|
|
HangingUp,
|
|
|
|
Ended,
|
2017-05-12 16:09:34 +00:00
|
|
|
EndedByOtherDevice,
|
2017-04-19 20:25:48 +00:00
|
|
|
ExchangingKeys,
|
|
|
|
Waiting,
|
|
|
|
Requesting,
|
|
|
|
WaitingIncoming,
|
|
|
|
Ringing,
|
|
|
|
Busy,
|
2023-02-27 15:30:50 +00:00
|
|
|
WaitingUserConfirmation,
|
2017-04-19 20:25:48 +00:00
|
|
|
};
|
2020-07-16 16:23:55 +00:00
|
|
|
[[nodiscard]] State state() const {
|
2020-06-25 17:57:36 +00:00
|
|
|
return _state.current();
|
2017-04-19 20:25:48 +00:00
|
|
|
}
|
2020-07-16 16:23:55 +00:00
|
|
|
[[nodiscard]] rpl::producer<State> stateValue() const {
|
2020-06-25 17:57:36 +00:00
|
|
|
return _state.value();
|
2017-04-19 20:25:48 +00:00
|
|
|
}
|
2017-04-25 16:45:41 +00:00
|
|
|
|
2020-08-18 14:00:33 +00:00
|
|
|
[[nodiscard]] rpl::producer<Error> errors() const {
|
|
|
|
return _errors.events();
|
|
|
|
}
|
|
|
|
|
2020-08-05 12:11:18 +00:00
|
|
|
enum class RemoteAudioState {
|
|
|
|
Muted,
|
|
|
|
Active,
|
2020-07-16 16:23:55 +00:00
|
|
|
};
|
2020-08-05 12:11:18 +00:00
|
|
|
[[nodiscard]] RemoteAudioState remoteAudioState() const {
|
|
|
|
return _remoteAudioState.current();
|
|
|
|
}
|
|
|
|
[[nodiscard]] auto remoteAudioStateValue() const
|
|
|
|
-> rpl::producer<RemoteAudioState> {
|
|
|
|
return _remoteAudioState.value();
|
|
|
|
}
|
|
|
|
|
2020-08-11 09:59:48 +00:00
|
|
|
[[nodiscard]] Webrtc::VideoState remoteVideoState() const {
|
2020-08-05 12:11:18 +00:00
|
|
|
return _remoteVideoState.current();
|
2020-07-16 16:23:55 +00:00
|
|
|
}
|
2020-08-05 12:11:18 +00:00
|
|
|
[[nodiscard]] auto remoteVideoStateValue() const
|
2020-08-11 09:59:48 +00:00
|
|
|
-> rpl::producer<Webrtc::VideoState> {
|
2020-08-05 12:11:18 +00:00
|
|
|
return _remoteVideoState.value();
|
2020-07-16 16:23:55 +00:00
|
|
|
}
|
|
|
|
|
2018-05-27 08:24:47 +00:00
|
|
|
static constexpr auto kSignalBarStarting = -1;
|
|
|
|
static constexpr auto kSignalBarFinished = -2;
|
|
|
|
static constexpr auto kSignalBarCount = 4;
|
2020-07-16 16:23:55 +00:00
|
|
|
[[nodiscard]] rpl::producer<int> signalBarCountValue() const {
|
|
|
|
return _signalBarCount.value();
|
|
|
|
}
|
|
|
|
|
|
|
|
void setMuted(bool mute);
|
|
|
|
[[nodiscard]] bool muted() const {
|
|
|
|
return _muted.current();
|
|
|
|
}
|
|
|
|
[[nodiscard]] rpl::producer<bool> mutedValue() const {
|
|
|
|
return _muted.value();
|
2018-05-27 08:24:47 +00:00
|
|
|
}
|
|
|
|
|
2020-08-11 09:59:48 +00:00
|
|
|
[[nodiscard]] not_null<Webrtc::VideoTrack*> videoIncoming() const;
|
|
|
|
[[nodiscard]] not_null<Webrtc::VideoTrack*> videoOutgoing() const;
|
2020-05-19 06:33:57 +00:00
|
|
|
|
2019-02-19 06:57:53 +00:00
|
|
|
crl::time getDurationMs() const;
|
2017-05-07 19:09:20 +00:00
|
|
|
float64 getWaitingSoundPeakValue() const;
|
2017-04-19 20:25:48 +00:00
|
|
|
|
2023-02-27 15:30:50 +00:00
|
|
|
void applyUserConfirmation();
|
2017-04-19 20:25:48 +00:00
|
|
|
void answer();
|
|
|
|
void hangup();
|
2017-04-29 18:00:27 +00:00
|
|
|
void redial();
|
2017-04-25 16:45:41 +00:00
|
|
|
|
|
|
|
bool isKeyShaForFingerprintReady() const;
|
2018-03-27 12:16:00 +00:00
|
|
|
bytes::vector getKeyShaForFingerprint() const;
|
2017-04-19 20:25:48 +00:00
|
|
|
|
2017-05-04 12:28:37 +00:00
|
|
|
QString getDebugLog() const;
|
2019-03-12 11:16:58 +00:00
|
|
|
|
2020-08-21 10:50:13 +00:00
|
|
|
void setCurrentAudioDevice(bool input, const QString &deviceId);
|
2020-11-27 10:35:16 +00:00
|
|
|
//void setAudioVolume(bool input, float level);
|
2019-01-05 11:08:02 +00:00
|
|
|
void setAudioDuckingEnabled(bool enabled);
|
2017-05-04 12:28:37 +00:00
|
|
|
|
2021-07-27 18:09:13 +00:00
|
|
|
void setCurrentCameraDevice(const QString &deviceId);
|
|
|
|
[[nodiscard]] QString videoDeviceId() const {
|
|
|
|
return _videoCaptureDeviceId;
|
|
|
|
}
|
|
|
|
|
|
|
|
[[nodiscard]] bool isSharingVideo() const;
|
|
|
|
[[nodiscard]] bool isSharingCamera() const;
|
|
|
|
[[nodiscard]] bool isSharingScreen() const;
|
|
|
|
[[nodiscard]] QString cameraSharingDeviceId() const;
|
|
|
|
[[nodiscard]] QString screenSharingDeviceId() const;
|
|
|
|
void toggleCameraSharing(bool enabled);
|
|
|
|
void toggleScreenSharing(std::optional<QString> uniqueId);
|
|
|
|
|
2020-06-25 17:57:36 +00:00
|
|
|
[[nodiscard]] rpl::lifetime &lifetime() {
|
|
|
|
return _lifetime;
|
|
|
|
}
|
|
|
|
|
2017-04-19 18:06:01 +00:00
|
|
|
~Call();
|
|
|
|
|
|
|
|
private:
|
2017-05-09 12:06:21 +00:00
|
|
|
enum class FinishType {
|
|
|
|
None,
|
|
|
|
Ended,
|
|
|
|
Failed,
|
|
|
|
};
|
2020-11-24 11:56:46 +00:00
|
|
|
|
2021-10-22 00:14:48 +00:00
|
|
|
void handleRequestError(const QString &error);
|
2020-05-14 14:12:07 +00:00
|
|
|
void handleControllerError(const QString &error);
|
2020-08-05 12:11:18 +00:00
|
|
|
void finish(
|
|
|
|
FinishType type,
|
|
|
|
const MTPPhoneCallDiscardReason &reason
|
|
|
|
= MTP_phoneCallDiscardReasonDisconnect());
|
2017-04-19 20:25:48 +00:00
|
|
|
void startOutgoing();
|
|
|
|
void startIncoming();
|
2017-05-03 13:43:01 +00:00
|
|
|
void startWaitingTrack();
|
2020-05-19 06:33:57 +00:00
|
|
|
void sendSignalingData(const QByteArray &data);
|
2017-04-19 18:06:01 +00:00
|
|
|
|
2018-03-27 12:16:00 +00:00
|
|
|
void generateModExpFirst(bytes::const_span randomSeed);
|
2020-08-05 12:11:18 +00:00
|
|
|
void handleControllerStateChange(tgcalls::State state);
|
2020-05-14 14:12:07 +00:00
|
|
|
void handleControllerBarCountChange(int count);
|
2017-04-19 18:06:01 +00:00
|
|
|
void createAndStartController(const MTPDphoneCall &call);
|
|
|
|
|
2017-04-19 20:25:48 +00:00
|
|
|
template <typename T>
|
|
|
|
bool checkCallCommonFields(const T &call);
|
2017-04-19 18:06:01 +00:00
|
|
|
bool checkCallFields(const MTPDphoneCall &call);
|
|
|
|
bool checkCallFields(const MTPDphoneCallAccepted &call);
|
|
|
|
|
2018-09-30 15:42:50 +00:00
|
|
|
void actuallyAnswer();
|
2017-04-19 18:06:01 +00:00
|
|
|
void confirmAcceptedCall(const MTPDphoneCallAccepted &call);
|
2017-04-19 20:25:48 +00:00
|
|
|
void startConfirmedCall(const MTPDphoneCall &call);
|
|
|
|
void setState(State state);
|
|
|
|
void setStateQueued(State state);
|
2020-05-14 14:12:07 +00:00
|
|
|
void setFailedQueued(const QString &error);
|
2018-05-27 08:24:47 +00:00
|
|
|
void setSignalBarCount(int count);
|
2017-04-29 18:00:27 +00:00
|
|
|
void destroyController();
|
2017-04-19 18:06:01 +00:00
|
|
|
|
2020-07-31 14:36:35 +00:00
|
|
|
void setupOutgoingVideo();
|
2020-08-05 12:11:18 +00:00
|
|
|
void updateRemoteMediaState(
|
|
|
|
tgcalls::AudioState audio,
|
|
|
|
tgcalls::VideoState video);
|
2020-07-31 14:36:35 +00:00
|
|
|
|
|
|
|
const not_null<Delegate*> _delegate;
|
|
|
|
const not_null<UserData*> _user;
|
2019-11-27 08:02:56 +00:00
|
|
|
MTP::Sender _api;
|
2017-04-19 20:25:48 +00:00
|
|
|
Type _type = Type::Outgoing;
|
2023-03-02 05:52:24 +00:00
|
|
|
rpl::variable<State> _state = State::Starting;
|
2021-10-22 00:14:48 +00:00
|
|
|
rpl::variable<RemoteAudioState> _remoteAudioState =
|
|
|
|
RemoteAudioState::Active;
|
2020-08-11 09:59:48 +00:00
|
|
|
rpl::variable<Webrtc::VideoState> _remoteVideoState;
|
2020-08-18 14:00:33 +00:00
|
|
|
rpl::event_stream<Error> _errors;
|
2017-05-09 12:06:21 +00:00
|
|
|
FinishType _finishAfterRequestingCall = FinishType::None;
|
2017-05-10 07:58:02 +00:00
|
|
|
bool _answerAfterDhConfigReceived = false;
|
2020-07-16 16:23:55 +00:00
|
|
|
rpl::variable<int> _signalBarCount = kSignalBarStarting;
|
2019-02-19 06:57:53 +00:00
|
|
|
crl::time _startTime = 0;
|
2017-04-28 17:16:14 +00:00
|
|
|
base::DelayedCallTimer _finishByTimeoutTimer;
|
|
|
|
base::Timer _discardByTimeoutTimer;
|
|
|
|
|
2020-07-16 16:23:55 +00:00
|
|
|
rpl::variable<bool> _muted = false;
|
2017-04-19 20:25:48 +00:00
|
|
|
|
|
|
|
DhConfig _dhConfig;
|
2018-03-27 12:16:00 +00:00
|
|
|
bytes::vector _ga;
|
|
|
|
bytes::vector _gb;
|
|
|
|
bytes::vector _gaHash;
|
|
|
|
bytes::vector _randomPower;
|
2017-04-24 12:16:38 +00:00
|
|
|
MTP::AuthKey::Data _authKey;
|
2017-04-19 20:25:48 +00:00
|
|
|
|
2021-11-01 00:56:30 +00:00
|
|
|
CallId _id = 0;
|
2017-04-19 18:06:01 +00:00
|
|
|
uint64 _accessHash = 0;
|
|
|
|
uint64 _keyFingerprint = 0;
|
|
|
|
|
2020-07-09 17:38:26 +00:00
|
|
|
std::unique_ptr<tgcalls::Instance> _instance;
|
2020-07-16 16:23:55 +00:00
|
|
|
std::shared_ptr<tgcalls::VideoCaptureInterface> _videoCapture;
|
2021-07-27 18:09:13 +00:00
|
|
|
QString _videoCaptureDeviceId;
|
|
|
|
bool _videoCaptureIsScreencast = false;
|
2020-08-11 09:59:48 +00:00
|
|
|
const std::unique_ptr<Webrtc::VideoTrack> _videoIncoming;
|
|
|
|
const std::unique_ptr<Webrtc::VideoTrack> _videoOutgoing;
|
2017-04-19 18:06:01 +00:00
|
|
|
|
2017-05-03 13:43:01 +00:00
|
|
|
std::unique_ptr<Media::Audio::Track> _waitingTrack;
|
|
|
|
|
2020-06-25 17:57:36 +00:00
|
|
|
rpl::lifetime _lifetime;
|
|
|
|
|
2017-04-19 18:06:01 +00:00
|
|
|
};
|
|
|
|
|
2020-05-14 14:12:07 +00:00
|
|
|
void UpdateConfig(const std::string &data);
|
2017-04-28 15:00:16 +00:00
|
|
|
|
2017-04-19 18:06:01 +00:00
|
|
|
} // namespace Calls
|