Reverse user and chat profile photos.

This commit is contained in:
John Preston 2017-11-23 13:58:12 +04:00
parent da71938d18
commit 67d4eb688a
10 changed files with 134 additions and 61 deletions

View File

@ -27,6 +27,11 @@ inline Type take(Type &value) {
return std::exchange(value, Type {}); return std::exchange(value, Type {});
} }
template <typename Type>
inline Type duplicate(const Type &value) {
return value;
}
template <typename Type, size_t Size> template <typename Type, size_t Size>
inline constexpr size_t array_size(const Type(&)[Size]) { inline constexpr size_t array_size(const Type(&)[Size]) {
return Size; return Size;

View File

@ -24,19 +24,31 @@ namespace base {
namespace functors { namespace functors {
struct abs_helper { struct abs_helper {
template <typename Type, template <
typename Type,
typename = decltype(0 < std::declval<Type>()), typename = decltype(0 < std::declval<Type>()),
typename = decltype(-std::declval<Type>())> typename = decltype(-std::declval<Type>())>
constexpr Type operator()(Type value) const { constexpr Type operator()(Type value) const {
return (0 < value) ? value : (-value); return (0 < value) ? value : (-value);
} }
}; };
constexpr auto abs = abs_helper {}; constexpr auto abs = abs_helper{};
template <typename Type> constexpr auto add = [](auto value) {
inline auto add(Type a) { return [value](auto other) {
return [a](auto b) { return a + b; }; return value + other;
};
}; };
struct negate_helper {
template <
typename Type,
typename = decltype(-std::declval<Type>())>
constexpr Type operator()(Type value) const {
return -value;
}
};
constexpr auto negate = negate_helper{};
} // namespace functors } // namespace functors
} // namespace base } // namespace base

View File

@ -22,27 +22,27 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
namespace qthelp { namespace qthelp {
QMap<QString, QString> url_parse_params(const QString &params, UrlParamNameTransform transform) { QMap<QString, QString> url_parse_params(
QMap<QString, QString> result; const QString &params,
UrlParamNameTransform transform) {
auto result = QMap<QString, QString>();
auto transformParamName = [transform](const QString &name) { const auto transformParamName = [transform](const QString &name) {
if (transform == UrlParamNameTransform::ToLower) { if (transform == UrlParamNameTransform::ToLower) {
return name.toLower(); return name.toLower();
} }
return name; return name;
}; };
for (const auto &param : params.split('&')) {
auto paramsList = params.split('&');
for_const (auto &param, paramsList) {
// Skip params without a name (starting with '='). // Skip params without a name (starting with '=').
if (auto separatorPosition = param.indexOf('=')) { if (auto separatorPosition = param.indexOf('=')) {
auto paramName = param; const auto paramName = transformParamName(
auto paramValue = QString(); (separatorPosition > 0)
if (separatorPosition > 0) { ? param.mid(0, separatorPosition)
paramName = param.mid(0, separatorPosition); : param);
paramValue = url_decode(param.mid(separatorPosition + 1)); const auto paramValue = (separatorPosition > 0)
} ? url_decode(param.mid(separatorPosition + 1))
paramName = transformParamName(paramName); : QString();
if (!result.contains(paramName)) { if (!result.contains(paramName)) {
result.insert(paramName, paramValue); result.insert(paramName, paramValue);
} }

View File

@ -154,13 +154,13 @@ SearchResult ParseSearchResult(
result.messageIds.reserve(messages->size()); result.messageIds.reserve(messages->size());
for (auto &message : *messages) { for (auto &message : *messages) {
if (auto item = App::histories().addNewMessage(message, addType)) { if (auto item = App::histories().addNewMessage(message, addType)) {
auto itemId = item->id;
if ((type == Storage::SharedMediaType::kCount) if ((type == Storage::SharedMediaType::kCount)
|| item->sharedMediaTypes().test(type)) { || item->sharedMediaTypes().test(type)) {
auto itemId = item->id;
result.messageIds.push_back(itemId); result.messageIds.push_back(itemId);
accumulate_min(result.noSkipRange.from, itemId);
accumulate_max(result.noSkipRange.till, itemId);
} }
accumulate_min(result.noSkipRange.from, itemId);
accumulate_max(result.noSkipRange.till, itemId);
} }
} }
if (messageId && result.messageIds.empty()) { if (messageId && result.messageIds.empty()) {

View File

@ -207,11 +207,15 @@ base::optional<int> SharedMediaWithLastSlice::fullCount() const {
_isolatedLastPhoto | [](bool isolated) { return isolated ? 1 : 0; }); _isolatedLastPhoto | [](bool isolated) { return isolated ? 1 : 0; });
} }
base::optional<int> SharedMediaWithLastSlice::skippedBefore() const { base::optional<int> SharedMediaWithLastSlice::skippedBeforeImpl() const {
return _slice.skippedBefore(); return _slice.skippedBefore();
} }
base::optional<int> SharedMediaWithLastSlice::skippedAfter() const { base::optional<int> SharedMediaWithLastSlice::skippedBefore() const {
return _reversed ? skippedAfterImpl() : skippedBeforeImpl();
}
base::optional<int> SharedMediaWithLastSlice::skippedAfterImpl() const {
return isolatedInSlice() return isolatedInSlice()
? Add( ? Add(
_slice.skippedAfter(), _slice.skippedAfter(),
@ -219,7 +223,11 @@ base::optional<int> SharedMediaWithLastSlice::skippedAfter() const {
: (lastPhotoSkip() | [](int) { return 0; }); : (lastPhotoSkip() | [](int) { return 0; });
} }
base::optional<int> SharedMediaWithLastSlice::indexOf(Value value) const { base::optional<int> SharedMediaWithLastSlice::skippedAfter() const {
return _reversed ? skippedBeforeImpl() : skippedAfterImpl();
}
base::optional<int> SharedMediaWithLastSlice::indexOfImpl(Value value) const {
return base::get_if<FullMsgId>(&value) return base::get_if<FullMsgId>(&value)
? _slice.indexOf(*base::get_if<FullMsgId>(&value)) ? _slice.indexOf(*base::get_if<FullMsgId>(&value))
: (isolatedInSlice() : (isolatedInSlice()
@ -228,6 +236,13 @@ base::optional<int> SharedMediaWithLastSlice::indexOf(Value value) const {
: Add(_slice.size() - 1, lastPhotoSkip()); : Add(_slice.size() - 1, lastPhotoSkip());
} }
base::optional<int> SharedMediaWithLastSlice::indexOf(Value value) const {
auto result = indexOfImpl(value);
return _reversed
? (result | func::negate | func::add(size() - 1))
: result;
}
int SharedMediaWithLastSlice::size() const { int SharedMediaWithLastSlice::size() const {
return _slice.size() return _slice.size()
+ ((!isolatedInSlice() && lastPhotoSkip() == 1) ? 1 : 0); + ((!isolatedInSlice() && lastPhotoSkip() == 1) ? 1 : 0);
@ -236,6 +251,9 @@ int SharedMediaWithLastSlice::size() const {
SharedMediaWithLastSlice::Value SharedMediaWithLastSlice::operator[](int index) const { SharedMediaWithLastSlice::Value SharedMediaWithLastSlice::operator[](int index) const {
Expects(index >= 0 && index < size()); Expects(index >= 0 && index < size());
if (_reversed) {
index = size() - index - 1;
}
return (index < _slice.size()) return (index < _slice.size())
? Value(_slice[index]) ? Value(_slice[index])
: Value(App::photo(_lastPhotoId)); : Value(App::photo(_lastPhotoId));
@ -252,6 +270,10 @@ base::optional<int> SharedMediaWithLastSlice::distance(
return base::none; return base::none;
} }
void SharedMediaWithLastSlice::reverse() {
_reversed = !_reversed;
}
PhotoId SharedMediaWithLastSlice::LastPeerPhotoId(PeerId peerId) { PhotoId SharedMediaWithLastSlice::LastPeerPhotoId(PeerId peerId) {
if (auto peer = App::peerLoaded(peerId)) { if (auto peer = App::peerLoaded(peerId)) {
return peer->photoId; return peer->photoId;
@ -332,3 +354,14 @@ rpl::producer<SharedMediaWithLastSlice> SharedMediaWithLastViewer(
}); });
}; };
} }
rpl::producer<SharedMediaWithLastSlice> SharedMediaWithLastReversedViewer(
SharedMediaWithLastSlice::Key key,
int limitBefore,
int limitAfter) {
return SharedMediaWithLastViewer(key, limitBefore, limitAfter)
| rpl::map([](SharedMediaWithLastSlice &&slice) {
slice.reverse();
return std::move(slice);
});
}

View File

@ -116,6 +116,8 @@ public:
Value operator[](int index) const; Value operator[](int index) const;
base::optional<int> distance(const Key &a, const Key &b) const; base::optional<int> distance(const Key &a, const Key &b) const;
void reverse();
static SparseIdsMergedSlice::Key ViewerKey(const Key &key) { static SparseIdsMergedSlice::Key ViewerKey(const Key &key) {
return { return {
key.peerId, key.peerId,
@ -174,11 +176,16 @@ private:
| [](bool isolated) { return isolated ? 1 : 0; }; | [](bool isolated) { return isolated ? 1 : 0; };
} }
base::optional<int> skippedBeforeImpl() const;
base::optional<int> skippedAfterImpl() const;
base::optional<int> indexOfImpl(Value fullId) const;
Key _key; Key _key;
SparseIdsMergedSlice _slice; SparseIdsMergedSlice _slice;
base::optional<SparseIdsMergedSlice> _ending; base::optional<SparseIdsMergedSlice> _ending;
PhotoId _lastPhotoId = 0; PhotoId _lastPhotoId = 0;
base::optional<bool> _isolatedLastPhoto; base::optional<bool> _isolatedLastPhoto;
bool _reversed = false;
}; };
@ -186,3 +193,8 @@ rpl::producer<SharedMediaWithLastSlice> SharedMediaWithLastViewer(
SharedMediaWithLastSlice::Key key, SharedMediaWithLastSlice::Key key,
int limitBefore, int limitBefore,
int limitAfter); int limitAfter);
rpl::producer<SharedMediaWithLastSlice> SharedMediaWithLastReversedViewer(
SharedMediaWithLastSlice::Key key,
int limitBefore,
int limitAfter);

View File

@ -66,22 +66,27 @@ UserPhotosSlice::UserPhotosSlice(Key key)
{}, {},
base::none, base::none,
base::none, base::none,
0) { base::none) {
} }
UserPhotosSlice::UserPhotosSlice( UserPhotosSlice::UserPhotosSlice(
Key key, Key key,
const std::deque<PhotoId> &ids, std::deque<PhotoId> &&ids,
base::optional<int> fullCount, base::optional<int> fullCount,
base::optional<int> skippedBefore, base::optional<int> skippedBefore,
int skippedAfter) base::optional<int> skippedAfter)
: _key(key) : _key(key)
, _ids(ids) , _ids(std::move(ids))
, _fullCount(fullCount) , _fullCount(fullCount)
, _skippedBefore(skippedBefore) , _skippedBefore(skippedBefore)
, _skippedAfter(skippedAfter) { , _skippedAfter(skippedAfter) {
} }
void UserPhotosSlice::reverse() {
ranges::reverse(_ids);
std::swap(_skippedBefore, _skippedAfter);
}
base::optional<int> UserPhotosSlice::indexOf(PhotoId photoId) const { base::optional<int> UserPhotosSlice::indexOf(PhotoId photoId) const {
auto it = ranges::find(_ids, photoId); auto it = ranges::find(_ids, photoId);
if (it != _ids.end()) { if (it != _ids.end()) {
@ -109,30 +114,13 @@ base::optional<int> UserPhotosSlice::distance(const Key &a, const Key &b) const
return base::none; return base::none;
} }
QString UserPhotosSlice::debug() const {
auto before = _skippedBefore
? (*_skippedBefore
? ('(' + QString::number(*_skippedBefore) + ").. ")
: QString())
: QString(".. ");
auto after = _skippedAfter
? (" ..(" + QString::number(_skippedAfter) + ')')
: QString(" ..");
auto middle = (size() > 2)
? QString::number((*this)[0]) + " .. " + QString::number((*this)[size() - 1])
: (size() > 1)
? QString::number((*this)[0]) + ' ' + QString::number((*this)[1])
: ((size() > 0) ? QString::number((*this)[0]) : QString());
return before + middle + after;
}
UserPhotosSliceBuilder::UserPhotosSliceBuilder( UserPhotosSliceBuilder::UserPhotosSliceBuilder(
Key key, Key key,
int limitBefore, int limitBefore,
int limitAfter) int limitAfter)
: _key(key) : _key(key)
, _limitBefore(limitBefore) , _limitBefore(limitBefore)
, _limitAfter(limitAfter) { , _limitAfter(limitAfter) {
} }
bool UserPhotosSliceBuilder::applyUpdate(const Storage::UserPhotosResult &update) { bool UserPhotosSliceBuilder::applyUpdate(const Storage::UserPhotosResult &update) {
@ -209,7 +197,12 @@ void UserPhotosSliceBuilder::sliceToLimits() {
} }
UserPhotosSlice UserPhotosSliceBuilder::snapshot() const { UserPhotosSlice UserPhotosSliceBuilder::snapshot() const {
return UserPhotosSlice(_key, _ids, _fullCount, _skippedBefore, _skippedAfter); return UserPhotosSlice(
_key,
base::duplicate(_ids),
_fullCount,
_skippedBefore,
_skippedAfter);
} }
rpl::producer<UserPhotosSlice> UserPhotosViewer( rpl::producer<UserPhotosSlice> UserPhotosViewer(
@ -249,3 +242,15 @@ rpl::producer<UserPhotosSlice> UserPhotosViewer(
return lifetime; return lifetime;
}; };
} }
rpl::producer<UserPhotosSlice> UserPhotosReversedViewer(
UserPhotosSlice::Key key,
int limitBefore,
int limitAfter) {
return UserPhotosViewer(key, limitBefore, limitAfter)
| rpl::map([](UserPhotosSlice &&slice) {
slice.reverse();
return std::move(slice);
});
}

View File

@ -30,29 +30,29 @@ public:
UserPhotosSlice(Key key); UserPhotosSlice(Key key);
UserPhotosSlice( UserPhotosSlice(
Key key, Key key,
const std::deque<PhotoId> &ids, std::deque<PhotoId> &&ids,
base::optional<int> fullCount, base::optional<int> fullCount,
base::optional<int> skippedBefore, base::optional<int> skippedBefore,
int skippedAfter); base::optional<int> skippedAfter);
void reverse();
const Key &key() const { return _key; } const Key &key() const { return _key; }
base::optional<int> fullCount() const { return _fullCount; } base::optional<int> fullCount() const { return _fullCount; }
base::optional<int> skippedBefore() const { return _skippedBefore; } base::optional<int> skippedBefore() const { return _skippedBefore; }
int skippedAfter() const { return _skippedAfter; } base::optional<int> skippedAfter() const { return _skippedAfter; }
base::optional<int> indexOf(PhotoId msgId) const; base::optional<int> indexOf(PhotoId msgId) const;
int size() const { return _ids.size(); } int size() const { return _ids.size(); }
PhotoId operator[](int index) const; PhotoId operator[](int index) const;
base::optional<int> distance(const Key &a, const Key &b) const; base::optional<int> distance(const Key &a, const Key &b) const;
QString debug() const;
private: private:
Key _key; Key _key;
std::deque<PhotoId> _ids; std::deque<PhotoId> _ids;
base::optional<int> _fullCount; base::optional<int> _fullCount;
base::optional<int> _skippedBefore; base::optional<int> _skippedBefore;
int _skippedAfter = 0; base::optional<int> _skippedAfter;
friend class UserPhotosSliceBuilder; friend class UserPhotosSliceBuilder;
@ -62,3 +62,8 @@ rpl::producer<UserPhotosSlice> UserPhotosViewer(
UserPhotosSlice::Key key, UserPhotosSlice::Key key,
int limitBefore, int limitBefore,
int limitAfter); int limitAfter);
rpl::producer<UserPhotosSlice> UserPhotosReversedViewer(
UserPhotosSlice::Key key,
int limitBefore,
int limitAfter);

View File

@ -1071,7 +1071,10 @@ bool MediaView::validSharedMedia() const {
void MediaView::validateSharedMedia() { void MediaView::validateSharedMedia() {
if (auto key = sharedMediaKey()) { if (auto key = sharedMediaKey()) {
_sharedMedia = std::make_unique<SharedMedia>(*key); _sharedMedia = std::make_unique<SharedMedia>(*key);
SharedMediaWithLastViewer( auto viewer = (key->type == SharedMediaType::ChatPhoto)
? SharedMediaWithLastReversedViewer
: SharedMediaWithLastViewer;
viewer(
*key, *key,
kIdsLimit, kIdsLimit,
kIdsLimit kIdsLimit
@ -1134,7 +1137,7 @@ bool MediaView::validUserPhotos() const {
void MediaView::validateUserPhotos() { void MediaView::validateUserPhotos() {
if (auto key = userPhotosKey()) { if (auto key = userPhotosKey()) {
_userPhotos = std::make_unique<UserPhotos>(*key); _userPhotos = std::make_unique<UserPhotos>(*key);
UserPhotosViewer( UserPhotosReversedViewer(
*key, *key,
kIdsLimit, kIdsLimit,
kIdsLimit kIdsLimit
@ -2811,6 +2814,7 @@ void MediaView::updateImage() {
} }
void MediaView::findCurrent() { void MediaView::findCurrent() {
using namespace rpl::mappers;
if (_sharedMediaData) { if (_sharedMediaData) {
_index = _msgid _index = _msgid
? _sharedMediaData->indexOf(_msgid) ? _sharedMediaData->indexOf(_msgid)

View File

@ -76,9 +76,6 @@ int SparseIdsList::addRangeItemsAndCountNew(
const Range &messages, const Range &messages,
MsgRange noSkipRange) { MsgRange noSkipRange) {
Expects(noSkipRange.from <= noSkipRange.till); Expects(noSkipRange.from <= noSkipRange.till);
if (messages.begin() == messages.end()) {
return 0;
}
auto uniteFrom = ranges::lower_bound( auto uniteFrom = ranges::lower_bound(
_slices, _slices,