Add file size limit to autoplay settings.

This commit is contained in:
John Preston 2019-12-24 13:09:04 +03:00
parent 8a3506af89
commit 616957e46f
15 changed files with 290 additions and 200 deletions

View File

@ -415,8 +415,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
"lng_settings_system_integration" = "System integration"; "lng_settings_system_integration" = "System integration";
"lng_settings_performance" = "Performance"; "lng_settings_performance" = "Performance";
"lng_settings_enable_animations" = "Enable animations"; "lng_settings_enable_animations" = "Enable animations";
"lng_settings_autoplay_gifs" = "Autoplay GIFs";
"lng_settings_autoplay_videos" = "Autoplay Videos";
"lng_settings_sensitive_title" = "Sensitive content"; "lng_settings_sensitive_title" = "Sensitive content";
"lng_settings_sensitive_disable_filtering" = "Disable filtering"; "lng_settings_sensitive_disable_filtering" = "Disable filtering";
"lng_settings_sensitive_about" = "Display sensitive media in public channels on all your Telegram devices."; "lng_settings_sensitive_about" = "Display sensitive media in public channels on all your Telegram devices.";
@ -1154,12 +1152,11 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
"lng_media_auto_in_groups" = "In groups"; "lng_media_auto_in_groups" = "In groups";
"lng_media_auto_in_channels" = "In channels"; "lng_media_auto_in_channels" = "In channels";
"lng_media_auto_title" = "Automatically download"; "lng_media_auto_title" = "Automatically download";
"lng_media_auto_play" = "Autoplay";
"lng_media_photo_title" = "Photos"; "lng_media_photo_title" = "Photos";
"lng_media_video_title" = "Video files"; "lng_media_video_title" = "Video files";
"lng_media_audio_title" = "Voice messages";
"lng_media_video_messages_title" = "Round video messages"; "lng_media_video_messages_title" = "Round video messages";
"lng_media_file_title" = "Files"; "lng_media_file_title" = "Files";
"lng_media_music_title" = "Music";
"lng_media_animation_title" = "Animated GIFs"; "lng_media_animation_title" = "Animated GIFs";
"lng_media_size_limit" = "Limit by size"; "lng_media_size_limit" = "Limit by size";
"lng_media_size_up_to" = "up to {size}"; "lng_media_size_up_to" = "up to {size}";

View File

@ -25,8 +25,53 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
namespace { namespace {
constexpr auto kMegabyte = 1024 * 1024; constexpr auto kMegabyte = 1024 * 1024;
constexpr auto kDefaultLimit = 10 * kMegabyte; constexpr auto kDefaultDownloadLimit = 10 * kMegabyte;
constexpr auto kDefaultAutoPlayLimit = 50 * kMegabyte;
using Type = Data::AutoDownload::Type;
not_null<int*> AddSizeLimitSlider(
not_null<Ui::VerticalLayout*> container,
const base::flat_map<Type, int> &values,
int defaultValue) {
using namespace Settings;
using Pair = base::flat_map<Type, int>::value_type;
const auto limits = Ui::CreateChild<rpl::event_stream<int>>(
container.get());
const auto currentLimit = ranges::max_element(
values,
std::less<>(),
[](Pair pair) { return pair.second; })->second;
const auto initialLimit = currentLimit ? currentLimit : defaultValue;
const auto result = Ui::CreateChild<int>(container.get(), initialLimit);
AddButtonWithLabel(
container,
tr::lng_media_size_limit(),
limits->events_starting_with_copy(
initialLimit
) | rpl::map([](int value) {
return tr::lng_media_size_up_to(
tr::now,
lt_size,
QString::number(value / kMegabyte) + " MB");
}),
st::autoDownloadLimitButton
)->setAttribute(Qt::WA_TransparentForMouseEvents);
const auto slider = container->add(
object_ptr<Ui::MediaSlider>(container, st::autoDownloadLimitSlider),
st::autoDownloadLimitPadding);
slider->resize(st::autoDownloadLimitSlider.seekSize);
slider->setPseudoDiscrete(
Export::View::kSizeValueCount,
Export::View::SizeLimitByIndex,
*result,
[=](int value) {
*result = value;
limits->fire_copy(value);
});
return result;
}
} // namespace } // namespace
AutoDownloadBox::AutoDownloadBox( AutoDownloadBox::AutoDownloadBox(
@ -42,12 +87,13 @@ void AutoDownloadBox::prepare() {
} }
void AutoDownloadBox::setupContent() { void AutoDownloadBox::setupContent() {
using namespace rpl::mappers;
using namespace Settings; using namespace Settings;
using namespace Data::AutoDownload; using namespace Data::AutoDownload;
using namespace rpl::mappers;
using Type = Data::AutoDownload::Type; using Type = Data::AutoDownload::Type;
using Pair = base::flat_map<Type, int>::value_type;
setTitle(tr::lng_media_auto_title()); setTitle(tr::lng_profile_settings_section());
const auto settings = &_session->settings().autoDownload(); const auto settings = &_session->settings().autoDownload();
const auto checked = [=](Source source, Type type) { const auto checked = [=](Source source, Type type) {
@ -60,11 +106,10 @@ void AutoDownloadBox::setupContent() {
this, this,
std::move(wrap))); std::move(wrap)));
const auto values = Ui::CreateChild<base::flat_map<Type, int>>(content); const auto add = [&](
const auto add = [&](Type type, rpl::producer<QString> label) { not_null<base::flat_map<Type, int>*> values,
if (ranges::find(kStreamedTypes, type) != end(kStreamedTypes)) { Type type,
return; rpl::producer<QString> label) {
}
const auto value = settings->bytesLimit(_source, type); const auto value = settings->bytesLimit(_source, type);
AddButton( AddButton(
content, content,
@ -78,78 +123,78 @@ void AutoDownloadBox::setupContent() {
}, content->lifetime()); }, content->lifetime());
values->emplace(type, value); values->emplace(type, value);
}; };
add(Type::Photo, tr::lng_media_photo_title());
add(Type::VoiceMessage, tr::lng_media_audio_title());
add(Type::VideoMessage, tr::lng_media_video_messages_title());
add(Type::Video, tr::lng_media_video_title());
add(Type::File, tr::lng_media_file_title());
add(Type::Music, tr::lng_media_music_title());
add(Type::GIF, tr::lng_media_animation_title());
const auto limits = Ui::CreateChild<rpl::event_stream<int>>(content); AddSubsectionTitle(content, tr::lng_media_auto_title());
using Pair = base::flat_map<Type, int>::value_type;
const auto settingsLimit = ranges::max_element( const auto downloadValues = Ui::CreateChild<base::flat_map<Type, int>>(
*values, content);
std::less<>(), add(downloadValues, Type::Photo, tr::lng_media_photo_title());
[](Pair pair) { return pair.second; })->second; add(downloadValues, Type::File, tr::lng_media_file_title());
const auto initialLimit = settingsLimit ? settingsLimit : kDefaultLimit;
const auto limit = Ui::CreateChild<int>(content, initialLimit); const auto downloadLimit = AddSizeLimitSlider(
AddButtonWithLabel(
content, content,
tr::lng_media_size_limit(), *downloadValues,
limits->events_starting_with_copy( kDefaultDownloadLimit);
initialLimit
) | rpl::map([](int value) { AddSkip(content);
return tr::lng_media_size_up_to( AddSubsectionTitle(content, tr::lng_media_auto_play());
tr::now,
lt_size, const auto autoPlayValues = Ui::CreateChild<base::flat_map<Type, int>>(
QString::number(value / kMegabyte) + " MB"); content);
}), add(
st::autoDownloadLimitButton autoPlayValues,
)->setAttribute(Qt::WA_TransparentForMouseEvents); Type::AutoPlayVideoMessage,
const auto slider = content->add( tr::lng_media_video_messages_title());
object_ptr<Ui::MediaSlider>(content, st::autoDownloadLimitSlider), add(autoPlayValues, Type::AutoPlayVideo, tr::lng_media_video_title());
st::autoDownloadLimitPadding); add(autoPlayValues, Type::AutoPlayGIF, tr::lng_media_animation_title());
slider->resize(st::autoDownloadLimitSlider.seekSize);
slider->setPseudoDiscrete( const auto autoPlayLimit = AddSizeLimitSlider(
Export::View::kSizeValueCount, content,
Export::View::SizeLimitByIndex, *autoPlayValues,
*limit, kDefaultAutoPlayLimit);
[=](int value) {
*limit = value; const auto limitByType = [=](Type type) {
limits->fire_copy(value); return (ranges::find(kAutoPlayTypes, type) != end(kAutoPlayTypes))
}); ? *autoPlayLimit
: *downloadLimit;
};
addButton(tr::lng_connection_save(), [=] { addButton(tr::lng_connection_save(), [=] {
auto allowMore = ranges::view::all( auto &&values = ranges::view::concat(
*values *downloadValues,
) | ranges::view::filter([&](Pair pair) { *autoPlayValues);
auto allowMore = values | ranges::view::filter([&](Pair pair) {
const auto [type, enabled] = pair; const auto [type, enabled] = pair;
const auto value = enabled ? *limit : 0; const auto value = enabled ? limitByType(type) : 0;
const auto old = settings->bytesLimit(_source, type); const auto old = settings->bytesLimit(_source, type);
return (old < value); return (old < value);
}) | ranges::view::transform([](Pair pair) { }) | ranges::view::transform([](Pair pair) {
return pair.first; return pair.first;
}); });
const auto less = ranges::find_if(*autoPlayValues, [&](Pair pair) {
const auto [type, enabled] = pair;
const auto value = enabled ? limitByType(type) : 0;
return value < settings->bytesLimit(_source, type);
}) != end(*autoPlayValues);
const auto allowMoreTypes = base::flat_set<Type>( const auto allowMoreTypes = base::flat_set<Type>(
allowMore.begin(), allowMore.begin(),
allowMore.end()); allowMore.end());
const auto changed = ranges::find_if(*values, [&](Pair pair) { const auto changed = ranges::find_if(values, [&](Pair pair) {
const auto [type, enabled] = pair; const auto [type, enabled] = pair;
const auto value = enabled ? *limit : 0; const auto value = enabled ? limitByType(type) : 0;
return settings->bytesLimit(_source, type) != value; return value != settings->bytesLimit(_source, type);
}) != end(*values); }) != end(values);
const auto &kHidden = kStreamedTypes; const auto &kHidden = kStreamedTypes;
const auto hiddenChanged = ranges::find_if(kHidden, [&](Type type) { const auto hiddenChanged = ranges::find_if(kHidden, [&](Type type) {
const auto now = settings->bytesLimit(_source, type); const auto now = settings->bytesLimit(_source, type);
return (now > 0) && (now != *limit); return (now > 0) && (now != limitByType(type));
}) != end(kHidden); }) != end(kHidden);
if (changed) { if (changed) {
for (const auto [type, enabled] : *values) { for (const auto [type, enabled] : values) {
const auto value = enabled ? *limit : 0; const auto value = enabled ? limitByType(type) : 0;
settings->setBytesLimit(_source, type, value); settings->setBytesLimit(_source, type, value);
} }
} }
@ -157,7 +202,10 @@ void AutoDownloadBox::setupContent() {
for (const auto type : kHidden) { for (const auto type : kHidden) {
const auto now = settings->bytesLimit(_source, type); const auto now = settings->bytesLimit(_source, type);
if (now > 0) { if (now > 0) {
settings->setBytesLimit(_source, type, *limit); settings->setBytesLimit(
_source,
type,
limitByType(type));
} }
} }
} }
@ -171,6 +219,9 @@ void AutoDownloadBox::setupContent() {
!= allowMoreTypes.end()) { != allowMoreTypes.end()) {
_session->data().documentLoadSettingsChanged(); _session->data().documentLoadSettingsChanged();
} }
if (less) {
_session->data().checkPlayingVideoFiles();
}
closeBox(); closeBox();
}); });
addButton(tr::lng_cancel(), [=] { closeBox(); }); addButton(tr::lng_cancel(), [=] { closeBox(); });

View File

@ -20,7 +20,9 @@ namespace AutoDownload {
namespace { namespace {
constexpr auto kDefaultMaxSize = 8 * 1024 * 1024; constexpr auto kDefaultMaxSize = 8 * 1024 * 1024;
constexpr auto kVersion = char(1); constexpr auto kDefaultAutoPlaySize = 50 * 1024 * 1024;
constexpr auto kVersion1 = char(1);
constexpr auto kVersion = char(2);
template <typename Enum> template <typename Enum>
auto enums_view(int from, int till) { auto enums_view(int from, int till) {
@ -38,13 +40,16 @@ auto enums_view(int till) {
void SetDefaultsForSource(Full &data, Source source) { void SetDefaultsForSource(Full &data, Source source) {
data.setBytesLimit(source, Type::Photo, kDefaultMaxSize); data.setBytesLimit(source, Type::Photo, kDefaultMaxSize);
data.setBytesLimit(source, Type::VoiceMessage, kDefaultMaxSize); data.setBytesLimit(source, Type::VoiceMessage, kDefaultMaxSize);
data.setBytesLimit(source, Type::VideoMessage, kDefaultMaxSize); data.setBytesLimit(
data.setBytesLimit(source, Type::GIF, kDefaultMaxSize); source,
Type::AutoPlayVideoMessage,
kDefaultAutoPlaySize);
data.setBytesLimit(source, Type::AutoPlayGIF, kDefaultAutoPlaySize);
const auto channelsFileLimit = (source == Source::Channel) const auto channelsFileLimit = (source == Source::Channel)
? 0 ? 0
: kDefaultMaxSize; : kDefaultMaxSize;
data.setBytesLimit(source, Type::File, channelsFileLimit); data.setBytesLimit(source, Type::File, channelsFileLimit);
data.setBytesLimit(source, Type::Video, channelsFileLimit); data.setBytesLimit(source, Type::AutoPlayVideo, kDefaultAutoPlaySize);
data.setBytesLimit(source, Type::Music, channelsFileLimit); data.setBytesLimit(source, Type::Music, channelsFileLimit);
} }
@ -69,19 +74,12 @@ Source SourceFromPeer(not_null<PeerData*> peer) {
} }
} }
Type TypeFromDocument(not_null<DocumentData*> document) { Type AutoPlayTypeFromDocument(not_null<DocumentData*> document) {
if (document->isSong()) { return document->isVideoFile()
return Type::Music; ? Type::AutoPlayVideo
} else if (document->isVoiceMessage()) { : document->isVideoMessage()
return Type::VoiceMessage; ? Type::AutoPlayVideoMessage
} else if (document->isVideoMessage()) { : Type::AutoPlayGIF;
return Type::VideoMessage;
} else if (document->isAnimation()) {
return Type::GIF;
} else if (document->isVideoFile()) {
return Type::Video;
}
return Type::File;
} }
} // namespace } // namespace
@ -223,7 +221,7 @@ bool Full::setFromSerialized(const QByteArray &serialized) {
stream >> version; stream >> version;
if (stream.status() != QDataStream::Ok) { if (stream.status() != QDataStream::Ok) {
return false; return false;
} else if (version != kVersion) { } else if (version != kVersion && version != kVersion1) {
return false; return false;
} }
auto temp = Full(); auto temp = Full();
@ -236,6 +234,15 @@ bool Full::setFromSerialized(const QByteArray &serialized) {
} }
} }
} }
if (version == kVersion1) {
for (const auto source : enums_view<Source>(kSourcesCount)) {
for (const auto type : kAutoPlayTypes) {
temp.setBytesLimit(source, type, std::max(
temp.bytesLimit(source, type),
kDefaultAutoPlaySize));
}
}
}
_data = temp._data; _data = temp._data;
return true; return true;
} }
@ -259,7 +266,7 @@ bool Should(
} }
return data.shouldDownload( return data.shouldDownload(
SourceFromPeer(peer), SourceFromPeer(peer),
TypeFromDocument(document), Type::File,
document->size); document->size);
} }
@ -269,11 +276,10 @@ bool Should(
if (document->sticker()) { if (document->sticker()) {
return true; return true;
} }
const auto type = TypeFromDocument(document);
const auto size = document->size; const auto size = document->size;
return data.shouldDownload(Source::User, type, size) return data.shouldDownload(Source::User, Type::File, size)
|| data.shouldDownload(Source::Group, type, size) || data.shouldDownload(Source::Group, Type::File, size)
|| data.shouldDownload(Source::Channel, type, size); || data.shouldDownload(Source::Channel, Type::File, size);
} }
bool Should( bool Should(
@ -286,5 +292,25 @@ bool Should(
image->bytesSize()); image->bytesSize());
} }
bool ShouldAutoPlay(
const Full &data,
not_null<PeerData*> peer,
not_null<DocumentData*> document) {
return data.shouldDownload(
SourceFromPeer(peer),
AutoPlayTypeFromDocument(document),
document->size);
}
Full WithDisabledAutoPlay(const Full &data) {
auto result = data;
for (const auto source : enums_view<Source>(kSourcesCount)) {
for (const auto type : kAutoPlayTypes) {
result.setBytesLimit(source, type, 0);
}
}
return result;
}
} // namespace AutoDownload } // namespace AutoDownload
} // namespace Data } // namespace Data

View File

@ -27,21 +27,24 @@ enum class Source {
constexpr auto kSourcesCount = 3; constexpr auto kSourcesCount = 3;
enum class Type { enum class Type {
Photo = 0x00, Photo = 0x00,
Video = 0x01, AutoPlayVideo = 0x01,
VoiceMessage = 0x02, VoiceMessage = 0x02,
VideoMessage = 0x03, AutoPlayVideoMessage = 0x03,
Music = 0x04, Music = 0x04,
GIF = 0x05, AutoPlayGIF = 0x05,
File = 0x06, File = 0x06,
};
inline constexpr auto kAutoPlayTypes = {
Type::AutoPlayVideo,
Type::AutoPlayVideoMessage,
Type::AutoPlayGIF,
}; };
inline constexpr auto kStreamedTypes = { inline constexpr auto kStreamedTypes = {
Type::Video,
Type::Music,
Type::VoiceMessage, Type::VoiceMessage,
Type::VideoMessage, Type::Music,
Type::GIF,
}; };
constexpr auto kTypesCount = 7; constexpr auto kTypesCount = 7;
@ -85,34 +88,44 @@ class Full {
public: public:
void setBytesLimit(Source source, Type type, int bytesLimit); void setBytesLimit(Source source, Type type, int bytesLimit);
bool shouldDownload(Source source, Type type, int fileSize) const; [[nodiscard]] bool shouldDownload(
int bytesLimit(Source source, Type type) const; Source source,
Type type,
int fileSize) const;
[[nodiscard]] int bytesLimit(Source source, Type type) const;
QByteArray serialize() const; [[nodiscard]] QByteArray serialize() const;
bool setFromSerialized(const QByteArray &serialized); bool setFromSerialized(const QByteArray &serialized);
static Full FullDisabled(); [[nodiscard]] static Full FullDisabled();
private: private:
const Set &set(Source source) const; [[nodiscard]] const Set &set(Source source) const;
Set &set(Source source); [[nodiscard]] Set &set(Source source);
const Set &setOrDefault(Source source, Type type) const; [[nodiscard]] const Set &setOrDefault(Source source, Type type) const;
std::array<Set, kSourcesCount> _data; std::array<Set, kSourcesCount> _data;
}; };
bool Should( [[nodiscard]] bool Should(
const Full &data, const Full &data,
not_null<PeerData*> peer, not_null<PeerData*> peer,
not_null<DocumentData*> document); not_null<DocumentData*> document);
bool Should( [[nodiscard]] bool Should(
const Full &data, const Full &data,
not_null<DocumentData*> document); not_null<DocumentData*> document);
bool Should( [[nodiscard]] bool Should(
const Full &data, const Full &data,
not_null<PeerData*> peer, not_null<PeerData*> peer,
not_null<Images::Source*> image); not_null<Images::Source*> image);
[[nodiscard]] bool ShouldAutoPlay(
const Full &data,
not_null<PeerData*> peer,
not_null<DocumentData*> document);
[[nodiscard]] Full WithDisabledAutoPlay(const Full &data);
} // namespace AutoDownload } // namespace AutoDownload
} // namespace Data } // namespace Data

View File

@ -1046,15 +1046,22 @@ void InnerWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
cancelContextDownload(document); cancelContextDownload(document);
}); });
} else { } else {
if (document->loaded() const auto itemId = view
&& document->isGifv() ? view->data()->fullId()
&& !document->session().settings().autoplayGifs()) { : FullMsgId();
const auto itemId = view if (const auto item = document->session().data().message(itemId)) {
? view->data()->fullId() const auto notAutoplayedGif = [&] {
: FullMsgId(); return document->isGifv()
_menu->addAction(tr::lng_context_open_gif(tr::now), [=] { && Data::AutoDownload::ShouldAutoPlay(
openContextGif(itemId); document->session().settings().autoDownload(),
}); item->history()->peer,
document);
}();
if (notAutoplayedGif) {
_menu->addAction(tr::lng_context_open_gif(tr::now), [=] {
openContextGif(itemId);
});
}
} }
if (!document->filepath(DocumentData::FilePathResolve::Checked).isEmpty()) { if (!document->filepath(DocumentData::FilePathResolve::Checked).isEmpty()) {
_menu->addAction(Platform::IsMac() ? tr::lng_context_show_in_finder(tr::now) : tr::lng_context_show_in_folder(tr::now), [=] { _menu->addAction(Platform::IsMac() ? tr::lng_context_show_in_finder(tr::now) : tr::lng_context_show_in_folder(tr::now), [=] {

View File

@ -1568,8 +1568,16 @@ void HistoryInner::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
const auto lnkIsVideo = document->isVideoFile(); const auto lnkIsVideo = document->isVideoFile();
const auto lnkIsVoice = document->isVoiceMessage(); const auto lnkIsVoice = document->isVoiceMessage();
const auto lnkIsAudio = document->isAudioFile(); const auto lnkIsAudio = document->isAudioFile();
if (document->loaded() && document->isGifv()) { if (document->isGifv()) {
if (!document->session().settings().autoplayGifs()) { const auto notAutoplayedGif = [&] {
return item
&& document->isGifv()
&& Data::AutoDownload::ShouldAutoPlay(
document->session().settings().autoDownload(),
item->history()->peer,
document);
}();
if (notAutoplayedGif) {
_menu->addAction(tr::lng_context_open_gif(tr::now), [=] { _menu->addAction(tr::lng_context_open_gif(tr::now), [=] {
openContextGif(itemId); openContextGif(itemId);
}); });

View File

@ -168,12 +168,19 @@ void AddDocumentActions(
}); });
return; return;
} }
if (document->loaded() if (const auto item = document->session().data().message(contextId)) {
&& document->isGifv() const auto notAutoplayedGif = [&] {
&& !document->session().settings().autoplayGifs()) { return document->isGifv()
menu->addAction(tr::lng_context_open_gif(tr::now), [=] { && Data::AutoDownload::ShouldAutoPlay(
OpenGif(contextId); document->session().settings().autoDownload(),
}); item->history()->peer,
document);
}();
if (notAutoplayedGif) {
menu->addAction(tr::lng_context_open_gif(tr::now), [=] {
OpenGif(contextId);
});
}
} }
if (document->sticker() if (document->sticker()
&& document->sticker()->set.type() != mtpc_inputStickerSetEmpty) { && document->sticker()->set.type() != mtpc_inputStickerSetEmpty) {

View File

@ -211,7 +211,9 @@ void Document::draw(Painter &p, const QRect &r, TextSelection selection, crl::ti
const auto cornerDownload = downloadInCorner(); const auto cornerDownload = downloadInCorner();
_data->automaticLoad(_realParent->fullId(), _parent->data()); if (!_data->canBePlayed()) {
_data->automaticLoad(_realParent->fullId(), _parent->data());
}
bool loaded = _data->loaded(), displayLoading = _data->displayLoading(); bool loaded = _data->loaded(), displayLoading = _data->displayLoading();
bool selected = (selection == FullSelection); bool selected = (selection == FullSelection);

View File

@ -238,9 +238,10 @@ bool Gif::downloadInCorner() const {
} }
bool Gif::autoplayEnabled() const { bool Gif::autoplayEnabled() const {
return _data->isVideoFile() return Data::AutoDownload::ShouldAutoPlay(
? history()->session().settings().autoplayVideos() _data->session().settings().autoDownload(),
: history()->session().settings().autoplayGifs(); _realParent->history()->peer,
_data);
} }
void Gif::draw(Painter &p, const QRect &r, TextSelection selection, crl::time ms) const { void Gif::draw(Painter &p, const QRect &r, TextSelection selection, crl::time ms) const {
@ -415,8 +416,7 @@ void Gif::draw(Painter &p, const QRect &r, TextSelection selection, crl::time ms
if (radial if (radial
|| (!streamingMode || (!streamingMode
&& ((!_data->loaded() && !_data->loading()) && ((!_data->loaded() && !_data->loading()) || !autoplay))) {
|| !autoplayEnabled()))) {
const auto radialOpacity = item->isSending() const auto radialOpacity = item->isSending()
? 1. ? 1.
: streamed : streamed
@ -923,8 +923,7 @@ void Gif::drawGrouped(
if (radial if (radial
|| (!streamingMode || (!streamingMode
&& ((!_data->loaded() && !_data->loading()) && ((!_data->loaded() && !_data->loading()) || !autoplay))) {
|| !autoplayEnabled()))) {
const auto radialOpacity = item->isSending() const auto radialOpacity = item->isSending()
? 1. ? 1.
: streamed : streamed

View File

@ -19,6 +19,8 @@ namespace {
constexpr auto kAutoLockTimeoutLateMs = crl::time(3000); constexpr auto kAutoLockTimeoutLateMs = crl::time(3000);
constexpr auto kLegacyCallsPeerToPeerNobody = 4; constexpr auto kLegacyCallsPeerToPeerNobody = 4;
constexpr auto kVersionTag = -1;
constexpr auto kVersion = 1;
} // namespace } // namespace
@ -45,6 +47,7 @@ QByteArray Settings::serialize() const {
{ {
QDataStream stream(&result, QIODevice::WriteOnly); QDataStream stream(&result, QIODevice::WriteOnly);
stream.setVersion(QDataStream::Qt_5_1); stream.setVersion(QDataStream::Qt_5_1);
stream << qint32(kVersionTag) << qint32(kVersion);
stream << static_cast<qint32>(_variables.selectorTab); stream << static_cast<qint32>(_variables.selectorTab);
stream << qint32(_variables.lastSeenWarningSeen ? 1 : 0); stream << qint32(_variables.lastSeenWarningSeen ? 1 : 0);
stream << qint32(_variables.tabbedSelectorSectionEnabled ? 1 : 0); stream << qint32(_variables.tabbedSelectorSectionEnabled ? 1 : 0);
@ -83,14 +86,13 @@ QByteArray Settings::serialize() const {
stream << qint32(_variables.notifyAboutPinned.current() ? 1 : 0); stream << qint32(_variables.notifyAboutPinned.current() ? 1 : 0);
stream << qint32(_variables.archiveInMainMenu.current() ? 1 : 0); stream << qint32(_variables.archiveInMainMenu.current() ? 1 : 0);
stream << qint32(_variables.skipArchiveInSearch.current() ? 1 : 0); stream << qint32(_variables.skipArchiveInSearch.current() ? 1 : 0);
stream << qint32(_variables.autoplayGifs ? 1 : 0); stream << qint32(0);// LEGACY _variables.autoplayGifs ? 1 : 0);
stream << qint32(_variables.loopAnimatedStickers ? 1 : 0); stream << qint32(_variables.loopAnimatedStickers ? 1 : 0);
stream << qint32(_variables.largeEmoji.current() ? 1 : 0); stream << qint32(_variables.largeEmoji.current() ? 1 : 0);
stream << qint32(_variables.replaceEmoji.current() ? 1 : 0); stream << qint32(_variables.replaceEmoji.current() ? 1 : 0);
stream << qint32(_variables.suggestEmoji ? 1 : 0); stream << qint32(_variables.suggestEmoji ? 1 : 0);
stream << qint32(_variables.suggestStickersByEmoji ? 1 : 0); stream << qint32(_variables.suggestStickersByEmoji ? 1 : 0);
stream << qint32(_variables.spellcheckerEnabled.current() ? 1 : 0); stream << qint32(_variables.spellcheckerEnabled.current() ? 1 : 0);
stream << qint32(_variables.autoplayVideos ? 1 : 0);
} }
return result; return result;
} }
@ -102,6 +104,8 @@ void Settings::constructFromSerialized(const QByteArray &serialized) {
QDataStream stream(serialized); QDataStream stream(serialized);
stream.setVersion(QDataStream::Qt_5_1); stream.setVersion(QDataStream::Qt_5_1);
qint32 versionTag = 0;
qint32 version = 0;
qint32 selectorTab = static_cast<qint32>(ChatHelpers::SelectorTab::Emoji); qint32 selectorTab = static_cast<qint32>(ChatHelpers::SelectorTab::Emoji);
qint32 lastSeenWarningSeen = 0; qint32 lastSeenWarningSeen = 0;
qint32 tabbedSelectorSectionEnabled = 1; qint32 tabbedSelectorSectionEnabled = 1;
@ -131,16 +135,21 @@ void Settings::constructFromSerialized(const QByteArray &serialized) {
qint32 notifyAboutPinned = _variables.notifyAboutPinned.current() ? 1 : 0; qint32 notifyAboutPinned = _variables.notifyAboutPinned.current() ? 1 : 0;
qint32 archiveInMainMenu = _variables.archiveInMainMenu.current() ? 1 : 0; qint32 archiveInMainMenu = _variables.archiveInMainMenu.current() ? 1 : 0;
qint32 skipArchiveInSearch = _variables.skipArchiveInSearch.current() ? 1 : 0; qint32 skipArchiveInSearch = _variables.skipArchiveInSearch.current() ? 1 : 0;
qint32 autoplayGifs = _variables.autoplayGifs ? 1 : 0; qint32 autoplayGifs = 1;
qint32 loopAnimatedStickers = _variables.loopAnimatedStickers ? 1 : 0; qint32 loopAnimatedStickers = _variables.loopAnimatedStickers ? 1 : 0;
qint32 largeEmoji = _variables.largeEmoji.current() ? 1 : 0; qint32 largeEmoji = _variables.largeEmoji.current() ? 1 : 0;
qint32 replaceEmoji = _variables.replaceEmoji.current() ? 1 : 0; qint32 replaceEmoji = _variables.replaceEmoji.current() ? 1 : 0;
qint32 suggestEmoji = _variables.suggestEmoji ? 1 : 0; qint32 suggestEmoji = _variables.suggestEmoji ? 1 : 0;
qint32 suggestStickersByEmoji = _variables.suggestStickersByEmoji ? 1 : 0; qint32 suggestStickersByEmoji = _variables.suggestStickersByEmoji ? 1 : 0;
qint32 spellcheckerEnabled = _variables.spellcheckerEnabled.current() ? 1 : 0; qint32 spellcheckerEnabled = _variables.spellcheckerEnabled.current() ? 1 : 0;
qint32 autoplayVideos = _variables.autoplayVideos ? 1 : 0;
stream >> selectorTab; stream >> versionTag;
if (versionTag == kVersionTag) {
stream >> version;
stream >> selectorTab;
} else {
selectorTab = versionTag;
}
stream >> lastSeenWarningSeen; stream >> lastSeenWarningSeen;
if (!stream.atEnd()) { if (!stream.atEnd()) {
stream >> tabbedSelectorSectionEnabled; stream >> tabbedSelectorSectionEnabled;
@ -241,9 +250,6 @@ void Settings::constructFromSerialized(const QByteArray &serialized) {
if (!stream.atEnd()) { if (!stream.atEnd()) {
stream >> spellcheckerEnabled; stream >> spellcheckerEnabled;
} }
if (!stream.atEnd()) {
stream >> autoplayVideos;
}
if (stream.status() != QDataStream::Ok) { if (stream.status() != QDataStream::Ok) {
LOG(("App Error: " LOG(("App Error: "
"Bad data for Main::Settings::constructFromSerialized()")); "Bad data for Main::Settings::constructFromSerialized()"));
@ -253,6 +259,13 @@ void Settings::constructFromSerialized(const QByteArray &serialized) {
&& !_variables.autoDownload.setFromSerialized(autoDownload)) { && !_variables.autoDownload.setFromSerialized(autoDownload)) {
return; return;
} }
if (!version) {
if (!autoplayGifs) {
using namespace Data::AutoDownload;
_variables.autoDownload = WithDisabledAutoPlay(
_variables.autoDownload);
}
}
auto uncheckedTab = static_cast<ChatHelpers::SelectorTab>(selectorTab); auto uncheckedTab = static_cast<ChatHelpers::SelectorTab>(selectorTab);
switch (uncheckedTab) { switch (uncheckedTab) {
@ -317,14 +330,12 @@ void Settings::constructFromSerialized(const QByteArray &serialized) {
_variables.notifyAboutPinned = (notifyAboutPinned == 1); _variables.notifyAboutPinned = (notifyAboutPinned == 1);
_variables.archiveInMainMenu = (archiveInMainMenu == 1); _variables.archiveInMainMenu = (archiveInMainMenu == 1);
_variables.skipArchiveInSearch = (skipArchiveInSearch == 1); _variables.skipArchiveInSearch = (skipArchiveInSearch == 1);
_variables.autoplayGifs = (autoplayGifs == 1);
_variables.loopAnimatedStickers = (loopAnimatedStickers == 1); _variables.loopAnimatedStickers = (loopAnimatedStickers == 1);
_variables.largeEmoji = (largeEmoji == 1); _variables.largeEmoji = (largeEmoji == 1);
_variables.replaceEmoji = (replaceEmoji == 1); _variables.replaceEmoji = (replaceEmoji == 1);
_variables.suggestEmoji = (suggestEmoji == 1); _variables.suggestEmoji = (suggestEmoji == 1);
_variables.suggestStickersByEmoji = (suggestStickersByEmoji == 1); _variables.suggestStickersByEmoji = (suggestStickersByEmoji == 1);
_variables.spellcheckerEnabled = (spellcheckerEnabled == 1); _variables.spellcheckerEnabled = (spellcheckerEnabled == 1);
_variables.autoplayVideos = (autoplayVideos == 1);
} }
void Settings::setSupportChatsTimeSlice(int slice) { void Settings::setSupportChatsTimeSlice(int slice) {

View File

@ -198,18 +198,6 @@ public:
void setExeLaunchWarning(bool warning) { void setExeLaunchWarning(bool warning) {
_variables.exeLaunchWarning = warning; _variables.exeLaunchWarning = warning;
} }
[[nodiscard]] bool autoplayGifs() const {
return _variables.autoplayGifs;
}
void setAutoplayGifs(bool value) {
_variables.autoplayGifs = value;
}
[[nodiscard]] bool autoplayVideos() const {
return _variables.autoplayVideos;
}
void setAutoplayVideos(bool value) {
_variables.autoplayVideos = value;
}
[[nodiscard]] bool loopAnimatedStickers() const { [[nodiscard]] bool loopAnimatedStickers() const {
return _variables.loopAnimatedStickers; return _variables.loopAnimatedStickers;
} }
@ -283,8 +271,6 @@ private:
rpl::variable<bool> archiveInMainMenu = false; rpl::variable<bool> archiveInMainMenu = false;
rpl::variable<bool> notifyAboutPinned = true; rpl::variable<bool> notifyAboutPinned = true;
rpl::variable<bool> skipArchiveInSearch = false; rpl::variable<bool> skipArchiveInSearch = false;
bool autoplayGifs = true;
bool autoplayVideos = true;
bool loopAnimatedStickers = true; bool loopAnimatedStickers = true;
rpl::variable<bool> largeEmoji = true; rpl::variable<bool> largeEmoji = true;
rpl::variable<bool> replaceEmoji = true; rpl::variable<bool> replaceEmoji = true;

View File

@ -1856,14 +1856,15 @@ void OverlayWidget::displayDocument(
_doc->dimensions.height()); _doc->dimensions.height());
} }
} else { } else {
_doc->automaticLoad(fileOrigin(), item);
if (_doc->canBePlayed() && initStreaming()) { if (_doc->canBePlayed() && initStreaming()) {
} else if (_doc->isVideoFile()) { } else if (_doc->isVideoFile()) {
_doc->automaticLoad(fileOrigin(), item);
initStreamingThumbnail(); initStreamingThumbnail();
} else if (_doc->isTheme()) { } else if (_doc->isTheme()) {
_doc->automaticLoad(fileOrigin(), item);
initThemePreview(); initThemePreview();
} else { } else {
_doc->automaticLoad(fileOrigin(), item);
auto &location = _doc->location(true); auto &location = _doc->location(true);
if (location.accessEnable()) { if (location.accessEnable()) {
const auto &path = location.name(); const auto &path = location.name();
@ -3216,7 +3217,9 @@ void OverlayWidget::preloadData(int delta) {
image->load(fileOrigin()); image->load(fileOrigin());
} else { } else {
(*document)->loadThumbnail(fileOrigin()); (*document)->loadThumbnail(fileOrigin());
(*document)->automaticLoad(fileOrigin(), entity.item); if (!(*document)->canBePlayed()) {
(*document)->automaticLoad(fileOrigin(), entity.item);
}
} }
} }
} }

View File

@ -429,7 +429,6 @@ void Video::paint(Painter &p, const QRect &clip, TextSelection selection, const
const auto thumbLoaded = _data->hasThumbnail() const auto thumbLoaded = _data->hasThumbnail()
&& _data->thumbnail()->loaded(); && _data->thumbnail()->loaded();
_data->automaticLoad(parent()->fullId(), parent());
bool loaded = _data->loaded(), displayLoading = _data->displayLoading(); bool loaded = _data->loaded(), displayLoading = _data->displayLoading();
if (displayLoading) { if (displayLoading) {
ensureRadial(); ensureRadial();
@ -621,8 +620,6 @@ void Voice::initDimensions() {
void Voice::paint(Painter &p, const QRect &clip, TextSelection selection, const PaintContext *context) { void Voice::paint(Painter &p, const QRect &clip, TextSelection selection, const PaintContext *context) {
bool selected = (selection == FullSelection); bool selected = (selection == FullSelection);
_data->automaticLoad(parent()->fullId(), parent());
bool loaded = _data->loaded(), displayLoading = _data->displayLoading(); bool loaded = _data->loaded(), displayLoading = _data->displayLoading();
if (displayLoading) { if (displayLoading) {

View File

@ -452,41 +452,6 @@ void SetupPerformance(
not_null<Window::SessionController*> controller, not_null<Window::SessionController*> controller,
not_null<Ui::VerticalLayout*> container) { not_null<Ui::VerticalLayout*> container) {
SetupAnimations(container); SetupAnimations(container);
const auto session = &controller->session();
AddButton(
container,
tr::lng_settings_autoplay_gifs(),
st::settingsButton
)->toggleOn(
rpl::single(session->settings().autoplayGifs())
)->toggledValue(
) | rpl::filter([=](bool enabled) {
return (enabled != session->settings().autoplayGifs());
}) | rpl::start_with_next([=](bool enabled) {
session->settings().setAutoplayGifs(enabled);
if (!enabled) {
session->data().checkPlayingVideoFiles();
}
session->saveSettingsDelayed();
}, container->lifetime());
AddButton(
container,
tr::lng_settings_autoplay_videos(),
st::settingsButton
)->toggleOn(
rpl::single(session->settings().autoplayVideos())
)->toggledValue(
) | rpl::filter([=](bool enabled) {
return (enabled != session->settings().autoplayVideos());
}) | rpl::start_with_next([=](bool enabled) {
session->settings().setAutoplayVideos(enabled);
if (!enabled) {
session->data().checkPlayingVideoFiles();
}
session->saveSettingsDelayed();
}, container->lifetime());
} }
void SetupSystemIntegration( void SetupSystemIntegration(

View File

@ -1117,8 +1117,8 @@ bool _readSetting(quint32 blockId, QDataStream &stream, int version, ReadSetting
}; };
set(Type::Photo, photo); set(Type::Photo, photo);
set(Type::VoiceMessage, audio); set(Type::VoiceMessage, audio);
set(Type::GIF, gif); set(Type::AutoPlayGIF, gif);
set(Type::VideoMessage, gif); set(Type::AutoPlayVideoMessage, gif);
} break; } break;
case dbiAutoPlayOld: { case dbiAutoPlayOld: {
@ -1126,7 +1126,25 @@ bool _readSetting(quint32 blockId, QDataStream &stream, int version, ReadSetting
stream >> gif; stream >> gif;
if (!_checkStreamStatus(stream)) return false; if (!_checkStreamStatus(stream)) return false;
GetStoredSessionSettings().setAutoplayGifs(gif == 1); if (!gif) {
using namespace Data::AutoDownload;
auto &settings = GetStoredSessionSettings().autoDownload();
const auto types = {
Type::AutoPlayGIF,
Type::AutoPlayVideo,
Type::AutoPlayVideoMessage,
};
const auto sources = {
Source::User,
Source::Group,
Source::Channel
};
for (const auto source : sources) {
for (const auto type : types) {
settings.setBytesLimit(source, type, 0);
}
}
}
} break; } break;
case dbiDialogsMode: { case dbiDialogsMode: {