From c76f604204609a7e5e4e7c809c07441a6723707d Mon Sep 17 00:00:00 2001 From: vng Date: Mon, 26 Apr 2021 20:45:58 +0300 Subject: [PATCH] [desktop] Fixed downloader progress. Signed-off-by: vng --- map/framework.cpp | 4 +- map/framework.hpp | 2 +- qt/draw_widget.cpp | 54 -------------------- qt/draw_widget.hpp | 13 ----- qt/mainwindow.cpp | 120 ++++++++++++++++++++++++-------------------- qt/mainwindow.hpp | 2 +- storage/storage.cpp | 14 ++---- storage/storage.hpp | 2 +- 8 files changed, 74 insertions(+), 137 deletions(-) diff --git a/map/framework.cpp b/map/framework.cpp index 06cc438227..fa9a4cfce5 100644 --- a/map/framework.cpp +++ b/map/framework.cpp @@ -1144,9 +1144,9 @@ void Framework::OnUpdateCurrentCountry(m2::PointD const & pt, int zoomLevel) }); } -void Framework::SetCurrentCountryChangedListener(TCurrentCountryChanged const & listener) +void Framework::SetCurrentCountryChangedListener(TCurrentCountryChanged listener) { - m_currentCountryChanged = listener; + m_currentCountryChanged = std::move(listener); m_lastReportedCountry = kInvalidCountryId; } diff --git a/map/framework.hpp b/map/framework.hpp index bce66be2ed..c7de352efd 100644 --- a/map/framework.hpp +++ b/map/framework.hpp @@ -359,7 +359,7 @@ public: using TCurrentCountryChanged = std::function; storage::CountryId const & GetLastReportedCountry() { return m_lastReportedCountry; } /// Guarantees that listener is called in the main thread context. - void SetCurrentCountryChangedListener(TCurrentCountryChanged const & listener); + void SetCurrentCountryChangedListener(TCurrentCountryChanged listener); std::vector GetRegionsCountryIdByRect(m2::RectD const & rect, bool rough) const; std::vector GetMwmsByRect(m2::RectD const & rect, bool rough) const; diff --git a/qt/draw_widget.cpp b/qt/draw_widget.cpp index 60284245eb..42c9041679 100644 --- a/qt/draw_widget.cpp +++ b/qt/draw_widget.cpp @@ -108,20 +108,11 @@ DrawWidget::DrawWidget(Framework & framework, bool apiOpenGLES3, std::unique_ptr routingManager.SetRouteRecommendationListener( [this](RoutingManager::Recommendation r) { OnRouteRecommendation(r); }); - m_framework.SetCurrentCountryChangedListener([this](storage::CountryId const & countryId) { - m_countryId = countryId; - UpdateCountryStatus(countryId); - }); - if (screenshotParams != nullptr) { m_ratio = screenshotParams->m_dpiScale; m_screenshoter = std::make_unique(*screenshotParams, m_framework, this); } - QTimer * countryStatusTimer = new QTimer(this); - VERIFY(connect(countryStatusTimer, SIGNAL(timeout()), this, SLOT(OnUpdateCountryStatusByTimer())), ()); - countryStatusTimer->setSingleShot(false); - countryStatusTimer->start(1000); } DrawWidget::~DrawWidget() @@ -129,51 +120,6 @@ DrawWidget::~DrawWidget() delete m_rubberBand; } -void DrawWidget::UpdateCountryStatus(storage::CountryId const & countryId) -{ - if (m_currentCountryChanged != nullptr) - { - std::string countryName = countryId; - auto status = m_framework.GetStorage().CountryStatusEx(countryId); - - uint8_t percentage = 0; - downloader::Progress progress; - if (!countryId.empty()) - { - storage::NodeAttrs nodeAttrs; - m_framework.GetStorage().GetNodeAttrs(countryId, nodeAttrs); - progress = nodeAttrs.m_downloadingProgress; - if (!progress.IsUnknown() && progress.m_bytesTotal != 0) - percentage = static_cast(100 * progress.m_bytesDownloaded / progress.m_bytesTotal); - } - - m_currentCountryChanged(countryId, countryName, status, progress.m_bytesTotal, percentage); - } -} - -void DrawWidget::OnUpdateCountryStatusByTimer() -{ - if (!m_countryId.empty()) - UpdateCountryStatus(m_countryId); -} - -void DrawWidget::SetCurrentCountryChangedListener(DrawWidget::TCurrentCountryChanged const & listener) -{ - m_currentCountryChanged = listener; -} - -void DrawWidget::DownloadCountry(storage::CountryId const & countryId) -{ - m_framework.GetStorage().DownloadNode(countryId); - if (!m_countryId.empty()) - UpdateCountryStatus(m_countryId); -} - -void DrawWidget::RetryToDownloadCountry(storage::CountryId const & countryId) -{ - // TODO @bykoianko -} - void DrawWidget::PrepareShutdown() { auto & routingManager = m_framework.GetRoutingManager(); diff --git a/qt/draw_widget.hpp b/qt/draw_widget.hpp index 8276995284..14e1a2dec4 100644 --- a/qt/draw_widget.hpp +++ b/qt/draw_widget.hpp @@ -47,7 +47,6 @@ public Q_SLOTS: void ChoosePositionModeEnable(); void ChoosePositionModeDisable(); - void OnUpdateCountryStatusByTimer(); public: DrawWidget(Framework & framework, bool apiOpenGLES3, std::unique_ptr && screenshotParams, @@ -72,13 +71,6 @@ public: void SetRuler(bool enabled); - using TCurrentCountryChanged = std::function; - void SetCurrentCountryChangedListener(TCurrentCountryChanged const & listener); - - void DownloadCountry(storage::CountryId const & countryId); - void RetryToDownloadCountry(storage::CountryId const & countryId); - RouteMarkType GetRoutePointAddMode() const { return m_routePointAddMode; } void SetRoutePointAddMode(RouteMarkType mode) { m_routePointAddMode = mode; } void FollowRoute(); @@ -109,8 +101,6 @@ private: void SubmitBookmark(m2::PointD const & pt); void ShowPlacePage(); - void UpdateCountryStatus(storage::CountryId const & countryId); - void VisualizeMwmsBordersInRect(m2::RectD const & rect, bool withVertices, bool fromPackedPolygon, bool boundingBox); @@ -121,9 +111,6 @@ private: bool m_emulatingLocation; - TCurrentCountryChanged m_currentCountryChanged; - storage::CountryId m_countryId; - public: enum class SelectionMode { diff --git a/qt/mainwindow.cpp b/qt/mainwindow.cpp index 6ea786d101..0658db9978 100644 --- a/qt/mainwindow.cpp +++ b/qt/mainwindow.cpp @@ -553,6 +553,11 @@ void MainWindow::CreateNavigationBar() addToolBar(Qt::RightToolBarArea, pToolBar); } +Framework & MainWindow::GetFramework() const +{ + return m_pDrawWidget->GetFramework(); +} + void MainWindow::CreateCountryStatusControls() { QHBoxLayout * mainLayout = new QHBoxLayout(); @@ -572,64 +577,69 @@ void MainWindow::CreateCountryStatusControls() m_pDrawWidget->setLayout(mainLayout); - m_pDrawWidget->SetCurrentCountryChangedListener([this](storage::CountryId const & countryId, - std::string const & countryName, - storage::Status status, - uint64_t sizeInBytes, uint8_t progress) { + auto const OnCountryChanged = [this](storage::CountryId const & countryId) + { + m_downloadButton->setVisible(false); + m_retryButton->setVisible(false); + m_downloadingStatusLabel->setVisible(false); + m_lastCountry = countryId; - if (m_lastCountry.empty() || status == storage::Status::OnDisk || status == storage::Status::OnDiskOutOfDate) + // Called by Framework in World zoom level. + if (countryId.empty()) + return; + + auto const & storage = GetFramework().GetStorage(); + auto status = storage.CountryStatusEx(countryId); + auto const & countryName = countryId; + + if (status == storage::Status::NotDownloaded) { - m_downloadButton->setVisible(false); - m_retryButton->setVisible(false); - m_downloadingStatusLabel->setVisible(false); + m_downloadButton->setVisible(true); + + std::string units; + size_t sizeToDownload = 0; + FormatMapSize(storage.CountrySizeInBytes(countryId).second, units, sizeToDownload); + std::stringstream str; + str << "Download (" << countryName << ") " << sizeToDownload << units; + m_downloadButton->setText(str.str().c_str()); } - else + else if (status == storage::Status::Downloading) { - if (status == storage::Status::NotDownloaded) - { - m_downloadButton->setVisible(true); - m_retryButton->setVisible(false); - m_downloadingStatusLabel->setVisible(false); - - std::string units; - size_t sizeToDownload = 0; - FormatMapSize(sizeInBytes, units, sizeToDownload); - std::stringstream str; - str << "Download (" << countryName << ") " << sizeToDownload << units; - m_downloadButton->setText(str.str().c_str()); - } - else if (status == storage::Status::Downloading) - { - m_downloadButton->setVisible(false); - m_retryButton->setVisible(false); - m_downloadingStatusLabel->setVisible(true); - - std::stringstream str; - str << "Downloading (" << countryName << ") " << (int)progress << "%"; - m_downloadingStatusLabel->setText(str.str().c_str()); - } - else if (status == storage::Status::InQueue) - { - m_downloadButton->setVisible(false); - m_retryButton->setVisible(false); - m_downloadingStatusLabel->setVisible(true); - - std::stringstream str; - str << countryName << " is waiting for downloading"; - m_downloadingStatusLabel->setText(str.str().c_str()); - } - else - { - m_downloadButton->setVisible(false); - m_retryButton->setVisible(true); - m_downloadingStatusLabel->setVisible(false); - - std::stringstream str; - str << "Retry to download " << countryName; - m_retryButton->setText(str.str().c_str()); - } + m_downloadingStatusLabel->setVisible(true); } - }); + else if (status == storage::Status::InQueue) + { + m_downloadingStatusLabel->setVisible(true); + + std::stringstream str; + str << countryName << " is waiting for downloading"; + m_downloadingStatusLabel->setText(str.str().c_str()); + } + else if (status != storage::Status::OnDisk && status != storage::Status::OnDiskOutOfDate) + { + m_retryButton->setVisible(true); + + std::stringstream str; + str << "Retry to download " << countryName; + m_retryButton->setText(str.str().c_str()); + } + }; + + GetFramework().SetCurrentCountryChangedListener(OnCountryChanged); + + GetFramework().GetStorage().Subscribe( + [this, onChanged = std::move(OnCountryChanged)](storage::CountryId const & countryId) + { + // Storage also calls notifications for parents, but we are interested in leafs only. + if (GetFramework().GetStorage().IsLeaf(countryId)) + onChanged(countryId); + }, + [this](storage::CountryId const & countryId, downloader::Progress const & progress) + { + std::stringstream str; + str << "Downloading (" << countryId << ") " << progress.m_bytesDownloaded * 100 / progress.m_bytesTotal << "%"; + m_downloadingStatusLabel->setText(str.str().c_str()); + }); } void MainWindow::OnAbout() @@ -975,12 +985,12 @@ void MainWindow::closeEvent(QCloseEvent * e) void MainWindow::OnDownloadClicked() { - m_pDrawWidget->DownloadCountry(m_lastCountry); + GetFramework().GetStorage().DownloadNode(m_lastCountry); } void MainWindow::OnRetryDownloadClicked() { - m_pDrawWidget->RetryToDownloadCountry(m_lastCountry); + GetFramework().GetStorage().RetryDownloadNode(m_lastCountry); } void MainWindow::SetEnabledTraffic(bool enable) diff --git a/qt/mainwindow.hpp b/qt/mainwindow.hpp index 6695739118..cadbe46815 100644 --- a/qt/mainwindow.hpp +++ b/qt/mainwindow.hpp @@ -80,7 +80,7 @@ public: static void SetDefaultSurfaceFormat(bool apiOpenGLES3); protected: - std::string GetIniFile(); + Framework & GetFramework() const; void OnLocationError(location::TLocationError errorCode) override; void OnLocationUpdated(location::GpsInfo const & info) override; diff --git a/storage/storage.cpp b/storage/storage.cpp index 650232e071..363cea06e4 100644 --- a/storage/storage.cpp +++ b/storage/storage.cpp @@ -588,19 +588,13 @@ void Storage::LoadCountriesFile(string const & pathToCountriesFile) } } -int Storage::Subscribe(ChangeCountryFunction const & change, ProgressFunction const & progress) +int Storage::Subscribe(ChangeCountryFunction change, ProgressFunction progress) { CHECK_THREAD_CHECKER(m_threadChecker, ()); - CountryObservers obs; - - obs.m_changeCountryFn = change; - obs.m_progressFn = progress; - obs.m_slotId = ++m_currentSlotId; - - m_observers.push_back(obs); - - return obs.m_slotId; + int const id = ++m_currentSlotId; + m_observers.push_back({ std::move(change), std::move(progress), id }); + return id; } void Storage::Unsubscribe(int slotId) diff --git a/storage/storage.hpp b/storage/storage.hpp index 5320a100a0..02dcca1578 100644 --- a/storage/storage.hpp +++ b/storage/storage.hpp @@ -498,7 +498,7 @@ public: /// Guarantees that change and progress are called in the main thread context. /// @return unique identifier that should be used with Unsubscribe function - int Subscribe(ChangeCountryFunction const & change, ProgressFunction const & progress); + int Subscribe(ChangeCountryFunction change, ProgressFunction progress); void Unsubscribe(int slotId); /// Returns information about selected counties downloading progress.