/* 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_main.h" #include "settings/settings_common.h" #include "settings/settings_codes.h" #include "boxes/language_box.h" #include "boxes/confirm_box.h" #include "boxes/about_box.h" #include "ui/wrap/vertical_layout.h" #include "ui/widgets/labels.h" #include "ui/widgets/discrete_sliders.h" #include "info/profile/info_profile_button.h" #include "info/profile/info_profile_cover.h" #include "lang/lang_keys.h" #include "storage/localstorage.h" #include "auth_session.h" #include "apiwrap.h" #include "core/file_utilities.h" #include "styles/style_settings.h" namespace Settings { void SetupLanguageButton( not_null container, bool icon) { const auto button = AddButtonWithLabel( container, lng_settings_language, Lang::Viewer(lng_language_name), icon ? st::settingsSectionButton : st::settingsButton, icon ? &st::settingsIconLanguage : nullptr); const auto guard = Ui::AttachAsChild(button, base::binary_guard()); button->addClickHandler([=] { *guard = LanguageBox::Show(); }); } void SetupSections( not_null container, Fn showOther) { AddDivider(container); AddSkip(container); const auto addSection = [&]( LangKey label, Type type, const style::icon *icon) { AddButton( container, label, st::settingsSectionButton, icon )->addClickHandler([=] { showOther(type); }); }; addSection( lng_settings_information, Type::Information, &st::settingsIconInformation); addSection( lng_settings_section_notify, Type::Notifications, &st::settingsIconNotifications); addSection( lng_settings_section_privacy, Type::PrivacySecurity, &st::settingsIconPrivacySecurity); addSection( lng_settings_section_chat_settings, Type::Chat, &st::settingsIconChat); addSection( lng_settings_advanced, Type::Advanced, &st::settingsIconGeneral); SetupLanguageButton(container); AddSkip(container); } bool HasInterfaceScale() { return !cRetina(); } void SetupInterfaceScale( not_null container, bool icon) { if (!HasInterfaceScale()) { return; } const auto toggled = Ui::AttachAsChild( container, rpl::event_stream()); const auto switched = (cConfigScale() == dbisAuto) || (cConfigScale() == cScreenScale()); const auto button = AddButton( container, lng_settings_default_scale, icon ? st::settingsSectionButton : st::settingsButton, icon ? &st::settingsIconInterfaceScale : nullptr )->toggleOn(toggled->events_starting_with_copy(switched)); const auto slider = container->add( object_ptr(container, st::settingsSlider), icon ? st::settingsScalePadding : st::settingsBigScalePadding); const auto inSetScale = Ui::AttachAsChild(container, false); const auto setScale = std::make_shared>(); *setScale = [=](DBIScale scale) { if (*inSetScale) return; *inSetScale = true; const auto guard = gsl::finally([=] { *inSetScale = false; }); if (scale == cScreenScale()) { scale = dbisAuto; } toggled->fire(scale == dbisAuto); const auto applying = scale; if (scale == dbisAuto) { scale = cScreenScale(); } slider->setActiveSection(scale - 1); if (cEvalScale(scale) != cEvalScale(cRealScale())) { const auto confirmed = crl::guard(button, [=] { cSetConfigScale(applying); Local::writeSettings(); App::restart(); }); const auto cancelled = crl::guard(button, [=] { App::CallDelayed( st::defaultSettingsSlider.duration, button, [=] { (*setScale)(cRealScale()); }); }); Ui::show(Box( lang(lng_settings_need_restart), lang(lng_settings_restart_now), confirmed, cancelled)); } else { cSetConfigScale(scale); Local::writeSettings(); } }; button->toggledValue( ) | rpl::start_with_next([=](bool checked) { auto scale = checked ? dbisAuto : cEvalScale(cConfigScale()); if (scale == cScreenScale()) { if (scale != cScale()) { scale = cScale(); } else { switch (scale) { case dbisOne: scale = dbisOneAndQuarter; break; case dbisOneAndQuarter: scale = dbisOne; break; case dbisOneAndHalf: scale = dbisOneAndQuarter; break; case dbisTwo: scale = dbisOneAndHalf; break; } } } (*setScale)(scale); }, button->lifetime()); const auto label = [](DBIScale scale) { switch (scale) { case dbisOne: return qsl("100%"); case dbisOneAndQuarter: return qsl("125%"); case dbisOneAndHalf: return qsl("150%"); case dbisTwo: return qsl("200%"); } Unexpected("Value in scale label."); }; const auto scaleByIndex = [](int index) { switch (index) { case 0: return dbisOne; case 1: return dbisOneAndQuarter; case 2: return dbisOneAndHalf; case 3: return dbisTwo; } Unexpected("Index in scaleByIndex."); }; slider->addSection(label(dbisOne)); slider->addSection(label(dbisOneAndQuarter)); slider->addSection(label(dbisOneAndHalf)); slider->addSection(label(dbisTwo)); slider->setActiveSectionFast(cEvalScale(cConfigScale()) - 1); slider->sectionActivated( ) | rpl::start_with_next([=](int section) { (*setScale)(scaleByIndex(section)); }, slider->lifetime()); } void OpenFaq() { QDesktopServices::openUrl(telegramFaqLink()); } void SetupFaq(not_null container, bool icon) { AddButton( container, lng_settings_faq, icon ? st::settingsSectionButton : st::settingsButton, icon ? &st::settingsIconFaq : nullptr )->addClickHandler(OpenFaq); } void SetupHelp(not_null container) { AddDivider(container); AddSkip(container); SetupFaq(container); if (AuthSession::Exists()) { const auto button = AddButton( container, lng_settings_ask_question, st::settingsSectionButton); button->addClickHandler([=] { const auto ready = crl::guard(button, [](const MTPUser &data) { const auto users = MTP_vector(1, data); if (const auto user = App::feedUsers(users)) { Ui::showPeerHistory(user, ShowAtUnreadMsgId); } }); const auto sure = crl::guard(button, [=] { Auth().api().requestSupportContact(ready); }); auto box = Box( lang(lng_settings_ask_sure), lang(lng_settings_ask_ok), lang(lng_settings_faq_button), sure, OpenFaq); box->setStrictCancel(true); Ui::show(std::move(box)); }); } AddSkip(container); } Main::Main( QWidget *parent, not_null controller, not_null self) : Section(parent) , _self(self) { setupContent(controller); } void Main::keyPressEvent(QKeyEvent *e) { CodesFeedString(e->text()); return Section::keyPressEvent(e); } void Main::setupContent(not_null controller) { const auto content = Ui::CreateChild(this); const auto cover = content->add(object_ptr( content, _self, controller)); cover->setOnlineCount(rpl::single(0)); SetupSections(content, [=](Type type) { _showOther.fire_copy(type); }); if (HasInterfaceScale()) { AddDivider(content); AddSkip(content); SetupInterfaceScale(content); AddSkip(content); } SetupHelp(content); Ui::ResizeFitChild(this, content); // If we load this in advance it won't jump when we open its' section. Auth().api().reloadPasswordState(); } rpl::producer Main::sectionShowOther() { return _showOther.events(); } } // namespace Settings