/* 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 #include "storage/storage_facade.h" #include "storage/storage_sparse_ids_list.h" namespace Storage { // Allow forward declarations. enum class SharedMediaType : signed char { Photo, Video, PhotoVideo, MusicFile, File, VoiceFile, Link, ChatPhoto, RoundVoiceFile, GIF, RoundFile, Pinned, kCount, }; constexpr auto kSharedMediaTypeCount = static_cast(SharedMediaType::kCount); constexpr bool IsValidSharedMediaType(SharedMediaType type) { return (static_cast(type) >= 0) && (static_cast(type) < kSharedMediaTypeCount); } using SharedMediaTypesMask = base::enum_mask; struct SharedMediaAddNew { SharedMediaAddNew( PeerId peerId, SharedMediaTypesMask types, MsgId messageId) : peerId(peerId) , messageId(messageId) , types(types) { } PeerId peerId = 0; MsgId messageId = 0; SharedMediaTypesMask types; }; struct SharedMediaAddExisting { SharedMediaAddExisting( PeerId peerId, MsgId topicRootId, SharedMediaTypesMask types, MsgId messageId, MsgRange noSkipRange) : peerId(peerId) , topicRootId(topicRootId) , messageId(messageId) , noSkipRange(noSkipRange) , types(types) { } PeerId peerId = 0; MsgId topicRootId = 0; MsgId messageId = 0; MsgRange noSkipRange; SharedMediaTypesMask types; }; struct SharedMediaAddSlice { SharedMediaAddSlice( PeerId peerId, MsgId topicRootId, SharedMediaType type, std::vector &&messageIds, MsgRange noSkipRange, std::optional count = std::nullopt) : peerId(peerId) , topicRootId(topicRootId) , messageIds(std::move(messageIds)) , noSkipRange(noSkipRange) , type(type) , count(count) { } PeerId peerId = 0; MsgId topicRootId = 0; std::vector messageIds; MsgRange noSkipRange; SharedMediaType type = SharedMediaType::kCount; std::optional count; }; struct SharedMediaRemoveOne { SharedMediaRemoveOne( PeerId peerId, SharedMediaTypesMask types, MsgId messageId) : peerId(peerId) , messageId(messageId) , types(types) { } PeerId peerId = 0; MsgId messageId = 0; SharedMediaTypesMask types; }; struct SharedMediaRemoveAll { SharedMediaRemoveAll( PeerId peerId, SharedMediaTypesMask types = SharedMediaTypesMask::All()) : peerId(peerId) , types(types) { } PeerId peerId = 0; SharedMediaTypesMask types; }; struct SharedMediaInvalidateBottom { SharedMediaInvalidateBottom(PeerId peerId) : peerId(peerId) { } PeerId peerId = 0; }; struct SharedMediaKey { SharedMediaKey( PeerId peerId, MsgId topicRootId, SharedMediaType type, MsgId messageId) : peerId(peerId) , topicRootId(topicRootId) , type(type) , messageId(messageId) { } friend inline constexpr auto operator<=>( const SharedMediaKey &, const SharedMediaKey &) = default; PeerId peerId = 0; MsgId topicRootId = 0; SharedMediaType type = SharedMediaType::kCount; MsgId messageId = 0; }; struct SharedMediaQuery { SharedMediaQuery( SharedMediaKey key, int limitBefore, int limitAfter) : key(key) , limitBefore(limitBefore) , limitAfter(limitAfter) { } SharedMediaKey key; int limitBefore = 0; int limitAfter = 0; }; using SharedMediaResult = SparseIdsListResult; struct SharedMediaSliceUpdate { SharedMediaSliceUpdate( PeerId peerId, MsgId topicRootId, SharedMediaType type, const SparseIdsSliceUpdate &data) : peerId(peerId) , topicRootId(topicRootId) , type(type) , data(data) { } PeerId peerId = 0; MsgId topicRootId = 0; SharedMediaType type = SharedMediaType::kCount; SparseIdsSliceUpdate data; }; class SharedMedia { public: using Type = SharedMediaType; void add(SharedMediaAddNew &&query); void add(SharedMediaAddExisting &&query); void add(SharedMediaAddSlice &&query); void remove(SharedMediaRemoveOne &&query); void remove(SharedMediaRemoveAll &&query); void invalidate(SharedMediaInvalidateBottom &&query); rpl::producer query(SharedMediaQuery &&query) const; SharedMediaResult snapshot(const SharedMediaQuery &query) const; bool empty(const SharedMediaKey &key) const; rpl::producer sliceUpdated() const; rpl::producer oneRemoved() const; rpl::producer allRemoved() const; rpl::producer bottomInvalidated() const; private: struct Key { PeerId peerId = 0; MsgId topicRootId = 0; friend inline constexpr auto operator<=>(Key, Key) = default; }; using Lists = std::array; std::map::iterator enforceLists(Key key); std::map _lists; rpl::event_stream _sliceUpdated; rpl::event_stream _oneRemoved; rpl::event_stream _allRemoved; rpl::event_stream _bottomInvalidated; rpl::lifetime _lifetime; }; } // namespace Storage