diff --git a/Telegram/Resources/langs/lang.strings b/Telegram/Resources/langs/lang.strings index f46ba6968..3b9d28d95 100644 --- a/Telegram/Resources/langs/lang.strings +++ b/Telegram/Resources/langs/lang.strings @@ -436,7 +436,9 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL "lng_local_storage_round#other" = "{count} video messages"; "lng_local_storage_animation#one" = "{count} animation"; "lng_local_storage_animation#other" = "{count} animations"; +"lng_local_storage_media" = "Media cache"; "lng_local_storage_size_limit" = "Total size limit: {size}"; +"lng_local_storage_media_limit" = "Media cache limit: {size}"; "lng_local_storage_time_limit" = "Clear files older than: {limit}"; "lng_local_storage_limit_weeks#one" = "{count} week"; "lng_local_storage_limit_weeks#other" = "{count} weeks"; diff --git a/Telegram/SourceFiles/boxes/local_storage_box.cpp b/Telegram/SourceFiles/boxes/local_storage_box.cpp index 1db946a0f..22192a63d 100644 --- a/Telegram/SourceFiles/boxes/local_storage_box.cpp +++ b/Telegram/SourceFiles/boxes/local_storage_box.cpp @@ -26,19 +26,34 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL namespace { -constexpr auto kSizeLimitsCount = 20; +constexpr auto kMegabyte = int64(1024 * 1024); +constexpr auto kTotalSizeLimitsCount = 18; +constexpr auto kMediaSizeLimitsCount = 18; +constexpr auto kMinimalSizeLimit = 100 * kMegabyte; constexpr auto kTimeLimitsCount = 16; constexpr auto kMaxTimeLimitValue = std::numeric_limits::max(); +constexpr auto kFakeMediaCacheTag = uint16(0xFFFF); -int64 SizeLimitInMB(int index) { - if (index < 10) { - return int64(index + 1) * 100; +int64 TotalSizeLimitInMB(int index) { + if (index < 8) { + return int64(index + 2) * 100; } - return int64(index - 9) * 1024; + return int64(index - 7) * 1024; } -int64 SizeLimit(int index) { - return SizeLimitInMB(index) * 1024 * 1024; +int64 TotalSizeLimit(int index) { + return TotalSizeLimitInMB(index) * kMegabyte; +} + +int64 MediaSizeLimitInMB(int index) { + if (index < 9) { + return int64(index + 1) * 100; + } + return int64(index - 8) * 1024; +} + +int64 MediaSizeLimit(int index) { + return MediaSizeLimitInMB(index) * kMegabyte; } QString SizeLimitText(int64 limit) { @@ -255,20 +270,30 @@ QString LocalStorageBox::Row::sizeText(const Database::TaggedSummary &data) cons LocalStorageBox::LocalStorageBox( QWidget*, not_null db, + not_null dbBig, CreateTag) -: _db(db) { +: _db(db) +, _dbBig(dbBig) { const auto &settings = Local::cacheSettings(); - _sizeLimit = settings.totalSizeLimit; + const auto &settingsBig = Local::cacheBigFileSettings(); + _totalSizeLimit = settings.totalSizeLimit + settingsBig.totalSizeLimit; + _mediaSizeLimit = settingsBig.totalSizeLimit; _timeLimit = settings.totalTimeLimit; } -void LocalStorageBox::Show(not_null db) { +void LocalStorageBox::Show( + not_null db, + not_null dbBig) { auto shared = std::make_shared>( - Box(db, CreateTag())); + Box(db, dbBig, CreateTag())); const auto weak = shared->data(); - db->statsOnMain( - ) | rpl::start_with_next([=](Database::Stats &&stats) { - weak->update(std::move(stats)); + rpl::combine( + db->statsOnMain(), + dbBig->statsOnMain() + ) | rpl::start_with_next([=]( + Database::Stats &&stats, + Database::Stats &&statsBig) { + weak->update(std::move(stats), std::move(statsBig)); if (auto &strong = *shared) { Ui::show(std::move(strong)); } @@ -285,7 +310,7 @@ void LocalStorageBox::prepare() { void LocalStorageBox::updateRow( not_null*> row, - Database::TaggedSummary *data) { + const Database::TaggedSummary *data) { const auto summary = (_rows.find(0)->second == row); const auto shown = (data && data->count && data->totalSize) || summary; if (shown) { @@ -294,28 +319,45 @@ void LocalStorageBox::updateRow( row->toggle(shown, anim::type::normal); } -void LocalStorageBox::update(Database::Stats &&stats) { +void LocalStorageBox::update( + Database::Stats &&stats, + Database::Stats &&statsBig) { _stats = std::move(stats); + _statsBig = std::move(statsBig); if (const auto i = _rows.find(0); i != end(_rows)) { - i->second->entity()->toggleProgress(_stats.clearing); + i->second->entity()->toggleProgress( + _stats.clearing || _statsBig.clearing); } for (const auto &entry : _rows) { - if (entry.first) { + if (entry.first == kFakeMediaCacheTag) { + updateRow(entry.second, &_statsBig.full); + } else if (entry.first) { const auto i = _stats.tagged.find(entry.first); updateRow( entry.second, (i != end(_stats.tagged)) ? &i->second : nullptr); } else { - updateRow(entry.second, &_stats.full); + const auto full = summary(); + updateRow(entry.second, &full); } } } -void LocalStorageBox::clearByTag(uint8 tag) { - if (tag) { +auto LocalStorageBox::summary() const -> Database::TaggedSummary { + auto result = _stats.full; + result.count += _statsBig.full.count; + result.totalSize += _statsBig.full.totalSize; + return result; +} + +void LocalStorageBox::clearByTag(uint16 tag) { + if (tag == kFakeMediaCacheTag) { + _dbBig->clear(); + } else if (tag) { _db->clearByTag(tag); } else { _db->clear(); + _dbBig->clear(); Ui::Emoji::ClearIrrelevantCache(); } } @@ -325,7 +367,7 @@ void LocalStorageBox::setupControls() { object_ptr(this), st::contactsMultiSelect.scroll); const auto createRow = [&]( - uint8 tag, + uint16 tag, Fn title, Fn clear, const Database::TaggedSummary &data) { @@ -363,11 +405,14 @@ void LocalStorageBox::setupControls() { auto summaryTitle = [](size_type) { return lang(lng_local_storage_summary); }; + auto mediaCacheTitle = [](size_type) { + return lang(lng_local_storage_media); + }; createRow( 0, std::move(summaryTitle), langFactory(lng_local_storage_clear), - _stats.full); + summary()); setupLimits(container); const auto shadow = container->add(object_ptr>( container, @@ -378,6 +423,11 @@ void LocalStorageBox::setupControls() { createTagRow(Data::kVoiceMessageCacheTag, lng_local_storage_voice); createTagRow(Data::kVideoMessageCacheTag, lng_local_storage_round); createTagRow(Data::kAnimationCacheTag, lng_local_storage_animation); + tracker.track(createRow( + kFakeMediaCacheTag, + std::move(mediaCacheTitle), + langFactory(lng_local_storage_clear_some), + _statsBig.full)); shadow->toggleOn( std::move(tracker).atLeastOneShownValue() ); @@ -393,7 +443,7 @@ template < typename Convert, typename Callback, typename> -void LocalStorageBox::createLimitsSlider( +not_null LocalStorageBox::createLimitsSlider( not_null container, int valuesCount, Convert &&convert, @@ -414,6 +464,60 @@ void LocalStorageBox::createLimitsSlider( [=, callback = std::forward(callback)](Value value) { callback(label, value); }); + return slider; +} + +void LocalStorageBox::updateMediaLimit() { + const auto good = [&](int64 mediaLimit) { + return (_totalSizeLimit - mediaLimit >= kMinimalSizeLimit); + }; + if (good(_mediaSizeLimit) || !_mediaSlider || !_mediaLabel) { + return; + } + auto index = 1; + while ((index < kMediaSizeLimitsCount) + && (MediaSizeLimit(index) * 2 <= _totalSizeLimit)) { + ++index; + } + --index; + _mediaSizeLimit = MediaSizeLimit(index); + _mediaSlider->setValue(index / float64(kMediaSizeLimitsCount - 1)); + updateMediaLabel(); + + Ensures(good(_mediaSizeLimit)); +} + +void LocalStorageBox::updateTotalLimit() { + const auto good = [&](int64 totalLimit) { + return (totalLimit - _mediaSizeLimit >= kMinimalSizeLimit); + }; + if (good(_totalSizeLimit) || !_totalSlider || !_totalLabel) { + return; + } + auto index = kTotalSizeLimitsCount - 1; + while ((index > 0) + && (TotalSizeLimit(index - 1) >= 2 * _mediaSizeLimit)) { + --index; + } + _totalSizeLimit = TotalSizeLimit(index); + _totalSlider->setValue(index / float64(kTotalSizeLimitsCount - 1)); + updateTotalLabel(); + + Ensures(good(_totalSizeLimit)); +} + +void LocalStorageBox::updateTotalLabel() { + Expects(_totalLabel != nullptr); + + const auto text = SizeLimitText(_totalSizeLimit); + _totalLabel->setText(lng_local_storage_size_limit(lt_size, text)); +} + +void LocalStorageBox::updateMediaLabel() { + Expects(_mediaLabel != nullptr); + + const auto text = SizeLimitText(_mediaSizeLimit); + _mediaLabel->setText(lng_local_storage_media_limit(lt_size, text)); } void LocalStorageBox::setupLimits(not_null container) { @@ -421,15 +525,29 @@ void LocalStorageBox::setupLimits(not_null container) { object_ptr(container), st::localStorageRowPadding); - createLimitsSlider( + _totalSlider = createLimitsSlider( container, - kSizeLimitsCount, - SizeLimit, - _sizeLimit, + kTotalSizeLimitsCount, + TotalSizeLimit, + _totalSizeLimit, [=](not_null label, int64 limit) { - const auto text = SizeLimitText(limit); - label->setText(lng_local_storage_size_limit(lt_size, text)); - _sizeLimit = limit; + _totalSizeLimit = limit; + _totalLabel = label; + updateTotalLabel(); + updateMediaLimit(); + limitsChanged(); + }); + + _mediaSlider = createLimitsSlider( + container, + kMediaSizeLimitsCount, + MediaSizeLimit, + _mediaSizeLimit, + [=](not_null label, int64 limit) { + _mediaSizeLimit = limit; + _mediaLabel = label; + updateMediaLabel(); + updateTotalLimit(); limitsChanged(); }); @@ -448,8 +566,12 @@ void LocalStorageBox::setupLimits(not_null container) { void LocalStorageBox::limitsChanged() { const auto &settings = Local::cacheSettings(); - const auto changed = (settings.totalSizeLimit != _sizeLimit) - || (settings.totalTimeLimit != _timeLimit); + const auto &settingsBig = Local::cacheBigFileSettings(); + const auto sizeLimit = _totalSizeLimit - _mediaSizeLimit; + const auto changed = (settings.totalSizeLimit != sizeLimit) + || (settingsBig.totalSizeLimit != _mediaSizeLimit) + || (settings.totalTimeLimit != _timeLimit) + || (settingsBig.totalTimeLimit != _timeLimit); if (_limitsChanged != changed) { _limitsChanged = changed; clearButtons(); @@ -468,10 +590,12 @@ void LocalStorageBox::save() { return; } auto update = Storage::Cache::Database::SettingsUpdate(); - update.totalSizeLimit = _sizeLimit; + update.totalSizeLimit = _totalSizeLimit - _mediaSizeLimit; update.totalTimeLimit = _timeLimit; - Local::updateCacheSettings(update); - Local::updateCacheBigFileSettings(update); + auto updateBig = Storage::Cache::Database::SettingsUpdate(); + updateBig.totalSizeLimit = _mediaSizeLimit; + update.totalTimeLimit = _timeLimit; + Local::updateCacheSettings(update, updateBig); Auth().data().cache().updateSettings(update); closeBox(); } diff --git a/Telegram/SourceFiles/boxes/local_storage_box.h b/Telegram/SourceFiles/boxes/local_storage_box.h index e3c36377c..5a9bc4e30 100644 --- a/Telegram/SourceFiles/boxes/local_storage_box.h +++ b/Telegram/SourceFiles/boxes/local_storage_box.h @@ -21,6 +21,7 @@ class VerticalLayout; template class SlideWrap; class LabelSimple; +class MediaSlider; } // namespace Ui class LocalStorageBox : public BoxContent { @@ -30,9 +31,13 @@ class LocalStorageBox : public BoxContent { public: using Database = Storage::Cache::Database; - LocalStorageBox(QWidget*, not_null db, CreateTag); + LocalStorageBox( + QWidget*, + not_null db, + not_null dbBig, + CreateTag); - static void Show(not_null db); + static void Show(not_null db, not_null dbBig); protected: void prepare() override; @@ -42,16 +47,22 @@ protected: private: class Row; - void clearByTag(uint8 tag); - void update(Database::Stats &&stats); + void clearByTag(uint16 tag); + void update(Database::Stats &&stats, Database::Stats &&statsBig); void updateRow( not_null*> row, - Database::TaggedSummary *data); + const Database::TaggedSummary *data); void setupControls(); void setupLimits(not_null container); + void updateMediaLimit(); + void updateTotalLimit(); + void updateTotalLabel(); + void updateMediaLabel(); void limitsChanged(); void save(); + Database::TaggedSummary summary() const; + template < typename Value, typename Convert, @@ -62,7 +73,7 @@ private: not_null, Value> && std::is_same_v()(1))>>> - void createLimitsSlider( + not_null createLimitsSlider( not_null container, int valuesCount, Convert &&convert, @@ -70,11 +81,18 @@ private: Callback &&callback); not_null _db; + not_null _dbBig; Database::Stats _stats; + Database::Stats _statsBig; - base::flat_map*>> _rows; + base::flat_map*>> _rows; + Ui::MediaSlider *_totalSlider = nullptr; + Ui::LabelSimple *_totalLabel = nullptr; + Ui::MediaSlider *_mediaSlider = nullptr; + Ui::LabelSimple *_mediaLabel = nullptr; - int64 _sizeLimit = 0; + int64 _totalSizeLimit = 0; + int64 _mediaSizeLimit = 0; size_type _timeLimit = 0; bool _limitsChanged = false; diff --git a/Telegram/SourceFiles/settings/settings_chat.cpp b/Telegram/SourceFiles/settings/settings_chat.cpp index c9d159ee8..32081b714 100644 --- a/Telegram/SourceFiles/settings/settings_chat.cpp +++ b/Telegram/SourceFiles/settings/settings_chat.cpp @@ -570,7 +570,9 @@ void SetupLocalStorage(not_null container) { lng_settings_manage_local_storage, st::settingsButton )->addClickHandler([] { - LocalStorageBox::Show(&Auth().data().cache()); + LocalStorageBox::Show( + &Auth().data().cache(), + &Auth().data().cacheBigFile()); }); } diff --git a/Telegram/SourceFiles/storage/localstorage.cpp b/Telegram/SourceFiles/storage/localstorage.cpp index 792ca7e24..626ed6297 100644 --- a/Telegram/SourceFiles/storage/localstorage.cpp +++ b/Telegram/SourceFiles/storage/localstorage.cpp @@ -602,12 +602,13 @@ enum { dbiTxtDomainString = 0x53, dbiThemeKey = 0x54, dbiTileBackground = 0x55, - dbiCacheSettings = 0x56, + dbiCacheSettingsOld = 0x56, dbiAnimationsDisabled = 0x57, dbiScalePercent = 0x58, dbiPlaybackSpeed = 0x59, dbiLanguagesKey = 0x5a, dbiCallSettings = 0x5b, + dbiCacheSettings = 0x5c, dbiEncryptedWithSalt = 333, dbiEncrypted = 444, @@ -1071,7 +1072,7 @@ bool _readSetting(quint32 blockId, QDataStream &stream, int version, ReadSetting cSetUseExternalVideoPlayer(v == 1); } break; - case dbiCacheSettings: { + case dbiCacheSettingsOld: { qint64 size; qint32 time; stream >> size >> time; @@ -1080,11 +1081,28 @@ bool _readSetting(quint32 blockId, QDataStream &stream, int version, ReadSetting || time < 0) { return false; } - _cacheTotalSizeLimit = size; _cacheTotalTimeLimit = time; _cacheBigFileTotalSizeLimit = size; - _cacheBigFileTotalTimeLimit = size; + _cacheBigFileTotalTimeLimit = time; + } break; + + case dbiCacheSettings: { + qint64 size, sizeBig; + qint32 time, timeBig; + stream >> size >> time >> sizeBig >> timeBig; + if (!_checkStreamStatus(stream) + || size <= Database::Settings().maxDataSize + || sizeBig <= Database::Settings().maxDataSize + || time < 0 + || timeBig < 0) { + return false; + } + + _cacheTotalSizeLimit = size; + _cacheTotalTimeLimit = time; + _cacheBigFileTotalSizeLimit = sizeBig; + _cacheBigFileTotalTimeLimit = timeBig; } break; case dbiAnimationsDisabled: { @@ -2108,8 +2126,7 @@ void _writeUserSettings() { data.stream << quint32(dbiModerateMode) << qint32(Global::ModerateModeEnabled() ? 1 : 0); data.stream << quint32(dbiAutoPlay) << qint32(cAutoPlayGif() ? 1 : 0); data.stream << quint32(dbiUseExternalVideoPlayer) << qint32(cUseExternalVideoPlayer()); - data.stream << quint32(dbiCacheSettings) << qint64(_cacheTotalSizeLimit) << qint32(_cacheTotalTimeLimit); - // #TODO streaming save _cacheBigFileTotal limits?.. + data.stream << quint32(dbiCacheSettings) << qint64(_cacheTotalSizeLimit) << qint32(_cacheTotalTimeLimit) << qint64(_cacheBigFileTotalSizeLimit) << qint32(_cacheBigFileTotalTimeLimit); if (!userData.isEmpty()) { data.stream << quint32(dbiAuthSessionSettings) << userData; } @@ -3208,16 +3225,24 @@ Storage::Cache::Database::Settings cacheSettings() { return result; } -void updateCacheSettings(Storage::Cache::Database::SettingsUpdate &update) { +void updateCacheSettings( + Storage::Cache::Database::SettingsUpdate &update, + Storage::Cache::Database::SettingsUpdate &updateBig) { Expects(update.totalSizeLimit > Database::Settings().maxDataSize); Expects(update.totalTimeLimit >= 0); + Expects(updateBig.totalSizeLimit > Database::Settings().maxDataSize); + Expects(updateBig.totalTimeLimit >= 0); if (_cacheTotalSizeLimit == update.totalSizeLimit - && _cacheTotalTimeLimit == update.totalTimeLimit) { + && _cacheTotalTimeLimit == update.totalTimeLimit + && _cacheBigFileTotalSizeLimit == updateBig.totalSizeLimit + && _cacheBigFileTotalTimeLimit == updateBig.totalTimeLimit) { return; } _cacheTotalSizeLimit = update.totalSizeLimit; _cacheTotalTimeLimit = update.totalTimeLimit; + _cacheBigFileTotalSizeLimit = updateBig.totalSizeLimit; + _cacheBigFileTotalTimeLimit = updateBig.totalTimeLimit; _writeUserSettings(); } @@ -3236,19 +3261,6 @@ Storage::Cache::Database::Settings cacheBigFileSettings() { return result; } -void updateCacheBigFileSettings(Storage::Cache::Database::SettingsUpdate &update) { - Expects(update.totalSizeLimit > Database::Settings().maxDataSize); - Expects(update.totalTimeLimit >= 0); - - if (_cacheBigFileTotalSizeLimit == update.totalSizeLimit - && _cacheBigFileTotalTimeLimit == update.totalTimeLimit) { - return; - } - _cacheBigFileTotalSizeLimit = update.totalSizeLimit; - _cacheBigFileTotalTimeLimit = update.totalTimeLimit; - //_writeUserSettings(); // #TODO streaming save those?.. -} - class CountWaveformTask : public Task { public: CountWaveformTask(DocumentData *doc) diff --git a/Telegram/SourceFiles/storage/localstorage.h b/Telegram/SourceFiles/storage/localstorage.h index 0f50d24ed..7a3d73aea 100644 --- a/Telegram/SourceFiles/storage/localstorage.h +++ b/Telegram/SourceFiles/storage/localstorage.h @@ -115,12 +115,13 @@ FileLocation readFileLocation(MediaKey location, bool check = true); Storage::EncryptionKey cacheKey(); QString cachePath(); Storage::Cache::Database::Settings cacheSettings(); -void updateCacheSettings(Storage::Cache::Database::SettingsUpdate &update); +void updateCacheSettings( + Storage::Cache::Database::SettingsUpdate &update, + Storage::Cache::Database::SettingsUpdate &updateBig); Storage::EncryptionKey cacheBigFileKey(); QString cacheBigFilePath(); Storage::Cache::Database::Settings cacheBigFileSettings(); -void updateCacheBigFileSettings(Storage::Cache::Database::SettingsUpdate &update); void countVoiceWaveform(DocumentData *document); diff --git a/Telegram/SourceFiles/ui/widgets/continuous_sliders.h b/Telegram/SourceFiles/ui/widgets/continuous_sliders.h index b09a759c0..079940ef9 100644 --- a/Telegram/SourceFiles/ui/widgets/continuous_sliders.h +++ b/Telegram/SourceFiles/ui/widgets/continuous_sliders.h @@ -152,6 +152,7 @@ public: setDirection(Ui::ContinuousSlider::Direction::Horizontal); const auto sectionsCount = (valuesCount - 1); + setValue(1.); for (auto index = index_type(); index != valuesCount; ++index) { if (current <= convert(index)) { setValue(index / float64(sectionsCount));