mirror of https://github.com/procxx/kepka.git
Refactor calls settings panel.
This commit is contained in:
parent
8711830f66
commit
65430d92ea
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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;
|
||||||
};
|
};
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in New Issue