2023-05-03 18:30:37 +00:00
|
|
|
/*
|
|
|
|
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_stories.h"
|
|
|
|
|
|
|
|
#include "api/api_text_entities.h"
|
2023-05-26 09:27:34 +00:00
|
|
|
#include "apiwrap.h"
|
|
|
|
#include "data/data_changes.h"
|
2023-05-03 18:30:37 +00:00
|
|
|
#include "data/data_document.h"
|
2023-05-26 09:27:34 +00:00
|
|
|
#include "data/data_file_origin.h"
|
2023-05-03 18:30:37 +00:00
|
|
|
#include "data/data_photo.h"
|
|
|
|
#include "data/data_session.h"
|
2023-05-26 09:27:34 +00:00
|
|
|
#include "lang/lang_keys.h"
|
2023-05-03 18:30:37 +00:00
|
|
|
#include "main/main_session.h"
|
2023-05-26 09:27:34 +00:00
|
|
|
#include "ui/text/text_utilities.h"
|
2023-05-03 18:30:37 +00:00
|
|
|
|
|
|
|
// #TODO stories testing
|
|
|
|
#include "data/data_user.h"
|
|
|
|
#include "history/history.h"
|
|
|
|
#include "history/history_item.h"
|
|
|
|
#include "storage/storage_shared_media.h"
|
|
|
|
|
|
|
|
namespace Data {
|
|
|
|
namespace {
|
|
|
|
|
2023-05-26 09:27:34 +00:00
|
|
|
constexpr auto kMaxResolveTogether = 100;
|
|
|
|
|
|
|
|
using UpdateFlag = StoryUpdate::Flag;
|
|
|
|
|
2023-05-03 18:30:37 +00:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
bool StoriesList::unread() const {
|
2023-05-26 07:21:19 +00:00
|
|
|
return !ids.empty() && readTill < ids.front();
|
|
|
|
}
|
|
|
|
|
|
|
|
Story::Story(
|
|
|
|
StoryId id,
|
|
|
|
not_null<PeerData*> peer,
|
|
|
|
StoryMedia media,
|
|
|
|
TimeId date)
|
|
|
|
: _id(id)
|
|
|
|
, _peer(peer)
|
|
|
|
, _media(std::move(media))
|
|
|
|
, _date(date) {
|
|
|
|
}
|
|
|
|
|
|
|
|
Session &Story::owner() const {
|
|
|
|
return _peer->owner();
|
|
|
|
}
|
|
|
|
|
|
|
|
Main::Session &Story::session() const {
|
|
|
|
return _peer->session();
|
|
|
|
}
|
|
|
|
|
|
|
|
not_null<PeerData*> Story::peer() const {
|
|
|
|
return _peer;
|
|
|
|
}
|
|
|
|
|
|
|
|
StoryId Story::id() const {
|
|
|
|
return _id;
|
|
|
|
}
|
|
|
|
|
2023-05-26 09:27:34 +00:00
|
|
|
FullStoryId Story::fullId() const {
|
|
|
|
return { _peer->id, _id };
|
|
|
|
}
|
|
|
|
|
2023-05-26 07:21:19 +00:00
|
|
|
TimeId Story::date() const {
|
|
|
|
return _date;
|
|
|
|
}
|
|
|
|
|
|
|
|
const StoryMedia &Story::media() const {
|
|
|
|
return _media;
|
|
|
|
}
|
|
|
|
|
|
|
|
PhotoData *Story::photo() const {
|
|
|
|
const auto result = std::get_if<not_null<PhotoData*>>(&_media.data);
|
|
|
|
return result ? result->get() : nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
DocumentData *Story::document() const {
|
|
|
|
const auto result = std::get_if<not_null<DocumentData*>>(&_media.data);
|
|
|
|
return result ? result->get() : nullptr;
|
|
|
|
}
|
|
|
|
|
2023-05-26 09:27:34 +00:00
|
|
|
bool Story::hasReplyPreview() const {
|
|
|
|
return v::match(_media.data, [](not_null<PhotoData*> photo) {
|
|
|
|
return !photo->isNull();
|
|
|
|
}, [](not_null<DocumentData*> document) {
|
|
|
|
return document->hasThumbnail();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
Image *Story::replyPreview() const {
|
|
|
|
return v::match(_media.data, [&](not_null<PhotoData*> photo) {
|
|
|
|
return photo->getReplyPreview(
|
|
|
|
Data::FileOriginStory(_peer->id, _id),
|
|
|
|
_peer,
|
|
|
|
false);
|
|
|
|
}, [&](not_null<DocumentData*> document) {
|
|
|
|
return document->getReplyPreview(
|
|
|
|
Data::FileOriginStory(_peer->id, _id),
|
|
|
|
_peer,
|
|
|
|
false);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
TextWithEntities Story::inReplyText() const {
|
|
|
|
const auto type = u"Story"_q;
|
|
|
|
return _caption.text.isEmpty()
|
|
|
|
? Ui::Text::PlainLink(type)
|
|
|
|
: tr::lng_dialogs_text_media(
|
|
|
|
tr::now,
|
|
|
|
lt_media_part,
|
|
|
|
tr::lng_dialogs_text_media_wrapped(
|
|
|
|
tr::now,
|
|
|
|
lt_media,
|
|
|
|
Ui::Text::PlainLink(type),
|
|
|
|
Ui::Text::WithEntities),
|
|
|
|
lt_caption,
|
|
|
|
_caption,
|
|
|
|
Ui::Text::WithEntities);
|
|
|
|
}
|
|
|
|
|
2023-05-26 07:21:19 +00:00
|
|
|
void Story::setPinned(bool pinned) {
|
|
|
|
_pinned = pinned;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Story::pinned() const {
|
|
|
|
return _pinned;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Story::setCaption(TextWithEntities &&caption) {
|
|
|
|
_caption = std::move(caption);
|
|
|
|
}
|
|
|
|
|
|
|
|
const TextWithEntities &Story::caption() const {
|
|
|
|
return _caption;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Story::apply(const MTPDstoryItem &data) {
|
|
|
|
_pinned = data.is_pinned();
|
|
|
|
_caption = TextWithEntities{
|
|
|
|
data.vcaption().value_or_empty(),
|
|
|
|
Api::EntitiesFromMTP(
|
|
|
|
&owner().session(),
|
|
|
|
data.ventities().value_or_empty()),
|
|
|
|
};
|
2023-05-03 18:30:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Stories::Stories(not_null<Session*> owner) : _owner(owner) {
|
|
|
|
}
|
|
|
|
|
|
|
|
Stories::~Stories() {
|
|
|
|
}
|
|
|
|
|
|
|
|
Session &Stories::owner() const {
|
|
|
|
return *_owner;
|
|
|
|
}
|
|
|
|
|
2023-05-26 09:27:34 +00:00
|
|
|
Main::Session &Stories::session() const {
|
|
|
|
return _owner->session();
|
|
|
|
}
|
|
|
|
|
2023-05-03 18:30:37 +00:00
|
|
|
void Stories::apply(const MTPDupdateStories &data) {
|
2023-05-26 14:48:33 +00:00
|
|
|
applyChanges(parse(data.vstories()));
|
2023-05-26 07:21:19 +00:00
|
|
|
_allChanged.fire({});
|
2023-05-03 18:30:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
StoriesList Stories::parse(const MTPUserStories &stories) {
|
|
|
|
const auto &data = stories.data();
|
|
|
|
const auto userId = UserId(data.vuser_id());
|
|
|
|
const auto readTill = data.vmax_read_id().value_or_empty();
|
|
|
|
const auto count = int(data.vstories().v.size());
|
|
|
|
auto result = StoriesList{
|
|
|
|
.user = _owner->user(userId),
|
|
|
|
.readTill = readTill,
|
|
|
|
.total = count,
|
|
|
|
};
|
|
|
|
const auto &list = data.vstories().v;
|
2023-05-26 07:21:19 +00:00
|
|
|
result.ids.reserve(list.size());
|
2023-05-03 18:30:37 +00:00
|
|
|
for (const auto &story : list) {
|
|
|
|
story.match([&](const MTPDstoryItem &data) {
|
2023-05-26 07:21:19 +00:00
|
|
|
if (const auto story = parse(result.user, data)) {
|
|
|
|
result.ids.push_back(story->id());
|
2023-05-03 18:30:37 +00:00
|
|
|
} else {
|
|
|
|
--result.total;
|
|
|
|
}
|
2023-05-26 07:21:19 +00:00
|
|
|
}, [&](const MTPDstoryItemSkipped &data) {
|
|
|
|
result.ids.push_back(data.vid().v);
|
|
|
|
}, [&](const MTPDstoryItemDeleted &data) {
|
2023-05-26 09:27:34 +00:00
|
|
|
applyDeleted({ peerFromUser(userId), data.vid().v });
|
2023-05-03 18:30:37 +00:00
|
|
|
--result.total;
|
|
|
|
});
|
|
|
|
}
|
2023-05-26 07:21:19 +00:00
|
|
|
result.total = std::max(result.total, int(result.ids.size()));
|
2023-05-03 18:30:37 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-05-26 07:21:19 +00:00
|
|
|
Story *Stories::parse(not_null<PeerData*> peer, const MTPDstoryItem &data) {
|
2023-05-03 18:30:37 +00:00
|
|
|
const auto id = data.vid().v;
|
2023-05-26 07:21:19 +00:00
|
|
|
auto &stories = _stories[peer->id];
|
|
|
|
const auto i = stories.find(id);
|
|
|
|
if (i != end(stories)) {
|
|
|
|
i->second->apply(data);
|
|
|
|
return i->second.get();
|
|
|
|
}
|
2023-05-03 18:30:37 +00:00
|
|
|
using MaybeMedia = std::optional<
|
|
|
|
std::variant<not_null<PhotoData*>, not_null<DocumentData*>>>;
|
|
|
|
const auto media = data.vmedia().match([&](
|
|
|
|
const MTPDmessageMediaPhoto &data) -> MaybeMedia {
|
|
|
|
if (const auto photo = data.vphoto()) {
|
|
|
|
const auto result = _owner->processPhoto(*photo);
|
|
|
|
if (!result->isNull()) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}, [&](const MTPDmessageMediaDocument &data) -> MaybeMedia {
|
|
|
|
if (const auto document = data.vdocument()) {
|
|
|
|
const auto result = _owner->processDocument(*document);
|
|
|
|
if (!result->isNull()
|
|
|
|
&& (result->isGifv() || result->isVideoFile())) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}, [](const auto &) { return MaybeMedia(); });
|
|
|
|
if (!media) {
|
2023-05-26 07:21:19 +00:00
|
|
|
return nullptr;
|
2023-05-03 18:30:37 +00:00
|
|
|
}
|
2023-05-26 07:21:19 +00:00
|
|
|
const auto result = stories.emplace(id, std::make_unique<Story>(
|
|
|
|
id,
|
|
|
|
peer,
|
|
|
|
StoryMedia{ *media },
|
|
|
|
data.vdate().v)).first->second.get();
|
|
|
|
result->apply(data);
|
|
|
|
return result;
|
2023-05-03 18:30:37 +00:00
|
|
|
}
|
|
|
|
|
2023-05-26 09:27:34 +00:00
|
|
|
void Stories::updateDependentMessages(not_null<Data::Story*> story) {
|
|
|
|
const auto i = _dependentMessages.find(story);
|
|
|
|
if (i != end(_dependentMessages)) {
|
|
|
|
for (const auto &dependent : i->second) {
|
|
|
|
dependent->updateDependencyItem();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
session().changes().storyUpdated(
|
|
|
|
story,
|
|
|
|
Data::StoryUpdate::Flag::Edited);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Stories::registerDependentMessage(
|
|
|
|
not_null<HistoryItem*> dependent,
|
|
|
|
not_null<Data::Story*> dependency) {
|
|
|
|
_dependentMessages[dependency].emplace(dependent);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Stories::unregisterDependentMessage(
|
|
|
|
not_null<HistoryItem*> dependent,
|
|
|
|
not_null<Data::Story*> dependency) {
|
|
|
|
const auto i = _dependentMessages.find(dependency);
|
|
|
|
if (i != end(_dependentMessages)) {
|
|
|
|
if (i->second.remove(dependent) && i->second.empty()) {
|
|
|
|
_dependentMessages.erase(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-03 18:30:37 +00:00
|
|
|
void Stories::loadMore() {
|
|
|
|
if (_loadMoreRequestId || _allLoaded) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const auto api = &_owner->session().api();
|
|
|
|
using Flag = MTPstories_GetAllStories::Flag;
|
|
|
|
_loadMoreRequestId = api->request(MTPstories_GetAllStories(
|
|
|
|
MTP_flags(_state.isEmpty() ? Flag(0) : Flag::f_next),
|
|
|
|
MTP_string(_state)
|
|
|
|
)).done([=](const MTPstories_AllStories &result) {
|
|
|
|
_loadMoreRequestId = 0;
|
|
|
|
|
|
|
|
result.match([&](const MTPDstories_allStories &data) {
|
|
|
|
_owner->processUsers(data.vusers());
|
|
|
|
_state = qs(data.vstate());
|
|
|
|
_allLoaded = !data.is_has_more();
|
|
|
|
for (const auto &single : data.vuser_stories().v) {
|
|
|
|
pushToBack(parse(single));
|
|
|
|
}
|
2023-05-26 07:21:19 +00:00
|
|
|
_allChanged.fire({});
|
2023-05-03 18:30:37 +00:00
|
|
|
}, [](const MTPDstories_allStoriesNotModified &) {
|
|
|
|
});
|
|
|
|
}).fail([=] {
|
|
|
|
_loadMoreRequestId = 0;
|
|
|
|
}).send();
|
|
|
|
}
|
|
|
|
|
2023-05-26 09:27:34 +00:00
|
|
|
void Stories::sendResolveRequests() {
|
|
|
|
if (!_resolveRequests.empty()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
struct Prepared {
|
|
|
|
QVector<MTPint> ids;
|
|
|
|
std::vector<Fn<void()>> callbacks;
|
|
|
|
};
|
|
|
|
auto leftToSend = kMaxResolveTogether;
|
|
|
|
auto byPeer = base::flat_map<PeerId, Prepared>();
|
|
|
|
for (auto i = begin(_resolves); i != end(_resolves);) {
|
|
|
|
auto &[peerId, ids] = *i;
|
|
|
|
auto &prepared = byPeer[peerId];
|
|
|
|
for (auto &[storyId, callbacks] : ids) {
|
|
|
|
prepared.ids.push_back(MTP_int(storyId));
|
|
|
|
prepared.callbacks.insert(
|
|
|
|
end(prepared.callbacks),
|
|
|
|
std::make_move_iterator(begin(callbacks)),
|
|
|
|
std::make_move_iterator(end(callbacks)));
|
|
|
|
if (!--leftToSend) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const auto sending = int(prepared.ids.size());
|
|
|
|
if (sending == ids.size()) {
|
|
|
|
i = _resolves.erase(i);
|
|
|
|
if (!leftToSend) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ids.erase(begin(ids), begin(ids) + sending);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const auto api = &_owner->session().api();
|
|
|
|
for (auto &entry : byPeer) {
|
|
|
|
const auto peerId = entry.first;
|
|
|
|
auto &prepared = entry.second;
|
|
|
|
const auto finish = [=, ids = prepared.ids](mtpRequestId id) {
|
|
|
|
for (const auto &id : ids) {
|
|
|
|
finalizeResolve({ peerId, id.v });
|
|
|
|
}
|
|
|
|
if (auto callbacks = _resolveRequests.take(id)) {
|
|
|
|
for (const auto &callback : *callbacks) {
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (_resolveRequests.empty() && !_resolves.empty()) {
|
|
|
|
crl::on_main(&session(), [=] { sendResolveRequests(); });
|
|
|
|
}
|
|
|
|
};
|
|
|
|
const auto user = _owner->session().data().peer(peerId)->asUser();
|
|
|
|
if (!user) {
|
|
|
|
_resolveRequests[0] = std::move(prepared.callbacks);
|
|
|
|
finish(0);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const auto requestId = api->request(MTPstories_GetStoriesByID(
|
|
|
|
user->inputUser,
|
|
|
|
MTP_vector<MTPint>(std::move(prepared.ids))
|
|
|
|
)).done([=](const MTPstories_Stories &result, mtpRequestId id) {
|
|
|
|
owner().processUsers(result.data().vusers());
|
|
|
|
processResolvedStories(user, result.data().vstories().v);
|
|
|
|
finish(id);
|
|
|
|
}).fail([=](const MTP::Error &error, mtpRequestId id) {
|
|
|
|
finish(id);
|
|
|
|
}).send();
|
|
|
|
_resolveRequests.emplace(requestId, std::move(prepared.callbacks));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Stories::processResolvedStories(
|
|
|
|
not_null<PeerData*> peer,
|
|
|
|
const QVector<MTPStoryItem> &list) {
|
|
|
|
for (const auto &item : list) {
|
|
|
|
item.match([&](const MTPDstoryItem &data) {
|
|
|
|
[[maybe_unused]] const auto story = parse(peer, data);
|
|
|
|
}, [&](const MTPDstoryItemSkipped &data) {
|
|
|
|
LOG(("API Error: Unexpected storyItemSkipped in resolve."));
|
|
|
|
}, [&](const MTPDstoryItemDeleted &data) {
|
|
|
|
applyDeleted({ peer->id, data.vid().v });
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Stories::finalizeResolve(FullStoryId id) {
|
|
|
|
const auto already = lookup(id);
|
|
|
|
if (!already.has_value() && already.error() == NoStory::Unknown) {
|
|
|
|
LOG(("API Error: Could not resolve story %1_%2"
|
|
|
|
).arg(id.peer.value
|
|
|
|
).arg(id.story));
|
|
|
|
applyDeleted(id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Stories::applyDeleted(FullStoryId id) {
|
|
|
|
const auto i = _stories.find(id.peer);
|
|
|
|
if (i != end(_stories)) {
|
|
|
|
const auto j = i->second.find(id.story);
|
|
|
|
if (j != end(i->second)) {
|
|
|
|
auto story = std::move(j->second);
|
|
|
|
i->second.erase(j);
|
|
|
|
session().changes().storyUpdated(
|
|
|
|
story.get(),
|
|
|
|
UpdateFlag::Destroyed);
|
2023-05-26 14:48:33 +00:00
|
|
|
removeDependencyStory(story.get());
|
2023-05-26 09:27:34 +00:00
|
|
|
if (i->second.empty()) {
|
|
|
|
_stories.erase(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-05-26 14:48:33 +00:00
|
|
|
const auto j = ranges::find(_all, id.peer, [](const StoriesList &list) {
|
|
|
|
return list.user->id;
|
|
|
|
});
|
|
|
|
if (j != end(_all)) {
|
|
|
|
const auto till = ranges::remove(j->ids, id.story);
|
|
|
|
const auto removed = int(std::distance(till, end(j->ids)));
|
|
|
|
if (till != end(j->ids)) {
|
|
|
|
j->ids.erase(till, end(j->ids));
|
|
|
|
j->total = std::max(j->total - removed, 0);
|
|
|
|
if (j->ids.empty()) {
|
|
|
|
_all.erase(j);
|
|
|
|
}
|
|
|
|
_allChanged.fire({});
|
|
|
|
}
|
|
|
|
}
|
2023-05-26 09:27:34 +00:00
|
|
|
_deleted.emplace(id);
|
|
|
|
}
|
|
|
|
|
2023-05-26 14:48:33 +00:00
|
|
|
void Stories::removeDependencyStory(not_null<Story*> story) {
|
|
|
|
const auto i = _dependentMessages.find(story);
|
|
|
|
if (i != end(_dependentMessages)) {
|
|
|
|
const auto items = std::move(i->second);
|
|
|
|
_dependentMessages.erase(i);
|
|
|
|
|
|
|
|
for (const auto &dependent : items) {
|
|
|
|
dependent->dependencyStoryRemoved(story);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-03 18:30:37 +00:00
|
|
|
const std::vector<StoriesList> &Stories::all() {
|
|
|
|
return _all;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Stories::allLoaded() const {
|
|
|
|
return _allLoaded;
|
|
|
|
}
|
|
|
|
|
2023-05-22 15:59:16 +00:00
|
|
|
rpl::producer<> Stories::allChanged() const {
|
|
|
|
return _allChanged.events();
|
|
|
|
}
|
|
|
|
|
2023-05-26 07:21:19 +00:00
|
|
|
base::expected<not_null<Story*>, NoStory> Stories::lookup(
|
|
|
|
FullStoryId id) const {
|
|
|
|
const auto i = _stories.find(id.peer);
|
|
|
|
if (i != end(_stories)) {
|
|
|
|
const auto j = i->second.find(id.story);
|
|
|
|
if (j != end(i->second)) {
|
|
|
|
return j->second.get();
|
2023-05-03 18:30:37 +00:00
|
|
|
}
|
2023-05-26 07:21:19 +00:00
|
|
|
}
|
|
|
|
return base::make_unexpected(
|
|
|
|
_deleted.contains(id) ? NoStory::Deleted : NoStory::Unknown);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Stories::resolve(FullStoryId id, Fn<void()> done) {
|
2023-05-26 09:27:34 +00:00
|
|
|
const auto already = lookup(id);
|
|
|
|
if (already.has_value() || already.error() != NoStory::Unknown) {
|
|
|
|
done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto &ids = _resolves[id.peer];
|
|
|
|
if (ids.empty()) {
|
|
|
|
crl::on_main(&session(), [=] {
|
|
|
|
sendResolveRequests();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
auto &callbacks = ids[id.story];
|
|
|
|
if (done) {
|
|
|
|
callbacks.push_back(std::move(done));
|
|
|
|
}
|
2023-05-03 18:30:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Stories::pushToBack(StoriesList &&list) {
|
|
|
|
const auto i = ranges::find(_all, list.user, &StoriesList::user);
|
|
|
|
if (i != end(_all)) {
|
2023-05-22 15:59:16 +00:00
|
|
|
if (*i == list) {
|
|
|
|
return;
|
|
|
|
}
|
2023-05-03 18:30:37 +00:00
|
|
|
*i = std::move(list);
|
|
|
|
} else {
|
|
|
|
_all.push_back(std::move(list));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-26 14:48:33 +00:00
|
|
|
void Stories::applyChanges(StoriesList &&list) {
|
2023-05-03 18:30:37 +00:00
|
|
|
const auto i = ranges::find(_all, list.user, &StoriesList::user);
|
|
|
|
if (i != end(_all)) {
|
2023-05-26 14:48:33 +00:00
|
|
|
auto added = false;
|
|
|
|
for (const auto id : list.ids) {
|
|
|
|
if (!ranges::contains(i->ids, id)) {
|
|
|
|
i->ids.insert(begin(i->ids), id);
|
|
|
|
++i->total;
|
|
|
|
added = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (added) {
|
|
|
|
ranges::rotate(begin(_all), i, i + 1);
|
|
|
|
}
|
|
|
|
} else if (!list.ids.empty()) {
|
2023-05-03 18:30:37 +00:00
|
|
|
_all.insert(begin(_all), std::move(list));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Data
|