895 lines
26 KiB
C++
895 lines
26 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
|
|
*/
|
|
#include "history/history_item_helpers.h"
|
|
|
|
#include "api/api_text_entities.h"
|
|
#include "boxes/premium_preview_box.h"
|
|
#include "calls/calls_instance.h"
|
|
#include "data/components/sponsored_messages.h"
|
|
#include "data/stickers/data_custom_emoji.h"
|
|
#include "data/notify/data_notify_settings.h"
|
|
#include "data/data_channel.h"
|
|
#include "data/data_chat.h"
|
|
#include "data/data_changes.h"
|
|
#include "data/data_document.h"
|
|
#include "data/data_group_call.h"
|
|
#include "data/data_forum.h"
|
|
#include "data/data_forum_topic.h"
|
|
#include "data/data_message_reactions.h"
|
|
#include "data/data_session.h"
|
|
#include "data/data_stories.h"
|
|
#include "data/data_user.h"
|
|
#include "history/history.h"
|
|
#include "history/history_item_components.h"
|
|
#include "main/main_account.h"
|
|
#include "main/main_domain.h"
|
|
#include "main/main_session.h"
|
|
#include "main/main_session_settings.h"
|
|
#include "menu/menu_sponsored.h"
|
|
#include "platform/platform_notifications_manager.h"
|
|
#include "window/window_controller.h"
|
|
#include "window/window_session_controller.h"
|
|
#include "apiwrap.h"
|
|
#include "base/unixtime.h"
|
|
#include "core/application.h"
|
|
#include "core/click_handler_types.h" // ClickHandlerContext.
|
|
#include "ui/text/format_values.h"
|
|
#include "ui/text/text_utilities.h"
|
|
#include "ui/toast/toast.h"
|
|
#include "ui/item_text_options.h"
|
|
#include "lang/lang_keys.h"
|
|
|
|
namespace {
|
|
|
|
bool PeerCallKnown(not_null<PeerData*> peer) {
|
|
if (peer->groupCall() != nullptr) {
|
|
return true;
|
|
} else if (const auto chat = peer->asChat()) {
|
|
return !(chat->flags() & ChatDataFlag::CallActive);
|
|
} else if (const auto channel = peer->asChannel()) {
|
|
return !(channel->flags() & ChannelDataFlag::CallActive);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
QString GetErrorTextForSending(
|
|
not_null<PeerData*> peer,
|
|
SendingErrorRequest request) {
|
|
const auto forum = request.topicRootId ? peer->forum() : nullptr;
|
|
const auto topic = forum
|
|
? forum->topicFor(request.topicRootId)
|
|
: nullptr;
|
|
const auto thread = topic
|
|
? not_null<Data::Thread*>(topic)
|
|
: peer->owner().history(peer);
|
|
if (request.story) {
|
|
if (const auto error = request.story->errorTextForForward(thread)) {
|
|
return *error;
|
|
}
|
|
}
|
|
if (request.forward) {
|
|
for (const auto &item : *request.forward) {
|
|
if (const auto error = item->errorTextForForward(thread)) {
|
|
return *error;
|
|
}
|
|
}
|
|
}
|
|
const auto hasText = (request.text && !request.text->empty());
|
|
if (hasText) {
|
|
const auto error = Data::RestrictionError(
|
|
peer,
|
|
ChatRestriction::SendOther);
|
|
if (error) {
|
|
return *error;
|
|
} else if (!Data::CanSendTexts(thread)) {
|
|
return tr::lng_forward_cant(tr::now);
|
|
}
|
|
}
|
|
if (peer->slowmodeApplied()) {
|
|
const auto count = (hasText ? 1 : 0)
|
|
+ (request.story ? 1 : 0)
|
|
+ (request.forward ? int(request.forward->size()) : 0);
|
|
if (const auto history = peer->owner().historyLoaded(peer)) {
|
|
if (!request.ignoreSlowmodeCountdown
|
|
&& (history->latestSendingMessage() != nullptr)
|
|
&& (count > 0)) {
|
|
return tr::lng_slowmode_no_many(tr::now);
|
|
}
|
|
}
|
|
if (request.text && request.text->text.size() > MaxMessageSize) {
|
|
return tr::lng_slowmode_too_long(tr::now);
|
|
} else if ((hasText || request.story) && count > 1) {
|
|
return tr::lng_slowmode_no_many(tr::now);
|
|
} else if (count > 1) {
|
|
const auto albumForward = [&] {
|
|
const auto first = request.forward->front();
|
|
if (const auto groupId = first->groupId()) {
|
|
for (const auto &item : *request.forward) {
|
|
if (item->groupId() != groupId) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}();
|
|
if (!albumForward) {
|
|
return tr::lng_slowmode_no_many(tr::now);
|
|
}
|
|
}
|
|
}
|
|
if (const auto left = peer->slowmodeSecondsLeft()) {
|
|
if (!request.ignoreSlowmodeCountdown) {
|
|
return tr::lng_slowmode_enabled(
|
|
tr::now,
|
|
lt_left,
|
|
Ui::FormatDurationWordsSlowmode(left));
|
|
}
|
|
}
|
|
|
|
return QString();
|
|
}
|
|
|
|
QString GetErrorTextForSending(
|
|
not_null<Data::Thread*> thread,
|
|
SendingErrorRequest request) {
|
|
request.topicRootId = thread->topicRootId();
|
|
return GetErrorTextForSending(thread->peer(), std::move(request));
|
|
}
|
|
|
|
void RequestDependentMessageItem(
|
|
not_null<HistoryItem*> item,
|
|
PeerId peerId,
|
|
MsgId msgId) {
|
|
if (!IsServerMsgId(msgId)) {
|
|
return;
|
|
}
|
|
const auto fullId = item->fullId();
|
|
const auto history = item->history();
|
|
const auto session = &history->session();
|
|
const auto done = [=] {
|
|
if (const auto item = session->data().message(fullId)) {
|
|
item->updateDependencyItem();
|
|
}
|
|
};
|
|
history->session().api().requestMessageData(
|
|
(peerId ? history->owner().peer(peerId) : history->peer),
|
|
msgId,
|
|
done);
|
|
}
|
|
|
|
void RequestDependentMessageStory(
|
|
not_null<HistoryItem*> item,
|
|
PeerId peerId,
|
|
StoryId storyId) {
|
|
const auto fullId = item->fullId();
|
|
const auto history = item->history();
|
|
const auto session = &history->session();
|
|
const auto done = [=] {
|
|
if (const auto item = session->data().message(fullId)) {
|
|
item->updateDependencyItem();
|
|
}
|
|
};
|
|
history->owner().stories().resolve(
|
|
{ peerId ? peerId : history->peer->id, storyId },
|
|
done);
|
|
}
|
|
|
|
MessageFlags NewMessageFlags(not_null<PeerData*> peer) {
|
|
return MessageFlag::BeingSent
|
|
| (peer->isSelf() ? MessageFlag() : MessageFlag::Outgoing);
|
|
}
|
|
|
|
bool ShouldSendSilent(
|
|
not_null<PeerData*> peer,
|
|
const Api::SendOptions &options) {
|
|
return options.silent
|
|
|| (peer->isBroadcast()
|
|
&& peer->owner().notifySettings().silentPosts(peer))
|
|
|| (peer->session().supportMode()
|
|
&& peer->session().settings().supportAllSilent());
|
|
}
|
|
|
|
HistoryItem *LookupReplyTo(not_null<History*> history, FullMsgId replyTo) {
|
|
return history->owner().message(replyTo);
|
|
}
|
|
|
|
MsgId LookupReplyToTop(not_null<History*> history, HistoryItem *replyTo) {
|
|
return (replyTo && replyTo->history() == history)
|
|
? replyTo->replyToTop()
|
|
: 0;
|
|
}
|
|
|
|
MsgId LookupReplyToTop(not_null<History*> history, FullReplyTo replyTo) {
|
|
return replyTo.topicRootId
|
|
? replyTo.topicRootId
|
|
: LookupReplyToTop(
|
|
history,
|
|
LookupReplyTo(history, replyTo.messageId));
|
|
}
|
|
|
|
bool LookupReplyIsTopicPost(HistoryItem *replyTo) {
|
|
return replyTo
|
|
&& (replyTo->topicRootId() != Data::ForumTopic::kGeneralId);
|
|
}
|
|
|
|
TextWithEntities DropDisallowedCustomEmoji(
|
|
not_null<PeerData*> to,
|
|
TextWithEntities text) {
|
|
if (to->session().premium() || to->isSelf()) {
|
|
return text;
|
|
}
|
|
const auto channel = to->asMegagroup();
|
|
const auto allowSetId = channel ? channel->mgInfo->emojiSet.id : 0;
|
|
if (!allowSetId) {
|
|
text.entities.erase(
|
|
ranges::remove(
|
|
text.entities,
|
|
EntityType::CustomEmoji,
|
|
&EntityInText::type),
|
|
text.entities.end());
|
|
} else {
|
|
const auto predicate = [&](const EntityInText &entity) {
|
|
if (entity.type() != EntityType::CustomEmoji) {
|
|
return false;
|
|
}
|
|
if (const auto id = Data::ParseCustomEmojiData(entity.data())) {
|
|
const auto document = to->owner().document(id);
|
|
if (const auto sticker = document->sticker()) {
|
|
if (sticker->set.id == allowSetId) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
};
|
|
text.entities.erase(
|
|
ranges::remove_if(text.entities, predicate),
|
|
text.entities.end());
|
|
}
|
|
return text;
|
|
}
|
|
|
|
Main::Session *SessionByUniqueId(uint64 sessionUniqueId) {
|
|
if (!sessionUniqueId) {
|
|
return nullptr;
|
|
}
|
|
for (const auto &[index, account] : Core::App().domain().accounts()) {
|
|
if (const auto session = account->maybeSession()) {
|
|
if (session->uniqueId() == sessionUniqueId) {
|
|
return session;
|
|
}
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
HistoryItem *MessageByGlobalId(GlobalMsgId globalId) {
|
|
const auto sessionId = globalId.itemId ? globalId.sessionUniqueId : 0;
|
|
if (const auto session = SessionByUniqueId(sessionId)) {
|
|
return session->data().message(globalId.itemId);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
QDateTime ItemDateTime(not_null<const HistoryItem*> item) {
|
|
return base::unixtime::parse(item->date());
|
|
}
|
|
|
|
QString ItemDateText(not_null<const HistoryItem*> item, bool isUntilOnline) {
|
|
const auto dateText = langDayOfMonthFull(ItemDateTime(item).date());
|
|
return !item->isScheduled()
|
|
? dateText
|
|
: isUntilOnline
|
|
? tr::lng_scheduled_date_until_online(tr::now)
|
|
: tr::lng_scheduled_date(tr::now, lt_date, dateText);
|
|
}
|
|
|
|
bool IsItemScheduledUntilOnline(not_null<const HistoryItem*> item) {
|
|
return item->isScheduled()
|
|
&& (item->date() == Api::kScheduledUntilOnlineTimestamp);
|
|
}
|
|
|
|
ClickHandlerPtr JumpToMessageClickHandler(
|
|
not_null<HistoryItem*> item,
|
|
FullMsgId returnToId,
|
|
TextWithEntities highlightPart,
|
|
int highlightPartOffsetHint) {
|
|
return JumpToMessageClickHandler(
|
|
item->history()->peer,
|
|
item->id,
|
|
returnToId,
|
|
std::move(highlightPart),
|
|
highlightPartOffsetHint);
|
|
}
|
|
|
|
ClickHandlerPtr JumpToMessageClickHandler(
|
|
not_null<PeerData*> peer,
|
|
MsgId msgId,
|
|
FullMsgId returnToId,
|
|
TextWithEntities highlightPart,
|
|
int highlightPartOffsetHint) {
|
|
return std::make_shared<LambdaClickHandler>([=] {
|
|
const auto separate = Core::App().separateWindowForPeer(peer);
|
|
const auto controller = separate
|
|
? separate->sessionController()
|
|
: peer->session().tryResolveWindow();
|
|
if (controller) {
|
|
auto params = Window::SectionShow{
|
|
Window::SectionShow::Way::Forward
|
|
};
|
|
params.highlightPart = highlightPart;
|
|
params.highlightPartOffsetHint = highlightPartOffsetHint;
|
|
params.origin = Window::SectionShow::OriginMessage{
|
|
returnToId
|
|
};
|
|
if (const auto item = peer->owner().message(peer, msgId)) {
|
|
controller->showMessage(item, params);
|
|
} else {
|
|
controller->showPeerHistory(peer, params, msgId);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
ClickHandlerPtr JumpToStoryClickHandler(not_null<Data::Story*> story) {
|
|
return JumpToStoryClickHandler(story->peer(), story->id());
|
|
}
|
|
|
|
ClickHandlerPtr JumpToStoryClickHandler(
|
|
not_null<PeerData*> peer,
|
|
StoryId storyId) {
|
|
return std::make_shared<LambdaClickHandler>([=] {
|
|
const auto separate = Core::App().separateWindowForPeer(peer);
|
|
const auto controller = separate
|
|
? separate->sessionController()
|
|
: peer->session().tryResolveWindow();
|
|
if (controller) {
|
|
controller->openPeerStory(
|
|
peer,
|
|
storyId,
|
|
{ Data::StoriesContextSingle() });
|
|
}
|
|
});
|
|
}
|
|
|
|
ClickHandlerPtr HideSponsoredClickHandler() {
|
|
return std::make_shared<LambdaClickHandler>([=](ClickContext context) {
|
|
const auto my = context.other.value<ClickHandlerContext>();
|
|
if (const auto controller = my.sessionWindow.get()) {
|
|
const auto &session = controller->session();
|
|
if (session.premium()) {
|
|
using Result = Data::SponsoredReportResult;
|
|
session.sponsoredMessages().createReportCallback(
|
|
my.itemId)(Result::Id("-1"), [](const auto &) {});
|
|
} else {
|
|
ShowPremiumPreviewBox(controller, PremiumFeature::NoAds);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
ClickHandlerPtr ReportSponsoredClickHandler(not_null<HistoryItem*> item) {
|
|
return std::make_shared<LambdaClickHandler>([=](ClickContext context) {
|
|
const auto my = context.other.value<ClickHandlerContext>();
|
|
if (const auto controller = my.sessionWindow.get()) {
|
|
Menu::ShowSponsored(
|
|
controller->widget(),
|
|
controller->uiShow(),
|
|
item);
|
|
}
|
|
});
|
|
}
|
|
|
|
MessageFlags FlagsFromMTP(
|
|
MsgId id,
|
|
MTPDmessage::Flags flags,
|
|
MessageFlags localFlags) {
|
|
using Flag = MessageFlag;
|
|
using MTP = MTPDmessage::Flag;
|
|
return localFlags
|
|
| (IsServerMsgId(id) ? Flag::HistoryEntry : Flag())
|
|
| ((flags & MTP::f_out) ? Flag::Outgoing : Flag())
|
|
| ((flags & MTP::f_mentioned) ? Flag::MentionsMe : Flag())
|
|
| ((flags & MTP::f_media_unread) ? Flag::MediaIsUnread : Flag())
|
|
| ((flags & MTP::f_silent) ? Flag::Silent : Flag())
|
|
| ((flags & MTP::f_post) ? Flag::Post : Flag())
|
|
| ((flags & MTP::f_legacy) ? Flag::Legacy : Flag())
|
|
| ((flags & MTP::f_edit_hide) ? Flag::HideEdited : Flag())
|
|
| ((flags & MTP::f_pinned) ? Flag::Pinned : Flag())
|
|
| ((flags & MTP::f_from_id) ? Flag::HasFromId : Flag())
|
|
| ((flags & MTP::f_reply_to) ? Flag::HasReplyInfo : Flag())
|
|
| ((flags & MTP::f_reply_markup) ? Flag::HasReplyMarkup : Flag())
|
|
| ((flags & MTP::f_quick_reply_shortcut_id)
|
|
? Flag::ShortcutMessage
|
|
: Flag())
|
|
| ((flags & MTP::f_from_scheduled)
|
|
? Flag::IsOrWasScheduled
|
|
: Flag())
|
|
| ((flags & MTP::f_views) ? Flag::HasViews : Flag())
|
|
| ((flags & MTP::f_noforwards) ? Flag::NoForwards : Flag())
|
|
| ((flags & MTP::f_invert_media) ? Flag::InvertMedia : Flag());
|
|
}
|
|
|
|
MessageFlags FlagsFromMTP(
|
|
MsgId id,
|
|
MTPDmessageService::Flags flags,
|
|
MessageFlags localFlags) {
|
|
using Flag = MessageFlag;
|
|
using MTP = MTPDmessageService::Flag;
|
|
return localFlags
|
|
| (IsServerMsgId(id) ? Flag::HistoryEntry : Flag())
|
|
| ((flags & MTP::f_out) ? Flag::Outgoing : Flag())
|
|
| ((flags & MTP::f_mentioned) ? Flag::MentionsMe : Flag())
|
|
| ((flags & MTP::f_media_unread) ? Flag::MediaIsUnread : Flag())
|
|
| ((flags & MTP::f_silent) ? Flag::Silent : Flag())
|
|
| ((flags & MTP::f_post) ? Flag::Post : Flag())
|
|
| ((flags & MTP::f_legacy) ? Flag::Legacy : Flag())
|
|
| ((flags & MTP::f_from_id) ? Flag::HasFromId : Flag())
|
|
| ((flags & MTP::f_reply_to) ? Flag::HasReplyInfo : Flag());
|
|
}
|
|
|
|
MTPMessageReplyHeader NewMessageReplyHeader(const Api::SendAction &action) {
|
|
if (const auto replyTo = action.replyTo) {
|
|
if (replyTo.storyId) {
|
|
return MTP_messageReplyStoryHeader(
|
|
peerToMTP(replyTo.storyId.peer),
|
|
MTP_int(replyTo.storyId.story));
|
|
}
|
|
using Flag = MTPDmessageReplyHeader::Flag;
|
|
const auto historyPeer = action.history->peer->id;
|
|
const auto externalPeerId = (replyTo.messageId.peer == historyPeer)
|
|
? PeerId()
|
|
: replyTo.messageId.peer;
|
|
const auto replyToTop = LookupReplyToTop(action.history, replyTo);
|
|
auto quoteEntities = Api::EntitiesToMTP(
|
|
&action.history->session(),
|
|
replyTo.quote.entities,
|
|
Api::ConvertOption::SkipLocal);
|
|
return MTP_messageReplyHeader(
|
|
MTP_flags(Flag::f_reply_to_msg_id
|
|
| (replyToTop ? Flag::f_reply_to_top_id : Flag())
|
|
| (externalPeerId ? Flag::f_reply_to_peer_id : Flag())
|
|
| (replyTo.quote.empty()
|
|
? Flag()
|
|
: (Flag::f_quote
|
|
| Flag::f_quote_text
|
|
| Flag::f_quote_offset))
|
|
| (quoteEntities.v.empty()
|
|
? Flag()
|
|
: Flag::f_quote_entities)),
|
|
MTP_int(replyTo.messageId.msg),
|
|
peerToMTP(externalPeerId),
|
|
MTPMessageFwdHeader(), // reply_from
|
|
MTPMessageMedia(), // reply_media
|
|
MTP_int(replyToTop),
|
|
MTP_string(replyTo.quote.text),
|
|
quoteEntities,
|
|
MTP_int(replyTo.quoteOffset));
|
|
}
|
|
return MTPMessageReplyHeader();
|
|
}
|
|
|
|
MediaCheckResult CheckMessageMedia(const MTPMessageMedia &media) {
|
|
using Result = MediaCheckResult;
|
|
return media.match([](const MTPDmessageMediaEmpty &) {
|
|
return Result::Good;
|
|
}, [](const MTPDmessageMediaContact &) {
|
|
return Result::Good;
|
|
}, [](const MTPDmessageMediaGeo &data) {
|
|
return data.vgeo().match([](const MTPDgeoPoint &) {
|
|
return Result::Good;
|
|
}, [](const MTPDgeoPointEmpty &) {
|
|
return Result::Empty;
|
|
});
|
|
}, [](const MTPDmessageMediaVenue &data) {
|
|
return data.vgeo().match([](const MTPDgeoPoint &) {
|
|
return Result::Good;
|
|
}, [](const MTPDgeoPointEmpty &) {
|
|
return Result::Empty;
|
|
});
|
|
}, [](const MTPDmessageMediaGeoLive &data) {
|
|
return data.vgeo().match([](const MTPDgeoPoint &) {
|
|
return Result::Good;
|
|
}, [](const MTPDgeoPointEmpty &) {
|
|
return Result::Empty;
|
|
});
|
|
}, [](const MTPDmessageMediaPhoto &data) {
|
|
const auto photo = data.vphoto();
|
|
if (data.vttl_seconds()) {
|
|
return Result::HasUnsupportedTimeToLive;
|
|
} else if (!photo) {
|
|
return Result::Empty;
|
|
}
|
|
return photo->match([](const MTPDphoto &) {
|
|
return Result::Good;
|
|
}, [](const MTPDphotoEmpty &) {
|
|
return Result::Empty;
|
|
});
|
|
}, [](const MTPDmessageMediaDocument &data) {
|
|
const auto document = data.vdocument();
|
|
if (data.vttl_seconds()) {
|
|
if (data.is_video()) {
|
|
return Result::HasUnsupportedTimeToLive;
|
|
} else if (!document) {
|
|
return Result::HasExpiredMediaTimeToLive;
|
|
}
|
|
} else if (!document) {
|
|
return Result::Empty;
|
|
}
|
|
return document->match([](const MTPDdocument &) {
|
|
return Result::Good;
|
|
}, [](const MTPDdocumentEmpty &) {
|
|
return Result::Empty;
|
|
});
|
|
}, [](const MTPDmessageMediaWebPage &data) {
|
|
return data.vwebpage().match([](const MTPDwebPage &) {
|
|
return Result::Good;
|
|
}, [](const MTPDwebPageEmpty &) {
|
|
return Result::Good;
|
|
}, [](const MTPDwebPagePending &) {
|
|
return Result::Good;
|
|
}, [](const MTPDwebPageNotModified &) {
|
|
return Result::Unsupported;
|
|
});
|
|
}, [](const MTPDmessageMediaGame &data) {
|
|
return data.vgame().match([](const MTPDgame &) {
|
|
return Result::Good;
|
|
});
|
|
}, [](const MTPDmessageMediaInvoice &) {
|
|
return Result::Good;
|
|
}, [](const MTPDmessageMediaPoll &) {
|
|
return Result::Good;
|
|
}, [](const MTPDmessageMediaDice &) {
|
|
return Result::Good;
|
|
}, [](const MTPDmessageMediaStory &data) {
|
|
return data.is_via_mention()
|
|
? Result::HasStoryMention
|
|
: Result::Good;
|
|
}, [](const MTPDmessageMediaGiveaway &) {
|
|
return Result::Good;
|
|
}, [](const MTPDmessageMediaGiveawayResults &) {
|
|
return Result::Good;
|
|
}, [](const MTPDmessageMediaUnsupported &) {
|
|
return Result::Unsupported;
|
|
});
|
|
}
|
|
|
|
[[nodiscard]] CallId CallIdFromInput(const MTPInputGroupCall &data) {
|
|
return data.match([&](const MTPDinputGroupCall &data) {
|
|
return data.vid().v;
|
|
});
|
|
}
|
|
|
|
std::vector<not_null<UserData*>> ParseInvitedToCallUsers(
|
|
not_null<HistoryItem*> item,
|
|
const QVector<MTPlong> &users) {
|
|
auto &owner = item->history()->owner();
|
|
return ranges::views::all(
|
|
users
|
|
) | ranges::views::transform([&](const MTPlong &id) {
|
|
return owner.user(id.v);
|
|
}) | ranges::to_vector;
|
|
}
|
|
|
|
PreparedServiceText GenerateJoinedText(
|
|
not_null<History*> history,
|
|
not_null<UserData*> inviter,
|
|
bool viaRequest) {
|
|
if (inviter->id != history->session().userPeerId()) {
|
|
auto result = PreparedServiceText();
|
|
result.links.push_back(inviter->createOpenLink());
|
|
result.text = (history->peer->isMegagroup()
|
|
? tr::lng_action_add_you_group
|
|
: tr::lng_action_add_you)(
|
|
tr::now,
|
|
lt_from,
|
|
Ui::Text::Link(inviter->name(), QString()),
|
|
Ui::Text::WithEntities);
|
|
return result;
|
|
} else if (history->peer->isMegagroup()) {
|
|
if (viaRequest) {
|
|
return { tr::lng_action_you_joined_by_request(
|
|
tr::now,
|
|
Ui::Text::WithEntities) };
|
|
}
|
|
auto self = history->session().user();
|
|
auto result = PreparedServiceText();
|
|
result.links.push_back(self->createOpenLink());
|
|
result.text = tr::lng_action_user_joined(
|
|
tr::now,
|
|
lt_from,
|
|
Ui::Text::Link(self->name(), QString()),
|
|
Ui::Text::WithEntities);
|
|
return result;
|
|
}
|
|
return { viaRequest
|
|
? tr::lng_action_you_joined_by_request_channel(
|
|
tr::now,
|
|
Ui::Text::WithEntities)
|
|
: tr::lng_action_you_joined(tr::now, Ui::Text::WithEntities) };
|
|
}
|
|
|
|
not_null<HistoryItem*> GenerateJoinedMessage(
|
|
not_null<History*> history,
|
|
TimeId inviteDate,
|
|
not_null<UserData*> inviter,
|
|
bool viaRequest) {
|
|
return history->makeMessage({
|
|
.id = history->owner().nextLocalMessageId(),
|
|
.flags = MessageFlag::Local | MessageFlag::ShowSimilarChannels,
|
|
.date = inviteDate,
|
|
}, GenerateJoinedText(history, inviter, viaRequest));
|
|
}
|
|
|
|
std::optional<bool> PeerHasThisCall(
|
|
not_null<PeerData*> peer,
|
|
CallId id) {
|
|
const auto call = peer->groupCall();
|
|
return call
|
|
? std::make_optional(call->id() == id)
|
|
: PeerCallKnown(peer)
|
|
? std::make_optional(false)
|
|
: std::nullopt;
|
|
}
|
|
[[nodiscard]] rpl::producer<bool> PeerHasThisCallValue(
|
|
not_null<PeerData*> peer,
|
|
CallId id) {
|
|
return peer->session().changes().peerFlagsValue(
|
|
peer,
|
|
Data::PeerUpdate::Flag::GroupCall
|
|
) | rpl::filter([=] {
|
|
return PeerCallKnown(peer);
|
|
}) | rpl::map([=] {
|
|
const auto call = peer->groupCall();
|
|
return (call && call->id() == id);
|
|
}) | rpl::distinct_until_changed(
|
|
) | rpl::take_while([=](bool hasThisCall) {
|
|
return hasThisCall;
|
|
}) | rpl::then(
|
|
rpl::single(false)
|
|
);
|
|
}
|
|
|
|
[[nodiscard]] ClickHandlerPtr GroupCallClickHandler(
|
|
not_null<PeerData*> peer,
|
|
CallId callId) {
|
|
return std::make_shared<LambdaClickHandler>([=] {
|
|
const auto call = peer->groupCall();
|
|
if (call && call->id() == callId) {
|
|
const auto &windows = peer->session().windows();
|
|
if (windows.empty()) {
|
|
Core::App().domain().activate(&peer->session().account());
|
|
if (windows.empty()) {
|
|
return;
|
|
}
|
|
}
|
|
windows.front()->startOrJoinGroupCall(peer, {});
|
|
}
|
|
});
|
|
}
|
|
|
|
[[nodiscard]] MessageFlags FinalizeMessageFlags(
|
|
not_null<History*> history,
|
|
MessageFlags flags) {
|
|
if (!(flags & MessageFlag::FakeHistoryItem)
|
|
&& !(flags & MessageFlag::IsOrWasScheduled)
|
|
&& !(flags & MessageFlag::ShortcutMessage)
|
|
&& !(flags & MessageFlag::AdminLogEntry)) {
|
|
flags |= MessageFlag::HistoryEntry;
|
|
if (history->peer->isSelf()) {
|
|
flags |= MessageFlag::ReactionsAreTags;
|
|
}
|
|
}
|
|
return flags;
|
|
}
|
|
|
|
using OnStackUsers = std::array<UserData*, kMaxUnreadReactions>;
|
|
[[nodiscard]] OnStackUsers LookupRecentUnreadReactedUsers(
|
|
not_null<HistoryItem*> item) {
|
|
auto result = OnStackUsers();
|
|
auto index = 0;
|
|
for (const auto &[emoji, reactions] : item->recentReactions()) {
|
|
for (const auto &reaction : reactions) {
|
|
if (!reaction.unread) {
|
|
continue;
|
|
}
|
|
if (const auto user = reaction.peer->asUser()) {
|
|
result[index++] = user;
|
|
if (index == result.size()) {
|
|
return result;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void CheckReactionNotificationSchedule(
|
|
not_null<HistoryItem*> item,
|
|
const OnStackUsers &wasUsers) {
|
|
// Call to addToUnreadThings may have read the reaction already.
|
|
if (!item->hasUnreadReaction()) {
|
|
return;
|
|
}
|
|
for (const auto &[emoji, reactions] : item->recentReactions()) {
|
|
for (const auto &reaction : reactions) {
|
|
if (!reaction.unread) {
|
|
continue;
|
|
}
|
|
const auto user = reaction.peer->asUser();
|
|
if (!user
|
|
|| !user->isContact()
|
|
|| ranges::contains(wasUsers, user)) {
|
|
continue;
|
|
}
|
|
using Status = PeerData::BlockStatus;
|
|
if (user->blockStatus() == Status::Unknown) {
|
|
user->updateFull();
|
|
}
|
|
const auto notification = Data::ItemNotification{
|
|
.item = item,
|
|
.reactionSender = user,
|
|
.type = Data::ItemNotificationType::Reaction,
|
|
};
|
|
item->notificationThread()->pushNotification(notification);
|
|
Core::App().notifications().schedule(notification);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
[[nodiscard]] MessageFlags NewForwardedFlags(
|
|
not_null<PeerData*> peer,
|
|
PeerId from,
|
|
not_null<HistoryItem*> fwd) {
|
|
auto result = NewMessageFlags(peer);
|
|
if (from) {
|
|
result |= MessageFlag::HasFromId;
|
|
}
|
|
if (const auto media = fwd->media()) {
|
|
if ((!peer->isChannel() || peer->isMegagroup())
|
|
&& media->forwardedBecomesUnread()) {
|
|
result |= MessageFlag::MediaIsUnread;
|
|
}
|
|
}
|
|
if (fwd->hasViews()) {
|
|
result |= MessageFlag::HasViews;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
[[nodiscard]] bool CopyMarkupToForward(not_null<const HistoryItem*> item) {
|
|
auto mediaOriginal = item->media();
|
|
if (mediaOriginal && mediaOriginal->game()) {
|
|
// Copy inline keyboard when forwarding messages with a game.
|
|
return true;
|
|
}
|
|
const auto markup = item->inlineReplyMarkup();
|
|
if (!markup) {
|
|
return false;
|
|
}
|
|
using Type = HistoryMessageMarkupButton::Type;
|
|
for (const auto &row : markup->data.rows) {
|
|
for (const auto &button : row) {
|
|
const auto switchInline = (button.type == Type::SwitchInline)
|
|
|| (button.type == Type::SwitchInlineSame);
|
|
const auto url = (button.type == Type::Url)
|
|
|| (button.type == Type::Auth);
|
|
if ((!switchInline || !item->viaBot()) && !url) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
[[nodiscard]] TextWithEntities EnsureNonEmpty(
|
|
const TextWithEntities &text) {
|
|
return !text.text.isEmpty() ? text : TextWithEntities{ u":-("_q };
|
|
}
|
|
|
|
[[nodiscard]] TextWithEntities UnsupportedMessageText() {
|
|
const auto siteLink = u"https://desktop.telegram.org"_q;
|
|
auto result = TextWithEntities{
|
|
tr::lng_message_unsupported(tr::now, lt_link, siteLink)
|
|
};
|
|
TextUtilities::ParseEntities(result, Ui::ItemTextNoMonoOptions().flags);
|
|
result.entities.push_front(
|
|
EntityInText(EntityType::Italic, 0, result.text.size()));
|
|
return result;
|
|
}
|
|
|
|
void ShowTrialTranscribesToast(int left, TimeId until) {
|
|
const auto window = Core::App().activeWindow();
|
|
if (!window) {
|
|
return;
|
|
}
|
|
const auto filter = [=](const auto &...) {
|
|
if (const auto controller = window->sessionController()) {
|
|
ShowPremiumPreviewBox(controller, PremiumFeature::VoiceToText);
|
|
window->activate();
|
|
}
|
|
return false;
|
|
};
|
|
const auto date = langDateTime(base::unixtime::parse(until));
|
|
constexpr auto kToastDuration = crl::time(4000);
|
|
const auto text = left
|
|
? tr::lng_audio_transcribe_trials_left(
|
|
tr::now,
|
|
lt_count,
|
|
left,
|
|
lt_date,
|
|
{ date },
|
|
Ui::Text::WithEntities)
|
|
: tr::lng_audio_transcribe_trials_over(
|
|
tr::now,
|
|
lt_date,
|
|
Ui::Text::Bold(date),
|
|
lt_link,
|
|
Ui::Text::Link(tr::lng_settings_privacy_premium_link(tr::now)),
|
|
Ui::Text::WithEntities);
|
|
window->uiShow()->showToast(Ui::Toast::Config{
|
|
.text = text,
|
|
.duration = kToastDuration,
|
|
.filter = filter,
|
|
});
|
|
}
|
|
|
|
void ClearMediaAsExpired(not_null<HistoryItem*> item) {
|
|
if (const auto media = item->media()) {
|
|
if (!media->ttlSeconds()) {
|
|
return;
|
|
}
|
|
if (const auto document = media->document()) {
|
|
item->applyEditionToHistoryCleared();
|
|
auto text = (document->isVideoFile()
|
|
? tr::lng_ttl_video_expired
|
|
: document->isVoiceMessage()
|
|
? tr::lng_ttl_voice_expired
|
|
: document->isVideoMessage()
|
|
? tr::lng_ttl_round_expired
|
|
: tr::lng_message_empty)(tr::now, Ui::Text::WithEntities);
|
|
item->updateServiceText(PreparedServiceText{ std::move(text) });
|
|
} else if (const auto photo = media->photo()) {
|
|
item->applyEditionToHistoryCleared();
|
|
item->updateServiceText(PreparedServiceText{
|
|
tr::lng_ttl_photo_expired(tr::now, Ui::Text::WithEntities)
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
int ItemsForwardSendersCount(const HistoryItemsList &list) {
|
|
auto peers = base::flat_set<not_null<PeerData*>>();
|
|
auto names = base::flat_set<QString>();
|
|
for (const auto &item : list) {
|
|
if (const auto peer = item->originalSender()) {
|
|
peers.emplace(peer);
|
|
} else {
|
|
names.emplace(item->originalHiddenSenderInfo()->name);
|
|
}
|
|
}
|
|
return int(peers.size()) + int(names.size());
|
|
}
|
|
|
|
int ItemsForwardCaptionsCount(const HistoryItemsList &list) {
|
|
auto result = 0;
|
|
for (const auto &item : list) {
|
|
if (const auto media = item->media()) {
|
|
if (!item->originalText().text.isEmpty()
|
|
&& media->allowsEditCaption()) {
|
|
++result;
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|