/* 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_global_ttl.h" #include "api/api_self_destruct.h" #include "apiwrap.h" #include "boxes/peer_list_controllers.h" #include "data/data_changes.h" #include "data/data_chat.h" #include "data/data_peer.h" #include "data/data_session.h" #include "history/history.h" #include "lang/lang_keys.h" #include "lottie/lottie_icon.h" #include "main/main_session.h" #include "menu/menu_ttl_validator.h" #include "settings/settings_common_session.h" #include "ui/boxes/confirm_box.h" #include "ui/painter.h" #include "ui/vertical_list.h" #include "ui/text/format_values.h" #include "ui/text/text_utilities.h" #include "ui/widgets/buttons.h" #include "ui/widgets/checkbox.h" #include "ui/widgets/labels.h" #include "ui/wrap/vertical_layout.h" #include "window/window_session_controller.h" #include "styles/style_boxes.h" #include "styles/style_layers.h" #include "styles/style_settings.h" #include "styles/style_calls.h" namespace Settings { namespace { class TTLRow : public ChatsListBoxController::Row { public: using ChatsListBoxController::Row::Row; void paintStatusText( Painter &p, const style::PeerListItem &st, int x, int y, int availableWidth, int outerWidth, bool selected) override; }; void TTLRow::paintStatusText( Painter &p, const style::PeerListItem &st, int x, int y, int availableWidth, int outerWidth, bool selected) { auto icon = history()->peer->messagesTTL() ? &st::settingsTTLChatsOn : &st::settingsTTLChatsOff; icon->paint( p, x + st::callArrowPosition.x(), y + st::callArrowPosition.y(), outerWidth); auto shift = st::callArrowPosition.x() + icon->width() + st::callArrowSkip; x += shift; availableWidth -= shift; PeerListRow::paintStatusText( p, st, x, y, availableWidth, outerWidth, selected); } class TTLChatsBoxController : public ChatsListBoxController { public: TTLChatsBoxController(not_null session); Main::Session &session() const override; void rowClicked(not_null row) override; protected: void prepareViewHook() override; std::unique_ptr createRow(not_null history) override; private: const not_null _session; rpl::lifetime _lifetime; }; TTLChatsBoxController::TTLChatsBoxController(not_null session) : ChatsListBoxController(session) , _session(session) { } Main::Session &TTLChatsBoxController::session() const { return *_session; } void TTLChatsBoxController::prepareViewHook() { delegate()->peerListSetTitle(tr::lng_settings_ttl_title()); } void TTLChatsBoxController::rowClicked(not_null row) { if (!TTLMenu::TTLValidator(nullptr, row->peer()).can()) { delegate()->peerListUiShow()->showToast( { tr::lng_settings_ttl_select_chats_sorry(tr::now) }); return; } delegate()->peerListSetRowChecked(row, !row->checked()); } std::unique_ptr TTLChatsBoxController::createRow( not_null history) { if (history->peer->isSelf() || history->peer->isRepliesChat()) { return nullptr; } else if (history->peer->isChat() && history->peer->asChat()->amIn()) { } else if (history->peer->isMegagroup()) { } else if (!TTLMenu::TTLValidator(nullptr, history->peer).can()) { return nullptr; } if (session().data().contactsNoChatsList()->contains({ history })) { return nullptr; } auto result = std::make_unique(history); const auto applyStatus = [=, raw = result.get()] { const auto ttl = history->peer->messagesTTL(); raw->setCustomStatus( ttl ? tr::lng_settings_ttl_select_chats_status( tr::now, lt_after_duration, Ui::FormatTTLAfter(ttl)) : tr::lng_settings_ttl_select_chats_status_disabled(tr::now), ttl); }; applyStatus(); return result; } void SetupTopContent( not_null parent, rpl::producer<> showFinished) { const auto divider = Ui::CreateChild(parent.get()); const auto verticalLayout = parent->add( object_ptr(parent.get())); auto icon = CreateLottieIcon( verticalLayout, { .name = u"ttl"_q, .sizeOverride = { st::settingsCloudPasswordIconSize, st::settingsCloudPasswordIconSize, }, }, st::settingsFilterIconPadding); std::move( showFinished ) | rpl::start_with_next([animate = std::move(icon.animate)] { animate(anim::repeat::loop); }, verticalLayout->lifetime()); verticalLayout->add(std::move(icon.widget)); verticalLayout->geometryValue( ) | rpl::start_with_next([=](const QRect &r) { divider->setGeometry(r); }, divider->lifetime()); } } // namespace class GlobalTTL : public Section { public: GlobalTTL( QWidget *parent, not_null controller); [[nodiscard]] rpl::producer title() override; void setupContent(); void showFinished() override final; private: void rebuildButtons(TimeId currentTTL) const; void showSure(TimeId ttl, bool rebuild) const; void request(TimeId ttl) const; const not_null _controller; const std::shared_ptr _group; const std::shared_ptr _show; not_null _buttons; rpl::event_stream<> _showFinished; rpl::lifetime _requestLifetime; }; GlobalTTL::GlobalTTL( QWidget *parent, not_null controller) : Section(parent) , _controller(controller) , _group(std::make_shared(0)) , _show(controller->uiShow()) , _buttons(Ui::CreateChild(this)) { setupContent(); } rpl::producer GlobalTTL::title() { return tr::lng_settings_ttl_title(); } void GlobalTTL::request(TimeId ttl) const { _controller->session().api().selfDestruct().updateDefaultHistoryTTL(ttl); } void GlobalTTL::showSure(TimeId ttl, bool rebuild) const { const auto ttlText = Ui::FormatTTLAfter(ttl); const auto confirmed = [=] { if (rebuild) { rebuildButtons(ttl); } _group->setChangedCallback([=](int value) { _group->setChangedCallback(nullptr); _show->showToast(tr::lng_settings_ttl_after_toast( tr::now, lt_after_duration, { .text = ttlText }, Ui::Text::WithEntities)); _show->hideLayer(); // Don't use close(). }); request(ttl); }; if (_group->value()) { confirmed(); return; } _show->showBox(Ui::MakeConfirmBox({ .text = tr::lng_settings_ttl_after_sure( lt_after_duration, rpl::single(ttlText)), .confirmed = confirmed, .cancelled = [=](Fn &&close) { _group->setChangedCallback(nullptr); close(); }, .confirmText = tr::lng_sure_enable(), })); } void GlobalTTL::rebuildButtons(TimeId currentTTL) const { auto ttls = std::vector{ 0, 3600 * 24, 3600 * 24 * 7, 3600 * 24 * 31, }; if (!ranges::contains(ttls, currentTTL)) { ttls.push_back(currentTTL); ranges::sort(ttls); } if (_buttons->count() > ttls.size()) { return; } _buttons->clear(); for (const auto &ttl : ttls) { const auto ttlText = Ui::FormatTTLAfter(ttl); const auto button = _buttons->add(object_ptr( _buttons, (!ttl) ? tr::lng_settings_ttl_after_off() : tr::lng_settings_ttl_after( lt_after_duration, rpl::single(ttlText)), st::settingsButtonNoIcon)); button->setClickedCallback([=] { if (_group->current() == ttl) { return; } if (!ttl) { _group->setChangedCallback(nullptr); request(ttl); return; } showSure(ttl, false); }); const auto radio = Ui::CreateChild( button, _group, ttl, QString()); radio->setAttribute(Qt::WA_TransparentForMouseEvents); radio->show(); button->sizeValue( ) | rpl::start_with_next([=] { radio->moveToRight(0, radio->checkRect().top()); }, radio->lifetime()); } _buttons->resizeToWidth(width()); } void GlobalTTL::setupContent() { setFocusPolicy(Qt::StrongFocus); setFocus(); const auto content = Ui::CreateChild(this); SetupTopContent(content, _showFinished.events()); Ui::AddSkip(content); Ui::AddSubsectionTitle(content, tr::lng_settings_ttl_after_subtitle()); content->add(object_ptr::fromRaw(_buttons)); { const auto &apiTTL = _controller->session().api().selfDestruct(); const auto rebuild = [=](TimeId period) { rebuildButtons(period); _group->setValue(period); }; rebuild(apiTTL.periodDefaultHistoryTTLCurrent()); apiTTL.periodDefaultHistoryTTL( ) | rpl::start_with_next(rebuild, content->lifetime()); } const auto show = _controller->uiShow(); content->add(object_ptr( content, tr::lng_settings_ttl_after_custom(), st::settingsButtonNoIcon))->setClickedCallback([=] { struct Args { std::shared_ptr show; TimeId startTtl; rpl::producer about; Fn callback; }; show->showBox(Box(TTLMenu::TTLBox, TTLMenu::Args{ .show = show, .startTtl = _group->current(), .callback = [=](TimeId ttl, Fn) { showSure(ttl, true); }, .hideDisable = true, })); }); Ui::AddSkip(content); auto footer = object_ptr( content, tr::lng_settings_ttl_after_about( lt_link, tr::lng_settings_ttl_after_about_link( ) | rpl::map([](QString s) { return Ui::Text::Link(s, 1); }), Ui::Text::WithEntities), st::boxDividerLabel); footer->setLink(1, std::make_shared([=] { const auto session = &_controller->session(); auto controller = std::make_unique(session); auto initBox = [=, controller = controller.get()]( not_null box) { box->addButton(tr::lng_settings_apply(), crl::guard(this, [=] { const auto &peers = box->collectSelectedRows(); if (peers.empty()) { return; } const auto &apiTTL = session->api().selfDestruct(); const auto ttl = apiTTL.periodDefaultHistoryTTLCurrent(); for (const auto &peer : peers) { peer->session().api().request(MTPmessages_SetHistoryTTL( peer->input, MTP_int(ttl) )).done([=](const MTPUpdates &result) { peer->session().api().applyUpdates(result); }).send(); } box->showToast(ttl ? tr::lng_settings_ttl_select_chats_toast( tr::now, lt_count, peers.size(), lt_duration, { .text = Ui::FormatTTL(ttl) }, Ui::Text::WithEntities) : tr::lng_settings_ttl_select_chats_disabled_toast( tr::now, lt_count, peers.size(), Ui::Text::WithEntities)); box->closeBox(); })); box->addButton(tr::lng_cancel(), [=] { box->closeBox(); }); }; _controller->show( Box(std::move(controller), std::move(initBox))); })); content->add(object_ptr( content, std::move(footer), st::defaultBoxDividerLabelPadding)); Ui::ResizeFitChild(this, content); } void GlobalTTL::showFinished() { _showFinished.fire({}); } Type GlobalTTLId() { return GlobalTTL::Id(); } } // namespace Settings