650 lines
14 KiB
C++
650 lines
14 KiB
C++
/*
|
|
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
|
|
|
|
#include "data/data_peer_id.h"
|
|
|
|
class FileLoader;
|
|
|
|
namespace Storage {
|
|
namespace Cache {
|
|
struct Key;
|
|
} // namespace Cache
|
|
} // namespace Storage
|
|
|
|
namespace Data {
|
|
struct UpdatedFileReferences;
|
|
} // namespace Data
|
|
|
|
enum LoadFromCloudSetting {
|
|
LoadFromCloudOrLocal,
|
|
LoadFromLocalOnly,
|
|
};
|
|
|
|
enum LoadToCacheSetting {
|
|
LoadToFileOnly,
|
|
LoadToCacheAsWell,
|
|
};
|
|
|
|
using InMemoryKey = std::pair<uint64, uint64>;
|
|
|
|
namespace std {
|
|
|
|
template<>
|
|
struct hash<InMemoryKey> {
|
|
size_t operator()(InMemoryKey value) const {
|
|
auto seed = hash<uint64>()(value.first);
|
|
seed ^= hash<uint64>()(value.second)
|
|
+ std::size_t(0x9e3779b9)
|
|
+ (seed << 6) + (seed >> 2);
|
|
return seed;
|
|
}
|
|
|
|
};
|
|
|
|
} // namespace std
|
|
|
|
class StorageFileLocation {
|
|
public:
|
|
// Those are used in serialization, don't change.
|
|
enum class Type : uint8 {
|
|
Legacy = 0x00,
|
|
Encrypted = 0x01,
|
|
Document = 0x02,
|
|
Secure = 0x03,
|
|
Takeout = 0x04,
|
|
Photo = 0x05,
|
|
PeerPhoto = 0x06,
|
|
StickerSetThumb = 0x07,
|
|
GroupCallStream = 0x08,
|
|
};
|
|
|
|
StorageFileLocation() = default;
|
|
StorageFileLocation(
|
|
int32 dcId,
|
|
UserId self,
|
|
const MTPInputFileLocation &tl);
|
|
|
|
[[nodiscard]] StorageFileLocation convertToModern(
|
|
Type type,
|
|
uint64 id,
|
|
uint64 accessHash) const;
|
|
|
|
[[nodiscard]] int32 dcId() const;
|
|
[[nodiscard]] uint64 objectId() const;
|
|
[[nodiscard]] MTPInputFileLocation tl(UserId self) const;
|
|
|
|
[[nodiscard]] QByteArray serialize() const;
|
|
[[nodiscard]] int serializeSize() const;
|
|
[[nodiscard]] static std::optional<StorageFileLocation> FromSerialized(
|
|
const QByteArray &serialized);
|
|
|
|
[[nodiscard]] Type type() const;
|
|
[[nodiscard]] bool valid() const;
|
|
[[nodiscard]] bool isLegacy() const;
|
|
[[nodiscard]] Storage::Cache::Key cacheKey() const;
|
|
[[nodiscard]] Storage::Cache::Key bigFileBaseCacheKey() const;
|
|
|
|
// We have to allow checking this because of a serialization bug.
|
|
[[nodiscard]] bool isDocumentThumbnail() const;
|
|
|
|
[[nodiscard]] QByteArray fileReference() const;
|
|
bool refreshFileReference(const Data::UpdatedFileReferences &updates);
|
|
bool refreshFileReference(const QByteArray &data);
|
|
|
|
[[nodiscard]] static const StorageFileLocation &Invalid();
|
|
|
|
friend bool operator==(
|
|
const StorageFileLocation &a,
|
|
const StorageFileLocation &b);
|
|
friend bool operator<(
|
|
const StorageFileLocation &a,
|
|
const StorageFileLocation &b);
|
|
|
|
private:
|
|
uint16 _dcId = 0;
|
|
Type _type = Type::Legacy;
|
|
uint8 _sizeLetter = 0;
|
|
int32 _localId = 0;
|
|
uint64 _id = 0;
|
|
uint64 _accessHash = 0;
|
|
uint64 _volumeId = 0;
|
|
PeerId _inMessagePeerId = 0;
|
|
uint32 _inMessageId = 0;
|
|
QByteArray _fileReference;
|
|
|
|
};
|
|
|
|
inline bool operator!=(
|
|
const StorageFileLocation &a,
|
|
const StorageFileLocation &b) {
|
|
return !(a == b);
|
|
}
|
|
|
|
inline bool operator>(
|
|
const StorageFileLocation &a,
|
|
const StorageFileLocation &b) {
|
|
return (b < a);
|
|
}
|
|
|
|
inline bool operator<=(
|
|
const StorageFileLocation &a,
|
|
const StorageFileLocation &b) {
|
|
return !(b < a);
|
|
}
|
|
|
|
inline bool operator>=(
|
|
const StorageFileLocation &a,
|
|
const StorageFileLocation &b) {
|
|
return !(a < b);
|
|
}
|
|
|
|
class StorageImageLocation {
|
|
public:
|
|
StorageImageLocation() = default;
|
|
StorageImageLocation(
|
|
const StorageFileLocation &file,
|
|
int width,
|
|
int height);
|
|
|
|
[[nodiscard]] QByteArray serialize() const;
|
|
[[nodiscard]] int serializeSize() const;
|
|
[[nodiscard]] static std::optional<StorageImageLocation> FromSerialized(
|
|
const QByteArray &serialized);
|
|
|
|
[[nodiscard]] StorageImageLocation convertToModern(
|
|
StorageFileLocation::Type type,
|
|
uint64 id,
|
|
uint64 accessHash) const {
|
|
return StorageImageLocation(
|
|
_file.convertToModern(type, id, accessHash),
|
|
_width,
|
|
_height);
|
|
}
|
|
|
|
[[nodiscard]] const StorageFileLocation &file() const {
|
|
return _file;
|
|
}
|
|
[[nodiscard]] int width() const {
|
|
return _width;
|
|
}
|
|
[[nodiscard]] int height() const {
|
|
return _height;
|
|
}
|
|
|
|
void setSize(int width, int height) {
|
|
_width = width;
|
|
_height = height;
|
|
}
|
|
|
|
[[nodiscard]] StorageFileLocation::Type type() const {
|
|
return _file.type();
|
|
}
|
|
[[nodiscard]] bool valid() const {
|
|
return _file.valid();
|
|
}
|
|
[[nodiscard]] bool isLegacy() const {
|
|
return _file.isLegacy();
|
|
}
|
|
[[nodiscard]] QByteArray fileReference() const {
|
|
return _file.fileReference();
|
|
}
|
|
bool refreshFileReference(const QByteArray &data) {
|
|
return _file.refreshFileReference(data);
|
|
}
|
|
bool refreshFileReference(const Data::UpdatedFileReferences &updates) {
|
|
return _file.refreshFileReference(updates);
|
|
}
|
|
|
|
[[nodiscard]] static const StorageImageLocation &Invalid() {
|
|
static auto result = StorageImageLocation();
|
|
return result;
|
|
}
|
|
|
|
friend inline bool operator==(
|
|
const StorageImageLocation &a,
|
|
const StorageImageLocation &b) {
|
|
return (a._file == b._file);
|
|
}
|
|
friend inline bool operator<(
|
|
const StorageImageLocation &a,
|
|
const StorageImageLocation &b) {
|
|
return (a._file < b._file);
|
|
}
|
|
|
|
private:
|
|
StorageFileLocation _file;
|
|
int _width = 0;
|
|
int _height = 0;
|
|
|
|
};
|
|
|
|
inline bool operator!=(
|
|
const StorageImageLocation &a,
|
|
const StorageImageLocation &b) {
|
|
return !(a == b);
|
|
}
|
|
|
|
inline bool operator>(
|
|
const StorageImageLocation &a,
|
|
const StorageImageLocation &b) {
|
|
return (b < a);
|
|
}
|
|
|
|
inline bool operator<=(
|
|
const StorageImageLocation &a,
|
|
const StorageImageLocation &b) {
|
|
return !(b < a);
|
|
}
|
|
|
|
inline bool operator>=(
|
|
const StorageImageLocation &a,
|
|
const StorageImageLocation &b) {
|
|
return !(a < b);
|
|
}
|
|
|
|
class WebFileLocation {
|
|
public:
|
|
WebFileLocation() = default;
|
|
WebFileLocation(const QByteArray &url, uint64 accessHash)
|
|
: _accessHash(accessHash)
|
|
, _url(url) {
|
|
}
|
|
bool isNull() const {
|
|
return _url.isEmpty();
|
|
}
|
|
uint64 accessHash() const {
|
|
return _accessHash;
|
|
}
|
|
const QByteArray &url() const {
|
|
return _url;
|
|
}
|
|
|
|
static WebFileLocation Null;
|
|
|
|
private:
|
|
uint64 _accessHash = 0;
|
|
QByteArray _url;
|
|
|
|
friend inline bool operator==(
|
|
const WebFileLocation &a,
|
|
const WebFileLocation &b) {
|
|
return (a._accessHash == b._accessHash)
|
|
&& (a._url == b._url);
|
|
}
|
|
friend inline bool operator<(
|
|
const WebFileLocation &a,
|
|
const WebFileLocation &b) {
|
|
return std::tie(a._accessHash, a._url)
|
|
< std::tie(b._accessHash, b._url);
|
|
}
|
|
|
|
};
|
|
|
|
inline bool operator!=(const WebFileLocation &a, const WebFileLocation &b) {
|
|
return !(a == b);
|
|
}
|
|
|
|
inline bool operator>(const WebFileLocation &a, const WebFileLocation &b) {
|
|
return (b < a);
|
|
}
|
|
|
|
inline bool operator<=(const WebFileLocation &a, const WebFileLocation &b) {
|
|
return !(b < a);
|
|
}
|
|
|
|
inline bool operator>=(const WebFileLocation &a, const WebFileLocation &b) {
|
|
return !(a < b);
|
|
}
|
|
|
|
struct GeoPointLocation {
|
|
float64 lat = 0.;
|
|
float64 lon = 0.;
|
|
uint64 access = 0;
|
|
int32 width = 0;
|
|
int32 height = 0;
|
|
int32 zoom = 0;
|
|
int32 scale = 0;
|
|
};
|
|
|
|
inline bool operator==(
|
|
const GeoPointLocation &a,
|
|
const GeoPointLocation &b) {
|
|
return (a.lat == b.lat)
|
|
&& (a.lon == b.lon)
|
|
&& (a.access == b.access)
|
|
&& (a.width == b.width)
|
|
&& (a.height == b.height)
|
|
&& (a.zoom == b.zoom)
|
|
&& (a.scale == b.scale);
|
|
}
|
|
|
|
inline bool operator<(
|
|
const GeoPointLocation &a,
|
|
const GeoPointLocation &b) {
|
|
return std::tie(
|
|
a.access,
|
|
a.lat,
|
|
a.lon,
|
|
a.width,
|
|
a.height,
|
|
a.zoom,
|
|
a.scale)
|
|
< std::tie(
|
|
b.access,
|
|
b.lat,
|
|
b.lon,
|
|
b.width,
|
|
b.height,
|
|
b.zoom,
|
|
b.scale);
|
|
}
|
|
|
|
inline bool operator!=(
|
|
const GeoPointLocation &a,
|
|
const GeoPointLocation &b) {
|
|
return !(a == b);
|
|
}
|
|
|
|
inline bool operator>(
|
|
const GeoPointLocation &a,
|
|
const GeoPointLocation &b) {
|
|
return (b < a);
|
|
}
|
|
|
|
inline bool operator<=(
|
|
const GeoPointLocation &a,
|
|
const GeoPointLocation &b) {
|
|
return !(b < a);
|
|
}
|
|
|
|
inline bool operator>=(
|
|
const GeoPointLocation &a,
|
|
const GeoPointLocation &b) {
|
|
return !(a < b);
|
|
}
|
|
|
|
struct PlainUrlLocation {
|
|
QString url;
|
|
|
|
friend inline bool operator==(
|
|
const PlainUrlLocation &a,
|
|
const PlainUrlLocation &b) {
|
|
return (a.url == b.url);
|
|
}
|
|
friend inline bool operator<(
|
|
const PlainUrlLocation &a,
|
|
const PlainUrlLocation &b) {
|
|
return (a.url < b.url);
|
|
}
|
|
};
|
|
|
|
inline bool operator!=(
|
|
const PlainUrlLocation &a,
|
|
const PlainUrlLocation &b) {
|
|
return !(a == b);
|
|
}
|
|
|
|
inline bool operator>(
|
|
const PlainUrlLocation &a,
|
|
const PlainUrlLocation &b) {
|
|
return (b < a);
|
|
}
|
|
|
|
inline bool operator<=(
|
|
const PlainUrlLocation &a,
|
|
const PlainUrlLocation &b) {
|
|
return !(b < a);
|
|
}
|
|
|
|
inline bool operator>=(
|
|
const PlainUrlLocation &a,
|
|
const PlainUrlLocation &b) {
|
|
return !(a < b);
|
|
}
|
|
|
|
struct InMemoryLocation {
|
|
QByteArray bytes;
|
|
|
|
friend inline bool operator==(
|
|
const InMemoryLocation &a,
|
|
const InMemoryLocation &b) {
|
|
return (a.bytes == b.bytes);
|
|
}
|
|
friend inline bool operator<(
|
|
const InMemoryLocation &a,
|
|
const InMemoryLocation &b) {
|
|
return (a.bytes < b.bytes);
|
|
}
|
|
};
|
|
|
|
inline bool operator!=(
|
|
const InMemoryLocation &a,
|
|
const InMemoryLocation &b) {
|
|
return !(a == b);
|
|
}
|
|
|
|
inline bool operator>(
|
|
const InMemoryLocation &a,
|
|
const InMemoryLocation &b) {
|
|
return (b < a);
|
|
}
|
|
|
|
inline bool operator<=(
|
|
const InMemoryLocation &a,
|
|
const InMemoryLocation &b) {
|
|
return !(b < a);
|
|
}
|
|
|
|
inline bool operator>=(
|
|
const InMemoryLocation &a,
|
|
const InMemoryLocation &b) {
|
|
return !(a < b);
|
|
}
|
|
|
|
class DownloadLocation {
|
|
public:
|
|
std::variant<
|
|
StorageFileLocation,
|
|
WebFileLocation,
|
|
GeoPointLocation,
|
|
PlainUrlLocation,
|
|
InMemoryLocation> data;
|
|
|
|
[[nodiscard]] QByteArray serialize() const;
|
|
[[nodiscard]] int serializeSize() const;
|
|
[[nodiscard]] static std::optional<DownloadLocation> FromSerialized(
|
|
const QByteArray &serialized);
|
|
|
|
[[nodiscard]] DownloadLocation convertToModern(
|
|
StorageFileLocation::Type type,
|
|
uint64 id,
|
|
uint64 accessHash) const;
|
|
|
|
[[nodiscard]] Storage::Cache::Key cacheKey() const;
|
|
[[nodiscard]] Storage::Cache::Key bigFileBaseCacheKey() const;
|
|
[[nodiscard]] bool valid() const;
|
|
[[nodiscard]] bool isLegacy() const;
|
|
[[nodiscard]] QByteArray fileReference() const;
|
|
bool refreshFileReference(const QByteArray &data);
|
|
bool refreshFileReference(const Data::UpdatedFileReferences &updates);
|
|
|
|
friend inline bool operator==(
|
|
const DownloadLocation &a,
|
|
const DownloadLocation &b) {
|
|
return (a.data == b.data);
|
|
}
|
|
friend inline bool operator<(
|
|
const DownloadLocation &a,
|
|
const DownloadLocation &b) {
|
|
return (a.data < b.data);
|
|
}
|
|
|
|
};
|
|
|
|
inline bool operator!=(
|
|
const DownloadLocation &a,
|
|
const DownloadLocation &b) {
|
|
return !(a == b);
|
|
}
|
|
|
|
inline bool operator>(
|
|
const DownloadLocation &a,
|
|
const DownloadLocation &b) {
|
|
return (b < a);
|
|
}
|
|
|
|
inline bool operator<=(
|
|
const DownloadLocation &a,
|
|
const DownloadLocation &b) {
|
|
return !(b < a);
|
|
}
|
|
|
|
inline bool operator>=(
|
|
const DownloadLocation &a,
|
|
const DownloadLocation &b) {
|
|
return !(a < b);
|
|
}
|
|
|
|
class ImageLocation {
|
|
public:
|
|
ImageLocation() = default;
|
|
ImageLocation(
|
|
const DownloadLocation &file,
|
|
int width,
|
|
int height);
|
|
|
|
[[nodiscard]] QByteArray serialize() const;
|
|
[[nodiscard]] int serializeSize() const;
|
|
[[nodiscard]] static std::optional<ImageLocation> FromSerialized(
|
|
const QByteArray &serialized);
|
|
|
|
[[nodiscard]] ImageLocation convertToModern(
|
|
StorageFileLocation::Type type,
|
|
uint64 id,
|
|
uint64 accessHash) const {
|
|
return ImageLocation(
|
|
_file.convertToModern(type, id, accessHash),
|
|
_width,
|
|
_height);
|
|
}
|
|
|
|
[[nodiscard]] const DownloadLocation &file() const {
|
|
return _file;
|
|
}
|
|
[[nodiscard]] int width() const {
|
|
return _width;
|
|
}
|
|
[[nodiscard]] int height() const {
|
|
return _height;
|
|
}
|
|
|
|
void setSize(int width, int height) {
|
|
_width = width;
|
|
_height = height;
|
|
}
|
|
|
|
[[nodiscard]] bool valid() const {
|
|
return _file.valid();
|
|
}
|
|
[[nodiscard]] bool isLegacy() const {
|
|
return _file.isLegacy();
|
|
}
|
|
[[nodiscard]] QByteArray fileReference() const {
|
|
return _file.fileReference();
|
|
}
|
|
bool refreshFileReference(const QByteArray &data) {
|
|
return _file.refreshFileReference(data);
|
|
}
|
|
bool refreshFileReference(const Data::UpdatedFileReferences &updates) {
|
|
return _file.refreshFileReference(updates);
|
|
}
|
|
|
|
[[nodiscard]] static const ImageLocation &Invalid() {
|
|
static auto result = ImageLocation();
|
|
return result;
|
|
}
|
|
|
|
friend inline bool operator==(
|
|
const ImageLocation &a,
|
|
const ImageLocation &b) {
|
|
return (a._file == b._file);
|
|
}
|
|
friend inline bool operator<(
|
|
const ImageLocation &a,
|
|
const ImageLocation &b) {
|
|
return (a._file < b._file);
|
|
}
|
|
|
|
private:
|
|
DownloadLocation _file;
|
|
int _width = 0;
|
|
int _height = 0;
|
|
|
|
};
|
|
|
|
inline bool operator!=(
|
|
const ImageLocation &a,
|
|
const ImageLocation &b) {
|
|
return !(a == b);
|
|
}
|
|
|
|
inline bool operator>(
|
|
const ImageLocation &a,
|
|
const ImageLocation &b) {
|
|
return (b < a);
|
|
}
|
|
|
|
inline bool operator<=(
|
|
const ImageLocation &a,
|
|
const ImageLocation &b) {
|
|
return !(b < a);
|
|
}
|
|
|
|
inline bool operator>=(
|
|
const ImageLocation &a,
|
|
const ImageLocation &b) {
|
|
return !(a < b);
|
|
}
|
|
|
|
struct ImageWithLocation {
|
|
ImageLocation location;
|
|
QByteArray bytes;
|
|
QImage preloaded;
|
|
int bytesCount = 0;
|
|
int progressivePartSize = 0;
|
|
};
|
|
|
|
InMemoryKey inMemoryKey(const StorageFileLocation &location);
|
|
|
|
inline InMemoryKey inMemoryKey(const StorageImageLocation &location) {
|
|
return inMemoryKey(location.file());
|
|
}
|
|
|
|
InMemoryKey inMemoryKey(const WebFileLocation &location);
|
|
InMemoryKey inMemoryKey(const GeoPointLocation &location);
|
|
InMemoryKey inMemoryKey(const PlainUrlLocation &location);
|
|
InMemoryKey inMemoryKey(const InMemoryLocation &location);
|
|
InMemoryKey inMemoryKey(const DownloadLocation &location);
|
|
|
|
inline InMemoryKey inMemoryKey(const ImageLocation &location) {
|
|
return inMemoryKey(location.file());
|
|
}
|
|
|
|
inline QSize shrinkToKeepAspect(int32 width, int32 height, int32 towidth, int32 toheight) {
|
|
int32 w = qMax(width, 1), h = qMax(height, 1);
|
|
if (w * toheight > h * towidth) {
|
|
h = qRound(h * towidth / float64(w));
|
|
w = towidth;
|
|
} else {
|
|
w = qRound(w * toheight / float64(h));
|
|
h = toheight;
|
|
}
|
|
return QSize(qMax(w, 1), qMax(h, 1));
|
|
}
|