826 lines
23 KiB
C++
826 lines
23 KiB
C++
/*
|
|
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_advanced.h"
|
|
|
|
#include "settings/settings_common.h"
|
|
#include "settings/settings_chat.h"
|
|
#include "settings/settings_experimental.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/buttons.h"
|
|
#include "ui/gl/gl_detection.h"
|
|
#include "ui/text/text_utilities.h" // Ui::Text::ToUpper
|
|
#include "ui/text/format_values.h"
|
|
#include "ui/boxes/single_choice_box.h"
|
|
#include "boxes/connection_box.h"
|
|
#include "boxes/about_box.h"
|
|
#include "ui/boxes/confirm_box.h"
|
|
#include "platform/platform_specific.h"
|
|
#include "ui/platform/ui_platform_window.h"
|
|
#include "base/platform/base_platform_info.h"
|
|
#include "window/window_controller.h"
|
|
#include "window/window_session_controller.h"
|
|
#include "lang/lang_keys.h"
|
|
#include "core/update_checker.h"
|
|
#include "core/application.h"
|
|
#include "storage/localstorage.h"
|
|
#include "storage/storage_domain.h"
|
|
#include "data/data_session.h"
|
|
#include "main/main_account.h"
|
|
#include "main/main_domain.h"
|
|
#include "main/main_session.h"
|
|
#include "mtproto/facade.h"
|
|
#include "styles/style_settings.h"
|
|
|
|
#ifdef Q_OS_MAC
|
|
#include "base/platform/mac/base_confirm_quit.h"
|
|
#endif // Q_OS_MAC
|
|
|
|
#ifndef TDESKTOP_DISABLE_SPELLCHECK
|
|
#include "boxes/dictionaries_manager.h"
|
|
#include "chat_helpers/spellchecker_common.h"
|
|
#include "spellcheck/platform/platform_spellcheck.h"
|
|
#endif // !TDESKTOP_DISABLE_SPELLCHECK
|
|
|
|
namespace Settings {
|
|
|
|
void SetupConnectionType(
|
|
not_null<Window::Controller*> controller,
|
|
not_null<Main::Account*> account,
|
|
not_null<Ui::VerticalLayout*> container) {
|
|
const auto connectionType = [=] {
|
|
const auto transport = account->mtp().dctransport();
|
|
if (!Core::App().settings().proxy().isEnabled()) {
|
|
return transport.isEmpty()
|
|
? tr::lng_connection_auto_connecting(tr::now)
|
|
: tr::lng_connection_auto(tr::now, lt_transport, transport);
|
|
} else {
|
|
return transport.isEmpty()
|
|
? tr::lng_connection_proxy_connecting(tr::now)
|
|
: tr::lng_connection_proxy(tr::now, lt_transport, transport);
|
|
}
|
|
};
|
|
const auto button = AddButtonWithLabel(
|
|
container,
|
|
tr::lng_settings_connection_type(),
|
|
rpl::merge(
|
|
Core::App().settings().proxy().connectionTypeChanges(),
|
|
// Handle language switch.
|
|
tr::lng_connection_auto_connecting() | rpl::to_empty
|
|
) | rpl::map(connectionType),
|
|
st::settingsButton,
|
|
{ &st::settingsIconArrows, kIconGreen });
|
|
button->addClickHandler([=] {
|
|
controller->show(ProxiesBoxController::CreateOwningBox(account));
|
|
});
|
|
}
|
|
|
|
bool HasUpdate() {
|
|
return !Core::UpdaterDisabled();
|
|
}
|
|
|
|
void SetupUpdate(
|
|
not_null<Ui::VerticalLayout*> container,
|
|
Fn<void(Type)> showOther) {
|
|
if (!HasUpdate()) {
|
|
return;
|
|
}
|
|
|
|
const auto texts = Ui::CreateChild<rpl::event_stream<QString>>(
|
|
container.get());
|
|
const auto downloading = Ui::CreateChild<rpl::event_stream<bool>>(
|
|
container.get());
|
|
const auto version = tr::lng_settings_current_version(
|
|
tr::now,
|
|
lt_version,
|
|
currentVersionText());
|
|
const auto toggle = AddButton(
|
|
container,
|
|
tr::lng_settings_update_automatically(),
|
|
st::settingsUpdateToggle);
|
|
const auto label = Ui::CreateChild<Ui::FlatLabel>(
|
|
toggle.get(),
|
|
texts->events(),
|
|
st::settingsUpdateState);
|
|
|
|
const auto options = container->add(
|
|
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
|
|
container,
|
|
object_ptr<Ui::VerticalLayout>(container)));
|
|
const auto inner = options->entity();
|
|
const auto install = cAlphaVersion() ? nullptr : AddButton(
|
|
inner,
|
|
tr::lng_settings_install_beta(),
|
|
st::settingsButtonNoIcon).get();
|
|
|
|
if (showOther) {
|
|
const auto experimental = inner->add(
|
|
object_ptr<Ui::SlideWrap<Button>>(
|
|
inner,
|
|
CreateButton(
|
|
inner,
|
|
tr::lng_settings_experimental(),
|
|
st::settingsButtonNoIcon)));
|
|
if (!install) {
|
|
experimental->toggle(true, anim::type::instant);
|
|
} else {
|
|
experimental->toggleOn(install->toggledValue());
|
|
}
|
|
experimental->entity()->setClickedCallback([=] {
|
|
showOther(Experimental::Id());
|
|
});
|
|
}
|
|
|
|
const auto check = AddButton(
|
|
inner,
|
|
tr::lng_settings_check_now(),
|
|
st::settingsButtonNoIcon);
|
|
const auto update = Ui::CreateChild<Button>(
|
|
check.get(),
|
|
tr::lng_update_telegram() | Ui::Text::ToUpper(),
|
|
st::settingsUpdate);
|
|
update->hide();
|
|
check->widthValue() | rpl::start_with_next([=](int width) {
|
|
update->resizeToWidth(width);
|
|
update->moveToLeft(0, 0);
|
|
}, update->lifetime());
|
|
|
|
rpl::combine(
|
|
toggle->widthValue(),
|
|
label->widthValue()
|
|
) | rpl::start_with_next([=] {
|
|
label->moveToLeft(
|
|
st::settingsUpdateStatePosition.x(),
|
|
st::settingsUpdateStatePosition.y());
|
|
}, label->lifetime());
|
|
label->setAttribute(Qt::WA_TransparentForMouseEvents);
|
|
|
|
const auto showDownloadProgress = [=](int64 ready, int64 total) {
|
|
texts->fire(tr::lng_settings_downloading_update(
|
|
tr::now,
|
|
lt_progress,
|
|
Ui::FormatDownloadText(ready, total)));
|
|
downloading->fire(true);
|
|
};
|
|
const auto setDefaultStatus = [=](const Core::UpdateChecker &checker) {
|
|
using State = Core::UpdateChecker::State;
|
|
const auto state = checker.state();
|
|
switch (state) {
|
|
case State::Download:
|
|
showDownloadProgress(checker.already(), checker.size());
|
|
break;
|
|
case State::Ready:
|
|
texts->fire(tr::lng_settings_update_ready(tr::now));
|
|
update->show();
|
|
break;
|
|
default:
|
|
texts->fire_copy(version);
|
|
break;
|
|
}
|
|
};
|
|
|
|
toggle->toggleOn(rpl::single(cAutoUpdate()));
|
|
toggle->toggledValue(
|
|
) | rpl::filter([](bool toggled) {
|
|
return (toggled != cAutoUpdate());
|
|
}) | rpl::start_with_next([=](bool toggled) {
|
|
cSetAutoUpdate(toggled);
|
|
|
|
Local::writeSettings();
|
|
Core::UpdateChecker checker;
|
|
if (cAutoUpdate()) {
|
|
checker.start();
|
|
} else {
|
|
checker.stop();
|
|
setDefaultStatus(checker);
|
|
}
|
|
}, toggle->lifetime());
|
|
|
|
if (install) {
|
|
install->toggleOn(rpl::single(cInstallBetaVersion()));
|
|
install->toggledValue(
|
|
) | rpl::filter([](bool toggled) {
|
|
return (toggled != cInstallBetaVersion());
|
|
}) | rpl::start_with_next([=](bool toggled) {
|
|
cSetInstallBetaVersion(toggled);
|
|
Core::App().writeInstallBetaVersionsSetting();
|
|
|
|
Core::UpdateChecker checker;
|
|
checker.stop();
|
|
if (toggled) {
|
|
cSetLastUpdateCheck(0);
|
|
}
|
|
checker.start();
|
|
}, toggle->lifetime());
|
|
}
|
|
|
|
Core::UpdateChecker checker;
|
|
options->toggleOn(rpl::combine(
|
|
toggle->toggledValue(),
|
|
downloading->events_starting_with(
|
|
checker.state() == Core::UpdateChecker::State::Download)
|
|
) | rpl::map([](bool check, bool downloading) {
|
|
return check && !downloading;
|
|
}));
|
|
|
|
checker.checking() | rpl::start_with_next([=] {
|
|
options->setAttribute(Qt::WA_TransparentForMouseEvents);
|
|
texts->fire(tr::lng_settings_update_checking(tr::now));
|
|
downloading->fire(false);
|
|
}, options->lifetime());
|
|
checker.isLatest() | rpl::start_with_next([=] {
|
|
options->setAttribute(Qt::WA_TransparentForMouseEvents, false);
|
|
texts->fire(tr::lng_settings_latest_installed(tr::now));
|
|
downloading->fire(false);
|
|
}, options->lifetime());
|
|
checker.progress(
|
|
) | rpl::start_with_next([=](Core::UpdateChecker::Progress progress) {
|
|
showDownloadProgress(progress.already, progress.size);
|
|
}, options->lifetime());
|
|
checker.failed() | rpl::start_with_next([=] {
|
|
options->setAttribute(Qt::WA_TransparentForMouseEvents, false);
|
|
texts->fire(tr::lng_settings_update_fail(tr::now));
|
|
downloading->fire(false);
|
|
}, options->lifetime());
|
|
checker.ready() | rpl::start_with_next([=] {
|
|
options->setAttribute(Qt::WA_TransparentForMouseEvents, false);
|
|
texts->fire(tr::lng_settings_update_ready(tr::now));
|
|
update->show();
|
|
downloading->fire(false);
|
|
}, options->lifetime());
|
|
|
|
setDefaultStatus(checker);
|
|
|
|
check->addClickHandler([] {
|
|
Core::UpdateChecker checker;
|
|
|
|
cSetLastUpdateCheck(0);
|
|
checker.start();
|
|
});
|
|
update->addClickHandler([] {
|
|
if (!Core::UpdaterDisabled()) {
|
|
Core::checkReadyUpdate();
|
|
}
|
|
Core::Restart();
|
|
});
|
|
}
|
|
|
|
bool HasSystemSpellchecker() {
|
|
#ifdef TDESKTOP_DISABLE_SPELLCHECK
|
|
return false;
|
|
#endif // TDESKTOP_DISABLE_SPELLCHECK
|
|
return true;
|
|
}
|
|
|
|
void SetupSpellchecker(
|
|
not_null<Window::SessionController*> controller,
|
|
not_null<Ui::VerticalLayout*> container) {
|
|
#ifndef TDESKTOP_DISABLE_SPELLCHECK
|
|
const auto session = &controller->session();
|
|
const auto settings = &Core::App().settings();
|
|
const auto isSystem = Platform::Spellchecker::IsSystemSpellchecker();
|
|
const auto button = AddButton(
|
|
container,
|
|
isSystem
|
|
? tr::lng_settings_system_spellchecker()
|
|
: tr::lng_settings_custom_spellchecker(),
|
|
st::settingsButtonNoIcon
|
|
)->toggleOn(
|
|
rpl::single(settings->spellcheckerEnabled())
|
|
);
|
|
|
|
button->toggledValue(
|
|
) | rpl::filter([=](bool enabled) {
|
|
return (enabled != settings->spellcheckerEnabled());
|
|
}) | rpl::start_with_next([=](bool enabled) {
|
|
settings->setSpellcheckerEnabled(enabled);
|
|
Core::App().saveSettingsDelayed();
|
|
}, container->lifetime());
|
|
|
|
if (isSystem) {
|
|
return;
|
|
}
|
|
|
|
const auto sliding = container->add(
|
|
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
|
|
container,
|
|
object_ptr<Ui::VerticalLayout>(container)));
|
|
|
|
AddButton(
|
|
sliding->entity(),
|
|
tr::lng_settings_auto_download_dictionaries(),
|
|
st::settingsButtonNoIcon
|
|
)->toggleOn(
|
|
rpl::single(settings->autoDownloadDictionaries())
|
|
)->toggledValue(
|
|
) | rpl::filter([=](bool enabled) {
|
|
return (enabled != settings->autoDownloadDictionaries());
|
|
}) | rpl::start_with_next([=](bool enabled) {
|
|
settings->setAutoDownloadDictionaries(enabled);
|
|
Core::App().saveSettingsDelayed();
|
|
}, sliding->entity()->lifetime());
|
|
|
|
AddButtonWithLabel(
|
|
sliding->entity(),
|
|
tr::lng_settings_manage_dictionaries(),
|
|
Spellchecker::ButtonManageDictsState(session),
|
|
st::settingsButtonNoIcon
|
|
)->addClickHandler([=] {
|
|
controller->show(
|
|
Box<Ui::ManageDictionariesBox>(&controller->session()));
|
|
});
|
|
|
|
button->toggledValue(
|
|
) | rpl::start_with_next([=](bool enabled) {
|
|
sliding->toggle(enabled, anim::type::normal);
|
|
}, container->lifetime());
|
|
#endif // !TDESKTOP_DISABLE_SPELLCHECK
|
|
}
|
|
|
|
void SetupSystemIntegrationContent(
|
|
Window::SessionController *controller,
|
|
not_null<Ui::VerticalLayout*> container) {
|
|
using WorkMode = Core::Settings::WorkMode;
|
|
|
|
const auto checkbox = [&](rpl::producer<QString> &&label, bool checked) {
|
|
return object_ptr<Ui::Checkbox>(
|
|
container,
|
|
std::move(label),
|
|
checked,
|
|
st::settingsCheckbox);
|
|
};
|
|
const auto addCheckbox = [&](
|
|
rpl::producer<QString> &&label,
|
|
bool checked) {
|
|
return container->add(
|
|
checkbox(std::move(label), checked),
|
|
st::settingsCheckboxPadding);
|
|
};
|
|
const auto addSlidingCheckbox = [&](
|
|
rpl::producer<QString> &&label,
|
|
bool checked) {
|
|
return container->add(
|
|
object_ptr<Ui::SlideWrap<Ui::Checkbox>>(
|
|
container,
|
|
checkbox(std::move(label), checked),
|
|
st::settingsCheckboxPadding));
|
|
};
|
|
|
|
if (Platform::TrayIconSupported()) {
|
|
const auto trayEnabled = [] {
|
|
const auto workMode = Core::App().settings().workMode();
|
|
return (workMode == WorkMode::TrayOnly)
|
|
|| (workMode == WorkMode::WindowAndTray);
|
|
};
|
|
const auto tray = addCheckbox(
|
|
tr::lng_settings_workmode_tray(),
|
|
trayEnabled());
|
|
|
|
const auto taskbarEnabled = [] {
|
|
const auto workMode = Core::App().settings().workMode();
|
|
return (workMode == WorkMode::WindowOnly)
|
|
|| (workMode == WorkMode::WindowAndTray);
|
|
};
|
|
const auto taskbar = Platform::SkipTaskbarSupported()
|
|
? addCheckbox(
|
|
tr::lng_settings_workmode_window(),
|
|
taskbarEnabled())
|
|
: nullptr;
|
|
|
|
const auto updateWorkmode = [=] {
|
|
const auto newMode = tray->checked()
|
|
? ((!taskbar || taskbar->checked())
|
|
? WorkMode::WindowAndTray
|
|
: WorkMode::TrayOnly)
|
|
: WorkMode::WindowOnly;
|
|
if ((newMode == WorkMode::WindowAndTray
|
|
|| newMode == WorkMode::TrayOnly)
|
|
&& Core::App().settings().workMode() != newMode) {
|
|
cSetSeenTrayTooltip(false);
|
|
}
|
|
Core::App().settings().setWorkMode(newMode);
|
|
Core::App().saveSettingsDelayed();
|
|
};
|
|
|
|
tray->checkedChanges(
|
|
) | rpl::filter([=](bool checked) {
|
|
return (checked != trayEnabled());
|
|
}) | rpl::start_with_next([=](bool checked) {
|
|
if (!checked && taskbar && !taskbar->checked()) {
|
|
taskbar->setChecked(true);
|
|
} else {
|
|
updateWorkmode();
|
|
}
|
|
}, tray->lifetime());
|
|
|
|
if (taskbar) {
|
|
taskbar->checkedChanges(
|
|
) | rpl::filter([=](bool checked) {
|
|
return (checked != taskbarEnabled());
|
|
}) | rpl::start_with_next([=](bool checked) {
|
|
if (!checked && !tray->checked()) {
|
|
tray->setChecked(true);
|
|
} else {
|
|
updateWorkmode();
|
|
}
|
|
}, taskbar->lifetime());
|
|
}
|
|
}
|
|
|
|
#ifdef Q_OS_MAC
|
|
const auto warnBeforeQuit = addCheckbox(
|
|
tr::lng_settings_mac_warn_before_quit(
|
|
lt_text,
|
|
rpl::single(Platform::ConfirmQuit::QuitKeysString())),
|
|
Core::App().settings().macWarnBeforeQuit());
|
|
warnBeforeQuit->checkedChanges(
|
|
) | rpl::filter([=](bool checked) {
|
|
return (checked != Core::App().settings().macWarnBeforeQuit());
|
|
}) | rpl::start_with_next([=](bool checked) {
|
|
Core::App().settings().setMacWarnBeforeQuit(checked);
|
|
Core::App().saveSettingsDelayed();
|
|
}, warnBeforeQuit->lifetime());
|
|
#else // Q_OS_MAC
|
|
const auto closeToTaskbar = addSlidingCheckbox(
|
|
tr::lng_settings_close_to_taskbar(),
|
|
Core::App().settings().closeToTaskbar());
|
|
|
|
const auto closeToTaskbarShown = std::make_shared<rpl::variable<bool>>(false);
|
|
Core::App().settings().workModeValue(
|
|
) | rpl::start_with_next([=](WorkMode workMode) {
|
|
*closeToTaskbarShown = (workMode == WorkMode::WindowOnly)
|
|
|| !Platform::TrayIconSupported();
|
|
}, closeToTaskbar->lifetime());
|
|
|
|
closeToTaskbar->toggleOn(closeToTaskbarShown->value());
|
|
closeToTaskbar->entity()->checkedChanges(
|
|
) | rpl::filter([=](bool checked) {
|
|
return (checked != Core::App().settings().closeToTaskbar());
|
|
}) | rpl::start_with_next([=](bool checked) {
|
|
Core::App().settings().setCloseToTaskbar(checked);
|
|
Local::writeSettings();
|
|
}, closeToTaskbar->lifetime());
|
|
#endif // Q_OS_MAC
|
|
|
|
if (Ui::Platform::NativeWindowFrameSupported()) {
|
|
const auto nativeFrame = addCheckbox(
|
|
tr::lng_settings_native_frame(),
|
|
Core::App().settings().nativeWindowFrame());
|
|
|
|
nativeFrame->checkedChanges(
|
|
) | rpl::filter([](bool checked) {
|
|
return (checked != Core::App().settings().nativeWindowFrame());
|
|
}) | rpl::start_with_next([=](bool checked) {
|
|
Core::App().settings().setNativeWindowFrame(checked);
|
|
Core::App().saveSettingsDelayed();
|
|
}, nativeFrame->lifetime());
|
|
}
|
|
|
|
if (Platform::AutostartSupported() && controller) {
|
|
const auto minimizedToggled = [=] {
|
|
return cStartMinimized()
|
|
&& !controller->session().domain().local().hasLocalPasscode();
|
|
};
|
|
|
|
const auto autostart = addCheckbox(
|
|
tr::lng_settings_auto_start(),
|
|
cAutoStart());
|
|
const auto minimized = addSlidingCheckbox(
|
|
tr::lng_settings_start_min(),
|
|
minimizedToggled());
|
|
|
|
autostart->checkedChanges(
|
|
) | rpl::filter([](bool checked) {
|
|
return (checked != cAutoStart());
|
|
}) | rpl::start_with_next([=](bool checked) {
|
|
cSetAutoStart(checked);
|
|
Platform::AutostartToggle(checked, crl::guard(autostart, [=](
|
|
bool enabled) {
|
|
autostart->setChecked(enabled);
|
|
if (enabled || !minimized->entity()->checked()) {
|
|
Local::writeSettings();
|
|
} else {
|
|
minimized->entity()->setChecked(false);
|
|
}
|
|
}));
|
|
}, autostart->lifetime());
|
|
|
|
Platform::AutostartRequestStateFromSystem(crl::guard(
|
|
controller,
|
|
[=](bool enabled) { autostart->setChecked(enabled); }));
|
|
|
|
minimized->toggleOn(autostart->checkedValue());
|
|
minimized->entity()->checkedChanges(
|
|
) | rpl::filter([=](bool checked) {
|
|
return (checked != minimizedToggled());
|
|
}) | rpl::start_with_next([=](bool checked) {
|
|
if (controller->session().domain().local().hasLocalPasscode()) {
|
|
minimized->entity()->setChecked(false);
|
|
controller->show(Ui::MakeInformBox(
|
|
tr::lng_error_start_minimized_passcoded()));
|
|
} else {
|
|
cSetStartMinimized(checked);
|
|
Local::writeSettings();
|
|
}
|
|
}, minimized->lifetime());
|
|
|
|
controller->session().domain().local().localPasscodeChanged(
|
|
) | rpl::start_with_next([=] {
|
|
minimized->entity()->setChecked(minimizedToggled());
|
|
}, minimized->lifetime());
|
|
}
|
|
|
|
if (Platform::IsWindows() && !Platform::IsWindowsStoreBuild()) {
|
|
const auto sendto = addCheckbox(
|
|
tr::lng_settings_add_sendto(),
|
|
cSendToMenu());
|
|
|
|
sendto->checkedChanges(
|
|
) | rpl::filter([](bool checked) {
|
|
return (checked != cSendToMenu());
|
|
}) | rpl::start_with_next([](bool checked) {
|
|
cSetSendToMenu(checked);
|
|
psSendToMenu(checked);
|
|
Local::writeSettings();
|
|
}, sendto->lifetime());
|
|
}
|
|
}
|
|
|
|
void SetupSystemIntegrationOptions(
|
|
not_null<Window::SessionController*> controller,
|
|
not_null<Ui::VerticalLayout*> container) {
|
|
auto wrap = object_ptr<Ui::VerticalLayout>(container);
|
|
SetupSystemIntegrationContent(controller, wrap.data());
|
|
if (wrap->count() > 0) {
|
|
container->add(object_ptr<Ui::OverrideMargins>(
|
|
container,
|
|
std::move(wrap)));
|
|
|
|
AddSkip(container, st::settingsCheckboxesSkip);
|
|
}
|
|
}
|
|
|
|
void SetupAnimations(not_null<Ui::VerticalLayout*> container) {
|
|
AddButton(
|
|
container,
|
|
tr::lng_settings_enable_animations(),
|
|
st::settingsButtonNoIcon
|
|
)->toggleOn(
|
|
rpl::single(!anim::Disabled())
|
|
)->toggledValue(
|
|
) | rpl::filter([](bool enabled) {
|
|
return (enabled == anim::Disabled());
|
|
}) | rpl::start_with_next([](bool enabled) {
|
|
anim::SetDisabled(!enabled);
|
|
Local::writeSettings();
|
|
}, container->lifetime());
|
|
}
|
|
|
|
void SetupHardwareAcceleration(not_null<Ui::VerticalLayout*> container) {
|
|
const auto settings = &Core::App().settings();
|
|
AddButton(
|
|
container,
|
|
tr::lng_settings_enable_hwaccel(),
|
|
st::settingsButtonNoIcon
|
|
)->toggleOn(
|
|
rpl::single(settings->hardwareAcceleratedVideo())
|
|
)->toggledValue(
|
|
) | rpl::filter([=](bool enabled) {
|
|
return (enabled != settings->hardwareAcceleratedVideo());
|
|
}) | rpl::start_with_next([=](bool enabled) {
|
|
settings->setHardwareAcceleratedVideo(enabled);
|
|
Core::App().saveSettingsDelayed();
|
|
}, container->lifetime());
|
|
}
|
|
|
|
#ifdef Q_OS_WIN
|
|
void SetupANGLE(
|
|
not_null<Window::SessionController*> controller,
|
|
not_null<Ui::VerticalLayout*> container) {
|
|
using ANGLE = Ui::GL::ANGLE;
|
|
const auto options = std::vector{
|
|
tr::lng_settings_angle_backend_auto(tr::now),
|
|
tr::lng_settings_angle_backend_d3d11(tr::now),
|
|
tr::lng_settings_angle_backend_d3d9(tr::now),
|
|
tr::lng_settings_angle_backend_d3d11on12(tr::now),
|
|
tr::lng_settings_angle_backend_opengl(tr::now),
|
|
tr::lng_settings_angle_backend_disabled(tr::now),
|
|
};
|
|
const auto backendIndex = [] {
|
|
if (Core::App().settings().disableOpenGL()) {
|
|
return 5;
|
|
} else switch (Ui::GL::CurrentANGLE()) {
|
|
case ANGLE::Auto: return 0;
|
|
case ANGLE::D3D11: return 1;
|
|
case ANGLE::D3D9: return 2;
|
|
case ANGLE::D3D11on12: return 3;
|
|
case ANGLE::OpenGL: return 4;
|
|
}
|
|
Unexpected("Ui::GL::CurrentANGLE value in SetupANGLE.");
|
|
}();
|
|
const auto button = AddButtonWithLabel(
|
|
container,
|
|
tr::lng_settings_angle_backend(),
|
|
rpl::single(options[backendIndex]),
|
|
st::settingsButtonNoIcon);
|
|
button->addClickHandler([=] {
|
|
controller->show(Box([=](not_null<Ui::GenericBox*> box) {
|
|
const auto save = [=](int index) {
|
|
if (index == backendIndex) {
|
|
return;
|
|
}
|
|
const auto confirmed = crl::guard(button, [=] {
|
|
const auto nowDisabled = (index == 5);
|
|
if (!nowDisabled) {
|
|
Ui::GL::ChangeANGLE([&] {
|
|
switch (index) {
|
|
case 0: return ANGLE::Auto;
|
|
case 1: return ANGLE::D3D11;
|
|
case 2: return ANGLE::D3D9;
|
|
case 3: return ANGLE::D3D11on12;
|
|
case 4: return ANGLE::OpenGL;
|
|
}
|
|
Unexpected("Index in SetupANGLE.");
|
|
}());
|
|
}
|
|
const auto wasDisabled = (backendIndex == 5);
|
|
if (nowDisabled != wasDisabled) {
|
|
Core::App().settings().setDisableOpenGL(nowDisabled);
|
|
Local::writeSettings();
|
|
}
|
|
Core::Restart();
|
|
});
|
|
controller->show(Ui::MakeConfirmBox({
|
|
.text = tr::lng_settings_need_restart(),
|
|
.confirmed = confirmed,
|
|
.confirmText = tr::lng_settings_restart_now(),
|
|
}));
|
|
};
|
|
SingleChoiceBox(box, {
|
|
.title = tr::lng_settings_angle_backend(),
|
|
.options = options,
|
|
.initialSelection = backendIndex,
|
|
.callback = save,
|
|
});
|
|
}));
|
|
});
|
|
}
|
|
#endif // Q_OS_WIN
|
|
|
|
void SetupOpenGL(
|
|
not_null<Window::SessionController*> controller,
|
|
not_null<Ui::VerticalLayout*> container) {
|
|
const auto toggles = container->lifetime().make_state<
|
|
rpl::event_stream<bool>
|
|
>();
|
|
const auto button = AddButton(
|
|
container,
|
|
tr::lng_settings_enable_opengl(),
|
|
st::settingsButtonNoIcon
|
|
)->toggleOn(
|
|
toggles->events_starting_with_copy(
|
|
!Core::App().settings().disableOpenGL())
|
|
);
|
|
button->toggledValue(
|
|
) | rpl::filter([](bool enabled) {
|
|
return (enabled == Core::App().settings().disableOpenGL());
|
|
}) | rpl::start_with_next([=](bool enabled) {
|
|
const auto confirmed = crl::guard(button, [=] {
|
|
Core::App().settings().setDisableOpenGL(!enabled);
|
|
Local::writeSettings();
|
|
Core::Restart();
|
|
});
|
|
const auto cancelled = crl::guard(button, [=](Fn<void()> close) {
|
|
toggles->fire(!enabled);
|
|
close();
|
|
});
|
|
controller->show(Ui::MakeConfirmBox({
|
|
.text = tr::lng_settings_need_restart(),
|
|
.confirmed = confirmed,
|
|
.cancelled = cancelled,
|
|
.confirmText = tr::lng_settings_restart_now(),
|
|
}));
|
|
}, container->lifetime());
|
|
}
|
|
|
|
void SetupPerformance(
|
|
not_null<Window::SessionController*> controller,
|
|
not_null<Ui::VerticalLayout*> container) {
|
|
SetupAnimations(container);
|
|
SetupHardwareAcceleration(container);
|
|
#ifdef Q_OS_WIN
|
|
SetupANGLE(controller, container);
|
|
#else // Q_OS_WIN
|
|
if constexpr (!Platform::IsMac()) {
|
|
SetupOpenGL(controller, container);
|
|
}
|
|
#endif // Q_OS_WIN
|
|
}
|
|
|
|
void SetupSystemIntegration(
|
|
not_null<Window::SessionController*> controller,
|
|
not_null<Ui::VerticalLayout*> container,
|
|
Fn<void(Type)> showOther) {
|
|
AddDivider(container);
|
|
AddSkip(container);
|
|
AddSubsectionTitle(container, tr::lng_settings_system_integration());
|
|
SetupSystemIntegrationOptions(controller, container);
|
|
AddSkip(container);
|
|
}
|
|
|
|
Advanced::Advanced(
|
|
QWidget *parent,
|
|
not_null<Window::SessionController*> controller)
|
|
: Section(parent) {
|
|
setupContent(controller);
|
|
}
|
|
|
|
rpl::producer<QString> Advanced::title() {
|
|
return tr::lng_settings_advanced();
|
|
}
|
|
|
|
rpl::producer<Type> Advanced::sectionShowOther() {
|
|
return _showOther.events();
|
|
}
|
|
|
|
void Advanced::setupContent(not_null<Window::SessionController*> controller) {
|
|
const auto content = Ui::CreateChild<Ui::VerticalLayout>(this);
|
|
|
|
auto empty = true;
|
|
const auto addDivider = [&] {
|
|
if (empty) {
|
|
empty = false;
|
|
} else {
|
|
AddDivider(content);
|
|
}
|
|
};
|
|
const auto addUpdate = [&] {
|
|
if (HasUpdate()) {
|
|
addDivider();
|
|
AddSkip(content);
|
|
AddSubsectionTitle(content, tr::lng_settings_version_info());
|
|
SetupUpdate(content, [=](Type type) {
|
|
_showOther.fire_copy(type);
|
|
});
|
|
AddSkip(content);
|
|
}
|
|
};
|
|
if (!cAutoUpdate()) {
|
|
addUpdate();
|
|
}
|
|
addDivider();
|
|
SetupDataStorage(controller, content);
|
|
SetupAutoDownload(controller, content);
|
|
SetupSystemIntegration(controller, content, [=](Type type) {
|
|
_showOther.fire_copy(type);
|
|
});
|
|
empty = false;
|
|
|
|
AddDivider(content);
|
|
AddSkip(content);
|
|
AddSubsectionTitle(content, tr::lng_settings_performance());
|
|
SetupPerformance(controller, content);
|
|
AddSkip(content);
|
|
|
|
if (HasSystemSpellchecker()) {
|
|
AddDivider(content);
|
|
AddSkip(content);
|
|
AddSubsectionTitle(content, tr::lng_settings_spellchecker());
|
|
SetupSpellchecker(controller, content);
|
|
AddSkip(content);
|
|
}
|
|
|
|
if (cAutoUpdate()) {
|
|
addUpdate();
|
|
}
|
|
if (!HasUpdate()) {
|
|
AddSkip(content);
|
|
AddDivider(content);
|
|
AddSkip(content);
|
|
content->add(
|
|
CreateButton(
|
|
content,
|
|
tr::lng_settings_experimental(),
|
|
st::settingsButtonNoIcon)
|
|
)->setClickedCallback([=] {
|
|
_showOther.fire_copy(Experimental::Id());
|
|
});
|
|
}
|
|
|
|
AddSkip(content);
|
|
AddDivider(content);
|
|
AddSkip(content);
|
|
SetupExport(controller, content);
|
|
|
|
Ui::ResizeFitChild(this, content);
|
|
}
|
|
|
|
} // namespace Settings
|