mirror of https://github.com/procxx/kepka.git
Upload wallpapers to the cloud.
This commit is contained in:
parent
890aacaeee
commit
95565c39ed
|
@ -5142,7 +5142,6 @@ void ApiWrap::photoUploadReady(
|
||||||
)).done(applier).afterRequest(history->sendRequestId).send();
|
)).done(applier).afterRequest(history->sendRequestId).send();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ApiWrap::clearPeerPhoto(not_null<PhotoData*> photo) {
|
void ApiWrap::clearPeerPhoto(not_null<PhotoData*> photo) {
|
||||||
|
|
|
@ -110,7 +110,7 @@ BackgroundBox::Inner::Inner(QWidget *parent) : RpWidget(parent)
|
||||||
, _check(std::make_unique<Ui::RoundCheckbox>(st::overviewCheck, [=] { update(); })) {
|
, _check(std::make_unique<Ui::RoundCheckbox>(st::overviewCheck, [=] { update(); })) {
|
||||||
_check->setChecked(true, Ui::RoundCheckbox::SetStyle::Fast);
|
_check->setChecked(true, Ui::RoundCheckbox::SetStyle::Fast);
|
||||||
if (Auth().data().wallpapers().empty()) {
|
if (Auth().data().wallpapers().empty()) {
|
||||||
resize(kBackgroundsInRow * (st::backgroundSize.width() + st::backgroundPadding) + st::backgroundPadding, 2 * (st::backgroundSize.height() + st::backgroundPadding) + st::backgroundPadding);
|
resize(st::boxWideWidth, 2 * (st::backgroundSize.height() + st::backgroundPadding) + st::backgroundPadding);
|
||||||
} else {
|
} else {
|
||||||
updatePapers();
|
updatePapers();
|
||||||
}
|
}
|
||||||
|
@ -169,7 +169,7 @@ void BackgroundBox::Inner::updatePapers() {
|
||||||
const auto rows = (count / kBackgroundsInRow)
|
const auto rows = (count / kBackgroundsInRow)
|
||||||
+ (count % kBackgroundsInRow ? 1 : 0);
|
+ (count % kBackgroundsInRow ? 1 : 0);
|
||||||
|
|
||||||
resize(kBackgroundsInRow * (st::backgroundSize.width() + st::backgroundPadding) + st::backgroundPadding, rows * (st::backgroundSize.height() + st::backgroundPadding) + st::backgroundPadding);
|
resize(st::boxWideWidth, rows * (st::backgroundSize.height() + st::backgroundPadding) + st::backgroundPadding);
|
||||||
|
|
||||||
const auto preload = kBackgroundsInRow * 3;
|
const auto preload = kBackgroundsInRow * 3;
|
||||||
for (const auto &paper : _papers | ranges::view::take(preload)) {
|
for (const auto &paper : _papers | ranges::view::take(preload)) {
|
||||||
|
|
|
@ -943,7 +943,7 @@ backgroundCheck: ServiceCheck {
|
||||||
diameter: 18px;
|
diameter: 18px;
|
||||||
shift: 2px;
|
shift: 2px;
|
||||||
thickness: 2px;
|
thickness: 2px;
|
||||||
tip: point(8px, 13px);
|
tip: point(7px, 13px);
|
||||||
small: 3px;
|
small: 3px;
|
||||||
large: 6px;
|
large: 6px;
|
||||||
stroke: 2px;
|
stroke: 2px;
|
||||||
|
|
|
@ -0,0 +1,598 @@
|
||||||
|
/*
|
||||||
|
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 "data/data_wall_paper.h"
|
||||||
|
|
||||||
|
#include "data/data_document.h"
|
||||||
|
#include "data/data_file_origin.h"
|
||||||
|
#include "data/data_session.h"
|
||||||
|
#include "storage/serialize_common.h"
|
||||||
|
#include "core/application.h"
|
||||||
|
#include "auth_session.h"
|
||||||
|
|
||||||
|
namespace Data {
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
constexpr auto FromLegacyBackgroundId(int32 legacyId) -> WallPaperId {
|
||||||
|
return uint64(0xFFFFFFFF00000000ULL) | uint64(uint32(legacyId));
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr auto kUninitializedBackground = FromLegacyBackgroundId(-999);
|
||||||
|
constexpr auto kTestingThemeBackground = FromLegacyBackgroundId(-666);
|
||||||
|
constexpr auto kTestingDefaultBackground = FromLegacyBackgroundId(-665);
|
||||||
|
constexpr auto kTestingEditorBackground = FromLegacyBackgroundId(-664);
|
||||||
|
constexpr auto kThemeBackground = FromLegacyBackgroundId(-2);
|
||||||
|
constexpr auto kCustomBackground = FromLegacyBackgroundId(-1);
|
||||||
|
constexpr auto kLegacy1DefaultBackground = FromLegacyBackgroundId(0);
|
||||||
|
constexpr auto kDefaultBackground = 5947530738516623361;
|
||||||
|
constexpr auto kIncorrectDefaultBackground = FromLegacyBackgroundId(105);
|
||||||
|
|
||||||
|
quint32 SerializeMaybeColor(std::optional<QColor> color) {
|
||||||
|
return color
|
||||||
|
? ((quint32(std::clamp(color->red(), 0, 255)) << 16)
|
||||||
|
| (quint32(std::clamp(color->green(), 0, 255)) << 8)
|
||||||
|
| quint32(std::clamp(color->blue(), 0, 255)))
|
||||||
|
: quint32(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<QColor> MaybeColorFromSerialized(quint32 serialized) {
|
||||||
|
return (serialized == quint32(-1))
|
||||||
|
? std::nullopt
|
||||||
|
: std::make_optional(QColor(
|
||||||
|
int((serialized >> 16) & 0xFFU),
|
||||||
|
int((serialized >> 8) & 0xFFU),
|
||||||
|
int(serialized & 0xFFU)));
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<QColor> ColorFromString(const QString &string) {
|
||||||
|
if (string.size() != 6) {
|
||||||
|
return {};
|
||||||
|
} else if (ranges::find_if(string, [](QChar ch) {
|
||||||
|
return (ch < 'a' || ch > 'f')
|
||||||
|
&& (ch < 'A' || ch > 'F')
|
||||||
|
&& (ch < '0' || ch > '9');
|
||||||
|
}) != string.end()) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
const auto component = [](const QString &text, int index) {
|
||||||
|
const auto decimal = [](QChar hex) {
|
||||||
|
const auto code = hex.unicode();
|
||||||
|
return (code >= '0' && code <= '9')
|
||||||
|
? int(code - '0')
|
||||||
|
: (code >= 'a' && code <= 'f')
|
||||||
|
? int(code - 'a' + 0x0a)
|
||||||
|
: int(code - 'A' + 0x0a);
|
||||||
|
};
|
||||||
|
index *= 2;
|
||||||
|
return decimal(text[index]) * 0x10 + decimal(text[index + 1]);
|
||||||
|
};
|
||||||
|
return QColor(
|
||||||
|
component(string, 0),
|
||||||
|
component(string, 1),
|
||||||
|
component(string, 2),
|
||||||
|
255);
|
||||||
|
}
|
||||||
|
|
||||||
|
QString StringFromColor(QColor color) {
|
||||||
|
const auto component = [](int value) {
|
||||||
|
const auto hex = [](int value) {
|
||||||
|
value = std::clamp(value, 0, 15);
|
||||||
|
return (value > 9)
|
||||||
|
? ('a' + (value - 10))
|
||||||
|
: ('0' + value);
|
||||||
|
};
|
||||||
|
return QString() + hex(value / 16) + hex(value % 16);
|
||||||
|
};
|
||||||
|
return component(color.red())
|
||||||
|
+ component(color.green())
|
||||||
|
+ component(color.blue());
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
WallPaper::WallPaper(WallPaperId id) : _id(id) {
|
||||||
|
}
|
||||||
|
|
||||||
|
void WallPaper::setLocalImageAsThumbnail(std::shared_ptr<Image> image) {
|
||||||
|
Expects(IsDefaultWallPaper(*this)
|
||||||
|
|| IsLegacy1DefaultWallPaper(*this)
|
||||||
|
|| IsCustomWallPaper(*this));
|
||||||
|
Expects(_thumbnail == nullptr);
|
||||||
|
|
||||||
|
_thumbnail = std::move(image);
|
||||||
|
}
|
||||||
|
|
||||||
|
WallPaperId WallPaper::id() const {
|
||||||
|
return _id;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<QColor> WallPaper::backgroundColor() const {
|
||||||
|
return _backgroundColor;
|
||||||
|
}
|
||||||
|
|
||||||
|
DocumentData *WallPaper::document() const {
|
||||||
|
return _document;
|
||||||
|
}
|
||||||
|
|
||||||
|
Image *WallPaper::thumbnail() const {
|
||||||
|
return _thumbnail
|
||||||
|
? _thumbnail.get()
|
||||||
|
: _document
|
||||||
|
? _document->thumbnail()
|
||||||
|
: nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WallPaper::isPattern() const {
|
||||||
|
return _flags & MTPDwallPaper::Flag::f_pattern;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WallPaper::isDefault() const {
|
||||||
|
return _flags & MTPDwallPaper::Flag::f_default;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WallPaper::isCreator() const {
|
||||||
|
return _flags & MTPDwallPaper::Flag::f_creator;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WallPaper::isDark() const {
|
||||||
|
return _flags & MTPDwallPaper::Flag::f_dark;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WallPaper::isLocal() const {
|
||||||
|
return !document() && thumbnail();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WallPaper::isBlurred() const {
|
||||||
|
return _settings & MTPDwallPaperSettings::Flag::f_blur;
|
||||||
|
}
|
||||||
|
|
||||||
|
int WallPaper::patternIntensity() const {
|
||||||
|
return _intensity;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WallPaper::hasShareUrl() const {
|
||||||
|
return !_slug.isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
QString WallPaper::shareUrl() const {
|
||||||
|
if (!hasShareUrl()) {
|
||||||
|
return QString();
|
||||||
|
}
|
||||||
|
const auto base = Core::App().createInternalLinkFull("bg/" + _slug);
|
||||||
|
auto params = QStringList();
|
||||||
|
if (isPattern()) {
|
||||||
|
if (_backgroundColor) {
|
||||||
|
params.push_back("bg_color=" + StringFromColor(*_backgroundColor));
|
||||||
|
}
|
||||||
|
if (_intensity) {
|
||||||
|
params.push_back("intensity=" + QString::number(_intensity));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
auto mode = QStringList();
|
||||||
|
if (_settings & MTPDwallPaperSettings::Flag::f_blur) {
|
||||||
|
mode.push_back("blur");
|
||||||
|
}
|
||||||
|
if (_settings & MTPDwallPaperSettings::Flag::f_motion) {
|
||||||
|
mode.push_back("motion");
|
||||||
|
}
|
||||||
|
if (!mode.isEmpty()) {
|
||||||
|
params.push_back("mode=" + mode.join('+'));
|
||||||
|
}
|
||||||
|
return params.isEmpty()
|
||||||
|
? base
|
||||||
|
: base + '?' + params.join('&');
|
||||||
|
}
|
||||||
|
|
||||||
|
void WallPaper::loadThumbnail() const {
|
||||||
|
if (_thumbnail) {
|
||||||
|
_thumbnail->load(fileOrigin());
|
||||||
|
}
|
||||||
|
if (_document) {
|
||||||
|
_document->loadThumbnail(fileOrigin());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void WallPaper::loadDocument() const {
|
||||||
|
if (_document) {
|
||||||
|
_document->save(fileOrigin(), QString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
FileOrigin WallPaper::fileOrigin() const {
|
||||||
|
return FileOriginWallpaper(_id, _accessHash);
|
||||||
|
}
|
||||||
|
|
||||||
|
MTPWallPaperSettings WallPaper::mtpSettings() const {
|
||||||
|
return MTP_wallPaperSettings(
|
||||||
|
MTP_flags(_settings),
|
||||||
|
(_backgroundColor
|
||||||
|
? MTP_int(SerializeMaybeColor(_backgroundColor))
|
||||||
|
: MTP_int(0)),
|
||||||
|
MTP_int(_intensity));
|
||||||
|
}
|
||||||
|
|
||||||
|
WallPaper WallPaper::withUrlParams(
|
||||||
|
const QMap<QString, QString> ¶ms) const {
|
||||||
|
using Flag = MTPDwallPaperSettings::Flag;
|
||||||
|
|
||||||
|
auto result = *this;
|
||||||
|
result._settings = Flag(0);
|
||||||
|
result._backgroundColor = ColorFromString(_slug);
|
||||||
|
result._intensity = kDefaultIntensity;
|
||||||
|
|
||||||
|
if (auto mode = params.value("mode"); !mode.isEmpty()) {
|
||||||
|
const auto list = mode.replace('+', ' ').split(' ');
|
||||||
|
for (const auto &change : list) {
|
||||||
|
if (change == qstr("blur")) {
|
||||||
|
result._settings |= Flag::f_blur;
|
||||||
|
} else if (change == qstr("motion")) {
|
||||||
|
result._settings |= Flag::f_motion;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (const auto color = ColorFromString(params.value("bg_color"))) {
|
||||||
|
result._settings |= Flag::f_background_color;
|
||||||
|
result._backgroundColor = color;
|
||||||
|
}
|
||||||
|
if (const auto string = params.value("intensity"); !string.isEmpty()) {
|
||||||
|
auto ok = false;
|
||||||
|
const auto intensity = string.toInt(&ok);
|
||||||
|
if (ok && base::in_range(intensity, 0, 101)) {
|
||||||
|
result._settings |= Flag::f_intensity;
|
||||||
|
result._intensity = intensity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
WallPaper WallPaper::withBlurred(bool blurred) const {
|
||||||
|
using Flag = MTPDwallPaperSettings::Flag;
|
||||||
|
|
||||||
|
auto result = *this;
|
||||||
|
if (blurred) {
|
||||||
|
result._settings |= Flag::f_blur;
|
||||||
|
} else {
|
||||||
|
result._settings &= ~Flag::f_blur;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
WallPaper WallPaper::withPatternIntensity(int intensity) const {
|
||||||
|
using Flag = MTPDwallPaperSettings::Flag;
|
||||||
|
|
||||||
|
auto result = *this;
|
||||||
|
result._settings |= Flag::f_intensity;
|
||||||
|
result._intensity = intensity;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
WallPaper WallPaper::withBackgroundColor(QColor color) const {
|
||||||
|
using Flag = MTPDwallPaperSettings::Flag;
|
||||||
|
|
||||||
|
auto result = *this;
|
||||||
|
result._settings |= Flag::f_background_color;
|
||||||
|
result._backgroundColor = color;
|
||||||
|
if (ColorFromString(_slug)) {
|
||||||
|
result._slug = StringFromColor(color);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
WallPaper WallPaper::withParamsFrom(const WallPaper &other) const {
|
||||||
|
auto result = *this;
|
||||||
|
result._settings = other._settings;
|
||||||
|
if (other._backgroundColor || !ColorFromString(_slug)) {
|
||||||
|
result._backgroundColor = other._backgroundColor;
|
||||||
|
if (ColorFromString(_slug)) {
|
||||||
|
result._slug = StringFromColor(*result._backgroundColor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
result._intensity = other._intensity;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
WallPaper WallPaper::withoutImageData() const {
|
||||||
|
auto result = *this;
|
||||||
|
result._thumbnail = nullptr;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<WallPaper> WallPaper::Create(const MTPWallPaper &data) {
|
||||||
|
return data.match([](const MTPDwallPaper &data) {
|
||||||
|
return Create(data);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<WallPaper> WallPaper::Create(const MTPDwallPaper &data) {
|
||||||
|
using Flag = MTPDwallPaper::Flag;
|
||||||
|
|
||||||
|
const auto document = Auth().data().processDocument(
|
||||||
|
data.vdocument);
|
||||||
|
if (!document->checkWallPaperProperties()) {
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
auto result = WallPaper(data.vid.v);
|
||||||
|
result._accessHash = data.vaccess_hash.v;
|
||||||
|
result._flags = data.vflags.v;
|
||||||
|
result._slug = qs(data.vslug);
|
||||||
|
result._document = document;
|
||||||
|
if (data.has_settings()) {
|
||||||
|
const auto isPattern = ((result._flags & Flag::f_pattern) != 0);
|
||||||
|
data.vsettings.match([&](const MTPDwallPaperSettings &data) {
|
||||||
|
using Flag = MTPDwallPaperSettings::Flag;
|
||||||
|
|
||||||
|
result._settings = data.vflags.v;
|
||||||
|
if (isPattern && data.has_background_color()) {
|
||||||
|
result._backgroundColor = MaybeColorFromSerialized(
|
||||||
|
data.vbackground_color.v);
|
||||||
|
} else {
|
||||||
|
result._settings &= ~Flag::f_background_color;
|
||||||
|
}
|
||||||
|
if (isPattern && data.has_intensity()) {
|
||||||
|
result._intensity = data.vintensity.v;
|
||||||
|
} else {
|
||||||
|
result._settings &= ~Flag::f_intensity;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
QByteArray WallPaper::serialize() const {
|
||||||
|
auto size = sizeof(quint64) // _id
|
||||||
|
+ sizeof(quint64) // _accessHash
|
||||||
|
+ sizeof(qint32) // _flags
|
||||||
|
+ Serialize::stringSize(_slug)
|
||||||
|
+ sizeof(qint32) // _settings
|
||||||
|
+ sizeof(quint32) // _backgroundColor
|
||||||
|
+ sizeof(qint32); // _intensity
|
||||||
|
|
||||||
|
auto result = QByteArray();
|
||||||
|
result.reserve(size);
|
||||||
|
{
|
||||||
|
auto stream = QDataStream(&result, QIODevice::WriteOnly);
|
||||||
|
stream.setVersion(QDataStream::Qt_5_1);
|
||||||
|
stream
|
||||||
|
<< quint64(_id)
|
||||||
|
<< quint64(_accessHash)
|
||||||
|
<< qint32(_flags)
|
||||||
|
<< _slug
|
||||||
|
<< qint32(_settings)
|
||||||
|
<< SerializeMaybeColor(_backgroundColor)
|
||||||
|
<< qint32(_intensity);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<WallPaper> WallPaper::FromSerialized(
|
||||||
|
const QByteArray &serialized) {
|
||||||
|
if (serialized.isEmpty()) {
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto id = quint64();
|
||||||
|
auto accessHash = quint64();
|
||||||
|
auto flags = qint32();
|
||||||
|
auto slug = QString();
|
||||||
|
auto settings = qint32();
|
||||||
|
auto backgroundColor = quint32();
|
||||||
|
auto intensity = qint32();
|
||||||
|
|
||||||
|
auto stream = QDataStream(serialized);
|
||||||
|
stream.setVersion(QDataStream::Qt_5_1);
|
||||||
|
stream
|
||||||
|
>> id
|
||||||
|
>> accessHash
|
||||||
|
>> flags
|
||||||
|
>> slug
|
||||||
|
>> settings
|
||||||
|
>> backgroundColor
|
||||||
|
>> intensity;
|
||||||
|
if (stream.status() != QDataStream::Ok) {
|
||||||
|
return std::nullopt;
|
||||||
|
} else if (intensity < 0 || intensity > 100) {
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
auto result = WallPaper(id);
|
||||||
|
result._accessHash = accessHash;
|
||||||
|
result._flags = MTPDwallPaper::Flags::from_raw(flags);
|
||||||
|
result._slug = slug;
|
||||||
|
result._settings = MTPDwallPaperSettings::Flags::from_raw(settings);
|
||||||
|
result._backgroundColor = MaybeColorFromSerialized(backgroundColor);
|
||||||
|
result._intensity = intensity;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<WallPaper> WallPaper::FromLegacySerialized(
|
||||||
|
quint64 id,
|
||||||
|
quint64 accessHash,
|
||||||
|
quint32 flags,
|
||||||
|
QString slug) {
|
||||||
|
auto result = WallPaper(id);
|
||||||
|
result._accessHash = accessHash;
|
||||||
|
result._flags = MTPDwallPaper::Flags::from_raw(flags);
|
||||||
|
result._slug = slug;
|
||||||
|
result._backgroundColor = ColorFromString(slug);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<WallPaper> WallPaper::FromLegacyId(qint32 legacyId) {
|
||||||
|
auto result = WallPaper(FromLegacyBackgroundId(legacyId));
|
||||||
|
if (!IsCustomWallPaper(result)) {
|
||||||
|
result._flags = MTPDwallPaper::Flag::f_default;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<WallPaper> WallPaper::FromColorSlug(const QString &slug) {
|
||||||
|
if (const auto color = ColorFromString(slug)) {
|
||||||
|
auto result = CustomWallPaper();
|
||||||
|
result._slug = slug;
|
||||||
|
result._backgroundColor = color;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
WallPaper ThemeWallPaper() {
|
||||||
|
return WallPaper(kThemeBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsThemeWallPaper(const WallPaper &paper) {
|
||||||
|
return (paper.id() == kThemeBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
WallPaper CustomWallPaper() {
|
||||||
|
return WallPaper(kCustomBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsCustomWallPaper(const WallPaper &paper) {
|
||||||
|
return (paper.id() == kCustomBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
WallPaper Legacy1DefaultWallPaper() {
|
||||||
|
return WallPaper(kLegacy1DefaultBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsLegacy1DefaultWallPaper(const WallPaper &paper) {
|
||||||
|
return (paper.id() == kLegacy1DefaultBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
WallPaper DefaultWallPaper() {
|
||||||
|
return WallPaper(kDefaultBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsDefaultWallPaper(const WallPaper &paper) {
|
||||||
|
return (paper.id() == kDefaultBackground)
|
||||||
|
|| (paper.id() == kIncorrectDefaultBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
QColor PatternColor(QColor background) {
|
||||||
|
const auto hue = background.hueF();
|
||||||
|
const auto saturation = background.saturationF();
|
||||||
|
const auto value = background.valueF();
|
||||||
|
return QColor::fromHsvF(
|
||||||
|
hue,
|
||||||
|
std::min(1.0, saturation + 0.05 + 0.1 * (1. - saturation)),
|
||||||
|
(value > 0.5
|
||||||
|
? std::max(0., value * 0.65)
|
||||||
|
: std::max(0., std::min(1., 1. - value * 0.65))),
|
||||||
|
0.4
|
||||||
|
).toRgb();
|
||||||
|
}
|
||||||
|
|
||||||
|
QImage PreparePatternImage(
|
||||||
|
QImage image,
|
||||||
|
QColor bg,
|
||||||
|
QColor fg,
|
||||||
|
int intensity) {
|
||||||
|
if (image.format() != QImage::Format_ARGB32_Premultiplied) {
|
||||||
|
image = std::move(image).convertToFormat(
|
||||||
|
QImage::Format_ARGB32_Premultiplied);
|
||||||
|
}
|
||||||
|
// Similar to ColorizePattern.
|
||||||
|
// But here we set bg to all 'alpha=0' pixels and fg to opaque ones.
|
||||||
|
|
||||||
|
const auto width = image.width();
|
||||||
|
const auto height = image.height();
|
||||||
|
const auto alpha = anim::interpolate(
|
||||||
|
0,
|
||||||
|
255,
|
||||||
|
fg.alphaF() * std::clamp(intensity / 100., 0., 1.));
|
||||||
|
if (!alpha) {
|
||||||
|
image.fill(bg);
|
||||||
|
return image;
|
||||||
|
}
|
||||||
|
fg.setAlpha(255);
|
||||||
|
const auto patternBg = anim::shifted(bg);
|
||||||
|
const auto patternFg = anim::shifted(fg);
|
||||||
|
|
||||||
|
const auto resultBytesPerPixel = (image.depth() >> 3);
|
||||||
|
constexpr auto resultIntsPerPixel = 1;
|
||||||
|
const auto resultIntsPerLine = (image.bytesPerLine() >> 2);
|
||||||
|
const auto resultIntsAdded = resultIntsPerLine - width * resultIntsPerPixel;
|
||||||
|
auto resultInts = reinterpret_cast<uint32*>(image.bits());
|
||||||
|
Assert(resultIntsAdded >= 0);
|
||||||
|
Assert(image.depth() == static_cast<int>((resultIntsPerPixel * sizeof(uint32)) << 3));
|
||||||
|
Assert(image.bytesPerLine() == (resultIntsPerLine << 2));
|
||||||
|
|
||||||
|
const auto maskBytesPerPixel = (image.depth() >> 3);
|
||||||
|
const auto maskBytesPerLine = image.bytesPerLine();
|
||||||
|
const auto maskBytesAdded = maskBytesPerLine - width * maskBytesPerPixel;
|
||||||
|
|
||||||
|
// We want to read the last byte of four available.
|
||||||
|
// This is the difference with style::colorizeImage.
|
||||||
|
auto maskBytes = image.constBits() + (maskBytesPerPixel - 1);
|
||||||
|
Assert(maskBytesAdded >= 0);
|
||||||
|
Assert(image.depth() == (maskBytesPerPixel << 3));
|
||||||
|
for (auto y = 0; y != height; ++y) {
|
||||||
|
for (auto x = 0; x != width; ++x) {
|
||||||
|
const auto maskOpacity = static_cast<anim::ShiftedMultiplier>(
|
||||||
|
*maskBytes) + 1;
|
||||||
|
const auto fgOpacity = (maskOpacity * alpha) >> 8;
|
||||||
|
const auto bgOpacity = 256 - fgOpacity;
|
||||||
|
*resultInts = anim::unshifted(
|
||||||
|
patternBg * bgOpacity + patternFg * fgOpacity);
|
||||||
|
maskBytes += maskBytesPerPixel;
|
||||||
|
resultInts += resultIntsPerPixel;
|
||||||
|
}
|
||||||
|
maskBytes += maskBytesAdded;
|
||||||
|
resultInts += resultIntsAdded;
|
||||||
|
}
|
||||||
|
return image;
|
||||||
|
}
|
||||||
|
|
||||||
|
QImage PrepareBlurredBackground(QImage image) {
|
||||||
|
constexpr auto kSize = 900;
|
||||||
|
constexpr auto kRadius = 24;
|
||||||
|
if (image.width() > kSize || image.height() > kSize) {
|
||||||
|
image = image.scaled(
|
||||||
|
kSize,
|
||||||
|
kSize,
|
||||||
|
Qt::KeepAspectRatio,
|
||||||
|
Qt::SmoothTransformation);
|
||||||
|
}
|
||||||
|
return Images::BlurLargeImage(image, kRadius);
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace details {
|
||||||
|
|
||||||
|
WallPaper UninitializedWallPaper() {
|
||||||
|
return WallPaper(kUninitializedBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsUninitializedWallPaper(const WallPaper &paper) {
|
||||||
|
return (paper.id() == kUninitializedBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
WallPaper TestingThemeWallPaper() {
|
||||||
|
return WallPaper(kTestingThemeBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsTestingThemeWallPaper(const WallPaper &paper) {
|
||||||
|
return (paper.id() == kTestingThemeBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
WallPaper TestingDefaultWallPaper() {
|
||||||
|
return WallPaper(kTestingDefaultBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsTestingDefaultWallPaper(const WallPaper &paper) {
|
||||||
|
return (paper.id() == kTestingDefaultBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
WallPaper TestingEditorWallPaper() {
|
||||||
|
return WallPaper(kTestingEditorBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsTestingEditorWallPaper(const WallPaper &paper) {
|
||||||
|
return (paper.id() == kTestingEditorBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace details
|
||||||
|
} // namespace Data
|
|
@ -0,0 +1,113 @@
|
||||||
|
/*
|
||||||
|
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
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
class Image;
|
||||||
|
|
||||||
|
namespace Data {
|
||||||
|
|
||||||
|
struct FileOrigin;
|
||||||
|
|
||||||
|
class WallPaper {
|
||||||
|
public:
|
||||||
|
explicit WallPaper(WallPaperId id);
|
||||||
|
|
||||||
|
void setLocalImageAsThumbnail(std::shared_ptr<Image> image);
|
||||||
|
|
||||||
|
[[nodiscard]] WallPaperId id() const;
|
||||||
|
[[nodiscard]] std::optional<QColor> backgroundColor() const;
|
||||||
|
[[nodiscard]] DocumentData *document() const;
|
||||||
|
[[nodiscard]] Image *thumbnail() const;
|
||||||
|
[[nodiscard]] bool isPattern() const;
|
||||||
|
[[nodiscard]] bool isDefault() const;
|
||||||
|
[[nodiscard]] bool isCreator() const;
|
||||||
|
[[nodiscard]] bool isDark() const;
|
||||||
|
[[nodiscard]] bool isLocal() const;
|
||||||
|
[[nodiscard]] bool isBlurred() const;
|
||||||
|
[[nodiscard]] int patternIntensity() const;
|
||||||
|
[[nodiscard]] bool hasShareUrl() const;
|
||||||
|
[[nodiscard]] QString shareUrl() const;
|
||||||
|
|
||||||
|
void loadDocument() const;
|
||||||
|
void loadThumbnail() const;
|
||||||
|
[[nodiscard]] FileOrigin fileOrigin() const;
|
||||||
|
[[nodiscard]] MTPWallPaperSettings mtpSettings() const;
|
||||||
|
|
||||||
|
[[nodiscard]] WallPaper withUrlParams(
|
||||||
|
const QMap<QString, QString> ¶ms) const;
|
||||||
|
[[nodiscard]] WallPaper withBlurred(bool blurred) const;
|
||||||
|
[[nodiscard]] WallPaper withPatternIntensity(int intensity) const;
|
||||||
|
[[nodiscard]] WallPaper withBackgroundColor(QColor color) const;
|
||||||
|
[[nodiscard]] WallPaper withParamsFrom(const WallPaper &other) const;
|
||||||
|
[[nodiscard]] WallPaper withoutImageData() const;
|
||||||
|
|
||||||
|
[[nodiscard]] static std::optional<WallPaper> Create(
|
||||||
|
const MTPWallPaper &data);
|
||||||
|
[[nodiscard]] static std::optional<WallPaper> Create(
|
||||||
|
const MTPDwallPaper &data);
|
||||||
|
|
||||||
|
[[nodiscard]] QByteArray serialize() const;
|
||||||
|
[[nodiscard]] static std::optional<WallPaper> FromSerialized(
|
||||||
|
const QByteArray &serialized);
|
||||||
|
[[nodiscard]] static std::optional<WallPaper> FromLegacySerialized(
|
||||||
|
quint64 id,
|
||||||
|
quint64 accessHash,
|
||||||
|
quint32 flags,
|
||||||
|
QString slug);
|
||||||
|
[[nodiscard]] static std::optional<WallPaper> FromLegacyId(
|
||||||
|
qint32 legacyId);
|
||||||
|
[[nodiscard]] static std::optional<WallPaper> FromColorSlug(
|
||||||
|
const QString &slug);
|
||||||
|
|
||||||
|
private:
|
||||||
|
static constexpr auto kDefaultIntensity = 40;
|
||||||
|
|
||||||
|
WallPaperId _id = WallPaperId();
|
||||||
|
uint64 _accessHash = 0;
|
||||||
|
MTPDwallPaper::Flags _flags;
|
||||||
|
QString _slug;
|
||||||
|
|
||||||
|
MTPDwallPaperSettings::Flags _settings;
|
||||||
|
std::optional<QColor> _backgroundColor;
|
||||||
|
int _intensity = kDefaultIntensity;
|
||||||
|
|
||||||
|
DocumentData *_document = nullptr;
|
||||||
|
std::shared_ptr<Image> _thumbnail;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
[[nodiscard]] WallPaper ThemeWallPaper();
|
||||||
|
[[nodiscard]] bool IsThemeWallPaper(const WallPaper &paper);
|
||||||
|
[[nodiscard]] WallPaper CustomWallPaper();
|
||||||
|
[[nodiscard]] bool IsCustomWallPaper(const WallPaper &paper);
|
||||||
|
[[nodiscard]] WallPaper Legacy1DefaultWallPaper();
|
||||||
|
[[nodiscard]] bool IsLegacy1DefaultWallPaper(const WallPaper &paper);
|
||||||
|
[[nodiscard]] WallPaper DefaultWallPaper();
|
||||||
|
[[nodiscard]] bool IsDefaultWallPaper(const WallPaper &paper);
|
||||||
|
|
||||||
|
QColor PatternColor(QColor background);
|
||||||
|
QImage PreparePatternImage(
|
||||||
|
QImage image,
|
||||||
|
QColor bg,
|
||||||
|
QColor fg,
|
||||||
|
int intensity);
|
||||||
|
QImage PrepareBlurredBackground(QImage image);
|
||||||
|
|
||||||
|
namespace details {
|
||||||
|
|
||||||
|
[[nodiscard]] WallPaper UninitializedWallPaper();
|
||||||
|
[[nodiscard]] bool IsUninitializedWallPaper(const WallPaper &paper);
|
||||||
|
[[nodiscard]] WallPaper TestingThemeWallPaper();
|
||||||
|
[[nodiscard]] bool IsTestingThemeWallPaper(const WallPaper &paper);
|
||||||
|
[[nodiscard]] WallPaper TestingDefaultWallPaper();
|
||||||
|
[[nodiscard]] bool IsTestingDefaultWallPaper(const WallPaper &paper);
|
||||||
|
[[nodiscard]] WallPaper TestingEditorWallPaper();
|
||||||
|
[[nodiscard]] bool IsTestingEditorWallPaper(const WallPaper &paper);
|
||||||
|
|
||||||
|
} // namespace details
|
||||||
|
} // namespace Data
|
|
@ -75,7 +75,9 @@ struct Uploader::File {
|
||||||
|
|
||||||
Uploader::File::File(const SendMediaReady &media) : media(media) {
|
Uploader::File::File(const SendMediaReady &media) : media(media) {
|
||||||
partsCount = media.parts.size();
|
partsCount = media.parts.size();
|
||||||
if (type() == SendMediaType::File || type() == SendMediaType::Audio) {
|
if (type() == SendMediaType::File
|
||||||
|
|| type() == SendMediaType::WallPaper
|
||||||
|
|| type() == SendMediaType::Audio) {
|
||||||
setDocSize(media.file.isEmpty()
|
setDocSize(media.file.isEmpty()
|
||||||
? media.data.size()
|
? media.data.size()
|
||||||
: media.filesize);
|
: media.filesize);
|
||||||
|
@ -89,7 +91,9 @@ Uploader::File::File(const std::shared_ptr<FileLoadResult> &file)
|
||||||
|| type() == SendMediaType::Secure)
|
|| type() == SendMediaType::Secure)
|
||||||
? file->fileparts.size()
|
? file->fileparts.size()
|
||||||
: file->thumbparts.size();
|
: file->thumbparts.size();
|
||||||
if (type() == SendMediaType::File || type() == SendMediaType::Audio) {
|
if (type() == SendMediaType::File
|
||||||
|
|| type() == SendMediaType::WallPaper
|
||||||
|
|| type() == SendMediaType::Audio) {
|
||||||
setDocSize(file->filesize);
|
setDocSize(file->filesize);
|
||||||
} else {
|
} else {
|
||||||
docSize = docPartSize = docPartsCount = 0;
|
docSize = docPartSize = docPartsCount = 0;
|
||||||
|
@ -149,6 +153,7 @@ void Uploader::uploadMedia(
|
||||||
if (media.type == SendMediaType::Photo) {
|
if (media.type == SendMediaType::Photo) {
|
||||||
Auth().data().processPhoto(media.photo, media.photoThumbs);
|
Auth().data().processPhoto(media.photo, media.photoThumbs);
|
||||||
} else if (media.type == SendMediaType::File
|
} else if (media.type == SendMediaType::File
|
||||||
|
|| media.type == SendMediaType::WallPaper
|
||||||
|| media.type == SendMediaType::Audio) {
|
|| media.type == SendMediaType::Audio) {
|
||||||
const auto document = media.photoThumbs.empty()
|
const auto document = media.photoThumbs.empty()
|
||||||
? Auth().data().processDocument(media.document)
|
? Auth().data().processDocument(media.document)
|
||||||
|
@ -157,6 +162,9 @@ void Uploader::uploadMedia(
|
||||||
base::duplicate(media.photoThumbs.front().second));
|
base::duplicate(media.photoThumbs.front().second));
|
||||||
if (!media.data.isEmpty()) {
|
if (!media.data.isEmpty()) {
|
||||||
document->setData(media.data);
|
document->setData(media.data);
|
||||||
|
if (media.type == SendMediaType::WallPaper) {
|
||||||
|
document->checkWallPaperProperties();
|
||||||
|
}
|
||||||
if (document->saveToCache()
|
if (document->saveToCache()
|
||||||
&& media.data.size() <= Storage::kMaxFileInMemory) {
|
&& media.data.size() <= Storage::kMaxFileInMemory) {
|
||||||
Auth().data().cache().put(
|
Auth().data().cache().put(
|
||||||
|
@ -184,6 +192,7 @@ void Uploader::upload(
|
||||||
photo->uploadingData = std::make_unique<Data::UploadState>(
|
photo->uploadingData = std::make_unique<Data::UploadState>(
|
||||||
file->partssize);
|
file->partssize);
|
||||||
} else if (file->type == SendMediaType::File
|
} else if (file->type == SendMediaType::File
|
||||||
|
|| file->type == SendMediaType::WallPaper
|
||||||
|| file->type == SendMediaType::Audio) {
|
|| file->type == SendMediaType::Audio) {
|
||||||
const auto document = file->thumb.isNull()
|
const auto document = file->thumb.isNull()
|
||||||
? Auth().data().processDocument(file->document)
|
? Auth().data().processDocument(file->document)
|
||||||
|
@ -197,6 +206,9 @@ void Uploader::upload(
|
||||||
std::move(file->goodThumbnailBytes));
|
std::move(file->goodThumbnailBytes));
|
||||||
if (!file->content.isEmpty()) {
|
if (!file->content.isEmpty()) {
|
||||||
document->setData(file->content);
|
document->setData(file->content);
|
||||||
|
if (file->type == SendMediaType::WallPaper) {
|
||||||
|
document->checkWallPaperProperties();
|
||||||
|
}
|
||||||
if (document->saveToCache()
|
if (document->saveToCache()
|
||||||
&& file->content.size() <= Storage::kMaxFileInMemory) {
|
&& file->content.size() <= Storage::kMaxFileInMemory) {
|
||||||
Auth().data().cache().put(
|
Auth().data().cache().put(
|
||||||
|
@ -220,6 +232,7 @@ void Uploader::currentFailed() {
|
||||||
if (j->second.type() == SendMediaType::Photo) {
|
if (j->second.type() == SendMediaType::Photo) {
|
||||||
_photoFailed.fire_copy(j->first);
|
_photoFailed.fire_copy(j->first);
|
||||||
} else if (j->second.type() == SendMediaType::File
|
} else if (j->second.type() == SendMediaType::File
|
||||||
|
|| j->second.type() == SendMediaType::WallPaper
|
||||||
|| j->second.type() == SendMediaType::Audio) {
|
|| j->second.type() == SendMediaType::Audio) {
|
||||||
const auto document = Auth().data().document(j->second.id());
|
const auto document = Auth().data().document(j->second.id());
|
||||||
if (document->uploading()) {
|
if (document->uploading()) {
|
||||||
|
@ -318,6 +331,7 @@ void Uploader::sendNext() {
|
||||||
MTP_bytes(md5));
|
MTP_bytes(md5));
|
||||||
_photoReady.fire({ uploadingId, silent, file });
|
_photoReady.fire({ uploadingId, silent, file });
|
||||||
} else if (uploadingData.type() == SendMediaType::File
|
} else if (uploadingData.type() == SendMediaType::File
|
||||||
|
|| uploadingData.type() == SendMediaType::WallPaper
|
||||||
|| uploadingData.type() == SendMediaType::Audio) {
|
|| uploadingData.type() == SendMediaType::Audio) {
|
||||||
QByteArray docMd5(32, Qt::Uninitialized);
|
QByteArray docMd5(32, Qt::Uninitialized);
|
||||||
hashMd5Hex(uploadingData.md5Hash.result(), docMd5.data());
|
hashMd5Hex(uploadingData.md5Hash.result(), docMd5.data());
|
||||||
|
@ -389,6 +403,7 @@ void Uploader::sendNext() {
|
||||||
* uploadingData.docPartSize;
|
* uploadingData.docPartSize;
|
||||||
toSend = content.mid(offset, uploadingData.docPartSize);
|
toSend = content.mid(offset, uploadingData.docPartSize);
|
||||||
if ((uploadingData.type() == SendMediaType::File
|
if ((uploadingData.type() == SendMediaType::File
|
||||||
|
|| uploadingData.type() == SendMediaType::WallPaper
|
||||||
|| uploadingData.type() == SendMediaType::Audio)
|
|| uploadingData.type() == SendMediaType::Audio)
|
||||||
&& uploadingData.docSentParts <= kUseBigFilesFrom) {
|
&& uploadingData.docSentParts <= kUseBigFilesFrom) {
|
||||||
uploadingData.md5Hash.feed(toSend.constData(), toSend.size());
|
uploadingData.md5Hash.feed(toSend.constData(), toSend.size());
|
||||||
|
@ -530,6 +545,7 @@ void Uploader::partLoaded(const MTPBool &result, mtpRequestId requestId) {
|
||||||
}
|
}
|
||||||
_photoProgress.fire_copy(fullId);
|
_photoProgress.fire_copy(fullId);
|
||||||
} else if (file.type() == SendMediaType::File
|
} else if (file.type() == SendMediaType::File
|
||||||
|
|| file.type() == SendMediaType::WallPaper
|
||||||
|| file.type() == SendMediaType::Audio) {
|
|| file.type() == SendMediaType::Audio) {
|
||||||
const auto document = Auth().data().document(file.id());
|
const auto document = Auth().data().document(file.id());
|
||||||
if (document->uploading()) {
|
if (document->uploading()) {
|
||||||
|
|
|
@ -234,6 +234,71 @@ SendMediaReady PreparePeerPhoto(PeerId peerId, QImage &&image) {
|
||||||
0);
|
0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SendMediaReady PrepareWallPaper(const QImage &image) {
|
||||||
|
PreparedPhotoThumbs thumbnails;
|
||||||
|
QVector<MTPPhotoSize> sizes;
|
||||||
|
|
||||||
|
QByteArray jpeg;
|
||||||
|
QBuffer jpegBuffer(&jpeg);
|
||||||
|
image.save(&jpegBuffer, "JPG", 87);
|
||||||
|
|
||||||
|
const auto scaled = [&](int size) {
|
||||||
|
return image.scaled(
|
||||||
|
size,
|
||||||
|
size,
|
||||||
|
Qt::KeepAspectRatio,
|
||||||
|
Qt::SmoothTransformation);
|
||||||
|
};
|
||||||
|
const auto push = [&](const char *type, QImage &&image) {
|
||||||
|
sizes.push_back(MTP_photoSize(
|
||||||
|
MTP_string(type),
|
||||||
|
MTP_fileLocationUnavailable(
|
||||||
|
MTP_long(0),
|
||||||
|
MTP_int(0),
|
||||||
|
MTP_long(0)),
|
||||||
|
MTP_int(image.width()),
|
||||||
|
MTP_int(image.height()), MTP_int(0)));
|
||||||
|
thumbnails.emplace(type[0], std::move(image));
|
||||||
|
};
|
||||||
|
push("s", scaled(320));
|
||||||
|
|
||||||
|
const auto filename = qsl("wallpaper.jpg");
|
||||||
|
auto attributes = QVector<MTPDocumentAttribute>(
|
||||||
|
1,
|
||||||
|
MTP_documentAttributeFilename(MTP_string(filename)));
|
||||||
|
attributes.push_back(MTP_documentAttributeImageSize(
|
||||||
|
MTP_int(image.width()),
|
||||||
|
MTP_int(image.height())));
|
||||||
|
const auto id = rand_value<DocumentId>();
|
||||||
|
const auto document = MTP_document(
|
||||||
|
MTP_flags(0),
|
||||||
|
MTP_long(id),
|
||||||
|
MTP_long(0),
|
||||||
|
MTP_bytes(QByteArray()),
|
||||||
|
MTP_int(unixtime()),
|
||||||
|
MTP_string("image/jpeg"),
|
||||||
|
MTP_int(jpeg.size()),
|
||||||
|
MTP_vector<MTPPhotoSize>(sizes),
|
||||||
|
MTP_int(MTP::maindc()),
|
||||||
|
MTP_vector<MTPDocumentAttribute>(attributes));
|
||||||
|
|
||||||
|
return SendMediaReady(
|
||||||
|
SendMediaType::WallPaper,
|
||||||
|
QString(), // filepath
|
||||||
|
filename,
|
||||||
|
jpeg.size(),
|
||||||
|
jpeg,
|
||||||
|
id,
|
||||||
|
0,
|
||||||
|
QString(),
|
||||||
|
PeerId(),
|
||||||
|
MTP_photoEmpty(MTP_long(0)),
|
||||||
|
thumbnails,
|
||||||
|
document,
|
||||||
|
QByteArray(),
|
||||||
|
0);
|
||||||
|
}
|
||||||
|
|
||||||
TaskQueue::TaskQueue(TimeMs stopTimeoutMs) {
|
TaskQueue::TaskQueue(TimeMs stopTimeoutMs) {
|
||||||
if (stopTimeoutMs > 0) {
|
if (stopTimeoutMs > 0) {
|
||||||
_stopTimer = new QTimer(this);
|
_stopTimer = new QTimer(this);
|
||||||
|
|
|
@ -20,6 +20,7 @@ enum class SendMediaType {
|
||||||
Photo,
|
Photo,
|
||||||
Audio,
|
Audio,
|
||||||
File,
|
File,
|
||||||
|
WallPaper,
|
||||||
Secure,
|
Secure,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -83,6 +84,7 @@ struct SendMediaReady {
|
||||||
};
|
};
|
||||||
|
|
||||||
SendMediaReady PreparePeerPhoto(PeerId peerId, QImage &&image);
|
SendMediaReady PreparePeerPhoto(PeerId peerId, QImage &&image);
|
||||||
|
SendMediaReady PrepareWallPaper(const QImage &image);
|
||||||
|
|
||||||
using TaskId = void*; // no interface, just id
|
using TaskId = void*; // no interface, just id
|
||||||
|
|
||||||
|
|
|
@ -10,592 +10,19 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "window/themes/window_theme_preview.h"
|
#include "window/themes/window_theme_preview.h"
|
||||||
#include "mainwidget.h"
|
#include "mainwidget.h"
|
||||||
#include "auth_session.h"
|
#include "auth_session.h"
|
||||||
#include "core/application.h"
|
#include "apiwrap.h"
|
||||||
#include "storage/serialize_common.h"
|
|
||||||
#include "data/data_document.h"
|
|
||||||
#include "data/data_session.h"
|
|
||||||
#include "storage/localstorage.h"
|
#include "storage/localstorage.h"
|
||||||
|
#include "storage/localimageloader.h"
|
||||||
|
#include "storage/file_upload.h"
|
||||||
#include "base/parse_helper.h"
|
#include "base/parse_helper.h"
|
||||||
#include "base/zlib_help.h"
|
#include "base/zlib_help.h"
|
||||||
|
#include "data/data_session.h"
|
||||||
#include "ui/image/image.h"
|
#include "ui/image/image.h"
|
||||||
#include "boxes/background_box.h"
|
#include "boxes/background_box.h"
|
||||||
|
#include "core/application.h"
|
||||||
#include "styles/style_widgets.h"
|
#include "styles/style_widgets.h"
|
||||||
#include "styles/style_history.h"
|
#include "styles/style_history.h"
|
||||||
|
|
||||||
namespace Data {
|
|
||||||
namespace {
|
|
||||||
|
|
||||||
constexpr auto FromLegacyBackgroundId(int32 legacyId) -> WallPaperId {
|
|
||||||
return uint64(0xFFFFFFFF00000000ULL) | uint64(uint32(legacyId));
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr auto kUninitializedBackground = FromLegacyBackgroundId(-999);
|
|
||||||
constexpr auto kTestingThemeBackground = FromLegacyBackgroundId(-666);
|
|
||||||
constexpr auto kTestingDefaultBackground = FromLegacyBackgroundId(-665);
|
|
||||||
constexpr auto kTestingEditorBackground = FromLegacyBackgroundId(-664);
|
|
||||||
constexpr auto kThemeBackground = FromLegacyBackgroundId(-2);
|
|
||||||
constexpr auto kCustomBackground = FromLegacyBackgroundId(-1);
|
|
||||||
constexpr auto kLegacy1DefaultBackground = FromLegacyBackgroundId(0);
|
|
||||||
constexpr auto kDefaultBackground = 5947530738516623361;
|
|
||||||
constexpr auto kIncorrectDefaultBackground = FromLegacyBackgroundId(105);
|
|
||||||
|
|
||||||
quint32 SerializeMaybeColor(std::optional<QColor> color) {
|
|
||||||
return color
|
|
||||||
? ((quint32(std::clamp(color->red(), 0, 255)) << 16)
|
|
||||||
| (quint32(std::clamp(color->green(), 0, 255)) << 8)
|
|
||||||
| quint32(std::clamp(color->blue(), 0, 255)))
|
|
||||||
: quint32(-1);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::optional<QColor> MaybeColorFromSerialized(quint32 serialized) {
|
|
||||||
return (serialized == quint32(-1))
|
|
||||||
? std::nullopt
|
|
||||||
: std::make_optional(QColor(
|
|
||||||
int((serialized >> 16) & 0xFFU),
|
|
||||||
int((serialized >> 8) & 0xFFU),
|
|
||||||
int(serialized & 0xFFU)));
|
|
||||||
}
|
|
||||||
|
|
||||||
std::optional<QColor> ColorFromString(const QString &string) {
|
|
||||||
if (string.size() != 6) {
|
|
||||||
return {};
|
|
||||||
} else if (ranges::find_if(string, [](QChar ch) {
|
|
||||||
return (ch < 'a' || ch > 'f')
|
|
||||||
&& (ch < 'A' || ch > 'F')
|
|
||||||
&& (ch < '0' || ch > '9');
|
|
||||||
}) != string.end()) {
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
const auto component = [](const QString &text, int index) {
|
|
||||||
const auto decimal = [](QChar hex) {
|
|
||||||
const auto code = hex.unicode();
|
|
||||||
return (code >= '0' && code <= '9')
|
|
||||||
? int(code - '0')
|
|
||||||
: (code >= 'a' && code <= 'f')
|
|
||||||
? int(code - 'a' + 0x0a)
|
|
||||||
: int(code - 'A' + 0x0a);
|
|
||||||
};
|
|
||||||
index *= 2;
|
|
||||||
return decimal(text[index]) * 0x10 + decimal(text[index + 1]);
|
|
||||||
};
|
|
||||||
return QColor(
|
|
||||||
component(string, 0),
|
|
||||||
component(string, 1),
|
|
||||||
component(string, 2),
|
|
||||||
255);
|
|
||||||
}
|
|
||||||
|
|
||||||
QString StringFromColor(QColor color) {
|
|
||||||
const auto component = [](int value) {
|
|
||||||
const auto hex = [](int value) {
|
|
||||||
value = std::clamp(value, 0, 15);
|
|
||||||
return (value > 9)
|
|
||||||
? ('a' + (value - 10))
|
|
||||||
: ('0' + value);
|
|
||||||
};
|
|
||||||
return QString() + hex(value / 16) + hex(value % 16);
|
|
||||||
};
|
|
||||||
return component(color.red())
|
|
||||||
+ component(color.green())
|
|
||||||
+ component(color.blue());
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace
|
|
||||||
|
|
||||||
WallPaper::WallPaper(WallPaperId id) : _id(id) {
|
|
||||||
}
|
|
||||||
|
|
||||||
void WallPaper::setLocalImageAsThumbnail(std::shared_ptr<Image> image) {
|
|
||||||
Expects(IsDefaultWallPaper(*this)
|
|
||||||
|| IsLegacy1DefaultWallPaper(*this)
|
|
||||||
|| IsCustomWallPaper(*this));
|
|
||||||
Expects(_thumbnail == nullptr);
|
|
||||||
|
|
||||||
_thumbnail = std::move(image);
|
|
||||||
}
|
|
||||||
|
|
||||||
WallPaperId WallPaper::id() const {
|
|
||||||
return _id;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::optional<QColor> WallPaper::backgroundColor() const {
|
|
||||||
return _backgroundColor;
|
|
||||||
}
|
|
||||||
|
|
||||||
DocumentData *WallPaper::document() const {
|
|
||||||
return _document;
|
|
||||||
}
|
|
||||||
|
|
||||||
Image *WallPaper::thumbnail() const {
|
|
||||||
return _thumbnail
|
|
||||||
? _thumbnail.get()
|
|
||||||
: _document
|
|
||||||
? _document->thumbnail()
|
|
||||||
: nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool WallPaper::isPattern() const {
|
|
||||||
return _flags & MTPDwallPaper::Flag::f_pattern;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool WallPaper::isDefault() const {
|
|
||||||
return _flags & MTPDwallPaper::Flag::f_default;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool WallPaper::isCreator() const {
|
|
||||||
return _flags & MTPDwallPaper::Flag::f_creator;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool WallPaper::isDark() const {
|
|
||||||
return _flags & MTPDwallPaper::Flag::f_dark;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool WallPaper::isLocal() const {
|
|
||||||
return !document() && thumbnail();
|
|
||||||
}
|
|
||||||
|
|
||||||
bool WallPaper::isBlurred() const {
|
|
||||||
return _settings & MTPDwallPaperSettings::Flag::f_blur;
|
|
||||||
}
|
|
||||||
|
|
||||||
int WallPaper::patternIntensity() const {
|
|
||||||
return _intensity;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool WallPaper::hasShareUrl() const {
|
|
||||||
return !_slug.isEmpty();
|
|
||||||
}
|
|
||||||
|
|
||||||
QString WallPaper::shareUrl() const {
|
|
||||||
if (!hasShareUrl()) {
|
|
||||||
return QString();
|
|
||||||
}
|
|
||||||
const auto base = Core::App().createInternalLinkFull("bg/" + _slug);
|
|
||||||
auto params = QStringList();
|
|
||||||
if (isPattern()) {
|
|
||||||
if (_backgroundColor) {
|
|
||||||
params.push_back("bg_color=" + StringFromColor(*_backgroundColor));
|
|
||||||
}
|
|
||||||
if (_intensity) {
|
|
||||||
params.push_back("intensity=" + QString::number(_intensity));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
auto mode = QStringList();
|
|
||||||
if (_settings & MTPDwallPaperSettings::Flag::f_blur) {
|
|
||||||
mode.push_back("blur");
|
|
||||||
}
|
|
||||||
if (_settings & MTPDwallPaperSettings::Flag::f_motion) {
|
|
||||||
mode.push_back("motion");
|
|
||||||
}
|
|
||||||
if (!mode.isEmpty()) {
|
|
||||||
params.push_back("mode=" + mode.join('+'));
|
|
||||||
}
|
|
||||||
return params.isEmpty()
|
|
||||||
? base
|
|
||||||
: base + '?' + params.join('&');
|
|
||||||
}
|
|
||||||
|
|
||||||
void WallPaper::loadThumbnail() const {
|
|
||||||
if (_thumbnail) {
|
|
||||||
_thumbnail->load(fileOrigin());
|
|
||||||
}
|
|
||||||
if (_document) {
|
|
||||||
_document->loadThumbnail(fileOrigin());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void WallPaper::loadDocument() const {
|
|
||||||
if (_document) {
|
|
||||||
_document->save(fileOrigin(), QString());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
FileOrigin WallPaper::fileOrigin() const {
|
|
||||||
return FileOriginWallpaper(_id, _accessHash);
|
|
||||||
}
|
|
||||||
|
|
||||||
WallPaper WallPaper::withUrlParams(
|
|
||||||
const QMap<QString, QString> ¶ms) const {
|
|
||||||
using Flag = MTPDwallPaperSettings::Flag;
|
|
||||||
|
|
||||||
auto result = *this;
|
|
||||||
result._settings = Flag(0);
|
|
||||||
result._backgroundColor = ColorFromString(_slug);
|
|
||||||
result._intensity = kDefaultIntensity;
|
|
||||||
|
|
||||||
if (auto mode = params.value("mode"); !mode.isEmpty()) {
|
|
||||||
const auto list = mode.replace('+', ' ').split(' ');
|
|
||||||
for (const auto &change : list) {
|
|
||||||
if (change == qstr("blur")) {
|
|
||||||
result._settings |= Flag::f_blur;
|
|
||||||
} else if (change == qstr("motion")) {
|
|
||||||
result._settings |= Flag::f_motion;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (const auto color = ColorFromString(params.value("bg_color"))) {
|
|
||||||
result._settings |= Flag::f_background_color;
|
|
||||||
result._backgroundColor = color;
|
|
||||||
}
|
|
||||||
if (const auto string = params.value("intensity"); !string.isEmpty()) {
|
|
||||||
auto ok = false;
|
|
||||||
const auto intensity = string.toInt(&ok);
|
|
||||||
if (ok && base::in_range(intensity, 0, 101)) {
|
|
||||||
result._settings |= Flag::f_intensity;
|
|
||||||
result._intensity = intensity;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
WallPaper WallPaper::withBlurred(bool blurred) const {
|
|
||||||
using Flag = MTPDwallPaperSettings::Flag;
|
|
||||||
|
|
||||||
auto result = *this;
|
|
||||||
if (blurred) {
|
|
||||||
result._settings |= Flag::f_blur;
|
|
||||||
} else {
|
|
||||||
result._settings &= ~Flag::f_blur;
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
WallPaper WallPaper::withPatternIntensity(int intensity) const {
|
|
||||||
using Flag = MTPDwallPaperSettings::Flag;
|
|
||||||
|
|
||||||
auto result = *this;
|
|
||||||
result._settings |= Flag::f_intensity;
|
|
||||||
result._intensity = intensity;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
WallPaper WallPaper::withBackgroundColor(QColor color) const {
|
|
||||||
using Flag = MTPDwallPaperSettings::Flag;
|
|
||||||
|
|
||||||
auto result = *this;
|
|
||||||
result._settings |= Flag::f_background_color;
|
|
||||||
result._backgroundColor = color;
|
|
||||||
if (ColorFromString(_slug)) {
|
|
||||||
result._slug = StringFromColor(color);
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
WallPaper WallPaper::withParamsFrom(const WallPaper &other) const {
|
|
||||||
auto result = *this;
|
|
||||||
result._settings = other._settings;
|
|
||||||
if (other._backgroundColor || !ColorFromString(_slug)) {
|
|
||||||
result._backgroundColor = other._backgroundColor;
|
|
||||||
if (ColorFromString(_slug)) {
|
|
||||||
result._slug = StringFromColor(*result._backgroundColor);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
result._intensity = other._intensity;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
WallPaper WallPaper::withoutImageData() const {
|
|
||||||
auto result = *this;
|
|
||||||
result._thumbnail = nullptr;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::optional<WallPaper> WallPaper::Create(const MTPWallPaper &data) {
|
|
||||||
return data.match([](const MTPDwallPaper &data) {
|
|
||||||
return Create(data);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
std::optional<WallPaper> WallPaper::Create(const MTPDwallPaper &data) {
|
|
||||||
using Flag = MTPDwallPaper::Flag;
|
|
||||||
|
|
||||||
const auto document = Auth().data().processDocument(
|
|
||||||
data.vdocument);
|
|
||||||
if (!document->checkWallPaperProperties()) {
|
|
||||||
return std::nullopt;
|
|
||||||
}
|
|
||||||
auto result = WallPaper(data.vid.v);
|
|
||||||
result._accessHash = data.vaccess_hash.v;
|
|
||||||
result._flags = data.vflags.v;
|
|
||||||
result._slug = qs(data.vslug);
|
|
||||||
result._document = document;
|
|
||||||
if (data.has_settings()) {
|
|
||||||
const auto isPattern = ((result._flags & Flag::f_pattern) != 0);
|
|
||||||
data.vsettings.match([&](const MTPDwallPaperSettings &data) {
|
|
||||||
using Flag = MTPDwallPaperSettings::Flag;
|
|
||||||
|
|
||||||
result._settings = data.vflags.v;
|
|
||||||
if (isPattern && data.has_background_color()) {
|
|
||||||
result._backgroundColor = MaybeColorFromSerialized(
|
|
||||||
data.vbackground_color.v);
|
|
||||||
} else {
|
|
||||||
result._settings &= ~Flag::f_background_color;
|
|
||||||
}
|
|
||||||
if (isPattern && data.has_intensity()) {
|
|
||||||
result._intensity = data.vintensity.v;
|
|
||||||
} else {
|
|
||||||
result._settings &= ~Flag::f_intensity;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
QByteArray WallPaper::serialize() const {
|
|
||||||
auto size = sizeof(quint64) // _id
|
|
||||||
+ sizeof(quint64) // _accessHash
|
|
||||||
+ sizeof(qint32) // _flags
|
|
||||||
+ Serialize::stringSize(_slug)
|
|
||||||
+ sizeof(qint32) // _settings
|
|
||||||
+ sizeof(quint32) // _backgroundColor
|
|
||||||
+ sizeof(qint32); // _intensity
|
|
||||||
|
|
||||||
auto result = QByteArray();
|
|
||||||
result.reserve(size);
|
|
||||||
{
|
|
||||||
auto stream = QDataStream(&result, QIODevice::WriteOnly);
|
|
||||||
stream.setVersion(QDataStream::Qt_5_1);
|
|
||||||
stream
|
|
||||||
<< quint64(_id)
|
|
||||||
<< quint64(_accessHash)
|
|
||||||
<< qint32(_flags)
|
|
||||||
<< _slug
|
|
||||||
<< qint32(_settings)
|
|
||||||
<< SerializeMaybeColor(_backgroundColor)
|
|
||||||
<< qint32(_intensity);
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::optional<WallPaper> WallPaper::FromSerialized(
|
|
||||||
const QByteArray &serialized) {
|
|
||||||
if (serialized.isEmpty()) {
|
|
||||||
return std::nullopt;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto id = quint64();
|
|
||||||
auto accessHash = quint64();
|
|
||||||
auto flags = qint32();
|
|
||||||
auto slug = QString();
|
|
||||||
auto settings = qint32();
|
|
||||||
auto backgroundColor = quint32();
|
|
||||||
auto intensity = qint32();
|
|
||||||
|
|
||||||
auto stream = QDataStream(serialized);
|
|
||||||
stream.setVersion(QDataStream::Qt_5_1);
|
|
||||||
stream
|
|
||||||
>> id
|
|
||||||
>> accessHash
|
|
||||||
>> flags
|
|
||||||
>> slug
|
|
||||||
>> settings
|
|
||||||
>> backgroundColor
|
|
||||||
>> intensity;
|
|
||||||
if (stream.status() != QDataStream::Ok) {
|
|
||||||
return std::nullopt;
|
|
||||||
} else if (intensity < 0 || intensity > 100) {
|
|
||||||
return std::nullopt;
|
|
||||||
}
|
|
||||||
auto result = WallPaper(id);
|
|
||||||
result._accessHash = accessHash;
|
|
||||||
result._flags = MTPDwallPaper::Flags::from_raw(flags);
|
|
||||||
result._slug = slug;
|
|
||||||
result._settings = MTPDwallPaperSettings::Flags::from_raw(settings);
|
|
||||||
result._backgroundColor = MaybeColorFromSerialized(backgroundColor);
|
|
||||||
result._intensity = intensity;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::optional<WallPaper> WallPaper::FromLegacySerialized(
|
|
||||||
quint64 id,
|
|
||||||
quint64 accessHash,
|
|
||||||
quint32 flags,
|
|
||||||
QString slug) {
|
|
||||||
auto result = WallPaper(id);
|
|
||||||
result._accessHash = accessHash;
|
|
||||||
result._flags = MTPDwallPaper::Flags::from_raw(flags);
|
|
||||||
result._slug = slug;
|
|
||||||
result._backgroundColor = ColorFromString(slug);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::optional<WallPaper> WallPaper::FromLegacyId(qint32 legacyId) {
|
|
||||||
auto result = WallPaper(FromLegacyBackgroundId(legacyId));
|
|
||||||
if (!IsCustomWallPaper(result)) {
|
|
||||||
result._flags = MTPDwallPaper::Flag::f_default;
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::optional<WallPaper> WallPaper::FromColorSlug(const QString &slug) {
|
|
||||||
if (const auto color = ColorFromString(slug)) {
|
|
||||||
auto result = CustomWallPaper();
|
|
||||||
result._slug = slug;
|
|
||||||
result._backgroundColor = color;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
return std::nullopt;
|
|
||||||
}
|
|
||||||
|
|
||||||
WallPaper ThemeWallPaper() {
|
|
||||||
return WallPaper(kThemeBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IsThemeWallPaper(const WallPaper &paper) {
|
|
||||||
return (paper.id() == kThemeBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
WallPaper CustomWallPaper() {
|
|
||||||
return WallPaper(kCustomBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IsCustomWallPaper(const WallPaper &paper) {
|
|
||||||
return (paper.id() == kCustomBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
WallPaper Legacy1DefaultWallPaper() {
|
|
||||||
return WallPaper(kLegacy1DefaultBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IsLegacy1DefaultWallPaper(const WallPaper &paper) {
|
|
||||||
return (paper.id() == kLegacy1DefaultBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
WallPaper DefaultWallPaper() {
|
|
||||||
return WallPaper(kDefaultBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IsDefaultWallPaper(const WallPaper &paper) {
|
|
||||||
return (paper.id() == kDefaultBackground)
|
|
||||||
|| (paper.id() == kIncorrectDefaultBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
QColor PatternColor(QColor background) {
|
|
||||||
const auto hue = background.hueF();
|
|
||||||
const auto saturation = background.saturationF();
|
|
||||||
const auto value = background.valueF();
|
|
||||||
return QColor::fromHsvF(
|
|
||||||
hue,
|
|
||||||
std::min(1.0, saturation + 0.05 + 0.1 * (1. - saturation)),
|
|
||||||
(value > 0.5
|
|
||||||
? std::max(0., value * 0.65)
|
|
||||||
: std::max(0., std::min(1., 1. - value * 0.65))),
|
|
||||||
0.4
|
|
||||||
).toRgb();
|
|
||||||
}
|
|
||||||
|
|
||||||
QImage PreparePatternImage(
|
|
||||||
QImage image,
|
|
||||||
QColor bg,
|
|
||||||
QColor fg,
|
|
||||||
int intensity) {
|
|
||||||
if (image.format() != QImage::Format_ARGB32_Premultiplied) {
|
|
||||||
image = std::move(image).convertToFormat(
|
|
||||||
QImage::Format_ARGB32_Premultiplied);
|
|
||||||
}
|
|
||||||
// Similar to ColorizePattern.
|
|
||||||
// But here we set bg to all 'alpha=0' pixels and fg to opaque ones.
|
|
||||||
|
|
||||||
const auto width = image.width();
|
|
||||||
const auto height = image.height();
|
|
||||||
const auto alpha = anim::interpolate(
|
|
||||||
0,
|
|
||||||
255,
|
|
||||||
fg.alphaF() * std::clamp(intensity / 100., 0., 1.));
|
|
||||||
if (!alpha) {
|
|
||||||
image.fill(bg);
|
|
||||||
return image;
|
|
||||||
}
|
|
||||||
fg.setAlpha(255);
|
|
||||||
const auto patternBg = anim::shifted(bg);
|
|
||||||
const auto patternFg = anim::shifted(fg);
|
|
||||||
|
|
||||||
const auto resultBytesPerPixel = (image.depth() >> 3);
|
|
||||||
constexpr auto resultIntsPerPixel = 1;
|
|
||||||
const auto resultIntsPerLine = (image.bytesPerLine() >> 2);
|
|
||||||
const auto resultIntsAdded = resultIntsPerLine - width * resultIntsPerPixel;
|
|
||||||
auto resultInts = reinterpret_cast<uint32*>(image.bits());
|
|
||||||
Assert(resultIntsAdded >= 0);
|
|
||||||
Assert(image.depth() == static_cast<int>((resultIntsPerPixel * sizeof(uint32)) << 3));
|
|
||||||
Assert(image.bytesPerLine() == (resultIntsPerLine << 2));
|
|
||||||
|
|
||||||
const auto maskBytesPerPixel = (image.depth() >> 3);
|
|
||||||
const auto maskBytesPerLine = image.bytesPerLine();
|
|
||||||
const auto maskBytesAdded = maskBytesPerLine - width * maskBytesPerPixel;
|
|
||||||
|
|
||||||
// We want to read the last byte of four available.
|
|
||||||
// This is the difference with style::colorizeImage.
|
|
||||||
auto maskBytes = image.constBits() + (maskBytesPerPixel - 1);
|
|
||||||
Assert(maskBytesAdded >= 0);
|
|
||||||
Assert(image.depth() == (maskBytesPerPixel << 3));
|
|
||||||
for (auto y = 0; y != height; ++y) {
|
|
||||||
for (auto x = 0; x != width; ++x) {
|
|
||||||
const auto maskOpacity = static_cast<anim::ShiftedMultiplier>(
|
|
||||||
*maskBytes) + 1;
|
|
||||||
const auto fgOpacity = (maskOpacity * alpha) >> 8;
|
|
||||||
const auto bgOpacity = 256 - fgOpacity;
|
|
||||||
*resultInts = anim::unshifted(
|
|
||||||
patternBg * bgOpacity + patternFg * fgOpacity);
|
|
||||||
maskBytes += maskBytesPerPixel;
|
|
||||||
resultInts += resultIntsPerPixel;
|
|
||||||
}
|
|
||||||
maskBytes += maskBytesAdded;
|
|
||||||
resultInts += resultIntsAdded;
|
|
||||||
}
|
|
||||||
return image;
|
|
||||||
}
|
|
||||||
|
|
||||||
QImage PrepareBlurredBackground(QImage image) {
|
|
||||||
constexpr auto kSize = 900;
|
|
||||||
constexpr auto kRadius = 24;
|
|
||||||
if (image.width() > kSize || image.height() > kSize) {
|
|
||||||
image = image.scaled(
|
|
||||||
kSize,
|
|
||||||
kSize,
|
|
||||||
Qt::KeepAspectRatio,
|
|
||||||
Qt::SmoothTransformation);
|
|
||||||
}
|
|
||||||
return Images::BlurLargeImage(image, kRadius);
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace details {
|
|
||||||
|
|
||||||
WallPaper UninitializedWallPaper() {
|
|
||||||
return WallPaper(kUninitializedBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IsUninitializedWallPaper(const WallPaper &paper) {
|
|
||||||
return (paper.id() == kUninitializedBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
WallPaper TestingThemeWallPaper() {
|
|
||||||
return WallPaper(kTestingThemeBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IsTestingThemeWallPaper(const WallPaper &paper) {
|
|
||||||
return (paper.id() == kTestingThemeBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
WallPaper TestingDefaultWallPaper() {
|
|
||||||
return WallPaper(kTestingDefaultBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IsTestingDefaultWallPaper(const WallPaper &paper) {
|
|
||||||
return (paper.id() == kTestingDefaultBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
WallPaper TestingEditorWallPaper() {
|
|
||||||
return WallPaper(kTestingEditorBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IsTestingEditorWallPaper(const WallPaper &paper) {
|
|
||||||
return (paper.id() == kTestingEditorBackground);
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace details
|
|
||||||
} // namespace Data
|
|
||||||
|
|
||||||
namespace Window {
|
namespace Window {
|
||||||
namespace Theme {
|
namespace Theme {
|
||||||
namespace {
|
namespace {
|
||||||
|
@ -954,9 +381,71 @@ void ChatBackground::start() {
|
||||||
if (!Local::readBackground()) {
|
if (!Local::readBackground()) {
|
||||||
set(Data::ThemeWallPaper());
|
set(Data::ThemeWallPaper());
|
||||||
}
|
}
|
||||||
|
refreshSession();
|
||||||
|
subscribe(Core::App().authSessionChanged(), [=] {
|
||||||
|
refreshSession();
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ChatBackground::refreshSession() {
|
||||||
|
const auto session = AuthSession::Exists() ? &Auth() : nullptr;
|
||||||
|
if (_session != session) {
|
||||||
|
_session = session;
|
||||||
|
checkUploadWallPaper();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ChatBackground::checkUploadWallPaper() {
|
||||||
|
if (!_session) {
|
||||||
|
_wallPaperUploadLifetime = rpl::lifetime();
|
||||||
|
_wallPaperUploadId = FullMsgId();
|
||||||
|
_wallPaperRequestId = 0;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (const auto id = base::take(_wallPaperUploadId)) {
|
||||||
|
_session->uploader().cancel(id);
|
||||||
|
}
|
||||||
|
if (const auto id = base::take(_wallPaperRequestId)) {
|
||||||
|
_session->api().request(id).cancel();
|
||||||
|
}
|
||||||
|
if (!Data::IsCustomWallPaper(_paper) || _original.isNull()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto ready = PrepareWallPaper(_original);
|
||||||
|
const auto documentId = ready.id;
|
||||||
|
_wallPaperUploadId = FullMsgId(0, clientMsgId());
|
||||||
|
_session->uploader().uploadMedia(_wallPaperUploadId, ready);
|
||||||
|
if (_wallPaperUploadLifetime) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_wallPaperUploadLifetime = _session->uploader().documentReady(
|
||||||
|
) | rpl::start_with_next([=](const Storage::UploadedDocument &data) {
|
||||||
|
if (data.fullId != _wallPaperUploadId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_wallPaperUploadId = FullMsgId();
|
||||||
|
_wallPaperRequestId = _session->api().request(
|
||||||
|
MTPaccount_UploadWallPaper(
|
||||||
|
data.file,
|
||||||
|
MTP_string("image/jpeg"),
|
||||||
|
_paper.mtpSettings()
|
||||||
|
)
|
||||||
|
).done([=](const MTPWallPaper &result) {
|
||||||
|
result.match([&](const MTPDwallPaper &data) {
|
||||||
|
_session->data().documentConvert(
|
||||||
|
_session->data().document(documentId),
|
||||||
|
data.vdocument);
|
||||||
|
});
|
||||||
|
if (const auto paper = Data::WallPaper::Create(result)) {
|
||||||
|
_paper = *paper;
|
||||||
|
writeNewBackgroundSettings();
|
||||||
|
}
|
||||||
|
}).send();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
void ChatBackground::set(const Data::WallPaper &paper, QImage image) {
|
void ChatBackground::set(const Data::WallPaper &paper, QImage image) {
|
||||||
image = ProcessBackgroundImage(std::move(image));
|
image = ProcessBackgroundImage(std::move(image));
|
||||||
|
|
||||||
|
@ -1041,6 +530,7 @@ void ChatBackground::set(const Data::WallPaper &paper, QImage image) {
|
||||||
notify(BackgroundUpdate(BackgroundUpdate::Type::TestingTheme, tile()), true);
|
notify(BackgroundUpdate(BackgroundUpdate::Type::TestingTheme, tile()), true);
|
||||||
notify(BackgroundUpdate(BackgroundUpdate::Type::ApplyingTheme, tile()), true);
|
notify(BackgroundUpdate(BackgroundUpdate::Type::ApplyingTheme, tile()), true);
|
||||||
}
|
}
|
||||||
|
checkUploadWallPaper();
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChatBackground::setPreparedImage(QImage original, QImage prepared) {
|
void ChatBackground::setPreparedImage(QImage original, QImage prepared) {
|
||||||
|
|
|
@ -7,109 +7,9 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
*/
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
class Image;
|
#include "data/data_wall_paper.h"
|
||||||
|
|
||||||
namespace Data {
|
class AuthSession;
|
||||||
|
|
||||||
struct FileOrigin;
|
|
||||||
|
|
||||||
class WallPaper {
|
|
||||||
public:
|
|
||||||
explicit WallPaper(WallPaperId id);
|
|
||||||
|
|
||||||
void setLocalImageAsThumbnail(std::shared_ptr<Image> image);
|
|
||||||
|
|
||||||
[[nodiscard]] WallPaperId id() const;
|
|
||||||
[[nodiscard]] std::optional<QColor> backgroundColor() const;
|
|
||||||
[[nodiscard]] DocumentData *document() const;
|
|
||||||
[[nodiscard]] Image *thumbnail() const;
|
|
||||||
[[nodiscard]] bool isPattern() const;
|
|
||||||
[[nodiscard]] bool isDefault() const;
|
|
||||||
[[nodiscard]] bool isCreator() const;
|
|
||||||
[[nodiscard]] bool isDark() const;
|
|
||||||
[[nodiscard]] bool isLocal() const;
|
|
||||||
[[nodiscard]] bool isBlurred() const;
|
|
||||||
[[nodiscard]] int patternIntensity() const;
|
|
||||||
[[nodiscard]] bool hasShareUrl() const;
|
|
||||||
[[nodiscard]] QString shareUrl() const;
|
|
||||||
|
|
||||||
void loadDocument() const;
|
|
||||||
void loadThumbnail() const;
|
|
||||||
[[nodiscard]] FileOrigin fileOrigin() const;
|
|
||||||
|
|
||||||
[[nodiscard]] WallPaper withUrlParams(
|
|
||||||
const QMap<QString, QString> ¶ms) const;
|
|
||||||
[[nodiscard]] WallPaper withBlurred(bool blurred) const;
|
|
||||||
[[nodiscard]] WallPaper withPatternIntensity(int intensity) const;
|
|
||||||
[[nodiscard]] WallPaper withBackgroundColor(QColor color) const;
|
|
||||||
[[nodiscard]] WallPaper withParamsFrom(const WallPaper &other) const;
|
|
||||||
[[nodiscard]] WallPaper withoutImageData() const;
|
|
||||||
|
|
||||||
[[nodiscard]] static std::optional<WallPaper> Create(
|
|
||||||
const MTPWallPaper &data);
|
|
||||||
[[nodiscard]] static std::optional<WallPaper> Create(
|
|
||||||
const MTPDwallPaper &data);
|
|
||||||
|
|
||||||
[[nodiscard]] QByteArray serialize() const;
|
|
||||||
[[nodiscard]] static std::optional<WallPaper> FromSerialized(
|
|
||||||
const QByteArray &serialized);
|
|
||||||
[[nodiscard]] static std::optional<WallPaper> FromLegacySerialized(
|
|
||||||
quint64 id,
|
|
||||||
quint64 accessHash,
|
|
||||||
quint32 flags,
|
|
||||||
QString slug);
|
|
||||||
[[nodiscard]] static std::optional<WallPaper> FromLegacyId(
|
|
||||||
qint32 legacyId);
|
|
||||||
[[nodiscard]] static std::optional<WallPaper> FromColorSlug(
|
|
||||||
const QString &slug);
|
|
||||||
|
|
||||||
private:
|
|
||||||
static constexpr auto kDefaultIntensity = 40;
|
|
||||||
|
|
||||||
WallPaperId _id = WallPaperId();
|
|
||||||
uint64 _accessHash = 0;
|
|
||||||
MTPDwallPaper::Flags _flags;
|
|
||||||
QString _slug;
|
|
||||||
|
|
||||||
MTPDwallPaperSettings::Flags _settings;
|
|
||||||
std::optional<QColor> _backgroundColor;
|
|
||||||
int _intensity = kDefaultIntensity;
|
|
||||||
|
|
||||||
DocumentData *_document = nullptr;
|
|
||||||
std::shared_ptr<Image> _thumbnail;
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
[[nodiscard]] WallPaper ThemeWallPaper();
|
|
||||||
[[nodiscard]] bool IsThemeWallPaper(const WallPaper &paper);
|
|
||||||
[[nodiscard]] WallPaper CustomWallPaper();
|
|
||||||
[[nodiscard]] bool IsCustomWallPaper(const WallPaper &paper);
|
|
||||||
[[nodiscard]] WallPaper Legacy1DefaultWallPaper();
|
|
||||||
[[nodiscard]] bool IsLegacy1DefaultWallPaper(const WallPaper &paper);
|
|
||||||
[[nodiscard]] WallPaper DefaultWallPaper();
|
|
||||||
[[nodiscard]] bool IsDefaultWallPaper(const WallPaper &paper);
|
|
||||||
|
|
||||||
QColor PatternColor(QColor background);
|
|
||||||
QImage PreparePatternImage(
|
|
||||||
QImage image,
|
|
||||||
QColor bg,
|
|
||||||
QColor fg,
|
|
||||||
int intensity);
|
|
||||||
QImage PrepareBlurredBackground(QImage image);
|
|
||||||
|
|
||||||
namespace details {
|
|
||||||
|
|
||||||
[[nodiscard]] WallPaper UninitializedWallPaper();
|
|
||||||
[[nodiscard]] bool IsUninitializedWallPaper(const WallPaper &paper);
|
|
||||||
[[nodiscard]] WallPaper TestingThemeWallPaper();
|
|
||||||
[[nodiscard]] bool IsTestingThemeWallPaper(const WallPaper &paper);
|
|
||||||
[[nodiscard]] WallPaper TestingDefaultWallPaper();
|
|
||||||
[[nodiscard]] bool IsTestingDefaultWallPaper(const WallPaper &paper);
|
|
||||||
[[nodiscard]] WallPaper TestingEditorWallPaper();
|
|
||||||
[[nodiscard]] bool IsTestingEditorWallPaper(const WallPaper &paper);
|
|
||||||
|
|
||||||
} // namespace details
|
|
||||||
} // namespace Data
|
|
||||||
|
|
||||||
namespace Window {
|
namespace Window {
|
||||||
namespace Theme {
|
namespace Theme {
|
||||||
|
@ -185,7 +85,9 @@ struct BackgroundUpdate {
|
||||||
bool tiled;
|
bool tiled;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ChatBackground : public base::Observable<BackgroundUpdate> {
|
class ChatBackground
|
||||||
|
: public base::Observable<BackgroundUpdate>
|
||||||
|
, private base::Subscriber {
|
||||||
public:
|
public:
|
||||||
ChatBackground();
|
ChatBackground();
|
||||||
|
|
||||||
|
@ -251,6 +153,8 @@ private:
|
||||||
void keepApplied(const QString &path, bool write);
|
void keepApplied(const QString &path, bool write);
|
||||||
[[nodiscard]] bool isNonDefaultThemeOrBackground();
|
[[nodiscard]] bool isNonDefaultThemeOrBackground();
|
||||||
[[nodiscard]] bool isNonDefaultBackground();
|
[[nodiscard]] bool isNonDefaultBackground();
|
||||||
|
void refreshSession();
|
||||||
|
void checkUploadWallPaper();
|
||||||
|
|
||||||
friend bool IsNightMode();
|
friend bool IsNightMode();
|
||||||
friend void SetNightModeValue(bool nightMode);
|
friend void SetNightModeValue(bool nightMode);
|
||||||
|
@ -259,6 +163,7 @@ private:
|
||||||
friend void KeepApplied();
|
friend void KeepApplied();
|
||||||
friend bool IsNonDefaultBackground();
|
friend bool IsNonDefaultBackground();
|
||||||
|
|
||||||
|
AuthSession *_session = nullptr;
|
||||||
Data::WallPaper _paper = Data::details::UninitializedWallPaper();
|
Data::WallPaper _paper = Data::details::UninitializedWallPaper();
|
||||||
std::optional<QColor> _paperColor;
|
std::optional<QColor> _paperColor;
|
||||||
QImage _original;
|
QImage _original;
|
||||||
|
@ -278,6 +183,9 @@ private:
|
||||||
bool _tileForRevert = false;
|
bool _tileForRevert = false;
|
||||||
|
|
||||||
std::vector<AdjustableColor> _adjustableColors;
|
std::vector<AdjustableColor> _adjustableColors;
|
||||||
|
FullMsgId _wallPaperUploadId;
|
||||||
|
mtpRequestId _wallPaperRequestId = 0;
|
||||||
|
rpl::lifetime _wallPaperUploadLifetime;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -207,6 +207,8 @@
|
||||||
<(src_loc)/data/data_user.h
|
<(src_loc)/data/data_user.h
|
||||||
<(src_loc)/data/data_user_photos.cpp
|
<(src_loc)/data/data_user_photos.cpp
|
||||||
<(src_loc)/data/data_user_photos.h
|
<(src_loc)/data/data_user_photos.h
|
||||||
|
<(src_loc)/data/data_wall_paper.cpp
|
||||||
|
<(src_loc)/data/data_wall_paper.h
|
||||||
<(src_loc)/data/data_web_page.cpp
|
<(src_loc)/data/data_web_page.cpp
|
||||||
<(src_loc)/data/data_web_page.h
|
<(src_loc)/data/data_web_page.h
|
||||||
<(src_loc)/dialogs/dialogs_entry.cpp
|
<(src_loc)/dialogs/dialogs_entry.cpp
|
||||||
|
|
Loading…
Reference in New Issue