| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446 |
- /*
- 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 "boxes/dictionaries_manager.h"
- #ifndef TDESKTOP_DISABLE_SPELLCHECK
- #include "base/event_filter.h"
- #include "chat_helpers/spellchecker_common.h"
- #include "core/application.h"
- #include "core/core_settings.h"
- #include "lang/lang_keys.h"
- #include "main/main_account.h"
- #include "main/main_session.h"
- #include "mainwidget.h"
- #include "mtproto/dedicated_file_loader.h"
- #include "spellcheck/spellcheck_utils.h"
- #include "ui/wrap/vertical_layout.h"
- #include "ui/widgets/buttons.h"
- #include "ui/widgets/labels.h"
- #include "ui/widgets/multi_select.h"
- #include "ui/widgets/popup_menu.h"
- #include "ui/wrap/slide_wrap.h"
- #include "ui/effects/animations.h"
- #include "styles/style_layers.h"
- #include "styles/style_settings.h"
- #include "styles/style_boxes.h"
- #include "styles/style_menu_icons.h"
- namespace Ui {
- namespace {
- using Dictionaries = std::vector<int>;
- using namespace Storage::CloudBlob;
- using Loading = MTP::DedicatedLoader::Progress;
- using DictState = BlobState;
- using QueryCallback = Fn<void(const QString &)>;
- constexpr auto kMaxQueryLength = 15;
- class Inner : public Ui::RpWidget {
- public:
- Inner(
- QWidget *parent,
- not_null<Main::Session*> session,
- Dictionaries enabledDictionaries);
- Dictionaries enabledRows() const;
- QueryCallback queryCallback() const;
- private:
- void setupContent(
- not_null<Main::Session*> session,
- Dictionaries enabledDictionaries);
- Dictionaries _enabledRows;
- QueryCallback _queryCallback;
- };
- inline auto DictExists(int langId) {
- return Spellchecker::DictionaryExists(langId);
- }
- inline auto FilterEnabledDict(Dictionaries dicts) {
- return dicts | ranges::views::filter(
- DictExists
- ) | ranges::to_vector;
- }
- DictState ComputeState(int id, bool enabled) {
- const auto result = enabled ? DictState(Active()) : DictState(Ready());
- if (DictExists(id)) {
- return result;
- }
- return Available{ Spellchecker::GetDownloadSize(id) };
- }
- QString StateDescription(const DictState &state) {
- return StateDescription(
- state,
- tr::lng_settings_manage_enabled_dictionary);
- }
- auto CreateMultiSelect(QWidget *parent) {
- const auto result = Ui::CreateChild<Ui::MultiSelect>(
- parent,
- st::defaultMultiSelect,
- tr::lng_participant_filter());
- result->resizeToWidth(st::boxWidth);
- result->moveToLeft(0, 0);
- return result;
- }
- Inner::Inner(
- QWidget *parent,
- not_null<Main::Session*> session,
- Dictionaries enabledDictionaries)
- : RpWidget(parent) {
- setupContent(session, std::move(enabledDictionaries));
- }
- QueryCallback Inner::queryCallback() const {
- return _queryCallback;
- }
- Dictionaries Inner::enabledRows() const {
- return _enabledRows;
- }
- auto AddButtonWithLoader(
- not_null<Ui::VerticalLayout*> content,
- not_null<Main::Session*> session,
- const Spellchecker::Dict &dict,
- bool buttonEnabled,
- rpl::producer<QStringView> query) {
- const auto id = dict.id;
- buttonEnabled &= DictExists(id);
- const auto locale = Spellchecker::LocaleFromLangId(id);
- const std::vector<QString> indexList = {
- dict.name,
- QLocale::languageToString(locale.language()),
- QLocale::countryToString(locale.country())
- };
- const auto wrap = content->add(
- object_ptr<Ui::SlideWrap<Ui::SettingsButton>>(
- content,
- object_ptr<Ui::SettingsButton>(
- content,
- rpl::single(dict.name),
- st::dictionariesSectionButton
- )
- )
- );
- const auto button = wrap->entity();
- std::move(
- query
- ) | rpl::start_with_next([=](auto string) {
- wrap->toggle(
- ranges::any_of(indexList, [&](const QString &s) {
- return s.startsWith(string, Qt::CaseInsensitive);
- }),
- anim::type::instant);
- }, button->lifetime());
- using Loader = Spellchecker::DictLoader;
- using GlobalLoaderPtr = std::shared_ptr<base::unique_qptr<Loader>>;
- const auto localLoader = button->lifetime()
- .make_state<base::unique_qptr<Loader>>();
- const auto localLoaderValues = button->lifetime()
- .make_state<rpl::event_stream<Loader*>>();
- const auto setLocalLoader = [=](base::unique_qptr<Loader> loader) {
- *localLoader = std::move(loader);
- localLoaderValues->fire(localLoader->get());
- };
- const auto destroyLocalLoader = [=] {
- setLocalLoader(nullptr);
- };
- const auto buttonState = button->lifetime()
- .make_state<rpl::variable<DictState>>();
- const auto dictionaryRemoved = button->lifetime()
- .make_state<rpl::event_stream<>>();
- const auto dictionaryFromGlobalLoader = button->lifetime()
- .make_state<rpl::event_stream<>>();
- const auto globalLoader = button->lifetime()
- .make_state<GlobalLoaderPtr>();
- const auto rawGlobalLoaderPtr = [=]() -> Loader* {
- if (!globalLoader || !*globalLoader || !*globalLoader->get()) {
- return nullptr;
- }
- return globalLoader->get()->get();
- };
- const auto setGlobalLoaderPtr = [=](GlobalLoaderPtr loader) {
- if (localLoader->get()) {
- if (loader && loader->get()) {
- loader->get()->destroy();
- }
- return;
- }
- *globalLoader = std::move(loader);
- localLoaderValues->fire(rawGlobalLoaderPtr());
- if (rawGlobalLoaderPtr()) {
- dictionaryFromGlobalLoader->fire({});
- }
- };
- Spellchecker::GlobalLoaderChanged(
- ) | rpl::start_with_next([=](int langId) {
- if (!langId && rawGlobalLoaderPtr()) {
- setGlobalLoaderPtr(nullptr);
- } else if (langId == id) {
- setGlobalLoaderPtr(Spellchecker::GlobalLoader());
- }
- }, button->lifetime());
- const auto label = Ui::CreateChild<Ui::FlatLabel>(
- button,
- buttonState->value() | rpl::map(StateDescription),
- st::settingsUpdateState);
- label->setAttribute(Qt::WA_TransparentForMouseEvents);
- rpl::combine(
- button->widthValue(),
- label->widthValue()
- ) | rpl::start_with_next([=] {
- label->moveToLeft(
- st::settingsUpdateStatePosition.x(),
- st::settingsUpdateStatePosition.y());
- }, label->lifetime());
- buttonState->value(
- ) | rpl::start_with_next([=](const DictState &state) {
- const auto isToggledSet = v::is<Active>(state);
- const auto toggled = isToggledSet ? 1. : 0.;
- const auto over = !button->isDisabled()
- && (button->isDown() || button->isOver());
- if (toggled == 0. && !over) {
- label->setTextColorOverride(std::nullopt);
- } else {
- label->setTextColorOverride(anim::color(
- over ? st::contactsStatusFgOver : st::contactsStatusFg,
- st::contactsStatusFgOnline,
- toggled));
- }
- }, label->lifetime());
- button->toggleOn(
- rpl::single(
- buttonEnabled
- ) | rpl::then(
- rpl::merge(
- // Events to toggle on.
- dictionaryFromGlobalLoader->events() | rpl::map_to(true),
- // Events to toggle off.
- rpl::merge(
- dictionaryRemoved->events(),
- buttonState->value(
- ) | rpl::filter([](const DictState &state) {
- return v::is<Failed>(state);
- }) | rpl::to_empty
- ) | rpl::map_to(false)
- )
- )
- );
- *buttonState = localLoaderValues->events_starting_with(
- rawGlobalLoaderPtr() ? rawGlobalLoaderPtr() : localLoader->get()
- ) | rpl::map([=](Loader *loader) {
- return (loader && loader->id() == id)
- ? loader->state()
- : rpl::single(
- buttonEnabled
- ) | rpl::then(
- rpl::merge(
- dictionaryRemoved->events() | rpl::map_to(false),
- button->toggledValue()
- )
- ) | rpl::map([=](auto enabled) {
- return ComputeState(id, enabled);
- });
- }) | rpl::flatten_latest(
- ) | rpl::filter([=](const DictState &state) {
- return !v::is<Failed>(buttonState->current())
- || !v::is<Available>(state);
- });
- button->toggledValue(
- ) | rpl::start_with_next([=](bool toggled) {
- const auto &state = buttonState->current();
- if (toggled && (v::is<Available>(state) || v::is<Failed>(state))) {
- const auto weak = Ui::MakeWeak(button);
- setLocalLoader(base::make_unique_q<Loader>(
- QCoreApplication::instance(),
- session,
- id,
- Spellchecker::GetDownloadLocation(id),
- Spellchecker::DictPathByLangId(id),
- Spellchecker::GetDownloadSize(id),
- crl::guard(weak, destroyLocalLoader)));
- } else if (!toggled && v::is<Loading>(state)) {
- if (const auto g = rawGlobalLoaderPtr()) {
- g->destroy();
- return;
- }
- if (localLoader && localLoader->get()->id() == id) {
- destroyLocalLoader();
- }
- }
- }, button->lifetime());
- const auto contextMenu = button->lifetime()
- .make_state<base::unique_qptr<Ui::PopupMenu>>();
- const auto showMenu = [=] {
- if (!DictExists(id)) {
- return false;
- }
- *contextMenu = base::make_unique_q<Ui::PopupMenu>(
- button,
- st::popupMenuWithIcons);
- contextMenu->get()->addAction(
- tr::lng_settings_manage_remove_dictionary(tr::now), [=] {
- Spellchecker::RemoveDictionary(id);
- dictionaryRemoved->fire({});
- }, &st::menuIconDelete);
- contextMenu->get()->popup(QCursor::pos());
- return true;
- };
- base::install_event_filter(button, [=](not_null<QEvent*> e) {
- if (e->type() == QEvent::ContextMenu && showMenu()) {
- return base::EventFilterResult::Cancel;
- }
- return base::EventFilterResult::Continue;
- });
- if (const auto g = Spellchecker::GlobalLoader()) {
- if (g.get() && g->get()->id() == id) {
- setGlobalLoaderPtr(g);
- }
- }
- return button;
- }
- void Inner::setupContent(
- not_null<Main::Session*> session,
- Dictionaries enabledDictionaries) {
- const auto content = Ui::CreateChild<Ui::VerticalLayout>(this);
- const auto queryStream = content->lifetime()
- .make_state<rpl::event_stream<QStringView>>();
- for (const auto &dict : Spellchecker::Dictionaries()) {
- const auto id = dict.id;
- const auto row = AddButtonWithLoader(
- content,
- session,
- dict,
- ranges::contains(enabledDictionaries, id),
- queryStream->events());
- row->toggledValue(
- ) | rpl::start_with_next([=](auto enabled) {
- if (enabled) {
- _enabledRows.push_back(id);
- } else {
- auto &rows = _enabledRows;
- rows.erase(ranges::remove(rows, id), end(rows));
- }
- }, row->lifetime());
- }
- _queryCallback = [=](const QString &query) {
- if (query.size() >= kMaxQueryLength) {
- return;
- }
- queryStream->fire_copy(query);
- };
- content->resizeToWidth(st::boxWidth);
- Ui::ResizeFitChild(this, content);
- }
- } // namespace
- ManageDictionariesBox::ManageDictionariesBox(
- QWidget*,
- not_null<Main::Session*> session)
- : _session(session) {
- }
- void ManageDictionariesBox::setInnerFocus() {
- _setInnerFocus();
- }
- void ManageDictionariesBox::prepare() {
- const auto multiSelect = CreateMultiSelect(this);
- const auto inner = setInnerWidget(
- object_ptr<Inner>(
- this,
- _session,
- Core::App().settings().dictionariesEnabled()),
- st::boxScroll,
- multiSelect->height()
- );
- multiSelect->setQueryChangedCallback(inner->queryCallback());
- _setInnerFocus = [=] {
- multiSelect->setInnerFocus();
- };
- // The initial list of enabled rows may differ from the list of languages
- // in settings, so we should store it when box opens
- // and save it when box closes (don't do it when "Save" was pressed).
- const auto initialEnabledRows = inner->enabledRows();
- setTitle(tr::lng_settings_manage_dictionaries());
- addButton(tr::lng_settings_save(), [=] {
- Core::App().settings().setDictionariesEnabled(
- FilterEnabledDict(inner->enabledRows()));
- Core::App().saveSettingsDelayed();
- // Ignore boxClosing() when the Save button was pressed.
- lifetime().destroy();
- closeBox();
- });
- addButton(tr::lng_close(), [=] { closeBox(); });
- boxClosing() | rpl::start_with_next([=] {
- Core::App().settings().setDictionariesEnabled(
- FilterEnabledDict(initialEnabledRows));
- Core::App().saveSettingsDelayed();
- }, lifetime());
- setDimensionsToContent(st::boxWidth, inner);
- using namespace rpl::mappers;
- const auto max = lifetime().make_state<int>(0);
- rpl::combine(
- inner->heightValue(),
- multiSelect->heightValue(),
- _1 + _2
- ) | rpl::start_with_next([=](int height) {
- using std::min;
- accumulate_max(*max, height);
- setDimensions(st::boxWidth, min(*max, st::boxMaxListHeight), true);
- }, inner->lifetime());
- }
- } // namespace Ui
- #endif // !TDESKTOP_DISABLE_SPELLCHECK
|