/* 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" #include "ui/boxes/confirm_box.h" #include "ui/vertical_list.h" #include "platform/platform_specific.h" #include "main/main_session.h" #include "lang/lang_keys.h" #include "styles/style_settings.h" #include "ui/widgets/continuous_sliders.h" #include "window/window_session_controller.h" #include "core/application.h" #include "core/core_settings.h" #include "calls/calls_call.h" #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" namespace Settings { namespace { using namespace Webrtc; [[nodiscard]] rpl::producer DeviceNameValue( DeviceType type, rpl::producer 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 Calls::Calls( QWidget *parent, not_null controller) : Section(parent) , _controller(controller) { // Request valid value of calls disabled flag. controller->session().api().authorizations().reload(); setupContent(); requestPermissionAndStartTestingMicrophone(); } Calls::~Calls() = default; rpl::producer Calls::title() { return tr::lng_settings_section_call_settings(); } Webrtc::VideoTrack *Calls::AddCameraSubsection( std::shared_ptr show, not_null content, bool saveToSettings) { auto &lifetime = content->lifetime(); const auto hasCall = (Core::App().calls().currentCall() != nullptr); const auto cameraNameStream = lifetime.make_state< rpl::event_stream >(); auto capturerOwner = lifetime.make_state< std::shared_ptr >(); const auto track = lifetime.make_state( (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 box) { SingleChoiceBox(box, { .title = tr::lng_settings_call_camera(), .options = options, .initialSelection = currentOption, .callback = save, }); })); }); const auto bubbleWrap = content->add(object_ptr(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; } void Calls::sectionSaveChanges(FnMut done) { if (_micTester) { _micTester.reset(); } done(); } void Calls::setupContent() { const auto content = Ui::CreateChild(this); Ui::AddSkip(content); Ui::AddSubsectionTitle(content, tr::lng_settings_call_section_output()); //auto playbackIdWithFallback = DeviceIdValueWithFallback( // Core::App().settings().callPlaybackDeviceIdValue(), // Core::App().settings().playbackDeviceIdValue()); auto playbackIdWithFallback = [] { return DeviceIdOrDefault( Core::App().settings().playbackDeviceIdValue()); }; AddButtonWithLabel( content, tr::lng_settings_call_output_device(), PlaybackDeviceNameValue(playbackIdWithFallback()), st::settingsButtonNoIcon )->addClickHandler([=] { _controller->show(ChoosePlaybackDeviceBox( playbackIdWithFallback(), crl::guard(this, [=](const QString &id) { //Core::App().settings().setCallPlaybackDeviceId(id); Core::App().settings().setPlaybackDeviceId(id); Core::App().saveSettingsDelayed(); }))); }); Ui::AddSkip(content); Ui::AddDivider(content); Ui::AddSkip(content); Ui::AddSubsectionTitle(content, tr::lng_settings_call_section_input()); //auto captureIdWithFallback = DeviceIdValueWithFallback( // Core::App().settings().callCaptureDeviceIdValue(), // Core::App().settings().captureDeviceIdValue()); auto captureIdWithFallback = [] { return DeviceIdOrDefault( Core::App().settings().captureDeviceIdValue()); }; AddButtonWithLabel( content, tr::lng_settings_call_input_device(), CaptureDeviceNameValue(captureIdWithFallback()), st::settingsButtonNoIcon )->addClickHandler([=] { _controller->show(ChooseCaptureDeviceBox( captureIdWithFallback(), crl::guard(this, [=](const QString &id) { //Core::App().settings().setCallCaptureDeviceId(id); Core::App().settings().setCaptureDeviceId(id); Core::App().saveSettingsDelayed(); if (_micTester) { _micTester->setDeviceId(id); } }))); }); _micTestLevel = content->add( object_ptr( 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); }); 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()); const auto api = &_controller->session().api(); content->add(object_ptr( 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( content, tr::lng_settings_call_open_system_prefs(), st::settingsButtonNoIcon ))->addClickHandler([=] { using namespace ::Platform; const auto opened = OpenSystemSettings(SystemSettingsType::Audio); if (!opened) { _controller->show( Ui::MakeInformBox(tr::lng_linux_no_audio_prefs())); } }); Ui::AddSkip(content); Ui::ResizeFitChild(this, content); } void Calls::requestPermissionAndStartTestingMicrophone() { using namespace ::Platform; const auto status = GetPermissionStatus( PermissionType::Microphone); if (status == PermissionStatus::Granted) { startTestingMicrophone(); } else if (status == PermissionStatus::CanRequest) { const auto startTestingChecked = crl::guard(this, [=]( PermissionStatus status) { if (status == PermissionStatus::Granted) { crl::on_main(crl::guard(this, [=] { startTestingMicrophone(); })); } }); RequestPermission( PermissionType::Microphone, startTestingChecked); } else { const auto showSystemSettings = [controller = _controller] { OpenSystemSettingsForPermission( PermissionType::Microphone); controller->hideLayer(); }; _controller->show(Ui::MakeConfirmBox({ .text = tr::lng_no_mic_permission(), .confirmed = showSystemSettings, .confirmText = tr::lng_menu_settings(), })); } } void Calls::startTestingMicrophone() { _levelUpdateTimer.callEach(kMicTestUpdateInterval); _micTester = std::make_unique( Core::App().settings().callAudioBackend(), Core::App().settings().callCaptureDeviceId()); } rpl::producer PlaybackDeviceNameValue(rpl::producer id) { return DeviceNameValue(DeviceType::Playback, std::move(id)); } rpl::producer CaptureDeviceNameValue(rpl::producer id) { return DeviceNameValue(DeviceType::Capture, std::move(id)); } void ChooseAudioDeviceBox( not_null box, rpl::producer title, rpl::producer> devicesValue, rpl::producer currentId, Fn 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(layout, skip)); if (!st) { st = &st::defaultBoxCheckbox; } if (!radioSt) { radioSt = &st::defaultRadio; } struct State { std::vector list; base::flat_map ids; rpl::variable currentId; QString currentName; bool ignoreValueChange = false; }; const auto state = box->lifetime().make_state(); state->currentId = std::move(currentId); const auto group = std::make_shared(); const auto fake = std::make_shared(0); const auto buttons = layout->add(object_ptr(layout)); const auto other = layout->add(object_ptr(layout)); const auto margins = QMargins( st::boxPadding.left() + st::boxOptionListPadding.left(), 0, st::boxPadding.right(), st::boxOptionListSkip); const auto def = buttons->add( object_ptr( 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( 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(); } } } }; std::move( devicesValue ) | rpl::start_with_next([=](std::vector &&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( 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(); } }); } object_ptr ChoosePlaybackDeviceBox( rpl::producer currentId, Fn 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 ChooseCaptureDeviceBox( rpl::producer currentId, Fn 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); } } // namespace Settings