tdesktop/Telegram/SourceFiles/core/click_handler_types.cpp
2023-03-07 08:42:29 +03:00

359 lines
10 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 "core/click_handler_types.h"
#include "lang/lang_keys.h"
#include "core/application.h"
#include "core/local_url_handlers.h"
#include "mainwidget.h"
#include "mainwindow.h"
#include "main/main_session.h"
#include "ui/boxes/confirm_box.h"
#include "ui/text/text_entity.h"
#include "ui/toast/toast.h"
#include "base/qthelp_regex.h"
#include "base/qt/qt_key_modifiers.h"
#include "storage/storage_account.h"
#include "history/history.h"
#include "history/view/history_view_element.h"
#include "history/history_item.h"
#include "data/data_user.h"
#include "data/data_session.h"
#include "window/window_controller.h"
#include "window/window_session_controller.h"
#include "styles/style_layers.h"
namespace {
// Possible context owners: media viewer, profile, history widget.
void SearchByHashtag(ClickContext context, const QString &tag) {
const auto my = context.other.value<ClickHandlerContext>();
const auto controller = my.sessionWindow.get();
if (!controller) {
return;
}
if (controller->openedFolder().current()) {
controller->closeFolder();
}
controller->widget()->ui_hideSettingsAndLayer(anim::type::normal);
Core::App().hideMediaView();
auto &data = controller->session().data();
const auto inPeer = my.peer
? my.peer
: my.itemId
? data.message(my.itemId)->history()->peer.get()
: nullptr;
controller->content()->searchMessages(
tag + ' ',
(inPeer && !inPeer->isUser())
? data.history(inPeer).get()
: Dialogs::Key());
}
} // namespace
bool UrlRequiresConfirmation(const QUrl &url) {
using namespace qthelp;
return !regex_match(
"(^|\\.)("
"telegram\\.(org|me|dog)"
"|t\\.me"
"|te\\.?legra\\.ph"
"|graph\\.org"
"|fragment\\.com"
"|telesco\\.pe"
")$",
url.host(),
RegExOption::CaseInsensitive);
}
QString HiddenUrlClickHandler::copyToClipboardText() const {
return url().startsWith(u"internal:url:"_q)
? url().mid(u"internal:url:"_q.size())
: url();
}
QString HiddenUrlClickHandler::copyToClipboardContextItemText() const {
return url().isEmpty()
? QString()
: !url().startsWith(u"internal:"_q)
? UrlClickHandler::copyToClipboardContextItemText()
: url().startsWith(u"internal:url:"_q)
? UrlClickHandler::copyToClipboardContextItemText()
: QString();
}
QString HiddenUrlClickHandler::dragText() const {
const auto result = HiddenUrlClickHandler::copyToClipboardText();
return result.startsWith(u"internal:"_q) ? QString() : result;
}
void HiddenUrlClickHandler::Open(QString url, QVariant context) {
url = Core::TryConvertUrlToLocal(url);
if (Core::InternalPassportLink(url)) {
return;
}
const auto open = [=] {
UrlClickHandler::Open(url, context);
};
if (url.startsWith(u"tg://"_q, Qt::CaseInsensitive)
|| url.startsWith(u"internal:"_q, Qt::CaseInsensitive)) {
UrlClickHandler::Open(url, QVariant::fromValue([&] {
auto result = context.value<ClickHandlerContext>();
result.mayShowConfirmation = !base::IsCtrlPressed();
return result;
}()));
} else {
const auto parsedUrl = QUrl::fromUserInput(url);
if (UrlRequiresConfirmation(parsedUrl) && !base::IsCtrlPressed()) {
Core::App().hideMediaView();
const auto displayed = parsedUrl.isValid()
? parsedUrl.toDisplayString()
: url;
const auto displayUrl = !IsSuspicious(displayed)
? displayed
: parsedUrl.isValid()
? QString::fromUtf8(parsedUrl.toEncoded())
: ShowEncoded(displayed);
const auto my = context.value<ClickHandlerContext>();
const auto controller = my.sessionWindow.get();
const auto use = controller
? &controller->window()
: Core::App().activeWindow();
auto box = Box([=](not_null<Ui::GenericBox*> box) {
Ui::ConfirmBox(box, {
.text = (tr::lng_open_this_link(tr::now)),
.confirmed = [=](Fn<void()> hide) { hide(); open(); },
.confirmText = tr::lng_open_link(),
});
const auto &st = st::boxLabel;
box->addSkip(st.style.lineHeight - st::boxPadding.bottom());
const auto url = box->addRow(
object_ptr<Ui::FlatLabel>(box, displayUrl, st));
url->setSelectable(true);
url->setContextCopyText(tr::lng_context_copy_link(tr::now));
});
if (my.show) {
my.show->showBox(std::move(box));
} else if (use) {
use->show(std::move(box), Ui::LayerOption::KeepOther);
}
} else {
open();
}
}
}
void BotGameUrlClickHandler::onClick(ClickContext context) const {
const auto url = Core::TryConvertUrlToLocal(this->url());
if (Core::InternalPassportLink(url)) {
return;
}
const auto open = [=] {
UrlClickHandler::Open(url, context.other);
};
if (url.startsWith(u"tg://"_q, Qt::CaseInsensitive)) {
open();
} else if (!_bot
|| _bot->isVerified()
|| _bot->session().local().isBotTrustedOpenGame(_bot->id)) {
open();
} else {
const auto my = context.other.value<ClickHandlerContext>();
if (const auto controller = my.sessionWindow.get()) {
const auto callback = [=, bot = _bot](Fn<void()> close) {
close();
bot->session().local().markBotTrustedOpenGame(bot->id);
open();
};
controller->show(Ui::MakeConfirmBox({
.text = tr::lng_allow_bot_pass(
tr::now,
lt_bot_name,
_bot->name()),
.confirmed = callback,
.confirmText = tr::lng_allow_bot(),
}));
}
}
}
auto HiddenUrlClickHandler::getTextEntity() const -> TextEntity {
return { EntityType::CustomUrl, url() };
}
QString MentionClickHandler::copyToClipboardContextItemText() const {
return tr::lng_context_copy_mention(tr::now);
}
void MentionClickHandler::onClick(ClickContext context) const {
const auto button = context.button;
if (button == Qt::LeftButton || button == Qt::MiddleButton) {
const auto my = context.other.value<ClickHandlerContext>();
const auto controller = my.sessionWindow.get();
const auto use = controller
? controller
: Core::App().activeWindow()
? Core::App().activeWindow()->sessionController()
: nullptr;
if (use) {
using Info = Window::SessionNavigation::PeerByLinkInfo;
use->showPeerByLink(Info{
.usernameOrId = _tag.mid(1),
.resolveType = Window::ResolveType::Mention,
});
}
}
}
auto MentionClickHandler::getTextEntity() const -> TextEntity {
return { EntityType::Mention };
}
void MentionNameClickHandler::onClick(ClickContext context) const {
const auto button = context.button;
if (button == Qt::LeftButton || button == Qt::MiddleButton) {
const auto my = context.other.value<ClickHandlerContext>();
if (const auto controller = my.sessionWindow.get()) {
if (auto user = _session->data().userLoaded(_userId)) {
controller->showPeerInfo(user);
}
}
}
}
auto MentionNameClickHandler::getTextEntity() const -> TextEntity {
const auto data = TextUtilities::MentionNameDataFromFields({
.selfId = _session->userId().bare,
.userId = _userId.bare,
.accessHash = _accessHash,
});
return { EntityType::MentionName, data };
}
QString MentionNameClickHandler::tooltip() const {
if (const auto user = _session->data().userLoaded(_userId)) {
const auto name = user->name();
if (name != _text) {
return name;
}
}
return QString();
}
QString HashtagClickHandler::copyToClipboardContextItemText() const {
return tr::lng_context_copy_hashtag(tr::now);
}
void HashtagClickHandler::onClick(ClickContext context) const {
const auto button = context.button;
if (button == Qt::LeftButton || button == Qt::MiddleButton) {
SearchByHashtag(context, _tag);
}
}
auto HashtagClickHandler::getTextEntity() const -> TextEntity {
return { EntityType::Hashtag };
}
QString CashtagClickHandler::copyToClipboardContextItemText() const {
return tr::lng_context_copy_hashtag(tr::now);
}
void CashtagClickHandler::onClick(ClickContext context) const {
const auto button = context.button;
if (button == Qt::LeftButton || button == Qt::MiddleButton) {
SearchByHashtag(context, _tag);
}
}
auto CashtagClickHandler::getTextEntity() const -> TextEntity {
return { EntityType::Cashtag };
}
void BotCommandClickHandler::onClick(ClickContext context) const {
const auto button = context.button;
if (button != Qt::LeftButton && button != Qt::MiddleButton) {
return;
}
const auto my = context.other.value<ClickHandlerContext>();
if (const auto delegate = my.elementDelegate ? my.elementDelegate() : nullptr) {
delegate->elementSendBotCommand(_cmd, my.itemId);
} else if (const auto controller = my.sessionWindow.get()) {
auto &data = controller->session().data();
const auto peer = my.peer
? my.peer
: my.itemId
? data.message(my.itemId)->history()->peer.get()
: nullptr;
// Can't find context.
if (!peer) {
return;
}
controller->widget()->ui_hideSettingsAndLayer(anim::type::normal);
Core::App().hideMediaView();
controller->content()->sendBotCommand({
.peer = peer,
.command = _cmd,
.context = my.itemId,
.replyTo = 0,
});
}
}
auto BotCommandClickHandler::getTextEntity() const -> TextEntity {
return { EntityType::BotCommand };
}
MonospaceClickHandler::MonospaceClickHandler(
const QString &text,
EntityType type)
: _text(text)
, _entity({ type }) {
}
void MonospaceClickHandler::onClick(ClickContext context) const {
const auto button = context.button;
if (button != Qt::LeftButton && button != Qt::MiddleButton) {
return;
}
const auto my = context.other.value<ClickHandlerContext>();
if (const auto controller = my.sessionWindow.get()) {
auto &data = controller->session().data();
const auto item = data.message(my.itemId);
const auto hasCopyRestriction = item
&& (!item->history()->peer->allowsForwarding()
|| item->forbidsForward());
const auto toastParent = Window::Show(controller).toastParent();
if (hasCopyRestriction) {
Ui::Toast::Show(
toastParent,
item->history()->peer->isBroadcast()
? tr::lng_error_nocopy_channel(tr::now)
: tr::lng_error_nocopy_group(tr::now));
return;
}
Ui::Toast::Show(toastParent, tr::lng_text_copied(tr::now));
}
TextUtilities::SetClipboardText(TextForMimeData::Simple(_text.trimmed()));
}
auto MonospaceClickHandler::getTextEntity() const -> TextEntity {
return _entity;
}
QString MonospaceClickHandler::url() const {
return _text;
}