2
0
mirror of https://github.com/telegramdesktop/tdesktop synced 2025-08-25 11:47:29 +00:00
tdesktop/Telegram/SourceFiles/settings/settings_calls.cpp

668 lines
18 KiB
C++
Raw Normal View History

2019-01-05 14:08:02 +03:00
/*
This file is part of Telegram Desktop,
the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#include "settings/settings_calls.h"
#include "ui/wrap/vertical_layout.h"
#include "ui/wrap/slide_wrap.h"
#include "ui/widgets/labels.h"
#include "ui/widgets/checkbox.h"
#include "ui/widgets/level_meter.h"
#include "ui/widgets/buttons.h"
#include "ui/boxes/single_choice_box.h"
2021-10-19 00:36:55 +03:00
#include "ui/boxes/confirm_box.h"
#include "ui/vertical_list.h"
2019-01-05 14:08:02 +03:00
#include "platform/platform_specific.h"
#include "main/main_session.h"
2019-01-05 14:08:02 +03:00
#include "lang/lang_keys.h"
#include "styles/style_settings.h"
#include "ui/widgets/continuous_sliders.h"
2019-08-06 17:40:08 +01:00
#include "window/window_session_controller.h"
#include "core/application.h"
#include "core/core_settings.h"
#include "calls/calls_call.h"
2019-01-05 14:08:02 +03:00
#include "calls/calls_instance.h"
#include "calls/calls_video_bubble.h"
#include "apiwrap.h"
#include "api/api_authorizations.h"
#include "webrtc/webrtc_environment.h"
#include "webrtc/webrtc_media_devices.h"
#include "webrtc/webrtc_video_track.h"
#include "webrtc/webrtc_audio_input_tester.h"
#include "webrtc/webrtc_create_adm.h" // Webrtc::Backend.
#include "tgcalls/VideoCaptureInterface.h"
#include "styles/style_layers.h"
2019-01-05 14:08:02 +03:00
namespace Settings {
namespace {
using namespace Webrtc;
[[nodiscard]] rpl::producer<QString> DeviceNameValue(
DeviceType type,
rpl::producer<QString> id) {
return std::move(id) | rpl::map([type](const QString &id) {
const auto list = Core::App().mediaDevices().devices(type);
const auto i = ranges::find(list, id, &DeviceInfo::id);
return (i != end(list))
? i->name
: tr::lng_settings_call_device_default(tr::now);
});
}
} // namespace
2019-01-05 14:08:02 +03:00
2019-07-24 16:00:30 +02:00
Calls::Calls(
QWidget *parent,
not_null<Window::SessionController*> controller)
: Section(parent)
, _controller(controller) {
// Request valid value of calls disabled flag.
controller->session().api().authorizations().reload();
setupContent();
requestPermissionAndStartTestingMicrophone();
2019-01-05 14:08:02 +03:00
}
2020-11-28 19:18:51 +03:00
Calls::~Calls() = default;
2019-01-05 14:08:02 +03:00
rpl::producer<QString> Calls::title() {
2024-01-23 15:06:55 +04:00
return tr::lng_settings_section_devices();
}
Webrtc::VideoTrack *Calls::AddCameraSubsection(
std::shared_ptr<Ui::Show> show,
not_null<Ui::VerticalLayout*> content,
bool saveToSettings) {
auto &lifetime = content->lifetime();
const auto hasCall = (Core::App().calls().currentCall() != nullptr);
const auto cameraNameStream = lifetime.make_state<
rpl::event_stream<QString>
>();
auto capturerOwner = lifetime.make_state<
std::shared_ptr<tgcalls::VideoCaptureInterface>
>();
const auto track = lifetime.make_state<VideoTrack>(
(hasCall
? VideoState::Inactive
: VideoState::Active));
const auto currentCameraName = [&] {
const auto cameras = GetVideoInputList();
const auto i = ranges::find(
cameras,
Core::App().settings().cameraDeviceId(),
&VideoInput::id);
return (i != end(cameras))
? i->name
: tr::lng_settings_call_device_default(tr::now);
}();
AddButtonWithLabel(
content,
tr::lng_settings_call_input_device(),
rpl::single(
currentCameraName
) | rpl::then(
cameraNameStream->events()
),
st::settingsButtonNoIcon
)->addClickHandler([=] {
const auto &devices = GetVideoInputList();
const auto options = ranges::views::concat(
ranges::views::single(
tr::lng_settings_call_device_default(tr::now)),
devices | ranges::views::transform(&VideoInput::name)
) | ranges::to_vector;
const auto i = ranges::find(
devices,
Core::App().settings().cameraDeviceId(),
&VideoInput::id);
const auto currentOption = (i != end(devices))
? int(i - begin(devices) + 1)
: 0;
const auto save = crl::guard(content, [=](int option) {
cameraNameStream->fire_copy(options[option]);
const auto deviceId = option
? devices[option - 1].id
: kDefaultDeviceId;
if (saveToSettings) {
Core::App().settings().setCameraDeviceId(deviceId);
Core::App().saveSettingsDelayed();
}
if (*capturerOwner) {
(*capturerOwner)->switchToDevice(
deviceId.toStdString(),
false);
}
});
show->showBox(Box([=](not_null<Ui::GenericBox*> box) {
SingleChoiceBox(box, {
.title = tr::lng_settings_call_camera(),
.options = options,
.initialSelection = currentOption,
.callback = save,
});
}));
});
const auto bubbleWrap = content->add(object_ptr<Ui::RpWidget>(content));
const auto bubble = lifetime.make_state<::Calls::VideoBubble>(
bubbleWrap,
track);
const auto padding = st::settingsButtonNoIcon.padding.left();
const auto top = st::boxRoundShadow.extend.top();
const auto bottom = st::boxRoundShadow.extend.bottom();
auto frameSize = track->renderNextFrame(
) | rpl::map([=] {
return track->frameSize();
}) | rpl::filter([=](QSize size) {
return !size.isEmpty()
&& !Core::App().calls().currentCall()
&& !Core::App().calls().currentGroupCall();
});
auto bubbleWidth = bubbleWrap->widthValue(
) | rpl::filter([=](int width) {
return width > 2 * padding + 1;
});
rpl::combine(
std::move(bubbleWidth),
std::move(frameSize)
) | rpl::start_with_next([=](int width, QSize frame) {
const auto useWidth = (width - 2 * padding);
const auto useHeight = std::min(
((useWidth * frame.height()) / frame.width()),
(useWidth * 480) / 640);
bubbleWrap->resize(width, top + useHeight + bottom);
bubble->updateGeometry(
::Calls::VideoBubble::DragMode::None,
QRect(padding, top, useWidth, useHeight));
bubbleWrap->update();
}, bubbleWrap->lifetime());
using namespace rpl::mappers;
const auto checkCapturer = [=] {
if (*capturerOwner
|| Core::App().calls().currentCall()
|| Core::App().calls().currentGroupCall()) {
return;
}
*capturerOwner = Core::App().calls().getVideoCapture(
Core::App().settings().cameraDeviceId(),
false);
(*capturerOwner)->setPreferredAspectRatio(0.);
track->setState(VideoState::Active);
(*capturerOwner)->setState(tgcalls::VideoState::Active);
(*capturerOwner)->setOutput(track->sink());
};
rpl::combine(
Core::App().calls().currentCallValue(),
Core::App().calls().currentGroupCallValue(),
_1 || _2
) | rpl::start_with_next([=](bool has) {
if (has) {
track->setState(VideoState::Inactive);
bubbleWrap->resize(bubbleWrap->width(), 0);
*capturerOwner = nullptr;
} else {
crl::on_main(content, checkCapturer);
}
}, lifetime);
return track;
}
2019-01-11 14:07:56 +04:00
void Calls::sectionSaveChanges(FnMut<void()> done) {
2019-01-05 14:08:02 +03:00
if (_micTester) {
_micTester.reset();
}
done();
}
void Calls::setupContent() {
2019-01-05 14:08:02 +03:00
const auto content = Ui::CreateChild<Ui::VerticalLayout>(this);
2024-01-23 15:06:55 +04:00
const auto settings = &Core::App().settings();
2019-01-05 14:08:02 +03:00
Ui::AddSkip(content);
Ui::AddSubsectionTitle(content, tr::lng_settings_call_section_output());
2024-01-23 15:06:55 +04:00
const auto playbackIdWithFallback = [=] {
return DeviceIdOrDefault(settings->playbackDeviceIdValue());
};
2019-01-11 14:07:56 +04:00
AddButtonWithLabel(
2019-01-05 14:08:02 +03:00
content,
2019-06-18 14:16:43 +02:00
tr::lng_settings_call_output_device(),
PlaybackDeviceNameValue(playbackIdWithFallback()),
st::settingsButtonNoIcon
2019-01-11 14:07:56 +04:00
)->addClickHandler([=] {
_controller->show(ChoosePlaybackDeviceBox(
playbackIdWithFallback(),
crl::guard(this, [=](const QString &id) {
2024-01-23 15:06:55 +04:00
settings->setPlaybackDeviceId(id);
Core::App().saveSettingsDelayed();
})));
2019-01-05 14:08:02 +03:00
});
Ui::AddSkip(content);
Ui::AddDivider(content);
Ui::AddSkip(content);
Ui::AddSubsectionTitle(content, tr::lng_settings_call_section_input());
2024-01-23 15:06:55 +04:00
const auto captureIdWithFallback = [=] {
return DeviceIdOrDefault(settings->captureDeviceIdValue());
};
2019-01-11 14:07:56 +04:00
AddButtonWithLabel(
2019-01-05 14:08:02 +03:00
content,
2019-06-18 14:16:43 +02:00
tr::lng_settings_call_input_device(),
CaptureDeviceNameValue(captureIdWithFallback()),
st::settingsButtonNoIcon
2019-01-11 14:07:56 +04:00
)->addClickHandler([=] {
_controller->show(ChooseCaptureDeviceBox(
captureIdWithFallback(),
crl::guard(this, [=](const QString &id) {
2024-01-23 15:06:55 +04:00
settings->setCaptureDeviceId(id);
Core::App().saveSettingsDelayed();
if (_micTester) {
_micTester->setDeviceId(id);
}
})));
2019-01-05 14:08:02 +03:00
});
2019-01-11 14:07:56 +04:00
_micTestLevel = content->add(
object_ptr<Ui::LevelMeter>(
content,
st::defaultLevelMeter),
st::settingsLevelMeterPadding);
_micTestLevel->resize(QSize(0, st::defaultLevelMeter.height));
_levelUpdateTimer.setCallback([=] {
const auto was = _micLevel;
_micLevel = _micTester->getAndResetLevel();
_micLevelAnimation.start([=] {
_micTestLevel->setValue(_micLevelAnimation.value(_micLevel));
}, was, _micLevel, kMicTestAnimationDuration);
2019-01-05 14:08:02 +03:00
});
Ui::AddSkip(content);
Ui::AddDivider(content);
2024-01-23 15:06:55 +04:00
Ui::AddSkip(content);
Ui::AddSubsectionTitle(content, tr::lng_settings_devices_calls());
const auto orDefault = [](const QString &value) {
return value.isEmpty() ? kDefaultDeviceId : value;
};
const auto same = content->add(object_ptr<Ui::SettingsButton>(
content,
tr::lng_settings_devices_calls_same(),
st::settingsButtonNoIcon));
same->toggleOn(rpl::combine(
settings->callPlaybackDeviceIdValue(),
settings->callCaptureDeviceIdValue()
) | rpl::map([](const QString &playback, const QString &capture) {
return playback.isEmpty() && capture.isEmpty();
}));
same->toggledValue() | rpl::filter([=](bool toggled) {
const auto empty = settings->callPlaybackDeviceId().isEmpty()
&& settings->callCaptureDeviceId().isEmpty();
return (empty != toggled);
}) | rpl::start_with_next([=](bool toggled) {
if (toggled) {
settings->setCallPlaybackDeviceId(QString());
settings->setCallCaptureDeviceId(QString());
} else {
settings->setCallPlaybackDeviceId(
orDefault(settings->playbackDeviceId()));
settings->setCallCaptureDeviceId(
orDefault(settings->captureDeviceId()));
}
Core::App().saveSettingsDelayed();
}, same->lifetime());
const auto different = content->add(
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
content,
object_ptr<Ui::VerticalLayout>(content)));
const auto calls = different->entity();
const auto callPlaybackIdWithFallback = [=] {
return DeviceIdValueWithFallback(
settings->callPlaybackDeviceIdValue(),
settings->playbackDeviceIdValue());
};
AddButtonWithLabel(
calls,
tr::lng_group_call_speakers(),
PlaybackDeviceNameValue(callPlaybackIdWithFallback()),
st::settingsButtonNoIcon
)->addClickHandler([=] {
_controller->show(ChoosePlaybackDeviceBox(
callPlaybackIdWithFallback(),
crl::guard(this, [=](const QString &id) {
settings->setCallPlaybackDeviceId(orDefault(id));
Core::App().saveSettingsDelayed();
})));
});
const auto callCaptureIdWithFallback = [=] {
return DeviceIdValueWithFallback(
settings->callCaptureDeviceIdValue(),
settings->captureDeviceIdValue());
};
AddButtonWithLabel(
calls,
tr::lng_group_call_microphone(),
CaptureDeviceNameValue(callCaptureIdWithFallback()),
st::settingsButtonNoIcon
)->addClickHandler([=] {
_controller->show(ChooseCaptureDeviceBox(
callCaptureIdWithFallback(),
crl::guard(this, [=](const QString &id) {
settings->setCallCaptureDeviceId(orDefault(id));
Core::App().saveSettingsDelayed();
//if (_micTester) {
// _micTester->setDeviceId(id);
//}
})));
});
different->toggleOn(same->toggledValue() | rpl::map(!rpl::mappers::_1));
Ui::AddSkip(content);
Ui::AddDivider(content);
if (!GetVideoInputList().empty()) {
Ui::AddSkip(content);
Ui::AddSubsectionTitle(content, tr::lng_settings_call_camera());
AddCameraSubsection(_controller->uiShow(), content, true);
Ui::AddSkip(content);
Ui::AddDivider(content);
}
Ui::AddSkip(content);
Ui::AddSubsectionTitle(content, tr::lng_settings_call_section_other());
2019-01-05 14:08:02 +03:00
const auto api = &_controller->session().api();
content->add(object_ptr<Ui::SettingsButton>(
content,
tr::lng_settings_call_accept_calls(),
st::settingsButtonNoIcon
))->toggleOn(
api->authorizations().callsDisabledHereValue(
) | rpl::map(!rpl::mappers::_1)
)->toggledChanges(
) | rpl::filter([=](bool value) {
return (value == api->authorizations().callsDisabledHere());
}) | start_with_next([=](bool value) {
api->authorizations().toggleCallsDisabledHere(!value);
}, content->lifetime());
content->add(object_ptr<Ui::SettingsButton>(
2019-01-11 14:07:56 +04:00
content,
2019-06-18 14:16:43 +02:00
tr::lng_settings_call_open_system_prefs(),
st::settingsButtonNoIcon
))->addClickHandler([=] {
using namespace ::Platform;
const auto opened = OpenSystemSettings(SystemSettingsType::Audio);
2019-01-11 14:07:56 +04:00
if (!opened) {
_controller->show(
Ui::MakeInformBox(tr::lng_linux_no_audio_prefs()));
2019-01-05 14:08:02 +03:00
}
});
Ui::AddSkip(content);
2019-01-05 14:08:02 +03:00
Ui::ResizeFitChild(this, content);
}
2019-01-11 14:07:56 +04:00
void Calls::requestPermissionAndStartTestingMicrophone() {
using namespace ::Platform;
const auto status = GetPermissionStatus(
PermissionType::Microphone);
if (status == PermissionStatus::Granted) {
2019-01-05 14:08:02 +03:00
startTestingMicrophone();
} else if (status == PermissionStatus::CanRequest) {
2019-01-11 14:07:56 +04:00
const auto startTestingChecked = crl::guard(this, [=](
PermissionStatus status) {
if (status == PermissionStatus::Granted) {
2019-01-11 14:07:56 +04:00
crl::on_main(crl::guard(this, [=] {
2019-01-05 14:08:02 +03:00
startTestingMicrophone();
}));
}
2019-01-11 14:07:56 +04:00
});
RequestPermission(
PermissionType::Microphone,
2019-01-11 14:07:56 +04:00
startTestingChecked);
2019-01-05 14:08:02 +03:00
} else {
const auto showSystemSettings = [controller = _controller] {
OpenSystemSettingsForPermission(
PermissionType::Microphone);
controller->hideLayer();
2019-01-11 14:07:56 +04:00
};
_controller->show(Ui::MakeConfirmBox({
.text = tr::lng_no_mic_permission(),
.confirmed = showSystemSettings,
.confirmText = tr::lng_menu_settings(),
}));
2019-01-05 14:08:02 +03:00
}
}
2019-01-11 14:07:56 +04:00
void Calls::startTestingMicrophone() {
_levelUpdateTimer.callEach(kMicTestUpdateInterval);
_micTester = std::make_unique<AudioInputTester>(
Core::App().settings().callAudioBackend(),
Core::App().settings().callCaptureDeviceId());
2019-01-05 14:08:02 +03:00
}
rpl::producer<QString> PlaybackDeviceNameValue(rpl::producer<QString> id) {
return DeviceNameValue(DeviceType::Playback, std::move(id));
2020-11-28 19:18:51 +03:00
}
rpl::producer<QString> CaptureDeviceNameValue(rpl::producer<QString> id) {
return DeviceNameValue(DeviceType::Capture, std::move(id));
2020-11-28 19:18:51 +03:00
}
void ChooseAudioDeviceBox(
not_null<Ui::GenericBox*> box,
rpl::producer<QString> title,
rpl::producer<std::vector<DeviceInfo>> devicesValue,
rpl::producer<QString> currentId,
Fn<void(QString id)> chosen,
const style::Checkbox *st,
const style::Radio *radioSt) {
box->setTitle(std::move(title));
box->addButton(tr::lng_box_ok(), [=] { box->closeBox(); });
const auto layout = box->verticalLayout();
const auto skip = st::boxOptionListPadding.top()
+ st::defaultBoxCheckbox.margin.top();
layout->add(object_ptr<Ui::FixedHeightWidget>(layout, skip));
if (!st) {
st = &st::defaultBoxCheckbox;
}
if (!radioSt) {
radioSt = &st::defaultRadio;
}
struct State {
std::vector<DeviceInfo> list;
base::flat_map<int, QString> ids;
rpl::variable<QString> currentId;
QString currentName;
bool ignoreValueChange = false;
};
const auto state = box->lifetime().make_state<State>();
state->currentId = std::move(currentId);
const auto group = std::make_shared<Ui::RadiobuttonGroup>();
const auto fake = std::make_shared<Ui::RadiobuttonGroup>(0);
const auto buttons = layout->add(object_ptr<Ui::VerticalLayout>(layout));
const auto other = layout->add(object_ptr<Ui::VerticalLayout>(layout));
const auto margins = QMargins(
st::boxPadding.left() + st::boxOptionListPadding.left(),
0,
st::boxPadding.right(),
st::boxOptionListSkip);
const auto def = buttons->add(
object_ptr<Ui::Radiobutton>(
buttons,
group,
0,
tr::lng_settings_call_device_default(tr::now),
*st,
*radioSt),
margins);
const auto selectCurrent = [=](QString current) {
state->ignoreValueChange = true;
const auto guard = gsl::finally([&] {
state->ignoreValueChange = false;
});
if (current.isEmpty() || current == kDefaultDeviceId) {
group->setValue(0);
other->clear();
} else {
auto found = false;
for (const auto &[index, id] : state->ids) {
if (id == current) {
group->setValue(index);
found = true;
break;
}
}
if (found) {
other->clear();
} else {
group->setValue(0);
const auto i = ranges::find(
state->list,
current,
&DeviceInfo::id);
if (i != end(state->list)) {
const auto button = other->add(
object_ptr<Ui::Radiobutton>(
other,
fake,
0,
i->name,
*st,
*radioSt),
margins);
button->show();
button->setDisabled(true);
button->finishAnimating();
button->setAttribute(Qt::WA_TransparentForMouseEvents);
while (other->count() > 1) {
delete other->widgetAt(1);
}
if (const auto width = box->width()) {
other->resizeToWidth(width);
}
} else {
other->clear();
}
}
}
2020-11-28 19:18:51 +03:00
};
std::move(
devicesValue
) | rpl::start_with_next([=](std::vector<DeviceInfo> &&list) {
auto count = buttons->count();
auto index = 1;
state->ids.clear();
state->list = std::move(list);
state->ignoreValueChange = true;
const auto guard = gsl::finally([&] {
state->ignoreValueChange = false;
});
const auto current = state->currentId.current();
for (const auto &info : state->list) {
if (info.inactive) {
continue;
} else if (current == info.id) {
group->setValue(index);
}
const auto button = buttons->insert(
index,
object_ptr<Ui::Radiobutton>(
buttons,
group,
index,
info.name,
*st,
*radioSt),
margins);
button->show();
button->finishAnimating();
state->ids.emplace(index, info.id);
if (index < count) {
delete buttons->widgetAt(index + 1);
}
++index;
}
while (index < count) {
delete buttons->widgetAt(index);
--count;
}
if (const auto width = box->width()) {
buttons->resizeToWidth(width);
}
selectCurrent(current);
}, box->lifetime());
state->currentId.changes(
) | rpl::start_with_next(selectCurrent, box->lifetime());
def->finishAnimating();
group->setChangedCallback([=](int value) {
if (state->ignoreValueChange) {
return;
}
const auto weak = Ui::MakeWeak(box);
const auto i = state->ids.find(value);
chosen((i != end(state->ids)) ? i->second : kDefaultDeviceId);
if (weak) {
box->closeBox();
}
});
2020-11-28 19:18:51 +03:00
}
object_ptr<Ui::GenericBox> ChoosePlaybackDeviceBox(
rpl::producer<QString> currentId,
Fn<void(QString id)> chosen,
const style::Checkbox *st,
const style::Radio *radioSt) {
return Box(
ChooseAudioDeviceBox,
tr::lng_settings_call_output_device(),
Core::App().mediaDevices().devicesValue(DeviceType::Playback),
std::move(currentId),
std::move(chosen),
st,
radioSt);
}
object_ptr<Ui::GenericBox> ChooseCaptureDeviceBox(
rpl::producer<QString> currentId,
Fn<void(QString id)> chosen,
const style::Checkbox *st,
const style::Radio *radioSt) {
return Box(
ChooseAudioDeviceBox,
tr::lng_settings_call_input_device(),
Core::App().mediaDevices().devicesValue(DeviceType::Capture),
std::move(currentId),
std::move(chosen),
st,
radioSt);
2020-11-28 19:18:51 +03:00
}
2019-01-05 14:08:02 +03:00
} // namespace Settings