Refactor calls settings panel.

This commit is contained in:
John Preston 2019-01-11 14:07:56 +04:00
parent 8711830f66
commit 65430d92ea
10 changed files with 284 additions and 193 deletions

View File

@ -13,27 +13,48 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "ui/widgets/checkbox.h" #include "ui/widgets/checkbox.h"
#include "styles/style_boxes.h" #include "styles/style_boxes.h"
SingleChoiceBox::SingleChoiceBox(
QWidget*,
LangKey title,
const std::vector<QString> &optionTexts,
int initialSelection,
Fn<void(int)> callback)
: _title(title)
, _optionTexts(optionTexts)
, _initialSelection(initialSelection)
, _callback(callback) {
}
void SingleChoiceBox::prepare() { void SingleChoiceBox::prepare() {
setTitle(langFactory(_title)); setTitle(langFactory(_title));
addButton(langFactory(lng_box_ok), [this] { closeBox(); }); addButton(langFactory(lng_box_ok), [=] { closeBox(); });
auto group = std::make_shared<Ui::RadiobuttonGroup>(_initialSelection); const auto group = std::make_shared<Ui::RadiobuttonGroup>(_initialSelection);
auto y = st::boxOptionListPadding.top() + st::autolockButton.margin.top(); auto y = st::boxOptionListPadding.top()
auto count = int(_optionTexts.size()); + st::autolockButton.margin.top();
_options.reserve(count); _options.reserve(_optionTexts.size());
auto i = 0; auto i = 0;
for (const auto &text : _optionTexts) { for (const auto &text : _optionTexts) {
_options.emplace_back(this, group, i, text, st::autolockButton); _options.emplace_back(this, group, i, text, st::autolockButton);
_options.back()->moveToLeft(st::boxPadding.left() + st::boxOptionListPadding.left(), y); _options.back()->moveToLeft(
st::boxPadding.left() + st::boxOptionListPadding.left(),
y);
y += _options.back()->heightNoMargins() + st::boxOptionListSkip; y += _options.back()->heightNoMargins() + st::boxOptionListSkip;
i++; i++;
} }
group->setChangedCallback([this](int value) { group->setChangedCallback([=](int value) {
const auto weak = make_weak(this);
_callback(value); _callback(value);
closeBox(); if (weak) {
closeBox();
}
}); });
setDimensions(st::autolockWidth, st::boxOptionListPadding.top() + count * _options.back()->heightNoMargins() + (count - 1) * st::boxOptionListSkip + st::boxOptionListPadding.bottom() + st::boxPadding.bottom()); const auto height = y
- st::boxOptionListSkip
+ st::boxOptionListPadding.bottom()
+ st::boxPadding.bottom();
setDimensions(st::autolockWidth, height);
} }

View File

@ -18,8 +18,12 @@ class Radiobutton;
class SingleChoiceBox : public BoxContent { class SingleChoiceBox : public BoxContent {
public: public:
SingleChoiceBox(QWidget*, LangKey title, std::vector<QString> optionTexts, int initialSelection, Fn<void(int)> callback) : _title(title), _optionTexts(optionTexts), _initialSelection(initialSelection), _callback(callback) { SingleChoiceBox(
} QWidget*,
LangKey title,
const std::vector<QString> &optionTexts,
int initialSelection,
Fn<void(int)> callback);
protected: protected:
void prepare() override; void prepare() override;

View File

@ -491,24 +491,23 @@ void OpenSystemSettingsForPermission(PermissionType type) {
bool OpenSystemSettings(SystemSettingsType type) { bool OpenSystemSettings(SystemSettingsType type) {
if (type == SystemSettingsType::Audio) { if (type == SystemSettingsType::Audio) {
bool succeeded = false; auto options = std::vector<QString>();
const auto add = [&](const char *option) {
options.emplace_back(option);
};
if (DesktopEnvironment::IsUnity()) { if (DesktopEnvironment::IsUnity()) {
succeeded |= QProcess::startDetached(qsl("unity-control-center sound")); add("unity-control-center sound");
} else if (DesktopEnvironment::IsKDE()) { } else if (DesktopEnvironment::IsKDE()) {
succeeded |= QProcess::startDetached(qsl("kcmshell5 kcm_pulseaudio")); add("kcmshell5 kcm_pulseaudio");
if (!succeeded) { add("kcmshell4 phonon");
succeeded |= QProcess::startDetached(qsl("kcmshell4 phonon"));
}
} else if (DesktopEnvironment::IsGnome()) { } else if (DesktopEnvironment::IsGnome()) {
succeeded |= QProcess::startDetached(qsl("gnome-control-center sound")); add("gnome-control-center sound");
} }
if (!succeeded) { add("pavucontrol");
succeeded |= QProcess::startDetached(qsl("pavucontrol")); add("alsamixergui");
} return ranges::find_if(options, [](const QString &command) {
if (!succeeded) { return QProcess::startDetached(command);
succeeded |= QProcess::startDetached(qsl("alsamixergui")); }) != end(options);
}
return succeeded;
} }
return true; return true;
} }

View File

@ -334,9 +334,9 @@ void OpenSystemSettingsForPermission(PermissionType type) {
bool OpenSystemSettings(SystemSettingsType type) { bool OpenSystemSettings(SystemSettingsType type) {
switch (type) { switch (type) {
case SystemSettingsType::Audio: case SystemSettingsType::Audio:
[[NSWorkspace sharedWorkspace] openFile:@"/System/Library/PreferencePanes/Sound.prefPane"]; [[NSWorkspace sharedWorkspace] openFile:@"/System/Library/PreferencePanes/Sound.prefPane"];
break; break;
} }
return true; return true;
} }

View File

@ -652,14 +652,24 @@ void RequestPermission(PermissionType type, Fn<void(PermissionStatus)> resultCal
} }
void OpenSystemSettingsForPermission(PermissionType type) { void OpenSystemSettingsForPermission(PermissionType type) {
if (type==PermissionType::Microphone) { if (type == PermissionType::Microphone) {
ShellExecute(NULL, L"open", L"ms-settings:privacy-microphone", NULL, NULL, SW_SHOWDEFAULT); crl::on_main([] {
ShellExecute(
nullptr,
L"open",
L"ms-settings:privacy-microphone",
nullptr,
nullptr,
SW_SHOWDEFAULT);
});
} }
} }
bool OpenSystemSettings(SystemSettingsType type) { bool OpenSystemSettings(SystemSettingsType type) {
if (type == SystemSettingsType::Audio) { if (type == SystemSettingsType::Audio) {
WinExec("control.exe mmsys.cpl", SW_SHOW); crl::on_main([] {
WinExec("control.exe mmsys.cpl", SW_SHOW);
});
} }
return true; return true;
} }

View File

@ -43,13 +43,13 @@ Calls::Calls(QWidget *parent, UserData *self)
setupContent(); setupContent();
} }
Calls::~Calls(){ Calls::~Calls() {
if (_needWriteSettings) { if (_needWriteSettings) {
Local::writeUserSettings(); Local::writeUserSettings();
} }
} }
void Calls::sectionSaveChanges(FnMut<void()> done){ void Calls::sectionSaveChanges(FnMut<void()> done) {
if (_micTester) { if (_micTester) {
_micTester.reset(); _micTester.reset();
} }
@ -57,181 +57,219 @@ void Calls::sectionSaveChanges(FnMut<void()> done){
} }
void Calls::setupContent() { void Calls::setupContent() {
using namespace tgvoip;
const auto content = Ui::CreateChild<Ui::VerticalLayout>(this); const auto content = Ui::CreateChild<Ui::VerticalLayout>(this);
const auto getId = [](const auto &device) {
return QString::fromStdString(device.id);
};
const auto getName = [](const auto &device) {
return QString::fromStdString(device.displayName);
};
QString currentOutputName; const auto currentOutputName = [&] {
if (Global::CallOutputDeviceID() == qsl("default")) { if (Global::CallOutputDeviceID() == qsl("default")) {
currentOutputName = lang(lng_settings_call_device_default); return lang(lng_settings_call_device_default);
} else {
std::vector<tgvoip::AudioOutputDevice> outputDevices = tgvoip::VoIPController::EnumerateAudioOutputs();
currentOutputName=Global::CallOutputDeviceID();
for (auto &dev : outputDevices) {
if (QString::fromUtf8(dev.id.c_str()) == Global::CallOutputDeviceID()) {
currentOutputName = QString::fromUtf8(dev.displayName.c_str());
break;
}
} }
} const auto &list = VoIPController::EnumerateAudioOutputs();
const auto i = ranges::find(
list,
Global::CallOutputDeviceID(),
getId);
return (i != end(list))
? getName(*i)
: Global::CallOutputDeviceID();
}();
QString currentInputName; const auto currentInputName = [&] {
if (Global::CallInputDeviceID() == qsl("default")) { if (Global::CallInputDeviceID() == qsl("default")) {
currentInputName = lang(lng_settings_call_device_default); return lang(lng_settings_call_device_default);
} else {
std::vector<tgvoip::AudioInputDevice> inputDevices = tgvoip::VoIPController::EnumerateAudioInputs();
currentInputName = Global::CallInputDeviceID();
for (auto &dev : inputDevices) {
if (QString::fromUtf8(dev.id.c_str()) == Global::CallInputDeviceID()) {
currentInputName = QString::fromUtf8(dev.displayName.c_str());
break;
}
} }
} const auto &list = VoIPController::EnumerateAudioInputs();
const auto i = ranges::find(
list,
Global::CallInputDeviceID(),
getId);
return (i != end(list))
? getName(*i)
: Global::CallInputDeviceID();
}();
AddSkip(content); AddSkip(content);
AddSubsectionTitle(content, lng_settings_call_section_output); AddSubsectionTitle(content, lng_settings_call_section_output);
const auto outputButton = AddButtonWithLabel( AddButtonWithLabel(
content, content,
lng_settings_call_output_device, lng_settings_call_output_device,
rpl::single(currentOutputName) | rpl::then(_outputNameStream.events()), rpl::single(
st::settingsButton); currentOutputName
outputButton->addClickHandler([this] { ) | rpl::then(
int selectedOption = 0; _outputNameStream.events()
std::vector<tgvoip::AudioOutputDevice> devices = tgvoip::VoIPController::EnumerateAudioOutputs(); ),
std::vector<QString> options; st::settingsButton
options.push_back(lang(lng_settings_call_device_default)); )->addClickHandler([=] {
int i = 1; const auto &devices = VoIPController::EnumerateAudioOutputs();
for (auto &device : devices) { const auto options = ranges::view::concat(
QString displayName = QString::fromUtf8(device.displayName.c_str()); ranges::view::single(lang(lng_settings_call_device_default)),
options.push_back(displayName); devices | ranges::view::transform(getName)
if (QString::fromUtf8(device.id.c_str()) == Global::CallOutputDeviceID()) { ) | ranges::to_vector;
selectedOption = i; const auto i = ranges::find(
} devices,
i++; Global::CallOutputDeviceID(),
} getId);
const auto save = crl::guard(this, [=](int selectedOption) { const auto currentOption = (i != end(devices))
QString name = options[selectedOption]; ? int(i - begin(devices) + 1)
_outputNameStream.fire(std::move(name)); : 0;
std::string selectedDeviceID; const auto save = crl::guard(this, [=](int option) {
if (selectedOption == 0) { _outputNameStream.fire_copy(options[option]);
selectedDeviceID = "default"; const auto deviceId = option
} else { ? devices[option - 1].id
selectedDeviceID = devices[selectedOption-1].id; : "default";
} Global::SetCallOutputDeviceID(QString::fromStdString(deviceId));
Global::SetCallOutputDeviceID(QString::fromStdString(selectedDeviceID));
Local::writeUserSettings(); Local::writeUserSettings();
if (const auto call = ::Calls::Current().currentCall()) {
::Calls::Call *currentCall = ::Calls::Current().currentCall(); call->setCurrentAudioDevice(false, deviceId);
if (currentCall) {
currentCall->setCurrentAudioDevice(false, selectedDeviceID);
} }
}); });
Ui::show(Box<SingleChoiceBox>(lng_settings_call_output_device, options, selectedOption, save)); Ui::show(Box<SingleChoiceBox>(
lng_settings_call_output_device,
options,
currentOption,
save));
}); });
const auto outputLabel = content->add(object_ptr<Ui::LabelSimple>(content, st::settingsAudioVolumeLabel), st::settingsAudioVolumeLabelPadding); const auto outputLabel = content->add(
const auto outputSlider = content->add(object_ptr<Ui::MediaSlider>(content, st::settingsAudioVolumeSlider), st::settingsAudioVolumeSliderPadding); object_ptr<Ui::LabelSimple>(
auto updateOutputLabel = [outputLabel](int value){ content,
QString percent = QString::number(value); st::settingsAudioVolumeLabel),
outputLabel->setText(lng_settings_call_output_volume(lt_percent, percent)); st::settingsAudioVolumeLabelPadding);
const auto outputSlider = content->add(
object_ptr<Ui::MediaSlider>(
content,
st::settingsAudioVolumeSlider),
st::settingsAudioVolumeSliderPadding);
const auto updateOutputLabel = [=](int value) {
const auto percent = QString::number(value);
outputLabel->setText(
lng_settings_call_output_volume(lt_percent, percent));
};
const auto updateOutputVolume = [=](int value) {
_needWriteSettings = true;
updateOutputLabel(value);
Global::SetCallOutputVolume(value);
if (const auto call = ::Calls::Current().currentCall()) {
call->setAudioVolume(false, value / 100.0f);
}
}; };
outputSlider->resize(st::settingsAudioVolumeSlider.seekSize); outputSlider->resize(st::settingsAudioVolumeSlider.seekSize);
outputSlider->setPseudoDiscrete( outputSlider->setPseudoDiscrete(
201, 201,
[](int val){ [](int val) { return val; },
return val;
},
Global::CallOutputVolume(), Global::CallOutputVolume(),
[updateOutputLabel, this](int value) { updateOutputVolume);
_needWriteSettings = true;
updateOutputLabel(value);
Global::SetCallOutputVolume(value);
::Calls::Call* currentCall = ::Calls::Current().currentCall();
if (currentCall) {
currentCall->setAudioVolume(false, value/100.0f);
}
});
updateOutputLabel(Global::CallOutputVolume()); updateOutputLabel(Global::CallOutputVolume());
AddSkip(content); AddSkip(content);
AddDivider(content); AddDivider(content);
AddSkip(content); AddSkip(content);
AddSubsectionTitle(content, lng_settings_call_section_input); AddSubsectionTitle(content, lng_settings_call_section_input);
const auto inputButton = AddButtonWithLabel( AddButtonWithLabel(
content, content,
lng_settings_call_input_device, lng_settings_call_input_device,
rpl::single(currentInputName) | rpl::then(_inputNameStream.events()), rpl::single(
st::settingsButton); currentInputName
inputButton->addClickHandler([this] { ) | rpl::then(
int selectedOption = 0; _inputNameStream.events()
std::vector<tgvoip::AudioInputDevice> devices = tgvoip::VoIPController::EnumerateAudioInputs(); ),
std::vector<QString> options; st::settingsButton
options.push_back(lang(lng_settings_call_device_default)); )->addClickHandler([=] {
int i = 1; const auto &devices = VoIPController::EnumerateAudioInputs();
for (auto &device : devices) { const auto options = ranges::view::concat(
QString displayName = QString::fromUtf8(device.displayName.c_str()); ranges::view::single(lang(lng_settings_call_device_default)),
options.push_back(displayName); devices | ranges::view::transform(getName)
if(QString::fromUtf8(device.id.c_str()) == Global::CallInputDeviceID()) ) | ranges::to_vector;
selectedOption = i; const auto i = ranges::find(
i++; devices,
} Global::CallInputDeviceID(),
const auto save = crl::guard(this, [=](int selectedOption) { getId);
QString name=options[selectedOption]; const auto currentOption = (i != end(devices))
_inputNameStream.fire(std::move(name)); ? int(i - begin(devices) + 1)
std::string selectedDeviceID; : 0;
if (selectedOption == 0) { const auto save = crl::guard(this, [=](int option) {
selectedDeviceID = "default"; _inputNameStream.fire_copy(options[option]);
} else { const auto deviceId = option
selectedDeviceID = devices[selectedOption - 1].id; ? devices[option - 1].id
} : "default";
Global::SetCallInputDeviceID(QString::fromUtf8(selectedDeviceID.c_str())); Global::SetCallInputDeviceID(QString::fromStdString(deviceId));
Local::writeUserSettings(); Local::writeUserSettings();
if (_micTester) { if (_micTester) {
stopTestingMicrophone(); stopTestingMicrophone();
} }
::Calls::Call *currentCall = ::Calls::Current().currentCall(); if (const auto call = ::Calls::Current().currentCall()) {
if(currentCall){ call->setCurrentAudioDevice(true, deviceId);
currentCall->setCurrentAudioDevice(true, selectedDeviceID);
} }
}); });
Ui::show(Box<SingleChoiceBox>(lng_settings_call_input_device, options, selectedOption, save)); Ui::show(Box<SingleChoiceBox>(
lng_settings_call_input_device,
options,
currentOption,
save));
}); });
const auto inputLabel = content->add(object_ptr<Ui::LabelSimple>(content, st::settingsAudioVolumeLabel), st::settingsAudioVolumeLabelPadding); const auto inputLabel = content->add(
const auto inputSlider = content->add(object_ptr<Ui::MediaSlider>(content, st::settingsAudioVolumeSlider), st::settingsAudioVolumeSliderPadding); object_ptr<Ui::LabelSimple>(
auto updateInputLabel = [inputLabel](int value){ content,
QString percent = QString::number(value); st::settingsAudioVolumeLabel),
inputLabel->setText(lng_settings_call_input_volume(lt_percent, percent)); st::settingsAudioVolumeLabelPadding);
const auto inputSlider = content->add(
object_ptr<Ui::MediaSlider>(
content,
st::settingsAudioVolumeSlider),
st::settingsAudioVolumeSliderPadding);
const auto updateInputLabel = [=](int value) {
const auto percent = QString::number(value);
inputLabel->setText(
lng_settings_call_input_volume(lt_percent, percent));
};
const auto updateInputVolume = [=](int value) {
_needWriteSettings = true;
updateInputLabel(value);
Global::SetCallInputVolume(value);
::Calls::Call *currentCall = ::Calls::Current().currentCall();
if (currentCall) {
currentCall->setAudioVolume(true, value / 100.0f);
}
}; };
inputSlider->resize(st::settingsAudioVolumeSlider.seekSize); inputSlider->resize(st::settingsAudioVolumeSlider.seekSize);
inputSlider->setPseudoDiscrete(101, inputSlider->setPseudoDiscrete(101,
[](int val){ [](int val) { return val; },
return val;
},
Global::CallInputVolume(), Global::CallInputVolume(),
[updateInputLabel, this](int value) { updateInputVolume);
_needWriteSettings = true;
updateInputLabel(value);
Global::SetCallInputVolume(value);
::Calls::Call *currentCall = ::Calls::Current().currentCall();
if (currentCall) {
currentCall->setAudioVolume(true, value / 100.0f);
}
});
updateInputLabel(Global::CallInputVolume()); updateInputLabel(Global::CallInputVolume());
_micTestButton=AddButton(content, rpl::single(lang(lng_settings_call_test_mic)) | rpl::then(_micTestTextStream.events()), st::settingsButton); AddButton(
content,
_micTestLevel=content->add(object_ptr<Ui::LevelMeter>(content, st::defaultLevelMeter), st::settingsLevelMeterPadding); rpl::single(
_micTestLevel->resize(QSize(0, st::defaultLevelMeter.height)); lang(lng_settings_call_test_mic)
) | rpl::then(
_micTestButton->addClickHandler([this]{ _micTestTextStream.events()
),
st::settingsButton
)->addClickHandler([=] {
if (!_micTester) { if (!_micTester) {
requestPermissionAndStartTestingMicrophone(); requestPermissionAndStartTestingMicrophone();
} else { } else {
stopTestingMicrophone(); stopTestingMicrophone();
} }
}); });
_levelUpdateTimer.setCallback([this](){
_micTestLevel = content->add(
object_ptr<Ui::LevelMeter>(
content,
st::defaultLevelMeter),
st::settingsLevelMeterPadding);
_micTestLevel->resize(QSize(0, st::defaultLevelMeter.height));
_levelUpdateTimer.setCallback([=] {
_micTestLevel->setValue(_micTester->GetAndResetLevel()); _micTestLevel->setValue(_micTester->GetAndResetLevel());
}); });
@ -252,16 +290,20 @@ void Calls::setupContent() {
}) | rpl::start_with_next([](bool enabled) { }) | rpl::start_with_next([](bool enabled) {
Global::SetCallAudioDuckingEnabled(enabled); Global::SetCallAudioDuckingEnabled(enabled);
Local::writeUserSettings(); Local::writeUserSettings();
::Calls::Call *currentCall = ::Calls::Current().currentCall(); if (const auto call = ::Calls::Current().currentCall()) {
if (currentCall) { call->setAudioDuckingEnabled(enabled);
currentCall->setAudioDuckingEnabled(enabled);
} }
}, content->lifetime()); }, content->lifetime());
#endif // Q_OS_MAC #endif // Q_OS_MAC
const auto systemSettingsButton=AddButton(content, lng_settings_call_open_system_prefs, st::settingsButton); AddButton(
systemSettingsButton->addClickHandler([]{ content,
if (!Platform::OpenSystemSettings(Platform::SystemSettingsType::Audio)) { lng_settings_call_open_system_prefs,
st::settingsButton
)->addClickHandler([] {
const auto opened = Platform::OpenSystemSettings(
Platform::SystemSettingsType::Audio);
if (!opened) {
Ui::show(Box<InformBox>(lang(lng_linux_no_audio_prefs))); Ui::show(Box<InformBox>(lang(lng_linux_no_audio_prefs)));
} }
}); });
@ -270,37 +312,48 @@ void Calls::setupContent() {
Ui::ResizeFitChild(this, content); Ui::ResizeFitChild(this, content);
} }
void Calls::requestPermissionAndStartTestingMicrophone(){ void Calls::requestPermissionAndStartTestingMicrophone() {
Platform::PermissionStatus status = Platform::GetPermissionStatus(Platform::PermissionType::Microphone); const auto status = Platform::GetPermissionStatus(
Platform::PermissionType::Microphone);
if (status == Platform::PermissionStatus::Granted) { if (status == Platform::PermissionStatus::Granted) {
startTestingMicrophone(); startTestingMicrophone();
} else if (status == Platform::PermissionStatus::CanRequest) { } else if (status == Platform::PermissionStatus::CanRequest) {
Platform::RequestPermission(Platform::PermissionType::Microphone, crl::guard(this, [this](Platform::PermissionStatus status) { const auto startTestingChecked = crl::guard(this, [=](
Platform::PermissionStatus status) {
if (status == Platform::PermissionStatus::Granted) { if (status == Platform::PermissionStatus::Granted) {
crl::on_main(crl::guard(this, [this]{ crl::on_main(crl::guard(this, [=] {
startTestingMicrophone(); startTestingMicrophone();
})); }));
} }
})); });
Platform::RequestPermission(
Platform::PermissionType::Microphone,
startTestingChecked);
} else { } else {
Ui::show(Box<ConfirmBox>(lang(lng_no_mic_permission), lang(lng_menu_settings), crl::guard(this, [] { const auto showSystemSettings = [] {
Platform::OpenSystemSettingsForPermission(Platform::PermissionType::Microphone); Platform::OpenSystemSettingsForPermission(
Platform::PermissionType::Microphone);
Ui::hideLayer(); Ui::hideLayer();
}))); };
Ui::show(Box<ConfirmBox>(
lang(lng_no_mic_permission),
lang(lng_menu_settings),
showSystemSettings));
} }
} }
void Calls::startTestingMicrophone(){ void Calls::startTestingMicrophone() {
_micTestTextStream.fire(lang(lng_settings_call_stop_mic_test)); _micTestTextStream.fire(lang(lng_settings_call_stop_mic_test));
_levelUpdateTimer.callEach(50); _levelUpdateTimer.callEach(50);
_micTester = std::make_unique<tgvoip::AudioInputTester>(Global::CallInputDeviceID().toStdString()); _micTester = std::make_unique<tgvoip::AudioInputTester>(
Global::CallInputDeviceID().toStdString());
if (_micTester->Failed()) { if (_micTester->Failed()) {
Ui::show(Box<InformBox>(lang(lng_call_error_audio_io))); Ui::show(Box<InformBox>(lang(lng_call_error_audio_io)));
stopTestingMicrophone(); stopTestingMicrophone();
} }
} }
void Calls::stopTestingMicrophone(){ void Calls::stopTestingMicrophone() {
_micTestTextStream.fire(lang(lng_settings_call_test_mic)); _micTestTextStream.fire(lang(lng_settings_call_test_mic));
_levelUpdateTimer.cancel(); _levelUpdateTimer.cancel();
_micTester.reset(); _micTester.reset();

View File

@ -41,7 +41,6 @@ private:
rpl::event_stream<QString> _micTestTextStream; rpl::event_stream<QString> _micTestTextStream;
bool _needWriteSettings = false; bool _needWriteSettings = false;
std::unique_ptr<tgvoip::AudioInputTester> _micTester; std::unique_ptr<tgvoip::AudioInputTester> _micTester;
Button *_micTestButton = nullptr;
Ui::LevelMeter *_micTestLevel = nullptr; Ui::LevelMeter *_micTestLevel = nullptr;
base::Timer _levelUpdateTimer; base::Timer _levelUpdateTimer;
}; };

View File

@ -603,11 +603,11 @@ void RadiobuttonGroup::setValue(int value) {
} }
_hasValue = true; _hasValue = true;
_value = value; _value = value;
for (auto button : _buttons) { for (const auto button : _buttons) {
button->handleNewGroupValue(_value); button->handleNewGroupValue(_value);
} }
if (_changedCallback) { if (const auto callback = _changedCallback) {
_changedCallback(_value); callback(_value);
} }
} }

View File

@ -9,33 +9,36 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
namespace Ui { namespace Ui {
LevelMeter::LevelMeter(QWidget *parent, const style::LevelMeter &st) : RpWidget(parent), _st(st){ LevelMeter::LevelMeter(QWidget *parent, const style::LevelMeter &st)
: RpWidget(parent)
, _st(st) {
} }
void LevelMeter::setValue(float value){ void LevelMeter::setValue(float value) {
_value = value; _value = value;
repaint(); repaint();
} }
void LevelMeter::paintEvent(QPaintEvent* event){ void LevelMeter::paintEvent(QPaintEvent* event) {
Painter p(this); Painter p(this);
PainterHighQualityEnabler hq(p); PainterHighQualityEnabler hq(p);
p.setPen(Qt::NoPen); p.setPen(Qt::NoPen);
auto activeFg = _st.activeFg; const auto activeFg = _st.activeFg;
auto inactiveFg = _st.inactiveFg; const auto inactiveFg = _st.inactiveFg;
auto radius = _st.lineWidth / 2; const auto radius = _st.lineWidth / 2;
QRect rect(0, 0, _st.lineWidth, height()); const auto rect = QRect(0, 0, _st.lineWidth, height());
p.setBrush(activeFg); p.setBrush(activeFg);
for (auto i = 0; i < _st.lineCount; ++i) { for (auto i = 0; i < _st.lineCount; ++i) {
float valueAtLine = (float)(i + 1) / _st.lineCount; const auto valueAtLine = (float)(i + 1) / _st.lineCount;
if (valueAtLine > _value) { if (valueAtLine > _value) {
p.setBrush(inactiveFg); p.setBrush(inactiveFg);
} }
rect.moveLeft((_st.lineWidth + _st.lineSpacing) * i); p.drawRoundedRect(
p.drawRoundedRect(rect, radius, radius); rect.translated((_st.lineWidth + _st.lineSpacing) * i, 0),
radius,
radius);
} }
} }

View File

@ -15,6 +15,7 @@ namespace Ui {
class LevelMeter : public RpWidget { class LevelMeter : public RpWidget {
public: public:
LevelMeter(QWidget *parent, const style::LevelMeter& st); LevelMeter(QWidget *parent, const style::LevelMeter& st);
void setValue(float value); void setValue(float value);
protected: protected:
@ -22,7 +23,8 @@ protected:
private: private:
const style::LevelMeter &_st; const style::LevelMeter &_st;
float _value=0.0f; float _value = 0.0f;
}; };
} // namespace Ui } // namespace Ui