/* 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 "intro/intro_widget.h" #include "intro/intro_start.h" #include "intro/intro_phone.h" #include "intro/intro_qr.h" #include "intro/intro_code.h" #include "intro/intro_signup.h" #include "intro/intro_password_check.h" #include "lang/lang_keys.h" #include "lang/lang_instance.h" #include "lang/lang_cloud_manager.h" #include "storage/localstorage.h" #include "main/main_account.h" #include "main/main_domain.h" #include "main/main_session.h" #include "mainwindow.h" #include "data/data_user.h" #include "data/data_countries.h" #include "boxes/confirm_box.h" #include "ui/text/text_utilities.h" #include "ui/widgets/buttons.h" #include "ui/widgets/labels.h" #include "ui/wrap/fade_wrap.h" #include "core/update_checker.h" #include "core/application.h" #include "mtproto/mtproto_dc_options.h" #include "window/window_slide_animation.h" #include "window/window_connecting_widget.h" #include "window/section_widget.h" #include "base/platform/base_platform_info.h" #include "api/api_text_entities.h" #include "app.h" #include "styles/style_layers.h" #include "styles/style_intro.h" namespace Intro { namespace { using namespace ::Intro::details; [[nodiscard]] QString ComputeNewAccountCountry() { if (const auto parent = Core::App().domain().maybeLastOrSomeAuthedAccount()) { if (const auto session = parent->maybeSession()) { const auto iso = ::Data::CountryISO2ByPhone( session->user()->phone()); if (!iso.isEmpty()) { return iso; } } } return Platform::SystemCountry(); } } // namespace Widget::Widget( QWidget *parent, not_null account, EnterPoint point) : RpWidget(parent) , _account(account) , _back(this, object_ptr(this, st::introBackButton)) , _settings( this, object_ptr( this, tr::lng_menu_settings(), st::defaultBoxButton)) , _next( this, object_ptr(this, nullptr, st::introNextButton)) , _connecting(std::make_unique( this, account, rpl::single(true))) { Core::App().setDefaultFloatPlayerDelegate(floatPlayerDelegate()); getData()->country = ComputeNewAccountCountry(); _account->mtpValue( ) | rpl::start_with_next([=](not_null instance) { _api.emplace(instance); crl::on_main(this, [=] { createLanguageLink(); }); }, lifetime()); switch (point) { case EnterPoint::Start: getNearestDC(); appendStep(new StartWidget(this, _account, getData())); break; case EnterPoint::Phone: appendStep(new PhoneWidget(this, _account, getData())); break; case EnterPoint::Qr: appendStep(new QrWidget(this, _account, getData())); break; default: Unexpected("Enter point in Intro::Widget::Widget."); } fixOrder(); subscribe(Lang::CurrentCloudManager().firstLanguageSuggestion(), [=] { createLanguageLink(); }); _account->mtpUpdates( ) | rpl::start_with_next([=](const MTPUpdates &updates) { handleUpdates(updates); }, lifetime()); _back->entity()->setClickedCallback([=] { backRequested(); }); _back->hide(anim::type::instant); _next->entity()->setClickedCallback([=] { getStep()->submit(); }); if (_changeLanguage) { _changeLanguage->finishAnimating(); } Lang::Updated( ) | rpl::start_with_next([=] { refreshLang(); }, lifetime()); show(); showControls(); getStep()->showFast(); setInnerFocus(); cSetPasswordRecovered(false); if (!Core::UpdaterDisabled()) { Core::UpdateChecker checker; checker.start(); rpl::merge( rpl::single(rpl::empty_value()), checker.isLatest(), checker.failed(), checker.ready() ) | rpl::start_with_next([=] { checkUpdateStatus(); }, lifetime()); } } rpl::producer<> Widget::showSettingsRequested() const { return _settings->entity()->clicks() | rpl::to_empty; } not_null Widget::floatPlayerDelegate() { return static_cast(this); } auto Widget::floatPlayerSectionDelegate() -> not_null { return static_cast(this); } not_null Widget::floatPlayerWidget() { return this; } auto Widget::floatPlayerGetSection(Window::Column column) -> not_null { return this; } void Widget::floatPlayerEnumerateSections(Fn widget, Window::Column widgetColumn)> callback) { callback(this, Window::Column::Second); } bool Widget::floatPlayerIsVisible(not_null item) { return false; } QRect Widget::floatPlayerAvailableRect() { return mapToGlobal(rect()); } bool Widget::floatPlayerHandleWheelEvent(QEvent *e) { return false; } void Widget::refreshLang() { _changeLanguage.destroy(); createLanguageLink(); InvokeQueued(this, [this] { updateControlsGeometry(); }); } void Widget::handleUpdates(const MTPUpdates &updates) { updates.match([&](const MTPDupdateShort &data) { handleUpdate(data.vupdate()); }, [&](const MTPDupdates &data) { for (const auto &update : data.vupdates().v) { handleUpdate(update); } }, [&](const MTPDupdatesCombined &data) { for (const auto &update : data.vupdates().v) { handleUpdate(update); } }, [](const auto &) {}); } void Widget::handleUpdate(const MTPUpdate &update) { update.match([&](const MTPDupdateDcOptions &data) { _account->mtp().dcOptions().addFromList(data.vdc_options()); }, [&](const MTPDupdateConfig &data) { _account->mtp().requestConfig(); }, [&](const MTPDupdateServiceNotification &data) { const auto text = TextWithEntities{ qs(data.vmessage()), Api::EntitiesFromMTP(nullptr, data.ventities().v) }; Ui::show(Box(text)); }, [](const auto &) {}); } void Widget::createLanguageLink() { if (_changeLanguage || Core::App().domain().maybeLastOrSomeAuthedAccount()) { return; } const auto createLink = [=]( const QString &text, const QString &languageId) { _changeLanguage.create( this, object_ptr(this, text)); _changeLanguage->hide(anim::type::instant); _changeLanguage->entity()->setClickedCallback([=] { Lang::CurrentCloudManager().switchToLanguage(languageId); }); _changeLanguage->toggle( !_resetAccount && !_terms && _nextShown, anim::type::normal); updateControlsGeometry(); }; const auto currentId = Lang::LanguageIdOrDefault(Lang::Id()); const auto defaultId = Lang::DefaultLanguageId(); const auto suggested = Lang::CurrentCloudManager().suggestedLanguage(); if (currentId != defaultId) { createLink( Lang::GetOriginalValue(tr::lng_switch_to_this.base), defaultId); } else if (!suggested.isEmpty() && suggested != currentId && _api) { _api->request(MTPlangpack_GetStrings( MTP_string(Lang::CloudLangPackName()), MTP_string(suggested), MTP_vector(1, MTP_string("lng_switch_to_this")) )).done([=](const MTPVector &result) { const auto strings = Lang::Instance::ParseStrings(result); const auto i = strings.find(tr::lng_switch_to_this.base); if (i != strings.end()) { createLink(i->second, suggested); } }).send(); } } void Widget::checkUpdateStatus() { Expects(!Core::UpdaterDisabled()); if (Core::UpdateChecker().state() == Core::UpdateChecker::State::Ready) { if (_update) return; _update.create( this, object_ptr( this, tr::lng_menu_update(), st::defaultBoxButton)); if (!_a_show.animating()) { _update->setVisible(true); } const auto stepHasCover = getStep()->hasCover(); _update->toggle(!stepHasCover, anim::type::instant); _update->entity()->setClickedCallback([] { Core::checkReadyUpdate(); App::restart(); }); } else { if (!_update) return; _update.destroy(); } updateControlsGeometry(); } void Widget::setInnerFocus() { if (getStep()->animating()) { setFocus(); } else { getStep()->setInnerFocus(); } } void Widget::historyMove(StackAction action, Animate animate) { Expects(_stepHistory.size() > 1); if (getStep()->animating()) { return; } auto wasStep = getStep((action == StackAction::Back) ? 0 : 1); if (action == StackAction::Back) { _stepHistory.pop_back(); wasStep->cancelled(); } else if (action == StackAction::Replace) { _stepHistory.erase(_stepHistory.end() - 2); } if (_resetAccount) { hideAndDestroy(std::exchange(_resetAccount, { nullptr })); } if (_terms) { hideAndDestroy(std::exchange(_terms, { nullptr })); } getStep()->finishInit(); getStep()->prepareShowAnimated(wasStep); if (wasStep->hasCover() != getStep()->hasCover()) { _nextTopFrom = wasStep->contentTop() + st::introNextTop; _controlsTopFrom = wasStep->hasCover() ? st::introCoverHeight : 0; _coverShownAnimation.start([this] { updateControlsGeometry(); }, 0., 1., st::introCoverDuration, wasStep->hasCover() ? anim::linear : anim::easeOutCirc); } _stepLifetime.destroy(); if (action == StackAction::Forward || action == StackAction::Replace) { wasStep->finished(); } if (action == StackAction::Back || action == StackAction::Replace) { delete base::take(wasStep); } _back->toggle(getStep()->hasBack(), anim::type::normal); auto stepHasCover = getStep()->hasCover(); _settings->toggle(!stepHasCover, anim::type::normal); if (_update) { _update->toggle(!stepHasCover, anim::type::normal); } setupNextButton(); if (_resetAccount) _resetAccount->show(anim::type::normal); if (_terms) _terms->show(anim::type::normal); getStep()->showAnimated(animate); fixOrder(); } void Widget::hideAndDestroy(object_ptr> widget) { const auto weak = Ui::MakeWeak(widget.data()); widget->hide(anim::type::normal); widget->shownValue( ) | rpl::start_with_next([=](bool shown) { if (!shown && weak) { weak->deleteLater(); } }, widget->lifetime()); } void Widget::fixOrder() { _next->raise(); if (_update) _update->raise(); if (_changeLanguage) _changeLanguage->raise(); _settings->raise(); _back->raise(); floatPlayerRaiseAll(); _connecting->raise(); } void Widget::moveToStep(Step *step, StackAction action, Animate animate) { appendStep(step); _back->raise(); _settings->raise(); if (_update) { _update->raise(); } _connecting->raise(); historyMove(action, animate); } void Widget::appendStep(Step *step) { _stepHistory.push_back(step); step->setGeometry(rect()); step->setGoCallback([=](Step *step, StackAction action, Animate animate) { if (action == StackAction::Back) { historyMove(action, animate); } else { moveToStep(step, action, animate); } }); step->setShowResetCallback([=] { showResetButton(); }); step->setShowTermsCallback([=] { showTerms(); }); step->setCancelNearestDcCallback([=] { if (_api) { _api->request(base::take(_nearestDcRequestId)).cancel(); } }); step->setAcceptTermsCallback([=](Fn callback) { acceptTerms(callback); }); } void Widget::showResetButton() { if (!_resetAccount) { auto entity = object_ptr( this, tr::lng_signin_reset_account(), st::introResetButton); _resetAccount.create(this, std::move(entity)); _resetAccount->hide(anim::type::instant); _resetAccount->entity()->setClickedCallback([this] { resetAccount(); }); updateControlsGeometry(); } _resetAccount->show(anim::type::normal); if (_changeLanguage) { _changeLanguage->hide(anim::type::normal); } } void Widget::showTerms() { if (getData()->termsLock.text.text.isEmpty()) { _terms.destroy(); } else if (!_terms) { auto entity = object_ptr( this, tr::lng_terms_signup( lt_link, tr::lng_terms_signup_link() | Ui::Text::ToLink(), Ui::Text::WithEntities), st::introTermsLabel); _terms.create(this, std::move(entity)); _terms->entity()->setClickHandlerFilter([=]( const ClickHandlerPtr &handler, Qt::MouseButton button) { if (button == Qt::LeftButton) { showTerms(nullptr); } return false; }); updateControlsGeometry(); _terms->hide(anim::type::instant); } if (_changeLanguage) { _changeLanguage->toggle( !_terms && !_resetAccount && _nextShown, anim::type::normal); } } void Widget::acceptTerms(Fn callback) { showTerms(callback); } void Widget::resetAccount() { if (_resetRequest || !_api) { return; } Ui::show(Box(tr::lng_signin_sure_reset(tr::now), tr::lng_signin_reset(tr::now), st::attentionBoxButton, crl::guard(this, [this] { if (_resetRequest) { return; } _resetRequest = _api->request(MTPaccount_DeleteAccount( MTP_string("Forgot password") )).done([=](const MTPBool &result) { _resetRequest = 0; Ui::hideLayer(); if (getData()->phone.isEmpty()) { moveToStep( new QrWidget(this, _account, getData()), StackAction::Replace, Animate::Back); } else { moveToStep( new SignupWidget(this, _account, getData()), StackAction::Replace, Animate::Forward); } }).fail([=](const RPCError &error) { _resetRequest = 0; const auto &type = error.type(); if (type.startsWith(qstr("2FA_CONFIRM_WAIT_"))) { const auto seconds = type.mid(qstr("2FA_CONFIRM_WAIT_").size()).toInt(); const auto days = (seconds + 59) / 86400; const auto hours = ((seconds + 59) % 86400) / 3600; const auto minutes = ((seconds + 59) % 3600) / 60; auto when = tr::lng_signin_reset_minutes( tr::now, lt_count, minutes); if (days > 0) { const auto daysCount = tr::lng_signin_reset_days( tr::now, lt_count, days); const auto hoursCount = tr::lng_signin_reset_hours( tr::now, lt_count, hours); when = tr::lng_signin_reset_in_days( tr::now, lt_days_count, daysCount, lt_hours_count, hoursCount, lt_minutes_count, when); } else if (hours > 0) { const auto hoursCount = tr::lng_signin_reset_hours( tr::now, lt_count, hours); when = tr::lng_signin_reset_in_hours( tr::now, lt_hours_count, hoursCount, lt_minutes_count, when); } Ui::show(Box(tr::lng_signin_reset_wait( tr::now, lt_phone_number, App::formatPhone(getData()->phone), lt_when, when))); } else if (type == qstr("2FA_RECENT_CONFIRM")) { Ui::show(Box( tr::lng_signin_reset_cancelled(tr::now))); } else { Ui::hideLayer(); getStep()->showError(rpl::single(Lang::Hard::ServerError())); } }).send(); }))); } void Widget::getNearestDC() { if (!_api) { return; } _nearestDcRequestId = _api->request(MTPhelp_GetNearestDc( )).done([=](const MTPNearestDc &result) { _nearestDcRequestId = 0; const auto &nearest = result.c_nearestDc(); DEBUG_LOG(("Got nearest dc, country: %1, nearest: %2, this: %3" ).arg(qs(nearest.vcountry()) ).arg(nearest.vnearest_dc().v ).arg(nearest.vthis_dc().v)); _account->suggestMainDcId(nearest.vnearest_dc().v); const auto nearestCountry = qs(nearest.vcountry()); if (getData()->country != nearestCountry) { getData()->country = nearestCountry; getData()->updated.notify(); } }).send(); } void Widget::showTerms(Fn callback) { if (getData()->termsLock.text.text.isEmpty()) { return; } const auto weak = Ui::MakeWeak(this); const auto box = Ui::show(callback ? Box( getData()->termsLock, tr::lng_terms_agree(), tr::lng_terms_decline()) : Box( getData()->termsLock.text, tr::lng_box_ok(), nullptr)); box->setCloseByEscape(false); box->setCloseByOutsideClick(false); box->agreeClicks( ) | rpl::start_with_next([=] { if (callback) { callback(); } if (box) { box->closeBox(); } }, box->lifetime()); box->cancelClicks( ) | rpl::start_with_next([=] { const auto box = Ui::show(Box( TextWithEntities{ tr::lng_terms_signup_sorry(tr::now) }, tr::lng_intro_finish(), tr::lng_terms_decline())); box->agreeClicks( ) | rpl::start_with_next([=] { if (weak) { showTerms(callback); } }, box->lifetime()); box->cancelClicks( ) | rpl::start_with_next([=] { if (box) { box->closeBox(); } }, box->lifetime()); }, box->lifetime()); } void Widget::showControls() { getStep()->show(); setupNextButton(); _next->toggle(_nextShown, anim::type::instant); _nextShownAnimation.stop(); _connecting->setForceHidden(false); auto hasCover = getStep()->hasCover(); _settings->toggle(!hasCover, anim::type::instant); if (_update) { _update->toggle(!hasCover, anim::type::instant); } if (_changeLanguage) { _changeLanguage->toggle( !_resetAccount && !_terms && _nextShown, anim::type::instant); } if (_terms) { _terms->show(anim::type::instant); } _back->toggle(getStep()->hasBack(), anim::type::instant); } void Widget::setupNextButton() { _next->entity()->setText(getStep()->nextButtonText( ) | rpl::filter([](const QString &text) { return !text.isEmpty(); })); getStep()->nextButtonText( ) | rpl::map([](const QString &text) { return !text.isEmpty(); }) | rpl::filter([=](bool visible) { return visible != _nextShown; }) | rpl::start_with_next([=](bool visible) { _next->toggle(visible, anim::type::normal); _nextShown = visible; if (_changeLanguage) { _changeLanguage->toggle( !_resetAccount && !_terms && _nextShown, anim::type::normal); } _nextShownAnimation.start( [=] { updateControlsGeometry(); }, _nextShown ? 0. : 1., _nextShown ? 1. : 0., st::slideDuration); }, _stepLifetime); } void Widget::hideControls() { getStep()->hide(); _next->hide(anim::type::instant); _connecting->setForceHidden(true); _settings->hide(anim::type::instant); if (_update) _update->hide(anim::type::instant); if (_changeLanguage) _changeLanguage->hide(anim::type::instant); if (_terms) _terms->hide(anim::type::instant); _back->hide(anim::type::instant); } void Widget::showAnimated(const QPixmap &bgAnimCache, bool back) { _showBack = back; (_showBack ? _cacheOver : _cacheUnder) = bgAnimCache; _a_show.stop(); showControls(); floatPlayerHideAll(); (_showBack ? _cacheUnder : _cacheOver) = Ui::GrabWidget(this); hideControls(); floatPlayerShowVisible(); _a_show.start( [=] { animationCallback(); }, 0., 1., st::slideDuration, Window::SlideAnimation::transition()); show(); } void Widget::animationCallback() { update(); if (!_a_show.animating()) { _cacheUnder = _cacheOver = QPixmap(); showControls(); getStep()->activate(); } } void Widget::paintEvent(QPaintEvent *e) { bool trivial = (rect() == e->rect()); setMouseTracking(true); QPainter p(this); if (!trivial) { p.setClipRect(e->rect()); } p.fillRect(e->rect(), st::windowBg); auto progress = _a_show.value(1.); if (_a_show.animating()) { auto coordUnder = _showBack ? anim::interpolate(-st::slideShift, 0, progress) : anim::interpolate(0, -st::slideShift, progress); auto coordOver = _showBack ? anim::interpolate(0, width(), progress) : anim::interpolate(width(), 0, progress); auto shadow = _showBack ? (1. - progress) : progress; if (coordOver > 0) { p.drawPixmap(QRect(0, 0, coordOver, height()), _cacheUnder, QRect(-coordUnder * cRetinaFactor(), 0, coordOver * cRetinaFactor(), height() * cRetinaFactor())); p.setOpacity(shadow); p.fillRect(0, 0, coordOver, height(), st::slideFadeOutBg); p.setOpacity(1); } p.drawPixmap(coordOver, 0, _cacheOver); p.setOpacity(shadow); st::slideShadow.fill(p, QRect(coordOver - st::slideShadow.width(), 0, st::slideShadow.width(), height())); } } void Widget::resizeEvent(QResizeEvent *e) { if (_stepHistory.empty()) { return; } for (const auto step : _stepHistory) { step->setGeometry(rect()); } updateControlsGeometry(); floatPlayerAreaUpdated(); } void Widget::updateControlsGeometry() { auto shown = _coverShownAnimation.value(1.); auto controlsTopTo = getStep()->hasCover() ? st::introCoverHeight : 0; auto controlsTop = anim::interpolate(_controlsTopFrom, controlsTopTo, shown); _settings->moveToRight(st::introSettingsSkip, controlsTop + st::introSettingsSkip); if (_update) { _update->moveToRight(st::introSettingsSkip + _settings->width() + st::introSettingsSkip, _settings->y()); } _back->moveToLeft(0, controlsTop); auto nextTopTo = getStep()->contentTop() + st::introNextTop; auto nextTop = anim::interpolate(_nextTopFrom, nextTopTo, shown); const auto shownAmount = _nextShownAnimation.value(_nextShown ? 1. : 0.); const auto realNextTop = anim::interpolate( nextTop + st::introNextSlide, nextTop, shownAmount); _next->moveToLeft((width() - _next->width()) / 2, realNextTop); getStep()->setShowAnimationClipping(shownAmount > 0 ? QRect(0, 0, width(), realNextTop) : QRect()); if (_changeLanguage) { _changeLanguage->moveToLeft((width() - _changeLanguage->width()) / 2, _next->y() + _next->height() + _changeLanguage->height()); } if (_resetAccount) { _resetAccount->moveToLeft((width() - _resetAccount->width()) / 2, height() - st::introResetBottom - _resetAccount->height()); } if (_terms) { _terms->moveToLeft((width() - _terms->width()) / 2, height() - st::introTermsBottom - _terms->height()); } } void Widget::keyPressEvent(QKeyEvent *e) { if (_a_show.animating() || getStep()->animating()) return; if (e->key() == Qt::Key_Escape || e->key() == Qt::Key_Back) { if (getStep()->hasBack()) { backRequested(); } } else if (e->key() == Qt::Key_Enter || e->key() == Qt::Key_Return || e->key() == Qt::Key_Space) { getStep()->submit(); } } void Widget::backRequested() { if (_stepHistory.size() > 1) { historyMove(StackAction::Back, Animate::Back); } else if (const auto parent = Core::App().domain().maybeLastOrSomeAuthedAccount()) { Core::App().domain().activate(parent); } else { moveToStep( new StartWidget(this, _account, getData()), StackAction::Replace, Animate::Back); } } Widget::~Widget() { for (auto step : base::take(_stepHistory)) { delete step; } } } // namespace Intro